2012-12-03 16:48:01 -08:00
|
|
|
// Copyright 2012 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 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
// except according to those terms.
|
|
|
|
|
2012-08-14 19:25:42 -04:00
|
|
|
// NB: transitionary, de-mode-ing.
|
2012-10-04 19:58:31 -07:00
|
|
|
#[forbid(deprecated_mode)];
|
2012-08-10 18:20:03 -04:00
|
|
|
/**
|
|
|
|
* Concurrency-enabled mechanisms for sharing mutable and/or immutable state
|
|
|
|
* between tasks.
|
|
|
|
*/
|
|
|
|
|
2012-12-23 17:41:37 -05:00
|
|
|
use sync;
|
|
|
|
use sync::{Mutex, mutex_with_condvars, RWlock, rwlock_with_condvars};
|
2012-08-10 18:20:03 -04:00
|
|
|
|
2012-12-23 17:41:37 -05:00
|
|
|
use core::cast;
|
|
|
|
use core::pipes;
|
2013-01-08 19:37:25 -08:00
|
|
|
use core::prelude::*;
|
2012-12-23 17:41:37 -05:00
|
|
|
use core::private::{SharedMutableState, shared_mutable_state};
|
|
|
|
use core::private::{clone_shared_mutable_state, unwrap_shared_mutable_state};
|
|
|
|
use core::private::{get_shared_mutable_state, get_shared_immutable_state};
|
|
|
|
use core::ptr;
|
|
|
|
use core::task;
|
|
|
|
use core::util;
|
2012-08-13 19:45:17 -04:00
|
|
|
|
|
|
|
/// As sync::condvar, a mechanism for unlock-and-descheduling and signalling.
|
2012-09-28 16:05:33 -07:00
|
|
|
pub struct Condvar { is_mutex: bool, failed: &mut bool, cond: &sync::Condvar }
|
2012-08-13 19:45:17 -04:00
|
|
|
|
2012-08-26 21:28:36 -04:00
|
|
|
impl &Condvar {
|
2012-08-13 19:45:17 -04:00
|
|
|
/// Atomically exit the associated ARC and block until a signal is sent.
|
2012-08-14 23:35:12 -04:00
|
|
|
#[inline(always)]
|
|
|
|
fn wait() { 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(always)]
|
|
|
|
fn wait_on(condvar_id: uint) {
|
2012-08-13 19:45:17 -04:00
|
|
|
assert !*self.failed;
|
2012-08-14 23:35:12 -04:00
|
|
|
self.cond.wait_on(condvar_id);
|
2012-08-13 19:45:17 -04:00
|
|
|
// 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.
|
2012-08-14 23:35:12 -04:00
|
|
|
#[inline(always)]
|
|
|
|
fn signal() -> 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(always)]
|
|
|
|
fn signal_on(condvar_id: uint) -> bool {
|
2012-08-13 19:45:17 -04:00
|
|
|
assert !*self.failed;
|
2012-08-14 23:35:12 -04:00
|
|
|
self.cond.signal_on(condvar_id)
|
2012-08-13 19:45:17 -04:00
|
|
|
}
|
|
|
|
/// Wake up all blocked tasks. Returns the number of tasks woken.
|
2012-08-14 23:35:12 -04:00
|
|
|
#[inline(always)]
|
|
|
|
fn broadcast() -> uint { self.broadcast_on(0) }
|
|
|
|
/**
|
|
|
|
* Wake up all blocked tasks on a specified condvar (as
|
|
|
|
* sync::cond.broadcast_on). Returns Returns the number of tasks woken.
|
|
|
|
*/
|
|
|
|
#[inline(always)]
|
|
|
|
fn broadcast_on(condvar_id: uint) -> uint {
|
2012-08-13 19:45:17 -04:00
|
|
|
assert !*self.failed;
|
2012-08-14 23:35:12 -04:00
|
|
|
self.cond.broadcast_on(condvar_id)
|
2012-08-13 19:45:17 -04:00
|
|
|
}
|
|
|
|
}
|
2012-08-10 18:20:03 -04:00
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Immutable ARC
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/// An atomically reference counted wrapper for shared immutable state.
|
2013-01-28 10:46:43 -08:00
|
|
|
struct ARC<T> { x: SharedMutableState<T> }
|
2012-08-10 18:20:03 -04:00
|
|
|
|
|
|
|
/// Create an atomically reference counted wrapper.
|
2012-12-11 13:50:04 -08:00
|
|
|
pub fn ARC<T: Const Owned>(data: T) -> ARC<T> {
|
2012-09-10 17:50:48 -07:00
|
|
|
ARC { x: unsafe { shared_mutable_state(move data) } }
|
2012-08-10 18:20:03 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Access the underlying data in an atomically reference counted
|
|
|
|
* wrapper.
|
|
|
|
*/
|
2012-12-11 13:50:04 -08:00
|
|
|
pub fn get<T: Const Owned>(rc: &a/ARC<T>) -> &a/T {
|
2012-08-10 18:20:03 -04:00
|
|
|
unsafe { get_shared_immutable_state(&rc.x) }
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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.
|
|
|
|
*/
|
2012-12-11 13:50:04 -08:00
|
|
|
pub fn clone<T: Const Owned>(rc: &ARC<T>) -> ARC<T> {
|
2012-08-26 21:28:36 -04:00
|
|
|
ARC { x: unsafe { clone_shared_mutable_state(&rc.x) } }
|
2012-08-10 18:20:03 -04:00
|
|
|
}
|
|
|
|
|
2012-08-20 22:30:53 -04:00
|
|
|
/**
|
|
|
|
* Retrieve the data back out of the ARC. This function blocks until the
|
|
|
|
* reference given to it is the last existing one, and then unwrap the data
|
|
|
|
* instead of destroying it.
|
|
|
|
*
|
|
|
|
* If multiple tasks call unwrap, all but the first will fail. Do not call
|
|
|
|
* unwrap from a task that holds another reference to the same ARC; it is
|
|
|
|
* guaranteed to deadlock.
|
|
|
|
*/
|
2013-01-25 17:25:41 -08:00
|
|
|
pub fn unwrap<T: Const Owned>(rc: ARC<T>) -> T {
|
2012-10-23 11:11:23 -07:00
|
|
|
let ARC { x: x } = move rc;
|
2012-09-10 17:50:48 -07:00
|
|
|
unsafe { unwrap_shared_mutable_state(move x) }
|
2012-08-20 17:58:59 -04:00
|
|
|
}
|
|
|
|
|
2012-12-11 13:50:04 -08:00
|
|
|
impl<T: Const Owned> ARC<T>: Clone {
|
2012-11-26 16:12:47 -08:00
|
|
|
fn clone(&self) -> ARC<T> {
|
|
|
|
clone(self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-08-10 18:20:03 -04:00
|
|
|
/****************************************************************************
|
|
|
|
* Mutex protected ARC (unsafe)
|
|
|
|
****************************************************************************/
|
|
|
|
|
2012-08-15 13:55:20 -04:00
|
|
|
#[doc(hidden)]
|
2013-01-28 10:46:43 -08:00
|
|
|
struct MutexARCInner<T> { lock: Mutex, failed: bool, data: T }
|
2012-08-10 20:46:19 -04:00
|
|
|
/// An ARC with mutable data protected by a blocking mutex.
|
2013-01-28 10:46:43 -08:00
|
|
|
struct MutexARC<T> { x: SharedMutableState<MutexARCInner<T>> }
|
2012-08-10 20:46:19 -04:00
|
|
|
|
|
|
|
/// Create a mutex-protected ARC with the supplied data.
|
2012-12-11 13:50:04 -08:00
|
|
|
pub fn MutexARC<T: Owned>(user_data: T) -> MutexARC<T> {
|
2012-09-10 17:50:48 -07:00
|
|
|
mutex_arc_with_condvars(move user_data, 1)
|
2012-08-14 23:35:12 -04:00
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Create a mutex-protected ARC with the supplied data and a specified number
|
|
|
|
* of condvars (as sync::mutex_with_condvars).
|
|
|
|
*/
|
2012-12-11 13:50:04 -08:00
|
|
|
pub fn mutex_arc_with_condvars<T: Owned>(user_data: T,
|
2012-08-26 21:28:36 -04:00
|
|
|
num_condvars: uint) -> MutexARC<T> {
|
2012-08-14 23:35:12 -04:00
|
|
|
let data =
|
2012-08-26 21:28:36 -04:00
|
|
|
MutexARCInner { lock: mutex_with_condvars(num_condvars),
|
2012-09-18 22:35:42 -07:00
|
|
|
failed: false, data: move user_data };
|
2012-09-10 17:50:48 -07:00
|
|
|
MutexARC { x: unsafe { shared_mutable_state(move data) } }
|
2012-08-10 20:46:19 -04:00
|
|
|
}
|
|
|
|
|
2012-12-11 13:50:04 -08:00
|
|
|
impl<T: Owned> MutexARC<T>: Clone {
|
2012-08-10 20:46:19 -04:00
|
|
|
/// Duplicate a mutex-protected ARC, as arc::clone.
|
2012-11-26 16:12:47 -08:00
|
|
|
fn clone(&self) -> MutexARC<T> {
|
2012-08-10 20:46:19 -04:00
|
|
|
// NB: Cloning the underlying mutex is not necessary. Its reference
|
|
|
|
// count would be exactly the same as the shared state's.
|
2012-08-26 21:28:36 -04:00
|
|
|
MutexARC { x: unsafe { clone_shared_mutable_state(&self.x) } }
|
2012-08-10 20:46:19 -04:00
|
|
|
}
|
2012-11-26 16:12:47 -08:00
|
|
|
}
|
|
|
|
|
2012-12-11 13:50:04 -08:00
|
|
|
impl<T: Owned> &MutexARC<T> {
|
2012-08-10 20:46:19 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* The reason this function is 'unsafe' is because it is possible to
|
|
|
|
* construct a circular reference among multiple ARCs by mutating the
|
|
|
|
* underlying data. This creates potential for deadlock, but worse, this
|
|
|
|
* will guarantee a memory leak of all involved ARCs. Using mutex ARCs
|
|
|
|
* inside of other ARCs is safe in absence of circular references.
|
|
|
|
*
|
|
|
|
* If you wish to nest mutex_arcs, 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(always)]
|
|
|
|
unsafe fn access<U>(blk: fn(x: &mut T) -> U) -> U {
|
|
|
|
let state = unsafe { get_shared_mutable_state(&self.x) };
|
|
|
|
// Borrowck would complain about this if the function were not already
|
|
|
|
// unsafe. See borrow_rwlock, far below.
|
|
|
|
do (&state.lock).lock {
|
|
|
|
check_poison(true, state.failed);
|
2012-08-26 21:28:36 -04:00
|
|
|
let _z = PoisonOnFail(&mut state.failed);
|
2012-08-13 19:45:17 -04:00
|
|
|
blk(&mut state.data)
|
2012-08-10 20:46:19 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
/// As access(), but with a condvar, as sync::mutex.lock_cond().
|
|
|
|
#[inline(always)]
|
2012-08-26 21:28:36 -04:00
|
|
|
unsafe fn access_cond<U>(blk: fn(x: &x/mut T, c: &c/Condvar) -> U) -> U {
|
2012-08-10 20:46:19 -04:00
|
|
|
let state = unsafe { get_shared_mutable_state(&self.x) };
|
|
|
|
do (&state.lock).lock_cond |cond| {
|
|
|
|
check_poison(true, state.failed);
|
2012-08-26 21:28:36 -04:00
|
|
|
let _z = PoisonOnFail(&mut state.failed);
|
2012-08-13 19:45:17 -04:00
|
|
|
blk(&mut state.data,
|
2012-08-26 21:28:36 -04:00
|
|
|
&Condvar { is_mutex: true, failed: &mut state.failed,
|
2012-08-13 19:45:17 -04:00
|
|
|
cond: cond })
|
2012-08-10 20:46:19 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-08-20 22:30:53 -04:00
|
|
|
/**
|
|
|
|
* Retrieves the data, blocking until all other references are dropped,
|
|
|
|
* exactly as arc::unwrap.
|
|
|
|
*
|
|
|
|
* Will additionally fail if another task has failed while accessing the arc.
|
|
|
|
*/
|
2012-10-11 15:37:37 -07:00
|
|
|
// FIXME(#3724) make this a by-move method on the arc
|
2012-12-11 13:50:04 -08:00
|
|
|
pub fn unwrap_mutex_arc<T: Owned>(arc: MutexARC<T>) -> T {
|
2012-10-23 11:11:23 -07:00
|
|
|
let MutexARC { x: x } = move arc;
|
2012-09-10 17:50:48 -07:00
|
|
|
let inner = unsafe { unwrap_shared_mutable_state(move x) };
|
2012-10-23 11:11:23 -07:00
|
|
|
let MutexARCInner { failed: failed, data: data, _ } = move inner;
|
2012-08-20 17:58:59 -04:00
|
|
|
if failed {
|
2012-08-26 21:28:36 -04:00
|
|
|
fail ~"Can't unwrap poisoned MutexARC - another task failed inside!"
|
2012-08-20 17:58:59 -04:00
|
|
|
}
|
2012-09-10 17:50:48 -07:00
|
|
|
move data
|
2012-08-20 17:58:59 -04:00
|
|
|
}
|
|
|
|
|
2012-08-10 20:46:19 -04:00
|
|
|
// Common code for {mutex.access,rwlock.write}{,_cond}.
|
|
|
|
#[inline(always)]
|
2012-08-15 13:55:20 -04:00
|
|
|
#[doc(hidden)]
|
2012-08-10 20:46:19 -04:00
|
|
|
fn check_poison(is_mutex: bool, failed: bool) {
|
|
|
|
if failed {
|
|
|
|
if is_mutex {
|
2012-08-26 21:28:36 -04:00
|
|
|
fail ~"Poisoned MutexARC - another task failed inside!";
|
2012-08-10 20:46:19 -04:00
|
|
|
} else {
|
|
|
|
fail ~"Poisoned rw_arc - another task failed inside!";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-08-15 13:55:20 -04:00
|
|
|
#[doc(hidden)]
|
2012-08-26 21:28:36 -04:00
|
|
|
struct PoisonOnFail {
|
2013-01-23 18:15:06 -08:00
|
|
|
failed: *mut bool,
|
2012-11-13 21:38:18 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
impl PoisonOnFail : Drop {
|
2012-11-28 15:42:16 -08:00
|
|
|
fn finalize(&self) {
|
2013-01-23 18:15:06 -08:00
|
|
|
unsafe {
|
|
|
|
/* assert !*self.failed;
|
|
|
|
-- might be false in case of cond.wait() */
|
|
|
|
if task::failing() {
|
|
|
|
*self.failed = true;
|
|
|
|
}
|
|
|
|
}
|
2012-08-13 19:45:17 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-23 18:15:06 -08:00
|
|
|
fn PoisonOnFail(failed: &r/mut bool) -> PoisonOnFail {
|
2012-09-04 17:22:09 -07:00
|
|
|
PoisonOnFail {
|
2013-01-23 18:15:06 -08:00
|
|
|
failed: ptr::to_mut_unsafe_ptr(failed)
|
2012-09-04 17:22:09 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-08-10 18:20:03 -04:00
|
|
|
/****************************************************************************
|
|
|
|
* R/W lock protected ARC
|
|
|
|
****************************************************************************/
|
|
|
|
|
2012-08-15 13:55:20 -04:00
|
|
|
#[doc(hidden)]
|
2013-01-28 10:46:43 -08:00
|
|
|
struct RWARCInner<T> { lock: RWlock, failed: bool, data: T }
|
2012-08-10 20:46:19 -04:00
|
|
|
/**
|
|
|
|
* 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.
|
|
|
|
*/
|
2013-01-28 10:46:43 -08:00
|
|
|
struct RWARC<T> {
|
2012-09-07 14:50:47 -07:00
|
|
|
x: SharedMutableState<RWARCInner<T>>,
|
|
|
|
mut cant_nest: ()
|
2012-08-10 20:46:19 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Create a reader/writer ARC with the supplied data.
|
2012-12-11 13:50:04 -08:00
|
|
|
pub fn RWARC<T: Const Owned>(user_data: T) -> RWARC<T> {
|
2012-09-10 17:50:48 -07:00
|
|
|
rw_arc_with_condvars(move user_data, 1)
|
2012-08-14 23:35:12 -04:00
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Create a reader/writer ARC with the supplied data and a specified number
|
|
|
|
* of condvars (as sync::rwlock_with_condvars).
|
|
|
|
*/
|
2012-12-11 13:50:04 -08:00
|
|
|
pub fn rw_arc_with_condvars<T: Const Owned>(user_data: T,
|
2012-08-26 21:28:36 -04:00
|
|
|
num_condvars: uint) -> RWARC<T> {
|
2012-08-14 23:35:12 -04:00
|
|
|
let data =
|
2012-08-26 21:28:36 -04:00
|
|
|
RWARCInner { lock: rwlock_with_condvars(num_condvars),
|
2012-09-18 22:35:42 -07:00
|
|
|
failed: false, data: move user_data };
|
2012-09-10 17:50:48 -07:00
|
|
|
RWARC { x: unsafe { shared_mutable_state(move data) }, cant_nest: () }
|
2012-08-10 20:46:19 -04:00
|
|
|
}
|
|
|
|
|
2012-12-11 13:50:04 -08:00
|
|
|
impl<T: Const Owned> RWARC<T> {
|
2012-08-10 20:46:19 -04:00
|
|
|
/// Duplicate a rwlock-protected ARC, as arc::clone.
|
2012-11-26 16:12:47 -08:00
|
|
|
fn clone(&self) -> RWARC<T> {
|
2012-08-26 21:28:36 -04:00
|
|
|
RWARC { x: unsafe { clone_shared_mutable_state(&self.x) },
|
|
|
|
cant_nest: () }
|
2012-08-10 20:46:19 -04:00
|
|
|
}
|
|
|
|
|
2012-11-26 16:12:47 -08:00
|
|
|
}
|
|
|
|
|
2012-12-11 13:50:04 -08:00
|
|
|
impl<T: Const Owned> &RWARC<T> {
|
2012-08-10 20:46:19 -04:00
|
|
|
/**
|
|
|
|
* 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
|
2012-08-26 21:28:36 -04:00
|
|
|
* that other tasks won't block forever. As MutexARC.access, it will also
|
2012-08-10 20:46:19 -04:00
|
|
|
* poison the ARC, so subsequent readers and writers will both also fail.
|
|
|
|
*/
|
|
|
|
#[inline(always)]
|
|
|
|
fn write<U>(blk: fn(x: &mut T) -> U) -> U {
|
|
|
|
let state = unsafe { get_shared_mutable_state(&self.x) };
|
|
|
|
do borrow_rwlock(state).write {
|
|
|
|
check_poison(false, state.failed);
|
2012-08-26 21:28:36 -04:00
|
|
|
let _z = PoisonOnFail(&mut state.failed);
|
2012-08-13 19:45:17 -04:00
|
|
|
blk(&mut state.data)
|
2012-08-10 20:46:19 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
/// As write(), but with a condvar, as sync::rwlock.write_cond().
|
|
|
|
#[inline(always)]
|
2012-08-26 21:28:36 -04:00
|
|
|
fn write_cond<U>(blk: fn(x: &x/mut T, c: &c/Condvar) -> U) -> U {
|
2012-08-10 20:46:19 -04:00
|
|
|
let state = unsafe { get_shared_mutable_state(&self.x) };
|
|
|
|
do borrow_rwlock(state).write_cond |cond| {
|
|
|
|
check_poison(false, state.failed);
|
2012-08-26 21:28:36 -04:00
|
|
|
let _z = PoisonOnFail(&mut state.failed);
|
2012-08-13 19:45:17 -04:00
|
|
|
blk(&mut state.data,
|
2012-08-26 21:28:36 -04:00
|
|
|
&Condvar { is_mutex: false, failed: &mut state.failed,
|
2012-08-13 19:45:17 -04:00
|
|
|
cond: cond })
|
2012-08-10 20:46:19 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
fn read<U>(blk: fn(x: &T) -> U) -> U {
|
|
|
|
let state = unsafe { get_shared_immutable_state(&self.x) };
|
|
|
|
do (&state.lock).read {
|
|
|
|
check_poison(false, state.failed);
|
|
|
|
blk(&state.data)
|
|
|
|
}
|
|
|
|
}
|
2012-08-14 13:32:41 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* As write(), but with the ability to atomically 'downgrade' the lock.
|
2012-08-26 21:28:36 -04:00
|
|
|
* 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.
|
2012-08-14 13:32:41 -04:00
|
|
|
* ~~~
|
|
|
|
* do arc.write_downgrade |write_mode| {
|
|
|
|
* do (&write_mode).write_cond |state, condvar| {
|
|
|
|
* ... exclusive access with mutable state ...
|
|
|
|
* }
|
|
|
|
* let read_mode = arc.downgrade(write_mode);
|
|
|
|
* do (&read_mode).read |state| {
|
|
|
|
* ... shared access with immutable state ...
|
|
|
|
* }
|
|
|
|
* }
|
|
|
|
* ~~~
|
|
|
|
*/
|
2012-10-03 12:21:48 -07:00
|
|
|
fn write_downgrade<U>(blk: fn(v: RWWriteMode<T>) -> U) -> U {
|
2012-08-14 13:32:41 -04:00
|
|
|
let state = unsafe { get_shared_mutable_state(&self.x) };
|
|
|
|
do borrow_rwlock(state).write_downgrade |write_mode| {
|
|
|
|
check_poison(false, state.failed);
|
2012-09-10 17:50:48 -07:00
|
|
|
blk(RWWriteMode((&mut state.data, move write_mode,
|
2012-08-26 21:28:36 -04:00
|
|
|
PoisonOnFail(&mut state.failed))))
|
2012-08-14 13:32:41 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// To be called inside of the write_downgrade block.
|
2012-10-03 12:21:48 -07:00
|
|
|
fn downgrade(token: RWWriteMode/&a<T>) -> RWReadMode/&a<T> {
|
2012-08-14 13:32:41 -04:00
|
|
|
// The rwlock should assert that the token belongs to us for us.
|
|
|
|
let state = unsafe { get_shared_immutable_state(&self.x) };
|
2012-10-23 11:11:23 -07:00
|
|
|
let RWWriteMode((data, t, _poison)) = move token;
|
2012-08-14 13:32:41 -04:00
|
|
|
// Let readers in
|
2012-09-10 17:50:48 -07:00
|
|
|
let new_token = (&state.lock).downgrade(move t);
|
2012-08-14 13:32:41 -04:00
|
|
|
// 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.)
|
2012-09-18 17:34:08 -07:00
|
|
|
let new_data = unsafe { cast::transmute_immut(data) };
|
2012-08-14 13:32:41 -04:00
|
|
|
// Downgrade ensured the token belonged to us. Just a sanity check.
|
|
|
|
assert ptr::ref_eq(&state.data, new_data);
|
|
|
|
// Produce new token
|
2012-09-10 17:50:48 -07:00
|
|
|
RWReadMode((new_data, move new_token))
|
2012-08-14 13:32:41 -04:00
|
|
|
}
|
2012-08-10 20:46:19 -04:00
|
|
|
}
|
|
|
|
|
2012-08-20 22:30:53 -04:00
|
|
|
/**
|
|
|
|
* Retrieves the data, blocking until all other references are dropped,
|
|
|
|
* exactly as arc::unwrap.
|
|
|
|
*
|
|
|
|
* Will additionally fail if another task has failed while accessing the arc
|
|
|
|
* in write mode.
|
|
|
|
*/
|
2012-10-11 15:37:37 -07:00
|
|
|
// FIXME(#3724) make this a by-move method on the arc
|
2012-12-11 13:50:04 -08:00
|
|
|
pub fn unwrap_rw_arc<T: Const Owned>(arc: RWARC<T>) -> T {
|
2012-10-23 11:11:23 -07:00
|
|
|
let RWARC { x: x, _ } = move arc;
|
2012-09-10 17:50:48 -07:00
|
|
|
let inner = unsafe { unwrap_shared_mutable_state(move x) };
|
2012-10-23 11:11:23 -07:00
|
|
|
let RWARCInner { failed: failed, data: data, _ } = move inner;
|
2012-08-20 17:58:59 -04:00
|
|
|
if failed {
|
2012-08-26 21:28:36 -04:00
|
|
|
fail ~"Can't unwrap poisoned RWARC - another task failed inside!"
|
2012-08-20 17:58:59 -04:00
|
|
|
}
|
2012-09-10 17:50:48 -07:00
|
|
|
move data
|
2012-08-20 17:58:59 -04:00
|
|
|
}
|
|
|
|
|
2012-08-10 20:46:19 -04:00
|
|
|
// 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'.
|
2012-08-15 13:55:20 -04:00
|
|
|
#[doc(hidden)]
|
2012-12-11 13:50:04 -08:00
|
|
|
fn borrow_rwlock<T: Const Owned>(state: &r/mut RWARCInner<T>) -> &r/RWlock {
|
2013-01-23 18:15:06 -08:00
|
|
|
unsafe { cast::transmute(&mut state.lock) }
|
2012-08-10 20:46:19 -04:00
|
|
|
}
|
|
|
|
|
2012-08-14 13:32:41 -04:00
|
|
|
// FIXME (#3154) ice with struct/&<T> prevents these from being structs.
|
|
|
|
|
2012-08-26 21:28:36 -04:00
|
|
|
/// The "write permission" token used for RWARC.write_downgrade().
|
2013-01-28 10:46:43 -08:00
|
|
|
pub enum RWWriteMode<T> =
|
2012-08-26 21:28:36 -04:00
|
|
|
(&mut T, sync::RWlockWriteMode, PoisonOnFail);
|
|
|
|
/// The "read permission" token used for RWARC.write_downgrade().
|
2013-01-28 10:46:43 -08:00
|
|
|
pub enum RWReadMode<T> = (&T, sync::RWlockReadMode);
|
2012-08-14 13:32:41 -04:00
|
|
|
|
2012-12-11 13:50:04 -08:00
|
|
|
impl<T: Const Owned> &RWWriteMode<T> {
|
2012-08-26 21:28:36 -04:00
|
|
|
/// Access the pre-downgrade RWARC in write mode.
|
2012-08-14 13:32:41 -04:00
|
|
|
fn write<U>(blk: fn(x: &mut T) -> U) -> U {
|
|
|
|
match *self {
|
2013-01-11 21:01:42 -08:00
|
|
|
RWWriteMode((ref data, ref token, _)) => {
|
2012-08-23 16:51:53 -04:00
|
|
|
do token.write {
|
2013-01-11 21:01:42 -08:00
|
|
|
blk(&mut **data)
|
2012-08-14 13:32:41 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-08-26 21:28:36 -04:00
|
|
|
/// Access the pre-downgrade RWARC in write mode with a condvar.
|
|
|
|
fn write_cond<U>(blk: fn(x: &x/mut T, c: &c/Condvar) -> U) -> U {
|
2012-08-14 13:32:41 -04:00
|
|
|
match *self {
|
2013-01-11 21:01:42 -08:00
|
|
|
RWWriteMode((ref data, ref token, ref poison)) => {
|
2012-08-23 16:51:53 -04:00
|
|
|
do token.write_cond |cond| {
|
2013-01-23 18:15:06 -08:00
|
|
|
unsafe {
|
|
|
|
let cvar = Condvar {
|
|
|
|
is_mutex: false,
|
|
|
|
failed: &mut *poison.failed,
|
|
|
|
cond: cond
|
|
|
|
};
|
|
|
|
blk(&mut **data, &cvar)
|
|
|
|
}
|
2012-08-14 13:32:41 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-12-11 13:50:04 -08:00
|
|
|
impl<T: Const Owned> &RWReadMode<T> {
|
2012-08-14 13:32:41 -04:00
|
|
|
/// Access the post-downgrade rwlock in read mode.
|
|
|
|
fn read<U>(blk: fn(x: &T) -> U) -> U {
|
|
|
|
match *self {
|
2012-08-26 21:28:36 -04:00
|
|
|
RWReadMode((data, ref token)) => {
|
2012-08-23 16:51:53 -04:00
|
|
|
do token.read { blk(data) }
|
2012-08-14 13:32:41 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-08-10 18:20:03 -04:00
|
|
|
/****************************************************************************
|
|
|
|
* Tests
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
2013-01-08 19:37:25 -08:00
|
|
|
use core::prelude::*;
|
|
|
|
|
|
|
|
use arc::*;
|
2012-12-28 12:46:08 -08:00
|
|
|
use arc;
|
|
|
|
|
2012-12-27 18:24:18 -08:00
|
|
|
use core::option::{Some, None};
|
|
|
|
use core::option;
|
|
|
|
use core::pipes;
|
|
|
|
use core::task;
|
|
|
|
use core::vec;
|
2012-08-10 18:20:03 -04:00
|
|
|
|
|
|
|
#[test]
|
2013-01-29 12:06:09 -08:00
|
|
|
pub fn manually_share_arc() {
|
2012-08-10 18:20:03 -04:00
|
|
|
let v = ~[1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
|
2012-08-29 14:45:25 -07:00
|
|
|
let arc_v = arc::ARC(v);
|
2012-08-10 18:20:03 -04:00
|
|
|
|
2012-12-11 12:26:41 -08:00
|
|
|
let (p, c) = pipes::stream();
|
2012-08-10 18:20:03 -04:00
|
|
|
|
2012-09-18 22:35:42 -07:00
|
|
|
do task::spawn() |move c| {
|
2012-08-16 17:50:21 -07:00
|
|
|
let p = pipes::PortSet();
|
2012-08-14 14:17:27 -07:00
|
|
|
c.send(p.chan());
|
2012-08-10 18:20:03 -04:00
|
|
|
|
|
|
|
let arc_v = p.recv();
|
|
|
|
|
|
|
|
let v = *arc::get::<~[int]>(&arc_v);
|
|
|
|
assert v[3] == 4;
|
|
|
|
};
|
|
|
|
|
|
|
|
let c = p.recv();
|
|
|
|
c.send(arc::clone(&arc_v));
|
|
|
|
|
|
|
|
assert (*arc::get(&arc_v))[2] == 3;
|
|
|
|
|
|
|
|
log(info, arc_v);
|
|
|
|
}
|
2012-08-10 20:46:19 -04:00
|
|
|
|
2012-08-13 19:45:17 -04:00
|
|
|
#[test]
|
2013-01-29 12:06:09 -08:00
|
|
|
pub fn test_mutex_arc_condvar() {
|
2012-08-29 14:45:25 -07:00
|
|
|
let arc = ~MutexARC(false);
|
2012-08-13 19:45:17 -04:00
|
|
|
let arc2 = ~arc.clone();
|
2013-01-15 17:18:00 -08:00
|
|
|
let (p,c) = pipes::oneshot();
|
2012-09-18 22:35:42 -07:00
|
|
|
let (c,p) = (~mut Some(move c), ~mut Some(move p));
|
|
|
|
do task::spawn |move arc2, move p| {
|
2012-08-13 19:45:17 -04:00
|
|
|
// wait until parent gets in
|
|
|
|
pipes::recv_one(option::swap_unwrap(p));
|
|
|
|
do arc2.access_cond |state, cond| {
|
|
|
|
*state = true;
|
|
|
|
cond.signal();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
do arc.access_cond |state, cond| {
|
|
|
|
pipes::send_one(option::swap_unwrap(c), ());
|
|
|
|
assert !*state;
|
|
|
|
while !*state {
|
|
|
|
cond.wait();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#[test] #[should_fail] #[ignore(cfg(windows))]
|
2013-01-29 12:06:09 -08:00
|
|
|
pub fn test_arc_condvar_poison() {
|
2012-08-29 14:45:25 -07:00
|
|
|
let arc = ~MutexARC(1);
|
2012-08-13 19:45:17 -04:00
|
|
|
let arc2 = ~arc.clone();
|
2012-12-11 12:26:41 -08:00
|
|
|
let (p, c) = pipes::stream();
|
2012-08-13 19:45:17 -04:00
|
|
|
|
2012-09-18 22:35:42 -07:00
|
|
|
do task::spawn_unlinked |move arc2, move p| {
|
2012-08-13 19:45:17 -04:00
|
|
|
let _ = p.recv();
|
|
|
|
do arc2.access_cond |one, cond| {
|
|
|
|
cond.signal();
|
|
|
|
assert *one == 0; // Parent should fail when it wakes up.
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
do arc.access_cond |one, cond| {
|
|
|
|
c.send(());
|
|
|
|
while *one == 1 {
|
|
|
|
cond.wait();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-08-10 20:46:19 -04:00
|
|
|
#[test] #[should_fail] #[ignore(cfg(windows))]
|
2013-01-29 12:06:09 -08:00
|
|
|
pub fn test_mutex_arc_poison() {
|
2012-08-29 14:45:25 -07:00
|
|
|
let arc = ~MutexARC(1);
|
2012-08-10 20:46:19 -04:00
|
|
|
let arc2 = ~arc.clone();
|
2012-09-18 22:35:42 -07:00
|
|
|
do task::try |move arc2| {
|
2012-08-10 20:46:19 -04:00
|
|
|
do arc2.access |one| {
|
|
|
|
assert *one == 2;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
do arc.access |one| {
|
|
|
|
assert *one == 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#[test] #[should_fail] #[ignore(cfg(windows))]
|
2013-01-29 12:06:09 -08:00
|
|
|
pub fn test_mutex_arc_unwrap_poison() {
|
2012-08-29 14:45:25 -07:00
|
|
|
let arc = MutexARC(1);
|
2012-08-20 17:58:59 -04:00
|
|
|
let arc2 = ~(&arc).clone();
|
2012-12-11 12:26:41 -08:00
|
|
|
let (p, c) = pipes::stream();
|
2012-09-18 22:35:42 -07:00
|
|
|
do task::spawn |move c, move arc2| {
|
2012-08-20 17:58:59 -04:00
|
|
|
do arc2.access |one| {
|
|
|
|
c.send(());
|
|
|
|
assert *one == 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let _ = p.recv();
|
2012-09-18 22:35:42 -07:00
|
|
|
let one = unwrap_mutex_arc(move arc);
|
2012-08-20 17:58:59 -04:00
|
|
|
assert one == 1;
|
|
|
|
}
|
|
|
|
#[test] #[should_fail] #[ignore(cfg(windows))]
|
2013-01-29 12:06:09 -08:00
|
|
|
pub fn test_rw_arc_poison_wr() {
|
2012-08-29 14:45:25 -07:00
|
|
|
let arc = ~RWARC(1);
|
2012-08-10 20:46:19 -04:00
|
|
|
let arc2 = ~arc.clone();
|
2012-09-18 22:35:42 -07:00
|
|
|
do task::try |move arc2| {
|
2012-08-10 20:46:19 -04:00
|
|
|
do arc2.write |one| {
|
|
|
|
assert *one == 2;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
do arc.read |one| {
|
|
|
|
assert *one == 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#[test] #[should_fail] #[ignore(cfg(windows))]
|
2013-01-29 12:06:09 -08:00
|
|
|
pub fn test_rw_arc_poison_ww() {
|
2012-08-29 14:45:25 -07:00
|
|
|
let arc = ~RWARC(1);
|
2012-08-10 20:46:19 -04:00
|
|
|
let arc2 = ~arc.clone();
|
2012-09-18 22:35:42 -07:00
|
|
|
do task::try |move arc2| {
|
2012-08-10 20:46:19 -04:00
|
|
|
do arc2.write |one| {
|
|
|
|
assert *one == 2;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
do arc.write |one| {
|
|
|
|
assert *one == 1;
|
|
|
|
}
|
|
|
|
}
|
2012-08-14 13:32:41 -04:00
|
|
|
#[test] #[should_fail] #[ignore(cfg(windows))]
|
2013-01-29 12:06:09 -08:00
|
|
|
pub fn test_rw_arc_poison_dw() {
|
2012-08-29 14:45:25 -07:00
|
|
|
let arc = ~RWARC(1);
|
2012-08-14 13:32:41 -04:00
|
|
|
let arc2 = ~arc.clone();
|
2012-09-18 22:35:42 -07:00
|
|
|
do task::try |move arc2| {
|
2012-08-14 13:32:41 -04:00
|
|
|
do arc2.write_downgrade |write_mode| {
|
2012-08-23 16:51:53 -04:00
|
|
|
do (&write_mode).write |one| {
|
2012-08-14 13:32:41 -04:00
|
|
|
assert *one == 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
do arc.write |one| {
|
|
|
|
assert *one == 1;
|
|
|
|
}
|
|
|
|
}
|
2012-08-10 20:46:19 -04:00
|
|
|
#[test] #[ignore(cfg(windows))]
|
2013-01-29 12:06:09 -08:00
|
|
|
pub fn test_rw_arc_no_poison_rr() {
|
2012-08-29 14:45:25 -07:00
|
|
|
let arc = ~RWARC(1);
|
2012-08-10 20:46:19 -04:00
|
|
|
let arc2 = ~arc.clone();
|
2012-09-18 22:35:42 -07:00
|
|
|
do task::try |move arc2| {
|
2012-08-10 20:46:19 -04:00
|
|
|
do arc2.read |one| {
|
|
|
|
assert *one == 2;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
do arc.read |one| {
|
|
|
|
assert *one == 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#[test] #[ignore(cfg(windows))]
|
2013-01-29 12:06:09 -08:00
|
|
|
pub fn test_rw_arc_no_poison_rw() {
|
2012-08-29 14:45:25 -07:00
|
|
|
let arc = ~RWARC(1);
|
2012-08-10 20:46:19 -04:00
|
|
|
let arc2 = ~arc.clone();
|
2012-09-18 22:35:42 -07:00
|
|
|
do task::try |move arc2| {
|
2012-08-10 20:46:19 -04:00
|
|
|
do arc2.read |one| {
|
|
|
|
assert *one == 2;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
do arc.write |one| {
|
|
|
|
assert *one == 1;
|
|
|
|
}
|
|
|
|
}
|
2012-08-14 13:32:41 -04:00
|
|
|
#[test] #[ignore(cfg(windows))]
|
2013-01-29 12:06:09 -08:00
|
|
|
pub fn test_rw_arc_no_poison_dr() {
|
2012-08-29 14:45:25 -07:00
|
|
|
let arc = ~RWARC(1);
|
2012-08-14 13:32:41 -04:00
|
|
|
let arc2 = ~arc.clone();
|
2012-09-18 22:35:42 -07:00
|
|
|
do task::try |move arc2| {
|
2012-08-14 13:32:41 -04:00
|
|
|
do arc2.write_downgrade |write_mode| {
|
2012-09-18 22:35:42 -07:00
|
|
|
let read_mode = arc2.downgrade(move write_mode);
|
2012-08-23 16:51:53 -04:00
|
|
|
do (&read_mode).read |one| {
|
2012-08-14 13:32:41 -04:00
|
|
|
assert *one == 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
do arc.write |one| {
|
|
|
|
assert *one == 1;
|
|
|
|
}
|
|
|
|
}
|
2012-08-10 20:46:19 -04:00
|
|
|
#[test]
|
2013-01-29 12:06:09 -08:00
|
|
|
pub fn test_rw_arc() {
|
2012-08-29 14:45:25 -07:00
|
|
|
let arc = ~RWARC(0);
|
2012-08-10 20:46:19 -04:00
|
|
|
let arc2 = ~arc.clone();
|
2012-12-11 12:26:41 -08:00
|
|
|
let (p,c) = pipes::stream();
|
2012-08-10 20:46:19 -04:00
|
|
|
|
2012-09-18 22:35:42 -07:00
|
|
|
do task::spawn |move arc2, move c| {
|
2012-08-10 20:46:19 -04:00
|
|
|
do arc2.write |num| {
|
|
|
|
for 10.times {
|
|
|
|
let tmp = *num;
|
|
|
|
*num = -1;
|
|
|
|
task::yield();
|
|
|
|
*num = tmp + 1;
|
|
|
|
}
|
|
|
|
c.send(());
|
|
|
|
}
|
|
|
|
}
|
2012-09-18 21:41:37 -07:00
|
|
|
|
2012-08-10 20:46:19 -04:00
|
|
|
// Readers try to catch the writer in the act
|
|
|
|
let mut children = ~[];
|
|
|
|
for 5.times {
|
|
|
|
let arc3 = ~arc.clone();
|
2012-09-18 22:35:42 -07:00
|
|
|
do task::task().future_result(|+r| children.push(move r)).spawn
|
|
|
|
|move arc3| {
|
2012-08-10 20:46:19 -04:00
|
|
|
do arc3.read |num| {
|
|
|
|
assert *num >= 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-09-18 21:41:37 -07:00
|
|
|
|
2012-08-10 20:46:19 -04:00
|
|
|
// Wait for children to pass their asserts
|
2012-10-22 19:01:37 -07:00
|
|
|
for vec::each(children) |r| { r.recv(); }
|
2012-09-18 21:41:37 -07:00
|
|
|
|
2012-08-10 20:46:19 -04:00
|
|
|
// Wait for writer to finish
|
|
|
|
p.recv();
|
|
|
|
do arc.read |num| { assert *num == 10; }
|
|
|
|
}
|
2012-08-14 13:32:41 -04:00
|
|
|
#[test]
|
2013-01-29 12:06:09 -08:00
|
|
|
pub fn test_rw_downgrade() {
|
2012-08-14 13:32:41 -04:00
|
|
|
// (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.
|
2012-08-29 14:45:25 -07:00
|
|
|
let arc = ~RWARC(0);
|
2012-08-14 13:32:41 -04:00
|
|
|
|
|
|
|
// Reader tasks
|
|
|
|
let mut reader_convos = ~[];
|
|
|
|
for 10.times {
|
2012-12-11 12:26:41 -08:00
|
|
|
let ((rp1,rc1),(rp2,rc2)) = (pipes::stream(),pipes::stream());
|
2012-09-18 22:35:42 -07:00
|
|
|
reader_convos.push((move rc1, move rp2));
|
2012-08-14 13:32:41 -04:00
|
|
|
let arcn = ~arc.clone();
|
2012-09-18 22:35:42 -07:00
|
|
|
do task::spawn |move rp1, move rc2, move arcn| {
|
2012-08-14 13:32:41 -04:00
|
|
|
rp1.recv(); // wait for downgrader to give go-ahead
|
|
|
|
do arcn.read |state| {
|
|
|
|
assert *state == 31337;
|
|
|
|
rc2.send(());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Writer task
|
|
|
|
let arc2 = ~arc.clone();
|
2012-12-11 12:26:41 -08:00
|
|
|
let ((wp1,wc1),(wp2,wc2)) = (pipes::stream(),pipes::stream());
|
2012-09-18 22:35:42 -07:00
|
|
|
do task::spawn |move arc2, move wc2, move wp1| {
|
2012-08-14 13:32:41 -04:00
|
|
|
wp1.recv();
|
|
|
|
do arc2.write_cond |state, cond| {
|
|
|
|
assert *state == 0;
|
|
|
|
*state = 42;
|
|
|
|
cond.signal();
|
|
|
|
}
|
|
|
|
wp1.recv();
|
|
|
|
do arc2.write |state| {
|
|
|
|
// This shouldn't happen until after the downgrade read
|
|
|
|
// section, and all other readers, finish.
|
|
|
|
assert *state == 31337;
|
|
|
|
*state = 42;
|
|
|
|
}
|
|
|
|
wc2.send(());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Downgrader (us)
|
|
|
|
do arc.write_downgrade |write_mode| {
|
2012-08-23 16:51:53 -04:00
|
|
|
do (&write_mode).write_cond |state, cond| {
|
2012-08-14 13:32:41 -04:00
|
|
|
wc1.send(()); // send to another writer who will wake us up
|
|
|
|
while *state == 0 {
|
|
|
|
cond.wait();
|
|
|
|
}
|
|
|
|
assert *state == 42;
|
|
|
|
*state = 31337;
|
|
|
|
// send to other readers
|
2012-09-18 21:41:13 -07:00
|
|
|
for vec::each(reader_convos) |x| {
|
2012-09-18 21:41:37 -07:00
|
|
|
match *x {
|
2012-09-28 02:26:20 -07:00
|
|
|
(ref rc, _) => rc.send(()),
|
2012-08-14 13:32:41 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-09-18 22:35:42 -07:00
|
|
|
let read_mode = arc.downgrade(move write_mode);
|
2012-08-23 16:51:53 -04:00
|
|
|
do (&read_mode).read |state| {
|
2012-08-14 13:32:41 -04:00
|
|
|
// complete handshake with other readers
|
2012-09-18 21:41:13 -07:00
|
|
|
for vec::each(reader_convos) |x| {
|
2012-09-18 21:41:37 -07:00
|
|
|
match *x {
|
2012-09-28 02:26:20 -07:00
|
|
|
(_, ref rp) => rp.recv(),
|
2012-08-14 13:32:41 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
wc1.send(()); // tell writer to try again
|
|
|
|
assert *state == 31337;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
wp2.recv(); // complete handshake with writer
|
|
|
|
}
|
2012-08-10 18:20:03 -04:00
|
|
|
}
|