| use std::sync::atomic::AtomicUsize; |
| use std::sync::atomic::Ordering::{AcqRel, Acquire, Relaxed}; |
| use std::time::{Duration, Instant}; |
| use crossbeam_epoch::{self as epoch, Atomic, Collector, LocalHandle, Owned, Shared}; |
| fn worker(a: Arc<Atomic<AtomicUsize>>, handle: LocalHandle) -> usize { |
| let mut rng = rand::thread_rng(); |
| thread::sleep(Duration::from_millis(1)); |
| let timeout = Duration::from_millis(rng.gen_range(0..10)); |
| let now = Instant::now(); |
| while now.elapsed() < timeout { |
| let guard = &handle.pin(); |
| let p = a.swap(Owned::new(AtomicUsize::new(sum)), AcqRel, guard); |
| let p = a.load(Acquire, guard); |
| unsafe { p.deref().fetch_add(sum, Relaxed) } |
| sum = sum.wrapping_add(val); |
| let collector = Collector::new(); |
| let a = Arc::new(Atomic::new(AtomicUsize::new(777))); |
| let c = collector.clone(); |
| thread::spawn(move || worker(a, c.register())) |
| a.swap(Shared::null(), AcqRel, epoch::unprotected()) |