diff --git a/refuse-pool/src/lib.rs b/refuse-pool/src/lib.rs index e37826a..39fa828 100644 --- a/refuse-pool/src/lib.rs +++ b/refuse-pool/src/lib.rs @@ -185,8 +185,8 @@ impl RootString { } /// Returns a typeless reference to this string. - pub fn as_any(&self) -> AnyRef { - self.0.as_any() + pub const fn downgrade_any(&self) -> AnyRef { + self.0.downgrade_any() } /// Returns the number of root references to this string, `self` included. diff --git a/src/lib.rs b/src/lib.rs index ce47fc4..c153da6 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -1550,14 +1550,33 @@ where /// Returns an untyped "weak" reference erased to this root. #[must_use] - pub fn downgrade_any(&self) -> AnyRef { + pub const fn downgrade_any(&self) -> AnyRef { self.reference.as_any() } - /// Returns this reference as an untyped reference. + /// Returns an untyped root reference. #[must_use] - pub fn as_any(&self) -> AnyRef { - self.reference.as_any() + pub fn to_any_root(&self) -> AnyRoot { + let roots = &self.as_rooted().roots; + roots.fetch_add(1, Ordering::Acquire); + + AnyRoot { + rooted: self.data.cast(), + roots, + any: self.reference.as_any(), + } + } + + /// Returns this root as an untyped root. + pub fn into_any_root(self) -> AnyRoot { + // We transfer ownership of this reference to the AnyRoot, so we want to + // avoid calling drop on `self`. + let this = ManuallyDrop::new(self); + AnyRoot { + rooted: this.data.cast(), + roots: &this.as_rooted().roots, + any: this.reference.as_any(), + } } fn as_rooted(&self) -> &Rooted { @@ -1792,7 +1811,7 @@ where /// Returns this reference as an untyped reference. #[must_use] - pub fn as_any(self) -> AnyRef { + pub const fn as_any(self) -> AnyRef { self.any } @@ -2665,6 +2684,78 @@ impl TypeIndex { } } +/// A type-erased root garbage collected reference. +#[derive(Eq, PartialEq, PartialOrd, Ord)] +pub struct AnyRoot { + rooted: *const (), + roots: *const AtomicU64, + any: AnyRef, +} + +impl AnyRoot { + /// Loads a reference to the underlying data. Returns `None` if `T` is not + /// the type of the underlying data. + pub fn load(&self) -> Option<&T> + where + T: Collectable, + { + if TypeIndex::of::() == self.any.type_index { + let rooted = unsafe { &*self.rooted.cast::>() }; + Some(&rooted.value) + } else { + None + } + } + + /// Returns an untyped "weak" reference to this root. + pub const fn as_any(&self) -> AnyRef { + self.any + } +} + +impl Clone for AnyRoot { + fn clone(&self) -> Self { + unsafe { &*self.roots }.fetch_add(1, Ordering::Acquire); + Self { + rooted: self.rooted, + roots: self.roots, + any: self.any, + } + } +} + +impl Drop for AnyRoot { + fn drop(&mut self) { + if unsafe { &*self.roots }.fetch_sub(1, Ordering::Acquire) == 1 { + CollectorCommand::schedule_collect_if_needed(); + } + } +} + +impl Hash for AnyRoot { + fn hash(&self, state: &mut H) { + self.any.hash(state); + } +} + +impl From> for AnyRoot +where + T: Collectable, +{ + fn from(value: Root) -> Self { + value.into_any_root() + } +} + +// SAFETY: AnyRoot's usage of a pointer prevents auto implementation. +// `Collectable` requires `Send`, and `Root` ensures proper Send + Sync +// behavior in its memory accesses. +unsafe impl Send for AnyRoot {} +// SAFETY: AnyRoot's usage of a pointer prevents auto implementation. +// `Collectable` requires `Send`, and `Root` ensures proper Send + Sync +// behavior in its memory accesses. +unsafe impl Sync for AnyRoot {} + /// A type-erased garbage collected reference. #[derive(Clone, Copy, Eq, PartialEq, PartialOrd, Ord)] pub struct AnyRef { @@ -2787,7 +2878,7 @@ where T: Collectable, { fn from(value: &'_ Root) -> Self { - value.as_any() + value.downgrade_any() } } diff --git a/src/tests.rs b/src/tests.rs index cb57b71..2b1bfea 100644 --- a/src/tests.rs +++ b/src/tests.rs @@ -43,7 +43,7 @@ use crate::{CollectionGuard, Root}; fn casting() { let guard = CollectionGuard::acquire(); let value = Root::new(2_u32, &guard); - let any = value.as_any(); + let any = value.downgrade_any(); assert_eq!(any.downcast_ref().load(&guard), Some(&2_u32)); assert_eq!(any.downcast_ref::().load(&guard), None); }