Skip to content

Commit 6c4ccfd

Browse files
committed
Fix race between critnib_release() and free_leaf() in critnib
Fix race between critnib_release() and free_leaf() in critnib: critnib_release() decremented ref_count to 0 and (before it called c->cb_free_leaf(k->to_be_freed)) free_leaf() added this leaf to the c->deleted_leaf list and alloc_leaf() reused it and zeroed k->to_be_freed before it could be freed in critnib_release(). This patch fixes this issue. Signed-off-by: Lukasz Dorau <[email protected]>
1 parent fce2b13 commit 6c4ccfd

File tree

1 file changed

+22
-11
lines changed

1 file changed

+22
-11
lines changed

src/critnib/critnib.c

Lines changed: 22 additions & 11 deletions
Original file line numberDiff line numberDiff line change
@@ -331,18 +331,26 @@ static void free_leaf(struct critnib *__restrict c,
331331
return;
332332
}
333333

334+
// k should be added to the c->deleted_leaf list here
335+
// or in critnib_release() when the reference count drops to 0.
336+
utils_atomic_store_release_u8(&k->pending_deleted_leaf, 1);
337+
334338
if (c->cb_free_leaf) {
335339
uint64_t ref_count;
336340
utils_atomic_load_acquire_u64(&k->ref_count, &ref_count);
337341
if (ref_count > 0) {
338-
// k will be added to c->deleted_leaf in critnib_release()
342+
// k will be added to the c->deleted_leaf list in critnib_release()
339343
// when the reference count drops to 0.
340-
utils_atomic_store_release_u8(&k->pending_deleted_leaf, 1);
341344
return;
342345
}
343346
}
344347

345-
add_to_deleted_leaf_list(c, k);
348+
uint8_t expected = 1;
349+
uint8_t desired = 0;
350+
if (utils_compare_exchange_u8(&k->pending_deleted_leaf, &expected,
351+
&desired)) {
352+
add_to_deleted_leaf_list(c, k);
353+
}
346354
}
347355

348356
/*
@@ -392,8 +400,8 @@ int critnib_insert(struct critnib *c, word key, void *value, int update) {
392400
utils_atomic_store_release_u8(&k->pending_deleted_leaf, 0);
393401

394402
if (c->cb_free_leaf) {
395-
// mark the leaf as valid (ref_count == 1)
396-
utils_atomic_store_release_u64(&k->ref_count, 1ULL);
403+
// mark the leaf as valid (ref_count == 2)
404+
utils_atomic_store_release_u64(&k->ref_count, 2ULL);
397405
} else {
398406
// the reference counter is not used in this case
399407
utils_atomic_store_release_u64(&k->ref_count, 0ULL);
@@ -609,18 +617,21 @@ int critnib_release(struct critnib *c, void *ref) {
609617
}
610618

611619
/* decrement the reference count */
612-
if (utils_atomic_decrement_u64(&k->ref_count) == 0) {
620+
if (utils_atomic_decrement_u64(&k->ref_count) == 1) {
613621
void *to_be_freed = NULL;
614622
utils_atomic_load_acquire_ptr(&k->to_be_freed, &to_be_freed);
615623
if (to_be_freed) {
616624
utils_atomic_store_release_ptr(&k->to_be_freed, NULL);
617625
c->cb_free_leaf(c->leaf_allocator, to_be_freed);
618626
}
619-
uint8_t pending_deleted_leaf;
620-
utils_atomic_load_acquire_u8(&k->pending_deleted_leaf,
621-
&pending_deleted_leaf);
622-
if (pending_deleted_leaf) {
623-
utils_atomic_store_release_u8(&k->pending_deleted_leaf, 0);
627+
628+
// mark the leaf as not used (ref_count == 0)
629+
utils_atomic_store_release_u64(&k->ref_count, 0ULL);
630+
631+
uint8_t expected = 1;
632+
uint8_t desired = 0;
633+
if (utils_compare_exchange_u8(&k->pending_deleted_leaf, &expected,
634+
&desired)) {
624635
add_to_deleted_leaf_list(c, k);
625636
}
626637
}

0 commit comments

Comments
 (0)