rstubs/util/
ticket.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
use core::cell::UnsafeCell;
use core::ops::{Deref, DerefMut};
use core::sync::atomic::AtomicUsize;
use core::sync::atomic::Ordering::*;

/// By the use of Ticketlocks, it is possible to serialize blocks of code
/// that might run parallel on multiple CPU cores.
///
/// Synchronization is implemented using a lock and a ticket variable.
/// Once a thread tries to enter the critical area, it obtains a ticket by
/// atomic increment of the ticket variable and waits until the lock variable
/// is equal to its ticket.
/// When a thread leaves the critical area, it increments the lock variable by
/// one and thereby allows the next thread to enter the critical area.
#[derive(Debug, Default)]
pub struct Ticket<T> {
    serving: AtomicUsize,
    ticket: AtomicUsize,
    /// Wrap the mutable data inside an unsafe cell, so that the compiler doesn't optimize it away
    value: UnsafeCell<T>,
}

unsafe impl<T> Sync for Ticket<T> {}
unsafe impl<T: Send> Send for Ticket<T> {}

impl<T> Ticket<T> {
    /// Create a new ticket lock, protecting `value`
    pub const fn new(value: T) -> Self {
        Self {
            serving: AtomicUsize::new(0),
            ticket: AtomicUsize::new(0),
            value: UnsafeCell::new(value),
        }
    }
    /// Locks the lock and returns the protected data.
    ///
    /// The data is wrapped in a guard, that unlocks the lock if it goes out of scope.
    #[must_use]
    pub fn lock(&self) -> TicketGuard<T> {
        let ticket = self.ticket.fetch_add(1, Relaxed);
        while ticket != self.serving.load(Acquire) {
            // pausing the cpu
            core::hint::spin_loop();
        }
        TicketGuard { lock: self }
    }
    /// This is automatically called by the guard destructor.
    pub unsafe fn unlock(&self) {
        self.serving.fetch_add(1, Release);
    }
    /// If you really want to access the contents. This is not protected by race conditions!
    pub unsafe fn raw(&self) -> &mut T {
        &mut *self.value.get()
    }
}

/// Accessor for the locks data, that unlocks if it goes out of scope.
pub struct TicketGuard<'a, T> {
    lock: &'a Ticket<T>,
}
impl<T> Deref for TicketGuard<'_, T> {
    type Target = T;

    fn deref(&self) -> &Self::Target {
        unsafe { self.lock.raw() }
    }
}
impl<T> DerefMut for TicketGuard<'_, T> {
    fn deref_mut(&mut self) -> &mut Self::Target {
        unsafe { self.lock.raw() }
    }
}
impl<T> Drop for TicketGuard<'_, T> {
    fn drop(&mut self) {
        unsafe { self.lock.unlock() };
    }
}