// // Created by xajhuang on 2023/4/6. // #ifndef VCPE_BITSET_H #define VCPE_BITSET_H #include #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