From 6f72fe1e208d96917c806bb4895b7014c1bfe164 Mon Sep 17 00:00:00 2001 From: Nicolas Silva Date: Sat, 18 Mar 2017 15:27:39 +0100 Subject: [PATCH] Implement arc::Weak::new_ref. --- src/liballoc/arc.rs | 48 ++++++++++++++++++++++++++++++++------------- 1 file changed, 34 insertions(+), 14 deletions(-) diff --git a/src/liballoc/arc.rs b/src/liballoc/arc.rs index 3ec45628886f7..fba1c3c9ee736 100644 --- a/src/liballoc/arc.rs +++ b/src/liballoc/arc.rs @@ -822,6 +822,37 @@ impl Weak { } impl Weak { + /// Creates another pointer to the same inner value, increasing the + /// weak reference count. + /// + /// # Examples + /// + /// ``` + /// use std::sync::Arc; + /// + /// let weak_five = Arc::downgrade(&Arc::new(5)); + /// + /// let _other_weak_five = weak_five.new_ref(); + /// // weak_five and _other_weak_five point to the same value. + /// ``` + #[inline] + fn new_ref(&self) -> Weak { + // See comments in Arc::new_ref() 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 = self.inner().weak.fetch_add(1, Relaxed); + + // See comments in Arc::new_ref() for why we do this (for mem::forget). + if old_size > MAX_REFCOUNT { + unsafe { + abort(); + } + } + + return Weak { ptr: self.ptr }; + } + /// Upgrades the `Weak` pointer to an [`Arc`][arc], if possible. /// /// Returns [`None`][option] if the strong count has reached zero and the @@ -893,6 +924,8 @@ impl Clone for Weak { /// /// This creates another pointer to the same inner value, increasing the /// weak reference count. + /// This is equivallent to calling `new_ref` with the added genericity of being callable + /// through the `Clone` trait. /// /// # Examples /// @@ -905,20 +938,7 @@ impl Clone for Weak { /// ``` #[inline] fn clone(&self) -> Weak { - // 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 = self.inner().weak.fetch_add(1, Relaxed); - - // See comments in Arc::clone() for why we do this (for mem::forget). - if old_size > MAX_REFCOUNT { - unsafe { - abort(); - } - } - - return Weak { ptr: self.ptr }; + self.new_ref() } }