// SPDX-License-Identifier: GPL-2.0 //! A container that can be initialized at most once. use super::atomic::{ ordering::{Acquire, Relaxed, Release}, Atomic, }; use core::{cell::UnsafeCell, mem::MaybeUninit}; /// A container that can be populated at most once. Thread safe. /// /// Once the a [`SetOnce`] is populated, it remains populated by the same object for the /// lifetime `Self`. /// /// # Invariants /// /// - `init` may only increase in value. /// - `init` may only assume values in the range `0..=2`. /// - `init == 0` if and only if `value` is uninitialized. /// - `init == 1` if and only if there is exactly one thread with exclusive /// access to `self.value`. /// - `init == 2` if and only if `value` is initialized and valid for shared /// access. /// /// # Example /// /// ``` /// # use kernel::sync::SetOnce; /// let value = SetOnce::new(); /// assert_eq!(None, value.as_ref()); /// /// let status = value.populate(42u8); /// assert_eq!(true, status); /// assert_eq!(Some(&42u8), value.as_ref()); /// assert_eq!(Some(42u8), value.copy()); /// /// let status = value.populate(101u8); /// assert_eq!(false, status); /// assert_eq!(Some(&42u8), value.as_ref()); /// assert_eq!(Some(42u8), value.copy()); /// ``` pub struct SetOnce { init: Atomic, value: UnsafeCell>, } impl Default for SetOnce { fn default() -> Self { Self::new() } } impl SetOnce { /// Create a new [`SetOnce`]. /// /// The returned instance will be empty. pub const fn new() -> Self { // INVARIANT: The container is empty and we initialize `init` to `0`. Self { value: UnsafeCell::new(MaybeUninit::uninit()), init: Atomic::new(0), } } /// Get a reference to the contained object. /// /// Returns [`None`] if this [`SetOnce`] is empty. pub fn as_ref(&self) -> Option<&T> { if self.init.load(Acquire) == 2 { // SAFETY: By the type invariants of `Self`, `self.init == 2` means that `self.value` // is initialized and valid for shared access. Some(unsafe { &*self.value.get().cast() }) } else { None } } /// Populate the [`SetOnce`]. /// /// Returns `true` if the [`SetOnce`] was successfully populated. pub fn populate(&self, value: T) -> bool { // INVARIANT: If the swap succeeds: // - We increase `init`. // - We write the valid value `1` to `init`. // - Only one thread can succeed in this write, so we have exclusive access after the // write. if let Ok(0) = self.init.cmpxchg(0, 1, Relaxed) { // SAFETY: By the type invariants of `Self`, the fact that we succeeded in writing `1` // to `self.init` means we obtained exclusive access to `self.value`. unsafe { core::ptr::write(self.value.get().cast(), value) }; // INVARIANT: // - We increase `init`. // - We write the valid value `2` to `init`. // - We release our exclusive access to `self.value` and it is now valid for shared // access. self.init.store(2, Release); true } else { false } } /// Get a copy of the contained object. /// /// Returns [`None`] if the [`SetOnce`] is empty. pub fn copy(&self) -> Option where T: Copy, { self.as_ref().copied() } } impl Drop for SetOnce { fn drop(&mut self) { if *self.init.get_mut() == 2 { let value = self.value.get_mut(); // SAFETY: By the type invariants of `Self`, `self.init == 2` means that `self.value` // contains a valid value. We have exclusive access, as we hold a `mut` reference to // `self`. unsafe { value.assume_init_drop() }; } } }