2013-05-09 17:36:45 -07:00
|
|
|
// Copyright 2013 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.
|
|
|
|
|
|
|
|
use cast;
|
2013-07-02 13:37:19 -04:00
|
|
|
use cell::Cell;
|
|
|
|
use comm;
|
2013-05-09 17:36:45 -07:00
|
|
|
use libc;
|
2013-07-02 13:37:19 -04:00
|
|
|
use ptr;
|
2013-05-09 17:36:45 -07:00
|
|
|
use option::*;
|
|
|
|
use task;
|
|
|
|
use task::atomically;
|
2013-07-02 13:37:19 -04:00
|
|
|
use unstable::atomics::{AtomicOption,AtomicUint,Acquire,Release,SeqCst};
|
2013-05-09 17:36:45 -07:00
|
|
|
use unstable::finally::Finally;
|
|
|
|
use ops::Drop;
|
|
|
|
use clone::Clone;
|
2013-06-05 17:56:24 -07:00
|
|
|
use kinds::Send;
|
2013-05-09 17:36:45 -07:00
|
|
|
|
2013-05-09 20:27:42 -07:00
|
|
|
/// An atomically reference counted pointer.
|
|
|
|
///
|
|
|
|
/// Enforces no shared-memory safety.
|
|
|
|
pub struct UnsafeAtomicRcBox<T> {
|
|
|
|
data: *mut libc::c_void,
|
2013-05-09 17:36:45 -07:00
|
|
|
}
|
|
|
|
|
2013-05-09 20:27:42 -07:00
|
|
|
struct AtomicRcBoxData<T> {
|
2013-07-02 13:37:19 -04:00
|
|
|
count: AtomicUint,
|
|
|
|
// An unwrapper uses this protocol to communicate with the "other" task that
|
|
|
|
// drops the last refcount on an arc. Unfortunately this can't be a proper
|
|
|
|
// pipe protocol because the unwrapper has to access both stages at once.
|
|
|
|
// FIXME(#7544): Maybe use AtomicPtr instead (to avoid xchg in take() later)?
|
|
|
|
unwrapper: AtomicOption<(comm::ChanOne<()>, comm::PortOne<bool>)>,
|
|
|
|
// FIXME(#3224) should be able to make this non-option to save memory
|
2013-05-09 20:27:42 -07:00
|
|
|
data: Option<T>,
|
2013-05-09 17:36:45 -07:00
|
|
|
}
|
|
|
|
|
2013-06-05 17:56:24 -07:00
|
|
|
impl<T: Send> UnsafeAtomicRcBox<T> {
|
2013-05-09 20:27:42 -07:00
|
|
|
pub fn new(data: T) -> UnsafeAtomicRcBox<T> {
|
2013-05-09 17:36:45 -07:00
|
|
|
unsafe {
|
2013-07-02 13:37:19 -04:00
|
|
|
let data = ~AtomicRcBoxData { count: AtomicUint::new(1),
|
|
|
|
unwrapper: AtomicOption::empty(),
|
|
|
|
data: Some(data) };
|
2013-05-09 20:27:42 -07:00
|
|
|
let ptr = cast::transmute(data);
|
|
|
|
return UnsafeAtomicRcBox { data: ptr };
|
2013-05-09 17:36:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-18 14:45:18 -07:00
|
|
|
#[inline]
|
2013-05-09 20:27:42 -07:00
|
|
|
pub unsafe fn get(&self) -> *mut T
|
|
|
|
{
|
|
|
|
let mut data: ~AtomicRcBoxData<T> = cast::transmute(self.data);
|
2013-07-02 13:37:19 -04:00
|
|
|
assert!(data.count.load(Acquire) > 0); // no barrier is really needed
|
2013-05-09 20:27:42 -07:00
|
|
|
let r: *mut T = data.data.get_mut_ref();
|
|
|
|
cast::forget(data);
|
|
|
|
return r;
|
|
|
|
}
|
2013-05-09 17:36:45 -07:00
|
|
|
|
2013-06-18 14:45:18 -07:00
|
|
|
#[inline]
|
2013-05-09 20:27:42 -07:00
|
|
|
pub unsafe fn get_immut(&self) -> *T
|
|
|
|
{
|
2013-07-02 13:37:19 -04:00
|
|
|
let data: ~AtomicRcBoxData<T> = cast::transmute(self.data);
|
|
|
|
assert!(data.count.load(Acquire) > 0); // no barrier is really needed
|
|
|
|
let r: *T = data.data.get_ref();
|
2013-05-09 20:27:42 -07:00
|
|
|
cast::forget(data);
|
|
|
|
return r;
|
|
|
|
}
|
2013-07-02 13:37:19 -04:00
|
|
|
|
|
|
|
/// Wait until all other handles are dropped, then retrieve the enclosed
|
|
|
|
/// data. See extra::arc::ARC for specific semantics documentation.
|
|
|
|
/// If called when the task is already unkillable, unwrap will unkillably
|
|
|
|
/// block; otherwise, an unwrapping task can be killed by linked failure.
|
|
|
|
pub unsafe fn unwrap(self) -> T {
|
|
|
|
let this = Cell::new(self); // argh
|
|
|
|
do task::unkillable {
|
|
|
|
let mut this = this.take();
|
|
|
|
let mut data: ~AtomicRcBoxData<T> = cast::transmute(this.data);
|
|
|
|
// Set up the unwrap protocol.
|
|
|
|
let (p1,c1) = comm::oneshot(); // ()
|
|
|
|
let (p2,c2) = comm::oneshot(); // bool
|
|
|
|
// Try to put our server end in the unwrapper slot.
|
|
|
|
// This needs no barrier -- it's protected by the release barrier on
|
|
|
|
// the xadd, and the acquire+release barrier in the destructor's xadd.
|
|
|
|
// FIXME(#6598) Change Acquire to Relaxed.
|
|
|
|
if data.unwrapper.fill(~(c1,p2), Acquire).is_none() {
|
|
|
|
// Got in. Tell this handle's destructor not to run (we are now it).
|
|
|
|
this.data = ptr::mut_null();
|
|
|
|
// Drop our own reference.
|
|
|
|
let old_count = data.count.fetch_sub(1, Release);
|
|
|
|
assert!(old_count >= 1);
|
|
|
|
if old_count == 1 {
|
|
|
|
// We were the last owner. Can unwrap immediately.
|
|
|
|
// AtomicOption's destructor will free the server endpoint.
|
|
|
|
// FIXME(#3224): it should be like this
|
|
|
|
// let ~AtomicRcBoxData { data: user_data, _ } = data;
|
|
|
|
// user_data
|
|
|
|
data.data.take_unwrap()
|
|
|
|
} else {
|
|
|
|
// The *next* person who sees the refcount hit 0 will wake us.
|
|
|
|
let p1 = Cell::new(p1); // argh
|
|
|
|
// Unlike the above one, this cell is necessary. It will get
|
|
|
|
// taken either in the do block or in the finally block.
|
|
|
|
let c2_and_data = Cell::new((c2,data));
|
|
|
|
do (|| {
|
|
|
|
do task::rekillable { p1.take().recv(); }
|
|
|
|
// Got here. Back in the 'unkillable' without getting killed.
|
|
|
|
let (c2, data) = c2_and_data.take();
|
|
|
|
c2.send(true);
|
|
|
|
// FIXME(#3224): it should be like this
|
|
|
|
// let ~AtomicRcBoxData { data: user_data, _ } = data;
|
|
|
|
// user_data
|
|
|
|
let mut data = data;
|
|
|
|
data.data.take_unwrap()
|
|
|
|
}).finally {
|
|
|
|
if task::failing() {
|
|
|
|
// Killed during wait. Because this might happen while
|
|
|
|
// someone else still holds a reference, we can't free
|
|
|
|
// the data now; the "other" last refcount will free it.
|
|
|
|
let (c2, data) = c2_and_data.take();
|
|
|
|
c2.send(false);
|
|
|
|
cast::forget(data);
|
|
|
|
} else {
|
|
|
|
assert!(c2_and_data.is_empty());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// If 'put' returns the server end back to us, we were rejected;
|
|
|
|
// someone else was trying to unwrap. Avoid guaranteed deadlock.
|
|
|
|
cast::forget(data);
|
|
|
|
fail!("Another task is already unwrapping this ARC!");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-05-09 17:36:45 -07:00
|
|
|
}
|
|
|
|
|
2013-06-05 17:56:24 -07:00
|
|
|
impl<T: Send> Clone for UnsafeAtomicRcBox<T> {
|
2013-05-09 20:27:42 -07:00
|
|
|
fn clone(&self) -> UnsafeAtomicRcBox<T> {
|
|
|
|
unsafe {
|
|
|
|
let mut data: ~AtomicRcBoxData<T> = cast::transmute(self.data);
|
2013-07-02 13:37:19 -04:00
|
|
|
// This barrier might be unnecessary, but I'm not sure...
|
|
|
|
let old_count = data.count.fetch_add(1, Acquire);
|
|
|
|
assert!(old_count >= 1);
|
2013-05-09 20:27:42 -07:00
|
|
|
cast::forget(data);
|
|
|
|
return UnsafeAtomicRcBox { data: self.data };
|
|
|
|
}
|
|
|
|
}
|
2013-05-09 17:36:45 -07:00
|
|
|
}
|
|
|
|
|
2013-05-09 20:27:42 -07:00
|
|
|
#[unsafe_destructor]
|
|
|
|
impl<T> Drop for UnsafeAtomicRcBox<T>{
|
2013-06-20 21:06:13 -04:00
|
|
|
fn drop(&self) {
|
2013-05-09 17:36:45 -07:00
|
|
|
unsafe {
|
2013-07-02 13:37:19 -04:00
|
|
|
if self.data.is_null() {
|
|
|
|
return; // Happens when destructing an unwrapper's handle.
|
|
|
|
}
|
2013-05-09 20:27:42 -07:00
|
|
|
do task::unkillable {
|
|
|
|
let mut data: ~AtomicRcBoxData<T> = cast::transmute(self.data);
|
2013-07-02 13:37:19 -04:00
|
|
|
// Must be acquire+release, not just release, to make sure this
|
|
|
|
// doesn't get reordered to after the unwrapper pointer load.
|
|
|
|
let old_count = data.count.fetch_sub(1, SeqCst);
|
|
|
|
assert!(old_count >= 1);
|
|
|
|
if old_count == 1 {
|
|
|
|
// Were we really last, or should we hand off to an
|
|
|
|
// unwrapper? It's safe to not xchg because the unwrapper
|
|
|
|
// will set the unwrap lock *before* dropping his/her
|
|
|
|
// reference. In effect, being here means we're the only
|
|
|
|
// *awake* task with the data.
|
|
|
|
match data.unwrapper.take(Acquire) {
|
|
|
|
Some(~(message,response)) => {
|
|
|
|
// Send 'ready' and wait for a response.
|
|
|
|
message.send(());
|
|
|
|
// Unkillable wait. Message guaranteed to come.
|
|
|
|
if response.recv() {
|
|
|
|
// Other task got the data.
|
|
|
|
cast::forget(data);
|
|
|
|
} else {
|
|
|
|
// Other task was killed. drop glue takes over.
|
|
|
|
}
|
|
|
|
}
|
|
|
|
None => {
|
|
|
|
// drop glue takes over.
|
|
|
|
}
|
|
|
|
}
|
2013-05-09 20:27:42 -07:00
|
|
|
} else {
|
|
|
|
cast::forget(data);
|
|
|
|
}
|
|
|
|
}
|
2013-05-09 17:36:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-09 20:27:42 -07:00
|
|
|
|
2013-05-09 17:36:45 -07:00
|
|
|
/****************************************************************************/
|
|
|
|
|
|
|
|
#[allow(non_camel_case_types)] // runtime type
|
|
|
|
pub type rust_little_lock = *libc::c_void;
|
|
|
|
|
|
|
|
struct LittleLock {
|
|
|
|
l: rust_little_lock,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Drop for LittleLock {
|
2013-06-20 21:06:13 -04:00
|
|
|
fn drop(&self) {
|
2013-05-09 17:36:45 -07:00
|
|
|
unsafe {
|
|
|
|
rust_destroy_little_lock(self.l);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn LittleLock() -> LittleLock {
|
|
|
|
unsafe {
|
|
|
|
LittleLock {
|
|
|
|
l: rust_create_little_lock()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-31 15:17:22 -07:00
|
|
|
impl LittleLock {
|
2013-06-18 14:45:18 -07:00
|
|
|
#[inline]
|
2013-05-31 15:17:22 -07:00
|
|
|
pub unsafe fn lock<T>(&self, f: &fn() -> T) -> T {
|
2013-05-09 17:36:45 -07:00
|
|
|
do atomically {
|
|
|
|
rust_lock_little_lock(self.l);
|
|
|
|
do (|| {
|
|
|
|
f()
|
|
|
|
}).finally {
|
|
|
|
rust_unlock_little_lock(self.l);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct ExData<T> {
|
|
|
|
lock: LittleLock,
|
|
|
|
failed: bool,
|
|
|
|
data: T,
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* An arc over mutable data that is protected by a lock. For library use only.
|
2013-07-02 13:37:19 -04:00
|
|
|
*
|
|
|
|
* # Safety note
|
|
|
|
*
|
|
|
|
* This uses a pthread mutex, not one that's aware of the userspace scheduler.
|
|
|
|
* The user of an exclusive must be careful not to invoke any functions that may
|
|
|
|
* reschedule the task while holding the lock, or deadlock may result. If you
|
|
|
|
* need to block or yield while accessing shared state, use extra::sync::RWARC.
|
2013-05-09 17:36:45 -07:00
|
|
|
*/
|
|
|
|
pub struct Exclusive<T> {
|
2013-05-09 20:27:42 -07:00
|
|
|
x: UnsafeAtomicRcBox<ExData<T>>
|
2013-05-09 17:36:45 -07:00
|
|
|
}
|
|
|
|
|
2013-06-05 17:56:24 -07:00
|
|
|
pub fn exclusive<T:Send>(user_data: T) -> Exclusive<T> {
|
2013-05-09 17:36:45 -07:00
|
|
|
let data = ExData {
|
|
|
|
lock: LittleLock(),
|
|
|
|
failed: false,
|
|
|
|
data: user_data
|
|
|
|
};
|
|
|
|
Exclusive {
|
2013-05-09 20:27:42 -07:00
|
|
|
x: UnsafeAtomicRcBox::new(data)
|
2013-05-09 17:36:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-05 17:56:24 -07:00
|
|
|
impl<T:Send> Clone for Exclusive<T> {
|
2013-05-09 17:36:45 -07:00
|
|
|
// Duplicate an exclusive ARC, as std::arc::clone.
|
|
|
|
fn clone(&self) -> Exclusive<T> {
|
2013-05-09 20:27:42 -07:00
|
|
|
Exclusive { x: self.x.clone() }
|
2013-05-09 17:36:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-05 17:56:24 -07:00
|
|
|
impl<T:Send> Exclusive<T> {
|
2013-05-09 17:36:45 -07:00
|
|
|
// Exactly like std::arc::mutex_arc,access(), but with the little_lock
|
|
|
|
// instead of a proper mutex. Same reason for being unsafe.
|
|
|
|
//
|
|
|
|
// Currently, scheduling operations (i.e., yielding, receiving on a pipe,
|
|
|
|
// accessing the provided condition variable) are prohibited while inside
|
|
|
|
// the exclusive. Supporting that is a work in progress.
|
2013-06-18 14:45:18 -07:00
|
|
|
#[inline]
|
2013-05-31 15:17:22 -07:00
|
|
|
pub unsafe fn with<U>(&self, f: &fn(x: &mut T) -> U) -> U {
|
2013-05-09 20:27:42 -07:00
|
|
|
let rec = self.x.get();
|
2013-05-09 17:36:45 -07:00
|
|
|
do (*rec).lock.lock {
|
|
|
|
if (*rec).failed {
|
2013-05-06 00:18:51 +02:00
|
|
|
fail!("Poisoned exclusive - another task failed inside!");
|
2013-05-09 17:36:45 -07:00
|
|
|
}
|
|
|
|
(*rec).failed = true;
|
|
|
|
let result = f(&mut (*rec).data);
|
|
|
|
(*rec).failed = false;
|
|
|
|
result
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-18 14:45:18 -07:00
|
|
|
#[inline]
|
2013-05-31 15:17:22 -07:00
|
|
|
pub unsafe fn with_imm<U>(&self, f: &fn(x: &T) -> U) -> U {
|
2013-05-09 17:36:45 -07:00
|
|
|
do self.with |x| {
|
|
|
|
f(cast::transmute_immut(x))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-02 13:37:19 -04:00
|
|
|
pub fn unwrap(self) -> T {
|
|
|
|
let Exclusive { x: x } = self;
|
|
|
|
// Someday we might need to unkillably unwrap an exclusive, but not today.
|
|
|
|
let inner = unsafe { x.unwrap() };
|
|
|
|
let ExData { data: user_data, _ } = inner; // will destroy the LittleLock
|
|
|
|
user_data
|
2013-05-09 17:36:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extern {
|
|
|
|
fn rust_create_little_lock() -> rust_little_lock;
|
|
|
|
fn rust_destroy_little_lock(lock: rust_little_lock);
|
|
|
|
fn rust_lock_little_lock(lock: rust_little_lock);
|
|
|
|
fn rust_unlock_little_lock(lock: rust_little_lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
2013-05-21 17:36:59 -07:00
|
|
|
use super::*;
|
2013-07-02 13:37:19 -04:00
|
|
|
use cell::Cell;
|
2013-05-09 17:36:45 -07:00
|
|
|
use comm;
|
2013-07-02 13:37:19 -04:00
|
|
|
use option::*;
|
|
|
|
use super::{exclusive, UnsafeAtomicRcBox};
|
2013-05-09 17:36:45 -07:00
|
|
|
use task;
|
|
|
|
use uint;
|
2013-07-02 13:37:19 -04:00
|
|
|
use util;
|
2013-05-09 17:36:45 -07:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn exclusive_arc() {
|
2013-05-24 19:35:29 -07:00
|
|
|
unsafe {
|
|
|
|
let mut futures = ~[];
|
2013-05-09 17:36:45 -07:00
|
|
|
|
2013-05-24 19:35:29 -07:00
|
|
|
let num_tasks = 10;
|
|
|
|
let count = 10;
|
2013-05-09 17:36:45 -07:00
|
|
|
|
2013-05-24 19:35:29 -07:00
|
|
|
let total = exclusive(~0);
|
2013-05-09 17:36:45 -07:00
|
|
|
|
2013-05-24 19:35:29 -07:00
|
|
|
for uint::range(0, num_tasks) |_i| {
|
|
|
|
let total = total.clone();
|
|
|
|
let (port, chan) = comm::stream();
|
|
|
|
futures.push(port);
|
2013-05-09 17:36:45 -07:00
|
|
|
|
2013-05-24 19:35:29 -07:00
|
|
|
do task::spawn || {
|
|
|
|
for uint::range(0, count) |_i| {
|
|
|
|
do total.with |count| {
|
|
|
|
**count += 1;
|
|
|
|
}
|
2013-05-09 17:36:45 -07:00
|
|
|
}
|
2013-05-24 19:35:29 -07:00
|
|
|
chan.send(());
|
2013-05-09 17:36:45 -07:00
|
|
|
}
|
2013-05-24 19:35:29 -07:00
|
|
|
};
|
2013-05-09 17:36:45 -07:00
|
|
|
|
2013-06-21 08:29:53 -04:00
|
|
|
for futures.iter().advance |f| { f.recv() }
|
2013-05-09 17:36:45 -07:00
|
|
|
|
2013-05-24 19:35:29 -07:00
|
|
|
do total.with |total| {
|
|
|
|
assert!(**total == num_tasks * count)
|
|
|
|
};
|
|
|
|
}
|
2013-05-09 17:36:45 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test] #[should_fail] #[ignore(cfg(windows))]
|
|
|
|
fn exclusive_poison() {
|
2013-05-24 19:35:29 -07:00
|
|
|
unsafe {
|
|
|
|
// Tests that if one task fails inside of an exclusive, subsequent
|
|
|
|
// accesses will also fail.
|
|
|
|
let x = exclusive(1);
|
|
|
|
let x2 = x.clone();
|
|
|
|
do task::try || {
|
|
|
|
do x2.with |one| {
|
|
|
|
assert_eq!(*one, 2);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
do x.with |one| {
|
|
|
|
assert_eq!(*one, 1);
|
2013-05-09 17:36:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-07-02 13:37:19 -04:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn unsafe_unwrap_basic() {
|
|
|
|
unsafe {
|
|
|
|
let x = UnsafeAtomicRcBox::new(~~"hello");
|
|
|
|
assert!(x.unwrap() == ~~"hello");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn exclusive_unwrap_basic() {
|
|
|
|
// Unlike the above, also tests no double-freeing of the LittleLock.
|
|
|
|
let x = exclusive(~~"hello");
|
|
|
|
assert!(x.unwrap() == ~~"hello");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn exclusive_unwrap_contended() {
|
|
|
|
let x = exclusive(~~"hello");
|
|
|
|
let x2 = Cell::new(x.clone());
|
|
|
|
do task::spawn {
|
|
|
|
let x2 = x2.take();
|
|
|
|
unsafe { do x2.with |_hello| { } }
|
|
|
|
task::yield();
|
|
|
|
}
|
|
|
|
assert!(x.unwrap() == ~~"hello");
|
|
|
|
|
|
|
|
// Now try the same thing, but with the child task blocking.
|
|
|
|
let x = exclusive(~~"hello");
|
|
|
|
let x2 = Cell::new(x.clone());
|
|
|
|
let mut res = None;
|
|
|
|
let mut builder = task::task();
|
|
|
|
builder.future_result(|r| res = Some(r));
|
|
|
|
do builder.spawn {
|
|
|
|
let x2 = x2.take();
|
|
|
|
assert!(x2.unwrap() == ~~"hello");
|
|
|
|
}
|
|
|
|
// Have to get rid of our reference before blocking.
|
|
|
|
util::ignore(x);
|
|
|
|
res.unwrap().recv();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test] #[should_fail] #[ignore(cfg(windows))]
|
|
|
|
fn exclusive_unwrap_conflict() {
|
|
|
|
let x = exclusive(~~"hello");
|
|
|
|
let x2 = Cell::new(x.clone());
|
|
|
|
let mut res = None;
|
|
|
|
let mut builder = task::task();
|
|
|
|
builder.future_result(|r| res = Some(r));
|
|
|
|
do builder.spawn {
|
|
|
|
let x2 = x2.take();
|
|
|
|
assert!(x2.unwrap() == ~~"hello");
|
|
|
|
}
|
|
|
|
assert!(x.unwrap() == ~~"hello");
|
|
|
|
// See #4689 for why this can't be just "res.recv()".
|
|
|
|
assert!(res.unwrap().recv() == task::Success);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test] #[ignore(cfg(windows))]
|
|
|
|
fn exclusive_unwrap_deadlock() {
|
|
|
|
// This is not guaranteed to get to the deadlock before being killed,
|
|
|
|
// but it will show up sometimes, and if the deadlock were not there,
|
|
|
|
// the test would nondeterministically fail.
|
|
|
|
let result = do task::try {
|
|
|
|
// a task that has two references to the same exclusive will
|
|
|
|
// deadlock when it unwraps. nothing to be done about that.
|
|
|
|
let x = exclusive(~~"hello");
|
|
|
|
let x2 = x.clone();
|
|
|
|
do task::spawn {
|
|
|
|
for 10.times { task::yield(); } // try to let the unwrapper go
|
|
|
|
fail!(); // punt it awake from its deadlock
|
|
|
|
}
|
|
|
|
let _z = x.unwrap();
|
|
|
|
unsafe { do x2.with |_hello| { } }
|
|
|
|
};
|
|
|
|
assert!(result.is_err());
|
|
|
|
}
|
2013-05-09 17:36:45 -07:00
|
|
|
}
|