tokio-sync 0.1.8

Synchronization utilities.
Documentation
extern crate futures;
#[macro_use]
extern crate loom;

macro_rules! if_fuzz {
    ($($t:tt)*) => {
        $($t)*
    }
}

#[path = "../src/mpsc/list.rs"]
#[allow(warnings)]
mod list;

#[path = "../src/mpsc/block.rs"]
#[allow(warnings)]
mod block;

const BLOCK_CAP: usize = 2;

use loom::thread;

use std::sync::Arc;

#[test]
fn smoke() {
    use block::Read::*;

    const NUM_TX: usize = 2;
    const NUM_MSG: usize = 2;

    loom::fuzz(|| {
        let (tx, mut rx) = list::channel();
        let tx = Arc::new(tx);

        for th in 0..NUM_TX {
            let tx = tx.clone();

            thread::spawn(move || {
                for i in 0..NUM_MSG {
                    tx.push((th, i));
                }
                debug!(" + tx thread done");
            });
        }

        let mut next = vec![0; NUM_TX];

        loop {
            debug!(" + rx.pop()");
            match rx.pop(&tx) {
                Some(Value((th, v))) => {
                    debug!(" + pop() -> Some(Value({}))", v);
                    assert_eq!(v, next[th]);
                    next[th] += 1;

                    if next.iter().all(|&i| i == NUM_MSG) {
                        break;
                    }
                }
                Some(Closed) => {
                    panic!();
                }
                None => {
                    debug!(" + pop() -> None");
                    loom::yield_now();
                }
            }
        }
    });
}