spin::once

Struct Once

source
pub struct Once<T = (), R = Spin> {
    phantom: PhantomData<R>,
    status: AtomicStatus,
    data: UnsafeCell<MaybeUninit<T>>,
}
Expand description

A primitive that provides lazy one-time initialization.

Unlike its std::sync equivalent, this is generalized such that the closure returns a value to be stored by the Once (std::sync::Once can be trivially emulated with Once).

Because Once::new is const, this primitive may be used to safely initialize statics.

§Examples

use spin;

static START: spin::Once = spin::Once::new();

START.call_once(|| {
    // run initialization here
});

Fields§

§phantom: PhantomData<R>§status: AtomicStatus§data: UnsafeCell<MaybeUninit<T>>

Implementations§

source§

impl<T, R: RelaxStrategy> Once<T, R>

source

pub fn call_once<F: FnOnce() -> T>(&self, f: F) -> &T

Performs an initialization routine once and only once. The given closure will be executed if this is the first time call_once has been called, and otherwise the routine will not be invoked.

This method will block the calling thread if another initialization routine is currently running.

When this function returns, it is guaranteed that some initialization has run and completed (it may not be the closure specified). The returned pointer will point to the result from the closure that was run.

§Panics

This function will panic if the Once previously panicked while attempting to initialize. This is similar to the poisoning behaviour of std::sync’s primitives.

§Examples
use spin;

static INIT: spin::Once<usize> = spin::Once::new();

fn get_cached_val() -> usize {
    *INIT.call_once(expensive_computation)
}

fn expensive_computation() -> usize {
    // ...
}
source

pub fn try_call_once<F: FnOnce() -> Result<T, E>, E>( &self, f: F, ) -> Result<&T, E>

This method is similar to call_once, but allows the given closure to fail, and lets the Once in a uninitialized state if it does.

This method will block the calling thread if another initialization routine is currently running.

When this function returns without error, it is guaranteed that some initialization has run and completed (it may not be the closure specified). The returned reference will point to the result from the closure that was run.

§Panics

This function will panic if the Once previously panicked while attempting to initialize. This is similar to the poisoning behaviour of std::sync’s primitives.

§Examples
use spin;

static INIT: spin::Once<usize> = spin::Once::new();

fn get_cached_val() -> Result<usize, String> {
    INIT.try_call_once(expensive_fallible_computation).map(|x| *x)
}

fn expensive_fallible_computation() -> Result<usize, String> {
    // ...
}
source

fn try_call_once_slow<F: FnOnce() -> Result<T, E>, E>( &self, f: F, ) -> Result<&T, E>

source

pub fn wait(&self) -> &T

Spins until the Once contains a value.

Note that in releases prior to 0.7, this function had the behaviour of Once::poll.

§Panics

This function will panic if the Once previously panicked while attempting to initialize. This is similar to the poisoning behaviour of std::sync’s primitives.

source

pub fn poll(&self) -> Option<&T>

Like Once::get, but will spin if the Once is in the process of being initialized. If initialization has not even begun, None will be returned.

Note that in releases prior to 0.7, this function was named wait.

§Panics

This function will panic if the Once previously panicked while attempting to initialize. This is similar to the poisoning behaviour of std::sync’s primitives.

source§

impl<T, R> Once<T, R>

source

pub const INIT: Self = _

Initialization constant of Once.

source

pub const fn new() -> Self

Creates a new Once.

source

pub const fn initialized(data: T) -> Self

Creates a new initialized Once.

source

pub fn as_mut_ptr(&self) -> *mut T

Retrieve a pointer to the inner data.

While this method itself is safe, accessing the pointer before the Once has been initialized is UB, unless this method has already been written to from a pointer coming from this method.

source

unsafe fn force_get(&self) -> &T

Get a reference to the initialized instance. Must only be called once COMPLETE.

source

unsafe fn force_get_mut(&mut self) -> &mut T

Get a reference to the initialized instance. Must only be called once COMPLETE.

source

unsafe fn force_into_inner(self) -> T

Get a reference to the initialized instance. Must only be called once COMPLETE.

source

pub fn get(&self) -> Option<&T>

Returns a reference to the inner value if the Once has been initialized.

source

pub unsafe fn get_unchecked(&self) -> &T

Returns a reference to the inner value on the unchecked assumption that the Once has been initialized.

§Safety

This is extremely unsafe if the Once has not already been initialized because a reference to uninitialized memory will be returned, immediately triggering undefined behaviour (even if the reference goes unused). However, this can be useful in some instances for exposing the Once to FFI or when the overhead of atomically checking initialization is unacceptable and the Once has already been initialized.

source

pub fn get_mut(&mut self) -> Option<&mut T>

Returns a mutable reference to the inner value if the Once has been initialized.

Because this method requires a mutable reference to the Once, no synchronization overhead is required to access the inner value. In effect, it is zero-cost.

source

pub unsafe fn get_mut_unchecked(&mut self) -> &mut T

Returns a mutable reference to the inner value

§Safety

This is extremely unsafe if the Once has not already been initialized because a reference to uninitialized memory will be returned, immediately triggering undefined behaviour (even if the reference goes unused). However, this can be useful in some instances for exposing the Once to FFI or when the overhead of atomically checking initialization is unacceptable and the Once has already been initialized.

source

pub fn try_into_inner(self) -> Option<T>

Returns a the inner value if the Once has been initialized.

Because this method requires ownership of the Once, no synchronization overhead is required to access the inner value. In effect, it is zero-cost.

source

pub unsafe fn into_inner_unchecked(self) -> T

Returns a the inner value if the Once has been initialized.

§Safety

This is extremely unsafe if the Once has not already been initialized because a reference to uninitialized memory will be returned, immediately triggering undefined behaviour (even if the reference goes unused) This can be useful, if Once has already been initialized, and you want to bypass an option check.

source

pub fn is_completed(&self) -> bool

Checks whether the value has been initialized.

This is done using Acquire ordering, and therefore it is safe to access the value directly via get_unchecked if this returns true.

Trait Implementations§

source§

impl<T: Debug, R> Debug for Once<T, R>

source§

fn fmt(&self, f: &mut Formatter<'_>) -> Result

Formats the value using the given formatter. Read more
source§

impl<T, R> Default for Once<T, R>

source§

fn default() -> Self

Returns the “default value” for a type. Read more
source§

impl<T, R> Drop for Once<T, R>

source§

fn drop(&mut self)

Executes the destructor for this type. Read more
source§

impl<T, R> From<T> for Once<T, R>

source§

fn from(data: T) -> Self

Converts to this type from the input type.
source§

impl<T: Send, R> Send for Once<T, R>

source§

impl<T: Send + Sync, R> Sync for Once<T, R>

Auto Trait Implementations§

§

impl<T = (), R = Spin> !Freeze for Once<T, R>

§

impl<T = (), R = Spin> !RefUnwindSafe for Once<T, R>

§

impl<T, R> Unpin for Once<T, R>
where R: Unpin, T: Unpin,

§

impl<T, R> UnwindSafe for Once<T, R>
where R: UnwindSafe, T: UnwindSafe,

Blanket Implementations§

source§

impl<T> Any for T
where T: 'static + ?Sized,

source§

fn type_id(&self) -> TypeId

Gets the TypeId of self. Read more
source§

impl<T> Borrow<T> for T
where T: ?Sized,

source§

fn borrow(&self) -> &T

Immutably borrows from an owned value. Read more
source§

impl<T> BorrowMut<T> for T
where T: ?Sized,

source§

fn borrow_mut(&mut self) -> &mut T

Mutably borrows from an owned value. Read more
source§

impl<T> From<!> for T

source§

fn from(t: !) -> T

Converts to this type from the input type.
source§

impl<T> From<T> for T

source§

fn from(t: T) -> T

Returns the argument unchanged.

source§

impl<T, U> Into<U> for T
where U: From<T>,

source§

fn into(self) -> U

Calls U::from(self).

That is, this conversion is whatever the implementation of From<T> for U chooses to do.

source§

impl<T, U> TryFrom<U> for T
where U: Into<T>,

source§

type Error = Infallible

The type returned in the event of a conversion error.
source§

fn try_from(value: U) -> Result<T, <T as TryFrom<U>>::Error>

Performs the conversion.
source§

impl<T, U> TryInto<U> for T
where U: TryFrom<T>,

source§

type Error = <U as TryFrom<T>>::Error

The type returned in the event of a conversion error.
source§

fn try_into(self) -> Result<U, <U as TryFrom<T>>::Error>

Performs the conversion.