272 lines
8.9 KiB
C
272 lines
8.9 KiB
C
//
|
|
// Created by xajhuang on 2023/4/6.
|
|
//
|
|
|
|
#ifndef VCPE_BITSET_H
|
|
#define VCPE_BITSET_H
|
|
#include <common.h>
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
// For compatibility with MSVC with the use of `restrict`
|
|
#if (__STDC_VERSION__ >= 199901L) || (defined(__GNUC__) && defined(__STDC_VERSION__))
|
|
#define CBITSET_RESTRICT restrict
|
|
#else
|
|
#define CBITSET_RESTRICT
|
|
#endif // (__STDC_VERSION__ >= 199901L) || (defined(__GNUC__) && defined(__STDC_VERSION__ ))
|
|
|
|
struct bitset_s {
|
|
uint64_t *CBITSET_RESTRICT array;
|
|
size_t arraysize;
|
|
size_t capacity;
|
|
};
|
|
|
|
typedef struct bitset_s bitset_t;
|
|
|
|
/* Create a new bitset. Return NULL in case of failure. */
|
|
bitset_t *bitset_create(void);
|
|
|
|
/* Create a new bitset able to contain size bits. Return NULL in case of failure. */
|
|
bitset_t *bitset_create_with_capacity(size_t size);
|
|
|
|
/* Free memory. */
|
|
void bitset_free(bitset_t *bitset);
|
|
|
|
/* Set all bits to zero. */
|
|
void bitset_clear(bitset_t *bitset);
|
|
|
|
/* Create a copy */
|
|
bitset_t *bitset_copy(const bitset_t *bitset);
|
|
|
|
/* Resize the bitset. Return TRUE in case of success, FALSE for failure. Pad with zeroes new buffer areas if requested. */
|
|
int bitset_resize(bitset_t *bitset, size_t newarraysize, int padwithzeroes);
|
|
|
|
/* returns how many bytes of memory the backend buffer uses */
|
|
static inline size_t bitset_size_in_bytes(const bitset_t *bitset) {
|
|
return bitset->arraysize * sizeof(uint64_t);
|
|
}
|
|
|
|
/* returns how many bits can be accessed */
|
|
static inline size_t bitset_size_in_bits(const bitset_t *bitset) {
|
|
return bitset->arraysize * 64;
|
|
}
|
|
|
|
/* returns how many words (64-bit) of memory the backend buffer uses */
|
|
static inline size_t bitset_size_in_words(const bitset_t *bitset) {
|
|
return bitset->arraysize;
|
|
}
|
|
|
|
/* Grow the bitset so that it can support newarraysize * 64 bits with padding. Return TRUE in case of success, FALSE for failure. */
|
|
static inline int bitset_grow(bitset_t *bitset, size_t newarraysize) {
|
|
if (bitset->capacity < newarraysize) {
|
|
uint64_t *newarray;
|
|
bitset->capacity = newarraysize * 2;
|
|
if ((newarray = (uint64_t *)realloc(bitset->array, sizeof(uint64_t) * bitset->capacity)) == NULL) {
|
|
free(bitset->array);
|
|
return FALSE;
|
|
}
|
|
bitset->array = newarray;
|
|
}
|
|
memset(bitset->array + bitset->arraysize, 0, sizeof(uint64_t) * (newarraysize - bitset->arraysize));
|
|
bitset->arraysize = newarraysize;
|
|
return TRUE; // success!
|
|
}
|
|
|
|
/* attempts to recover unused memory, return FALSE in case of reallocation failure */
|
|
int bitset_trim(bitset_t *bitset);
|
|
|
|
/* shifts all bits by 's' positions so that the bitset representing values 1,2,10 would represent values 1+s, 2+s, 10+s */
|
|
void bitset_shift_left(bitset_t *bitset, size_t s);
|
|
|
|
/* shifts all bits by 's' positions so that the bitset representing values 1,2,10 would represent values 1-s, 2-s, 10-s, negative values are deleted */
|
|
void bitset_shift_right(bitset_t *bitset, size_t s);
|
|
|
|
/* Set the ith bit. Attempts to resize the bitset if needed (may silently fail) */
|
|
static inline void bitset_set(bitset_t *bitset, size_t i) {
|
|
size_t shiftedi = i >> 6;
|
|
if (shiftedi >= bitset->arraysize) {
|
|
if (!bitset_grow(bitset, shiftedi + 1)) {
|
|
return;
|
|
}
|
|
}
|
|
bitset->array[shiftedi] |= ((uint64_t)1) << (i % 64);
|
|
}
|
|
|
|
/* Set the ith bit to the specified value. Attempts to resize the bitset if needed (may silently fail) */
|
|
static inline void bitset_set_to_value(bitset_t *bitset, size_t i, int flag) {
|
|
size_t shiftedi = i >> 6;
|
|
uint64_t mask = ((uint64_t)1) << (i % 64);
|
|
uint64_t dynmask = ((uint64_t)flag) << (i % 64);
|
|
if (shiftedi >= bitset->arraysize) {
|
|
if (!bitset_grow(bitset, shiftedi + 1)) {
|
|
return;
|
|
}
|
|
}
|
|
uint64_t w = bitset->array[shiftedi];
|
|
w &= ~mask;
|
|
w |= dynmask;
|
|
bitset->array[shiftedi] = w;
|
|
}
|
|
|
|
static inline void bitset_cls_bit(bitset_t *bitset, size_t i) {
|
|
bitset_set_to_value(bitset, i, 0);
|
|
}
|
|
|
|
/* Get the value of the ith bit. */
|
|
static inline int bitset_get(const bitset_t *bitset, size_t i) {
|
|
size_t shiftedi = i >> 6;
|
|
if (shiftedi >= bitset->arraysize) {
|
|
return FALSE;
|
|
}
|
|
return (bitset->array[shiftedi] & (((uint64_t)1) << (i % 64))) != 0;
|
|
}
|
|
|
|
/* Count number of bits set. */
|
|
size_t bitset_count(const bitset_t *bitset);
|
|
|
|
/* Find the index of the first bit set. */
|
|
size_t bitset_minimum(const bitset_t *bitset);
|
|
|
|
/* Find the index of the last bit set. */
|
|
size_t bitset_maximum(const bitset_t *bitset);
|
|
|
|
/* compute the union in-place (to b1), returns TRUE if successful, to generate a new bitset first call bitset_copy */
|
|
int bitset_inplace_union(bitset_t *CBITSET_RESTRICT b1, const bitset_t *CBITSET_RESTRICT b2);
|
|
|
|
/* report the size of the union (without materializing it) */
|
|
size_t bitset_union_count(const bitset_t *CBITSET_RESTRICT b1, const bitset_t *CBITSET_RESTRICT b2);
|
|
|
|
/* compute the intersection in-place (to b1), to generate a new bitset first call bitset_copy */
|
|
void bitset_inplace_intersection(bitset_t *CBITSET_RESTRICT b1, const bitset_t *CBITSET_RESTRICT b2);
|
|
|
|
/* report the size of the intersection (without materializing it) */
|
|
size_t bitset_intersection_count(const bitset_t *CBITSET_RESTRICT b1, const bitset_t *CBITSET_RESTRICT b2);
|
|
|
|
/* returns TRUE if the bitsets contain no common elements */
|
|
int bitsets_disjoint(const bitset_t *b1, const bitset_t *b2);
|
|
|
|
/* returns TRUE if the bitsets contain any common elements */
|
|
int bitsets_intersect(const bitset_t *b1, const bitset_t *b2);
|
|
|
|
/* returns TRUE if b1 contains all the set bits of b2 */
|
|
int bitset_contains_all(const bitset_t *b1, const bitset_t *b2);
|
|
|
|
/* compute the difference in-place (to b1), to generate a new bitset first call bitset_copy */
|
|
void bitset_inplace_difference(bitset_t *CBITSET_RESTRICT b1, const bitset_t *CBITSET_RESTRICT b2);
|
|
|
|
/* compute the size of the difference */
|
|
size_t bitset_difference_count(const bitset_t *CBITSET_RESTRICT b1, const bitset_t *CBITSET_RESTRICT b2);
|
|
|
|
/* compute the symmetric difference in-place (to b1), return TRUE if successful, to generate a new bitset first call bitset_copy */
|
|
int bitset_inplace_symmetric_difference(bitset_t *CBITSET_RESTRICT b1, const bitset_t *CBITSET_RESTRICT b2);
|
|
|
|
/* compute the size of the symmetric difference */
|
|
size_t bitset_symmetric_difference_count(const bitset_t *CBITSET_RESTRICT b1, const bitset_t *CBITSET_RESTRICT b2);
|
|
|
|
/* iterate over the set bits
|
|
like so :
|
|
for(size_t i = 0; nextSetBit(b,&i) ; i++) {
|
|
//.....
|
|
}
|
|
*/
|
|
static inline int nextSetBit(const bitset_t *bitset, size_t *i) {
|
|
size_t x = *i >> 6;
|
|
if (x >= bitset->arraysize) {
|
|
return FALSE;
|
|
}
|
|
uint64_t w = bitset->array[x];
|
|
w >>= (*i & 63);
|
|
if (w != 0) {
|
|
*i += __builtin_ctzll(w);
|
|
return TRUE;
|
|
}
|
|
x++;
|
|
while (x < bitset->arraysize) {
|
|
w = bitset->array[x];
|
|
if (w != 0) {
|
|
*i = x * 64 + __builtin_ctzll(w);
|
|
return TRUE;
|
|
}
|
|
x++;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
/* Iterate over the set bits
|
|
like so :
|
|
size_t buffer[256];
|
|
size_t howmany = 0;
|
|
for(size_t startfrom = 0; (howmany = nextSetBits(b,buffer,256, &startfrom)) > 0 ; startfrom++) {
|
|
//.....
|
|
}
|
|
*/
|
|
static inline size_t nextSetBits(const bitset_t *bitset, size_t *buffer, size_t capacity, size_t *startfrom) {
|
|
if (capacity == 0) {
|
|
return 0; // sanity check
|
|
}
|
|
size_t x = *startfrom >> 6;
|
|
if (x >= bitset->arraysize) {
|
|
return 0; // nothing more to iterate over
|
|
}
|
|
uint64_t w = bitset->array[x];
|
|
w >>= (*startfrom & 63);
|
|
size_t howmany = 0;
|
|
size_t base = x << 6;
|
|
while (howmany < capacity) {
|
|
while (w != 0) {
|
|
uint64_t t = w & (~w + 1);
|
|
int r = __builtin_ctzll(w);
|
|
buffer[howmany++] = r + base;
|
|
if (howmany == capacity) {
|
|
goto end;
|
|
}
|
|
w ^= t;
|
|
}
|
|
x += 1;
|
|
if (x == bitset->arraysize) {
|
|
break;
|
|
}
|
|
base += 64;
|
|
w = bitset->array[x];
|
|
}
|
|
end:
|
|
if (howmany > 0) {
|
|
*startfrom = buffer[howmany - 1];
|
|
}
|
|
return howmany;
|
|
}
|
|
|
|
typedef int (*bitset_iterator)(size_t value, void *param);
|
|
|
|
// return TRUE if uninterrupted
|
|
static inline int bitset_for_each(const bitset_t *b, bitset_iterator iterator, void *ptr) {
|
|
size_t base = 0;
|
|
for (size_t i = 0; i < b->arraysize; ++i) {
|
|
uint64_t w = b->array[i];
|
|
while (w != 0) {
|
|
uint64_t t = w & (~w + 1);
|
|
int r = __builtin_ctzll(w);
|
|
if (!iterator(r + base, ptr)) {
|
|
return FALSE;
|
|
}
|
|
w ^= t;
|
|
}
|
|
base += 64;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
//static inline void bitset_print(const bitset_t *b) {
|
|
// printf("{");
|
|
// for (size_t i = 0; nextSetBit(b, &i); i++) {
|
|
// printf("%zu, ", i);
|
|
// }
|
|
// printf("}");
|
|
//}
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
#endif //VCPE_BITSET_H
|