[go: up one dir, main page]

sdd 4.4.0

Scalable lock-free delayed memory reclaimer
Documentation
use std::sync::atomic::Ordering::Relaxed;
use std::sync::{Arc, Mutex};

use loom::sync::atomic::{AtomicBool, AtomicUsize};
use loom::thread::{spawn, yield_now};

use crate::{AtomicOwned, AtomicShared, Guard, suspend};

struct A(AtomicBool, Arc<AtomicUsize>);
impl Drop for A {
    fn drop(&mut self) {
        self.0.store(true, Relaxed);
        self.1.fetch_add(1, Relaxed);
    }
}

static SERIALIZER: Mutex<()> = Mutex::new(());

#[test]
fn ebr_owned() {
    let _guard = SERIALIZER.lock().unwrap();
    loom::model(|| {
        let drop_count = Arc::new(AtomicUsize::new(0));
        let data_owned = AtomicOwned::new(A(AtomicBool::new(false), drop_count.clone()));

        let guard = Guard::new();
        let ptr = data_owned.load(Relaxed, &guard);

        let thread = spawn(move || {
            let guard = Guard::new();
            guard.accelerate();

            let ptr = data_owned.load(Relaxed, &guard);
            drop(data_owned);

            assert!(!ptr.as_ref().unwrap().0.load(Relaxed));
            drop(guard);

            while drop_count.load(Relaxed) != 1 {
                Guard::new().accelerate();
                yield_now();
            }

            assert!(suspend());
        });

        assert!(!ptr.as_ref().unwrap().0.load(Relaxed));
        drop(guard);

        assert!(thread.join().is_ok());
        assert!(suspend());
    });
}

#[test]
fn ebr_shared() {
    let _guard = SERIALIZER.lock().unwrap();
    loom::model(|| {
        let drop_count = Arc::new(AtomicUsize::new(0));
        let data_shared = AtomicShared::new(A(AtomicBool::new(false), drop_count.clone()));

        let guard = Guard::new();
        let ptr = data_shared.load(Relaxed, &guard);

        let thread = spawn(move || {
            let data_shared_clone = data_shared.get_shared(Relaxed, &Guard::new()).unwrap();
            drop(data_shared);

            let guard = Guard::new();
            guard.accelerate();

            let ptr = data_shared_clone.get_guarded_ptr(&guard);
            drop(data_shared_clone);

            assert!(!ptr.as_ref().unwrap().0.load(Relaxed));
            drop(guard);

            while drop_count.load(Relaxed) != 1 {
                Guard::new().accelerate();
                yield_now();
            }

            assert!(suspend());
        });

        assert!(!ptr.as_ref().unwrap().0.load(Relaxed));
        drop(guard);

        assert!(thread.join().is_ok());
        assert!(suspend());
    });
}