//! Defines [`Exclusive`]. use core::fmt; use core::future::Future; use core::marker::Tuple; use core::ops::{Coroutine, CoroutineState}; use core::pin::Pin; use core::task::{Context, Poll}; /// `Exclusive` provides only _mutable_ access, also referred to as _exclusive_ /// access to the underlying value. It provides no _immutable_, or _shared_ /// access to the underlying value. /// /// While this may seem not very useful, it allows `Exclusive` to _unconditionally_ /// implement [`Sync`]. Indeed, the safety requirements of `Sync` state that for `Exclusive` /// to be `Sync`, it must be sound to _share_ across threads, that is, it must be sound /// for `&Exclusive` to cross thread boundaries. By design, a `&Exclusive` has no API /// whatsoever, making it useless, thus harmless, thus memory safe. /// /// Certain constructs like [`Future`]s can only be used with _exclusive_ access, /// and are often `Send` but not `Sync`, so `Exclusive` can be used as hint to the /// Rust compiler that something is `Sync` in practice. /// /// ## Examples /// Using a non-`Sync` future prevents the wrapping struct from being `Sync` /// ```compile_fail /// use core::cell::Cell; /// /// async fn other() {} /// fn assert_sync(t: T) {} /// struct State { /// future: F /// } /// /// assert_sync(State { /// future: async { /// let cell = Cell::new(1); /// let cell_ref = &cell; /// other().await; /// let value = cell_ref.get(); /// } /// }); /// ``` /// /// `Exclusive` ensures the struct is `Sync` without stripping the future of its /// functionality. /// ``` /// #![feature(exclusive_wrapper)] /// use core::cell::Cell; /// use core::sync::Exclusive; /// /// async fn other() {} /// fn assert_sync(t: T) {} /// struct State { /// future: Exclusive /// } /// /// assert_sync(State { /// future: Exclusive::new(async { /// let cell = Cell::new(1); /// let cell_ref = &cell; /// other().await; /// let value = cell_ref.get(); /// }) /// }); /// ``` /// /// ## Parallels with a mutex /// In some sense, `Exclusive` can be thought of as a _compile-time_ version of /// a mutex, as the borrow-checker guarantees that only one `&mut` can exist /// for any value. This is a parallel with the fact that /// `&` and `&mut` references together can be thought of as a _compile-time_ /// version of a read-write lock. #[unstable(feature = "exclusive_wrapper", issue = "98407")] #[doc(alias = "SyncWrapper")] #[doc(alias = "SyncCell")] #[doc(alias = "Unique")] // `Exclusive` can't have `PartialOrd`, `Clone`, etc. impls as they would // use `&` access to the inner value, violating the `Sync` impl's safety // requirements. #[derive(Default)] #[repr(transparent)] pub struct Exclusive { inner: T, } // See `Exclusive`'s docs for justification. #[unstable(feature = "exclusive_wrapper", issue = "98407")] unsafe impl Sync for Exclusive {} #[unstable(feature = "exclusive_wrapper", issue = "98407")] impl fmt::Debug for Exclusive { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> Result<(), fmt::Error> { f.debug_struct("Exclusive").finish_non_exhaustive() } } impl Exclusive { /// Wrap a value in an `Exclusive` #[unstable(feature = "exclusive_wrapper", issue = "98407")] #[must_use] #[inline] pub const fn new(t: T) -> Self { Self { inner: t } } /// Unwrap the value contained in the `Exclusive` #[unstable(feature = "exclusive_wrapper", issue = "98407")] #[rustc_const_unstable(feature = "exclusive_wrapper", issue = "98407")] #[must_use] #[inline] pub const fn into_inner(self) -> T { self.inner } } impl Exclusive { /// Gets exclusive access to the underlying value. #[unstable(feature = "exclusive_wrapper", issue = "98407")] #[must_use] #[inline] pub const fn get_mut(&mut self) -> &mut T { &mut self.inner } /// Gets pinned exclusive access to the underlying value. /// /// `Exclusive` is considered to _structurally pin_ the underlying /// value, which means _unpinned_ `Exclusive`s can produce _unpinned_ /// access to the underlying value, but _pinned_ `Exclusive`s only /// produce _pinned_ access to the underlying value. #[unstable(feature = "exclusive_wrapper", issue = "98407")] #[must_use] #[inline] pub const fn get_pin_mut(self: Pin<&mut Self>) -> Pin<&mut T> { // SAFETY: `Exclusive` can only produce `&mut T` if itself is unpinned // `Pin::map_unchecked_mut` is not const, so we do this conversion manually unsafe { Pin::new_unchecked(&mut self.get_unchecked_mut().inner) } } /// Build a _mutable_ reference to an `Exclusive` from /// a _mutable_ reference to a `T`. This allows you to skip /// building an `Exclusive` with [`Exclusive::new`]. #[unstable(feature = "exclusive_wrapper", issue = "98407")] #[must_use] #[inline] pub const fn from_mut(r: &'_ mut T) -> &'_ mut Exclusive { // SAFETY: repr is ≥ C, so refs have the same layout; and `Exclusive` properties are `&mut`-agnostic unsafe { &mut *(r as *mut T as *mut Exclusive) } } /// Build a _pinned mutable_ reference to an `Exclusive` from /// a _pinned mutable_ reference to a `T`. This allows you to skip /// building an `Exclusive` with [`Exclusive::new`]. #[unstable(feature = "exclusive_wrapper", issue = "98407")] #[must_use] #[inline] pub const fn from_pin_mut(r: Pin<&'_ mut T>) -> Pin<&'_ mut Exclusive> { // SAFETY: `Exclusive` can only produce `&mut T` if itself is unpinned // `Pin::map_unchecked_mut` is not const, so we do this conversion manually unsafe { Pin::new_unchecked(Self::from_mut(r.get_unchecked_mut())) } } } #[unstable(feature = "exclusive_wrapper", issue = "98407")] #[rustc_const_unstable(feature = "const_convert", issue = "143773")] impl const From for Exclusive { #[inline] fn from(t: T) -> Self { Self::new(t) } } #[unstable(feature = "exclusive_wrapper", issue = "98407")] impl FnOnce for Exclusive where F: FnOnce, Args: Tuple, { type Output = F::Output; extern "rust-call" fn call_once(self, args: Args) -> Self::Output { self.into_inner().call_once(args) } } #[unstable(feature = "exclusive_wrapper", issue = "98407")] impl FnMut for Exclusive where F: FnMut, Args: Tuple, { extern "rust-call" fn call_mut(&mut self, args: Args) -> Self::Output { self.get_mut().call_mut(args) } } #[unstable(feature = "exclusive_wrapper", issue = "98407")] impl Future for Exclusive where T: Future + ?Sized, { type Output = T::Output; #[inline] fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll { self.get_pin_mut().poll(cx) } } #[unstable(feature = "coroutine_trait", issue = "43122")] // also #98407 impl Coroutine for Exclusive where G: Coroutine + ?Sized, { type Yield = G::Yield; type Return = G::Return; #[inline] fn resume(self: Pin<&mut Self>, arg: R) -> CoroutineState { G::resume(self.get_pin_mut(), arg) } }