diff options
| author | Colin Sherratt <colin.sherratt@gmail.com> | 2013-12-30 19:17:35 -0500 |
|---|---|---|
| committer | Colin Sherratt <colin.sherratt@gmail.com> | 2013-12-31 14:31:23 -0500 |
| commit | 06a8d59ded67dfaa589112ceb8a8e031bc575249 (patch) | |
| tree | c53ff4a39cc4f05a4a2d951dfdf6ace174d02de0 /src | |
| parent | 5ff7b283731b795930d1e6782ae1639c83595e91 (diff) | |
| download | rust-06a8d59ded67dfaa589112ceb8a8e031bc575249.tar.gz rust-06a8d59ded67dfaa589112ceb8a8e031bc575249.zip | |
Add a copy-on-write container.
Diffstat (limited to 'src')
| -rw-r--r-- | src/libextra/arc.rs | 108 | ||||
| -rw-r--r-- | src/libstd/sync/arc.rs | 8 |
2 files changed, 116 insertions, 0 deletions
diff --git a/src/libextra/arc.rs b/src/libextra/arc.rs index a411c4e9185..81b183cc0ed 100644 --- a/src/libextra/arc.rs +++ b/src/libextra/arc.rs @@ -550,6 +550,50 @@ impl<'a, T:Freeze + Send> RWReadMode<'a, T> { } /**************************************************************************** + * Copy-on-write Arc + ****************************************************************************/ + +pub struct CowArc<T> { priv x: UnsafeArc<T> } + +/// A Copy-on-write Arc functions the same way as an `arc` except it allows +/// mutation of the contents if there is only a single reference to +/// the data. If there are multiple references the data is automatically +/// cloned and the task modifies the cloned data in place of the shared data. +impl<T:Clone+Send> CowArc<T> { + /// Create a copy-on-write atomically reference counted wrapper + #[inline] + pub fn new(data: T) -> CowArc<T> { + CowArc { x: UnsafeArc::new(data) } + } + + #[inline] + pub fn get<'a>(&'a self) -> &'a T { + unsafe { &*self.x.get_immut() } + } + + /// get a mutable reference to the contents. If there are more then one + /// reference to the contents of the `CowArc` will be cloned + /// and this reference updated to point to the cloned data. + #[inline] + pub fn get_mut<'a>(&'a mut self) -> &'a mut T { + if !self.x.is_owned() { + *self = CowArc::new(self.get().clone()) + } + unsafe { &mut *self.x.get() } + } +} + +impl<T:Clone+Send> Clone for CowArc<T> { + /// Duplicate a Copy-on-write Arc. See arc::clone for more details. + #[inline] + fn clone(&self) -> CowArc<T> { + CowArc { x: self.x.clone() } + } +} + + + +/**************************************************************************** * Tests ****************************************************************************/ @@ -958,4 +1002,68 @@ mod tests { // and I wasn't sure why :( . This is a mediocre "next best" option. 8.times(|| test_rw_write_cond_downgrade_read_race_helper()); } + + #[test] + fn test_cowarc_clone() + { + let cow0 = CowArc::new(75u); + let cow1 = cow0.clone(); + let cow2 = cow1.clone(); + + assert!(75 == *cow0.get()); + assert!(75 == *cow1.get()); + assert!(75 == *cow2.get()); + + assert!(cow0.get() == cow1.get()); + assert!(cow0.get() == cow2.get()); + } + + #[test] + fn test_cowarc_clone_get_mut() + { + let mut cow0 = CowArc::new(75u); + let mut cow1 = cow0.clone(); + let mut cow2 = cow1.clone(); + + assert!(75 == *cow0.get_mut()); + assert!(75 == *cow1.get_mut()); + assert!(75 == *cow2.get_mut()); + + *cow0.get_mut() += 1; + *cow1.get_mut() += 2; + *cow2.get_mut() += 3; + + assert!(76 == *cow0.get()); + assert!(77 == *cow1.get()); + assert!(78 == *cow2.get()); + + // none should point to the same backing memory + assert!(cow0.get() != cow1.get()); + assert!(cow0.get() != cow2.get()); + assert!(cow1.get() != cow2.get()); + } + + #[test] + fn test_cowarc_clone_get_mut2() + { + let mut cow0 = CowArc::new(75u); + let cow1 = cow0.clone(); + let cow2 = cow1.clone(); + + assert!(75 == *cow0.get()); + assert!(75 == *cow1.get()); + assert!(75 == *cow2.get()); + + *cow0.get_mut() += 1; + + assert!(76 == *cow0.get()); + assert!(75 == *cow1.get()); + assert!(75 == *cow2.get()); + + // cow1 and cow2 should share the same contents + // cow0 should have a unique reference + assert!(cow0.get() != cow1.get()); + assert!(cow0.get() != cow2.get()); + assert!(cow1.get() == cow2.get()); + } } diff --git a/src/libstd/sync/arc.rs b/src/libstd/sync/arc.rs index 7b94a3acc2b..5c452018b9b 100644 --- a/src/libstd/sync/arc.rs +++ b/src/libstd/sync/arc.rs @@ -94,6 +94,14 @@ impl<T: Send> UnsafeArc<T> { return &(*self.data).data as *T; } } + + /// checks if this is the only reference to the arc protected data + #[inline] + pub fn is_owned(&self) -> bool { + unsafe { + (*self.data).count.load(Relaxed) == 1 + } + } } impl<T: Send> Clone for UnsafeArc<T> { |
