blob: bf3f1009f3eba16153babba08fe1e53739d63b84 [file] [log] [blame]
#pragma once
#include <inttypes.h>
#include <limits.h>
#include <stdarg.h>
#include <stdbool.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <time.h>
#include <shared/macro.h>
/* string handling functions and memory allocations */
/* ************************************************************************ */
#define streq(a, b) (strcmp((a), (b)) == 0)
#define strstartswith(a, b) (strncmp(a, b, strlen(b)) == 0)
char *strchr_replace(char *s, char c, char r);
_nonnull_all_ void *memdup(const void *p, size_t n);
/* module-related functions */
/* ************************************************************************ */
#define KMOD_EXTENSION_UNCOMPRESSED ".ko"
_must_check_ _nonnull_(1, 2) int alias_normalize(const char *alias,
char buf[static PATH_MAX], size_t *len);
_must_check_ int underscores(char *s);
_nonnull_(1, 2) char *modname_normalize(const char *modname, char buf[static PATH_MAX],
size_t *len);
_nonnull_(2) char *path_to_modname(const char *path, char buf[static PATH_MAX],
size_t *len);
_nonnull_all_ bool path_ends_with_kmod_ext(const char *path, size_t len);
/* read-like and fread-like functions */
/* ************************************************************************ */
_must_check_ _nonnull_(2) ssize_t pread_str_safe(int fd, char *buf, size_t buflen,
off_t off);
_must_check_ _nonnull_(2) ssize_t read_str_safe(int fd, char *buf, size_t buflen);
_nonnull_(2) ssize_t write_str_safe(int fd, const char *buf, size_t buflen);
_must_check_ _nonnull_(2) int read_str_long(int fd, long *value, int base);
_must_check_ _nonnull_(2) int read_str_ulong(int fd, unsigned long *value, int base);
_nonnull_(1) char *freadline_wrapped(FILE *fp, unsigned int *linenum);
/* path handling functions */
/* ************************************************************************ */
_must_check_ _nonnull_all_ char *path_make_absolute_cwd(const char *p);
static inline _must_check_ _nonnull_all_ bool path_is_absolute(const char *p)
{
return p[0] == '/';
}
int mkdir_p(const char *path, int len, mode_t mode);
int mkdir_parents(const char *path, mode_t mode);
unsigned long long stat_mstamp(const struct stat *st);
_nonnull_all_ int fd_lookup_path(int fd, char *path, size_t pathlen);
/* time-related functions
* ************************************************************************ */
#define USEC_PER_SEC 1000000ULL
#define USEC_PER_MSEC 1000ULL
#define MSEC_PER_SEC 1000ULL
#define NSEC_PER_MSEC 1000000ULL
unsigned long long ts_usec(const struct timespec *ts);
unsigned long long now_usec(void);
unsigned long long now_msec(void);
int sleep_until_msec(unsigned long long msec);
unsigned long long get_backoff_delta_msec(unsigned long long tend,
unsigned long long *delta);
/* endianness and alignments */
/* ************************************************************************ */
#define get_unaligned(ptr) \
({ \
struct __attribute__((packed)) { \
typeof(*(ptr)) __v; \
} *__p = (typeof(__p))(ptr); \
__p->__v; \
})
#define put_unaligned(val, ptr) \
do { \
struct __attribute__((packed)) { \
typeof(*(ptr)) __v; \
} *__p = (typeof(__p))(ptr); \
__p->__v = (val); \
} while (0)
static inline unsigned int align_power2(unsigned int u)
{
return 1 << ((sizeof(u) * 8) - __builtin_clz(u - 1));
}
/* misc */
/* ************************************************************************ */
static inline bool uadd32_overflow(uint32_t a, uint32_t b, uint32_t *res)
{
#if (HAVE___BUILTIN_UADD_OVERFLOW && __SIZEOF_INT__ == 4)
return __builtin_uadd_overflow(a, b, res);
#else
*res = a + b;
return UINT32_MAX - a < b;
#endif
}
static inline bool uadd64_overflow(uint64_t a, uint64_t b, uint64_t *res)
{
#if (HAVE___BUILTIN_UADDL_OVERFLOW && __SIZEOF_LONG__ == 8)
return __builtin_uaddl_overflow(a, b, res);
#elif (HAVE___BUILTIN_UADDLL_OVERFLOW && __SIZEOF_LONG_LONG__ == 8)
return __builtin_uaddll_overflow(a, b, res);
#else
*res = a + b;
return UINT64_MAX - a < b;
#endif
}
static inline bool uaddsz_overflow(size_t a, size_t b, size_t *res)
{
#if __SIZEOF_SIZE_T__ == 8
return uadd64_overflow(a, b, res);
#elif __SIZEOF_SIZE_T__ == 4
return uadd32_overflow(a, b, res);
#else
#error "Unknown sizeof(size_t)"
#endif
}
static inline bool umul32_overflow(uint32_t a, uint32_t b, uint32_t *res)
{
#if (HAVE___BUILTIN_UMUL_OVERFLOW && __SIZEOF_INT__ == 4)
return __builtin_umul_overflow(a, b, res);
#else
*res = a * b;
return UINT32_MAX / a < b;
#endif
}
static inline bool umul64_overflow(uint64_t a, uint64_t b, uint64_t *res)
{
#if (HAVE___BUILTIN_UMULL_OVERFLOW && __SIZEOF_LONG__ == 8)
return __builtin_umull_overflow(a, b, res);
#elif (HAVE___BUILTIN_UMULLL_OVERFLOW && __SIZEOF_LONG_LONG__ == 8)
return __builtin_umulll_overflow(a, b, res);
#else
*res = a * b;
return UINT64_MAX / a < b;
#endif
}
static inline bool umulll_overflow(unsigned long long a, unsigned long long b,
unsigned long long *res)
{
#if (HAVE___BUILTIN_UMULLL_OVERFLOW)
return __builtin_umulll_overflow(a, b, res);
#else
*res = a * b;
return UINT64_MAX / a < b;
#endif
}
static inline bool umulsz_overflow(size_t a, size_t b, size_t *res)
{
#if __SIZEOF_SIZE_T__ == 8
return umul64_overflow(a, b, res);
#elif __SIZEOF_SIZE_T__ == 4
return umul32_overflow(a, b, res);
#else
#error "Unknown sizeof(size_t)"
#endif
}
#define TAKE_PTR(x) \
({ \
typeof(x) x__ = x; \
(x) = NULL; \
x__; \
})
/* dlfcn helpers */
/* ************************************************************************ */
/*
* Load many various symbols from @filename.
* @dlp: pointer to the previous results of this call: it's set when it succeeds
* @filename: the library to dlopen() and look for symbols
* @...: or 1 more tuples created by DLSYM_ARG() with ( &var, "symbol name" ).
*/
_sentinel_ int dlsym_many(void **dlp, const char *filename, ...);
/*
* Helper to create tuples passed as arguments to dlsym_many().
* @symbol__: symbol to create arguments for. Example: DLSYM_ARG(foo) expands to
* `&sym_foo, "foo"`
*/
#define DLSYM_ARG(symbol__) &sym_##symbol__, STRINGIFY(symbol__),
/* For symbols being dynamically loaded */
#define DECLARE_DLSYM(symbol) static typeof(symbol) *sym_##symbol
/* Pointer indirection to support linking directly */
#define DECLARE_PTRSYM(symbol) static typeof(symbol) *sym_##symbol = symbol
/*
* Helper defines, to be done locally before including this header to switch between
* implementations
*/
#if defined(DLSYM_LOCALLY_ENABLED) && DLSYM_LOCALLY_ENABLED
#define DECLARE_SYM(sym__) DECLARE_DLSYM(sym__);
#else
#define DECLARE_SYM(sym__) DECLARE_PTRSYM(sym__);
#endif