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-04-04 21:17:50 -07:00
|
|
|
#[doc(hidden)];
|
|
|
|
|
2012-12-23 17:41:37 -05:00
|
|
|
use cast;
|
2012-12-27 17:53:04 -08:00
|
|
|
use iter;
|
2012-12-23 17:41:37 -05:00
|
|
|
use libc;
|
|
|
|
use option;
|
2013-02-11 12:11:23 +10:00
|
|
|
use pipes::{GenericChan, GenericPort};
|
2013-01-08 19:37:25 -08:00
|
|
|
use prelude::*;
|
2012-12-23 17:41:37 -05:00
|
|
|
use ptr;
|
2012-12-27 17:53:04 -08:00
|
|
|
use result;
|
2012-12-23 17:41:37 -05:00
|
|
|
use task;
|
|
|
|
use task::{TaskBuilder, atomically};
|
2012-12-27 17:53:04 -08:00
|
|
|
use uint;
|
2012-04-06 17:03:00 -07:00
|
|
|
|
2013-01-11 15:55:14 -08:00
|
|
|
#[path = "private/at_exit.rs"]
|
|
|
|
pub mod at_exit;
|
2013-01-12 23:27:46 -08:00
|
|
|
#[path = "private/global.rs"]
|
|
|
|
pub mod global;
|
2013-01-15 17:11:16 -08:00
|
|
|
#[path = "private/finally.rs"]
|
|
|
|
pub mod finally;
|
2013-01-15 19:53:35 -08:00
|
|
|
#[path = "private/weak_task.rs"]
|
|
|
|
pub mod weak_task;
|
2013-01-13 16:53:13 -08:00
|
|
|
#[path = "private/exchange_alloc.rs"]
|
|
|
|
pub mod exchange_alloc;
|
2013-02-20 17:57:15 +01:00
|
|
|
#[path = "private/intrinsics.rs"]
|
|
|
|
pub mod intrinsics;
|
2013-02-20 21:08:25 +01:00
|
|
|
#[path = "private/extfmt.rs"]
|
|
|
|
pub mod extfmt;
|
2013-01-11 15:55:14 -08:00
|
|
|
|
2012-07-03 16:11:00 -07:00
|
|
|
extern mod rustrt {
|
2013-01-29 11:47:18 -08:00
|
|
|
pub unsafe fn rust_create_little_lock() -> rust_little_lock;
|
|
|
|
pub unsafe fn rust_destroy_little_lock(lock: rust_little_lock);
|
|
|
|
pub unsafe fn rust_lock_little_lock(lock: rust_little_lock);
|
|
|
|
pub unsafe fn rust_unlock_little_lock(lock: rust_little_lock);
|
2012-09-18 14:51:40 -07:00
|
|
|
|
2013-01-29 11:47:18 -08:00
|
|
|
pub unsafe fn rust_raw_thread_start(f: &fn()) -> *raw_thread;
|
|
|
|
pub unsafe fn rust_raw_thread_join_delete(thread: *raw_thread);
|
2012-04-04 21:17:50 -07:00
|
|
|
}
|
|
|
|
|
2013-01-10 19:03:13 -08:00
|
|
|
#[allow(non_camel_case_types)] // runtime type
|
|
|
|
type raw_thread = libc::c_void;
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
2013-01-11 17:59:35 -08:00
|
|
|
Start a new thread outside of the current runtime context and wait
|
|
|
|
for it to terminate.
|
|
|
|
|
|
|
|
The executing thread has no access to a task pointer and will be using
|
|
|
|
a normal large stack.
|
2013-01-10 19:03:13 -08:00
|
|
|
*/
|
|
|
|
pub unsafe fn run_in_bare_thread(f: ~fn()) {
|
|
|
|
let (port, chan) = pipes::stream();
|
2013-01-18 14:21:31 -08:00
|
|
|
// FIXME #4525: Unfortunate that this creates an extra scheduler but it's
|
|
|
|
// necessary since rust_raw_thread_join_delete is blocking
|
2013-01-23 11:43:58 -08:00
|
|
|
do task::spawn_sched(task::SingleThreaded) {
|
|
|
|
unsafe {
|
|
|
|
let closure: &fn() = || {
|
|
|
|
f()
|
|
|
|
};
|
|
|
|
let thread = rustrt::rust_raw_thread_start(closure);
|
|
|
|
rustrt::rust_raw_thread_join_delete(thread);
|
|
|
|
chan.send(());
|
|
|
|
}
|
2013-01-10 19:03:13 -08:00
|
|
|
}
|
|
|
|
port.recv();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2013-01-23 11:43:58 -08:00
|
|
|
fn test_run_in_bare_thread() {
|
|
|
|
unsafe {
|
|
|
|
let i = 100;
|
|
|
|
do run_in_bare_thread {
|
|
|
|
assert i == 100;
|
|
|
|
}
|
2013-01-10 19:03:13 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-13 16:53:13 -08:00
|
|
|
#[test]
|
|
|
|
fn test_run_in_bare_thread_exchange() {
|
|
|
|
unsafe {
|
|
|
|
// Does the exchange heap work without the runtime?
|
|
|
|
let i = ~100;
|
|
|
|
do run_in_bare_thread {
|
|
|
|
assert i == ~100;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-22 21:47:14 -04:00
|
|
|
fn compare_and_swap(address: &mut int, oldval: int, newval: int) -> bool {
|
2013-01-23 16:29:31 -08:00
|
|
|
unsafe {
|
2013-02-20 17:57:15 +01:00
|
|
|
let old = intrinsics::atomic_cxchg(address, oldval, newval);
|
2013-01-23 16:29:31 -08:00
|
|
|
old == oldval
|
|
|
|
}
|
2012-10-22 21:47:14 -04:00
|
|
|
}
|
|
|
|
|
2012-09-18 14:51:40 -07:00
|
|
|
/****************************************************************************
|
|
|
|
* Shared state & exclusive ARC
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
// 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.
|
2012-11-03 13:38:16 -07:00
|
|
|
type UnwrapProto = ~mut Option<(pipes::ChanOne<()>, pipes::PortOne<bool>)>;
|
2012-09-18 14:51:40 -07:00
|
|
|
|
|
|
|
struct ArcData<T> {
|
|
|
|
mut count: libc::intptr_t,
|
2012-11-03 13:38:16 -07:00
|
|
|
mut unwrapper: int, // either a UnwrapProto or 0
|
2012-09-18 14:51:40 -07:00
|
|
|
// FIXME(#3224) should be able to make this non-option to save memory, and
|
|
|
|
// in unwrap() use "let ~ArcData { data: result, _ } = thing" to unwrap it
|
|
|
|
mut data: Option<T>,
|
|
|
|
}
|
|
|
|
|
|
|
|
struct ArcDestruct<T> {
|
|
|
|
mut data: *libc::c_void,
|
2013-01-23 11:43:58 -08:00
|
|
|
drop {
|
|
|
|
unsafe {
|
|
|
|
if self.data.is_null() {
|
|
|
|
return; // Happens when destructing an unwrapper's handle.
|
|
|
|
}
|
|
|
|
do task::unkillable {
|
|
|
|
let data: ~ArcData<T> = cast::reinterpret_cast(&self.data);
|
2013-02-20 20:41:24 +01:00
|
|
|
let new_count =
|
|
|
|
intrinsics::atomic_xsub(&mut data.count, 1) - 1;
|
2013-01-23 11:43:58 -08:00
|
|
|
assert new_count >= 0;
|
|
|
|
if new_count == 0 {
|
|
|
|
// 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.
|
|
|
|
if data.unwrapper != 0 {
|
|
|
|
let p: UnwrapProto =
|
|
|
|
cast::reinterpret_cast(&data.unwrapper);
|
|
|
|
let (message, response) = option::swap_unwrap(p);
|
|
|
|
// Send 'ready' and wait for a response.
|
2013-02-15 03:51:28 -05:00
|
|
|
pipes::send_one(message, ());
|
2013-01-23 11:43:58 -08:00
|
|
|
// Unkillable wait. Message guaranteed to come.
|
2013-02-15 03:51:28 -05:00
|
|
|
if pipes::recv_one(response) {
|
2013-01-23 11:43:58 -08:00
|
|
|
// Other task got the data.
|
2013-02-15 03:51:28 -05:00
|
|
|
cast::forget(data);
|
2013-01-23 11:43:58 -08:00
|
|
|
} else {
|
|
|
|
// Other task was killed. drop glue takes over.
|
|
|
|
}
|
2012-09-18 14:51:40 -07:00
|
|
|
} else {
|
2013-01-23 11:43:58 -08:00
|
|
|
// drop glue takes over.
|
2012-09-18 14:51:40 -07:00
|
|
|
}
|
|
|
|
} else {
|
2013-02-15 03:51:28 -05:00
|
|
|
cast::forget(data);
|
2012-09-18 14:51:40 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn ArcDestruct<T>(data: *libc::c_void) -> ArcDestruct<T> {
|
|
|
|
ArcDestruct {
|
|
|
|
data: data
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-20 17:07:17 -08:00
|
|
|
pub unsafe fn unwrap_shared_mutable_state<T:Owned>(rc: SharedMutableState<T>)
|
2012-09-18 14:51:40 -07:00
|
|
|
-> T {
|
|
|
|
struct DeathThroes<T> {
|
|
|
|
mut ptr: Option<~ArcData<T>>,
|
|
|
|
mut response: Option<pipes::ChanOne<bool>>,
|
2013-01-23 11:43:58 -08:00
|
|
|
drop {
|
|
|
|
unsafe {
|
|
|
|
let response = option::swap_unwrap(&mut self.response);
|
|
|
|
// In case we get killed early, we need to tell the person who
|
|
|
|
// tried to wake us whether they should hand-off the data to
|
|
|
|
// us.
|
|
|
|
if task::failing() {
|
2013-02-15 03:51:28 -05:00
|
|
|
pipes::send_one(response, false);
|
2013-01-23 11:43:58 -08:00
|
|
|
// Either this swap_unwrap or the one below (at "Got
|
|
|
|
// here") ought to run.
|
|
|
|
cast::forget(option::swap_unwrap(&mut self.ptr));
|
|
|
|
} else {
|
|
|
|
assert self.ptr.is_none();
|
2013-02-15 03:51:28 -05:00
|
|
|
pipes::send_one(response, true);
|
2013-01-23 11:43:58 -08:00
|
|
|
}
|
2012-09-18 14:51:40 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
do task::unkillable {
|
2012-09-18 17:34:08 -07:00
|
|
|
let ptr: ~ArcData<T> = cast::reinterpret_cast(&rc.data);
|
2013-01-15 17:18:00 -08:00
|
|
|
let (p1,c1) = pipes::oneshot(); // ()
|
|
|
|
let (p2,c2) = pipes::oneshot(); // bool
|
2013-02-15 03:51:28 -05:00
|
|
|
let server: UnwrapProto = ~mut Some((c1,p2));
|
|
|
|
let serverp: int = cast::transmute(server);
|
2012-09-18 14:51:40 -07:00
|
|
|
// Try to put our server end in the unwrapper slot.
|
2012-11-03 13:38:16 -07:00
|
|
|
if compare_and_swap(&mut ptr.unwrapper, 0, serverp) {
|
2012-09-18 14:51:40 -07:00
|
|
|
// Got in. Step 0: Tell destructor not to run. We are now it.
|
|
|
|
rc.data = ptr::null();
|
|
|
|
// Step 1 - drop our own reference.
|
2013-02-20 17:57:15 +01:00
|
|
|
let new_count = intrinsics::atomic_xsub(&mut ptr.count, 1) - 1;
|
2012-10-21 22:24:56 -04:00
|
|
|
//assert new_count >= 0;
|
|
|
|
if new_count == 0 {
|
|
|
|
// We were the last owner. Can unwrap immediately.
|
|
|
|
// Also we have to free the server endpoints.
|
2013-02-15 03:51:28 -05:00
|
|
|
let _server: UnwrapProto = cast::transmute(serverp);
|
2012-10-21 22:24:56 -04:00
|
|
|
option::swap_unwrap(&mut ptr.data)
|
2012-09-18 14:51:40 -07:00
|
|
|
// drop glue takes over.
|
|
|
|
} else {
|
|
|
|
// The *next* person who sees the refcount hit 0 will wake us.
|
|
|
|
let end_result =
|
2013-02-15 03:51:28 -05:00
|
|
|
DeathThroes { ptr: Some(ptr),
|
|
|
|
response: Some(c2) };
|
|
|
|
let mut p1 = Some(p1); // argh
|
2012-09-18 14:51:40 -07:00
|
|
|
do task::rekillable {
|
|
|
|
pipes::recv_one(option::swap_unwrap(&mut p1));
|
|
|
|
}
|
|
|
|
// Got here. Back in the 'unkillable' without getting killed.
|
|
|
|
// Recover ownership of ptr, then take the data out.
|
|
|
|
let ptr = option::swap_unwrap(&mut end_result.ptr);
|
|
|
|
option::swap_unwrap(&mut ptr.data)
|
|
|
|
// drop glue takes over.
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Somebody else was trying to unwrap. Avoid guaranteed deadlock.
|
2013-02-15 03:51:28 -05:00
|
|
|
cast::forget(ptr);
|
2012-09-18 14:51:40 -07:00
|
|
|
// Also we have to free the (rejected) server endpoints.
|
2013-02-15 03:51:28 -05:00
|
|
|
let _server: UnwrapProto = cast::transmute(serverp);
|
2013-02-11 19:26:38 -08:00
|
|
|
fail!(~"Another task is already unwrapping this ARC!");
|
2012-09-18 14:51:40 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* COMPLETELY UNSAFE. Used as a primitive for the safe versions in std::arc.
|
|
|
|
*
|
|
|
|
* Data races between tasks can result in crashes and, with sufficient
|
|
|
|
* cleverness, arbitrary type coercion.
|
|
|
|
*/
|
2013-01-28 10:46:43 -08:00
|
|
|
pub type SharedMutableState<T> = ArcDestruct<T>;
|
2012-09-18 14:51:40 -07:00
|
|
|
|
2013-02-20 17:07:17 -08:00
|
|
|
pub unsafe fn shared_mutable_state<T:Owned>(data: T) ->
|
2012-09-26 17:20:14 -07:00
|
|
|
SharedMutableState<T> {
|
2013-02-15 03:51:28 -05:00
|
|
|
let data = ~ArcData { count: 1, unwrapper: 0, data: Some(data) };
|
2012-09-18 14:51:40 -07:00
|
|
|
unsafe {
|
2013-02-15 03:51:28 -05:00
|
|
|
let ptr = cast::transmute(data);
|
2012-09-18 14:51:40 -07:00
|
|
|
ArcDestruct(ptr)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline(always)]
|
2013-02-20 17:07:17 -08:00
|
|
|
pub unsafe fn get_shared_mutable_state<T:Owned>(
|
2013-02-08 22:21:45 -08:00
|
|
|
rc: *SharedMutableState<T>) -> *mut T
|
|
|
|
{
|
2012-09-18 14:51:40 -07:00
|
|
|
unsafe {
|
2012-09-18 17:34:08 -07:00
|
|
|
let ptr: ~ArcData<T> = cast::reinterpret_cast(&(*rc).data);
|
2012-09-18 14:51:40 -07:00
|
|
|
assert ptr.count > 0;
|
2013-02-08 22:21:45 -08:00
|
|
|
let r = cast::transmute(option::get_ref(&ptr.data));
|
2013-02-15 03:51:28 -05:00
|
|
|
cast::forget(ptr);
|
2013-02-08 22:21:45 -08:00
|
|
|
return r;
|
2012-09-18 14:51:40 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#[inline(always)]
|
2013-02-20 17:07:17 -08:00
|
|
|
pub unsafe fn get_shared_immutable_state<T:Owned>(
|
2012-09-26 17:20:14 -07:00
|
|
|
rc: &a/SharedMutableState<T>) -> &a/T {
|
2012-09-18 14:51:40 -07:00
|
|
|
unsafe {
|
2012-09-18 17:34:08 -07:00
|
|
|
let ptr: ~ArcData<T> = cast::reinterpret_cast(&(*rc).data);
|
2012-09-18 14:51:40 -07:00
|
|
|
assert ptr.count > 0;
|
|
|
|
// Cast us back into the correct region
|
2012-09-18 17:34:08 -07:00
|
|
|
let r = cast::transmute_region(option::get_ref(&ptr.data));
|
2013-02-15 03:51:28 -05:00
|
|
|
cast::forget(ptr);
|
2012-09-18 14:51:40 -07:00
|
|
|
return r;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-20 17:07:17 -08:00
|
|
|
pub unsafe fn clone_shared_mutable_state<T:Owned>(rc: &SharedMutableState<T>)
|
2012-09-18 14:51:40 -07:00
|
|
|
-> SharedMutableState<T> {
|
|
|
|
unsafe {
|
2012-09-18 17:34:08 -07:00
|
|
|
let ptr: ~ArcData<T> = cast::reinterpret_cast(&(*rc).data);
|
2013-02-20 17:57:15 +01:00
|
|
|
let new_count = intrinsics::atomic_xadd(&mut ptr.count, 1) + 1;
|
2012-09-18 14:51:40 -07:00
|
|
|
assert new_count >= 2;
|
2013-02-15 03:51:28 -05:00
|
|
|
cast::forget(ptr);
|
2012-09-18 14:51:40 -07:00
|
|
|
}
|
|
|
|
ArcDestruct((*rc).data)
|
|
|
|
}
|
|
|
|
|
2013-02-20 17:07:17 -08:00
|
|
|
impl<T:Owned> Clone for SharedMutableState<T> {
|
2013-01-25 17:51:53 -08:00
|
|
|
fn clone(&self) -> SharedMutableState<T> {
|
|
|
|
unsafe {
|
|
|
|
clone_shared_mutable_state(self)
|
|
|
|
}
|
2013-01-12 23:27:46 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-09-18 14:51:40 -07:00
|
|
|
/****************************************************************************/
|
|
|
|
|
|
|
|
#[allow(non_camel_case_types)] // runtime type
|
|
|
|
type rust_little_lock = *libc::c_void;
|
|
|
|
|
|
|
|
struct LittleLock {
|
|
|
|
l: rust_little_lock,
|
2013-01-10 21:23:07 -08:00
|
|
|
drop {
|
|
|
|
unsafe {
|
|
|
|
rustrt::rust_destroy_little_lock(self.l);
|
|
|
|
}
|
|
|
|
}
|
2012-09-18 14:51:40 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
fn LittleLock() -> LittleLock {
|
2013-01-10 21:23:07 -08:00
|
|
|
unsafe {
|
|
|
|
LittleLock {
|
|
|
|
l: rustrt::rust_create_little_lock()
|
|
|
|
}
|
2012-09-18 14:51:40 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl LittleLock {
|
|
|
|
#[inline(always)]
|
|
|
|
unsafe fn lock<T>(f: fn() -> T) -> T {
|
|
|
|
struct Unlock {
|
|
|
|
l: rust_little_lock,
|
2013-01-10 21:23:07 -08:00
|
|
|
drop {
|
|
|
|
unsafe {
|
|
|
|
rustrt::rust_unlock_little_lock(self.l);
|
|
|
|
}
|
|
|
|
}
|
2012-09-18 14:51:40 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
fn Unlock(l: rust_little_lock) -> Unlock {
|
|
|
|
Unlock {
|
|
|
|
l: l
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
do atomically {
|
|
|
|
rustrt::rust_lock_little_lock(self.l);
|
|
|
|
let _r = Unlock(self.l);
|
|
|
|
f()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-28 10:46:43 -08:00
|
|
|
struct ExData<T> { lock: LittleLock, mut failed: bool, mut data: T, }
|
2012-09-18 14:51:40 -07:00
|
|
|
/**
|
|
|
|
* An arc over mutable data that is protected by a lock. For library use only.
|
|
|
|
*/
|
2013-01-28 10:46:43 -08:00
|
|
|
pub struct Exclusive<T> { x: SharedMutableState<ExData<T>> }
|
2012-09-18 14:51:40 -07:00
|
|
|
|
2013-02-20 17:07:17 -08:00
|
|
|
pub fn exclusive<T:Owned>(user_data: T) -> Exclusive<T> {
|
2012-09-18 14:51:40 -07:00
|
|
|
let data = ExData {
|
2013-02-15 03:51:28 -05:00
|
|
|
lock: LittleLock(), mut failed: false, mut data: user_data
|
2012-09-18 14:51:40 -07:00
|
|
|
};
|
2013-02-15 03:51:28 -05:00
|
|
|
Exclusive { x: unsafe { shared_mutable_state(data) } }
|
2012-09-18 14:51:40 -07:00
|
|
|
}
|
|
|
|
|
2013-02-20 17:07:17 -08:00
|
|
|
impl<T:Owned> Clone for Exclusive<T> {
|
2012-09-18 14:51:40 -07:00
|
|
|
// Duplicate an exclusive ARC, as std::arc::clone.
|
2012-11-26 16:12:47 -08:00
|
|
|
fn clone(&self) -> Exclusive<T> {
|
2012-09-18 14:51:40 -07:00
|
|
|
Exclusive { x: unsafe { clone_shared_mutable_state(&self.x) } }
|
|
|
|
}
|
2012-11-26 16:12:47 -08:00
|
|
|
}
|
2012-09-18 14:51:40 -07:00
|
|
|
|
2013-02-20 17:07:17 -08:00
|
|
|
impl<T:Owned> Exclusive<T> {
|
2012-09-18 14:51:40 -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.
|
|
|
|
#[inline(always)]
|
|
|
|
unsafe fn with<U>(f: fn(x: &mut T) -> U) -> U {
|
2013-02-08 22:21:45 -08:00
|
|
|
unsafe {
|
|
|
|
let rec = get_shared_mutable_state(&self.x);
|
|
|
|
do (*rec).lock.lock {
|
|
|
|
if (*rec).failed {
|
2013-02-11 19:26:38 -08:00
|
|
|
fail!(
|
|
|
|
~"Poisoned exclusive - another task failed inside!");
|
2013-02-08 22:21:45 -08:00
|
|
|
}
|
|
|
|
(*rec).failed = true;
|
|
|
|
let result = f(&mut (*rec).data);
|
|
|
|
(*rec).failed = false;
|
2013-02-15 03:51:28 -05:00
|
|
|
result
|
2012-09-18 14:51:40 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline(always)]
|
|
|
|
unsafe fn with_imm<U>(f: fn(x: &T) -> U) -> U {
|
|
|
|
do self.with |x| {
|
2012-09-18 17:34:08 -07:00
|
|
|
f(cast::transmute_immut(x))
|
2012-09-18 14:51:40 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-11 14:52:41 -07:00
|
|
|
// FIXME(#3724) make this a by-move method on the exclusive
|
2013-02-20 17:07:17 -08:00
|
|
|
pub fn unwrap_exclusive<T:Owned>(arc: Exclusive<T>) -> T {
|
2013-02-15 03:51:28 -05:00
|
|
|
let Exclusive { x: x } = arc;
|
|
|
|
let inner = unsafe { unwrap_shared_mutable_state(x) };
|
|
|
|
let ExData { data: data, _ } = inner;
|
|
|
|
data
|
2012-09-18 14:51:40 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
2012-09-26 17:20:14 -07:00
|
|
|
pub mod tests {
|
2013-01-08 19:37:25 -08:00
|
|
|
use core::option::{None, Some};
|
|
|
|
|
2012-12-27 17:53:04 -08:00
|
|
|
use option;
|
|
|
|
use pipes;
|
2013-01-08 19:37:25 -08:00
|
|
|
use private::{exclusive, unwrap_exclusive};
|
2012-12-27 17:53:04 -08:00
|
|
|
use result;
|
|
|
|
use task;
|
|
|
|
use uint;
|
|
|
|
|
2012-09-18 14:51:40 -07:00
|
|
|
#[test]
|
2012-09-26 17:20:14 -07:00
|
|
|
pub fn exclusive_arc() {
|
2012-09-18 14:51:40 -07:00
|
|
|
let mut futures = ~[];
|
|
|
|
|
2012-09-18 22:35:28 -07:00
|
|
|
let num_tasks = 10;
|
|
|
|
let count = 10;
|
2012-09-18 14:51:40 -07:00
|
|
|
|
2012-09-18 22:35:28 -07:00
|
|
|
let total = exclusive(~mut 0);
|
2012-09-18 14:51:40 -07:00
|
|
|
|
2012-09-18 22:35:28 -07:00
|
|
|
for uint::range(0, num_tasks) |_i| {
|
2012-09-18 14:51:40 -07:00
|
|
|
let total = total.clone();
|
2012-12-11 12:26:41 -08:00
|
|
|
let (port, chan) = pipes::stream();
|
2013-02-15 03:51:28 -05:00
|
|
|
futures.push(port);
|
2012-10-22 19:16:52 -07:00
|
|
|
|
2013-02-15 03:51:28 -05:00
|
|
|
do task::spawn || {
|
2012-09-18 22:35:28 -07:00
|
|
|
for uint::range(0, count) |_i| {
|
2012-09-18 14:51:40 -07:00
|
|
|
do total.with |count| {
|
2012-09-18 22:35:28 -07:00
|
|
|
**count += 1;
|
2012-09-18 14:51:40 -07:00
|
|
|
}
|
|
|
|
}
|
2012-10-22 19:16:52 -07:00
|
|
|
chan.send(());
|
|
|
|
}
|
2012-09-18 14:51:40 -07:00
|
|
|
};
|
|
|
|
|
2012-10-22 19:16:52 -07:00
|
|
|
for futures.each |f| { f.recv() }
|
2012-09-18 14:51:40 -07:00
|
|
|
|
|
|
|
do total.with |total| {
|
|
|
|
assert **total == num_tasks * count
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test] #[should_fail] #[ignore(cfg(windows))]
|
2012-09-26 17:20:14 -07:00
|
|
|
pub fn exclusive_poison() {
|
2012-09-18 14:51:40 -07:00
|
|
|
// Tests that if one task fails inside of an exclusive, subsequent
|
|
|
|
// accesses will also fail.
|
|
|
|
let x = exclusive(1);
|
|
|
|
let x2 = x.clone();
|
2013-02-15 03:51:28 -05:00
|
|
|
do task::try || {
|
2012-09-18 14:51:40 -07:00
|
|
|
do x2.with |one| {
|
|
|
|
assert *one == 2;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
do x.with |one| {
|
|
|
|
assert *one == 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2012-09-26 17:20:14 -07:00
|
|
|
pub fn exclusive_unwrap_basic() {
|
2012-09-18 14:51:40 -07:00
|
|
|
let x = exclusive(~~"hello");
|
2013-02-15 03:51:28 -05:00
|
|
|
assert unwrap_exclusive(x) == ~~"hello";
|
2012-09-18 14:51:40 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2012-09-26 17:20:14 -07:00
|
|
|
pub fn exclusive_unwrap_contended() {
|
2012-09-18 14:51:40 -07:00
|
|
|
let x = exclusive(~~"hello");
|
|
|
|
let x2 = ~mut Some(x.clone());
|
2013-02-15 03:51:28 -05:00
|
|
|
do task::spawn || {
|
2012-09-18 14:51:40 -07:00
|
|
|
let x2 = option::swap_unwrap(x2);
|
|
|
|
do x2.with |_hello| { }
|
|
|
|
task::yield();
|
|
|
|
}
|
2013-02-15 03:51:28 -05:00
|
|
|
assert unwrap_exclusive(x) == ~~"hello";
|
2012-09-18 14:51:40 -07:00
|
|
|
|
|
|
|
// Now try the same thing, but with the child task blocking.
|
|
|
|
let x = exclusive(~~"hello");
|
|
|
|
let x2 = ~mut Some(x.clone());
|
|
|
|
let mut res = None;
|
2013-02-15 03:51:28 -05:00
|
|
|
do task::task().future_result(|+r| res = Some(r)).spawn
|
|
|
|
|| {
|
2012-09-18 14:51:40 -07:00
|
|
|
let x2 = option::swap_unwrap(x2);
|
2013-02-15 03:51:28 -05:00
|
|
|
assert unwrap_exclusive(x2) == ~~"hello";
|
2012-09-18 14:51:40 -07:00
|
|
|
}
|
|
|
|
// Have to get rid of our reference before blocking.
|
2013-02-15 03:51:28 -05:00
|
|
|
{ let _x = x; } // FIXME(#3161) util::ignore doesn't work here
|
2012-09-18 14:51:40 -07:00
|
|
|
let res = option::swap_unwrap(&mut res);
|
2012-10-22 19:01:37 -07:00
|
|
|
res.recv();
|
2012-09-18 14:51:40 -07:00
|
|
|
}
|
|
|
|
|
2013-02-04 14:19:19 -08:00
|
|
|
#[test] #[should_fail] #[ignore(reason = "random red")]
|
2012-09-26 17:20:14 -07:00
|
|
|
pub fn exclusive_unwrap_conflict() {
|
2012-09-18 14:51:40 -07:00
|
|
|
let x = exclusive(~~"hello");
|
|
|
|
let x2 = ~mut Some(x.clone());
|
|
|
|
let mut res = None;
|
2013-02-15 03:51:28 -05:00
|
|
|
do task::task().future_result(|+r| res = Some(r)).spawn
|
|
|
|
|| {
|
2012-09-18 14:51:40 -07:00
|
|
|
let x2 = option::swap_unwrap(x2);
|
2013-02-15 03:51:28 -05:00
|
|
|
assert unwrap_exclusive(x2) == ~~"hello";
|
2012-09-18 14:51:40 -07:00
|
|
|
}
|
2013-02-15 03:51:28 -05:00
|
|
|
assert unwrap_exclusive(x) == ~~"hello";
|
2012-09-18 14:51:40 -07:00
|
|
|
let res = option::swap_unwrap(&mut res);
|
2012-10-22 19:01:37 -07:00
|
|
|
res.recv();
|
2012-09-18 14:51:40 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test] #[ignore(cfg(windows))]
|
2012-09-26 17:20:14 -07:00
|
|
|
pub fn exclusive_unwrap_deadlock() {
|
2012-09-18 14:51:40 -07:00
|
|
|
// 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
|
2013-02-11 19:26:38 -08:00
|
|
|
fail!(); // punt it awake from its deadlock
|
2012-09-18 14:51:40 -07:00
|
|
|
}
|
2013-02-15 03:51:28 -05:00
|
|
|
let _z = unwrap_exclusive(x);
|
2012-09-18 14:51:40 -07:00
|
|
|
do x2.with |_hello| { }
|
|
|
|
};
|
|
|
|
assert result.is_err();
|
|
|
|
}
|
|
|
|
}
|