/* SPDX-License-Identifier: BSD-3-Clause * Copyright (C) 2017 Intel Corporation. All rights reserved. * Copyright (c) 2019 Mellanox Technologies LTD. All rights reserved. */ /** \file * General utility functions */ #ifndef SPDK_UTIL_H #define SPDK_UTIL_H #include "spdk/stdinc.h" #ifdef __cplusplus extern "C" { #endif #define SPDK_CACHE_LINE_SIZE 64 #define spdk_min(a,b) (((a)<(b))?(a):(b)) #define spdk_max(a,b) (((a)>(b))?(a):(b)) #define SPDK_COUNTOF(arr) (sizeof(arr) / sizeof((arr)[0])) #define SPDK_CONTAINEROF(ptr, type, member) ((type *)((uintptr_t)ptr - offsetof(type, member))) /** * Get the size of a member of a struct. */ #define SPDK_SIZEOF_MEMBER(type, member) (sizeof(((type *)0)->member)) #define SPDK_SEC_TO_USEC 1000000ULL #define SPDK_SEC_TO_NSEC 1000000000ULL /* Ceiling division of unsigned integers */ #define SPDK_CEIL_DIV(x,y) (((x)+(y)-1)/(y)) /** * Macro to align a value to a given power-of-two. The resultant value * will be of the same type as the first parameter, and will be no * bigger than the first parameter. Second parameter must be a * power-of-two value. */ #define SPDK_ALIGN_FLOOR(val, align) \ (__typeof__(val))((val) & (~((__typeof__(val))((align) - 1)))) /** * Macro to align a value to a given power-of-two. The resultant value * will be of the same type as the first parameter, and will be no lower * than the first parameter. Second parameter must be a power-of-two * value. */ #define SPDK_ALIGN_CEIL(val, align) \ SPDK_ALIGN_FLOOR(((val) + ((__typeof__(val)) (align) - 1)), align) uint32_t spdk_u32log2(uint32_t x); static inline uint32_t spdk_align32pow2(uint32_t x) { return 1u << (1 + spdk_u32log2(x - 1)); } uint64_t spdk_u64log2(uint64_t x); static inline uint64_t spdk_align64pow2(uint64_t x) { return 1ULL << (1 + spdk_u64log2(x - 1)); } /** * Check if a uint32_t is a power of 2. */ static inline bool spdk_u32_is_pow2(uint32_t x) { if (x == 0) { return false; } return (x & (x - 1)) == 0; } /** * Check if a uint64_t is a power of 2. */ static inline bool spdk_u64_is_pow2(uint64_t x) { if (x == 0) { return false; } return (x & (x - 1)) == 0; } static inline uint64_t spdk_divide_round_up(uint64_t num, uint64_t divisor) { return (num + divisor - 1) / divisor; } /** * Copy the data described by the source iovec to the destination iovec. * * \return The number of bytes copied. */ size_t spdk_iovcpy(struct iovec *siov, size_t siovcnt, struct iovec *diov, size_t diovcnt); /** * Same as spdk_iovcpy(), but the src/dst buffers might overlap. * * \return The number of bytes copied. */ size_t spdk_iovmove(struct iovec *siov, size_t siovcnt, struct iovec *diov, size_t diovcnt); /** * An iovec iterator. Can be allocated on the stack. */ struct spdk_ioviter { struct iovec *siov; size_t siovcnt; struct iovec *diov; size_t diovcnt; size_t sidx; size_t didx; int siov_len; uint8_t *siov_base; int diov_len; uint8_t *diov_base; }; /** * Initialize and move to the first common segment of the two given * iovecs. See spdk_ioviter_next(). */ size_t spdk_ioviter_first(struct spdk_ioviter *iter, struct iovec *siov, size_t siovcnt, struct iovec *diov, size_t diovcnt, void **src, void **dst); /** * Move to the next segment in the iterator. * * This will iterate through the segments of the source and destination * and return the individual segments, one by one. For example, if the * source consists of one element of length 4k and the destination * consists of 4 elements each of length 1k, this function will return * 4 1k src+dst pairs of buffers, and then return 0 bytes to indicate * the iteration is complete on the fifth call. */ size_t spdk_ioviter_next(struct spdk_ioviter *iter, void **src, void **dst); /** * Copy iovs contents to buf through memcpy. */ void spdk_copy_iovs_to_buf(void *buf, size_t buf_len, struct iovec *iovs, int iovcnt); /** * Copy buf contents to iovs through memcpy. */ void spdk_copy_buf_to_iovs(struct iovec *iovs, int iovcnt, void *buf, size_t buf_len); /** * Scan build is really pessimistic and assumes that mempool functions can * dequeue NULL buffers even if they return success. This is obviously a false * positive, but the mempool dequeue can be done in a DPDK inline function that * we can't decorate with usual assert(buf != NULL). Instead, we'll * preinitialize the dequeued buffer array with some dummy objects. */ #define SPDK_CLANG_ANALYZER_PREINIT_PTR_ARRAY(arr, arr_size, buf_size) \ do { \ static char dummy_buf[buf_size]; \ int i; \ for (i = 0; i < arr_size; i++) { \ arr[i] = (void *)dummy_buf; \ } \ } while (0) /** * Add two sequence numbers s1 and s2 * * \param s1 First sequence number * \param s2 Second sequence number * * \return Sum of s1 and s2 based on serial number arithmetic. */ static inline uint32_t spdk_sn32_add(uint32_t s1, uint32_t s2) { return (uint32_t)(s1 + s2); } #define SPDK_SN32_CMPMAX (1U << (32 - 1)) /** * Compare if sequence number s1 is less than s2. * * \param s1 First sequence number * \param s2 Second sequence number * * \return true if s1 is less than s2, or false otherwise. */ static inline bool spdk_sn32_lt(uint32_t s1, uint32_t s2) { return (s1 != s2) && ((s1 < s2 && s2 - s1 < SPDK_SN32_CMPMAX) || (s1 > s2 && s1 - s2 > SPDK_SN32_CMPMAX)); } /** * Compare if sequence number s1 is greater than s2. * * \param s1 First sequence number * \param s2 Second sequence number * * \return true if s1 is greater than s2, or false otherwise. */ static inline bool spdk_sn32_gt(uint32_t s1, uint32_t s2) { return (s1 != s2) && ((s1 < s2 && s2 - s1 > SPDK_SN32_CMPMAX) || (s1 > s2 && s1 - s2 < SPDK_SN32_CMPMAX)); } #ifdef __cplusplus } #endif #endif