use std::sync::{Arc, Barrier}; use std::thread; use criterion::{criterion_group, criterion_main, Criterion}; const THREADS: usize = 16; const ITEMS: usize = 1000; fn treiber_stack(c: &mut Criterion) { c.bench_function("trieber_stack-haphazard", |b| { b.iter(run::>) }); c.bench_function("trieber_stack-crossbeam", |b| { b.iter(run::>) }); c.bench_function("trieber_stack-seize", |b| { b.iter(run::>) }); } trait Stack { fn new() -> Self; fn push(&self, value: T); fn pop(&self) -> Option; fn is_empty(&self) -> bool; } fn run() where T: Stack + Send + Sync + 'static, { let stack = Arc::new(T::new()); let barrier = Arc::new(Barrier::new(THREADS)); let handles = (0..THREADS - 1) .map(|_| { let stack = stack.clone(); let barrier = barrier.clone(); thread::spawn(move || { barrier.wait(); for i in 0..ITEMS { stack.push(i); assert!(stack.pop().is_some()); } }) }) .collect::>(); barrier.wait(); for i in 0..ITEMS { stack.push(i); assert!(stack.pop().is_some()); } for handle in handles { handle.join().unwrap(); } assert!(stack.pop().is_none()); assert!(stack.is_empty()); } criterion_group!(benches, treiber_stack); criterion_main!(benches); mod seize_stack { use super::Stack; use seize::{reclaim, Collector, Guard, Linked}; use std::mem::ManuallyDrop; use std::ptr::{self, NonNull}; use std::sync::atomic::{AtomicPtr, Ordering}; #[derive(Debug)] pub struct TreiberStack { head: AtomicPtr>>, collector: Collector, } #[derive(Debug)] struct Node { data: ManuallyDrop, next: *mut Linked>, } impl Stack for TreiberStack { fn new() -> TreiberStack { TreiberStack { head: AtomicPtr::new(ptr::null_mut()), collector: Collector::new().epoch_frequency(None), } } fn push(&self, value: T) { let node = self.collector.link_boxed(Node { data: ManuallyDrop::new(value), next: ptr::null_mut(), }); let guard = self.collector.enter(); loop { let head = guard.protect(&self.head, Ordering::Relaxed); unsafe { (*node).next = head } if self .head .compare_exchange(head, node, Ordering::Release, Ordering::Relaxed) .is_ok() { break; } } } fn pop(&self) -> Option { let guard = self.collector.enter(); loop { let head = NonNull::new(guard.protect(&self.head, Ordering::Acquire))?.as_ptr(); let next = unsafe { (*head).next }; if self .head .compare_exchange(head, next, Ordering::Relaxed, Ordering::Relaxed) .is_ok() { unsafe { let data = ptr::read(&(*head).data); self.collector .retire(head, reclaim::boxed::>>); return Some(ManuallyDrop::into_inner(data)); } } } } fn is_empty(&self) -> bool { let guard = self.collector.enter(); guard.protect(&self.head, Ordering::Relaxed).is_null() } } impl Drop for TreiberStack { fn drop(&mut self) { while self.pop().is_some() {} } } } mod haphazard_stack { use super::Stack; use haphazard::{Domain, HazardPointer}; use std::mem::ManuallyDrop; use std::ptr; use std::sync::atomic::{AtomicPtr, Ordering}; #[derive(Debug)] pub struct TreiberStack { head: AtomicPtr>, } #[derive(Debug)] struct Node { data: ManuallyDrop, next: *mut Node, } unsafe impl Send for Node {} unsafe impl Sync for Node {} impl Stack for TreiberStack { fn new() -> TreiberStack { TreiberStack { head: AtomicPtr::default(), } } fn push(&self, value: T) { let node = Box::into_raw(Box::new(Node { data: ManuallyDrop::new(value), next: ptr::null_mut(), })); let mut h = HazardPointer::new(); loop { let head = match h.protect_ptr(&self.head) { Some((ptr, _)) => ptr.as_ptr(), None => ptr::null_mut(), }; unsafe { (*node).next = head } if self .head .compare_exchange(head, node, Ordering::Release, Ordering::Relaxed) .is_ok() { break; } } } fn pop(&self) -> Option { let mut h = HazardPointer::new(); loop { let (head, _) = h.protect_ptr(&self.head)?; let next = unsafe { head.as_ref().next }; if self .head .compare_exchange(head.as_ptr(), next, Ordering::Relaxed, Ordering::Relaxed) .is_ok() { unsafe { let data = ptr::read(&head.as_ref().data); Domain::global().retire_ptr::<_, Box>>(head.as_ptr()); return Some(ManuallyDrop::into_inner(data)); } } } } fn is_empty(&self) -> bool { let mut h = HazardPointer::new(); unsafe { h.protect(&self.head) }.is_none() } } impl Drop for TreiberStack { fn drop(&mut self) { while self.pop().is_some() {} } } } mod crossbeam_stack { use super::Stack; use crossbeam_epoch::{Atomic, Owned, Shared}; use std::mem::ManuallyDrop; use std::ptr; use std::sync::atomic::Ordering; #[derive(Debug)] pub struct TreiberStack { head: Atomic>, } unsafe impl Send for TreiberStack {} unsafe impl Sync for TreiberStack {} #[derive(Debug)] struct Node { data: ManuallyDrop, next: *const Node, } impl Stack for TreiberStack { fn new() -> TreiberStack { TreiberStack { head: Atomic::null(), } } fn push(&self, value: T) { let guard = crossbeam_epoch::pin(); let mut node = Owned::new(Node { data: ManuallyDrop::new(value), next: ptr::null_mut(), }); loop { let head = self.head.load(Ordering::Relaxed, &guard); node.next = head.as_raw(); match self.head.compare_exchange( head, node, Ordering::Release, Ordering::Relaxed, &guard, ) { Ok(_) => break, Err(err) => node = err.new, } } } fn pop(&self) -> Option { let guard = crossbeam_epoch::pin(); loop { let head = self.head.load(Ordering::Acquire, &guard); if head.is_null() { return None; } let next = unsafe { head.deref().next }; if self .head .compare_exchange( head, Shared::from(next), Ordering::Relaxed, Ordering::Relaxed, &guard, ) .is_ok() { unsafe { let data = ptr::read(&head.deref().data); guard.defer_destroy(head); return Some(ManuallyDrop::into_inner(data)); } } } } fn is_empty(&self) -> bool { let guard = crossbeam_epoch::pin(); self.head.load(Ordering::Relaxed, &guard).is_null() } } impl Drop for TreiberStack { fn drop(&mut self) { while self.pop().is_some() {} } } }