#![feature(test)]
extern crate crossbeam_utils;
extern crate test;
use std::sync::Barrier;
use crossbeam_utils::atomic::AtomicCell;
use crossbeam_utils::thread;
#[bench]
fn load_u8(b: &mut test::Bencher) {
let a = AtomicCell::new(0u8);
let mut sum = 0;
b.iter(|| sum += a.load());
test::black_box(sum);
}
#[bench]
fn store_u8(b: &mut test::Bencher) {
let a = AtomicCell::new(0u8);
b.iter(|| a.store(1));
}
#[bench]
fn fetch_add_u8(b: &mut test::Bencher) {
let a = AtomicCell::new(0u8);
b.iter(|| a.fetch_add(1));
}
#[bench]
fn compare_and_swap_u8(b: &mut test::Bencher) {
let a = AtomicCell::new(0u8);
let mut i = 0;
b.iter(|| {
a.compare_and_swap(i, i.wrapping_add(1));
i = i.wrapping_add(1);
});
}
#[bench]
fn concurrent_load_u8(b: &mut test::Bencher) {
const THREADS: usize = 2;
const STEPS: usize = 1_000_000;
let start = Barrier::new(THREADS + 1);
let end = Barrier::new(THREADS + 1);
let exit = AtomicCell::new(false);
let a = AtomicCell::new(0u8);
thread::scope(|scope| {
for _ in 0..THREADS {
scope.spawn(|_| {
loop {
start.wait();
let mut sum = 0;
for _ in 0..STEPS {
sum += a.load();
}
test::black_box(sum);
end.wait();
if exit.load() {
break;
}
}
});
}
start.wait();
end.wait();
b.iter(|| {
start.wait();
end.wait();
});
start.wait();
exit.store(true);
end.wait();
}).unwrap();
}
#[bench]
fn load_usize(b: &mut test::Bencher) {
let a = AtomicCell::new(0usize);
let mut sum = 0;
b.iter(|| sum += a.load());
test::black_box(sum);
}
#[bench]
fn store_usize(b: &mut test::Bencher) {
let a = AtomicCell::new(0usize);
b.iter(|| a.store(1));
}
#[bench]
fn fetch_add_usize(b: &mut test::Bencher) {
let a = AtomicCell::new(0usize);
b.iter(|| a.fetch_add(1));
}
#[bench]
fn compare_and_swap_usize(b: &mut test::Bencher) {
let a = AtomicCell::new(0usize);
let mut i = 0;
b.iter(|| {
a.compare_and_swap(i, i.wrapping_add(1));
i = i.wrapping_add(1);
});
}
#[bench]
fn concurrent_load_usize(b: &mut test::Bencher) {
const THREADS: usize = 2;
const STEPS: usize = 1_000_000;
let start = Barrier::new(THREADS + 1);
let end = Barrier::new(THREADS + 1);
let exit = AtomicCell::new(false);
let a = AtomicCell::new(0usize);
thread::scope(|scope| {
for _ in 0..THREADS {
scope.spawn(|_| {
loop {
start.wait();
let mut sum = 0;
for _ in 0..STEPS {
sum += a.load();
}
test::black_box(sum);
end.wait();
if exit.load() {
break;
}
}
});
}
start.wait();
end.wait();
b.iter(|| {
start.wait();
end.wait();
});
start.wait();
exit.store(true);
end.wait();
}).unwrap();
}