From d400ba38fec4690bf71ae30b0981ab1042da6c66 Mon Sep 17 00:00:00 2001 From: Soerian Lieve Date: Sun, 21 Jan 2024 21:30:36 +0100 Subject: [PATCH 1/6] Add roaring64_bitmap_{add, remove}_range These are the open versions of `roaring64_bitmap_add_range_closed` and `roaring64_bitmap_remove_range_closed`. --- include/roaring/roaring64.h | 12 ++++++++++++ src/roaring64.c | 16 ++++++++++++++++ 2 files changed, 28 insertions(+) diff --git a/include/roaring/roaring64.h b/include/roaring/roaring64.h index ce9216fc6..b1491a722 100644 --- a/include/roaring/roaring64.h +++ b/include/roaring/roaring64.h @@ -102,6 +102,12 @@ void roaring64_bitmap_add_bulk(roaring64_bitmap_t *r, void roaring64_bitmap_add_many(roaring64_bitmap_t *r, size_t n_args, const uint64_t *vals); +/** + * Add all values in range [min, max). + */ +void roaring64_bitmap_add_range(roaring64_bitmap_t *r, uint64_t min, + uint64_t max); + /** * Add all values in range [min, max]. */ @@ -148,6 +154,12 @@ void roaring64_bitmap_remove_bulk(roaring64_bitmap_t *r, void roaring64_bitmap_remove_many(roaring64_bitmap_t *r, size_t n_args, const uint64_t *vals); +/** + * Remove all values in range [min, max). + */ +void roaring64_bitmap_remove_range(roaring64_bitmap_t *r, uint64_t min, + uint64_t max); + /** * Remove all values in range [min, max]. */ diff --git a/src/roaring64.c b/src/roaring64.c index 53914653b..d6031f3f7 100644 --- a/src/roaring64.c +++ b/src/roaring64.c @@ -297,6 +297,14 @@ static inline void add_range_closed_at(art_t *art, uint8_t *high48, art_insert(art, high48, (art_val_t *)leaf); } +void roaring64_bitmap_add_range(roaring64_bitmap_t *r, uint64_t min, + uint64_t max) { + if (min >= max) { + return; + } + roaring64_bitmap_add_range_closed(r, min, max - 1); +} + void roaring64_bitmap_add_range_closed(roaring64_bitmap_t *r, uint64_t min, uint64_t max) { if (min > max) { @@ -556,6 +564,14 @@ static inline void remove_range_closed_at(art_t *art, uint8_t *high48, } } +void roaring64_bitmap_remove_range(roaring64_bitmap_t *r, uint64_t min, + uint64_t max) { + if (min >= max) { + return; + } + roaring64_bitmap_remove_range_closed(r, min, max - 1); +} + void roaring64_bitmap_remove_range_closed(roaring64_bitmap_t *r, uint64_t min, uint64_t max) { if (min > max) { From 3a2584043dcd4dcab200b24dc522539c7e4a1c70 Mon Sep 17 00:00:00 2001 From: Soerian Lieve Date: Sun, 21 Jan 2024 21:32:58 +0100 Subject: [PATCH 2/6] Add roaring64_bitmap_contains_range --- include/roaring/roaring64.h | 6 +++++ src/roaring64.c | 51 +++++++++++++++++++++++++++++++++++++ tests/roaring64_unit.cpp | 50 ++++++++++++++++++++++++++++++++++++ 3 files changed, 107 insertions(+) diff --git a/include/roaring/roaring64.h b/include/roaring/roaring64.h index b1491a722..3d3f09843 100644 --- a/include/roaring/roaring64.h +++ b/include/roaring/roaring64.h @@ -171,6 +171,12 @@ void roaring64_bitmap_remove_range_closed(roaring64_bitmap_t *r, uint64_t min, */ bool roaring64_bitmap_contains(const roaring64_bitmap_t *r, uint64_t val); +/** + * Returns true if all values in the range [min, max) are present. + */ +bool roaring64_bitmap_contains_range(const roaring64_bitmap_t *r, uint64_t min, + uint64_t max); + /** * Check if an item is present using context from a previous insert or search * for faster search. diff --git a/src/roaring64.c b/src/roaring64.c index d6031f3f7..03d0caf47 100644 --- a/src/roaring64.c +++ b/src/roaring64.c @@ -346,6 +346,57 @@ bool roaring64_bitmap_contains(const roaring64_bitmap_t *r, uint64_t val) { return false; } +bool roaring64_bitmap_contains_range(const roaring64_bitmap_t *r, uint64_t min, + uint64_t max) { + if (min >= max) { + return true; + } + + uint8_t min_high48[ART_KEY_BYTES]; + uint16_t min_low16 = split_key(min, min_high48); + uint8_t max_high48[ART_KEY_BYTES]; + uint16_t max_low16 = split_key(max, max_high48); + uint64_t max_high48_bits = max & 0xFFFFFFFFFFFF0000; + + art_iterator_t it = art_lower_bound(&r->art, min_high48); + if (it.value == NULL) { + return false; + } + uint64_t prev_high48_bits = min & 0xFFFFFFFFFFFF0000; + while (it.value != NULL) { + uint64_t current_high48_bits = combine_key(it.key, 0); + if (current_high48_bits > max_high48_bits) { + return true; + } + if (current_high48_bits > prev_high48_bits + 0x10000) { + return false; + } + + leaf_t *leaf = (leaf_t *)it.value; + uint32_t container_min = 0; + if (compare_high48(it.key, min_high48) == 0) { + container_min = min_low16; + } + uint32_t container_max = 0xFFFF + 1; // Exclusive + if (compare_high48(it.key, max_high48) == 0) { + container_max = max_low16; + } + + // For the first and last containers we use container_contains_range, + // for the intermediate containers we can use container_is_full. + if (container_min == 0 && container_max == 0xFFFF + 1 && + !container_is_full(leaf->container, leaf->typecode)) { + return false; + } else if (!container_contains_range(leaf->container, container_min, + container_max, leaf->typecode)) { + return false; + } + prev_high48_bits = current_high48_bits; + art_iterator_next(&it); + } + return true; +} + bool roaring64_bitmap_contains_bulk(const roaring64_bitmap_t *r, roaring64_bulk_context_t *context, uint64_t val) { diff --git a/tests/roaring64_unit.cpp b/tests/roaring64_unit.cpp index 8a8a78b11..b0177c296 100644 --- a/tests/roaring64_unit.cpp +++ b/tests/roaring64_unit.cpp @@ -248,6 +248,55 @@ DEFINE_TEST(test_contains_bulk) { roaring64_bitmap_free(r); } +DEFINE_TEST(test_contains_range) { + { + // Empty bitmap. + roaring64_bitmap_t* r = roaring64_bitmap_create(); + assert_false(roaring64_bitmap_contains_range(r, 1, 10)); + roaring64_bitmap_free(r); + } + { + // Empty range. + roaring64_bitmap_t* r = roaring64_bitmap_create(); + roaring64_bitmap_add_range(r, 1, 10); + assert_true(roaring64_bitmap_contains_range(r, 1, 1)); + roaring64_bitmap_free(r); + } + { + // Range within one container. + roaring64_bitmap_t* r = roaring64_bitmap_create(); + roaring64_bitmap_add_range(r, 1, 10); + assert_true(roaring64_bitmap_contains_range(r, 1, 10)); + assert_false(roaring64_bitmap_contains_range(r, 1, 11)); + roaring64_bitmap_free(r); + } + { + // Range across two containers. + roaring64_bitmap_t* r = roaring64_bitmap_create(); + roaring64_bitmap_add_range(r, 1, (1 << 16) + 10); + assert_true(roaring64_bitmap_contains_range(r, 1, (1 << 16) + 10)); + assert_true(roaring64_bitmap_contains_range(r, 1, (1 << 16) - 1)); + assert_false(roaring64_bitmap_contains_range(r, 1, (1 << 16) + 11)); + roaring64_bitmap_free(r); + } + { + // Range across three containers. + roaring64_bitmap_t* r = roaring64_bitmap_create(); + roaring64_bitmap_add_range(r, 1, (2 << 16) + 10); + assert_true(roaring64_bitmap_contains_range(r, 1, (2 << 16) + 10)); + assert_false(roaring64_bitmap_contains_range(r, 1, (2 << 16) + 11)); + roaring64_bitmap_free(r); + } + { + // Container missing from range. + roaring64_bitmap_t* r = roaring64_bitmap_create(); + roaring64_bitmap_add_range(r, 1, (1 << 16) - 1); + roaring64_bitmap_add_range(r, (2 << 16), (3 << 16) - 1); + assert_false(roaring64_bitmap_contains_range(r, 1, (3 << 16) - 1)); + roaring64_bitmap_free(r); + } +} + DEFINE_TEST(test_select) { roaring64_bitmap_t* r = roaring64_bitmap_create(); for (uint64_t i = 0; i < 100; ++i) { @@ -1253,6 +1302,7 @@ int main() { cmocka_unit_test(test_add_bulk), cmocka_unit_test(test_add_many), cmocka_unit_test(test_add_range_closed), + cmocka_unit_test(test_contains_range), cmocka_unit_test(test_contains_bulk), cmocka_unit_test(test_select), cmocka_unit_test(test_rank), From 937608aa9d3418fa8396f8e59a78ed4a04735a12 Mon Sep 17 00:00:00 2001 From: Soerian Lieve Date: Sun, 21 Jan 2024 21:34:04 +0100 Subject: [PATCH 3/6] Add roaring64_bitmap_flip{_closed}{_inplace} --- include/roaring/roaring64.h | 31 +++++++ src/roaring64.c | 174 +++++++++++++++++++++++++++++++++++- tests/roaring64_unit.cpp | 93 +++++++++++++++++++ 3 files changed, 297 insertions(+), 1 deletion(-) diff --git a/include/roaring/roaring64.h b/include/roaring/roaring64.h index 3d3f09843..554a11f30 100644 --- a/include/roaring/roaring64.h +++ b/include/roaring/roaring64.h @@ -383,6 +383,37 @@ uint64_t roaring64_bitmap_andnot_cardinality(const roaring64_bitmap_t *r1, void roaring64_bitmap_andnot_inplace(roaring64_bitmap_t *r1, const roaring64_bitmap_t *r2); +/** + * Compute the negation of the bitmap in the interval [min, max). + * The number of negated values is `max - min`. Areas outside the range are + * passed through unchanged. + */ +roaring64_bitmap_t *roaring64_bitmap_flip(const roaring64_bitmap_t *r, + uint64_t min, uint64_t max); + +/** + * Compute the negation of the bitmap in the interval [min, max]. + * The number of negated values is `max - min + 1`. Areas outside the range are + * passed through unchanged. + */ +roaring64_bitmap_t *roaring64_bitmap_flip_closed(const roaring64_bitmap_t *r, + uint64_t min, uint64_t max); + +/** + * In-place version of `roaring64_bitmap_flip`. Compute the negation of the + * bitmap in the interval [min, max). The number of negated values is `max - + * min`. Areas outside the range are passed through unchanged. + */ +void roaring64_bitmap_flip_inplace(roaring64_bitmap_t *r, uint64_t min, + uint64_t max); +/** + * In-place version of `roaring64_bitmap_flip_closed`. Compute the negation of + * the bitmap in the interval [min, max]. The number of negated values is `max - + * min + 1`. Areas outside the range are passed through unchanged. + */ +void roaring64_bitmap_flip_closed_inplace(roaring64_bitmap_t *r, uint64_t min, + uint64_t max); + /** * Iterate over the bitmap elements. The function `iterator` is called once for * all the values with `ptr` (can be NULL) as the second parameter of each call. diff --git a/src/roaring64.c b/src/roaring64.c index 03d0caf47..f63754b7c 100644 --- a/src/roaring64.c +++ b/src/roaring64.c @@ -1395,12 +1395,184 @@ void roaring64_bitmap_andnot_inplace(roaring64_bitmap_t *r1, } } +/** + * Flips the leaf at high48 in the range [min, max), returning a new leaf with a + * new container. If the high48 key is not found in the existing bitmap, a new + * container is created. Returns null if the negation results in an empty range. + */ +static leaf_t *roaring64_flip_leaf(const roaring64_bitmap_t *r, + uint8_t high48[], uint32_t min, + uint32_t max) { + leaf_t *leaf1 = (leaf_t *)art_find(&r->art, high48); + container_t *container2; + uint8_t typecode2; + if (leaf1 == NULL) { + container2 = container_range_of_ones(min, max, &typecode2); + } else if (min == 0 && max > 0xFFFF) { + container2 = + container_not(leaf1->container, leaf1->typecode, &typecode2); + } else { + container2 = container_not_range(leaf1->container, leaf1->typecode, min, + max, &typecode2); + } + if (container_nonzero_cardinality(container2, typecode2)) { + return create_leaf(container2, typecode2); + } + container_free(container2, typecode2); + return NULL; +} + +/** + * Flips the leaf at high48 in the range [min, max). If the high48 key is not + * found in the bitmap, a new container is created. Deletes the leaf and + * associated container if the negation results in an empty range. + */ +static void roaring64_flip_leaf_inplace(roaring64_bitmap_t *r, uint8_t high48[], + uint32_t min, uint32_t max) { + leaf_t *leaf = (leaf_t *)art_find(&r->art, high48); + container_t *container2; + uint8_t typecode2; + if (leaf == NULL) { + // No container at this key, insert a full container. + container2 = container_range_of_ones(min, max, &typecode2); + art_insert(&r->art, high48, + (art_val_t *)create_leaf(container2, typecode2)); + return; + } + + if (min == 0 && max > 0xFFFF) { + container2 = + container_inot(leaf->container, leaf->typecode, &typecode2); + } else { + container2 = container_inot_range(leaf->container, leaf->typecode, min, + max, &typecode2); + } + + leaf->container = container2; + leaf->typecode = typecode2; + + if (!container_nonzero_cardinality(leaf->container, leaf->typecode)) { + art_erase(&r->art, high48); + container_free(leaf->container, leaf->typecode); + free_leaf(leaf); + } +} + +roaring64_bitmap_t *roaring64_bitmap_flip(const roaring64_bitmap_t *r, + uint64_t min, uint64_t max) { + if (min >= max) { + return roaring64_bitmap_copy(r); + } + return roaring64_bitmap_flip_closed(r, min, max - 1); +} + +roaring64_bitmap_t *roaring64_bitmap_flip_closed(const roaring64_bitmap_t *r1, + uint64_t min, uint64_t max) { + if (min > max) { + return roaring64_bitmap_copy(r1); + } + uint8_t min_high48_key[ART_KEY_BYTES]; + uint16_t min_low16 = split_key(min, min_high48_key); + uint8_t max_high48_key[ART_KEY_BYTES]; + uint16_t max_low16 = split_key(max, max_high48_key); + uint64_t min_high48_bits = (min & 0xFFFFFFFFFFFF0000ULL) >> 16; + uint64_t max_high48_bits = (max & 0xFFFFFFFFFFFF0000ULL) >> 16; + + roaring64_bitmap_t *r2 = roaring64_bitmap_create(); + art_iterator_t it = art_init_iterator(&r1->art, /*first=*/true); + + // Copy the containers before min unchanged. + while (it.value != NULL && compare_high48(it.key, min_high48_key) < 0) { + leaf_t *leaf1 = (leaf_t *)it.value; + uint8_t typecode2 = leaf1->typecode; + container_t *container2 = get_copy_of_container( + leaf1->container, &typecode2, /*copy_on_write=*/false); + art_insert(&r2->art, it.key, + (art_val_t *)create_leaf(container2, typecode2)); + art_iterator_next(&it); + } + + // Flip the range (including non-existent containers!) between min and max. + for (uint64_t high48_bits = min_high48_bits; high48_bits <= max_high48_bits; + high48_bits++) { + uint8_t current_high48_key[ART_KEY_BYTES]; + split_key(high48_bits << 16, current_high48_key); + + uint32_t min_container = 0; + if (high48_bits == min_high48_bits) { + min_container = min_low16; + } + uint32_t max_container = ((uint32_t)0xFFFF) + 1; // Exclusive range. + if (high48_bits == max_high48_bits) { + max_container = (uint32_t)(max_low16) + 1; // Exclusive. + } + + leaf_t *leaf = roaring64_flip_leaf(r1, current_high48_key, + min_container, max_container); + if (leaf != NULL) { + art_insert(&r2->art, current_high48_key, (art_val_t *)leaf); + } + } + + // Copy the containers after max unchanged. + it = art_upper_bound(&r1->art, max_high48_key); + while (it.value != NULL) { + leaf_t *leaf1 = (leaf_t *)it.value; + uint8_t typecode2 = leaf1->typecode; + container_t *container2 = get_copy_of_container( + leaf1->container, &typecode2, /*copy_on_write=*/false); + art_insert(&r2->art, it.key, + (art_val_t *)create_leaf(container2, typecode2)); + art_iterator_next(&it); + } + + return r2; +} + +void roaring64_bitmap_flip_inplace(roaring64_bitmap_t *r, uint64_t min, + uint64_t max) { + if (min >= max) { + return; + } + roaring64_bitmap_flip_closed_inplace(r, min, max - 1); +} + +void roaring64_bitmap_flip_closed_inplace(roaring64_bitmap_t *r, uint64_t min, + uint64_t max) { + if (min > max) { + return; + } + uint16_t min_low16 = (uint16_t)min; + uint16_t max_low16 = (uint16_t)max; + uint64_t min_high48_bits = (min & 0xFFFFFFFFFFFF0000ULL) >> 16; + uint64_t max_high48_bits = (max & 0xFFFFFFFFFFFF0000ULL) >> 16; + + // Flip the range (including non-existent containers!) between min and max. + for (uint64_t high48_bits = min_high48_bits; high48_bits <= max_high48_bits; + high48_bits++) { + uint8_t current_high48_key[ART_KEY_BYTES]; + split_key(high48_bits << 16, current_high48_key); + + uint32_t min_container = 0; + if (high48_bits == min_high48_bits) { + min_container = min_low16; + } + uint32_t max_container = ((uint32_t)0xFFFF) + 1; // Exclusive range. + if (high48_bits == max_high48_bits) { + max_container = (uint32_t)(max_low16) + 1; // Exclusive. + } + + roaring64_flip_leaf_inplace(r, current_high48_key, min_container, + max_container); + } +} + bool roaring64_bitmap_iterate(const roaring64_bitmap_t *r, roaring_iterator64 iterator, void *ptr) { art_iterator_t it = art_init_iterator(&r->art, /*first=*/true); while (it.value != NULL) { uint64_t high48 = combine_key(it.key, 0); - uint64_t high32 = high48 & 0xFFFFFFFF00000000; + uint64_t high32 = high48 & 0xFFFFFFFF00000000ULL; uint32_t low32 = high48; leaf_t *leaf = (leaf_t *)it.value; if (!container_iterate64(leaf->container, leaf->typecode, low32, diff --git a/tests/roaring64_unit.cpp b/tests/roaring64_unit.cpp index b0177c296..e247f94b6 100644 --- a/tests/roaring64_unit.cpp +++ b/tests/roaring64_unit.cpp @@ -1001,6 +1001,97 @@ DEFINE_TEST(test_andnot_inplace) { } } +DEFINE_TEST(test_flip) { + { + // Flipping an empty bitmap should result in a non-empty range. + roaring64_bitmap_t* r1 = roaring64_bitmap_create(); + roaring64_bitmap_t* r2 = roaring64_bitmap_flip(r1, 10, 100000); + assert_true(roaring64_bitmap_contains_range(r2, 10, 100000)); + + roaring64_bitmap_free(r1); + roaring64_bitmap_free(r2); + } + { + // Only the specified range should be flipped. + roaring64_bitmap_t* r1 = roaring64_bitmap_of(3, 1, 3, 6); + roaring64_bitmap_t* r2 = roaring64_bitmap_flip(r1, 2, 5); + roaring64_bitmap_t* r3 = roaring64_bitmap_of(4, 1, 2, 4, 6); + assert_true(roaring64_bitmap_equals(r2, r3)); + + roaring64_bitmap_free(r1); + roaring64_bitmap_free(r2); + roaring64_bitmap_free(r3); + } + { + // An empty range does nothing. + roaring64_bitmap_t* r1 = roaring64_bitmap_of(3, 1, 3, 6); + roaring64_bitmap_t* r2 = roaring64_bitmap_flip(r1, 3, 3); + assert_true(roaring64_bitmap_equals(r2, r1)); + + roaring64_bitmap_free(r1); + roaring64_bitmap_free(r2); + } + { + // A bitmap with values in all affected containers. + roaring64_bitmap_t* r1 = + roaring64_bitmap_of(3, (2 << 16), (3 << 16) + 1, (4 << 16) + 3); + roaring64_bitmap_t* r2 = + roaring64_bitmap_flip(r1, (2 << 16), (4 << 16) + 4); + roaring64_bitmap_t* r3 = + roaring64_bitmap_from_range((2 << 16) + 1, (4 << 16) + 3, 1); + roaring64_bitmap_remove(r3, (3 << 16) + 1); + assert_true(roaring64_bitmap_equals(r2, r3)); + + roaring64_bitmap_free(r1); + roaring64_bitmap_free(r2); + roaring64_bitmap_free(r3); + } +} + +DEFINE_TEST(test_flip_inplace) { + { + // Flipping an empty bitmap should result in a non-empty range. + roaring64_bitmap_t* r1 = roaring64_bitmap_create(); + roaring64_bitmap_flip_inplace(r1, 10, 100000); + assert_true(roaring64_bitmap_contains_range(r1, 10, 100000)); + + roaring64_bitmap_free(r1); + } + { + // Only the specified range should be flipped. + roaring64_bitmap_t* r1 = roaring64_bitmap_of(3, 1, 3, 6); + roaring64_bitmap_flip_inplace(r1, 2, 5); + roaring64_bitmap_t* r2 = roaring64_bitmap_of(4, 1, 2, 4, 6); + assert_true(roaring64_bitmap_equals(r1, r2)); + + roaring64_bitmap_free(r1); + roaring64_bitmap_free(r2); + } + { + // An empty range does nothing. + roaring64_bitmap_t* r1 = roaring64_bitmap_of(3, 1, 3, 6); + roaring64_bitmap_flip_inplace(r1, 3, 3); + roaring64_bitmap_t* r2 = roaring64_bitmap_of(3, 1, 3, 6); + assert_true(roaring64_bitmap_equals(r1, r2)); + + roaring64_bitmap_free(r1); + roaring64_bitmap_free(r2); + } + { + // A bitmap with values in all affected containers. + roaring64_bitmap_t* r1 = + roaring64_bitmap_of(3, (2 << 16), (3 << 16) + 1, (4 << 16) + 3); + roaring64_bitmap_flip_inplace(r1, (2 << 16), (4 << 16) + 4); + roaring64_bitmap_t* r2 = + roaring64_bitmap_from_range((2 << 16) + 1, (4 << 16) + 3, 1); + roaring64_bitmap_remove(r2, (3 << 16) + 1); + assert_true(roaring64_bitmap_equals(r1, r2)); + + roaring64_bitmap_free(r1); + roaring64_bitmap_free(r2); + } +} + bool roaring_iterator64_sumall(uint64_t value, void* param) { *(uint64_t*)param += value; return true; @@ -1334,6 +1425,8 @@ int main() { cmocka_unit_test(test_andnot), cmocka_unit_test(test_andnot_cardinality), cmocka_unit_test(test_andnot_inplace), + cmocka_unit_test(test_flip), + cmocka_unit_test(test_flip_inplace), cmocka_unit_test(test_iterate), cmocka_unit_test(test_iterator_create), cmocka_unit_test(test_iterator_create_last), From 72832b8fa82077f944b9cfab29e2bdd3e5fd49cb Mon Sep 17 00:00:00 2001 From: Soerian Lieve Date: Mon, 22 Jan 2024 23:16:45 +0100 Subject: [PATCH 4/6] Use ULL integers for bitmap_of in flip tests --- src/roaring64.c | 8 ++++---- tests/roaring64_unit.cpp | 22 +++++++++++----------- 2 files changed, 15 insertions(+), 15 deletions(-) diff --git a/src/roaring64.c b/src/roaring64.c index f63754b7c..ad936d7b6 100644 --- a/src/roaring64.c +++ b/src/roaring64.c @@ -1502,9 +1502,9 @@ roaring64_bitmap_t *roaring64_bitmap_flip_closed(const roaring64_bitmap_t *r1, if (high48_bits == min_high48_bits) { min_container = min_low16; } - uint32_t max_container = ((uint32_t)0xFFFF) + 1; // Exclusive range. + uint32_t max_container = 0xFFFF + 1; // Exclusive range. if (high48_bits == max_high48_bits) { - max_container = (uint32_t)(max_low16) + 1; // Exclusive. + max_container = max_low16 + 1; // Exclusive. } leaf_t *leaf = roaring64_flip_leaf(r1, current_high48_key, @@ -1557,9 +1557,9 @@ void roaring64_bitmap_flip_closed_inplace(roaring64_bitmap_t *r, uint64_t min, if (high48_bits == min_high48_bits) { min_container = min_low16; } - uint32_t max_container = ((uint32_t)0xFFFF) + 1; // Exclusive range. + uint32_t max_container = 0xFFFF + 1; // Exclusive range. if (high48_bits == max_high48_bits) { - max_container = (uint32_t)(max_low16) + 1; // Exclusive. + max_container = max_low16 + 1; // Exclusive. } roaring64_flip_leaf_inplace(r, current_high48_key, min_container, diff --git a/tests/roaring64_unit.cpp b/tests/roaring64_unit.cpp index e247f94b6..817177c42 100644 --- a/tests/roaring64_unit.cpp +++ b/tests/roaring64_unit.cpp @@ -1013,9 +1013,9 @@ DEFINE_TEST(test_flip) { } { // Only the specified range should be flipped. - roaring64_bitmap_t* r1 = roaring64_bitmap_of(3, 1, 3, 6); + roaring64_bitmap_t* r1 = roaring64_bitmap_of(3, 1ULL, 3ULL, 6ULL); roaring64_bitmap_t* r2 = roaring64_bitmap_flip(r1, 2, 5); - roaring64_bitmap_t* r3 = roaring64_bitmap_of(4, 1, 2, 4, 6); + roaring64_bitmap_t* r3 = roaring64_bitmap_of(4, 1ULL, 2ULL, 4ULL, 6ULL); assert_true(roaring64_bitmap_equals(r2, r3)); roaring64_bitmap_free(r1); @@ -1024,7 +1024,7 @@ DEFINE_TEST(test_flip) { } { // An empty range does nothing. - roaring64_bitmap_t* r1 = roaring64_bitmap_of(3, 1, 3, 6); + roaring64_bitmap_t* r1 = roaring64_bitmap_of(3, 1ULL, 3ULL, 6); roaring64_bitmap_t* r2 = roaring64_bitmap_flip(r1, 3, 3); assert_true(roaring64_bitmap_equals(r2, r1)); @@ -1033,8 +1033,8 @@ DEFINE_TEST(test_flip) { } { // A bitmap with values in all affected containers. - roaring64_bitmap_t* r1 = - roaring64_bitmap_of(3, (2 << 16), (3 << 16) + 1, (4 << 16) + 3); + roaring64_bitmap_t* r1 = roaring64_bitmap_of( + 3, (2ULL << 16), (3ULL << 16) + 1, (4ULL << 16) + 3); roaring64_bitmap_t* r2 = roaring64_bitmap_flip(r1, (2 << 16), (4 << 16) + 4); roaring64_bitmap_t* r3 = @@ -1059,9 +1059,9 @@ DEFINE_TEST(test_flip_inplace) { } { // Only the specified range should be flipped. - roaring64_bitmap_t* r1 = roaring64_bitmap_of(3, 1, 3, 6); + roaring64_bitmap_t* r1 = roaring64_bitmap_of(3, 1ULL, 3ULL, 6ULL); roaring64_bitmap_flip_inplace(r1, 2, 5); - roaring64_bitmap_t* r2 = roaring64_bitmap_of(4, 1, 2, 4, 6); + roaring64_bitmap_t* r2 = roaring64_bitmap_of(4, 1ULL, 2ULL, 4ULL, 6ULL); assert_true(roaring64_bitmap_equals(r1, r2)); roaring64_bitmap_free(r1); @@ -1069,9 +1069,9 @@ DEFINE_TEST(test_flip_inplace) { } { // An empty range does nothing. - roaring64_bitmap_t* r1 = roaring64_bitmap_of(3, 1, 3, 6); + roaring64_bitmap_t* r1 = roaring64_bitmap_of(3, 1ULL, 3ULL, 6ULL); roaring64_bitmap_flip_inplace(r1, 3, 3); - roaring64_bitmap_t* r2 = roaring64_bitmap_of(3, 1, 3, 6); + roaring64_bitmap_t* r2 = roaring64_bitmap_of(3, 1ULL, 3ULL, 6ULL); assert_true(roaring64_bitmap_equals(r1, r2)); roaring64_bitmap_free(r1); @@ -1079,8 +1079,8 @@ DEFINE_TEST(test_flip_inplace) { } { // A bitmap with values in all affected containers. - roaring64_bitmap_t* r1 = - roaring64_bitmap_of(3, (2 << 16), (3 << 16) + 1, (4 << 16) + 3); + roaring64_bitmap_t* r1 = roaring64_bitmap_of( + 3, (2ULL << 16), (3ULL << 16) + 1, (4ULL << 16) + 3); roaring64_bitmap_flip_inplace(r1, (2 << 16), (4 << 16) + 4); roaring64_bitmap_t* r2 = roaring64_bitmap_from_range((2 << 16) + 1, (4 << 16) + 3, 1); From d993f13daef982f564451b86a599d7dcbb5cb999 Mon Sep 17 00:00:00 2001 From: Soerian Lieve Date: Tue, 23 Jan 2024 21:29:37 +0100 Subject: [PATCH 5/6] Check final iterator state --- src/roaring64.c | 11 +++++++---- tests/roaring64_unit.cpp | 7 +++++++ 2 files changed, 14 insertions(+), 4 deletions(-) diff --git a/src/roaring64.c b/src/roaring64.c index ad936d7b6..6dd76dd63 100644 --- a/src/roaring64.c +++ b/src/roaring64.c @@ -366,6 +366,8 @@ bool roaring64_bitmap_contains_range(const roaring64_bitmap_t *r, uint64_t min, while (it.value != NULL) { uint64_t current_high48_bits = combine_key(it.key, 0); if (current_high48_bits > max_high48_bits) { + // We've passed the end of the range with all containers containing + // the range. return true; } if (current_high48_bits > prev_high48_bits + 0x10000) { @@ -384,9 +386,10 @@ bool roaring64_bitmap_contains_range(const roaring64_bitmap_t *r, uint64_t min, // For the first and last containers we use container_contains_range, // for the intermediate containers we can use container_is_full. - if (container_min == 0 && container_max == 0xFFFF + 1 && - !container_is_full(leaf->container, leaf->typecode)) { - return false; + if (container_min == 0 && container_max == 0xFFFF + 1) { + if (!container_is_full(leaf->container, leaf->typecode)) { + return false; + } } else if (!container_contains_range(leaf->container, container_min, container_max, leaf->typecode)) { return false; @@ -394,7 +397,7 @@ bool roaring64_bitmap_contains_range(const roaring64_bitmap_t *r, uint64_t min, prev_high48_bits = current_high48_bits; art_iterator_next(&it); } - return true; + return prev_high48_bits == max_high48_bits; } bool roaring64_bitmap_contains_bulk(const roaring64_bitmap_t *r, diff --git a/tests/roaring64_unit.cpp b/tests/roaring64_unit.cpp index 817177c42..b8a4c748f 100644 --- a/tests/roaring64_unit.cpp +++ b/tests/roaring64_unit.cpp @@ -295,6 +295,13 @@ DEFINE_TEST(test_contains_range) { assert_false(roaring64_bitmap_contains_range(r, 1, (3 << 16) - 1)); roaring64_bitmap_free(r); } + { + // Range larger than bitmap. + roaring64_bitmap_t* r = roaring64_bitmap_create(); + roaring64_bitmap_add_range(r, 1, 1 << 16); + assert_false(roaring64_bitmap_contains_range(r, 1, (1 << 16))); + roaring64_bitmap_free(r); + } } DEFINE_TEST(test_select) { From da9e15dee2ac720839b967690cf4e2df683581c3 Mon Sep 17 00:00:00 2001 From: Soerian Lieve Date: Tue, 23 Jan 2024 21:33:46 +0100 Subject: [PATCH 6/6] Add a few comments --- src/roaring64.c | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/src/roaring64.c b/src/roaring64.c index 6dd76dd63..293b2de16 100644 --- a/src/roaring64.c +++ b/src/roaring64.c @@ -371,6 +371,7 @@ bool roaring64_bitmap_contains_range(const roaring64_bitmap_t *r, uint64_t min, return true; } if (current_high48_bits > prev_high48_bits + 0x10000) { + // There is a gap in the iterator that falls in the range. return false; } @@ -1410,11 +1411,14 @@ static leaf_t *roaring64_flip_leaf(const roaring64_bitmap_t *r, container_t *container2; uint8_t typecode2; if (leaf1 == NULL) { + // No container at this key, create a full container. container2 = container_range_of_ones(min, max, &typecode2); } else if (min == 0 && max > 0xFFFF) { + // Flip whole container. container2 = container_not(leaf1->container, leaf1->typecode, &typecode2); } else { + // Partially flip a container. container2 = container_not_range(leaf1->container, leaf1->typecode, min, max, &typecode2); } @@ -1444,9 +1448,11 @@ static void roaring64_flip_leaf_inplace(roaring64_bitmap_t *r, uint8_t high48[], } if (min == 0 && max > 0xFFFF) { + // Flip whole container. container2 = container_inot(leaf->container, leaf->typecode, &typecode2); } else { + // Partially flip a container. container2 = container_inot_range(leaf->container, leaf->typecode, min, max, &typecode2); }