From fe9db7519bff01acedc89b1032b78b30e7abf045 Mon Sep 17 00:00:00 2001 From: Taiki Endo Date: Fri, 22 Dec 2023 00:09:32 +0900 Subject: [PATCH] util: Rewrite Arc based on std::sync::Arc --- .github/.cspell/project-dictionary.txt | 2 + .github/workflows/ci.yml | 8 +- portable-atomic-util/build.rs | 8 + portable-atomic-util/src/arc.rs | 2522 +++++++++++++++++++----- portable-atomic-util/src/lib.rs | 2 + portable-atomic-util/tests/arc.rs | 684 ++++++- tools/build.sh | 5 +- tools/test.sh | 20 +- 8 files changed, 2745 insertions(+), 506 deletions(-) diff --git a/.github/.cspell/project-dictionary.txt b/.github/.cspell/project-dictionary.txt index e24df63d0..99edb87fb 100644 --- a/.github/.cspell/project-dictionary.txt +++ b/.github/.cspell/project-dictionary.txt @@ -44,6 +44,7 @@ DESTDIR DEVEL dlsym DWCAS +elems espup exynos FIQs @@ -157,6 +158,7 @@ uapi uart umax umin +unclonable unistd unparse usart diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 3daa1fda7..0bc5af08d 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -271,12 +271,12 @@ jobs: if: (matrix.target == '' && !contains(matrix.rust, 'i686') || startsWith(matrix.target, 'x86_64')) || startsWith(matrix.target, 'aarch64') && !(contains(matrix.target, '-musl') && matrix.flags == '') || startsWith(matrix.target, 'armv5te') || matrix.target == 'arm-linux-androideabi' # outline-atomics is disabled by default on aarch64 musl with static linking and powerpc64 # powerpc64le- (little-endian) is skipped because it is pwr8 by default - - run: tools/test.sh -vv --tests $TARGET $BUILD_STD $RELEASE + - run: tools/test.sh -vv --tests $TARGET $BUILD_STD $RELEASE --exclude api-test env: RUSTDOCFLAGS: ${{ env.RUSTDOCFLAGS }} --cfg portable_atomic_outline_atomics RUSTFLAGS: ${{ env.RUSTFLAGS }} --cfg portable_atomic_outline_atomics if: startsWith(matrix.target, 'aarch64') && contains(matrix.target, '-musl') && matrix.flags == '' || startsWith(matrix.target, 'powerpc64-') - - run: tools/test.sh -vv --tests $TARGET $BUILD_STD $RELEASE + - run: tools/test.sh -vv --tests $TARGET $BUILD_STD $RELEASE --exclude api-test env: # Note: detect_false cfg is intended to make it easy for portable-atomic developers to # test cases such as has_cmpxchg16b == false, has_lse == false, @@ -398,6 +398,7 @@ jobs: matrix: include: - rust: '1.34' + - rust: '1.36' - rust: '1.59' - rust: stable - rust: beta @@ -519,6 +520,7 @@ jobs: persist-credentials: false - name: Install Rust run: rustup toolchain add nightly --no-self-update --component miri && rustup default nightly + - uses: taiki-e/install-action@cargo-hack # - run: sudo apt-get -o Acquire::Retries=10 -qq update && sudo apt-get -o Acquire::Retries=10 -o Dpkg::Use-Pty=0 install -y --no-install-recommends moreutils - run: echo "TARGET=--target=${{ matrix.target }}" >>"${GITHUB_ENV}" if: matrix.target != 'x86_64-unknown-linux-gnu' @@ -544,6 +546,7 @@ jobs: persist-credentials: false - name: Install Rust run: rustup toolchain add nightly --no-self-update && rustup default nightly + - uses: taiki-e/install-action@cargo-hack # - run: sudo apt-get -o Acquire::Retries=10 -qq update && sudo apt-get -o Acquire::Retries=10 -o Dpkg::Use-Pty=0 install -y --no-install-recommends moreutils - run: | echo "ASAN_OPTIONS=detect_stack_use_after_return=1" >>"${GITHUB_ENV}" @@ -578,6 +581,7 @@ jobs: persist-credentials: false - name: Install Rust run: rustup toolchain add nightly --no-self-update && rustup default nightly + - uses: taiki-e/install-action@cargo-hack # - run: sudo apt-get -o Acquire::Retries=10 -qq update && sudo apt-get -o Acquire::Retries=10 -o Dpkg::Use-Pty=0 install -y --no-install-recommends moreutils - uses: taiki-e/install-action@valgrind - run: tools/test.sh valgrind -vv diff --git a/portable-atomic-util/build.rs b/portable-atomic-util/build.rs index 468cd7a0c..7330b51b9 100644 --- a/portable-atomic-util/build.rs +++ b/portable-atomic-util/build.rs @@ -33,10 +33,18 @@ fn main() { if !version.probe(36, 2019, 4, 14) { println!("cargo:rustc-cfg=portable_atomic_no_alloc"); } + // MaybeUninit stabilized in Rust 1.36 (nightly-2019-05-21) https://github.com/rust-lang/rust/pull/60445 + if !version.probe(36, 2019, 5, 20) { + println!("cargo:rustc-cfg=portable_atomic_no_maybe_uninit"); + } // unsafe_op_in_unsafe_fn stabilized in Rust 1.52 (nightly-2021-03-11): https://github.com/rust-lang/rust/pull/79208 if !version.probe(52, 2021, 3, 10) { println!("cargo:rustc-cfg=portable_atomic_no_unsafe_op_in_unsafe_fn"); } + // https://github.com/rust-lang/rust/pull/84662 merged in Rust 1.56 (nightly-2021-08-02). + if !version.probe(56, 2021, 8, 1) { + println!("cargo:rustc-cfg=portable_atomic_no_core_unwind_safe"); + } if version.nightly { // `cfg(sanitize = "..")` is not stabilized. diff --git a/portable-atomic-util/src/arc.rs b/portable-atomic-util/src/arc.rs index 72a93b817..dcceac740 100644 --- a/portable-atomic-util/src/arc.rs +++ b/portable-atomic-util/src/arc.rs @@ -1,87 +1,78 @@ // SPDX-License-Identifier: Apache-2.0 OR MIT +// The module is based on alloc::sync::Arc, licensed under "Apache-2.0 OR MIT". +// +// The code has been adjusted to work with stable Rust. +// +// Source: https://github.com/rust-lang/rust/blob/5151b8c42712c473e7da56e213926b929d0212ef/library/alloc/src/sync.rs. +// +// Copyright & License of the original code: +// - https://github.com/rust-lang/rust/blob/5151b8c42712c473e7da56e213926b929d0212ef/COPYRIGHT +// - https://github.com/rust-lang/rust/blob/5151b8c42712c473e7da56e213926b929d0212ef/LICENSE-APACHE +// - https://github.com/rust-lang/rust/blob/5151b8c42712c473e7da56e213926b929d0212ef/LICENSE-MIT + //! An implementation of the standard library's `Arc` type using `portable-atomic`. +#![allow(clippy::undocumented_unsafe_blocks, clippy::must_use_candidate)] + use portable_atomic::{ - AtomicUsize, + self as atomic, hint, Ordering::{Acquire, Relaxed, Release}, }; -use alloc::boxed::Box; +use alloc::{alloc::handle_alloc_error, boxed::Box}; use core::{ - borrow::Borrow, - fmt, - hash::Hash, + alloc::Layout, + borrow, cmp, fmt, + hash::{Hash, Hasher}, isize, marker::PhantomData, - mem, + mem::{self, align_of_val, size_of_val}, ops::Deref, pin::Pin, ptr::{self, NonNull}, usize, }; -const MAX_REFCOUNT: usize = (isize::MAX as usize) - 1; +/// A soft limit on the amount of references that may be made to an `Arc`. +/// +/// Going above this limit will abort your program (although not +/// necessarily) at _exactly_ `MAX_REFCOUNT + 1` references. +/// Trying to go above it might call a `panic` (if not actually going above it). +/// +/// This is a global invariant, and also applies when using a compare-exchange loop. +/// +/// See comment in `Arc::clone`. +const MAX_REFCOUNT: usize = (isize::MAX) as usize; + +/// The error in case either counter reaches above `MAX_REFCOUNT`, and we can `panic` safely. +const INTERNAL_OVERFLOW_ERROR: &str = "Arc counter overflow"; #[cfg(not(portable_atomic_sanitize_thread))] macro_rules! acquire { - ($x:expr) => {{ - portable_atomic::fence(Acquire); - }}; + ($x:expr) => { + atomic::fence(Acquire) + }; } +// ThreadSanitizer does not support memory fences. To avoid false positive +// reports in Arc / Weak implementation use atomic loads for synchronization +// instead. #[cfg(portable_atomic_sanitize_thread)] macro_rules! acquire { - ($x:expr) => {{ - ($x).load(Acquire); - }}; -} - -/// The inner heap allocation of an `Arc`. -#[repr(C)] -struct Shared { - /// The reference count of the `Arc`. - header: Header, - - /// The value that is being reference counted. - value: T, -} - -#[repr(C)] -struct Header { - /// The current strong reference count. - /// - /// As long as this is greater than zero, the `value` is initialized. - strong: AtomicUsize, - - /// The weak reference count. - /// - /// This includes an intrinsic weak reference held by the strong pointers. - /// The allocation should be deallocated when this reaches zero. - weak: AtomicUsize, -} - -impl Shared { - /// Get the strong count. - fn strong(&self) -> &AtomicUsize { - &self.header.strong - } - - /// Get the weak count. - fn weak(&self) -> &AtomicUsize { - &self.header.weak - } + ($x:expr) => { + $x.load(Acquire) + }; } /// A thread-safe, strongly reference counted pointer. /// /// This is an equivalent to [`std::sync::Arc`], but using [`portable-atomic`] for synchronization. -/// See the documentation for the standard library's [`Arc`] for more details. +/// See the documentation for [`std::sync::Arc`] for more details. /// /// [`std::sync::Arc`]: https://doc.rust-lang.org/std/sync/struct.Arc.html /// [`portable-atomic`]: https://crates.io/crates/portable-atomic -/// [`Arc`]: https://doc.rust-lang.org/std/sync/struct.Arc.html /// /// # Examples /// @@ -100,20 +91,35 @@ impl Shared { /// # if cfg!(miri) { std::thread::sleep(std::time::Duration::from_millis(500)); } // wait for background threads closed: https://github.com/rust-lang/miri/issues/1371 /// ``` pub struct Arc { - /// The inner heap allocation. - shared: NonNull>, + ptr: NonNull>, + phantom: PhantomData>, +} + +unsafe impl Send for Arc {} +unsafe impl Sync for Arc {} + +#[cfg(not(portable_atomic_no_core_unwind_safe))] +impl core::panic::UnwindSafe for Arc {} +#[cfg(all(portable_atomic_no_core_unwind_safe, feature = "std"))] +impl std::panic::UnwindSafe for Arc {} - _marker: PhantomData>, +impl Arc { + unsafe fn from_inner(ptr: NonNull>) -> Self { + Self { ptr, phantom: PhantomData } + } + + unsafe fn from_ptr(ptr: *mut ArcInner) -> Self { + unsafe { Self::from_inner(NonNull::new_unchecked(ptr)) } + } } /// A weakly reference counted pointer. /// /// This is an equivalent to [`std::sync::Weak`], but using [`portable-atomic`] for synchronization. -/// See the documentation for the standard library's [`Weak`] for more details. +/// See the documentation for [`std::sync::Weak`] for more details. /// /// [`std::sync::Weak`]: https://doc.rust-lang.org/std/sync/struct.Weak.html /// [`portable-atomic`]: https://crates.io/crates/portable-atomic -/// [`Weak`]: https://doc.rust-lang.org/std/sync/struct.Weak.html /// /// # Examples /// @@ -132,703 +138,2195 @@ pub struct Arc { /// # t.join().unwrap(); // join thread to avoid https://github.com/rust-lang/miri/issues/1371 /// ``` pub struct Weak { - /// The inner heap allocation. - shared: NonNull>, + // This is a `NonNull` to allow optimizing the size of this type in enums, + // but it is not necessarily a valid pointer. + // `Weak::new` sets this to `usize::MAX` so that it doesn’t need + // to allocate space on the heap. That's not a value a real pointer + // will ever have because RcBox has alignment at least 2. + // This is only possible when `T: Sized`; unsized `T` never dangle. + ptr: NonNull>, } -// SAFETY: This value is accessible from many threads, it has to be Sync. -unsafe impl Send for Arc {} -// SAFETY: This value is accessible from many threads, it has to be Sync. -unsafe impl Sync for Arc {} -// SAFETY: This value is accessible from many threads, it has to be Sync. -unsafe impl Send for Weak {} -// SAFETY: This value is accessible from many threads, it has to be Sync. -unsafe impl Sync for Weak {} - -impl Unpin for Arc {} +unsafe impl Send for Weak {} +unsafe impl Sync for Weak {} -impl fmt::Debug for Arc { +impl fmt::Debug for Weak { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { - fmt::Debug::fmt(&**self, f) - } -} - -impl Clone for Arc { - fn clone(&self) -> Self { - // Bump the refcount. - let old_size = self.inner().strong().fetch_add(1, Relaxed); - - // Abort if the refcount overflowed. - if old_size > MAX_REFCOUNT { - abort(); - } - - // Return a new `Arc`. - // - // SAFETY: The refcount was incremented, so the allocation is still valid. - unsafe { Self::from_inner(self.shared) } - } -} - -impl Default for Arc { - fn default() -> Self { - Self::new(T::default()) - } -} - -impl> PartialEq> for Arc { - fn eq(&self, other: &Arc) -> bool { - PartialEq::eq(&**self, &**other) + write!(f, "(Weak)") } } -impl Eq for Arc {} +// This is repr(C) to future-proof against possible field-reordering, which +// would interfere with otherwise safe [into|from]_raw() of transmutable +// inner types. +#[repr(C)] +struct ArcInner { + strong: atomic::AtomicUsize, -impl> PartialOrd> for Arc { - fn partial_cmp(&self, other: &Arc) -> Option { - PartialOrd::partial_cmp(&**self, &**other) - } -} + // the value usize::MAX acts as a sentinel for temporarily "locking" the + // ability to upgrade weak pointers or downgrade strong ones; this is used + // to avoid races in `make_mut` and `get_mut`. + weak: atomic::AtomicUsize, -impl Ord for Arc { - fn cmp(&self, other: &Self) -> core::cmp::Ordering { - Ord::cmp(&**self, &**other) - } + data: T, } -impl Hash for Arc { - fn hash(&self, state: &mut H) { - (**self).hash(state); - } +/// Calculate layout for `ArcInner` using the inner value's layout +#[inline] +fn arc_inner_layout_for_value_layout(layout: Layout) -> Layout { + // Calculate layout using the given value layout. + // Previously, layout was calculated on the expression + // `&*(ptr as *const ArcInner)`, but this created a misaligned + // reference (see #54908). + pad_to_align(extend_layout(Layout::new::>(), layout).unwrap().0) } -impl Arc { - fn inner(&self) -> &Shared { - // SAFETY: self.shared is always a valid pointer to a `Shared`. - unsafe { self.shared.as_ref() } - } +unsafe impl Send for ArcInner {} +unsafe impl Sync for ArcInner {} - unsafe fn from_inner(ptr: NonNull>) -> Self { - Self { shared: ptr, _marker: PhantomData } +impl Arc { + /// Constructs a new `Arc`. + /// + /// # Examples + /// + /// ``` + /// use portable_atomic_util::Arc; + /// + /// let five = Arc::new(5); + /// ``` + #[inline] + pub fn new(data: T) -> Arc { + // Start the weak pointer count as 1 which is the weak pointer that's + // held by all the strong pointers (kinda), see std/rc.rs for more info + let x: Box<_> = Box::new(ArcInner { + strong: atomic::AtomicUsize::new(1), + weak: atomic::AtomicUsize::new(1), + data, + }); + unsafe { Self::from_inner(Box::leak(x).into()) } } - unsafe fn from_ptr(ptr: *mut Shared) -> Self { - // SAFETY: Invariant is checked by the caller. - unsafe { Self::from_inner(NonNull::new_unchecked(ptr)) } - } + /// Constructs a new `Arc` while giving you a `Weak` to the allocation, + /// to allow you to construct a `T` which holds a weak pointer to itself. + /// + /// Generally, a structure circularly referencing itself, either directly or + /// indirectly, should not hold a strong reference to itself to prevent a memory leak. + /// Using this function, you get access to the weak pointer during the + /// initialization of `T`, before the `Arc` is created, such that you can + /// clone and store it inside the `T`. + /// + /// `new_cyclic` first allocates the managed allocation for the `Arc`, + /// then calls your closure, giving it a `Weak` to this allocation, + /// and only afterwards completes the construction of the `Arc` by placing + /// the `T` returned from your closure into the allocation. + /// + /// Since the new `Arc` is not fully-constructed until `Arc::new_cyclic` + /// returns, calling [`upgrade`] on the weak reference inside your closure will + /// fail and result in a `None` value. + /// + /// # Panics + /// + /// If `data_fn` panics, the panic is propagated to the caller, and the + /// temporary [`Weak`] is dropped normally. + /// + /// # Example + /// + /// ``` + /// # #![allow(dead_code)] + /// use portable_atomic_util::{Arc, Weak}; + /// + /// struct Gadget { + /// me: Weak, + /// } + /// + /// impl Gadget { + /// /// Construct a reference counted Gadget. + /// fn new() -> Arc { + /// // `me` is a `Weak` pointing at the new allocation of the + /// // `Arc` we're constructing. + /// Arc::new_cyclic(|me| { + /// // Create the actual struct here. + /// Gadget { me: me.clone() } + /// }) + /// } + /// + /// /// Return a reference counted pointer to Self. + /// fn me(&self) -> Arc { + /// self.me.upgrade().unwrap() + /// } + /// } + /// ``` + /// [`upgrade`]: Weak::upgrade + #[cfg(not(portable_atomic_no_maybe_uninit))] + #[inline] + pub fn new_cyclic(data_fn: F) -> Arc + where + F: FnOnce(&Weak) -> T, + { + // Construct the inner in the "uninitialized" state with a single + // weak reference. + let uninit_ptr: NonNull<_> = Box::leak(Box::new(ArcInner { + strong: atomic::AtomicUsize::new(0), + weak: atomic::AtomicUsize::new(1), + data: mem::MaybeUninit::::uninit(), + })) + .into(); + let init_ptr = uninit_ptr.cast::>(); + + let weak = Weak { ptr: init_ptr }; + + // It's important we don't give up ownership of the weak pointer, or + // else the memory might be freed by the time `data_fn` returns. If + // we really wanted to pass ownership, we could create an additional + // weak pointer for ourselves, but this would result in additional + // updates to the weak reference count which might not be necessary + // otherwise. + let data = data_fn(&weak); + + // Now we can properly initialize the inner value and turn our weak + // reference into a strong reference. + let strong = unsafe { + let inner = init_ptr.as_ptr(); + let offset = data_offset(&data); + ptr::write(strict::byte_add(inner, offset) as *mut T, data); + + // The above write to the data field must be visible to any threads which + // observe a non-zero strong count. Therefore we need at least "Release" ordering + // in order to synchronize with the `compare_exchange_weak` in `Weak::upgrade`. + // + // "Acquire" ordering is not required. When considering the possible behaviors + // of `data_fn` we only need to look at what it could do with a reference to a + // non-upgradeable `Weak`: + // - It can *clone* the `Weak`, increasing the weak reference count. + // - It can drop those clones, decreasing the weak reference count (but never to zero). + // + // These side effects do not impact us in any way, and no other side effects are + // possible with safe code alone. + let prev_value = (*inner).strong.fetch_add(1, Release); + debug_assert_eq!(prev_value, 0, "No prior strong references should exist"); + + Arc::from_inner(init_ptr) + }; - unsafe fn get_mut_unchecked(this: &mut Self) -> &mut T { - // SAFETY: Since we have an exclusive reference, as certified by the caller, this - // dereference is valid. - unsafe { &mut (*this.shared.as_ptr()).value } + // Strong references should collectively own a shared weak reference, + // so don't run the destructor for our old weak reference. + mem::forget(weak); + strong } - #[inline(never)] - unsafe fn drop_slow(&mut self) { - // SAFETY: The refcount is zero, so these operations are valid. + #[cfg(not(portable_atomic_no_maybe_uninit))] + #[inline] + #[must_use] + fn new_uninit() -> Arc> { unsafe { - // Destroy the value itself. - ptr::drop_in_place(Self::get_mut_unchecked(self)); - - // Drop the intrinsic weak reference to deallocate. - drop(Weak::from_inner(self.shared)); + Arc::from_ptr(Arc::allocate_for_layout( + Layout::new::(), + |layout| Global.allocate(layout), + |ptr| ptr as *mut _, + )) } } - fn is_unique(&self) -> bool { - // "Lock" the weak counter so it can't be increased if we turn out to be the only - // strong reference. - if self.inner().weak().compare_exchange(1, usize::MAX, Acquire, Relaxed).is_ok() { - // There are no outside weak references, so we can check the strong count. - let strong = self.inner().strong().load(Acquire); - - // Restore the former weak count. - self.inner().weak().store(1, Release); - - // If there is only one strong reference, we are unique. - strong == 1 - } else { - // There are outside weak references, so we can't be unique. - false - } + /// Constructs a new `Pin>`. If `T` does not implement `Unpin`, then + /// `data` will be pinned in memory and unable to be moved. + #[must_use] + pub fn pin(data: T) -> Pin> { + unsafe { Pin::new_unchecked(Arc::new(data)) } } } impl Arc { - /// Create a new [`Arc`]. + /// Returns the inner value, if the `Arc` has exactly one strong reference. /// - /// # Example + /// Otherwise, an [`Err`] is returned with the same `Arc` that was + /// passed in. + /// + /// This will succeed even if there are outstanding weak references. + /// + /// It is strongly recommended to use [`Arc::into_inner`] instead if you don't + /// want to keep the `Arc` in the [`Err`] case. + /// Immediately dropping the [`Err`] payload, like in the expression + /// `Arc::try_unwrap(this).ok()`, can still cause the strong count to + /// drop to zero and the inner value of the `Arc` to be dropped: + /// For instance if two threads each execute this expression in parallel, then + /// there is a race condition. The threads could first both check whether they + /// have the last clone of their `Arc` via `Arc::try_unwrap`, and then + /// both drop their `Arc` in the call to [`ok`][`Result::ok`], + /// taking the strong count from two down to zero. + /// + /// # Examples /// /// ``` /// use portable_atomic_util::Arc; /// - /// let five = Arc::new(5); + /// let x = Arc::new(3); + /// assert_eq!(Arc::try_unwrap(x), Ok(3)); + /// + /// let x = Arc::new(4); + /// let _y = Arc::clone(&x); + /// assert_eq!(*Arc::try_unwrap(x).unwrap_err(), 4); /// ``` - pub fn new(item: T) -> Arc { - let shared = Box::into_raw(Box::new(Shared { - header: Header { strong: AtomicUsize::new(1), weak: AtomicUsize::new(1) }, - value: item, - })); + #[inline] + pub fn try_unwrap(this: Self) -> Result { + if this.inner().strong.compare_exchange(1, 0, Relaxed, Relaxed).is_err() { + return Err(this); + } + + acquire!(this.inner().strong); + + unsafe { + let elem = ptr::read(&this.ptr.as_ref().data); + + // Make a weak pointer to clean up the implicit strong-weak reference + let _weak = Weak { ptr: this.ptr }; + mem::forget(this); - // SAFETY: The newly created allocation is valid. - unsafe { Self::from_ptr(shared) } + Ok(elem) + } } - /// Create a new [`Arc`] whose pointer is pinned to the heap. + /// Returns the inner value, if the `Arc` has exactly one strong reference. /// - /// # Example + /// Otherwise, [`None`] is returned and the `Arc` is dropped. + /// + /// This will succeed even if there are outstanding weak references. + /// + /// If `Arc::into_inner` is called on every clone of this `Arc`, + /// it is guaranteed that exactly one of the calls returns the inner value. + /// This means in particular that the inner value is not dropped. + /// + /// The similar expression `Arc::try_unwrap(this).ok()` does not + /// offer such a guarantee. See the last example below + /// and the documentation of [`Arc::try_unwrap`]. + /// + /// # Examples + /// + /// Minimal example demonstrating the guarantee that `Arc::into_inner` gives. /// /// ``` /// use portable_atomic_util::Arc; /// - /// let five = Arc::pin(5); - /// ``` - pub fn pin(item: T) -> Pin> { - // SAFETY: The inner object is now pinned to the heap. - unsafe { Pin::new_unchecked(Self::new(item)) } - } - - /// Unwrap and try to get the inner value. + /// let x = Arc::new(3); + /// let y = Arc::clone(&x); /// - /// # Example + /// // Two threads calling `Arc::into_inner` on both clones of an `Arc`: + /// let x_thread = std::thread::spawn(|| Arc::into_inner(x)); + /// let y_thread = std::thread::spawn(|| Arc::into_inner(y)); /// + /// let x_inner_value = x_thread.join().unwrap(); + /// let y_inner_value = y_thread.join().unwrap(); + /// + /// // One of the threads is guaranteed to receive the inner value: + /// assert!(matches!((x_inner_value, y_inner_value), (None, Some(3)) | (Some(3), None))); + /// // The result could also be `(None, None)` if the threads called + /// // `Arc::try_unwrap(x).ok()` and `Arc::try_unwrap(y).ok()` instead. /// ``` - /// use portable_atomic_util::Arc; /// - /// let five = Arc::new(5); - /// assert_eq!(Arc::try_unwrap(five).unwrap(), 5); + /// A more practical example demonstrating the need for `Arc::into_inner`: + /// ``` + /// use portable_atomic_util::Arc; /// - /// let five = Arc::new(5); - /// let five2 = Arc::clone(&five); - /// assert!(Arc::try_unwrap(five).is_err()); + /// // Definition of a simple singly linked list using `Arc`: + /// #[derive(Clone)] + /// struct LinkedList(Option>>); + /// struct Node(T, Option>>); + /// + /// // Dropping a long `LinkedList` relying on the destructor of `Arc` + /// // can cause a stack overflow. To prevent this, we can provide a + /// // manual `Drop` implementation that does the destruction in a loop: + /// impl Drop for LinkedList { + /// fn drop(&mut self) { + /// let mut link = self.0.take(); + /// while let Some(arc_node) = link.take() { + /// if let Some(Node(_value, next)) = Arc::into_inner(arc_node) { + /// link = next; + /// } + /// } + /// } + /// } + /// + /// // Implementation of `new` and `push` omitted + /// impl LinkedList { + /// /* ... */ + /// # fn new() -> Self { + /// # LinkedList(None) + /// # } + /// # fn push(&mut self, x: T) { + /// # self.0 = Some(Arc::new(Node(x, self.0.take()))); + /// # } + /// } + /// + /// // The following code could have still caused a stack overflow + /// // despite the manual `Drop` impl if that `Drop` impl had used + /// // `Arc::try_unwrap(arc).ok()` instead of `Arc::into_inner(arc)`. + /// + /// // Create a long list and clone it + /// let mut x = LinkedList::new(); + /// # #[cfg(not(miri))] + /// for i in 0..100000 { + /// x.push(i); // Adds i to the front of x + /// } + /// # #[cfg(miri)] // Miri is slow + /// # for i in 0..100 { + /// # x.push(i); // Adds i to the front of x + /// # } + /// let y = x.clone(); + /// + /// // Drop the clones in parallel + /// let x_thread = std::thread::spawn(|| drop(x)); + /// let y_thread = std::thread::spawn(|| drop(y)); + /// x_thread.join().unwrap(); + /// y_thread.join().unwrap(); /// ``` - pub fn try_unwrap(this: Self) -> Result { - // Try to decrement the strong count. - if this.inner().strong().compare_exchange(1, 0, Relaxed, Relaxed).is_err() { - return Err(this); + #[inline] + pub fn into_inner(this: Self) -> Option { + // Make sure that the ordinary `Drop` implementation isn’t called as well + let mut this = mem::ManuallyDrop::new(this); + + // Following the implementation of `drop` and `drop_slow` + if this.inner().strong.fetch_sub(1, Release) != 1 { + return None; } - // Otherwise, we can safely unwrap the value. - acquire!(this.inner().strong()); + acquire!(this.inner().strong); - // SAFETY: These operations are valid since we hold the only reference. - unsafe { - let element = ptr::read(&this.inner().value); + // SAFETY: This mirrors the line + // + // unsafe { ptr::drop_in_place(Self::get_mut_unchecked(self)) }; + // + // in `drop_slow`. Instead of dropping the value behind the pointer, + // it is read and eventually returned; `ptr::read` has the same + // safety conditions as `ptr::drop_in_place`. - // Create a new weak pointer to deallocate. - let _weak = Weak::from_inner(this.shared); - mem::forget(this); + let inner = unsafe { ptr::read(Self::get_mut_unchecked(&mut this)) }; - Ok(element) - } + drop(Weak { ptr: this.ptr }); + + Some(inner) + } +} + +#[cfg(not(portable_atomic_no_maybe_uninit))] +impl Arc> { + #[must_use = "`self` will be dropped if the result is not used"] + #[inline] + unsafe fn assume_init(self) -> Arc { + let md_self = mem::ManuallyDrop::new(self); + unsafe { Arc::from_inner(md_self.ptr.cast::>()) } } } impl Arc { - /// Consume this `Arc` and get the raw pointer to the inner value. + /// Constructs an `Arc` from a raw pointer. /// - /// # Example + /// # Safety + /// + /// The raw pointer must have been previously returned by a call to + /// [`Arc::into_raw`][into_raw] where `U` must have the same size and + /// alignment as `T`. This is trivially true if `U` is `T`. + /// Note that if `U` is not `T` but has the same size and alignment, this is + /// basically like transmuting references of different types. See + /// [`mem::transmute`] for more information on what + /// restrictions apply in this case. + /// + /// The user of `from_raw` has to make sure a specific value of `T` is only + /// dropped once. + /// + /// This function is unsafe because improper use may lead to memory unsafety, + /// even if the returned `Arc` is never accessed. + /// + /// [into_raw]: Arc::into_raw + /// + /// # Examples /// /// ``` /// use portable_atomic_util::Arc; /// - /// let five = Arc::new(5u8); - /// let five_ptr = Arc::into_raw(five); + /// let x = Arc::new("hello".to_owned()); + /// let x_ptr = Arc::into_raw(x); + /// + /// unsafe { + /// // Convert back to an `Arc` to prevent leak. + /// let x = Arc::from_raw(x_ptr); + /// assert_eq!(&*x, "hello"); /// - /// // We should now free the pointer. - /// // SAFETY: The pointer is valid. - /// let five = unsafe { Arc::from_raw(five_ptr) }; - /// assert_eq!(&*five, &5u8); + /// // Further calls to `Arc::from_raw(x_ptr)` would be memory-unsafe. + /// } + /// + /// // The memory was freed when `x` went out of scope above, so `x_ptr` is now dangling! /// ``` - #[must_use] - pub fn into_raw(self) -> *const T { - // SAFETY: The refcount is incremented, so the allocation is still valid. - let ptr = self.as_ptr(); - mem::forget(self); - ptr + #[inline] + pub unsafe fn from_raw(ptr: *const T) -> Self { + unsafe { + let offset = data_offset(ptr); + + // Reverse the offset to find the original ArcInner. + let arc_ptr = strict::byte_sub(ptr as *mut T, offset) as *mut ArcInner; + + Self::from_ptr(arc_ptr) + } } - /// Get the raw pointer representing this `Arc`. + /// Increments the strong reference count on the `Arc` associated with the + /// provided pointer by one. /// - /// # Example + /// # Safety + /// + /// The pointer must have been obtained through `Arc::into_raw`, and the + /// associated `Arc` instance must be valid (i.e. the strong count must be at + /// least 1) for the duration of this method. + /// + /// # Examples /// /// ``` /// use portable_atomic_util::Arc; /// /// let five = Arc::new(5); - /// let five_ptr = Arc::as_ptr(&five); + /// + /// unsafe { + /// let ptr = Arc::into_raw(five); + /// Arc::increment_strong_count(ptr); + /// + /// // This assertion is deterministic because we haven't shared + /// // the `Arc` between threads. + /// let five = Arc::from_raw(ptr); + /// assert_eq!(2, Arc::strong_count(&five)); + /// # Arc::decrement_strong_count(ptr); // avoid memory leak + /// } /// ``` - #[must_use] - pub fn as_ptr(&self) -> *const T { - // Get offset of the value `T` into the allocation. - let offset = Self::value_offset(&**self); - - // Add the size of the header so that it points to the value. - let ptr = strict::map_addr(self.shared.as_ptr() as *mut u8, |addr| addr + offset); - - // Cast the pointer to the correct type. - strict::with_metadata_of(ptr, self.shared.as_ptr() as *mut T) + #[inline] + pub unsafe fn increment_strong_count(ptr: *const T) { + // Retain Arc, but don't touch refcount by wrapping in ManuallyDrop + let arc = unsafe { mem::ManuallyDrop::new(Arc::from_raw(ptr)) }; + // Now increase refcount, but don't drop new refcount either + let _arc_clone: mem::ManuallyDrop<_> = arc.clone(); } - /// Convert a raw pointer previously created by `into_raw` into a new `Arc`. + /// Decrements the strong reference count on the `Arc` associated with the + /// provided pointer by one. /// /// # Safety /// - /// This function can only be called with a pointer that was previously returned by `into_raw`. + /// The pointer must have been obtained through `Arc::into_raw`, and the + /// associated `Arc` instance must be valid (i.e. the strong count must be at + /// least 1) when invoking this method. This method can be used to release the final + /// `Arc` and backing storage, but **should not** be called after the final `Arc` has been + /// released. /// - /// # Example + /// # Examples /// - /// ```rust + /// ``` /// use portable_atomic_util::Arc; /// /// let five = Arc::new(5); - /// let five_ptr = Arc::into_raw(five); /// - /// // SAFETY: The pointer is valid. - /// let five = unsafe { Arc::from_raw(five_ptr) }; - /// assert_eq!(*five, 5); + /// unsafe { + /// let ptr = Arc::into_raw(five); + /// Arc::increment_strong_count(ptr); + /// + /// // Those assertions are deterministic because we haven't shared + /// // the `Arc` between threads. + /// let five = Arc::from_raw(ptr); + /// assert_eq!(2, Arc::strong_count(&five)); + /// Arc::decrement_strong_count(ptr); + /// assert_eq!(1, Arc::strong_count(&five)); + /// } /// ``` - #[must_use] - pub unsafe fn from_raw(ptr: *const T) -> Self { - // Get offset of the value `T` into the allocation. - let offset = Self::value_offset( - // Safety: The caller asserts that the pointer came from Arc::into_raw(), which should - // point to a valid instance of `T`. - unsafe { &*ptr }, - ); - - // Subtract the size of the header so that it points to the Shared allocation. - let new_ptr = strict::map_addr(ptr as *mut u8, |addr| addr - offset); - - // Cast the pointer to the correct type. - let shared = strict::with_metadata_of(new_ptr, ptr as *mut Shared); + #[inline] + pub unsafe fn decrement_strong_count(ptr: *const T) { + unsafe { drop(Arc::from_raw(ptr)) }; + } +} - // Safety: The caller ensures the original pointer came from a valid `Arc`. - unsafe { Self::from_ptr(shared) } +impl Arc { + /// Consumes the `Arc`, returning the wrapped pointer. + /// + /// To avoid a memory leak the pointer must be converted back to an `Arc` using + /// [`Arc::from_raw`]. + /// + /// # Examples + /// + /// ``` + /// use portable_atomic_util::Arc; + /// + /// let x = Arc::new("hello".to_owned()); + /// let x_ptr = Arc::into_raw(x); + /// assert_eq!(unsafe { &*x_ptr }, "hello"); + /// # unsafe { drop(Arc::from_raw(x_ptr)) } // avoid memory leak + /// ``` + #[must_use = "losing the pointer will leak memory"] + pub fn into_raw(this: Self) -> *const T { + let ptr = Self::as_ptr(&this); + mem::forget(this); + ptr } - /// Return the number of bytes into the `Shared` struct the held value is placed at. - fn value_offset(value: &T) -> usize { - // Get the alignment of `T`. - let align = mem::align_of_val::(value); + /// Provides a raw pointer to the data. + /// + /// The counts are not affected in any way and the `Arc` is not consumed. The pointer is valid for + /// as long as there are strong counts in the `Arc`. + /// + /// # Examples + /// + /// ``` + /// use portable_atomic_util::Arc; + /// + /// let x = Arc::new("hello".to_owned()); + /// let y = Arc::clone(&x); + /// let x_ptr = Arc::as_ptr(&x); + /// assert_eq!(x_ptr, Arc::as_ptr(&y)); + /// assert_eq!(unsafe { &*x_ptr }, "hello"); + /// ``` + #[must_use] + pub fn as_ptr(this: &Self) -> *const T { + let offset = unsafe { data_offset(&**this) }; - // Calculate the offset of the data relative to self. - let len = mem::size_of::
(); + let ptr: *mut ArcInner = NonNull::as_ptr(this.ptr); - // Shamelessly ripped off the unstable implementation of `Layout::padding_needed_for()`: - // - // Offset is: offset = (len + align - 1) & !(align - 1); - // - // We use modular arithmetic throughout: - // - // 1. align is guaranteed to be > 0, so align - 1 is never gonna overflow. - // - // 2. `len + align - 1` can overflow by at most `align - 1`, - // so the &-mask with `!(align - 1)` will ensure that in the - // case of overflow, `len_rounded_up` will itself be 0. - // Thus the returned padding, when added to `len`, yields 0, - // which trivially satisfies the alignment `align`. - // - // (Of course, attempts to allocate blocks of memory whose - // size and padding overflow in the above manner should cause - // the allocator to yield an error anyway.) - len.wrapping_add(align).wrapping_sub(1) & !align.wrapping_sub(1) + unsafe { strict::byte_add(ptr, offset) as *const T } } - /// Get a [`Weak`] reference from this `Arc`. + /// Creates a new [`Weak`] pointer to this allocation. /// - /// # Example + /// # Examples /// /// ``` /// use portable_atomic_util::Arc; /// /// let five = Arc::new(5); - /// let weak_five = Arc::downgrade(&five); /// - /// assert!(weak_five.upgrade().is_some()); + /// let weak_five = Arc::downgrade(&five); /// ``` + #[must_use = "this returns a new `Weak` pointer, \ + without modifying the original `Arc`"] #[allow(clippy::missing_panics_doc)] - #[must_use] pub fn downgrade(this: &Self) -> Weak { - let mut cur = this.inner().weak().load(Relaxed); + // This Relaxed is OK because we're checking the value in the CAS + // below. + let mut cur = this.inner().weak.load(Relaxed); loop { - // The weak counter may be "locked", so spin and reload if it is. + // check if the weak counter is currently "locked"; if so, spin. if cur == usize::MAX { - portable_atomic::hint::spin_loop(); - cur = this.inner().weak().load(Relaxed); + hint::spin_loop(); + cur = this.inner().weak.load(Relaxed); continue; } - // If the weak counter is greater than the maximum, panic. - // Panic instead of abort is okay because we didn't increment the weak counter yet. - assert!(cur <= MAX_REFCOUNT, "Arc counter overflow"); + // We can't allow the refcount to increase much past `MAX_REFCOUNT`. + assert!(cur <= MAX_REFCOUNT, "{}", INTERNAL_OVERFLOW_ERROR); - // Try to increment the weak counter. - match this.inner().weak().compare_exchange_weak(cur, cur + 1, Acquire, Relaxed) { + // NOTE: this code currently ignores the possibility of overflow + // into usize::MAX; in general both Rc and Arc need to be adjusted + // to deal with overflow. + + // Unlike with Clone(), we need this to be an Acquire read to + // synchronize with the write coming from `is_unique`, so that the + // events prior to that write happen before this read. + match this.inner().weak.compare_exchange_weak(cur, cur + 1, Acquire, Relaxed) { Ok(_) => { - // SAFETY: Now that the weak counter is incremented, the allocation for a weak pointer is valid. - return unsafe { Weak::from_inner(this.shared) }; + // Make sure we do not create a dangling Weak + debug_assert!(!is_dangling(this.ptr.as_ptr())); + return Weak { ptr: this.ptr }; } - Err(x) => cur = x, + Err(old) => cur = old, } } } - /// Get the number of weak pointers to this allocation. + /// Gets the number of [`Weak`] pointers to this allocation. /// - /// # Example + /// # Safety + /// + /// This method by itself is safe, but using it correctly requires extra care. + /// Another thread can change the weak count at any time, + /// including potentially between calling this method and acting on the result. + /// + /// # Examples /// /// ``` /// use portable_atomic_util::Arc; /// /// let five = Arc::new(5); - /// let weak_five = Arc::downgrade(&five); + /// let _weak_five = Arc::downgrade(&five); /// - /// assert_eq!(Arc::weak_count(&five), 1); + /// // This assertion is deterministic because we haven't shared + /// // the `Arc` or `Weak` between threads. + /// assert_eq!(1, Arc::weak_count(&five)); /// ``` + #[inline] #[must_use] pub fn weak_count(this: &Self) -> usize { - match this.inner().weak().load(Acquire) { - usize::MAX => 0, - cnt => cnt - 1, + let cnt = this.inner().weak.load(Relaxed); + // If the weak count is currently locked, the value of the + // count was 0 just before taking the lock. + if cnt == usize::MAX { + 0 + } else { + cnt - 1 } } - /// Get the number of strong pointers to this allocation. + /// Gets the number of strong (`Arc`) pointers to this allocation. /// - /// # Example + /// # Safety + /// + /// This method by itself is safe, but using it correctly requires extra care. + /// Another thread can change the strong count at any time, + /// including potentially between calling this method and acting on the result. + /// + /// # Examples /// /// ``` /// use portable_atomic_util::Arc; /// /// let five = Arc::new(5); - /// let five2 = Arc::clone(&five); + /// let _also_five = Arc::clone(&five); /// - /// assert_eq!(Arc::strong_count(&five), 2); + /// // This assertion is deterministic because we haven't shared + /// // the `Arc` between threads. + /// assert_eq!(2, Arc::strong_count(&five)); /// ``` + #[inline] #[must_use] pub fn strong_count(this: &Self) -> usize { - this.inner().strong().load(Acquire) + this.inner().strong.load(Relaxed) } - /// Increment the strong count of the `Arc` pointed to by `ptr` by one. - /// - /// # Safety - /// - /// The pointer must be a pointer previously returned by `Arc::into_raw`. + #[inline] + fn inner(&self) -> &ArcInner { + // This unsafety is ok because while this arc is alive we're guaranteed + // that the inner pointer is valid. Furthermore, we know that the + // `ArcInner` structure itself is `Sync` because the inner data is + // `Sync` as well, so we're ok loaning out an immutable pointer to these + // contents. + unsafe { self.ptr.as_ref() } + } + + // Non-inlined part of `drop`. + #[inline(never)] + unsafe fn drop_slow(&mut self) { + // Destroy the data at this time, even though we must not free the box + // allocation itself (there might still be weak pointers lying around). + unsafe { ptr::drop_in_place(Self::get_mut_unchecked(self)) }; + + // Drop the weak ref collectively held by all strong references + // Take a reference to `self.alloc` instead of cloning because 1. it'll + // last long enough, and 2. you should be able to drop `Arc`s with + // unclonable allocators + drop(Weak { ptr: self.ptr }); + } + + /// Returns `true` if the two `Arc`s point to the same allocation in a vein similar to + /// [`ptr::eq`]. This function ignores the metadata of `dyn Trait` pointers. /// - /// # Example + /// # Examples /// - /// ```rust + /// ``` /// use portable_atomic_util::Arc; /// /// let five = Arc::new(5); - /// let five_ptr = Arc::into_raw(five); + /// let same_five = Arc::clone(&five); + /// let other_five = Arc::new(5); /// - /// // SAFETY: The pointer is valid. - /// unsafe { Arc::increment_strong_count(five_ptr) }; + /// assert!(Arc::ptr_eq(&five, &same_five)); + /// assert!(!Arc::ptr_eq(&five, &other_five)); + /// ``` /// - /// // SAFETY: The pointer is valid. - /// let five2 = unsafe { Arc::from_raw(five_ptr) }; - /// assert_eq!(*five2, 5); + /// [`ptr::eq`]: core::ptr::eq "ptr::eq" + #[inline] + #[must_use] + pub fn ptr_eq(this: &Self, other: &Self) -> bool { + ptr::eq(this.ptr.as_ptr() as *const (), other.ptr.as_ptr() as *const ()) + } +} + +impl Arc { + /// Allocates an `ArcInner` with sufficient space for + /// a possibly-unsized inner value where the value has the layout provided. /// - /// // SAFETY: Since the refcount is incremented, we can get another. - /// let five3 = unsafe { Arc::from_raw(five_ptr) }; - /// assert_eq!(*five3, 5); - /// ``` - pub unsafe fn increment_strong_count(ptr: *const T) { - // Retain Arc, but don't touch refcount by wrapping in ManuallyDrop - // - // SAFETY: The caller must ensure that the pointer is valid. - let arc = unsafe { mem::ManuallyDrop::new(Arc::::from_raw(ptr)) }; + /// The function `mem_to_arc_inner` is called with the data pointer + /// and must return back a (potentially fat)-pointer for the `ArcInner`. + unsafe fn allocate_for_layout( + value_layout: Layout, + allocate: impl FnOnce(Layout) -> Option>, + mem_to_arc_inner: impl FnOnce(*mut u8) -> *mut ArcInner, + ) -> *mut ArcInner { + let layout = arc_inner_layout_for_value_layout(value_layout); - // Now increase refcount, but don't drop new refcount either - let _arc_clone: mem::ManuallyDrop<_> = arc.clone(); + let ptr = allocate(layout).unwrap_or_else(|| handle_alloc_error(layout)); + + unsafe { Self::initialize_arc_inner(ptr, layout, mem_to_arc_inner) } } - /// Decrement the strong count of the `Arc` pointed to by `ptr` by one. - /// - /// # Safety + unsafe fn initialize_arc_inner( + ptr: NonNull, + layout: Layout, + mem_to_arc_inner: impl FnOnce(*mut u8) -> *mut ArcInner, + ) -> *mut ArcInner { + let inner = mem_to_arc_inner(ptr.as_ptr()); + debug_assert_eq!(unsafe { Layout::for_value(&*inner) }, layout); + + // SAFETY: mem_to_arc_inner return a valid pointer to uninitialized ArcInner. + // ArcInner is repr(C), and strong and weak are the first and second fields and + // are the same type, so inner.cast::() is strong and + // inner.cast::().add(1) is weak. + unsafe { + let strong = inner.cast::(); + strong.write(atomic::AtomicUsize::new(1)); + let weak = strong.add(1); + weak.write(atomic::AtomicUsize::new(1)); + } + + inner + } +} + +impl Arc { + /// Allocates an `ArcInner` with sufficient space for an unsized inner value. + #[inline] + unsafe fn allocate_for_ptr(ptr: *const T) -> *mut ArcInner { + // Allocate for the `ArcInner` using the given value. + unsafe { + Arc::allocate_for_layout( + Layout::for_value(&*ptr), + |layout| Global.allocate(layout), + |mem| strict::with_metadata_of(mem, ptr as *mut ArcInner), + ) + } + } + + fn from_box(src: Box) -> Arc { + unsafe { + let value_size = size_of_val(&*src); + let offset = data_offset(&*src); + let ptr = Self::allocate_for_ptr(&*src); + + // Copy value as bytes + ptr::copy_nonoverlapping( + &*src as *const T as *const u8, + strict::byte_add(ptr, offset) as *mut u8, + value_size, + ); + + // Free the allocation without dropping its contents + let box_ptr = Box::into_raw(src); + let src = Box::from_raw(box_ptr as *mut mem::ManuallyDrop); + drop(src); + + Self::from_ptr(ptr) + } + } +} + +// Layout::array requires Rust 1.44 +// impl Arc<[T]> { +// /// Allocates an `ArcInner<[T]>` with the given length. +// unsafe fn allocate_for_slice(len: usize) -> *mut ArcInner<[T]> { +// unsafe { +// Self::allocate_for_layout( +// Layout::array::(len).unwrap(), +// |layout| Global.allocate(layout), +// // ptr::slice_from_raw_parts_mut requires Rust 1.42 +// |mem| slice::from_raw_parts_mut(mem as *mut T, len) as *mut _ as *mut ArcInner<[T]>, +// ) +// } +// } + +// /// Constructs an `Arc<[T]>` from an iterator known to be of a certain size. +// /// +// /// Behavior is undefined should the size be wrong. +// unsafe fn from_iter_exact(iter: impl Iterator, len: usize) -> Arc<[T]> { +// // Panic guard while cloning T elements. +// // In the event of a panic, elements that have been written +// // into the new ArcInner will be dropped, then the memory freed. +// struct Guard { +// mem: NonNull, +// elems: *mut T, +// layout: Layout, +// n_elems: usize, +// } + +// impl Drop for Guard { +// fn drop(&mut self) { +// unsafe { +// let slice = from_raw_parts_mut(self.elems, self.n_elems); +// ptr::drop_in_place(slice); + +// Global.deallocate(self.mem, self.layout); +// } +// } +// } + +// unsafe { +// let ptr = Self::allocate_for_slice(len); + +// let mem = ptr as *mut _ as *mut u8; +// let layout = Layout::for_value(&*ptr); + +// // Pointer to first element +// let elems = &mut (*ptr).data as *mut [T] as *mut T; + +// let mut guard = Guard { mem: NonNull::new_unchecked(mem), elems, layout, n_elems: 0 }; + +// for (i, item) in iter.enumerate() { +// ptr::write(elems.add(i), item); +// guard.n_elems += 1; +// } + +// // All clear. Forget the guard so it doesn't free the new ArcInner. +// mem::forget(guard); + +// Self::from_ptr(ptr) +// } +// } +// } + +impl Clone for Arc { + /// Makes a clone of the `Arc` pointer. /// - /// The pointer must be a pointer previously returned by `Arc::into_raw`. + /// This creates another pointer to the same allocation, increasing the + /// strong reference count. /// - /// # Example + /// # Examples /// - /// ```rust + /// ``` /// use portable_atomic_util::Arc; /// /// let five = Arc::new(5); - /// let five2 = Arc::clone(&five); /// - /// let five_ptr = Arc::into_raw(five); - /// - /// // SAFETY: The pointer is valid. - /// unsafe { Arc::decrement_strong_count(five_ptr) }; + /// let _ = Arc::clone(&five); /// ``` - pub unsafe fn decrement_strong_count(ptr: *const T) { - // SAFETY: The caller must ensure that the pointer is valid. - unsafe { mem::drop(Arc::from_raw(ptr)) }; + #[inline] + fn clone(&self) -> Self { + // Using a relaxed ordering is alright here, as knowledge of the + // original reference prevents other threads from erroneously deleting + // the object. + // + // As explained in the [Boost documentation][1], Increasing the + // reference counter can always be done with memory_order_relaxed: New + // references to an object can only be formed from an existing + // reference, and passing an existing reference from one thread to + // another must already provide any required synchronization. + // + // [1]: (www.boost.org/doc/libs/1_55_0/doc/html/atomic/usage_examples.html) + let old_size = self.inner().strong.fetch_add(1, Relaxed); + + // However we need to guard against massive refcounts in case someone is `mem::forget`ing + // Arcs. If we don't do this the count can overflow and users will use-after free. This + // branch will never be taken in any realistic program. We abort because such a program is + // incredibly degenerate, and we don't care to support it. + // + // This check is not 100% water-proof: we error when the refcount grows beyond `isize::MAX`. + // But we do that check *after* having done the increment, so there is a chance here that + // the worst already happened and we actually do overflow the `usize` counter. However, that + // requires the counter to grow from `isize::MAX` to `usize::MAX` between the increment + // above and the `abort` below, which seems exceedingly unlikely. + // + // This is a global invariant, and also applies when using a compare-exchange loop to increment + // counters in other methods. + // Otherwise, the counter could be brought to an almost-overflow using a compare-exchange loop, + // and then overflow using a few `fetch_add`s. + if old_size > MAX_REFCOUNT { + abort(); + } + + unsafe { Self::from_inner(self.ptr) } } +} + +impl Deref for Arc { + type Target = T; - /// Tell if two `Arc`s point to the same allocation. + #[inline] + fn deref(&self) -> &Self::Target { + &self.inner().data + } +} + +impl Arc { + /// Makes a mutable reference into the given `Arc`. /// - /// # Example + /// If there are other `Arc` pointers to the same allocation, then `make_mut` will + /// [`clone`] the inner value to a new allocation to ensure unique ownership. This is also + /// referred to as clone-on-write. + /// + /// However, if there are no other `Arc` pointers to this allocation, but some [`Weak`] + /// pointers, then the [`Weak`] pointers will be dissociated and the inner value will not + /// be cloned. + /// + /// See also [`get_mut`], which will fail rather than cloning the inner value + /// or dissociating [`Weak`] pointers. + /// + /// [`clone`]: Clone::clone + /// [`get_mut`]: Arc::get_mut + /// + /// # Examples /// /// ``` /// use portable_atomic_util::Arc; /// - /// let five = Arc::new(5); - /// let five2 = Arc::clone(&five); + /// let mut data = Arc::new(5); + /// + /// *Arc::make_mut(&mut data) += 1; // Won't clone anything + /// let mut other_data = Arc::clone(&data); // Won't clone inner data + /// *Arc::make_mut(&mut data) += 1; // Clones inner data + /// *Arc::make_mut(&mut data) += 1; // Won't clone anything + /// *Arc::make_mut(&mut other_data) *= 2; // Won't clone anything /// - /// assert!(Arc::ptr_eq(&five, &five2)); + /// // Now `data` and `other_data` point to different allocations. + /// assert_eq!(*data, 8); + /// assert_eq!(*other_data, 12); /// ``` - #[must_use] - pub fn ptr_eq(this: &Self, other: &Self) -> bool { - this.shared == other.shared - } - - /// Get a mutable pointer to the inner value if there are no other strong references. /// - /// # Example + /// [`Weak`] pointers will be dissociated: /// /// ``` /// use portable_atomic_util::Arc; /// - /// let mut five = Arc::new(5); - /// assert!(Arc::get_mut(&mut five).is_some()); + /// let mut data = Arc::new(75); + /// let weak = Arc::downgrade(&data); /// - /// let five2 = Arc::clone(&five); - /// assert!(Arc::get_mut(&mut five).is_none()); + /// assert!(75 == *data); + /// assert!(75 == *weak.upgrade().unwrap()); + /// + /// *Arc::make_mut(&mut data) += 1; + /// + /// assert!(76 == *data); + /// assert!(weak.upgrade().is_none()); /// ``` - #[must_use] - pub fn get_mut(this: &mut Self) -> Option<&mut T> { - if this.is_unique() { - // SAFETY: The pointer is unique. - unsafe { Some(Self::get_mut_unchecked(this)) } + #[cfg(not(portable_atomic_no_maybe_uninit))] + #[inline] + pub fn make_mut(this: &mut Self) -> &mut T { + // Note that we hold both a strong reference and a weak reference. + // Thus, releasing our strong reference only will not, by itself, cause + // the memory to be deallocated. + // + // Use Acquire to ensure that we see any writes to `weak` that happen + // before release writes (i.e., decrements) to `strong`. Since we hold a + // weak count, there's no chance the ArcInner itself could be + // deallocated. + if this.inner().strong.compare_exchange(1, 0, Acquire, Relaxed).is_err() { + // Another strong pointer exists, so we must clone. + // Pre-allocate memory to allow writing the cloned value directly. + let mut arc = Self::new_uninit(); + unsafe { + let data = Arc::get_mut_unchecked(&mut arc); + data.as_mut_ptr().write((**this).clone()); + *this = arc.assume_init(); + } + } else if this.inner().weak.load(Relaxed) != 1 { + // Relaxed suffices in the above because this is fundamentally an + // optimization: we are always racing with weak pointers being + // dropped. Worst case, we end up allocated a new Arc unnecessarily. + + // We removed the last strong ref, but there are additional weak + // refs remaining. We'll move the contents to a new Arc, and + // invalidate the other weak refs. + + // Note that it is not possible for the read of `weak` to yield + // usize::MAX (i.e., locked), since the weak count can only be + // locked by a thread with a strong reference. + + // Materialize our own implicit weak pointer, so that it can clean + // up the ArcInner as needed. + let _weak = Weak { ptr: this.ptr }; + + // Can just steal the data, all that's left are `Weak`s + let mut arc = Self::new_uninit(); + unsafe { + let data = Arc::get_mut_unchecked(&mut arc); + data.as_mut_ptr().copy_from_nonoverlapping(&**this, 1); + ptr::write(this, arc.assume_init()); + } } else { - None + // We were the sole reference of either kind; bump back up the + // strong ref count. + this.inner().strong.store(1, Release); } + + // As with `get_mut()`, the unsafety is ok because our reference was + // either unique to begin with, or became one upon cloning the contents. + unsafe { Self::get_mut_unchecked(this) } } -} -impl Arc { - /// Try to get the inner value or clone it. + /// If we have the only reference to `T` then unwrap it. Otherwise, clone `T` and return the + /// clone. /// - /// # Example + /// Assuming `arc_t` is of type `Arc`, this function is functionally equivalent to + /// `(*arc_t).clone()`, but will avoid cloning the inner value where possible. + /// + /// # Examples /// /// ``` /// use portable_atomic_util::Arc; - /// - /// let five = Arc::new(5); - /// let five2 = Arc::clone(&five); - /// - /// assert_eq!(Arc::unwrap_or_clone(five), 5); + /// use std::ptr; + /// + /// let inner = String::from("test"); + /// let ptr = inner.as_ptr(); + /// + /// let arc = Arc::new(inner); + /// let inner = Arc::unwrap_or_clone(arc); + /// // The inner value was not cloned + /// assert!(ptr::eq(ptr, inner.as_ptr())); + /// + /// let arc = Arc::new(inner); + /// let arc2 = arc.clone(); + /// let inner = Arc::unwrap_or_clone(arc); + /// // Because there were 2 references, we had to clone the inner value. + /// assert!(!ptr::eq(ptr, inner.as_ptr())); + /// // `arc2` is the last reference, so when we unwrap it we get back + /// // the original `String`. + /// let inner = Arc::unwrap_or_clone(arc2); + /// assert!(ptr::eq(ptr, inner.as_ptr())); /// ``` - #[must_use] + #[inline] pub fn unwrap_or_clone(this: Self) -> T { - Self::try_unwrap(this).unwrap_or_else(|this| (*this).clone()) + Arc::try_unwrap(this).unwrap_or_else(|arc| (*arc).clone()) } } -impl Deref for Arc { - type Target = T; - - fn deref(&self) -> &Self::Target { - &self.inner().value +impl Arc { + /// Returns a mutable reference into the given `Arc`, if there are + /// no other `Arc` or [`Weak`] pointers to the same allocation. + /// + /// Returns [`None`] otherwise, because it is not safe to + /// mutate a shared value. + /// + /// See also [`make_mut`][make_mut], which will [`clone`][clone] + /// the inner value when there are other `Arc` pointers. + /// + /// [make_mut]: Arc::make_mut + /// [clone]: Clone::clone + /// + /// # Examples + /// + /// ``` + /// use portable_atomic_util::Arc; + /// + /// let mut x = Arc::new(3); + /// *Arc::get_mut(&mut x).unwrap() = 4; + /// assert_eq!(*x, 4); + /// + /// let _y = Arc::clone(&x); + /// assert!(Arc::get_mut(&mut x).is_none()); + /// ``` + #[inline] + pub fn get_mut(this: &mut Self) -> Option<&mut T> { + if this.is_unique() { + // This unsafety is ok because we're guaranteed that the pointer + // returned is the *only* pointer that will ever be returned to T. Our + // reference count is guaranteed to be 1 at this point, and we required + // the Arc itself to be `mut`, so we're returning the only possible + // reference to the inner data. + unsafe { Some(Arc::get_mut_unchecked(this)) } + } else { + None + } } -} -impl AsRef for Arc { - fn as_ref(&self) -> &T { - &self.inner().value + #[inline] + unsafe fn get_mut_unchecked(this: &mut Self) -> &mut T { + // We are careful to *not* create a reference covering the "count" fields, as + // this would alias with concurrent access to the reference counts (e.g. by `Weak`). + unsafe { &mut (*this.ptr.as_ptr()).data } } -} -impl Borrow for Arc { - fn borrow(&self) -> &T { - &self.inner().value + /// Determine whether this is the unique reference (including weak refs) to + /// the underlying data. + /// + /// Note that this requires locking the weak ref count. + fn is_unique(&mut self) -> bool { + // lock the weak pointer count if we appear to be the sole weak pointer + // holder. + // + // The acquire label here ensures a happens-before relationship with any + // writes to `strong` (in particular in `Weak::upgrade`) prior to decrements + // of the `weak` count (via `Weak::drop`, which uses release). If the upgraded + // weak ref was never dropped, the CAS here will fail so we do not care to synchronize. + if self.inner().weak.compare_exchange(1, usize::MAX, Acquire, Relaxed).is_ok() { + // This needs to be an `Acquire` to synchronize with the decrement of the `strong` + // counter in `drop` -- the only access that happens when any but the last reference + // is being dropped. + let unique = self.inner().strong.load(Acquire) == 1; + + // The release write here synchronizes with a read in `downgrade`, + // effectively preventing the above read of `strong` from happening + // after the write. + self.inner().weak.store(1, Release); // release the lock + unique + } else { + false + } } } impl Drop for Arc { + /// Drops the `Arc`. + /// + /// This will decrement the strong reference count. If the strong reference + /// count reaches zero then the only other references (if any) are + /// [`Weak`], so we `drop` the inner value. + /// + /// # Examples + /// + /// ``` + /// use portable_atomic_util::Arc; + /// + /// struct Foo; + /// + /// impl Drop for Foo { + /// fn drop(&mut self) { + /// println!("dropped!"); + /// } + /// } + /// + /// let foo = Arc::new(Foo); + /// let foo2 = Arc::clone(&foo); + /// + /// drop(foo); // Doesn't print anything + /// drop(foo2); // Prints "dropped!" + /// ``` + #[inline] fn drop(&mut self) { - // Decrement the strong refcount. - if self.inner().strong().fetch_sub(1, Release) != 1 { + // Because `fetch_sub` is already atomic, we do not need to synchronize + // with other threads unless we are going to delete the object. This + // same logic applies to the below `fetch_sub` to the `weak` count. + if self.inner().strong.fetch_sub(1, Release) != 1 { return; } - // Ensure we're synchronized with other threads. - acquire!(self.inner().strong()); - - // Drop the value and deallocate. + // This fence is needed to prevent reordering of use of the data and + // deletion of the data. Because it is marked `Release`, the decreasing + // of the reference count synchronizes with this `Acquire` fence. This + // means that use of the data happens before decreasing the reference + // count, which happens before this fence, which happens before the + // deletion of the data. + // + // As explained in the [Boost documentation][1], + // + // > It is important to enforce any possible access to the object in one + // > thread (through an existing reference) to *happen before* deleting + // > the object in a different thread. This is achieved by a "release" + // > operation after dropping a reference (any access to the object + // > through this reference must obviously happened before), and an + // > "acquire" operation before deleting the object. + // + // In particular, while the contents of an Arc are usually immutable, it's + // possible to have interior writes to something like a Mutex. Since a + // Mutex is not acquired when it is deleted, we can't rely on its + // synchronization logic to make writes in thread A visible to a destructor + // running in thread B. // - // SAFETY: Since the strong count is zero, we have exclusive access to the inner value. - unsafe { self.drop_slow() } + // Also note that the Acquire fence here could probably be replaced with an + // Acquire load, which could improve performance in highly-contended + // situations. See [2]. + // + // [1]: (www.boost.org/doc/libs/1_55_0/doc/html/atomic/usage_examples.html) + // [2]: (https://github.com/rust-lang/rust/pull/41714) + acquire!(self.inner().strong); + + unsafe { + self.drop_slow(); + } } } -impl fmt::Debug for Weak { - fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { - f.write_str("Weak") +// TODO: CoerceUnsized is needed to cast Arc -> Arc +// impl Arc { +// /// Attempt to downcast the `Arc` to a concrete type. +// /// +// /// # Examples +// /// +// /// ``` +// /// use portable_atomic_util::Arc; +// /// use std::any::Any; +// /// +// /// fn print_if_string(value: Arc) { +// /// if let Ok(string) = value.downcast::() { +// /// println!("String ({}): {}", string.len(), string); +// /// } +// /// } +// /// +// /// let my_string = "Hello World".to_string(); +// /// print_if_string(Arc::new(my_string)); +// /// print_if_string(Arc::new(0i8)); +// /// ``` +// #[inline] +// pub fn downcast(self) -> Result, Self> +// where +// T: Any + Send + Sync, +// { +// if (*self).is::() { +// unsafe { +// let ptr = self.ptr.cast::>(); +// mem::forget(self); +// Ok(Arc::from_inner(ptr)) +// } +// } else { +// Err(self) +// } +// } +// } + +impl Weak { + /// Constructs a new `Weak`, without allocating any memory. + /// Calling [`upgrade`] on the return value always gives [`None`]. + /// + /// [`upgrade`]: Weak::upgrade + /// + /// # Examples + /// + /// ``` + /// use portable_atomic_util::Weak; + /// + /// let empty: Weak = Weak::new(); + /// assert!(empty.upgrade().is_none()); + /// ``` + #[inline] + #[must_use] + pub const fn new() -> Weak { + Weak { ptr: unsafe { NonNull::new_unchecked(strict::invalid::>(usize::MAX)) } } } } -impl Clone for Weak { - fn clone(&self) -> Self { - // Get the inner header. - let header = match self.inner() { - Some(header) => header, - None => { - // SAFETY: We can duplicate the dangling pointer as we please. - return unsafe { Self::from_inner(self.shared) }; - } - }; +/// Helper type to allow accessing the reference counts without +/// making any assertions about the data field. +struct WeakInner<'a> { + weak: &'a atomic::AtomicUsize, + strong: &'a atomic::AtomicUsize, +} - // Bump the weak refcount. - let old_size = header.weak.fetch_add(1, Relaxed); +impl Weak { + /// Converts a raw pointer previously created by [`into_raw`] back into `Weak`. + /// + /// This can be used to safely get a strong reference (by calling [`upgrade`] + /// later) or to deallocate the weak count by dropping the `Weak`. + /// + /// It takes ownership of one weak reference (with the exception of pointers created by [`new`], + /// as these don't own anything; the method still works on them). + /// + /// # Safety + /// + /// The pointer must have originated from the [`into_raw`] and must still own its potential + /// weak reference. + /// + /// It is allowed for the strong count to be 0 at the time of calling this. Nevertheless, this + /// takes ownership of one weak reference currently represented as a raw pointer (the weak + /// count is not modified by this operation) and therefore it must be paired with a previous + /// call to [`into_raw`]. + /// # Examples + /// + /// ``` + /// use portable_atomic_util::{Arc, Weak}; + /// + /// let strong = Arc::new("hello".to_owned()); + /// + /// let raw_1 = Arc::downgrade(&strong).into_raw(); + /// let raw_2 = Arc::downgrade(&strong).into_raw(); + /// + /// assert_eq!(2, Arc::weak_count(&strong)); + /// + /// assert_eq!("hello", &*unsafe { Weak::from_raw(raw_1) }.upgrade().unwrap()); + /// assert_eq!(1, Arc::weak_count(&strong)); + /// + /// drop(strong); + /// + /// // Decrement the last weak count. + /// assert!(unsafe { Weak::from_raw(raw_2) }.upgrade().is_none()); + /// ``` + /// + /// [`new`]: Weak::new + /// [`into_raw`]: Weak::into_raw + /// [`upgrade`]: Weak::upgrade + #[inline] + pub unsafe fn from_raw(ptr: *const T) -> Self { + // See Weak::as_ptr for context on how the input pointer is derived. - // Abort if the refcount overflowed. - if old_size > MAX_REFCOUNT { - abort(); - } + let ptr = if is_dangling(ptr as *mut T) { + // This is a dangling Weak. + ptr as *mut ArcInner + } else { + // Otherwise, we're guaranteed the pointer came from a non-dangling Weak. + // SAFETY: data_offset is safe to call, as ptr references a real (potentially dropped) T. + let offset = unsafe { data_offset(ptr) }; + // Thus, we reverse the offset to get the whole RcBox. + // SAFETY: the pointer originated from a Weak, so this offset is safe. + unsafe { strict::byte_sub(ptr as *mut T, offset) as *mut ArcInner } + }; - // Return a new `Weak`. - // - // SAFETY: self.shared is guaranteed to be valid now that the refcount is raised. - unsafe { Self::from_inner(self.shared) } + // SAFETY: we now have recovered the original Weak pointer, so can create the Weak. + Weak { ptr: unsafe { NonNull::new_unchecked(ptr) } } } } impl Weak { - fn inner(&self) -> Option<&Header> { - let ptr = self.shared.as_ptr(); + /// Returns a raw pointer to the object `T` pointed to by this `Weak`. + /// + /// The pointer is valid only if there are some strong references. The pointer may be dangling, + /// unaligned or even [`null`] otherwise. + /// + /// # Examples + /// + /// ``` + /// use portable_atomic_util::Arc; + /// use std::ptr; + /// + /// let strong = Arc::new("hello".to_owned()); + /// let weak = Arc::downgrade(&strong); + /// // Both point to the same object + /// assert!(ptr::eq(&*strong, weak.as_ptr())); + /// // The strong here keeps it alive, so we can still access the object. + /// assert_eq!("hello", unsafe { &*weak.as_ptr() }); + /// + /// drop(strong); + /// // But not any more. We can do weak.as_ptr(), but accessing the pointer would lead to + /// // undefined behavior. + /// // assert_eq!("hello", unsafe { &*weak.as_ptr() }); + /// ``` + /// + /// [`null`]: core::ptr::null "ptr::null" + #[must_use] + pub fn as_ptr(&self) -> *const T { + let ptr: *mut ArcInner = NonNull::as_ptr(self.ptr); + if is_dangling(ptr) { - // We are dangling. - None + // If the pointer is dangling, we return the sentinel directly. This cannot be + // a valid payload address, as the payload is at least as aligned as ArcInner (usize). + ptr as *const T } else { - // SAFETY: We know ptr is a valid pointer. - unsafe { - // Make sure we never take a reference to the `value` itself. - // - // SAFETY: Due to repr(C), Header will always come first. We take a reference - // to that. - let ptr = ptr as *mut Header; - Some(&*ptr) - } + let offset = unsafe { data_offset(&(*ptr).data) }; + + // SAFETY: if is_dangling returns false, then the pointer is dereferenceable. + // The payload may be dropped at this point, and we have to maintain provenance, + // so use raw pointer manipulation. + unsafe { strict::byte_add(ptr, offset) as *const T } } } - unsafe fn from_inner(ptr: NonNull>) -> Self { - Self { shared: ptr } + /// Consumes the `Weak` and turns it into a raw pointer. + /// + /// This converts the weak pointer into a raw pointer, while still preserving the ownership of + /// one weak reference (the weak count is not modified by this operation). It can be turned + /// back into the `Weak` with [`from_raw`]. + /// + /// The same restrictions of accessing the target of the pointer as with + /// [`as_ptr`] apply. + /// + /// # Examples + /// + /// ``` + /// use portable_atomic_util::{Arc, Weak}; + /// + /// let strong = Arc::new("hello".to_owned()); + /// let weak = Arc::downgrade(&strong); + /// let raw = weak.into_raw(); + /// + /// assert_eq!(1, Arc::weak_count(&strong)); + /// assert_eq!("hello", unsafe { &*raw }); + /// + /// drop(unsafe { Weak::from_raw(raw) }); + /// assert_eq!(0, Arc::weak_count(&strong)); + /// ``` + /// + /// [`from_raw`]: Weak::from_raw + /// [`as_ptr`]: Weak::as_ptr + #[must_use = "`self` will be dropped if the result is not used"] + pub fn into_raw(self) -> *const T { + let result = self.as_ptr(); + mem::forget(self); + result } +} - /// Try to upgrade this `Weak` pointer to a strong pointer. +impl Weak { + /// Attempts to upgrade the `Weak` pointer to an [`Arc`], delaying + /// dropping of the inner value if successful. /// - /// # Example + /// Returns [`None`] if the inner value has since been dropped. + /// + /// # Examples /// /// ``` /// use portable_atomic_util::Arc; /// /// let five = Arc::new(5); /// - /// let weak = Arc::downgrade(&five); - /// assert!(weak.upgrade().is_some()); + /// let weak_five = Arc::downgrade(&five); + /// + /// let strong_five: Option> = weak_five.upgrade(); + /// assert!(strong_five.is_some()); + /// + /// // Destroy all strong pointers. + /// drop(strong_five); + /// drop(five); + /// + /// assert!(weak_five.upgrade().is_none()); /// ``` - #[allow(clippy::missing_panics_doc)] - #[must_use] + #[must_use = "this returns a new `Arc`, \ + without modifying the original weak pointer"] pub fn upgrade(&self) -> Option> { - let header = self.inner()?; - let mut strong = header.strong.load(Relaxed); - - loop { - // If the strong count is zero, the allocation has been dropped. - if strong == 0 { + #[inline] + fn checked_increment(n: usize) -> Option { + // Any write of 0 we can observe leaves the field in permanently zero state. + if n == 0 { return None; } + // See comments in `Arc::clone` for why we do this (for `mem::forget`). + assert!(n <= MAX_REFCOUNT, "{}", INTERNAL_OVERFLOW_ERROR); + Some(n + 1) + } - // If the strong count is greater than the maximum, panic. - // Panic instead of abort is okay because we didn't increment the strong counter yet. - assert!(strong <= MAX_REFCOUNT, "Arc counter overflow"); - - // Try to increment the strong count. - match header.strong.compare_exchange_weak(strong, strong + 1, Acquire, Relaxed) { - Ok(_) => { - // SAFETY: Now that the strong counter is incremented, the allocation for an Arc is valid. - return unsafe { Some(Arc::from_inner(self.shared)) }; - } - Err(x) => strong = x, - } + // We use a CAS loop to increment the strong count instead of a + // fetch_add as this function should never take the reference count + // from zero to one. + // + // Relaxed is fine for the failure case because we don't have any expectations about the new state. + // Acquire is necessary for the success case to synchronize with `Arc::new_cyclic`, when the inner + // value can be initialized after `Weak` references have already been created. In that case, we + // expect to observe the fully initialized value. + if self.inner()?.strong.fetch_update(Acquire, Relaxed, checked_increment).is_ok() { + // SAFETY: pointer is not null, verified in checked_increment + unsafe { Some(Arc::from_inner(self.ptr)) } + } else { + None } } - /// Get the number of strong pointers to this allocation. + /// Gets the number of strong (`Arc`) pointers pointing to this allocation. + /// + /// If `self` was created using [`Weak::new`], this will return 0. #[must_use] pub fn strong_count(&self) -> usize { - self.inner().map_or(0, |header| header.strong.load(Acquire)) + if let Some(inner) = self.inner() { + inner.strong.load(Relaxed) + } else { + 0 + } } - /// Get the number of weak pointers to this allocation. + /// Gets an approximation of the number of `Weak` pointers pointing to this + /// allocation. + /// + /// If `self` was created using [`Weak::new`], or if there are no remaining + /// strong pointers, this will return 0. + /// + /// # Accuracy + /// + /// Due to implementation details, the returned value can be off by 1 in + /// either direction when other threads are manipulating any `Arc`s or + /// `Weak`s pointing to the same allocation. #[must_use] pub fn weak_count(&self) -> usize { - self.inner().map_or(0, |header| { - let weak = header.weak.load(Acquire); - let strong = header.strong.load(Acquire); - + if let Some(inner) = self.inner() { + let weak = inner.weak.load(Acquire); + let strong = inner.strong.load(Relaxed); if strong == 0 { 0 } else { + // Since we observed that there was at least one strong pointer + // after reading the weak count, we know that the implicit weak + // reference (present whenever any strong references are alive) + // was still around when we observed the weak count, and can + // therefore safely subtract it. weak - 1 } - }) + } else { + 0 + } } -} -impl Weak { - /// Create a new dangling `Weak` pointer. + /// Returns `None` when the pointer is dangling and there is no allocated `ArcInner`, + /// (i.e., when this `Weak` was created by `Weak::new`). + #[inline] + fn inner(&self) -> Option> { + let ptr = self.ptr.as_ptr(); + if is_dangling(ptr) { + None + } else { + // We are careful to *not* create a reference covering the "data" field, as + // the field may be mutated concurrently (for example, if the last `Arc` + // is dropped, the data field will be dropped in-place). + Some(unsafe { WeakInner { strong: &(*ptr).strong, weak: &(*ptr).weak } }) + } + } + + /// Returns `true` if the two `Weak`s point to the same allocation similar to [`ptr::eq`], or if + /// both don't point to any allocation (because they were created with `Weak::new()`). However, + /// this function ignores the metadata of `dyn Trait` pointers. /// - /// # Example + /// # Notes + /// + /// Since this compares pointers it means that `Weak::new()` will equal each + /// other, even though they don't point to any allocation. + /// + /// # Examples /// /// ``` - /// use portable_atomic_util::Weak; + /// use portable_atomic_util::Arc; + /// + /// let first_rc = Arc::new(5); + /// let first = Arc::downgrade(&first_rc); + /// let second = Arc::downgrade(&first_rc); + /// + /// assert!(first.ptr_eq(&second)); + /// + /// let third_rc = Arc::new(5); + /// let third = Arc::downgrade(&third_rc); + /// + /// assert!(!first.ptr_eq(&third)); + /// ``` + /// + /// Comparing `Weak::new`. /// - /// let weak = Weak::::new(); /// ``` + /// use portable_atomic_util::{Arc, Weak}; + /// + /// let first = Weak::new(); + /// let second = Weak::new(); + /// assert!(first.ptr_eq(&second)); + /// + /// let third_rc = Arc::new(()); + /// let third = Arc::downgrade(&third_rc); + /// assert!(!first.ptr_eq(&third)); + /// ``` + /// + /// [`ptr::eq`]: core::ptr::eq "ptr::eq" + #[inline] #[must_use] - pub fn new() -> Self { - Self { - // SAFETY: usize::MAX != 0, so this is always valid - shared: unsafe { NonNull::new_unchecked(strict::invalid(usize::MAX)) }, + pub fn ptr_eq(&self, other: &Self) -> bool { + ptr::eq(self.ptr.as_ptr() as *const (), other.ptr.as_ptr() as *const ()) + } +} + +impl Clone for Weak { + /// Makes a clone of the `Weak` pointer that points to the same allocation. + /// + /// # Examples + /// + /// ``` + /// use portable_atomic_util::{Arc, Weak}; + /// + /// let weak_five = Arc::downgrade(&Arc::new(5)); + /// + /// let _ = Weak::clone(&weak_five); + /// ``` + #[inline] + fn clone(&self) -> Weak { + let inner = if let Some(inner) = self.inner() { + inner + } else { + return Weak { ptr: self.ptr }; + }; + // See comments in Arc::clone() for why this is relaxed. This can use a + // fetch_add (ignoring the lock) because the weak count is only locked + // where are *no other* weak pointers in existence. (So we can't be + // running this code in that case). + let old_size = inner.weak.fetch_add(1, Relaxed); + + // See comments in Arc::clone() for why we do this (for mem::forget). + if old_size > MAX_REFCOUNT { + abort(); } + + Weak { ptr: self.ptr } } } impl Default for Weak { - fn default() -> Self { - Self::new() + /// Constructs a new `Weak`, without allocating memory. + /// Calling [`upgrade`] on the return value always + /// gives [`None`]. + /// + /// [`upgrade`]: Weak::upgrade + /// + /// # Examples + /// + /// ``` + /// use portable_atomic_util::Weak; + /// + /// let empty: Weak = Default::default(); + /// assert!(empty.upgrade().is_none()); + /// ``` + fn default() -> Weak { + Weak::new() } } impl Drop for Weak { + /// Drops the `Weak` pointer. + /// + /// # Examples + /// + /// ``` + /// use portable_atomic_util::{Arc, Weak}; + /// + /// struct Foo; + /// + /// impl Drop for Foo { + /// fn drop(&mut self) { + /// println!("dropped!"); + /// } + /// } + /// + /// let foo = Arc::new(Foo); + /// let weak_foo = Arc::downgrade(&foo); + /// let other_weak_foo = Weak::clone(&weak_foo); + /// + /// drop(weak_foo); // Doesn't print anything + /// drop(foo); // Prints "dropped!" + /// + /// assert!(other_weak_foo.upgrade().is_none()); + /// ``` fn drop(&mut self) { - // Try to get access to the inner shared pointer. - if let Some(inner) = self.inner() { - if inner.weak.fetch_sub(1, Release) == 1 { - acquire!(inner.weak); - - // Deallocate the memory. - let ptr = self.shared.as_ptr() as *mut Shared>; - // SAFETY: We know that the weak count is 0, so we can deallocate. - // Using ManuallyDrop here prevents T from being dropped twice. - mem::drop(unsafe { Box::from_raw(ptr) }); + // If we find out that we were the last weak pointer, then its time to + // deallocate the data entirely. See the discussion in Arc::drop() about + // the memory orderings + // + // It's not necessary to check for the locked state here, because the + // weak count can only be locked if there was precisely one weak ref, + // meaning that drop could only subsequently run ON that remaining weak + // ref, which can only happen after the lock is released. + let inner = if let Some(inner) = self.inner() { inner } else { return }; + + if inner.weak.fetch_sub(1, Release) == 1 { + acquire!(inner.weak); + unsafe { + let layout = Layout::for_value(&*self.ptr.as_ptr()); + Global.deallocate(self.ptr.cast::(), layout); } } } } +impl PartialEq for Arc { + /// Equality for two `Arc`s. + /// + /// Two `Arc`s are equal if their inner values are equal, even if they are + /// stored in different allocation. + /// + /// If `T` also implements `Eq` (implying reflexivity of equality), + /// two `Arc`s that point to the same allocation are always equal. + /// + /// # Examples + /// + /// ``` + /// use portable_atomic_util::Arc; + /// + /// let five = Arc::new(5); + /// + /// assert!(five == Arc::new(5)); + /// ``` + #[inline] + fn eq(&self, other: &Self) -> bool { + **self == **other + } +} + +impl PartialOrd for Arc { + /// Partial comparison for two `Arc`s. + /// + /// The two are compared by calling `partial_cmp()` on their inner values. + /// + /// # Examples + /// + /// ``` + /// use portable_atomic_util::Arc; + /// use std::cmp::Ordering; + /// + /// let five = Arc::new(5); + /// + /// assert_eq!(Some(Ordering::Less), five.partial_cmp(&Arc::new(6))); + /// ``` + fn partial_cmp(&self, other: &Self) -> Option { + (**self).partial_cmp(&**other) + } + + /// Less-than comparison for two `Arc`s. + /// + /// The two are compared by calling `<` on their inner values. + /// + /// # Examples + /// + /// ``` + /// use portable_atomic_util::Arc; + /// + /// let five = Arc::new(5); + /// + /// assert!(five < Arc::new(6)); + /// ``` + fn lt(&self, other: &Self) -> bool { + *(*self) < *(*other) + } + + /// 'Less than or equal to' comparison for two `Arc`s. + /// + /// The two are compared by calling `<=` on their inner values. + /// + /// # Examples + /// + /// ``` + /// use portable_atomic_util::Arc; + /// + /// let five = Arc::new(5); + /// + /// assert!(five <= Arc::new(5)); + /// ``` + fn le(&self, other: &Self) -> bool { + *(*self) <= *(*other) + } + + /// Greater-than comparison for two `Arc`s. + /// + /// The two are compared by calling `>` on their inner values. + /// + /// # Examples + /// + /// ``` + /// use portable_atomic_util::Arc; + /// + /// let five = Arc::new(5); + /// + /// assert!(five > Arc::new(4)); + /// ``` + fn gt(&self, other: &Self) -> bool { + *(*self) > *(*other) + } + + /// 'Greater than or equal to' comparison for two `Arc`s. + /// + /// The two are compared by calling `>=` on their inner values. + /// + /// # Examples + /// + /// ``` + /// use portable_atomic_util::Arc; + /// + /// let five = Arc::new(5); + /// + /// assert!(five >= Arc::new(5)); + /// ``` + fn ge(&self, other: &Self) -> bool { + *(*self) >= *(*other) + } +} +impl Ord for Arc { + /// Comparison for two `Arc`s. + /// + /// The two are compared by calling `cmp()` on their inner values. + /// + /// # Examples + /// + /// ``` + /// use portable_atomic_util::Arc; + /// use std::cmp::Ordering; + /// + /// let five = Arc::new(5); + /// + /// assert_eq!(Ordering::Less, five.cmp(&Arc::new(6))); + /// ``` + fn cmp(&self, other: &Self) -> cmp::Ordering { + (**self).cmp(&**other) + } +} +impl Eq for Arc {} + +impl fmt::Display for Arc { + fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { + fmt::Display::fmt(&**self, f) + } +} + +impl fmt::Debug for Arc { + fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { + fmt::Debug::fmt(&**self, f) + } +} + +impl fmt::Pointer for Arc { + fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { + fmt::Pointer::fmt(&(&**self as *const T), f) + } +} + +impl Default for Arc { + /// Creates a new `Arc`, with the `Default` value for `T`. + /// + /// # Examples + /// + /// ``` + /// use portable_atomic_util::Arc; + /// + /// let x: Arc = Default::default(); + /// assert_eq!(*x, 0); + /// ``` + fn default() -> Arc { + Arc::new(Default::default()) + } +} + +impl Hash for Arc { + fn hash(&self, state: &mut H) { + (**self).hash(state); + } +} + +impl From for Arc { + /// Converts a `T` into an `Arc` + /// + /// The conversion moves the value into a + /// newly allocated `Arc`. It is equivalent to + /// calling `Arc::new(t)`. + /// + /// # Example + /// + /// ``` + /// use portable_atomic_util::Arc; + /// let x = 5; + /// let arc = Arc::new(5); + /// + /// assert_eq!(Arc::from(x), arc); + /// ``` + fn from(t: T) -> Self { + Arc::new(t) + } +} + +// TODO: CoerceUnsized is needed to cast Arc<[T; N]> -> Arc<[T]> +// impl From<[T; N]> for Arc<[T]> { +// /// Converts a [`[T; N]`](prim@array) into an `Arc<[T]>`. +// /// +// /// The conversion moves the array into a newly allocated `Arc`. +// /// +// /// # Example +// /// +// /// ``` +// /// use portable_atomic_util::Arc; +// /// let original: [i32; 3] = [1, 2, 3]; +// /// let shared: Arc<[i32]> = Arc::from(original); +// /// assert_eq!(&[1, 2, 3], &shared[..]); +// /// ``` +// #[inline] +// fn from(v: [T; N]) -> Arc<[T]> { +// Arc::<[T; N]>::from(v) +// } +// } + +// Layout::array requires Rust 1.44 +// impl From<&[T]> for Arc<[T]> { +// /// Allocate a reference-counted slice and fill it by cloning `v`'s items. +// /// +// /// # Example +// /// +// /// ``` +// /// use portable_atomic_util::Arc; +// /// let original: &[i32] = &[1, 2, 3]; +// /// let shared: Arc<[i32]> = Arc::from(original); +// /// assert_eq!(&[1, 2, 3], &shared[..]); +// /// ``` +// #[inline] +// fn from(v: &[T]) -> Arc<[T]> { +// unsafe { Self::from_iter_exact(v.iter().cloned(), v.len()) } +// } +// } + +// Layout::array requires Rust 1.44 +// impl From<&str> for Arc { +// /// Allocate a reference-counted `str` and copy `v` into it. +// /// +// /// # Example +// /// +// /// ``` +// /// use portable_atomic_util::Arc; +// /// let shared: Arc = Arc::from("eggplant"); +// /// assert_eq!("eggplant", &shared[..]); +// /// ``` +// #[inline] +// fn from(v: &str) -> Arc { +// let arc = Arc::<[u8]>::from(v.as_bytes()); +// unsafe { Arc::from_raw(Arc::into_raw(arc) as *const str) } +// } +// } + +// Layout::array requires Rust 1.44 +// impl From for Arc { +// /// Allocate a reference-counted `str` and copy `v` into it. +// /// +// /// # Example +// /// +// /// ``` +// /// use portable_atomic_util::Arc; +// /// let unique: String = "eggplant".to_owned(); +// /// let shared: Arc = Arc::from(unique); +// /// assert_eq!("eggplant", &shared[..]); +// /// ``` +// #[inline] +// fn from(v: String) -> Self { +// Arc::from(&v[..]) +// } +// } + +impl From> for Arc { + /// Move a boxed object to a new, reference-counted allocation. + /// + /// # Example + /// + /// ``` + /// use portable_atomic_util::Arc; + /// let unique: Box = Box::from("eggplant"); + /// let shared: Arc = Arc::from(unique); + /// assert_eq!("eggplant", &shared[..]); + /// ``` + #[inline] + fn from(v: Box) -> Self { + Arc::from_box(v) + } +} + +// Layout::array requires Rust 1.44 +// impl From> for Arc<[T]> { +// /// Allocate a reference-counted slice and move `v`'s items into it. +// /// +// /// # Example +// /// +// /// ``` +// /// use portable_atomic_util::Arc; +// /// let unique: Vec = vec![1, 2, 3]; +// /// let shared: Arc<[i32]> = Arc::from(unique); +// /// assert_eq!(&[1, 2, 3], &shared[..]); +// /// ``` +// #[inline] +// fn from(v: Vec) -> Self { +// unsafe { +// let len = v.len(); +// let cap = v.capacity(); +// let vec_ptr = mem::ManuallyDrop::new(v).as_mut_ptr(); + +// let rc_ptr = Self::allocate_for_slice(len); +// ptr::copy_nonoverlapping(vec_ptr, &mut (*rc_ptr).data as *mut [T] as *mut T, len); + +// // Create a `Vec` with length 0, to deallocate the buffer +// // without dropping its contents or the allocator +// let _ = Vec::from_raw_parts(vec_ptr, 0, cap); + +// Self::from_ptr(rc_ptr) +// } +// } +// } + +// TODO: CoerceUnsized is needed to cast Arc> -> Arc +// impl<'a, B> From> for Arc +// where +// B: ToOwned + ?Sized, +// Arc: From<&'a B> + From, +// { +// /// Create an atomically reference-counted pointer from +// /// a clone-on-write pointer by copying its content. +// /// +// /// # Example +// /// +// /// ``` +// /// use portable_atomic_util::Arc; +// /// use std::borrow::Cow; +// /// let cow: Cow<'_, str> = Cow::Borrowed("eggplant"); +// /// let shared: Arc = Arc::from(cow); +// /// assert_eq!("eggplant", &shared[..]); +// /// ``` +// #[inline] +// fn from(cow: Cow<'a, B>) -> Self { +// match cow { +// Cow::Borrowed(s) => Arc::from(s), +// Cow::Owned(s) => Arc::from(s), +// } +// } +// } + +// TODO: CoerceUnsized is needed to cast Arc::from("..") -> Arc +// impl From> for Arc<[u8]> { +// /// Converts an atomically reference-counted string slice into a byte slice. +// /// +// /// # Example +// /// +// /// ``` +// /// use portable_atomic_util::Arc; +// /// let string: Arc = Arc::from("eggplant"); +// /// let bytes: Arc<[u8]> = Arc::from(string); +// /// assert_eq!("eggplant".as_bytes(), bytes.as_ref()); +// /// ``` +// #[inline] +// fn from(rc: Arc) -> Self { +// // SAFETY: `str` has the same layout as `[u8]`. +// unsafe { Arc::from_raw(Arc::into_raw(rc) as *const [u8]) } +// } +// } + +// TODO: const generics requires 1.51 +// impl TryFrom> for Arc<[T; N]> { +// type Error = Arc<[T]>; +// +// fn try_from(boxed_slice: Arc<[T]>) -> Result { +// if boxed_slice.len() == N { +// Ok(unsafe { Arc::from_raw(Arc::into_raw(boxed_slice) as *mut [T; N]) }) +// } else { +// Err(boxed_slice) +// } +// } +// } + +// Layout::array requires Rust 1.44 +// impl FromIterator for Arc<[T]> { +// /// Takes each element in the `Iterator` and collects it into an `Arc<[T]>`. +// /// +// /// # Performance characteristics +// /// +// /// ## The general case +// /// +// /// In the general case, collecting into `Arc<[T]>` is done by first +// /// collecting into a `Vec`. That is, when writing the following: +// /// +// /// ``` +// /// use portable_atomic_util::Arc; +// /// let evens: Arc<[u8]> = (0..10).filter(|&x| x % 2 == 0).collect(); +// /// # assert_eq!(&*evens, &[0, 2, 4, 6, 8]); +// /// ``` +// /// +// /// this behaves as if we wrote: +// /// +// /// ``` +// /// use portable_atomic_util::Arc; +// /// let evens: Arc<[u8]> = (0..10).filter(|&x| x % 2 == 0) +// /// .collect::>() // The first set of allocations happens here. +// /// .into(); // A second allocation for `Arc<[T]>` happens here. +// /// # assert_eq!(&*evens, &[0, 2, 4, 6, 8]); +// /// ``` +// /// +// /// This will allocate as many times as needed for constructing the `Vec` +// /// and then it will allocate once for turning the `Vec` into the `Arc<[T]>`. +// /// +// /// ## Iterators of known length +// /// +// /// When your `Iterator` implements `TrustedLen` and is of an exact size, +// /// a single allocation will be made for the `Arc<[T]>`. For example: +// /// +// /// ``` +// /// use portable_atomic_util::Arc; +// /// let evens: Arc<[u8]> = (0..10).collect(); // Just a single allocation happens here. +// /// # assert_eq!(&*evens, &*(0..10).collect::>()); +// /// ``` +// fn from_iter>(iter: I) -> Self { +// iter.into_iter().collect::>().into() +// } +// } + +impl borrow::Borrow for Arc { + fn borrow(&self) -> &T { + self + } +} + +impl AsRef for Arc { + fn as_ref(&self) -> &T { + self + } +} + +impl Unpin for Arc {} + +/// Get the offset within an `ArcInner` for the payload behind a pointer. +/// +/// # Safety +/// +/// The pointer must point to (and have valid metadata for) a previously +/// valid instance of T, but the T is allowed to be dropped. +unsafe fn data_offset(ptr: *const T) -> usize { + // Align the unsized value to the end of the ArcInner. + // Because RcBox is repr(C), it will always be the last field in memory. + // SAFETY: the caller ensure ptr is valid. Call align_of_val for + // ManuallyDrop to avoid creating &T if it was dropped. + unsafe { + data_offset_align(align_of_val::>( + &*(ptr as *const mem::ManuallyDrop), + )) + } +} + +#[inline] +fn data_offset_align(align: usize) -> usize { + let layout = Layout::new::>(); + layout.size() + padding_needed_for(layout, align) +} + +#[cfg(feature = "std")] +impl std::error::Error for Arc { + fn source(&self) -> Option<&(dyn std::error::Error + 'static)> { + std::error::Error::source(&**self) + } +} + +// Based on unstable Layout::padding_needed_for. +#[must_use] +#[inline] +fn padding_needed_for(layout: Layout, align: usize) -> usize { + let len = layout.size(); + + // Rounded up value is: + // len_rounded_up = (len + align - 1) & !(align - 1); + // and then we return the padding difference: `len_rounded_up - len`. + // + // We use modular arithmetic throughout: + // + // 1. align is guaranteed to be > 0, so align - 1 is always + // valid. + // + // 2. `len + align - 1` can overflow by at most `align - 1`, + // so the &-mask with `!(align - 1)` will ensure that in the + // case of overflow, `len_rounded_up` will itself be 0. + // Thus the returned padding, when added to `len`, yields 0, + // which trivially satisfies the alignment `align`. + // + // (Of course, attempts to allocate blocks of memory whose + // size and padding overflow in the above manner should cause + // the allocator to yield an error anyway.) + + let len_rounded_up = len.wrapping_add(align).wrapping_sub(1) & !align.wrapping_sub(1); + len_rounded_up.wrapping_sub(len) +} + +// Based on Layout::pad_to_align stabilized in Rust 1.44. +#[must_use] +#[inline] +fn pad_to_align(layout: Layout) -> Layout { + let pad = padding_needed_for(layout, layout.align()); + // This cannot overflow. Quoting from the invariant of Layout: + // > `size`, when rounded up to the nearest multiple of `align`, + // > must not overflow isize (i.e., the rounded value must be + // > less than or equal to `isize::MAX`) + let new_size = layout.size() + pad; + + // SAFETY: padded size is guaranteed to not exceed `isize::MAX`. + unsafe { Layout::from_size_align_unchecked(new_size, layout.align()) } +} + +// Based on Layout::extend stabilized in Rust 1.44. +#[inline] +fn extend_layout(layout: Layout, next: Layout) -> Option<(Layout, usize)> { + let new_align = cmp::max(layout.align(), next.align()); + let pad = padding_needed_for(layout, next.align()); + + let offset = layout.size().checked_add(pad)?; + let new_size = offset.checked_add(next.size())?; + + // The safe constructor is called here to enforce the isize size limit. + let layout = Layout::from_size_align(new_size, new_align).ok()?; + Some((layout, offset)) +} + +#[cfg(feature = "std")] +use std::process::abort; +#[cfg(not(feature = "std"))] +#[cold] fn abort() -> ! { struct Abort; - impl Drop for Abort { fn drop(&mut self) { panic!(); @@ -836,7 +2334,6 @@ fn abort() -> ! { } let _abort = Abort; - panic!("abort") } @@ -844,6 +2341,43 @@ fn is_dangling(ptr: *mut T) -> bool { ptr as *mut () as usize == usize::MAX } +// Based on unstable alloc::alloc::Global. +// +// Note: unlike alloc::alloc::Global that returns NonNull<[u8]>, +// this returns NonNull. +#[derive(Copy, Clone, Default, Debug)] +struct Global; +#[allow(clippy::trivially_copy_pass_by_ref, clippy::unused_self)] +impl Global { + #[inline] + fn allocate(&self, layout: Layout) -> Option> { + // Layout::dangling is unstable + #[must_use] + #[inline] + fn dangling(layout: Layout) -> NonNull { + // SAFETY: align is guaranteed to be non-zero + unsafe { NonNull::new_unchecked(strict::invalid::(layout.align())) } + } + + match layout.size() { + 0 => Some(dangling(layout)), + // SAFETY: `layout` is non-zero in size, + _size => unsafe { + let raw_ptr = alloc::alloc::alloc(layout); + NonNull::new(raw_ptr) + }, + } + } + #[inline] + unsafe fn deallocate(&self, ptr: NonNull, layout: Layout) { + if layout.size() != 0 { + // SAFETY: `layout` is non-zero in size, + // other conditions must be upheld by the caller + unsafe { alloc::alloc::dealloc(ptr.as_ptr(), layout) } + } + } +} + /// Emulate strict provenance. /// /// Once strict_provenance is stable, migrate to the standard library's APIs. @@ -878,25 +2412,19 @@ mod strict { other } - /// Replace the address portion of this pointer with a new address. + // Based on ::byte_add stabilized in Rust 1.75. #[inline] #[must_use] - pub(super) fn with_addr(ptr: *mut T, addr: usize) -> *mut T { - // FIXME(strict_provenance_magic): I am magic and should be a compiler intrinsic. - // - // In the mean-time, this operation is defined to be "as if" it was - // a wrapping_add, so we can emulate it as such. This should properly - // restore pointer provenance even under today's compiler. - let offset = addr.wrapping_sub(ptr as usize); - - // This is the canonical desugaring of this operation. - (ptr as *mut u8).wrapping_add(offset) as *mut T + pub(super) unsafe fn byte_add(ptr: *mut T, count: usize) -> *mut T { + // SAFETY: the caller must uphold the safety contract for `add`. + unsafe { with_metadata_of((ptr as *mut u8).add(count), ptr) } } - /// Run an operation of some kind on a pointer. + // Based on ::byte_sub stabilized in Rust 1.75. #[inline] #[must_use] - pub(super) fn map_addr(ptr: *mut T, f: impl FnOnce(usize) -> usize) -> *mut T { - with_addr(ptr, f(ptr as usize)) + pub(super) unsafe fn byte_sub(ptr: *mut T, count: usize) -> *mut T { + // SAFETY: the caller must uphold the safety contract for `sub`. + unsafe { with_metadata_of((ptr as *mut u8).sub(count), ptr) } } } diff --git a/portable-atomic-util/src/lib.rs b/portable-atomic-util/src/lib.rs index d9044ec80..f9f3ca75d 100644 --- a/portable-atomic-util/src/lib.rs +++ b/portable-atomic-util/src/lib.rs @@ -63,6 +63,8 @@ See [#1] for other primitives being considered for addition to this crate. #[cfg(all(feature = "alloc", not(portable_atomic_no_alloc)))] extern crate alloc; +#[cfg(feature = "std")] +extern crate std; #[cfg(all(feature = "std", portable_atomic_no_alloc))] extern crate std as alloc; diff --git a/portable-atomic-util/tests/arc.rs b/portable-atomic-util/tests/arc.rs index d822399de..3d23abfd2 100644 --- a/portable-atomic-util/tests/arc.rs +++ b/portable-atomic-util/tests/arc.rs @@ -4,6 +4,7 @@ use portable_atomic_util::Arc; +// https://github.com/taiki-e/portable-atomic/issues/37 #[test] fn over_aligned() { #[repr(align(128))] @@ -12,8 +13,689 @@ fn over_aligned() { let value = Arc::new(Aligned(128)); let ptr = Arc::into_raw(value); // SAFETY: `ptr` should always be a valid `Aligned`. - assert_eq!(unsafe { (&*ptr).0 }, 128); + assert_eq!(unsafe { (*ptr).0 }, 128); // SAFETY: `ptr` is a valid reference to an `Arc`. let value = unsafe { Arc::from_raw(ptr) }; assert_eq!(value.0, 128); } + +// https://github.com/rust-lang/rust/blob/5151b8c42712c473e7da56e213926b929d0212ef/library/alloc/src/sync/tests.rs +#[allow(clippy::many_single_char_names, clippy::undocumented_unsafe_blocks)] +pub mod alloc_tests { + use std::{ + sync::{ + atomic::Ordering::{Acquire, SeqCst}, + mpsc::channel, + Mutex, + }, + thread, + }; + + use portable_atomic as atomic; + use portable_atomic_util::{Arc, Weak}; + + struct Canary(*mut atomic::AtomicUsize); + + impl Drop for Canary { + fn drop(&mut self) { + unsafe { + match *self { + Canary(c) => { + (*c).fetch_add(1, SeqCst); + } + } + } + } + } + + #[test] + #[cfg_attr(target_os = "emscripten", ignore)] + fn manually_share_arc() { + let v = vec![1, 2, 3, 4, 5, 6, 7, 8, 9, 10]; + let arc_v = Arc::new(v); + + let (tx, rx) = channel(); + + let _t = thread::spawn(move || { + let arc_v: Arc> = rx.recv().unwrap(); + assert_eq!((*arc_v)[3], 4); + }); + + tx.send(arc_v.clone()).unwrap(); + + assert_eq!((*arc_v)[2], 3); + assert_eq!((*arc_v)[4], 5); + } + + #[test] + fn test_arc_get_mut() { + let mut x = Arc::new(3); + *Arc::get_mut(&mut x).unwrap() = 4; + assert_eq!(*x, 4); + let y = x.clone(); + assert!(Arc::get_mut(&mut x).is_none()); + drop(y); + assert!(Arc::get_mut(&mut x).is_some()); + let _w = Arc::downgrade(&x); + assert!(Arc::get_mut(&mut x).is_none()); + } + + #[test] + fn weak_counts() { + assert_eq!(Weak::weak_count(&Weak::::new()), 0); + assert_eq!(Weak::strong_count(&Weak::::new()), 0); + + let a = Arc::new(0); + let w = Arc::downgrade(&a); + assert_eq!(Weak::strong_count(&w), 1); + assert_eq!(Weak::weak_count(&w), 1); + let w2 = w.clone(); + assert_eq!(Weak::strong_count(&w), 1); + assert_eq!(Weak::weak_count(&w), 2); + assert_eq!(Weak::strong_count(&w2), 1); + assert_eq!(Weak::weak_count(&w2), 2); + drop(w); + assert_eq!(Weak::strong_count(&w2), 1); + assert_eq!(Weak::weak_count(&w2), 1); + let a2 = a.clone(); + assert_eq!(Weak::strong_count(&w2), 2); + assert_eq!(Weak::weak_count(&w2), 1); + drop(a2); + drop(a); + assert_eq!(Weak::strong_count(&w2), 0); + assert_eq!(Weak::weak_count(&w2), 0); + drop(w2); + } + + #[test] + fn try_unwrap() { + let x = Arc::new(3); + assert_eq!(Arc::try_unwrap(x), Ok(3)); + let x = Arc::new(4); + let _y = x.clone(); + assert_eq!(Arc::try_unwrap(x), Err(Arc::new(4))); + let x = Arc::new(5); + let _w = Arc::downgrade(&x); + assert_eq!(Arc::try_unwrap(x), Ok(5)); + } + + #[test] + fn into_inner() { + for _ in 0..100 + // ^ Increase chances of hitting potential race conditions + { + let x = Arc::new(3); + let y = Arc::clone(&x); + let r_thread = std::thread::spawn(|| Arc::into_inner(x)); + let s_thread = std::thread::spawn(|| Arc::into_inner(y)); + let r = r_thread.join().expect("r_thread panicked"); + let s = s_thread.join().expect("s_thread panicked"); + assert!( + matches!((r, s), (None, Some(3)) | (Some(3), None)), + "assertion failed: unexpected result `{:?}`\ + \n expected `(None, Some(3))` or `(Some(3), None)`", + (r, s), + ); + } + + let x = Arc::new(3); + assert_eq!(Arc::into_inner(x), Some(3)); + + let x = Arc::new(4); + let y = Arc::clone(&x); + assert_eq!(Arc::into_inner(x), None); + assert_eq!(Arc::into_inner(y), Some(4)); + + let x = Arc::new(5); + let _w = Arc::downgrade(&x); + assert_eq!(Arc::into_inner(x), Some(5)); + } + + #[test] + fn into_from_raw() { + let x = Arc::new(Box::new("hello")); + let y = x.clone(); + + let x_ptr = Arc::into_raw(x); + drop(y); + unsafe { + assert_eq!(**x_ptr, "hello"); + + let x = Arc::from_raw(x_ptr); + assert_eq!(**x, "hello"); + + assert_eq!(Arc::try_unwrap(x).map(|x| *x), Ok("hello")); + } + } + + // TODO: CoerceUnsized is needed to cast to Arc + // #[test] + // fn test_into_from_raw_unsized() { + // use std::{fmt::Display, string::ToString}; + + // let arc: Arc = Arc::from("foo"); + + // let ptr = Arc::into_raw(arc.clone()); + // let arc2 = unsafe { Arc::from_raw(ptr) }; + + // assert_eq!(unsafe { &*ptr }, "foo"); + // assert_eq!(arc, arc2); + + // let arc: Arc = Arc::new(123); + + // let ptr = Arc::into_raw(arc.clone()); + // let arc2 = unsafe { Arc::from_raw(ptr) }; + + // assert_eq!(unsafe { &*ptr }.to_string(), "123"); + // assert_eq!(arc2.to_string(), "123"); + // } + + #[test] + fn into_from_weak_raw() { + let x = Arc::new(Box::new("hello")); + let y = Arc::downgrade(&x); + + let y_ptr = Weak::into_raw(y); + unsafe { + assert_eq!(**y_ptr, "hello"); + + let y = Weak::from_raw(y_ptr); + let y_up = Weak::upgrade(&y).unwrap(); + assert_eq!(**y_up, "hello"); + drop(y_up); + + assert_eq!(Arc::try_unwrap(x).map(|x| *x), Ok("hello")); + } + } + + // TODO: CoerceUnsized is needed to cast to Arc + // #[test] + // fn test_into_from_weak_raw_unsized() { + // use std::{fmt::Display, string::ToString}; + + // let arc: Arc = Arc::from("foo"); + // let weak: Weak = Arc::downgrade(&arc); + + // let ptr = Weak::into_raw(weak.clone()); + // let weak2 = unsafe { Weak::from_raw(ptr) }; + + // assert_eq!(unsafe { &*ptr }, "foo"); + // assert!(weak.ptr_eq(&weak2)); + + // let arc: Arc = Arc::new(123); + // let weak: Weak = Arc::downgrade(&arc); + + // let ptr = Weak::into_raw(weak.clone()); + // let weak2 = unsafe { Weak::from_raw(ptr) }; + + // assert_eq!(unsafe { &*ptr }.to_string(), "123"); + // assert!(weak.ptr_eq(&weak2)); + // } + + #[cfg(not(portable_atomic_no_maybe_uninit))] + #[test] + fn test_cow_arc_clone_make_mut() { + let mut cow0 = Arc::new(75); + let mut cow1 = cow0.clone(); + let mut cow2 = cow1.clone(); + + assert!(75 == *Arc::make_mut(&mut cow0)); + assert!(75 == *Arc::make_mut(&mut cow1)); + assert!(75 == *Arc::make_mut(&mut cow2)); + + *Arc::make_mut(&mut cow0) += 1; + *Arc::make_mut(&mut cow1) += 2; + *Arc::make_mut(&mut cow2) += 3; + + assert!(76 == *cow0); + assert!(77 == *cow1); + assert!(78 == *cow2); + + // none should point to the same backing memory + assert!(*cow0 != *cow1); + assert!(*cow0 != *cow2); + assert!(*cow1 != *cow2); + } + + #[cfg(not(portable_atomic_no_maybe_uninit))] + #[test] + fn test_cow_arc_clone_unique2() { + let mut cow0 = Arc::new(75); + let cow1 = cow0.clone(); + let cow2 = cow1.clone(); + + assert!(75 == *cow0); + assert!(75 == *cow1); + assert!(75 == *cow2); + + *Arc::make_mut(&mut cow0) += 1; + assert!(76 == *cow0); + assert!(75 == *cow1); + assert!(75 == *cow2); + + // cow1 and cow2 should share the same contents + // cow0 should have a unique reference + assert!(*cow0 != *cow1); + assert!(*cow0 != *cow2); + assert!(*cow1 == *cow2); + } + + #[cfg(not(portable_atomic_no_maybe_uninit))] + #[test] + fn test_cow_arc_clone_weak() { + let mut cow0 = Arc::new(75); + let cow1_weak = Arc::downgrade(&cow0); + + assert!(75 == *cow0); + assert!(75 == *cow1_weak.upgrade().unwrap()); + + *Arc::make_mut(&mut cow0) += 1; + + assert!(76 == *cow0); + assert!(cow1_weak.upgrade().is_none()); + } + + #[test] + fn test_live() { + let x = Arc::new(5); + let y = Arc::downgrade(&x); + assert!(y.upgrade().is_some()); + } + + #[test] + fn test_dead() { + let x = Arc::new(5); + let y = Arc::downgrade(&x); + drop(x); + assert!(y.upgrade().is_none()); + } + + #[test] + fn weak_self_cyclic() { + struct Cycle { + x: Mutex>>, + } + + let a = Arc::new(Cycle { x: Mutex::new(None) }); + let b = Arc::downgrade(&a.clone()); + *a.x.lock().unwrap() = Some(b); + + // hopefully we don't double-free (or leak)... + } + + #[test] + fn drop_arc() { + let mut canary = atomic::AtomicUsize::new(0); + let x = Arc::new(Canary(&mut canary as *mut atomic::AtomicUsize)); + drop(x); + assert!(canary.load(Acquire) == 1); + } + + #[test] + fn drop_arc_weak() { + let mut canary = atomic::AtomicUsize::new(0); + let arc = Arc::new(Canary(&mut canary as *mut atomic::AtomicUsize)); + let arc_weak = Arc::downgrade(&arc); + assert!(canary.load(Acquire) == 0); + drop(arc); + assert!(canary.load(Acquire) == 1); + drop(arc_weak); + } + + #[test] + fn test_strong_count() { + let a = Arc::new(0); + assert!(Arc::strong_count(&a) == 1); + let w = Arc::downgrade(&a); + assert!(Arc::strong_count(&a) == 1); + let b = w.upgrade().expect(""); + assert!(Arc::strong_count(&b) == 2); + assert!(Arc::strong_count(&a) == 2); + drop(w); + drop(a); + assert!(Arc::strong_count(&b) == 1); + let c = b.clone(); + assert!(Arc::strong_count(&b) == 2); + assert!(Arc::strong_count(&c) == 2); + } + + #[test] + fn test_weak_count() { + let a = Arc::new(0); + assert!(Arc::strong_count(&a) == 1); + assert!(Arc::weak_count(&a) == 0); + let w = Arc::downgrade(&a); + assert!(Arc::strong_count(&a) == 1); + assert!(Arc::weak_count(&a) == 1); + let x = w.clone(); + assert!(Arc::weak_count(&a) == 2); + drop(w); + drop(x); + assert!(Arc::strong_count(&a) == 1); + assert!(Arc::weak_count(&a) == 0); + let c = a.clone(); + assert!(Arc::strong_count(&a) == 2); + assert!(Arc::weak_count(&a) == 0); + let d = Arc::downgrade(&c); + assert!(Arc::weak_count(&c) == 1); + assert!(Arc::strong_count(&c) == 2); + + drop(a); + drop(c); + drop(d); + } + + #[test] + fn show_arc() { + let a = Arc::new(5); + assert_eq!(format!("{:?}", a), "5"); + } + + // Make sure deriving works with Arc + #[derive(Eq, Ord, PartialEq, PartialOrd, Clone, Debug, Default)] + struct Foo { + inner: Arc, + } + + // TODO: CoerceUnsized is needed to cast to Arc<[..]> + // #[test] + // fn test_unsized() { + // let x: Arc<[i32]> = Arc::new([1, 2, 3]); + // assert_eq!(format!("{x:?}"), "[1, 2, 3]"); + // let y = Arc::downgrade(&x.clone()); + // drop(x); + // assert!(y.upgrade().is_none()); + // } + + #[test] + fn test_maybe_thin_unsized() { + // If/when custom thin DSTs exist, this test should be updated to use one + use std::ffi::{CStr, CString}; + + let x: Arc = Arc::from(CString::new("swordfish").unwrap().into_boxed_c_str()); + assert_eq!(format!("{:?}", x), "\"swordfish\""); + let y: Weak = Arc::downgrade(&x); + drop(x); + + // At this point, the weak points to a dropped DST + assert!(y.upgrade().is_none()); + // But we still need to be able to get the alloc layout to drop. + // CStr has no drop glue, but custom DSTs might, and need to work. + drop(y); + } + + #[test] + fn test_from_owned() { + let foo = 123; + let foo_arc = Arc::from(foo); + assert!(123 == *foo_arc); + } + + #[test] + fn test_new_weak() { + let foo: Weak = Weak::new(); + assert!(foo.upgrade().is_none()); + } + + #[test] + fn test_ptr_eq() { + let five = Arc::new(5); + let same_five = five.clone(); + let other_five = Arc::new(5); + + assert!(Arc::ptr_eq(&five, &same_five)); + assert!(!Arc::ptr_eq(&five, &other_five)); + } + + #[test] + #[cfg_attr(target_os = "emscripten", ignore)] + fn test_weak_count_locked() { + let mut a = Arc::new(atomic::AtomicBool::new(false)); + let a2 = a.clone(); + let t = thread::spawn(move || { + // Miri is too slow + let count = if cfg!(miri) { 1_000 } else { 1_000_000 }; + for _i in 0..count { + Arc::get_mut(&mut a); + } + a.store(true, SeqCst); + }); + + while !a2.load(SeqCst) { + let n = Arc::weak_count(&a2); + assert!(n < 2, "bad weak count: {}", n); + #[cfg(miri)] // Miri's scheduler does not guarantee liveness, and thus needs this hint. + std::hint::spin_loop(); + } + t.join().unwrap(); + } + + // TODO: CoerceUnsized is needed to cast to Arc + // #[test] + // fn test_from_str() { + // let r: Arc = Arc::from("foo"); + + // assert_eq!(&r[..], "foo"); + // } + + // TODO: CoerceUnsized is needed to cast to Arc<[..]> + // #[test] + // fn test_copy_from_slice() { + // let s: &[u32] = &[1, 2, 3]; + // let r: Arc<[u32]> = Arc::from(s); + + // assert_eq!(&r[..], [1, 2, 3]); + // } + + // TODO: CoerceUnsized is needed to cast to Arc<[..]> + // #[test] + // fn test_clone_from_slice() { + // #[derive(Clone, Debug, Eq, PartialEq)] + // struct X(u32); + + // let s: &[X] = &[X(1), X(2), X(3)]; + // let r: Arc<[X]> = Arc::from(s); + + // assert_eq!(&r[..], s); + // } + + // TODO: CoerceUnsized is needed to cast to Arc<[..]> + // #[test] + // #[should_panic] + // fn test_clone_from_slice_panic() { + // use std::string::{String, ToString}; + + // struct Fail(u32, String); + + // impl Clone for Fail { + // fn clone(&self) -> Fail { + // if self.0 == 2 { + // panic!(); + // } + // Fail(self.0, self.1.clone()) + // } + // } + + // let s: &[Fail] = + // &[Fail(0, "foo".to_string()), Fail(1, "bar".to_string()), Fail(2, "baz".to_string())]; + + // // Should panic, but not cause memory corruption + // let _r: Arc<[Fail]> = Arc::from(s); + // } + + #[test] + fn test_from_box() { + let b: Box = Box::new(123); + let r: Arc = Arc::from(b); + + assert_eq!(*r, 123); + } + + #[test] + fn test_from_box_str() { + let s = String::from("foo").into_boxed_str(); + let r: Arc = Arc::from(s); + + assert_eq!(&r[..], "foo"); + } + + #[test] + fn test_from_box_slice() { + let s = vec![1, 2, 3].into_boxed_slice(); + let r: Arc<[u32]> = Arc::from(s); + + assert_eq!(&r[..], [1, 2, 3]); + } + + #[test] + fn test_from_box_trait() { + use std::{fmt::Display, string::ToString}; + + let b: Box = Box::new(123); + let r: Arc = Arc::from(b); + + assert_eq!(r.to_string(), "123"); + } + + #[test] + fn test_from_box_trait_zero_sized() { + use std::fmt::Debug; + + let b: Box = Box::new(()); + let r: Arc = Arc::from(b); + + assert_eq!(format!("{:?}", r), "()"); + } + + // TODO: CoerceUnsized is needed to cast to Arc<[..]> + // #[test] + // fn test_from_vec() { + // let v = vec![1, 2, 3]; + // let r: Arc<[u32]> = Arc::from(v); + + // assert_eq!(&r[..], [1, 2, 3]); + // } + + // TODO: CoerceUnsized is needed to cast to Arc + // #[test] + // fn test_downcast() { + // use std::any::Any; + + // let r1: Arc = Arc::new(i32::MAX); + // let r2: Arc = Arc::new("abc"); + + // assert!(r1.clone().downcast::().is_err()); + + // let r1i32 = r1.downcast::(); + // assert!(r1i32.is_ok()); + // assert_eq!(r1i32.unwrap(), Arc::new(i32::MAX)); + + // assert!(r2.clone().downcast::().is_err()); + + // let r2str = r2.downcast::<&'static str>(); + // assert!(r2str.is_ok()); + // assert_eq!(r2str.unwrap(), Arc::new("abc")); + // } + + // TODO: CoerceUnsized is needed to cast to Arc<[..]> + // #[test] + // fn test_array_from_slice() { + // let v = vec![1, 2, 3]; + // let r: Arc<[u32]> = Arc::from(v); + + // let a: Result, _> = r.clone().try_into(); + // assert!(a.is_ok()); + + // let a: Result, _> = r.clone().try_into(); + // assert!(a.is_err()); + // } + + #[cfg(not(portable_atomic_no_maybe_uninit))] + #[test] + fn test_arc_cyclic_with_zero_refs() { + struct ZeroRefs { + inner: Weak, + } + let zero_refs = Arc::new_cyclic(|inner| { + assert_eq!(inner.strong_count(), 0); + assert!(inner.upgrade().is_none()); + ZeroRefs { inner: Weak::new() } + }); + + assert_eq!(Arc::strong_count(&zero_refs), 1); + assert_eq!(Arc::weak_count(&zero_refs), 0); + assert_eq!(zero_refs.inner.strong_count(), 0); + assert_eq!(zero_refs.inner.weak_count(), 0); + } + + #[cfg(not(portable_atomic_no_maybe_uninit))] + #[test] + fn test_arc_new_cyclic_one_ref() { + struct OneRef { + inner: Weak, + } + let one_ref = Arc::new_cyclic(|inner| { + assert_eq!(inner.strong_count(), 0); + assert!(inner.upgrade().is_none()); + OneRef { inner: inner.clone() } + }); + + assert_eq!(Arc::strong_count(&one_ref), 1); + assert_eq!(Arc::weak_count(&one_ref), 1); + + let one_ref2 = Weak::upgrade(&one_ref.inner).unwrap(); + assert!(Arc::ptr_eq(&one_ref, &one_ref2)); + + assert_eq!(Arc::strong_count(&one_ref), 2); + assert_eq!(Arc::weak_count(&one_ref), 1); + } + + #[cfg(not(portable_atomic_no_maybe_uninit))] + #[test] + fn test_arc_cyclic_two_refs() { + struct TwoRefs { + inner1: Weak, + inner2: Weak, + } + let two_refs = Arc::new_cyclic(|inner| { + assert_eq!(inner.strong_count(), 0); + assert!(inner.upgrade().is_none()); + + let inner1 = inner.clone(); + let inner2 = inner1.clone(); + + TwoRefs { inner1, inner2 } + }); + + assert_eq!(Arc::strong_count(&two_refs), 1); + assert_eq!(Arc::weak_count(&two_refs), 2); + + let two_refs1 = Weak::upgrade(&two_refs.inner1).unwrap(); + assert!(Arc::ptr_eq(&two_refs, &two_refs1)); + + let two_refs2 = Weak::upgrade(&two_refs.inner2).unwrap(); + assert!(Arc::ptr_eq(&two_refs, &two_refs2)); + + assert_eq!(Arc::strong_count(&two_refs), 3); + assert_eq!(Arc::weak_count(&two_refs), 2); + } + + /// Test for Arc::drop bug (https://github.com/rust-lang/rust/issues/55005) + #[test] + #[cfg(miri)] // relies on Stacked Borrows in Miri + fn arc_drop_dereferenceable_race() { + // The bug seems to take up to 700 iterations to reproduce with most seeds (tested 0-9). + for _ in 0..750 { + let arc_1 = Arc::new(()); + let arc_2 = arc_1.clone(); + let thread = thread::spawn(|| drop(arc_2)); + // Spin a bit; makes the race more likely to appear + let mut i = 0; + while i < 256 { + i += 1; + } + drop(arc_1); + thread.join().unwrap(); + } + } +} diff --git a/tools/build.sh b/tools/build.sh index a4c6e5fd6..826b09236 100755 --- a/tools/build.sh +++ b/tools/build.sh @@ -221,7 +221,7 @@ else if [[ -n "${TESTS:-}" ]]; then # TESTS=1 builds binaries, so cargo build requires toolchain and libraries. subcmd=check - base_args=("${subcmd}") + base_args=(hack "${subcmd}") else base_args=(hack "${subcmd}") fi @@ -247,7 +247,7 @@ if [[ "${rustc_version}" == *"nightly"* ]] || [[ "${rustc_version}" == *"dev"* ] rustup ${pre_args[@]+"${pre_args[@]}"} component add clippy &>/dev/null target_dir="${target_dir}/check-cfg" if [[ -n "${TESTS:-}" ]]; then - base_args=("${subcmd}" -Z check-cfg) + base_args=(hack "${subcmd}" -Z check-cfg) else base_args=(hack "${subcmd}" -Z check-cfg) fi @@ -383,6 +383,7 @@ build() { args+=( --tests --features "${test_features}" + --ignore-unknown-features --workspace --exclude bench --exclude portable-atomic-internal-codegen ) elif [[ -n "${TARGET_GROUP:-}" ]]; then diff --git a/tools/test.sh b/tools/test.sh index 23931a191..3e6b81668 100755 --- a/tools/test.sh +++ b/tools/test.sh @@ -42,10 +42,18 @@ x_cargo() { if [[ "${cmd}" == "miri" ]] && [[ -n "${MIRIFLAGS:-}" ]]; then echo "+ MIRIFLAGS='${MIRIFLAGS}' \\" fi - if [[ -n "${TS:-}" ]]; then - x "${cargo}" ${pre_args[@]+"${pre_args[@]}"} "$@" 2>&1 | "${TS}" -i '%.s ' + if type -P cargo-hack; then + if [[ -n "${TS:-}" ]]; then + x "${cargo}" ${pre_args[@]+"${pre_args[@]}"} hack "$@" 2>&1 | "${TS}" -i '%.s ' + else + x "${cargo}" ${pre_args[@]+"${pre_args[@]}"} hack "$@" + fi else - x "${cargo}" ${pre_args[@]+"${pre_args[@]}"} "$@" + if [[ -n "${TS:-}" ]]; then + x "${cargo}" ${pre_args[@]+"${pre_args[@]}"} "$@" 2>&1 | "${TS}" -i '%.s ' + else + x "${cargo}" ${pre_args[@]+"${pre_args[@]}"} "$@" + fi fi echo } @@ -168,7 +176,11 @@ if [[ -n "${target}" ]]; then fi fi fi -args+=(--features "${test_features}") +if type -P cargo-hack; then + args+=(--features "${test_features}" --ignore-unknown-features) +else + args+=(--features "${test_features}") +fi case "${cmd}" in build) ;; *) args+=(--workspace --exclude bench --exclude portable-atomic-internal-codegen) ;;