use crate::cell::UnsafeCell; use crate::fmt; use crate::marker::PhantomData; use crate::mem::MaybeUninit; use crate::panic::{RefUnwindSafe, UnwindSafe}; use crate::sync::Once; /// A synchronization primitive which can nominally be written to only once. /// /// This type is a thread-safe [`OnceCell`], and can be used in statics. /// In many simple cases, you can use [`LazyLock`] instead to get the benefits of this type /// with less effort: `LazyLock` "looks like" `&T` because it initializes with `F` on deref! /// Where OnceLock shines is when LazyLock is too simple to support a given case, as LazyLock /// doesn't allow additional inputs to its function after you call [`LazyLock::new(|| ...)`]. /// /// A `OnceLock` can be thought of as a safe abstraction over uninitialized data that becomes /// initialized once written. /// /// Unlike [`Mutex`](crate::sync::Mutex), `OnceLock` is never poisoned on panic. /// /// [`OnceCell`]: crate::cell::OnceCell /// [`LazyLock`]: crate::sync::LazyLock /// [`LazyLock::new(|| ...)`]: crate::sync::LazyLock::new /// /// # Examples /// /// Writing to a `OnceLock` from a separate thread: /// /// ``` /// use std::sync::OnceLock; /// /// static CELL: OnceLock = OnceLock::new(); /// /// // `OnceLock` has not been written to yet. /// assert!(CELL.get().is_none()); /// /// // Spawn a thread and write to `OnceLock`. /// std::thread::spawn(|| { /// let value = CELL.get_or_init(|| 12345); /// assert_eq!(value, &12345); /// }) /// .join() /// .unwrap(); /// /// // `OnceLock` now contains the value. /// assert_eq!( /// CELL.get(), /// Some(&12345), /// ); /// ``` /// /// You can use `OnceLock` to implement a type that requires "append-only" logic: /// /// ``` /// use std::sync::{OnceLock, atomic::{AtomicU32, Ordering}}; /// use std::thread; /// /// struct OnceList { /// data: OnceLock, /// next: OnceLock>>, /// } /// impl OnceList { /// const fn new() -> OnceList { /// OnceList { data: OnceLock::new(), next: OnceLock::new() } /// } /// fn push(&self, value: T) { /// // FIXME: this impl is concise, but is also slow for long lists or many threads. /// // as an exercise, consider how you might improve on it while preserving the behavior /// if let Err(value) = self.data.set(value) { /// let next = self.next.get_or_init(|| Box::new(OnceList::new())); /// next.push(value) /// }; /// } /// fn contains(&self, example: &T) -> bool /// where /// T: PartialEq, /// { /// self.data.get().map(|item| item == example).filter(|v| *v).unwrap_or_else(|| { /// self.next.get().map(|next| next.contains(example)).unwrap_or(false) /// }) /// } /// } /// /// // Let's exercise this new Sync append-only list by doing a little counting /// static LIST: OnceList = OnceList::new(); /// static COUNTER: AtomicU32 = AtomicU32::new(0); /// /// # const LEN: u32 = if cfg!(miri) { 50 } else { 1000 }; /// # /* /// const LEN: u32 = 1000; /// # */ /// thread::scope(|s| { /// for _ in 0..thread::available_parallelism().unwrap().get() { /// s.spawn(|| { /// while let i @ 0..LEN = COUNTER.fetch_add(1, Ordering::Relaxed) { /// LIST.push(i); /// } /// }); /// } /// }); /// /// for i in 0..LEN { /// assert!(LIST.contains(&i)); /// } /// /// ``` #[stable(feature = "once_cell", since = "1.70.0")] pub struct OnceLock { // FIXME(nonpoison_once): switch to nonpoison version once it is available once: Once, // Whether or not the value is initialized is tracked by `once.is_completed()`. value: UnsafeCell>, /// `PhantomData` to make sure dropck understands we're dropping T in our Drop impl. /// /// ```compile_fail,E0597 /// use std::sync::OnceLock; /// /// struct A<'a>(&'a str); /// /// impl<'a> Drop for A<'a> { /// fn drop(&mut self) {} /// } /// /// let cell = OnceLock::new(); /// { /// let s = String::new(); /// let _ = cell.set(A(&s)); /// } /// ``` _marker: PhantomData, } impl OnceLock { /// Creates a new uninitialized cell. #[inline] #[must_use] #[stable(feature = "once_cell", since = "1.70.0")] #[rustc_const_stable(feature = "once_cell", since = "1.70.0")] pub const fn new() -> OnceLock { OnceLock { once: Once::new(), value: UnsafeCell::new(MaybeUninit::uninit()), _marker: PhantomData, } } /// Gets the reference to the underlying value. /// /// Returns `None` if the cell is uninitialized, or being initialized. /// This method never blocks. #[inline] #[stable(feature = "once_cell", since = "1.70.0")] pub fn get(&self) -> Option<&T> { if self.is_initialized() { // Safe b/c checked is_initialized Some(unsafe { self.get_unchecked() }) } else { None } } /// Gets the mutable reference to the underlying value. /// /// Returns `None` if the cell is uninitialized. /// /// This method never blocks. Since it borrows the `OnceLock` mutably, /// it is statically guaranteed that no active borrows to the `OnceLock` /// exist, including from other threads. #[inline] #[stable(feature = "once_cell", since = "1.70.0")] pub fn get_mut(&mut self) -> Option<&mut T> { if self.is_initialized() { // Safe b/c checked is_initialized and we have a unique access Some(unsafe { self.get_unchecked_mut() }) } else { None } } /// Blocks the current thread until the cell is initialized. /// /// # Example /// /// Waiting for a computation on another thread to finish: /// ```rust /// use std::thread; /// use std::sync::OnceLock; /// /// let value = OnceLock::new(); /// /// thread::scope(|s| { /// s.spawn(|| value.set(1 + 1)); /// /// let result = value.wait(); /// assert_eq!(result, &2); /// }) /// ``` #[inline] #[stable(feature = "once_wait", since = "1.86.0")] pub fn wait(&self) -> &T { self.once.wait_force(); unsafe { self.get_unchecked() } } /// Initializes the contents of the cell to `value`. /// /// May block if another thread is currently attempting to initialize the cell. The cell is /// guaranteed to contain a value when `set` returns, though not necessarily the one provided. /// /// Returns `Ok(())` if the cell was uninitialized and /// `Err(value)` if the cell was already initialized. /// /// # Examples /// /// ``` /// use std::sync::OnceLock; /// /// static CELL: OnceLock = OnceLock::new(); /// /// fn main() { /// assert!(CELL.get().is_none()); /// /// std::thread::spawn(|| { /// assert_eq!(CELL.set(92), Ok(())); /// }).join().unwrap(); /// /// assert_eq!(CELL.set(62), Err(62)); /// assert_eq!(CELL.get(), Some(&92)); /// } /// ``` #[inline] #[stable(feature = "once_cell", since = "1.70.0")] pub fn set(&self, value: T) -> Result<(), T> { match self.try_insert(value) { Ok(_) => Ok(()), Err((_, value)) => Err(value), } } /// Initializes the contents of the cell to `value` if the cell was uninitialized, /// then returns a reference to it. /// /// May block if another thread is currently attempting to initialize the cell. The cell is /// guaranteed to contain a value when `try_insert` returns, though not necessarily the /// one provided. /// /// Returns `Ok(&value)` if the cell was uninitialized and /// `Err((¤t_value, value))` if it was already initialized. /// /// # Examples /// /// ``` /// #![feature(once_cell_try_insert)] /// /// use std::sync::OnceLock; /// /// static CELL: OnceLock = OnceLock::new(); /// /// fn main() { /// assert!(CELL.get().is_none()); /// /// std::thread::spawn(|| { /// assert_eq!(CELL.try_insert(92), Ok(&92)); /// }).join().unwrap(); /// /// assert_eq!(CELL.try_insert(62), Err((&92, 62))); /// assert_eq!(CELL.get(), Some(&92)); /// } /// ``` #[inline] #[unstable(feature = "once_cell_try_insert", issue = "116693")] pub fn try_insert(&self, value: T) -> Result<&T, (&T, T)> { let mut value = Some(value); let res = self.get_or_init(|| value.take().unwrap()); match value { None => Ok(res), Some(value) => Err((res, value)), } } /// Gets the contents of the cell, initializing it to `f()` if the cell /// was uninitialized. /// /// Many threads may call `get_or_init` concurrently with different /// initializing functions, but it is guaranteed that only one function /// will be executed if the function doesn't panic. /// /// # Panics /// /// If `f()` panics, the panic is propagated to the caller, and the cell /// remains uninitialized. /// /// It is an error to reentrantly initialize the cell from `f`. The /// exact outcome is unspecified. Current implementation deadlocks, but /// this may be changed to a panic in the future. /// /// # Examples /// /// ``` /// use std::sync::OnceLock; /// /// let cell = OnceLock::new(); /// let value = cell.get_or_init(|| 92); /// assert_eq!(value, &92); /// let value = cell.get_or_init(|| unreachable!()); /// assert_eq!(value, &92); /// ``` #[inline] #[stable(feature = "once_cell", since = "1.70.0")] pub fn get_or_init(&self, f: F) -> &T where F: FnOnce() -> T, { match self.get_or_try_init(|| Ok::(f())) { Ok(val) => val, } } /// Gets the mutable reference of the contents of the cell, initializing /// it to `f()` if the cell was uninitialized. /// /// This method never blocks. Since it borrows the `OnceLock` mutably, /// it is statically guaranteed that no active borrows to the `OnceLock` /// exist, including from other threads. /// /// # Panics /// /// If `f()` panics, the panic is propagated to the caller, and the cell /// remains uninitialized. /// /// # Examples /// /// ``` /// #![feature(once_cell_get_mut)] /// /// use std::sync::OnceLock; /// /// let mut cell = OnceLock::new(); /// let value = cell.get_mut_or_init(|| 92); /// assert_eq!(*value, 92); /// /// *value += 2; /// assert_eq!(*value, 94); /// /// let value = cell.get_mut_or_init(|| unreachable!()); /// assert_eq!(*value, 94); /// ``` #[inline] #[unstable(feature = "once_cell_get_mut", issue = "121641")] pub fn get_mut_or_init(&mut self, f: F) -> &mut T where F: FnOnce() -> T, { match self.get_mut_or_try_init(|| Ok::(f())) { Ok(val) => val, } } /// Gets the contents of the cell, initializing it to `f()` if /// the cell was uninitialized. If the cell was uninitialized /// and `f()` failed, an error is returned. /// /// # Panics /// /// If `f()` panics, the panic is propagated to the caller, and /// the cell remains uninitialized. /// /// It is an error to reentrantly initialize the cell from `f`. /// The exact outcome is unspecified. Current implementation /// deadlocks, but this may be changed to a panic in the future. /// /// # Examples /// /// ``` /// #![feature(once_cell_try)] /// /// use std::sync::OnceLock; /// /// let cell = OnceLock::new(); /// assert_eq!(cell.get_or_try_init(|| Err(())), Err(())); /// assert!(cell.get().is_none()); /// let value = cell.get_or_try_init(|| -> Result { /// Ok(92) /// }); /// assert_eq!(value, Ok(&92)); /// assert_eq!(cell.get(), Some(&92)) /// ``` #[inline] #[unstable(feature = "once_cell_try", issue = "109737")] pub fn get_or_try_init(&self, f: F) -> Result<&T, E> where F: FnOnce() -> Result, { // Fast path check // NOTE: We need to perform an acquire on the state in this method // in order to correctly synchronize `LazyLock::force`. This is // currently done by calling `self.get()`, which in turn calls // `self.is_initialized()`, which in turn performs the acquire. if let Some(value) = self.get() { return Ok(value); } self.initialize(f)?; debug_assert!(self.is_initialized()); // SAFETY: The inner value has been initialized Ok(unsafe { self.get_unchecked() }) } /// Gets the mutable reference of the contents of the cell, initializing /// it to `f()` if the cell was uninitialized. If the cell was uninitialized /// and `f()` failed, an error is returned. /// /// This method never blocks. Since it borrows the `OnceLock` mutably, /// it is statically guaranteed that no active borrows to the `OnceLock` /// exist, including from other threads. /// /// # Panics /// /// If `f()` panics, the panic is propagated to the caller, and /// the cell remains uninitialized. /// /// # Examples /// /// ``` /// #![feature(once_cell_get_mut)] /// /// use std::sync::OnceLock; /// /// let mut cell: OnceLock = OnceLock::new(); /// /// // Failed attempts to initialize the cell do not change its contents /// assert!(cell.get_mut_or_try_init(|| "not a number!".parse()).is_err()); /// assert!(cell.get().is_none()); /// /// let value = cell.get_mut_or_try_init(|| "1234".parse()); /// assert_eq!(value, Ok(&mut 1234)); /// *value.unwrap() += 2; /// assert_eq!(cell.get(), Some(&1236)) /// ``` #[inline] #[unstable(feature = "once_cell_get_mut", issue = "121641")] pub fn get_mut_or_try_init(&mut self, f: F) -> Result<&mut T, E> where F: FnOnce() -> Result, { if self.get().is_none() { self.initialize(f)?; } debug_assert!(self.is_initialized()); // SAFETY: The inner value has been initialized Ok(unsafe { self.get_unchecked_mut() }) } /// Consumes the `OnceLock`, returning the wrapped value. Returns /// `None` if the cell was uninitialized. /// /// # Examples /// /// ``` /// use std::sync::OnceLock; /// /// let cell: OnceLock = OnceLock::new(); /// assert_eq!(cell.into_inner(), None); /// /// let cell = OnceLock::new(); /// cell.set("hello".to_string()).unwrap(); /// assert_eq!(cell.into_inner(), Some("hello".to_string())); /// ``` #[inline] #[stable(feature = "once_cell", since = "1.70.0")] pub fn into_inner(mut self) -> Option { self.take() } /// Takes the value out of this `OnceLock`, moving it back to an uninitialized state. /// /// Has no effect and returns `None` if the `OnceLock` was uninitialized. /// /// Since this method borrows the `OnceLock` mutably, it is statically guaranteed that /// no active borrows to the `OnceLock` exist, including from other threads. /// /// # Examples /// /// ``` /// use std::sync::OnceLock; /// /// let mut cell: OnceLock = OnceLock::new(); /// assert_eq!(cell.take(), None); /// /// let mut cell = OnceLock::new(); /// cell.set("hello".to_string()).unwrap(); /// assert_eq!(cell.take(), Some("hello".to_string())); /// assert_eq!(cell.get(), None); /// ``` #[inline] #[stable(feature = "once_cell", since = "1.70.0")] pub fn take(&mut self) -> Option { if self.is_initialized() { self.once = Once::new(); // SAFETY: `self.value` is initialized and contains a valid `T`. // `self.once` is reset, so `is_initialized()` will be false again // which prevents the value from being read twice. unsafe { Some((&mut *self.value.get()).assume_init_read()) } } else { None } } #[inline] fn is_initialized(&self) -> bool { self.once.is_completed() } #[cold] #[optimize(size)] fn initialize(&self, f: F) -> Result<(), E> where F: FnOnce() -> Result, { let mut res: Result<(), E> = Ok(()); let slot = &self.value; // Ignore poisoning from other threads // If another thread panics, then we'll be able to run our closure self.once.call_once_force(|p| { match f() { Ok(value) => { unsafe { (&mut *slot.get()).write(value) }; } Err(e) => { res = Err(e); // Treat the underlying `Once` as poisoned since we // failed to initialize our value. p.poison(); } } }); res } /// # Safety /// /// The cell must be initialized #[inline] unsafe fn get_unchecked(&self) -> &T { debug_assert!(self.is_initialized()); unsafe { (&*self.value.get()).assume_init_ref() } } /// # Safety /// /// The cell must be initialized #[inline] unsafe fn get_unchecked_mut(&mut self) -> &mut T { debug_assert!(self.is_initialized()); unsafe { (&mut *self.value.get()).assume_init_mut() } } } // Why do we need `T: Send`? // Thread A creates a `OnceLock` and shares it with // scoped thread B, which fills the cell, which is // then destroyed by A. That is, destructor observes // a sent value. #[stable(feature = "once_cell", since = "1.70.0")] unsafe impl Sync for OnceLock {} #[stable(feature = "once_cell", since = "1.70.0")] unsafe impl Send for OnceLock {} #[stable(feature = "once_cell", since = "1.70.0")] impl RefUnwindSafe for OnceLock {} #[stable(feature = "once_cell", since = "1.70.0")] impl UnwindSafe for OnceLock {} #[stable(feature = "once_cell", since = "1.70.0")] impl Default for OnceLock { /// Creates a new uninitialized cell. /// /// # Example /// /// ``` /// use std::sync::OnceLock; /// /// fn main() { /// assert_eq!(OnceLock::<()>::new(), OnceLock::default()); /// } /// ``` #[inline] fn default() -> OnceLock { OnceLock::new() } } #[stable(feature = "once_cell", since = "1.70.0")] impl fmt::Debug for OnceLock { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { let mut d = f.debug_tuple("OnceLock"); match self.get() { Some(v) => d.field(v), None => d.field(&format_args!("")), }; d.finish() } } #[stable(feature = "once_cell", since = "1.70.0")] impl Clone for OnceLock { #[inline] fn clone(&self) -> OnceLock { let cell = Self::new(); if let Some(value) = self.get() { match cell.set(value.clone()) { Ok(()) => (), Err(_) => unreachable!(), } } cell } } #[stable(feature = "once_cell", since = "1.70.0")] impl From for OnceLock { /// Creates a new cell with its contents set to `value`. /// /// # Example /// /// ``` /// use std::sync::OnceLock; /// /// # fn main() -> Result<(), i32> { /// let a = OnceLock::from(3); /// let b = OnceLock::new(); /// b.set(3)?; /// assert_eq!(a, b); /// Ok(()) /// # } /// ``` #[inline] fn from(value: T) -> Self { let cell = Self::new(); match cell.set(value) { Ok(()) => cell, Err(_) => unreachable!(), } } } #[stable(feature = "once_cell", since = "1.70.0")] impl PartialEq for OnceLock { /// Equality for two `OnceLock`s. /// /// Two `OnceLock`s are equal if they either both contain values and their /// values are equal, or if neither contains a value. /// /// # Examples /// /// ``` /// use std::sync::OnceLock; /// /// let five = OnceLock::new(); /// five.set(5).unwrap(); /// /// let also_five = OnceLock::new(); /// also_five.set(5).unwrap(); /// /// assert!(five == also_five); /// /// assert!(OnceLock::::new() == OnceLock::::new()); /// ``` #[inline] fn eq(&self, other: &OnceLock) -> bool { self.get() == other.get() } } #[stable(feature = "once_cell", since = "1.70.0")] impl Eq for OnceLock {} #[stable(feature = "once_cell", since = "1.70.0")] unsafe impl<#[may_dangle] T> Drop for OnceLock { #[inline] fn drop(&mut self) { if self.is_initialized() { // SAFETY: The cell is initialized and being dropped, so it can't // be accessed again. We also don't touch the `T` other than // dropping it, which validates our usage of #[may_dangle]. unsafe { (&mut *self.value.get()).assume_init_drop() }; } } }