2
0
mirror of https://github.com/openvswitch/ovs synced 2025-09-01 14:55:18 +00:00

util: New function raw_ctz().

This will acquire a user in an upcoming commit.

Signed-off-by: Ben Pfaff <blp@nicira.com>
This commit is contained in:
Ben Pfaff
2012-08-21 10:47:22 -07:00
parent 3ca1de08b4
commit 0ee140fb69
2 changed files with 43 additions and 24 deletions

View File

@@ -790,38 +790,34 @@ log_2_ceil(uint32_t n)
return log_2_floor(n) + !IS_POW2(n); return log_2_floor(n) + !IS_POW2(n);
} }
/* Returns the number of trailing 0-bits in 'n', or 32 if 'n' is 0. */ /* Returns the number of trailing 0-bits in 'n'. Undefined if 'n' == 0. */
int
ctz(uint32_t n)
{
if (!n) {
return 32;
} else {
#if !defined(UINT_MAX) || !defined(UINT32_MAX) #if !defined(UINT_MAX) || !defined(UINT32_MAX)
#error "Someone screwed up the #includes." #error "Someone screwed up the #includes."
#elif __GNUC__ >= 4 && UINT_MAX == UINT32_MAX #elif __GNUC__ >= 4 && UINT_MAX == UINT32_MAX
return __builtin_ctz(n); /* Defined inline in util.h. */
#else #else
unsigned int k; static int
int count = 31; raw_ctz(uint32_t n)
{
unsigned int k;
int count = 31;
#define CTZ_STEP(X) \ #define CTZ_STEP(X) \
k = n << (X); \ k = n << (X); \
if (k) { \ if (k) { \
count -= X; \ count -= X; \
n = k; \ n = k; \
} }
CTZ_STEP(16); CTZ_STEP(16);
CTZ_STEP(8); CTZ_STEP(8);
CTZ_STEP(4); CTZ_STEP(4);
CTZ_STEP(2); CTZ_STEP(2);
CTZ_STEP(1); CTZ_STEP(1);
#undef CTZ_STEP #undef CTZ_STEP
return count; return count;
#endif
}
} }
#endif
/* Returns the number of 1-bits in 'x', between 0 and 32 inclusive. */ /* Returns the number of 1-bits in 'x', between 0 and 32 inclusive. */
int int

View File

@@ -242,9 +242,32 @@ char *xreadlink(const char *filename);
char *follow_symlinks(const char *filename); char *follow_symlinks(const char *filename);
void ignore(bool x OVS_UNUSED); void ignore(bool x OVS_UNUSED);
/* Returns the number of trailing 0-bits in 'n'. Undefined if 'n' == 0.
*
* This compiles to a single machine instruction ("bsf") with GCC on x86. */
#if !defined(UINT_MAX) || !defined(UINT32_MAX)
#error "Someone screwed up the #includes."
#elif __GNUC__ >= 4 && UINT_MAX == UINT32_MAX
static inline int
raw_ctz(uint32_t n)
{
return __builtin_ctz(n);
}
#else
/* Defined in util.c. */
int raw_ctz(uint32_t n);
#endif
/* Returns the number of trailing 0-bits in 'n', or 32 if 'n' is 0. */
static inline int
ctz(uint32_t n)
{
return n ? raw_ctz(n) : 32;
}
int log_2_floor(uint32_t); int log_2_floor(uint32_t);
int log_2_ceil(uint32_t); int log_2_ceil(uint32_t);
int ctz(uint32_t);
int popcount(uint32_t); int popcount(uint32_t);
bool is_all_zeros(const uint8_t *, size_t); bool is_all_zeros(const uint8_t *, size_t);