// Copyright 2012-2014 The Rust Project Developers. See the COPYRIGHT // file at the top-level directory of this distribution and at // http://rust-lang.org/COPYRIGHT. // // Licensed under the Apache License, Version 2.0 or the MIT license // , at your // option. This file may not be copied, modified, or distributed // except according to those terms. /*! * Concurrency-enabled mechanisms for sharing mutable and/or immutable state * between tasks. * * # Example * * In this example, a large vector of floats is shared between several tasks. * With simple pipes, without Arc, a copy would have to be made for each task. * * ```rust * extern crate sync; * extern crate rand; * * use std::slice; * use sync::Arc; * * fn main() { * let numbers = slice::from_fn(100, |i| (i as f32) * rand::random()); * let shared_numbers = Arc::new(numbers); * * for _ in range(0, 10) { * let (tx, rx) = channel(); * tx.send(shared_numbers.clone()); * * spawn(proc() { * let shared_numbers = rx.recv(); * let local_numbers = shared_numbers.get(); * * // Work with the local numbers * }); * } * } * ``` */ #[allow(missing_doc, dead_code)]; use sync; use sync::{Mutex, RWLock}; use std::cast; use std::kinds::marker; use std::sync::arc::UnsafeArc; use std::task; #[cfg(stage0)] use std::kinds::Share; /// As sync::condvar, a mechanism for unlock-and-descheduling and /// signaling, for use with the Arc types. pub struct ArcCondvar<'a> { priv is_mutex: bool, priv failed: &'a bool, priv cond: &'a sync::Condvar<'a> } impl<'a> ArcCondvar<'a> { /// Atomically exit the associated Arc and block until a signal is sent. #[inline] pub fn wait(&self) { self.wait_on(0) } /** * Atomically exit the associated Arc and block on a specified condvar * until a signal is sent on that same condvar (as sync::cond.wait_on). * * wait() is equivalent to wait_on(0). */ #[inline] pub fn wait_on(&self, condvar_id: uint) { assert!(!*self.failed); self.cond.wait_on(condvar_id); // This is why we need to wrap sync::condvar. check_poison(self.is_mutex, *self.failed); } /// Wake up a blocked task. Returns false if there was no blocked task. #[inline] pub fn signal(&self) -> bool { self.signal_on(0) } /** * Wake up a blocked task on a specified condvar (as * sync::cond.signal_on). Returns false if there was no blocked task. */ #[inline] pub fn signal_on(&self, condvar_id: uint) -> bool { assert!(!*self.failed); self.cond.signal_on(condvar_id) } /// Wake up all blocked tasks. Returns the number of tasks woken. #[inline] pub fn broadcast(&self) -> uint { self.broadcast_on(0) } /** * Wake up all blocked tasks on a specified condvar (as * sync::cond.broadcast_on). Returns the number of tasks woken. */ #[inline] pub fn broadcast_on(&self, condvar_id: uint) -> uint { assert!(!*self.failed); self.cond.broadcast_on(condvar_id) } } /**************************************************************************** * Immutable Arc ****************************************************************************/ /// An atomically reference counted wrapper for shared immutable state. pub struct Arc { priv x: UnsafeArc } /** * Access the underlying data in an atomically reference counted * wrapper. */ impl Arc { /// Create an atomically reference counted wrapper. #[inline] pub fn new(data: T) -> Arc { Arc { x: UnsafeArc::new(data) } } #[inline] pub fn get<'a>(&'a self) -> &'a T { unsafe { &*self.x.get_immut() } } } impl Clone for Arc { /** * Duplicate an atomically reference counted wrapper. * * The resulting two `arc` objects will point to the same underlying data * object. However, one of the `arc` objects can be sent to another task, * allowing them to share the underlying data. */ #[inline] fn clone(&self) -> Arc { Arc { x: self.x.clone() } } } /**************************************************************************** * Mutex protected Arc (unsafe) ****************************************************************************/ #[doc(hidden)] struct MutexArcInner { lock: Mutex, failed: bool, data: T } /// An Arc with mutable data protected by a blocking mutex. pub struct MutexArc { priv x: UnsafeArc>, priv marker: marker::NoFreeze, } impl Clone for MutexArc { /// Duplicate a mutex-protected Arc. See arc::clone for more details. #[inline] fn clone(&self) -> MutexArc { // NB: Cloning the underlying mutex is not necessary. Its reference // count would be exactly the same as the shared state's. MutexArc { x: self.x.clone(), marker: marker::NoFreeze, } } } impl MutexArc { /// Create a mutex-protected Arc with the supplied data. pub fn new(user_data: T) -> MutexArc { MutexArc::new_with_condvars(user_data, 1) } /** * Create a mutex-protected Arc with the supplied data and a specified number * of condvars (as sync::Mutex::new_with_condvars). */ pub fn new_with_condvars(user_data: T, num_condvars: uint) -> MutexArc { let data = MutexArcInner { lock: Mutex::new_with_condvars(num_condvars), failed: false, data: user_data }; MutexArc { x: UnsafeArc::new(data), marker: marker::NoFreeze, } } /** * Access the underlying mutable data with mutual exclusion from other * tasks. The argument closure will be run with the mutex locked; all * other tasks wishing to access the data will block until the closure * finishes running. * * If you wish to nest MutexArcs, one strategy for ensuring safety at * runtime is to add a "nesting level counter" inside the stored data, and * when traversing the arcs, assert that they monotonically decrease. * * # Failure * * Failing while inside the Arc will unlock the Arc while unwinding, so * that other tasks won't block forever. It will also poison the Arc: * any tasks that subsequently try to access it (including those already * blocked on the mutex) will also fail immediately. */ #[inline] pub fn access(&self, blk: |x: &mut T| -> U) -> U { let state = self.x.get(); unsafe { // Borrowck would complain about this if the code were // not already unsafe. See borrow_rwlock, far below. (&(*state).lock).lock(|| { check_poison(true, (*state).failed); let _z = PoisonOnFail::new(&mut (*state).failed); blk(&mut (*state).data) }) } } /// As access(), but with a condvar, as sync::mutex.lock_cond(). #[inline] pub fn access_cond(&self, blk: |x: &mut T, c: &ArcCondvar| -> U) -> U { let state = self.x.get(); unsafe { (&(*state).lock).lock_cond(|cond| { check_poison(true, (*state).failed); let _z = PoisonOnFail::new(&mut (*state).failed); blk(&mut (*state).data, &ArcCondvar {is_mutex: true, failed: &(*state).failed, cond: cond }) }) } } } // Common code for {mutex.access,rwlock.write}{,_cond}. #[inline] #[doc(hidden)] fn check_poison(is_mutex: bool, failed: bool) { if failed { if is_mutex { fail!("Poisoned MutexArc - another task failed inside!"); } else { fail!("Poisoned rw_arc - another task failed inside!"); } } } #[doc(hidden)] struct PoisonOnFail { flag: *mut bool, failed: bool, } impl Drop for PoisonOnFail { fn drop(&mut self) { unsafe { /* assert!(!*self.failed); -- might be false in case of cond.wait() */ if !self.failed && task::failing() { *self.flag = true; } } } } impl PoisonOnFail { fn new<'a>(flag: &'a mut bool) -> PoisonOnFail { PoisonOnFail { flag: flag, failed: task::failing() } } } /**************************************************************************** * R/W lock protected Arc ****************************************************************************/ #[doc(hidden)] struct RWArcInner { lock: RWLock, failed: bool, data: T } /** * A dual-mode Arc protected by a reader-writer lock. The data can be accessed * mutably or immutably, and immutably-accessing tasks may run concurrently. * * Unlike mutex_arcs, rw_arcs are safe, because they cannot be nested. */ pub struct RWArc { priv x: UnsafeArc>, priv marker: marker::NoFreeze, priv marker1: marker::NoShare, } impl Clone for RWArc { /// Duplicate a rwlock-protected Arc. See arc::clone for more details. #[inline] fn clone(&self) -> RWArc { RWArc { x: self.x.clone(), marker: marker::NoFreeze, marker1: marker::NoShare, } } } impl RWArc { /// Create a reader/writer Arc with the supplied data. pub fn new(user_data: T) -> RWArc { RWArc::new_with_condvars(user_data, 1) } /** * Create a reader/writer Arc with the supplied data and a specified number * of condvars (as sync::RWLock::new_with_condvars). */ pub fn new_with_condvars(user_data: T, num_condvars: uint) -> RWArc { let data = RWArcInner { lock: RWLock::new_with_condvars(num_condvars), failed: false, data: user_data }; RWArc { x: UnsafeArc::new(data), marker: marker::NoFreeze, marker1: marker::NoShare, } } /** * Access the underlying data mutably. Locks the rwlock in write mode; * other readers and writers will block. * * # Failure * * Failing while inside the Arc will unlock the Arc while unwinding, so * that other tasks won't block forever. As MutexArc.access, it will also * poison the Arc, so subsequent readers and writers will both also fail. */ #[inline] pub fn write(&self, blk: |x: &mut T| -> U) -> U { unsafe { let state = self.x.get(); (*borrow_rwlock(state)).write(|| { check_poison(false, (*state).failed); let _z = PoisonOnFail::new(&mut (*state).failed); blk(&mut (*state).data) }) } } /// As write(), but with a condvar, as sync::rwlock.write_cond(). #[inline] pub fn write_cond(&self, blk: |x: &mut T, c: &ArcCondvar| -> U) -> U { unsafe { let state = self.x.get(); (*borrow_rwlock(state)).write_cond(|cond| { check_poison(false, (*state).failed); let _z = PoisonOnFail::new(&mut (*state).failed); blk(&mut (*state).data, &ArcCondvar {is_mutex: false, failed: &(*state).failed, cond: cond}) }) } } /** * Access the underlying data immutably. May run concurrently with other * reading tasks. * * # Failure * * Failing will unlock the Arc while unwinding. However, unlike all other * access modes, this will not poison the Arc. */ pub fn read(&self, blk: |x: &T| -> U) -> U { unsafe { let state = self.x.get(); (*state).lock.read(|| { check_poison(false, (*state).failed); blk(&(*state).data) }) } } /** * As write(), but with the ability to atomically 'downgrade' the lock. * See sync::rwlock.write_downgrade(). The RWWriteMode token must be used * to obtain the &mut T, and can be transformed into a RWReadMode token by * calling downgrade(), after which a &T can be obtained instead. * * # Example * * ```rust * use sync::RWArc; * * let arc = RWArc::new(1); * arc.write_downgrade(|mut write_token| { * write_token.write_cond(|state, condvar| { * // ... exclusive access with mutable state ... * }); * let read_token = arc.downgrade(write_token); * read_token.read(|state| { * // ... shared access with immutable state ... * }); * }) * ``` */ pub fn write_downgrade(&self, blk: |v: RWWriteMode| -> U) -> U { unsafe { let state = self.x.get(); (*borrow_rwlock(state)).write_downgrade(|write_mode| { check_poison(false, (*state).failed); blk(RWWriteMode { data: &mut (*state).data, token: write_mode, poison: PoisonOnFail::new(&mut (*state).failed) }) }) } } /// To be called inside of the write_downgrade block. pub fn downgrade<'a>(&self, token: RWWriteMode<'a, T>) -> RWReadMode<'a, T> { unsafe { // The rwlock should assert that the token belongs to us for us. let state = self.x.get(); let RWWriteMode { data: data, token: t, poison: _poison } = token; // Let readers in let new_token = (*state).lock.downgrade(t); // Whatever region the input reference had, it will be safe to use // the same region for the output reference. (The only 'unsafe' part // of this cast is removing the mutability.) let new_data = data; // Downgrade ensured the token belonged to us. Just a sanity check. assert!((&(*state).data as *T as uint) == (new_data as *mut T as uint)); // Produce new token RWReadMode { data: new_data, token: new_token, } } } } // Borrowck rightly complains about immutably aliasing the rwlock in order to // lock it. This wraps the unsafety, with the justification that the 'lock' // field is never overwritten; only 'failed' and 'data'. #[doc(hidden)] fn borrow_rwlock(state: *mut RWArcInner) -> *RWLock { unsafe { cast::transmute(&(*state).lock) } } /// The "write permission" token used for RWArc.write_downgrade(). pub struct RWWriteMode<'a, T> { priv data: &'a mut T, priv token: sync::RWLockWriteMode<'a>, priv poison: PoisonOnFail, } /// The "read permission" token used for RWArc.write_downgrade(). pub struct RWReadMode<'a, T> { priv data: &'a T, priv token: sync::RWLockReadMode<'a>, } impl<'a, T: Share + Send> RWWriteMode<'a, T> { /// Access the pre-downgrade RWArc in write mode. pub fn write(&mut self, blk: |x: &mut T| -> U) -> U { match *self { RWWriteMode { data: &ref mut data, token: ref token, poison: _ } => { token.write(|| blk(data)) } } } /// Access the pre-downgrade RWArc in write mode with a condvar. pub fn write_cond(&mut self, blk: |x: &mut T, c: &ArcCondvar| -> U) -> U { match *self { RWWriteMode { data: &ref mut data, token: ref token, poison: ref poison } => { token.write_cond(|cond| { unsafe { let cvar = ArcCondvar { is_mutex: false, failed: &*poison.flag, cond: cond }; blk(data, &cvar) } }) } } } } impl<'a, T: Share + Send> RWReadMode<'a, T> { /// Access the post-downgrade rwlock in read mode. pub fn read(&self, blk: |x: &T| -> U) -> U { match *self { RWReadMode { data: data, token: ref token } => { token.read(|| blk(data)) } } } } /**************************************************************************** * Copy-on-write Arc ****************************************************************************/ pub struct CowArc { priv x: UnsafeArc } /// 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 CowArc { /// Create a copy-on-write atomically reference counted wrapper #[inline] pub fn new(data: T) -> CowArc { 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 Clone for CowArc { /// Duplicate a Copy-on-write Arc. See arc::clone for more details. fn clone(&self) -> CowArc { CowArc { x: self.x.clone() } } } /**************************************************************************** * Tests ****************************************************************************/ #[cfg(test)] mod tests { use super::{Arc, RWArc, MutexArc, CowArc}; use std::task; #[test] fn manually_share_arc() { let v = ~[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]; let arc_v = Arc::new(v); let (tx, rx) = channel(); task::spawn(proc() { let arc_v: Arc<~[int]> = rx.recv(); let v = arc_v.get().clone(); assert_eq!(v[3], 4); }); tx.send(arc_v.clone()); assert_eq!(arc_v.get()[2], 3); assert_eq!(arc_v.get()[4], 5); info!("{:?}", arc_v); } #[test] fn test_mutex_arc_condvar() { let arc = ~MutexArc::new(false); let arc2 = ~arc.clone(); let (tx, rx) = channel(); task::spawn(proc() { // wait until parent gets in rx.recv(); arc2.access_cond(|state, cond| { *state = true; cond.signal(); }) }); arc.access_cond(|state, cond| { tx.send(()); assert!(!*state); while !*state { cond.wait(); } }) } #[test] #[should_fail] fn test_arc_condvar_poison() { let arc = ~MutexArc::new(1); let arc2 = ~arc.clone(); let (tx, rx) = channel(); spawn(proc() { let _ = rx.recv(); arc2.access_cond(|one, cond| { cond.signal(); // Parent should fail when it wakes up. assert_eq!(*one, 0); }) }); arc.access_cond(|one, cond| { tx.send(()); while *one == 1 { cond.wait(); } }) } #[test] #[should_fail] fn test_mutex_arc_poison() { let arc = ~MutexArc::new(1); let arc2 = ~arc.clone(); let _ = task::try(proc() { arc2.access(|one| { assert_eq!(*one, 2); }) }); arc.access(|one| { assert_eq!(*one, 1); }) } #[test] fn test_mutex_arc_nested() { // Tests nested mutexes and access // to underlaying data. let arc = ~MutexArc::new(1); let arc2 = ~MutexArc::new(*arc); task::spawn(proc() { (*arc2).access(|mutex| { (*mutex).access(|one| { assert!(*one == 1); }) }) }); } #[test] fn test_mutex_arc_access_in_unwind() { let arc = MutexArc::new(1i); let arc2 = arc.clone(); let _ = task::try::<()>(proc() { struct Unwinder { i: MutexArc } impl Drop for Unwinder { fn drop(&mut self) { self.i.access(|num| *num += 1); } } let _u = Unwinder { i: arc2 }; fail!(); }); assert_eq!(2, arc.access(|n| *n)); } #[test] #[should_fail] fn test_rw_arc_poison_wr() { let arc = RWArc::new(1); let arc2 = arc.clone(); let _ = task::try(proc() { arc2.write(|one| { assert_eq!(*one, 2); }) }); arc.read(|one| { assert_eq!(*one, 1); }) } #[test] #[should_fail] fn test_rw_arc_poison_ww() { let arc = RWArc::new(1); let arc2 = arc.clone(); let _ = task::try(proc() { arc2.write(|one| { assert_eq!(*one, 2); }) }); arc.write(|one| { assert_eq!(*one, 1); }) } #[test] #[should_fail] fn test_rw_arc_poison_dw() { let arc = RWArc::new(1); let arc2 = arc.clone(); let _ = task::try(proc() { arc2.write_downgrade(|mut write_mode| { write_mode.write(|one| { assert_eq!(*one, 2); }) }) }); arc.write(|one| { assert_eq!(*one, 1); }) } #[test] fn test_rw_arc_no_poison_rr() { let arc = RWArc::new(1); let arc2 = arc.clone(); let _ = task::try(proc() { arc2.read(|one| { assert_eq!(*one, 2); }) }); arc.read(|one| { assert_eq!(*one, 1); }) } #[test] fn test_rw_arc_no_poison_rw() { let arc = RWArc::new(1); let arc2 = arc.clone(); let _ = task::try(proc() { arc2.read(|one| { assert_eq!(*one, 2); }) }); arc.write(|one| { assert_eq!(*one, 1); }) } #[test] fn test_rw_arc_no_poison_dr() { let arc = RWArc::new(1); let arc2 = arc.clone(); let _ = task::try(proc() { arc2.write_downgrade(|write_mode| { let read_mode = arc2.downgrade(write_mode); read_mode.read(|one| { assert_eq!(*one, 2); }) }) }); arc.write(|one| { assert_eq!(*one, 1); }) } #[test] fn test_rw_arc() { let arc = RWArc::new(0); let arc2 = arc.clone(); let (tx, rx) = channel(); task::spawn(proc() { arc2.write(|num| { for _ in range(0, 10) { let tmp = *num; *num = -1; task::deschedule(); *num = tmp + 1; } tx.send(()); }) }); // Readers try to catch the writer in the act let mut children = ~[]; for _ in range(0, 5) { let arc3 = arc.clone(); let mut builder = task::task(); children.push(builder.future_result()); builder.spawn(proc() { arc3.read(|num| { assert!(*num >= 0); }) }); } // Wait for children to pass their asserts for r in children.mut_iter() { let _ = r.recv(); } // Wait for writer to finish rx.recv(); arc.read(|num| { assert_eq!(*num, 10); }) } #[test] fn test_rw_arc_access_in_unwind() { let arc = RWArc::new(1i); let arc2 = arc.clone(); let _ = task::try::<()>(proc() { struct Unwinder { i: RWArc } impl Drop for Unwinder { fn drop(&mut self) { self.i.write(|num| *num += 1); } } let _u = Unwinder { i: arc2 }; fail!(); }); assert_eq!(2, arc.read(|n| *n)); } #[test] fn test_rw_downgrade() { // (1) A downgrader gets in write mode and does cond.wait. // (2) A writer gets in write mode, sets state to 42, and does signal. // (3) Downgrader wakes, sets state to 31337. // (4) tells writer and all other readers to contend as it downgrades. // (5) Writer attempts to set state back to 42, while downgraded task // and all reader tasks assert that it's 31337. let arc = RWArc::new(0); // Reader tasks let mut reader_convos = ~[]; for _ in range(0, 10) { let ((tx1, rx1), (tx2, rx2)) = (channel(), channel()); reader_convos.push((tx1, rx2)); let arcn = arc.clone(); task::spawn(proc() { rx1.recv(); // wait for downgrader to give go-ahead arcn.read(|state| { assert_eq!(*state, 31337); tx2.send(()); }) }); } // Writer task let arc2 = arc.clone(); let ((tx1, rx1), (tx2, rx2)) = (channel(), channel()); task::spawn(proc() { rx1.recv(); arc2.write_cond(|state, cond| { assert_eq!(*state, 0); *state = 42; cond.signal(); }); rx1.recv(); arc2.write(|state| { // This shouldn't happen until after the downgrade read // section, and all other readers, finish. assert_eq!(*state, 31337); *state = 42; }); tx2.send(()); }); // Downgrader (us) arc.write_downgrade(|mut write_mode| { write_mode.write_cond(|state, cond| { tx1.send(()); // send to another writer who will wake us up while *state == 0 { cond.wait(); } assert_eq!(*state, 42); *state = 31337; // send to other readers for &(ref mut rc, _) in reader_convos.mut_iter() { rc.send(()) } }); let read_mode = arc.downgrade(write_mode); read_mode.read(|state| { // complete handshake with other readers for &(_, ref mut rp) in reader_convos.mut_iter() { rp.recv() } tx1.send(()); // tell writer to try again assert_eq!(*state, 31337); }); }); rx2.recv(); // complete handshake with writer } #[cfg(test)] fn test_rw_write_cond_downgrade_read_race_helper() { // Tests that when a downgrader hands off the "reader cloud" lock // because of a contending reader, a writer can't race to get it // instead, which would result in readers_and_writers. This tests // the sync module rather than this one, but it's here because an // rwarc gives us extra shared state to help check for the race. // If you want to see this test fail, go to sync.rs and replace the // line in RWLock::write_cond() that looks like: // "blk(&ArcCondvar { order: opt_lock, ..*cond })" // with just "blk(cond)". let x = RWArc::new(true); let (tx, rx) = channel(); // writer task let xw = x.clone(); task::spawn(proc() { xw.write_cond(|state, c| { tx.send(()); // tell downgrader it's ok to go c.wait(); // The core of the test is here: the condvar reacquire path // must involve order_lock, so that it cannot race with a reader // trying to receive the "reader cloud lock hand-off". *state = false; }) }); rx.recv(); // wait for writer to get in x.write_downgrade(|mut write_mode| { write_mode.write_cond(|state, c| { assert!(*state); // make writer contend in the cond-reacquire path c.signal(); }); // make a reader task to trigger the "reader cloud lock" handoff let xr = x.clone(); let (tx, rx) = channel(); task::spawn(proc() { tx.send(()); xr.read(|_state| { }) }); rx.recv(); // wait for reader task to exist let read_mode = x.downgrade(write_mode); read_mode.read(|state| { // if writer mistakenly got in, make sure it mutates state // before we assert on it for _ in range(0, 5) { task::deschedule(); } // make sure writer didn't get in. assert!(*state); }) }); } #[test] fn test_rw_write_cond_downgrade_read_race() { // Ideally the above test case would have deschedule statements in it that // helped to expose the race nearly 100% of the time... but adding // deschedules in the intuitively-right locations made it even less likely, // and I wasn't sure why :( . This is a mediocre "next best" option. for _ in range(0, 8) { 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()); } }