lockfree_object_pool/
linear_page.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
use crate::page::{Page, PageId};
use std::ptr;
use std::sync::atomic::{AtomicPtr, Ordering};

pub struct LinearPage<T> {
    page: Page<T>,
    next: AtomicPtr<LinearPage<T>>,
}

impl<T> LinearPage<T> {
    #[inline]
    pub fn new<I>(init: I) -> Self
    where
        I: Fn() -> T,
    {
        Self {
            page: Page::new(init),
            next: AtomicPtr::new(ptr::null_mut()),
        }
    }

    #[inline]
    pub fn get_or_create_next<I>(&self, init: I) -> &Self
    where
        I: Fn() -> T,
    {
        let mut current = self.next.load(Ordering::Relaxed);
        if current.is_null() {
            let new = Box::into_raw(Box::new(LinearPage::<T>::new(init)));
            match self
                .next
                .compare_exchange(current, new, Ordering::SeqCst, Ordering::Relaxed)
            {
                Ok(_) => {
                    current = new;
                }
                Err(x) => {
                    unsafe {
                        // SAFETY: new was been allocated by Box::new
                        drop(Box::from_raw(new))
                    };
                    current = x;
                }
            }
        }
        unsafe {
            // SAFETY: there are no mutable references to current
            current.as_ref().unwrap()
        }
    }

    #[inline]
    pub fn alloc<I>(&self, init: I) -> (&Page<T>, PageId)
    where
        I: Fn() -> T + Clone,
    {
        let mut linear_page = self;
        loop {
            match linear_page.page.alloc() {
                Some(id) => {
                    return (&linear_page.page, id);
                }
                None => {
                    linear_page = linear_page.get_or_create_next(init.clone());
                }
            };
        }
    }
}

impl<T> Drop for LinearPage<T> {
    #[inline]
    fn drop(&mut self) {
        let current = self.next.load(Ordering::Relaxed);
        if !current.is_null() {
            unsafe {
                // SAFETY: current was allocated with Box::new
                drop(Box::from_raw(current))
            };
        }
    }
}