Skip to content

Commit

Permalink
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Return Some for only one thread when calling remove() (crossbeam-…
Browse files Browse the repository at this point in the history
Coder-256 authored and al8n committed Dec 24, 2024

Verified

This commit was signed with the committer’s verified signature.
al8n Al Liu
1 parent 20c2d30 commit 7fd3a03
Showing 2 changed files with 58 additions and 1 deletion.
6 changes: 5 additions & 1 deletion crossbeam-skiplist/src/base.rs
Original file line number Diff line number Diff line change
@@ -1158,8 +1158,12 @@ where
break;
}
}
return Some(entry);
} else {
// The node has already been marked.
n.decrement(guard);
return None;
}
return Some(entry);
}
}
}
53 changes: 53 additions & 0 deletions crossbeam-skiplist/tests/base.rs
Original file line number Diff line number Diff line change
@@ -970,3 +970,56 @@ fn comparable_get() {
assert_eq!(ent.key().b, 2);
assert_eq!(*ent.value(), 12);
}

// https://github.com/crossbeam-rs/crossbeam/pull/1143
#[cfg(target_has_atomic = "32")]
#[test]
fn remove_race() {
use std::sync::atomic::{AtomicU32, Ordering};

let nthreads = 16;
#[cfg(miri)]
let key_range = 100u32;
#[cfg(not(miri))]
let key_range = 100_000u32;

let guard = &epoch::pin();
let s = SkipList::new(epoch::default_collector().clone());

for x in 0..key_range {
s.insert(x, (), guard).release(guard);
}

let barrier1 = AtomicU32::new(nthreads);
let barrier2 = AtomicU32::new(nthreads);
let mut total_removed = AtomicU32::new(0);

std::thread::scope(|scope| {
for _ in 0..nthreads {
scope.spawn(|| {
let guard = &epoch::pin();
let mut removed_entries = Vec::with_capacity(key_range as usize);

barrier1.fetch_sub(1, Ordering::Relaxed);
while barrier1.load(Ordering::Acquire) != 0 {}

for x in 0..key_range {
if let Some(entry) = s.remove(&x, guard) {
removed_entries.push(entry);
}
}

barrier2.fetch_sub(1, Ordering::Relaxed);
while barrier2.load(Ordering::Acquire) != 0 {}

total_removed.fetch_add(removed_entries.len() as u32, Ordering::Relaxed);

for entry in removed_entries.drain(..) {
entry.release(guard);
}
});
}
});

assert_eq!(*total_removed.get_mut(), key_range);
}

0 comments on commit 7fd3a03

Please sign in to comment.