[go: up one dir, main page]

loom 0.2.9

Permutation testing for concurrent code
Documentation
#![deny(warnings, rust_2018_idioms)]

use loom::sync::atomic::AtomicUsize;
use loom::thread;

use std::cell::UnsafeCell;
use std::sync::atomic::Ordering::{AcqRel, Acquire, Release, SeqCst};
use std::sync::Arc;

#[test]
fn valid_get_mut() {
    loom::model(|| {
        let v1 = Arc::new(UnsafeCell::new(AtomicUsize::new(0)));
        let v2 = v1.clone();

        let th = thread::spawn(move || unsafe {
            (*v2.get()).store(1, SeqCst);
        });

        th.join().unwrap();

        let v = unsafe { *(*v1.get()).get_mut() };
        assert_eq!(1, v);
    });
}

#[test]
#[should_panic]
fn invalid_get_mut() {
    loom::model(|| {
        let v1 = Arc::new(UnsafeCell::new(AtomicUsize::new(0)));
        let v2 = v1.clone();

        thread::spawn(move || unsafe {
            (*v2.get()).store(1, SeqCst);
        });

        let _ = unsafe { *(*v1.get()).get_mut() };
    });
}

#[test]
#[ignore]
#[should_panic]
fn wut() {
    loom::model(|| {
        let a = Arc::new(AtomicUsize::new(0));
        let b = Arc::new(AtomicUsize::new(0));

        let a2 = a.clone();
        let b2 = b.clone();

        let th = thread::spawn(move || {
            a2.store(1, Release);
            b2.compare_and_swap(0, 2, AcqRel);
        });

        b.store(1, Release);
        a.compare_and_swap(0, 2, AcqRel);

        th.join().unwrap();

        let a_val = a.load(Acquire);
        let b_val = b.load(Acquire);

        if a_val == 2 && b_val == 2 {
            panic!();
        }
    });
}