2013-04-21 18:28:17 -05: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.
|
|
|
|
|
|
|
|
//! Language-level runtime services that should reasonably expected
|
|
|
|
//! to be available 'everywhere'. Local heaps, GC, unwinding,
|
|
|
|
//! local storage, and logging. Even a 'freestanding' Rust would likely want
|
|
|
|
//! to implement this.
|
|
|
|
|
2013-06-02 18:16:40 -05:00
|
|
|
use borrow;
|
2013-04-22 19:15:31 -05:00
|
|
|
use cast::transmute;
|
2013-06-22 03:09:06 -05:00
|
|
|
use cleanup;
|
2013-05-24 21:35:29 -05:00
|
|
|
use libc::{c_void, uintptr_t};
|
|
|
|
use ptr;
|
|
|
|
use prelude::*;
|
2013-06-14 01:31:19 -05:00
|
|
|
use option::{Option, Some, None};
|
2013-07-01 22:24:24 -05:00
|
|
|
use rt::kill::Death;
|
2013-05-19 17:45:39 -05:00
|
|
|
use rt::local::Local;
|
2013-05-07 17:57:15 -05:00
|
|
|
use rt::logging::StdErrLogger;
|
2013-05-24 21:35:29 -05:00
|
|
|
use super::local_heap::LocalHeap;
|
2013-06-26 18:41:00 -05:00
|
|
|
use rt::sched::{Scheduler, SchedHandle};
|
|
|
|
use rt::stack::{StackSegment, StackPool};
|
|
|
|
use rt::context::Context;
|
2013-07-16 12:42:12 -05:00
|
|
|
use task::spawn::Taskgroup;
|
2013-06-26 18:41:00 -05:00
|
|
|
use cell::Cell;
|
2013-04-21 18:28:17 -05:00
|
|
|
|
2013-05-19 03:04:01 -05:00
|
|
|
pub struct Task {
|
2013-04-21 18:28:17 -05:00
|
|
|
heap: LocalHeap,
|
|
|
|
gc: GarbageCollector,
|
|
|
|
storage: LocalStorage,
|
2013-04-27 20:57:15 -05:00
|
|
|
logger: StdErrLogger,
|
2013-06-14 01:16:27 -05:00
|
|
|
unwinder: Unwinder,
|
2013-06-15 21:31:46 -05:00
|
|
|
home: Option<SchedHome>,
|
2013-07-16 12:42:12 -05:00
|
|
|
taskgroup: Option<Taskgroup>,
|
2013-07-01 22:24:24 -05:00
|
|
|
death: Death,
|
2013-06-26 18:41:00 -05:00
|
|
|
destroyed: bool,
|
|
|
|
coroutine: Option<~Coroutine>
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct Coroutine {
|
|
|
|
/// The segment of stack on which the task is currently running or
|
|
|
|
/// if the task is blocked, on which the task will resume
|
|
|
|
/// execution.
|
|
|
|
priv current_stack_segment: StackSegment,
|
|
|
|
/// Always valid if the task is alive and not running.
|
|
|
|
saved_context: Context
|
|
|
|
}
|
|
|
|
|
|
|
|
pub enum SchedHome {
|
|
|
|
AnySched,
|
|
|
|
Sched(SchedHandle)
|
2013-04-21 18:28:17 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
pub struct GarbageCollector;
|
2013-07-09 11:40:50 -05:00
|
|
|
pub struct LocalStorage(*c_void, Option<extern "Rust" fn(*c_void)>);
|
2013-04-22 19:15:31 -05:00
|
|
|
|
|
|
|
pub struct Unwinder {
|
|
|
|
unwinding: bool,
|
|
|
|
}
|
2013-04-21 18:28:17 -05:00
|
|
|
|
2013-05-19 03:04:01 -05:00
|
|
|
impl Task {
|
2013-06-26 18:41:00 -05:00
|
|
|
|
|
|
|
pub fn new_root(stack_pool: &mut StackPool,
|
|
|
|
start: ~fn()) -> Task {
|
|
|
|
Task::new_root_homed(stack_pool, AnySched, start)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn new_child(&mut self,
|
|
|
|
stack_pool: &mut StackPool,
|
|
|
|
start: ~fn()) -> Task {
|
|
|
|
self.new_child_homed(stack_pool, AnySched, start)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn new_root_homed(stack_pool: &mut StackPool,
|
|
|
|
home: SchedHome,
|
|
|
|
start: ~fn()) -> Task {
|
2013-05-19 03:04:01 -05:00
|
|
|
Task {
|
2013-04-21 21:03:52 -05:00
|
|
|
heap: LocalHeap::new(),
|
2013-04-21 18:28:17 -05:00
|
|
|
gc: GarbageCollector,
|
2013-04-22 14:54:03 -05:00
|
|
|
storage: LocalStorage(ptr::null(), None),
|
2013-04-27 20:57:15 -05:00
|
|
|
logger: StdErrLogger,
|
2013-06-14 01:16:27 -05:00
|
|
|
unwinder: Unwinder { unwinding: false },
|
2013-06-26 18:41:00 -05:00
|
|
|
home: Some(home),
|
2013-07-12 21:45:19 -05:00
|
|
|
taskgroup: None,
|
2013-07-01 22:24:24 -05:00
|
|
|
death: Death::new(),
|
2013-06-26 18:41:00 -05:00
|
|
|
destroyed: false,
|
|
|
|
coroutine: Some(~Coroutine::new(stack_pool, start))
|
2013-04-23 17:11:28 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-26 18:41:00 -05:00
|
|
|
pub fn new_child_homed(&mut self,
|
|
|
|
stack_pool: &mut StackPool,
|
|
|
|
home: SchedHome,
|
|
|
|
start: ~fn()) -> Task {
|
2013-05-19 03:04:01 -05:00
|
|
|
Task {
|
2013-04-23 17:11:28 -05:00
|
|
|
heap: LocalHeap::new(),
|
|
|
|
gc: GarbageCollector,
|
|
|
|
storage: LocalStorage(ptr::null(), None),
|
2013-04-27 20:57:15 -05:00
|
|
|
logger: StdErrLogger,
|
2013-06-26 18:41:00 -05:00
|
|
|
home: Some(home),
|
2013-06-14 01:16:27 -05:00
|
|
|
unwinder: Unwinder { unwinding: false },
|
2013-07-12 21:45:19 -05:00
|
|
|
taskgroup: None,
|
2013-07-01 22:24:24 -05:00
|
|
|
// FIXME(#7544) make watching optional
|
|
|
|
death: self.death.new_child(),
|
2013-06-26 18:41:00 -05:00
|
|
|
destroyed: false,
|
|
|
|
coroutine: Some(~Coroutine::new(stack_pool, start))
|
2013-04-21 18:28:17 -05:00
|
|
|
}
|
|
|
|
}
|
2013-04-22 14:54:03 -05:00
|
|
|
|
2013-06-14 14:17:56 -05:00
|
|
|
pub fn give_home(&mut self, new_home: SchedHome) {
|
|
|
|
self.home = Some(new_home);
|
|
|
|
}
|
|
|
|
|
2013-04-22 19:15:31 -05:00
|
|
|
pub fn run(&mut self, f: &fn()) {
|
|
|
|
// This is just an assertion that `run` was called unsafely
|
2013-05-19 03:04:01 -05:00
|
|
|
// and this instance of Task is still accessible.
|
2013-06-10 17:29:02 -05:00
|
|
|
do Local::borrow::<Task, ()> |task| {
|
2013-06-02 18:16:40 -05:00
|
|
|
assert!(borrow::ref_eq(task, self));
|
2013-04-22 19:15:31 -05:00
|
|
|
}
|
|
|
|
|
2013-06-14 01:16:27 -05:00
|
|
|
self.unwinder.try(f);
|
2013-07-12 21:45:19 -05:00
|
|
|
{ let _ = self.taskgroup.take(); }
|
2013-07-01 22:24:24 -05:00
|
|
|
self.death.collect_failure(!self.unwinder.unwinding);
|
2013-04-22 19:15:31 -05:00
|
|
|
self.destroy();
|
|
|
|
}
|
|
|
|
|
2013-06-14 01:31:19 -05:00
|
|
|
/// must be called manually before finalization to clean up
|
2013-04-22 14:54:03 -05:00
|
|
|
/// thread-local resources. Some of the routines here expect
|
2013-05-19 03:04:01 -05:00
|
|
|
/// Task to be available recursively so this must be
|
|
|
|
/// called unsafely, without removing Task from
|
2013-04-22 14:54:03 -05:00
|
|
|
/// thread-local-storage.
|
2013-04-22 19:15:31 -05:00
|
|
|
fn destroy(&mut self) {
|
2013-06-26 18:41:00 -05:00
|
|
|
|
2013-06-10 17:29:02 -05:00
|
|
|
do Local::borrow::<Task, ()> |task| {
|
2013-06-02 18:16:40 -05:00
|
|
|
assert!(borrow::ref_eq(task, self));
|
2013-04-22 14:54:03 -05:00
|
|
|
}
|
2013-06-26 18:41:00 -05:00
|
|
|
|
2013-04-22 14:54:03 -05:00
|
|
|
match self.storage {
|
2013-04-23 17:16:04 -05:00
|
|
|
LocalStorage(ptr, Some(ref dtor)) => {
|
|
|
|
(*dtor)(ptr)
|
|
|
|
}
|
2013-04-22 14:54:03 -05:00
|
|
|
_ => ()
|
|
|
|
}
|
2013-06-22 03:09:06 -05:00
|
|
|
|
|
|
|
// Destroy remaining boxes
|
|
|
|
unsafe { cleanup::annihilate(); }
|
|
|
|
|
2013-04-22 14:54:03 -05:00
|
|
|
self.destroyed = true;
|
|
|
|
}
|
2013-06-26 18:41:00 -05:00
|
|
|
|
|
|
|
/// Check if *task* is currently home.
|
|
|
|
pub fn is_home(&self) -> bool {
|
|
|
|
do Local::borrow::<Scheduler,bool> |sched| {
|
|
|
|
match self.home {
|
|
|
|
Some(AnySched) => { false }
|
|
|
|
Some(Sched(SchedHandle { sched_id: ref id, _ })) => {
|
|
|
|
*id == sched.sched_id()
|
|
|
|
}
|
|
|
|
None => { rtabort!("task home of None") }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_home_no_tls(&self, sched: &~Scheduler) -> bool {
|
|
|
|
match self.home {
|
|
|
|
Some(AnySched) => { false }
|
|
|
|
Some(Sched(SchedHandle { sched_id: ref id, _ })) => {
|
|
|
|
*id == sched.sched_id()
|
|
|
|
}
|
|
|
|
None => {rtabort!("task home of None") }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_home_using_id(sched_id: uint) -> bool {
|
|
|
|
do Local::borrow::<Task,bool> |task| {
|
|
|
|
match task.home {
|
|
|
|
Some(Sched(SchedHandle { sched_id: ref id, _ })) => {
|
|
|
|
*id == sched_id
|
|
|
|
}
|
|
|
|
Some(AnySched) => { false }
|
|
|
|
None => { rtabort!("task home of None") }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Check if this *task* has a home.
|
|
|
|
pub fn homed(&self) -> bool {
|
|
|
|
match self.home {
|
|
|
|
Some(AnySched) => { false }
|
|
|
|
Some(Sched(_)) => { true }
|
|
|
|
None => {
|
|
|
|
rtabort!("task home of None")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// On a special scheduler?
|
|
|
|
pub fn on_special() -> bool {
|
|
|
|
do Local::borrow::<Scheduler,bool> |sched| {
|
2013-07-02 13:44:51 -05:00
|
|
|
!sched.run_anything
|
2013-06-26 18:41:00 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-22 14:54:03 -05:00
|
|
|
}
|
|
|
|
|
2013-05-19 03:04:01 -05:00
|
|
|
impl Drop for Task {
|
2013-06-20 20:06:13 -05:00
|
|
|
fn drop(&self) { assert!(self.destroyed) }
|
2013-04-21 18:28:17 -05:00
|
|
|
}
|
|
|
|
|
2013-06-26 18:41:00 -05:00
|
|
|
// Coroutines represent nothing more than a context and a stack
|
|
|
|
// segment.
|
|
|
|
|
|
|
|
impl Coroutine {
|
|
|
|
|
|
|
|
pub fn new(stack_pool: &mut StackPool, start: ~fn()) -> Coroutine {
|
|
|
|
static MIN_STACK_SIZE: uint = 100000; // XXX: Too much stack
|
|
|
|
|
|
|
|
let start = Coroutine::build_start_wrapper(start);
|
|
|
|
let mut stack = stack_pool.take_segment(MIN_STACK_SIZE);
|
|
|
|
let initial_context = Context::new(start, &mut stack);
|
|
|
|
Coroutine {
|
|
|
|
current_stack_segment: stack,
|
|
|
|
saved_context: initial_context
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn build_start_wrapper(start: ~fn()) -> ~fn() {
|
|
|
|
let start_cell = Cell::new(start);
|
|
|
|
let wrapper: ~fn() = || {
|
|
|
|
// First code after swap to this new context. Run our
|
|
|
|
// cleanup job.
|
|
|
|
unsafe {
|
|
|
|
let sched = Local::unsafe_borrow::<Scheduler>();
|
|
|
|
(*sched).run_cleanup_job();
|
|
|
|
|
|
|
|
let sched = Local::unsafe_borrow::<Scheduler>();
|
|
|
|
let task = (*sched).current_task.get_mut_ref();
|
|
|
|
|
|
|
|
do task.run {
|
|
|
|
// N.B. Removing `start` from the start wrapper
|
|
|
|
// closure by emptying a cell is critical for
|
|
|
|
// correctness. The ~Task pointer, and in turn the
|
|
|
|
// closure used to initialize the first call
|
|
|
|
// frame, is destroyed in the scheduler context,
|
|
|
|
// not task context. So any captured closures must
|
|
|
|
// not contain user-definable dtors that expect to
|
|
|
|
// be in task context. By moving `start` out of
|
|
|
|
// the closure, all the user code goes our of
|
|
|
|
// scope while the task is still running.
|
|
|
|
let start = start_cell.take();
|
|
|
|
start();
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
let sched = Local::take::<Scheduler>();
|
|
|
|
sched.terminate_current_task();
|
|
|
|
};
|
|
|
|
return wrapper;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Destroy coroutine and try to reuse stack segment.
|
|
|
|
pub fn recycle(~self, stack_pool: &mut StackPool) {
|
|
|
|
match self {
|
|
|
|
~Coroutine { current_stack_segment, _ } => {
|
|
|
|
stack_pool.give_segment(current_stack_segment);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-04-22 19:15:31 -05:00
|
|
|
// Just a sanity check to make sure we are catching a Rust-thrown exception
|
|
|
|
static UNWIND_TOKEN: uintptr_t = 839147;
|
|
|
|
|
|
|
|
impl Unwinder {
|
|
|
|
pub fn try(&mut self, f: &fn()) {
|
|
|
|
use sys::Closure;
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
let closure: Closure = transmute(f);
|
|
|
|
let code = transmute(closure.code);
|
|
|
|
let env = transmute(closure.env);
|
|
|
|
|
|
|
|
let token = rust_try(try_fn, code, env);
|
|
|
|
assert!(token == 0 || token == UNWIND_TOKEN);
|
|
|
|
}
|
|
|
|
|
|
|
|
extern fn try_fn(code: *c_void, env: *c_void) {
|
|
|
|
unsafe {
|
|
|
|
let closure: Closure = Closure {
|
|
|
|
code: transmute(code),
|
|
|
|
env: transmute(env),
|
|
|
|
};
|
|
|
|
let closure: &fn() = transmute(closure);
|
|
|
|
closure();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extern {
|
|
|
|
#[rust_stack]
|
|
|
|
fn rust_try(f: *u8, code: *c_void, data: *c_void) -> uintptr_t;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn begin_unwind(&mut self) -> ! {
|
|
|
|
self.unwinding = true;
|
|
|
|
unsafe {
|
|
|
|
rust_begin_unwind(UNWIND_TOKEN);
|
|
|
|
return transmute(());
|
|
|
|
}
|
|
|
|
extern {
|
|
|
|
fn rust_begin_unwind(token: uintptr_t);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-21 21:03:52 -05:00
|
|
|
#[cfg(test)]
|
|
|
|
mod test {
|
|
|
|
use rt::test::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn local_heap() {
|
|
|
|
do run_in_newsched_task() {
|
|
|
|
let a = @5;
|
|
|
|
let b = a;
|
|
|
|
assert!(*a == 5);
|
|
|
|
assert!(*b == 5);
|
|
|
|
}
|
|
|
|
}
|
2013-04-22 14:54:03 -05:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn tls() {
|
2013-07-11 00:14:40 -05:00
|
|
|
use local_data;
|
2013-04-22 14:54:03 -05:00
|
|
|
do run_in_newsched_task() {
|
2013-07-14 03:43:31 -05:00
|
|
|
static key: local_data::Key<@~str> = &local_data::Key;
|
2013-07-12 03:38:44 -05:00
|
|
|
local_data::set(key, @~"data");
|
|
|
|
assert!(*local_data::get(key, |k| k.map(|&k| *k)).get() == ~"data");
|
2013-07-14 03:43:31 -05:00
|
|
|
static key2: local_data::Key<@~str> = &local_data::Key;
|
2013-07-12 03:38:44 -05:00
|
|
|
local_data::set(key2, @~"data");
|
|
|
|
assert!(*local_data::get(key2, |k| k.map(|&k| *k)).get() == ~"data");
|
2013-04-22 14:54:03 -05:00
|
|
|
}
|
|
|
|
}
|
2013-04-22 19:15:31 -05:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn unwind() {
|
|
|
|
do run_in_newsched_task() {
|
2013-04-23 17:11:28 -05:00
|
|
|
let result = spawntask_try(||());
|
2013-06-26 18:41:00 -05:00
|
|
|
rtdebug!("trying first assert");
|
2013-04-22 19:15:31 -05:00
|
|
|
assert!(result.is_ok());
|
2013-04-23 17:11:28 -05:00
|
|
|
let result = spawntask_try(|| fail!());
|
2013-06-26 18:41:00 -05:00
|
|
|
rtdebug!("trying second assert");
|
2013-04-22 19:15:31 -05:00
|
|
|
assert!(result.is_err());
|
|
|
|
}
|
|
|
|
}
|
2013-05-06 20:24:37 -05:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn rng() {
|
|
|
|
do run_in_newsched_task() {
|
|
|
|
use rand::{rng, Rng};
|
2013-05-08 14:26:34 -05:00
|
|
|
let mut r = rng();
|
2013-05-06 20:24:37 -05:00
|
|
|
let _ = r.next();
|
|
|
|
}
|
|
|
|
}
|
2013-04-27 20:57:15 -05:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn logging() {
|
|
|
|
do run_in_newsched_task() {
|
|
|
|
info!("here i am. logging in a newsched task");
|
|
|
|
}
|
|
|
|
}
|
2013-05-17 01:12:22 -05:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn comm_oneshot() {
|
|
|
|
use comm::*;
|
|
|
|
|
|
|
|
do run_in_newsched_task {
|
|
|
|
let (port, chan) = oneshot();
|
|
|
|
send_one(chan, 10);
|
|
|
|
assert!(recv_one(port) == 10);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn comm_stream() {
|
|
|
|
use comm::*;
|
|
|
|
|
|
|
|
do run_in_newsched_task() {
|
2013-05-17 19:47:10 -05:00
|
|
|
let (port, chan) = stream();
|
2013-05-17 01:12:22 -05:00
|
|
|
chan.send(10);
|
|
|
|
assert!(port.recv() == 10);
|
|
|
|
}
|
|
|
|
}
|
2013-06-14 01:31:19 -05:00
|
|
|
|
2013-06-20 20:26:56 -05:00
|
|
|
#[test]
|
|
|
|
fn comm_shared_chan() {
|
|
|
|
use comm::*;
|
|
|
|
|
|
|
|
do run_in_newsched_task() {
|
|
|
|
let (port, chan) = stream();
|
|
|
|
let chan = SharedChan::new(chan);
|
|
|
|
chan.send(10);
|
|
|
|
assert!(port.recv() == 10);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-14 01:31:19 -05:00
|
|
|
#[test]
|
|
|
|
fn linked_failure() {
|
|
|
|
do run_in_newsched_task() {
|
|
|
|
let res = do spawntask_try {
|
|
|
|
spawntask_random(|| fail!());
|
|
|
|
};
|
|
|
|
assert!(res.is_err());
|
|
|
|
}
|
|
|
|
}
|
2013-06-22 03:09:06 -05:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn heap_cycles() {
|
|
|
|
use option::{Option, Some, None};
|
|
|
|
|
|
|
|
do run_in_newsched_task {
|
|
|
|
struct List {
|
|
|
|
next: Option<@mut List>,
|
|
|
|
}
|
|
|
|
|
|
|
|
let a = @mut List { next: None };
|
|
|
|
let b = @mut List { next: Some(a) };
|
|
|
|
|
|
|
|
a.next = Some(b);
|
|
|
|
}
|
|
|
|
}
|
2013-06-23 16:01:59 -05:00
|
|
|
|
|
|
|
// XXX: This is a copy of test_future_result in std::task.
|
|
|
|
// It can be removed once the scheduler is turned on by default.
|
|
|
|
#[test]
|
|
|
|
fn future_result() {
|
|
|
|
do run_in_newsched_task {
|
|
|
|
use option::{Some, None};
|
|
|
|
use task::*;
|
|
|
|
|
|
|
|
let mut result = None;
|
|
|
|
let mut builder = task();
|
|
|
|
builder.future_result(|r| result = Some(r));
|
|
|
|
do builder.spawn {}
|
|
|
|
assert_eq!(result.unwrap().recv(), Success);
|
|
|
|
|
|
|
|
result = None;
|
|
|
|
let mut builder = task();
|
|
|
|
builder.future_result(|r| result = Some(r));
|
|
|
|
builder.unlinked();
|
|
|
|
do builder.spawn {
|
|
|
|
fail!();
|
|
|
|
}
|
|
|
|
assert_eq!(result.unwrap().recv(), Failure);
|
|
|
|
}
|
|
|
|
}
|
2013-05-08 14:26:34 -05:00
|
|
|
}
|