| use crate::sync::atomic::Ordering::Relaxed; |
| use crate::sys::futex::{Futex, futex_wait, futex_wake, futex_wake_all}; |
| use crate::sys::sync::Mutex; |
| use crate::time::Duration; |
| |
| pub struct Condvar { |
| // The value of this atomic is simply incremented on every notification. |
| // This is used by `.wait()` to not miss any notifications after |
| // unlocking the mutex and before waiting for notifications. |
| futex: Futex, |
| } |
| |
| impl Condvar { |
| #[inline] |
| pub const fn new() -> Self { |
| Self { futex: Futex::new(0) } |
| } |
| |
| // All the memory orderings here are `Relaxed`, |
| // because synchronization is done by unlocking and locking the mutex. |
| |
| pub fn notify_one(&self) { |
| self.futex.fetch_add(1, Relaxed); |
| futex_wake(&self.futex); |
| } |
| |
| pub fn notify_all(&self) { |
| self.futex.fetch_add(1, Relaxed); |
| futex_wake_all(&self.futex); |
| } |
| |
| pub unsafe fn wait(&self, mutex: &Mutex) { |
| self.wait_optional_timeout(mutex, None); |
| } |
| |
| pub unsafe fn wait_timeout(&self, mutex: &Mutex, timeout: Duration) -> bool { |
| self.wait_optional_timeout(mutex, Some(timeout)) |
| } |
| |
| unsafe fn wait_optional_timeout(&self, mutex: &Mutex, timeout: Option<Duration>) -> bool { |
| // Examine the notification counter _before_ we unlock the mutex. |
| let futex_value = self.futex.load(Relaxed); |
| |
| // Unlock the mutex before going to sleep. |
| mutex.unlock(); |
| |
| // Wait, but only if there hasn't been any |
| // notification since we unlocked the mutex. |
| let r = futex_wait(&self.futex, futex_value, timeout); |
| |
| // Lock the mutex again. |
| mutex.lock(); |
| |
| r |
| } |
| } |