/* SPDX-License-Identifier: GPL-2.0 */ #ifndef KUBLK_UTILS_H #define KUBLK_UTILS_H #ifndef min #define min(a, b) ((a) < (b) ? (a) : (b)) #endif #define ARRAY_SIZE(x) (sizeof(x) / sizeof(x[0])) #ifndef offsetof #define offsetof(TYPE, MEMBER) ((size_t)&((TYPE *)0)->MEMBER) #endif #ifndef container_of #define container_of(ptr, type, member) ({ \ unsigned long __mptr = (unsigned long)(ptr); \ ((type *)(__mptr - offsetof(type, member))); }) #endif #define round_up(val, rnd) \ (((val) + ((rnd) - 1)) & ~((rnd) - 1)) /* small sized & per-thread allocator */ struct allocator { unsigned int size; cpu_set_t *set; }; static inline int allocator_init(struct allocator *a, unsigned size) { a->set = CPU_ALLOC(size); a->size = size; if (a->set) return 0; return -ENOMEM; } static inline void allocator_deinit(struct allocator *a) { CPU_FREE(a->set); a->set = NULL; a->size = 0; } static inline int allocator_get(struct allocator *a) { int i; for (i = 0; i < a->size; i += 1) { size_t set_size = CPU_ALLOC_SIZE(a->size); if (!CPU_ISSET_S(i, set_size, a->set)) { CPU_SET_S(i, set_size, a->set); return i; } } return -1; } static inline void allocator_put(struct allocator *a, int i) { size_t set_size = CPU_ALLOC_SIZE(a->size); if (i >= 0 && i < a->size) CPU_CLR_S(i, set_size, a->set); } static inline int allocator_get_val(struct allocator *a, int i) { size_t set_size = CPU_ALLOC_SIZE(a->size); return CPU_ISSET_S(i, set_size, a->set); } static inline unsigned int ilog2(unsigned int x) { if (x == 0) return 0; return (sizeof(x) * 8 - 1) - __builtin_clz(x); } #define UBLK_DBG_DEV (1U << 0) #define UBLK_DBG_THREAD (1U << 1) #define UBLK_DBG_IO_CMD (1U << 2) #define UBLK_DBG_IO (1U << 3) #define UBLK_DBG_CTRL_CMD (1U << 4) #define UBLK_LOG (1U << 5) extern unsigned int ublk_dbg_mask; static inline void ublk_err(const char *fmt, ...) { va_list ap; va_start(ap, fmt); vfprintf(stderr, fmt, ap); va_end(ap); } static inline void ublk_log(const char *fmt, ...) { if (ublk_dbg_mask & UBLK_LOG) { va_list ap; va_start(ap, fmt); vfprintf(stdout, fmt, ap); va_end(ap); } } static inline void ublk_dbg(int level, const char *fmt, ...) { if (level & ublk_dbg_mask) { va_list ap; va_start(ap, fmt); vfprintf(stdout, fmt, ap); va_end(ap); } } #define ublk_assert(x) do { \ if (!(x)) { \ ublk_err("%s %d: assert!\n", __func__, __LINE__); \ assert(x); \ } \ } while (0) #endif