/*- * BSD LICENSE * * Copyright(c) 2010-2014 Intel Corporation. All rights reserved. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in * the documentation and/or other materials provided with the * distribution. * * Neither the name of Intel Corporation nor the names of its * contributors may be used to endorse or promote products derived * from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #ifndef _RTE_COMMON_H_ #define _RTE_COMMON_H_ /** * @file * * Generic, commonly-used macro and inline function definitions * for DPDK. */ #ifdef __cplusplus extern "C" { #endif #include #include #include #include #include #ifndef typeof #define typeof __typeof__ #endif #ifndef asm #define asm __asm__ #endif /** C extension macro for environments lacking C11 features. */ #if !defined(__STDC_VERSION__) || __STDC_VERSION__ < 201112L #define RTE_STD_C11 __extension__ #else #define RTE_STD_C11 #endif #ifdef RTE_ARCH_STRICT_ALIGN typedef uint64_t unaligned_uint64_t __attribute__ ((aligned(1))); typedef uint32_t unaligned_uint32_t __attribute__ ((aligned(1))); typedef uint16_t unaligned_uint16_t __attribute__ ((aligned(1))); #else typedef uint64_t unaligned_uint64_t; typedef uint32_t unaligned_uint32_t; typedef uint16_t unaligned_uint16_t; #endif /** * Force alignment */ #define __rte_aligned(a) __attribute__((__aligned__(a))) /** * Force a structure to be packed */ #define __rte_packed __attribute__((__packed__)) /******* Macro to mark functions and fields scheduled for removal *****/ #define __rte_deprecated __attribute__((__deprecated__)) /*********** Macros to eliminate unused variable warnings ********/ /** * short definition to mark a function parameter unused */ #define __rte_unused __attribute__((__unused__)) /** * definition to mark a variable or function parameter as used so * as to avoid a compiler warning */ #define RTE_SET_USED(x) (void)(x) /*********** Macros for pointer arithmetic ********/ /** * add a byte-value offset from a pointer */ #define RTE_PTR_ADD(ptr, x) ((void*)((uintptr_t)(ptr) + (x))) /** * subtract a byte-value offset from a pointer */ #define RTE_PTR_SUB(ptr, x) ((void*)((uintptr_t)ptr - (x))) /** * get the difference between two pointer values, i.e. how far apart * in bytes are the locations they point two. It is assumed that * ptr1 is greater than ptr2. */ #define RTE_PTR_DIFF(ptr1, ptr2) ((uintptr_t)(ptr1) - (uintptr_t)(ptr2)) /*********** Macros/static functions for doing alignment ********/ /** * Macro to align a pointer to a given power-of-two. The resultant * pointer will be a pointer of the same type as the first parameter, and * point to an address no higher than the first parameter. Second parameter * must be a power-of-two value. */ #define RTE_PTR_ALIGN_FLOOR(ptr, align) \ ((typeof(ptr))RTE_ALIGN_FLOOR((uintptr_t)ptr, align)) /** * 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 RTE_ALIGN_FLOOR(val, align) \ (typeof(val))((val) & (~((typeof(val))((align) - 1)))) /** * Macro to align a pointer to a given power-of-two. The resultant * pointer will be a pointer of the same type as the first parameter, and * point to an address no lower than the first parameter. Second parameter * must be a power-of-two value. */ #define RTE_PTR_ALIGN_CEIL(ptr, align) \ RTE_PTR_ALIGN_FLOOR((typeof(ptr))RTE_PTR_ADD(ptr, (align) - 1), align) /** * 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 RTE_ALIGN_CEIL(val, align) \ RTE_ALIGN_FLOOR(((val) + ((typeof(val)) (align) - 1)), align) /** * Macro to align a pointer to a given power-of-two. The resultant * pointer will be a pointer of the same type as the first parameter, and * point to an address no lower than the first parameter. Second parameter * must be a power-of-two value. * This function is the same as RTE_PTR_ALIGN_CEIL */ #define RTE_PTR_ALIGN(ptr, align) RTE_PTR_ALIGN_CEIL(ptr, align) /** * 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. * This function is the same as RTE_ALIGN_CEIL */ #define RTE_ALIGN(val, align) RTE_ALIGN_CEIL(val, align) /** * Checks if a pointer is aligned to a given power-of-two value * * @param ptr * The pointer whose alignment is to be checked * @param align * The power-of-two value to which the ptr should be aligned * * @return * True(1) where the pointer is correctly aligned, false(0) otherwise */ static inline int rte_is_aligned(void *ptr, unsigned align) { return RTE_PTR_ALIGN(ptr, align) == ptr; } /*********** Macros for compile type checks ********/ /** * Triggers an error at compilation time if the condition is true. */ #ifndef __OPTIMIZE__ #define RTE_BUILD_BUG_ON(condition) ((void)sizeof(char[1 - 2*!!(condition)])) #else extern int RTE_BUILD_BUG_ON_detected_error; #define RTE_BUILD_BUG_ON(condition) do { \ ((void)sizeof(char[1 - 2*!!(condition)])); \ if (condition) \ RTE_BUILD_BUG_ON_detected_error = 1; \ } while(0) #endif /*********** Macros to work with powers of 2 ********/ /** * Returns true if n is a power of 2 * @param n * Number to check * @return 1 if true, 0 otherwise */ static inline int rte_is_power_of_2(uint32_t n) { return n && !(n & (n - 1)); } /** * Aligns input parameter to the next power of 2 * * @param x * The integer value to algin * * @return * Input parameter aligned to the next power of 2 */ static inline uint32_t rte_align32pow2(uint32_t x) { x--; x |= x >> 1; x |= x >> 2; x |= x >> 4; x |= x >> 8; x |= x >> 16; return x + 1; } /** * Aligns 64b input parameter to the next power of 2 * * @param v * The 64b value to align * * @return * Input parameter aligned to the next power of 2 */ static inline uint64_t rte_align64pow2(uint64_t v) { v--; v |= v >> 1; v |= v >> 2; v |= v >> 4; v |= v >> 8; v |= v >> 16; v |= v >> 32; return v + 1; } /*********** Macros for calculating min and max **********/ /** * Macro to return the minimum of two numbers */ #define RTE_MIN(a, b) \ __extension__ ({ \ typeof (a) _a = (a); \ typeof (b) _b = (b); \ _a < _b ? _a : _b; \ }) /** * Macro to return the maximum of two numbers */ #define RTE_MAX(a, b) \ __extension__ ({ \ typeof (a) _a = (a); \ typeof (b) _b = (b); \ _a > _b ? _a : _b; \ }) /*********** Other general functions / macros ********/ #ifdef __SSE2__ #include /** * PAUSE instruction for tight loops (avoid busy waiting) */ static inline void rte_pause (void) { _mm_pause(); } #else static inline void rte_pause(void) {} #endif /** * Searches the input parameter for the least significant set bit * (starting from zero). * If a least significant 1 bit is found, its bit index is returned. * If the content of the input parameter is zero, then the content of the return * value is undefined. * @param v * input parameter, should not be zero. * @return * least significant set bit in the input parameter. */ static inline uint32_t rte_bsf32(uint32_t v) { return __builtin_ctz(v); } #ifndef offsetof /** Return the offset of a field in a structure. */ #define offsetof(TYPE, MEMBER) __builtin_offsetof (TYPE, MEMBER) #endif /** * Return pointer to the wrapping struct instance. * * Example: * * struct wrapper { * ... * struct child c; * ... * }; * * struct child *x = obtain(...); * struct wrapper *w = container_of(x, struct wrapper, c); */ #ifndef container_of #define container_of(ptr, type, member) __extension__ ({ \ const typeof(((type *)0)->member) *_ptr = (ptr); \ (type *)(((uintptr_t)_ptr) - offsetof(type, member)); \ }) #endif #define _RTE_STR(x) #x /** Take a macro value and get a string version of it */ #define RTE_STR(x) _RTE_STR(x) /** * ISO C helpers to modify format strings using variadic macros. * This is a replacement for the ", ## __VA_ARGS__" GNU extension. * An empty %s argument is appended to avoid a dangling comma. */ #define RTE_FMT(fmt, ...) fmt "%.0s", __VA_ARGS__ "" #define RTE_FMT_HEAD(fmt, ...) fmt #define RTE_FMT_TAIL(fmt, ...) __VA_ARGS__ /** Mask value of type "tp" for the first "ln" bit set. */ #define RTE_LEN2MASK(ln, tp) \ ((tp)((uint64_t)-1 >> (sizeof(uint64_t) * CHAR_BIT - (ln)))) /** Number of elements in the array. */ #define RTE_DIM(a) (sizeof (a) / sizeof ((a)[0])) /** * Converts a numeric string to the equivalent uint64_t value. * As well as straight number conversion, also recognises the suffixes * k, m and g for kilobytes, megabytes and gigabytes respectively. * * If a negative number is passed in i.e. a string with the first non-black * character being "-", zero is returned. Zero is also returned in the case of * an error with the strtoull call in the function. * * @param str * String containing number to convert. * @return * Number. */ static inline uint64_t rte_str_to_size(const char *str) { char *endptr; unsigned long long size; while (isspace((int)*str)) str++; if (*str == '-') return 0; errno = 0; size = strtoull(str, &endptr, 0); if (errno) return 0; if (*endptr == ' ') endptr++; /* allow 1 space gap */ switch (*endptr){ case 'G': case 'g': size *= 1024; /* fall-through */ case 'M': case 'm': size *= 1024; /* fall-through */ case 'K': case 'k': size *= 1024; /* fall-through */ default: break; } return size; } /** * Function to terminate the application immediately, printing an error * message and returning the exit_code back to the shell. * * This function never returns * * @param exit_code * The exit code to be returned by the application * @param format * The format string to be used for printing the message. This can include * printf format characters which will be expanded using any further parameters * to the function. */ void rte_exit(int exit_code, const char *format, ...) __attribute__((noreturn)) __attribute__((format(printf, 2, 3))); #ifdef __cplusplus } #endif #endif