extern crate atom;
use std::thread;
use std::mem;
use std::sync::{Arc, Barrier};
use atom::*;
const THREADS: usize = 100;
#[derive(Debug)]
struct Link {
next: AtomSetOnce<Box<Link>>
}
impl Drop for Link {
fn drop(&mut self) {
while let Some(mut h) = self.next.atom().take() {
self.next = mem::replace(&mut h.next, AtomSetOnce::empty());
}
}
}
fn main() {
let b = Arc::new(Barrier::new(THREADS + 1));
let head = Arc::new(Link{next: AtomSetOnce::empty()});
for _ in (0..THREADS) {
let b = b.clone();
let head = head.clone();
thread::spawn(move || {
let mut hptr = &*head;
for _ in (0..10_000) {
let mut my_awesome_node = Box::new(Link {
next: AtomSetOnce::empty()
});
loop {
while let Some(h) = hptr.next.get() {
hptr = h;
}
my_awesome_node = match hptr.next.set_if_none(my_awesome_node) {
Some(v) => v,
None => break
};
}
}
b.wait();
});
}
b.wait();
let mut hptr = &*head;
let mut count = 0;
while let Some(h) = hptr.next.get() {
hptr = h;
count += 1;
}
println!("Using {} threads we wrote {} links at the same time!", THREADS, count);
}