40#include <rte_compat.h>
52#define RTE_BITSET_WORD_SIZE (sizeof(uint64_t))
58#define RTE_BITSET_WORD_BITS (RTE_BITSET_WORD_SIZE * CHAR_BIT)
63#define RTE_BITSET_NUM_WORDS(size) \
64 ((size + RTE_BITSET_WORD_BITS - 1) / RTE_BITSET_WORD_BITS)
70#define RTE_BITSET_SIZE(size) \
71 ((size_t)(RTE_BITSET_NUM_WORDS(size) * RTE_BITSET_WORD_SIZE))
73#define __RTE_BITSET_WORD_IDX(bit_num) ((bit_num) / RTE_BITSET_WORD_BITS)
74#define __RTE_BITSET_BIT_OFFSET(bit_num) ((bit_num) % RTE_BITSET_WORD_BITS)
75#define __RTE_BITSET_UNUSED(size) \
76 ((RTE_BITSET_NUM_WORDS(size) * RTE_BITSET_WORD_BITS) - (size))
77#define __RTE_BITSET_USED_MASK(size) \
78 (UINT64_MAX >> __RTE_BITSET_UNUSED(size))
80#define __RTE_BITSET_DELEGATE_N(fun, bitset, bit_num, ...) \
81 fun(&(bitset)[__RTE_BITSET_WORD_IDX(bit_num)], __RTE_BITSET_BIT_OFFSET(bit_num), \
85#define __RTE_BITSET_DELEGATE(fun, bitset, bit_num) \
86 fun(&(bitset)[__RTE_BITSET_WORD_IDX(bit_num)], __RTE_BITSET_BIT_OFFSET(bit_num))
103#define RTE_BITSET_DECLARE(name, size) \
104 uint64_t name[RTE_BITSET_NUM_WORDS(size)]
106#define __RTE_BITSET_FOREACH_LEFT(var, size, start_bit, len) \
107 ((len) - 1 - ((var) >= (start_bit) ? (var) - (start_bit) : (size) - (start_bit) + (var)))
109#define __RTE_BITSET_FOREACH(var, bitset, size, start_bit, len, flags) \
110 for ((var) = __rte_bitset_find(bitset, size, start_bit, len, flags); \
112 (var) = __RTE_BITSET_FOREACH_LEFT(var, size, start_bit, len) > 0 ? \
113 __rte_bitset_find(bitset, size, ((var) + 1) % (size), \
114 __RTE_BITSET_FOREACH_LEFT(var, size, start_bit, len), flags) : -1)
134#define RTE_BITSET_FOREACH_SET(var, bitset, size) \
135 __RTE_BITSET_FOREACH(var, bitset, size, 0, size, 0)
154#define RTE_BITSET_FOREACH_CLEAR(var, bitset, size) \
155 __RTE_BITSET_FOREACH(var, bitset, size, 0, size, __RTE_BITSET_FIND_FLAG_FIND_CLEAR)
180#define RTE_BITSET_FOREACH_SET_RANGE(var, bitset, size, start_bit, len) \
181 __RTE_BITSET_FOREACH(var, bitset, size, start_bit, len, 0)
206#define RTE_BITSET_FOREACH_CLEAR_RANGE(var, bitset, size, start_bit, len) \
207 __RTE_BITSET_FOREACH(var, bitset, size, start_bit, len, __RTE_BITSET_FIND_FLAG_FIND_CLEAR)
209#define RTE_BITSET_FOREACH_SET_WRAP(var, bitset, size, start_bit, len) \
210 __RTE_BITSET_FOREACH(var, bitset, size, start_bit, len, __RTE_BITSET_FIND_FLAG_WRAP)
212#define RTE_BITSET_FOREACH_CLEAR_WRAP(var, bitset, size, start_bit, len) \
213 __RTE_BITSET_FOREACH(var, bitset, size, start_bit, len, \
214 __RTE_BITSET_FIND_FLAG_WRAP | __RTE_BITSET_FIND_FLAG_FIND_CLEAR)
257#ifdef ALLOW_EXPERIMENTAL_API
258 return __RTE_BITSET_DELEGATE(
rte_bit_test, bitset, bit_num);
285#ifdef ALLOW_EXPERIMENTAL_API
286 __RTE_BITSET_DELEGATE(
rte_bit_set, bitset, bit_num);
313#ifdef ALLOW_EXPERIMENTAL_API
343#ifdef ALLOW_EXPERIMENTAL_API
344 __RTE_BITSET_DELEGATE_N(
rte_bit_assign, bitset, bit_num, bit_value);
372#ifdef ALLOW_EXPERIMENTAL_API
403#ifdef ALLOW_EXPERIMENTAL_API
439#ifdef ALLOW_EXPERIMENTAL_API
475#ifdef ALLOW_EXPERIMENTAL_API
513#ifdef ALLOW_EXPERIMENTAL_API
550#ifdef ALLOW_EXPERIMENTAL_API
593#ifdef ALLOW_EXPERIMENTAL_API
629 total +=
rte_popcount64(bitset[i] & __RTE_BITSET_USED_MASK(size));
651#ifdef ALLOW_EXPERIMENTAL_API
660#define __RTE_BITSET_FIND_FLAG_FIND_CLEAR (1U << 0)
661#define __RTE_BITSET_FIND_FLAG_WRAP (1U << 1)
665__rte_bitset_find_nowrap(
const uint64_t *bitset,
size_t __rte_unused size,
size_t start_bit,
666 size_t len,
bool find_clear)
670 size_t end_bit = start_bit + len;
672 RTE_ASSERT(end_bit <= size);
677 word_idx = __RTE_BITSET_WORD_IDX(start_bit);
678 offset = __RTE_BITSET_BIT_OFFSET(start_bit);
680 while (word_idx <= __RTE_BITSET_WORD_IDX(end_bit - 1)) {
683 word = bitset[word_idx];
690 size_t ffs = start_bit +
rte_bsf64(word);
713__rte_bitset_find(
const uint64_t *bitset,
size_t size,
size_t start_bit,
size_t len,
716#ifdef ALLOW_EXPERIMENTAL_API
717 bool find_clear = flags & __RTE_BITSET_FIND_FLAG_FIND_CLEAR;
718 bool may_wrap = flags & __RTE_BITSET_FIND_FLAG_WRAP;
719 bool does_wrap = (start_bit + len) > size;
722 RTE_ASSERT(len <= size);
724 RTE_ASSERT(!does_wrap);
726 if (may_wrap && does_wrap) {
727 size_t len0 = size - start_bit;
728 size_t len1 = len - len0;
730 rc = __rte_bitset_find_nowrap(bitset, size, start_bit, len0, find_clear);
732 rc = __rte_bitset_find_nowrap(bitset, size, 0, len1, find_clear);
734 rc = __rte_bitset_find_nowrap(bitset, size, start_bit, len, find_clear);
768#ifdef ALLOW_EXPERIMENTAL_API
769 return __rte_bitset_find(bitset, size, 0, size, 0);
804#ifdef ALLOW_EXPERIMENTAL_API
805 return __rte_bitset_find(bitset, size, start_bit, len, 0);
843#ifdef ALLOW_EXPERIMENTAL_API
844 return __rte_bitset_find(bitset, size, start_bit, len, __RTE_BITSET_FIND_FLAG_WRAP);
875#ifdef ALLOW_EXPERIMENTAL_API
876 return __rte_bitset_find(bitset, size, 0, size, __RTE_BITSET_FIND_FLAG_FIND_CLEAR);
911#ifdef ALLOW_EXPERIMENTAL_API
912 return __rte_bitset_find(bitset, size, start_bit, len, __RTE_BITSET_FIND_FLAG_FIND_CLEAR);
950#ifdef ALLOW_EXPERIMENTAL_API
951 return __rte_bitset_find(bitset, size, start_bit, len,
952 __RTE_BITSET_FIND_FLAG_FIND_CLEAR | __RTE_BITSET_FIND_FLAG_WRAP);
1008rte_bitset_or(uint64_t *dst_bitset,
const uint64_t *src_bitset0,
const uint64_t *src_bitset1,
1014 dst_bitset[i] = src_bitset0[i] | src_bitset1[i];
1038rte_bitset_and(uint64_t *dst_bitset,
const uint64_t *src_bitset0,
const uint64_t *src_bitset1,
1044 dst_bitset[i] = src_bitset0[i] & src_bitset1[i];
1068rte_bitset_xor(uint64_t *dst_bitset,
const uint64_t *src_bitset0,
const uint64_t *src_bitset1,
1074 dst_bitset[i] = src_bitset0[i] ^ src_bitset1[i];
1100 dst_bitset[i] = ~src_bitset[i];
1128 unsigned int dst_idx;
1131 int src_high_idx = dst_idx - src_word_offset;
1132 uint64_t low_bits = 0;
1133 uint64_t high_bits = 0;
1135 if (src_high_idx >= 0) {
1136 int src_low_idx = src_high_idx - 1;
1138 high_bits = src_bitset[src_high_idx] << src_bit_offset;
1140 if (src_bit_offset > 0 && src_low_idx >= 0)
1141 low_bits = src_bitset[src_low_idx] >>
1144 dst_bitset[dst_idx] = low_bits | high_bits;
1172 const uint64_t used_mask = __RTE_BITSET_USED_MASK(size);
1177 for (dst_idx = 0; dst_idx < num_words; dst_idx++) {
1178 int src_low_idx = src_word_offset + dst_idx;
1179 int src_high_idx = src_low_idx + 1;
1180 uint64_t src_low_word_bits = 0;
1181 uint64_t src_high_word_bits = 0;
1183 if (src_low_idx < num_words) {
1184 src_low_word_bits = src_bitset[src_low_idx];
1186 if (src_low_idx == (num_words - 1))
1187 src_low_word_bits &= used_mask;
1189 src_low_word_bits >>= src_bit_offset;
1191 if (src_bit_offset > 0 && src_high_idx < num_words) {
1192 src_high_word_bits = src_bitset[src_high_idx];
1194 if (src_high_idx == (num_words - 1))
1195 src_high_word_bits &= used_mask;
1200 dst_bitset[dst_idx] = src_low_word_bits | src_high_word_bits;
1224 uint64_t last_a, last_b;
1227 if (bitset_a[i] != bitset_b[i])
1230 last_a = bitset_a[i] << __RTE_BITSET_UNUSED(size);
1231 last_b = bitset_b[i] << __RTE_BITSET_UNUSED(size);
1233 return last_a == last_b;
#define rte_bit_atomic_assign(addr, nr, value, memory_order)
#define rte_bit_flip(addr, nr)
#define rte_bit_set(addr, nr)
#define rte_bit_atomic_flip(addr, nr, memory_order)
static uint32_t rte_bsf64(uint64_t v)
#define rte_bit_assign(addr, nr, value)
#define rte_bit_test(addr, nr)
static unsigned int rte_popcount64(uint64_t v)
#define rte_bit_atomic_set(addr, nr, memory_order)
#define rte_bit_clear(addr, nr)
#define rte_bit_atomic_clear(addr, nr, memory_order)
#define rte_bit_atomic_test(addr, nr, memory_order)
#define RTE_BITSET_WORD_BITS
static __rte_experimental ssize_t rte_bitset_find_clear_wrap(const uint64_t *bitset, size_t size, size_t start_bit, size_t len)
static __rte_experimental void rte_bitset_or(uint64_t *dst_bitset, const uint64_t *src_bitset0, const uint64_t *src_bitset1, size_t size)
static __rte_experimental size_t rte_bitset_count_set(const uint64_t *bitset, size_t size)
static __rte_experimental ssize_t rte_bitset_find_first_set(const uint64_t *bitset, size_t size)
static __rte_experimental void rte_bitset_assign(uint64_t *bitset, size_t bit_num, bool bit_value)
static __rte_experimental void rte_bitset_copy(uint64_t *__rte_restrict dst_bitset, const uint64_t *__rte_restrict src_bitset, size_t size)
static __rte_experimental ssize_t rte_bitset_find_first_clear(const uint64_t *bitset, size_t size)
static __rte_experimental void rte_bitset_and(uint64_t *dst_bitset, const uint64_t *src_bitset0, const uint64_t *src_bitset1, size_t size)
static __rte_experimental void rte_bitset_xor(uint64_t *dst_bitset, const uint64_t *src_bitset0, const uint64_t *src_bitset1, size_t size)
static __rte_experimental bool rte_bitset_atomic_test(const uint64_t *bitset, size_t bit_num, int memory_order)
static __rte_experimental void rte_bitset_shift_right(uint64_t *dst_bitset, const uint64_t *src_bitset, size_t size, size_t shift_bits)
static __rte_experimental ssize_t rte_bitset_find_set_wrap(const uint64_t *bitset, size_t size, size_t start_bit, size_t len)
static __rte_experimental void rte_bitset_set_all(uint64_t *bitset, size_t size)
static __rte_experimental void rte_bitset_init(uint64_t *bitset, size_t size)
static __rte_experimental void rte_bitset_complement(uint64_t *dst_bitset, const uint64_t *src_bitset, size_t size)
static __rte_experimental bool rte_bitset_equal(const uint64_t *bitset_a, const uint64_t *bitset_b, size_t size)
static __rte_experimental void rte_bitset_atomic_flip(uint64_t *bitset, size_t bit_num, int memory_order)
static __rte_experimental void rte_bitset_shift_left(uint64_t *dst_bitset, const uint64_t *src_bitset, size_t size, size_t shift_bits)
static __rte_experimental void rte_bitset_atomic_set(uint64_t *bitset, size_t bit_num, int memory_order)
static __rte_experimental void rte_bitset_flip(uint64_t *bitset, size_t bit_num)
static __rte_experimental void rte_bitset_clear(uint64_t *bitset, size_t bit_num)
static __rte_experimental bool rte_bitset_test(const uint64_t *bitset, size_t bit_num)
static __rte_experimental void rte_bitset_clear_all(uint64_t *bitset, size_t size)
static __rte_experimental void rte_bitset_atomic_assign(uint64_t *bitset, size_t bit_num, bool bit_value, int memory_order)
static __rte_experimental size_t rte_bitset_count_clear(const uint64_t *bitset, size_t size)
static __rte_experimental void rte_bitset_set(uint64_t *bitset, size_t bit_num)
static __rte_experimental void rte_bitset_atomic_clear(uint64_t *bitset, size_t bit_num, int memory_order)
static __rte_experimental ssize_t rte_bitset_find_clear(const uint64_t *bitset, size_t size, size_t start_bit, size_t len)
#define RTE_BITSET_NUM_WORDS(size)
#define RTE_BITSET_SIZE(size)
static __rte_experimental ssize_t rte_bitset_find_set(const uint64_t *bitset, size_t size, size_t start_bit, size_t len)
__rte_experimental ssize_t rte_bitset_to_str(const uint64_t *bitset, size_t size, char *buf, size_t capacity)
static void * rte_memcpy(void *dst, const void *src, size_t n)