2014-01-16 01:34:05 +08:00
|
|
|
// Copyright 2013-2014 The Rust Project Developers. See the COPYRIGHT
|
2013-04-21 16:28:17 -07:00
|
|
|
// 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-12-12 18:01:59 -08:00
|
|
|
use any::AnyOwnExt;
|
2013-12-18 09:57:58 -08:00
|
|
|
use cast;
|
2013-06-22 01:09:06 -07:00
|
|
|
use cleanup;
|
2014-01-06 16:48:51 -08:00
|
|
|
use clone::Clone;
|
2013-12-05 18:19:06 -08:00
|
|
|
use io::Writer;
|
2013-12-18 09:57:58 -08:00
|
|
|
use iter::{Iterator, Take};
|
2013-10-11 23:20:34 +02:00
|
|
|
use local_data;
|
2014-01-06 10:26:11 -08:00
|
|
|
use logging::Logger;
|
2013-12-12 18:01:59 -08:00
|
|
|
use ops::Drop;
|
2013-06-13 23:31:19 -07:00
|
|
|
use option::{Option, Some, None};
|
2013-12-12 18:01:59 -08:00
|
|
|
use prelude::drop;
|
|
|
|
use result::{Result, Ok, Err};
|
|
|
|
use rt::Runtime;
|
2013-05-19 15:45:39 -07:00
|
|
|
use rt::local::Local;
|
2013-12-12 18:01:59 -08:00
|
|
|
use rt::local_heap::LocalHeap;
|
|
|
|
use rt::rtio::LocalIo;
|
2013-12-15 17:17:07 -08:00
|
|
|
use rt::unwind::Unwinder;
|
2014-02-07 16:36:59 -08:00
|
|
|
use str::SendStr;
|
2013-12-12 18:01:59 -08:00
|
|
|
use sync::arc::UnsafeArc;
|
2013-12-30 00:55:27 -08:00
|
|
|
use sync::atomics::{AtomicUint, SeqCst};
|
2013-12-12 18:01:59 -08:00
|
|
|
use task::{TaskResult, TaskOpts};
|
2013-10-11 23:20:34 +02:00
|
|
|
use unstable::finally::Finally;
|
2013-12-12 18:01:59 -08:00
|
|
|
|
2014-01-16 01:34:05 +08:00
|
|
|
/// The Task struct represents all state associated with a rust
|
|
|
|
/// task. There are at this point two primary "subtypes" of task,
|
|
|
|
/// however instead of using a subtype we just have a "task_type" field
|
|
|
|
/// in the struct. This contains a pointer to another struct that holds
|
|
|
|
/// the type-specific state.
|
2013-05-19 01:04:01 -07:00
|
|
|
pub struct Task {
|
2013-10-20 11:21:30 +05:30
|
|
|
heap: LocalHeap,
|
2013-12-12 18:01:59 -08:00
|
|
|
gc: GarbageCollector,
|
2013-10-20 11:21:30 +05:30
|
|
|
storage: LocalStorage,
|
|
|
|
unwinder: Unwinder,
|
|
|
|
death: Death,
|
|
|
|
destroyed: bool,
|
|
|
|
name: Option<SendStr>,
|
2013-12-12 18:01:59 -08:00
|
|
|
|
2014-01-06 10:26:11 -08:00
|
|
|
logger: Option<~Logger>,
|
|
|
|
stdout: Option<~Writer>,
|
|
|
|
stderr: Option<~Writer>,
|
2013-12-04 19:51:29 -08:00
|
|
|
|
2013-12-12 18:01:59 -08:00
|
|
|
priv imp: Option<~Runtime>,
|
2013-07-19 14:25:05 -07:00
|
|
|
}
|
|
|
|
|
2013-12-12 18:01:59 -08:00
|
|
|
pub struct GarbageCollector;
|
|
|
|
pub struct LocalStorage(Option<local_data::Map>);
|
2013-06-26 16:41:00 -07:00
|
|
|
|
2013-12-12 18:01:59 -08:00
|
|
|
/// A handle to a blocked task. Usually this means having the ~Task pointer by
|
|
|
|
/// ownership, but if the task is killable, a killer can steal it at any time.
|
|
|
|
pub enum BlockedTask {
|
|
|
|
Owned(~Task),
|
|
|
|
Shared(UnsafeArc<AtomicUint>),
|
2013-06-26 16:41:00 -07:00
|
|
|
}
|
|
|
|
|
2013-12-12 18:01:59 -08:00
|
|
|
/// Per-task state related to task death, killing, failure, etc.
|
|
|
|
pub struct Death {
|
|
|
|
// Action to be done with the exit code. If set, also makes the task wait
|
|
|
|
// until all its watched children exit before collecting the status.
|
|
|
|
on_exit: Option<proc(TaskResult)>,
|
2013-04-21 16:28:17 -07:00
|
|
|
}
|
|
|
|
|
2014-01-14 22:32:24 -05:00
|
|
|
pub struct BlockedTasks {
|
2013-12-12 18:01:59 -08:00
|
|
|
priv inner: UnsafeArc<AtomicUint>,
|
|
|
|
}
|
2013-04-22 17:15:31 -07:00
|
|
|
|
2013-05-19 01:04:01 -07:00
|
|
|
impl Task {
|
2013-12-12 18:01:59 -08:00
|
|
|
pub fn new() -> Task {
|
2013-05-19 01:04:01 -07:00
|
|
|
Task {
|
2013-04-21 19:03:52 -07:00
|
|
|
heap: LocalHeap::new(),
|
2013-04-21 16:28:17 -07:00
|
|
|
gc: GarbageCollector,
|
2013-08-10 20:06:39 -07:00
|
|
|
storage: LocalStorage(None),
|
2013-12-12 18:01:59 -08:00
|
|
|
unwinder: Unwinder::new(),
|
2013-07-01 23:24:24 -04:00
|
|
|
death: Death::new(),
|
2013-06-26 16:41:00 -07:00
|
|
|
destroyed: false,
|
2013-07-30 19:20:59 -04:00
|
|
|
name: None,
|
2013-11-05 11:29:45 -08:00
|
|
|
logger: None,
|
2014-01-06 10:26:11 -08:00
|
|
|
stdout: None,
|
|
|
|
stderr: None,
|
2013-12-12 18:01:59 -08:00
|
|
|
imp: None,
|
2013-04-21 16:28:17 -07:00
|
|
|
}
|
|
|
|
}
|
2013-04-22 12:54:03 -07:00
|
|
|
|
2013-12-12 18:01:59 -08:00
|
|
|
/// Executes the given closure as if it's running inside this task. The task
|
|
|
|
/// is consumed upon entry, and the destroyed task is returned from this
|
|
|
|
/// function in order for the caller to free. This function is guaranteed to
|
|
|
|
/// not unwind because the closure specified is run inside of a `rust_try`
|
|
|
|
/// block. (this is the only try/catch block in the world).
|
|
|
|
///
|
|
|
|
/// This function is *not* meant to be abused as a "try/catch" block. This
|
|
|
|
/// is meant to be used at the absolute boundaries of a task's lifetime, and
|
|
|
|
/// only for that purpose.
|
|
|
|
pub fn run(~self, f: ||) -> ~Task {
|
|
|
|
// Need to put ourselves into TLS, but also need access to the unwinder.
|
|
|
|
// Unsafely get a handle to the task so we can continue to use it after
|
|
|
|
// putting it in tls (so we can invoke the unwinder).
|
|
|
|
let handle: *mut Task = unsafe {
|
|
|
|
*cast::transmute::<&~Task, &*mut Task>(&self)
|
|
|
|
};
|
|
|
|
Local::put(self);
|
2013-04-22 17:15:31 -07:00
|
|
|
|
2013-08-03 14:43:16 -07:00
|
|
|
// The only try/catch block in the world. Attempt to run the task's
|
|
|
|
// client-specified code and catch any failures.
|
2013-12-12 18:01:59 -08:00
|
|
|
let try_block = || {
|
2013-06-26 16:41:00 -07:00
|
|
|
|
2013-08-03 14:43:16 -07:00
|
|
|
// Run the task main function, then do some cleanup.
|
2013-11-20 14:17:12 -08:00
|
|
|
f.finally(|| {
|
2014-01-29 16:33:57 -08:00
|
|
|
#[allow(unused_must_use)]
|
2014-01-06 10:26:11 -08:00
|
|
|
fn close_outputs() {
|
|
|
|
let mut task = Local::borrow(None::<Task>);
|
|
|
|
let logger = task.get().logger.take();
|
|
|
|
let stderr = task.get().stderr.take();
|
|
|
|
let stdout = task.get().stdout.take();
|
|
|
|
drop(task);
|
|
|
|
drop(logger); // loggers are responsible for flushing
|
2014-01-29 16:33:57 -08:00
|
|
|
match stdout { Some(mut w) => { w.flush(); }, None => {} }
|
|
|
|
match stderr { Some(mut w) => { w.flush(); }, None => {} }
|
2013-12-12 18:01:59 -08:00
|
|
|
}
|
2013-10-24 17:30:36 -07:00
|
|
|
|
2014-01-06 10:26:11 -08:00
|
|
|
// First, flush/destroy the user stdout/logger because these
|
|
|
|
// destructors can run arbitrary code.
|
|
|
|
close_outputs();
|
|
|
|
|
2013-08-10 20:06:39 -07:00
|
|
|
// First, destroy task-local storage. This may run user dtors.
|
|
|
|
//
|
2013-08-05 00:32:46 -07:00
|
|
|
// FIXME #8302: Dear diary. I'm so tired and confused.
|
|
|
|
// There's some interaction in rustc between the box
|
|
|
|
// annihilator and the TLS dtor by which TLS is
|
|
|
|
// accessed from annihilated box dtors *after* TLS is
|
|
|
|
// destroyed. Somehow setting TLS back to null, as the
|
|
|
|
// old runtime did, makes this work, but I don't currently
|
|
|
|
// understand how. I would expect that, if the annihilator
|
|
|
|
// reinvokes TLS while TLS is uninitialized, that
|
|
|
|
// TLS would be reinitialized but never destroyed,
|
|
|
|
// but somehow this works. I have no idea what's going
|
|
|
|
// on but this seems to make things magically work. FML.
|
2013-08-10 20:06:39 -07:00
|
|
|
//
|
|
|
|
// (added after initial comment) A possible interaction here is
|
|
|
|
// that the destructors for the objects in TLS themselves invoke
|
|
|
|
// TLS, or possibly some destructors for those objects being
|
|
|
|
// annihilated invoke TLS. Sadly these two operations seemed to
|
|
|
|
// be intertwined, and miraculously work for now...
|
2013-12-12 18:01:59 -08:00
|
|
|
let mut task = Local::borrow(None::<Task>);
|
2013-11-01 18:06:31 -07:00
|
|
|
let storage_map = {
|
|
|
|
let task = task.get();
|
|
|
|
let LocalStorage(ref mut optmap) = task.storage;
|
|
|
|
optmap.take()
|
|
|
|
};
|
2013-12-12 18:01:59 -08:00
|
|
|
drop(task);
|
2013-11-01 18:06:31 -07:00
|
|
|
drop(storage_map);
|
2013-08-05 00:32:46 -07:00
|
|
|
|
2013-08-03 14:43:16 -07:00
|
|
|
// Destroy remaining boxes. Also may run user dtors.
|
|
|
|
unsafe { cleanup::annihilate(); }
|
2013-10-24 17:30:36 -07:00
|
|
|
|
2014-01-06 10:26:11 -08:00
|
|
|
// Finally, just in case user dtors printed/logged during TLS
|
|
|
|
// cleanup and annihilation, re-destroy stdout and the logger.
|
|
|
|
// Note that these will have been initialized with a
|
|
|
|
// runtime-provided type which we have control over what the
|
|
|
|
// destructor does.
|
|
|
|
close_outputs();
|
2013-11-20 14:17:12 -08:00
|
|
|
})
|
2013-12-12 18:01:59 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
unsafe { (*handle).unwinder.try(try_block); }
|
2013-06-22 01:09:06 -07:00
|
|
|
|
2013-12-13 21:14:08 -08:00
|
|
|
// Here we must unsafely borrow the task in order to not remove it from
|
|
|
|
// TLS. When collecting failure, we may attempt to send on a channel (or
|
|
|
|
// just run aribitrary code), so we must be sure to still have a local
|
|
|
|
// task in TLS.
|
2013-12-12 21:38:57 -08:00
|
|
|
unsafe {
|
|
|
|
let me: *mut Task = Local::unsafe_borrow();
|
|
|
|
(*me).death.collect_failure((*me).unwinder.result());
|
|
|
|
}
|
2013-12-12 18:01:59 -08:00
|
|
|
let mut me: ~Task = Local::take();
|
|
|
|
me.destroyed = true;
|
|
|
|
return me;
|
2013-04-22 12:54:03 -07:00
|
|
|
}
|
2013-06-26 16:41:00 -07:00
|
|
|
|
2013-12-12 18:01:59 -08:00
|
|
|
/// Inserts a runtime object into this task, transferring ownership to the
|
|
|
|
/// task. It is illegal to replace a previous runtime object in this task
|
|
|
|
/// with this argument.
|
|
|
|
pub fn put_runtime(&mut self, ops: ~Runtime) {
|
|
|
|
assert!(self.imp.is_none());
|
|
|
|
self.imp = Some(ops);
|
|
|
|
}
|
2013-06-26 16:41:00 -07:00
|
|
|
|
2013-12-12 18:01:59 -08:00
|
|
|
/// Attempts to extract the runtime as a specific type. If the runtime does
|
|
|
|
/// not have the provided type, then the runtime is not removed. If the
|
|
|
|
/// runtime does have the specified type, then it is removed and returned
|
|
|
|
/// (transfer of ownership).
|
|
|
|
///
|
|
|
|
/// It is recommended to only use this method when *absolutely necessary*.
|
|
|
|
/// This function may not be available in the future.
|
|
|
|
pub fn maybe_take_runtime<T: 'static>(&mut self) -> Option<~T> {
|
|
|
|
// This is a terrible, terrible function. The general idea here is to
|
|
|
|
// take the runtime, cast it to ~Any, check if it has the right type,
|
|
|
|
// and then re-cast it back if necessary. The method of doing this is
|
|
|
|
// pretty sketchy and involves shuffling vtables of trait objects
|
|
|
|
// around, but it gets the job done.
|
|
|
|
//
|
2014-01-26 03:43:42 -05:00
|
|
|
// FIXME: This function is a serious code smell and should be avoided at
|
2013-12-12 18:01:59 -08:00
|
|
|
// all costs. I have yet to think of a method to avoid this
|
|
|
|
// function, and I would be saddened if more usage of the function
|
|
|
|
// crops up.
|
|
|
|
unsafe {
|
|
|
|
let imp = self.imp.take_unwrap();
|
|
|
|
let &(vtable, _): &(uint, uint) = cast::transmute(&imp);
|
|
|
|
match imp.wrap().move::<T>() {
|
|
|
|
Ok(t) => Some(t),
|
|
|
|
Err(t) => {
|
|
|
|
let (_, obj): (uint, uint) = cast::transmute(t);
|
|
|
|
let obj: ~Runtime = cast::transmute((vtable, obj));
|
|
|
|
self.put_runtime(obj);
|
|
|
|
None
|
|
|
|
}
|
2013-06-26 16:41:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-12 18:01:59 -08:00
|
|
|
/// Spawns a sibling to this task. The newly spawned task is configured with
|
|
|
|
/// the `opts` structure and will run `f` as the body of its code.
|
|
|
|
pub fn spawn_sibling(mut ~self, opts: TaskOpts, f: proc()) {
|
|
|
|
let ops = self.imp.take_unwrap();
|
|
|
|
ops.spawn_sibling(self, opts, f)
|
2013-06-26 16:41:00 -07:00
|
|
|
}
|
|
|
|
|
2013-12-12 18:01:59 -08:00
|
|
|
/// Deschedules the current task, invoking `f` `amt` times. It is not
|
|
|
|
/// recommended to use this function directly, but rather communication
|
|
|
|
/// primitives in `std::comm` should be used.
|
|
|
|
pub fn deschedule(mut ~self, amt: uint,
|
|
|
|
f: |BlockedTask| -> Result<(), BlockedTask>) {
|
|
|
|
let ops = self.imp.take_unwrap();
|
|
|
|
ops.deschedule(amt, self, f)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Wakes up a previously blocked task, optionally specifiying whether the
|
|
|
|
/// current task can accept a change in scheduling. This function can only
|
|
|
|
/// be called on tasks that were previously blocked in `deschedule`.
|
2014-01-16 19:58:42 -08:00
|
|
|
pub fn reawaken(mut ~self) {
|
2013-12-12 18:01:59 -08:00
|
|
|
let ops = self.imp.take_unwrap();
|
2014-01-16 19:58:42 -08:00
|
|
|
ops.reawaken(self);
|
2013-12-12 18:01:59 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Yields control of this task to another task. This function will
|
|
|
|
/// eventually return, but possibly not immediately. This is used as an
|
|
|
|
/// opportunity to allow other tasks a chance to run.
|
|
|
|
pub fn yield_now(mut ~self) {
|
|
|
|
let ops = self.imp.take_unwrap();
|
|
|
|
ops.yield_now(self);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Similar to `yield_now`, except that this function may immediately return
|
|
|
|
/// without yielding (depending on what the runtime decides to do).
|
|
|
|
pub fn maybe_yield(mut ~self) {
|
|
|
|
let ops = self.imp.take_unwrap();
|
|
|
|
ops.maybe_yield(self);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Acquires a handle to the I/O factory that this task contains, normally
|
|
|
|
/// stored in the task's runtime. This factory may not always be available,
|
|
|
|
/// which is why the return type is `Option`
|
|
|
|
pub fn local_io<'a>(&'a mut self) -> Option<LocalIo<'a>> {
|
|
|
|
self.imp.get_mut_ref().local_io()
|
2013-06-26 16:41:00 -07:00
|
|
|
}
|
2014-01-04 00:06:13 -08:00
|
|
|
|
|
|
|
/// Returns the stack bounds for this task in (lo, hi) format. The stack
|
|
|
|
/// bounds may not be known for all tasks, so the return value may be
|
|
|
|
/// `None`.
|
2014-01-07 13:39:55 +11:00
|
|
|
pub fn stack_bounds(&self) -> (uint, uint) {
|
2014-01-04 00:06:13 -08:00
|
|
|
self.imp.get_ref().stack_bounds()
|
|
|
|
}
|
2014-01-16 19:58:42 -08:00
|
|
|
|
|
|
|
/// Returns whether it is legal for this task to block the OS thread that it
|
|
|
|
/// is running on.
|
|
|
|
pub fn can_block(&self) -> bool {
|
|
|
|
self.imp.get_ref().can_block()
|
|
|
|
}
|
2013-04-22 12:54:03 -07:00
|
|
|
}
|
|
|
|
|
2013-05-19 01:04:01 -07:00
|
|
|
impl Drop for Task {
|
2013-09-16 21:18:07 -04:00
|
|
|
fn drop(&mut self) {
|
2014-01-28 21:05:57 -05:00
|
|
|
rtdebug!("called drop for a task: {}", self as *mut Task as uint);
|
2013-10-24 17:30:36 -07:00
|
|
|
rtassert!(self.destroyed);
|
2013-07-19 14:25:05 -07:00
|
|
|
}
|
2013-04-21 16:28:17 -07:00
|
|
|
}
|
|
|
|
|
2014-01-14 22:32:24 -05:00
|
|
|
impl Iterator<BlockedTask> for BlockedTasks {
|
2013-12-12 18:01:59 -08:00
|
|
|
fn next(&mut self) -> Option<BlockedTask> {
|
|
|
|
Some(Shared(self.inner.clone()))
|
2013-06-26 16:41:00 -07:00
|
|
|
}
|
2013-12-12 18:01:59 -08:00
|
|
|
}
|
2013-06-26 16:41:00 -07:00
|
|
|
|
2013-12-12 18:01:59 -08:00
|
|
|
impl BlockedTask {
|
|
|
|
/// Returns Some if the task was successfully woken; None if already killed.
|
|
|
|
pub fn wake(self) -> Option<~Task> {
|
|
|
|
match self {
|
|
|
|
Owned(task) => Some(task),
|
|
|
|
Shared(arc) => unsafe {
|
|
|
|
match (*arc.get()).swap(0, SeqCst) {
|
|
|
|
0 => None,
|
|
|
|
n => Some(cast::transmute(n)),
|
|
|
|
}
|
|
|
|
}
|
2013-07-19 14:25:05 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-12 18:01:59 -08:00
|
|
|
// This assertion has two flavours because the wake involves an atomic op.
|
|
|
|
// In the faster version, destructors will fail dramatically instead.
|
|
|
|
#[cfg(not(test))] pub fn trash(self) { }
|
|
|
|
#[cfg(test)] pub fn trash(self) { assert!(self.wake().is_none()); }
|
2013-06-26 16:41:00 -07:00
|
|
|
|
2013-12-12 18:01:59 -08:00
|
|
|
/// Create a blocked task, unless the task was already killed.
|
|
|
|
pub fn block(task: ~Task) -> BlockedTask {
|
|
|
|
Owned(task)
|
|
|
|
}
|
2013-07-19 14:25:05 -07:00
|
|
|
|
2013-12-12 18:01:59 -08:00
|
|
|
/// Converts one blocked task handle to a list of many handles to the same.
|
2014-01-14 22:32:24 -05:00
|
|
|
pub fn make_selectable(self, num_handles: uint) -> Take<BlockedTasks>
|
2013-12-12 18:01:59 -08:00
|
|
|
{
|
|
|
|
let arc = match self {
|
|
|
|
Owned(task) => {
|
|
|
|
let flag = unsafe { AtomicUint::new(cast::transmute(task)) };
|
|
|
|
UnsafeArc::new(flag)
|
2013-06-26 16:41:00 -07:00
|
|
|
}
|
2013-12-12 18:01:59 -08:00
|
|
|
Shared(arc) => arc.clone(),
|
2013-06-26 16:41:00 -07:00
|
|
|
};
|
2014-01-14 22:32:24 -05:00
|
|
|
BlockedTasks{ inner: arc }.take(num_handles)
|
2013-06-26 16:41:00 -07:00
|
|
|
}
|
|
|
|
|
2013-12-12 18:01:59 -08:00
|
|
|
/// Convert to an unsafe uint value. Useful for storing in a pipe's state
|
|
|
|
/// flag.
|
|
|
|
#[inline]
|
|
|
|
pub unsafe fn cast_to_uint(self) -> uint {
|
2013-06-26 16:41:00 -07:00
|
|
|
match self {
|
2013-12-12 18:01:59 -08:00
|
|
|
Owned(task) => {
|
|
|
|
let blocked_task_ptr: uint = cast::transmute(task);
|
|
|
|
rtassert!(blocked_task_ptr & 0x1 == 0);
|
|
|
|
blocked_task_ptr
|
|
|
|
}
|
|
|
|
Shared(arc) => {
|
|
|
|
let blocked_task_ptr: uint = cast::transmute(~arc);
|
|
|
|
rtassert!(blocked_task_ptr & 0x1 == 0);
|
|
|
|
blocked_task_ptr | 0x1
|
2013-06-26 16:41:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-12 18:01:59 -08:00
|
|
|
/// Convert from an unsafe uint value. Useful for retrieving a pipe's state
|
|
|
|
/// flag.
|
|
|
|
#[inline]
|
|
|
|
pub unsafe fn cast_from_uint(blocked_task_ptr: uint) -> BlockedTask {
|
|
|
|
if blocked_task_ptr & 0x1 == 0 {
|
|
|
|
Owned(cast::transmute(blocked_task_ptr))
|
2013-10-17 01:40:33 -07:00
|
|
|
} else {
|
2013-12-12 18:01:59 -08:00
|
|
|
let ptr: ~UnsafeArc<AtomicUint> =
|
|
|
|
cast::transmute(blocked_task_ptr & !1);
|
|
|
|
Shared(*ptr)
|
2013-10-17 01:40:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-12 18:01:59 -08:00
|
|
|
impl Death {
|
|
|
|
pub fn new() -> Death {
|
|
|
|
Death { on_exit: None, }
|
|
|
|
}
|
2013-10-11 23:20:34 +02:00
|
|
|
|
2013-12-12 18:01:59 -08:00
|
|
|
/// Collect failure exit codes from children and propagate them to a parent.
|
|
|
|
pub fn collect_failure(&mut self, result: TaskResult) {
|
|
|
|
match self.on_exit.take() {
|
|
|
|
Some(f) => f(result),
|
|
|
|
None => {}
|
2013-10-11 23:20:34 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-12 18:01:59 -08:00
|
|
|
impl Drop for Death {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
// make this type noncopyable
|
2013-10-09 10:34:27 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-21 19:03:52 -07:00
|
|
|
#[cfg(test)]
|
|
|
|
mod test {
|
2013-10-27 20:12:40 +01:00
|
|
|
use super::*;
|
2013-12-05 18:19:06 -08:00
|
|
|
use prelude::*;
|
2013-12-12 21:38:57 -08:00
|
|
|
use task;
|
2013-04-21 19:03:52 -07:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn local_heap() {
|
2013-12-12 21:38:57 -08:00
|
|
|
let a = @5;
|
|
|
|
let b = a;
|
|
|
|
assert!(*a == 5);
|
|
|
|
assert!(*b == 5);
|
2013-04-21 19:03:52 -07:00
|
|
|
}
|
2013-04-22 12:54:03 -07:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn tls() {
|
2013-07-10 22:14:40 -07:00
|
|
|
use local_data;
|
2013-12-12 21:38:57 -08:00
|
|
|
local_data_key!(key: @~str)
|
|
|
|
local_data::set(key, @~"data");
|
|
|
|
assert!(*local_data::get(key, |k| k.map(|k| *k)).unwrap() == ~"data");
|
|
|
|
local_data_key!(key2: @~str)
|
|
|
|
local_data::set(key2, @~"data");
|
|
|
|
assert!(*local_data::get(key2, |k| k.map(|k| *k)).unwrap() == ~"data");
|
2013-04-22 12:54:03 -07:00
|
|
|
}
|
2013-04-22 17:15:31 -07:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn unwind() {
|
2013-12-12 21:38:57 -08:00
|
|
|
let result = task::try(proc()());
|
|
|
|
rtdebug!("trying first assert");
|
|
|
|
assert!(result.is_ok());
|
|
|
|
let result = task::try::<()>(proc() fail!());
|
|
|
|
rtdebug!("trying second assert");
|
|
|
|
assert!(result.is_err());
|
2013-04-22 17:15:31 -07:00
|
|
|
}
|
2013-05-06 18:24:37 -07:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn rng() {
|
2013-12-12 21:38:57 -08:00
|
|
|
use rand::{rng, Rng};
|
|
|
|
let mut r = rng();
|
|
|
|
let _ = r.next_u32();
|
2013-05-06 18:24:37 -07:00
|
|
|
}
|
2013-04-27 18:57:15 -07:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn logging() {
|
2013-12-12 21:38:57 -08:00
|
|
|
info!("here i am. logging in a newsched task");
|
2013-04-27 18:57:15 -07:00
|
|
|
}
|
2013-05-16 23:12:22 -07:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn comm_stream() {
|
2013-12-12 21:38:57 -08:00
|
|
|
let (port, chan) = Chan::new();
|
|
|
|
chan.send(10);
|
|
|
|
assert!(port.recv() == 10);
|
2013-05-16 23:12:22 -07:00
|
|
|
}
|
2013-06-13 23:31:19 -07:00
|
|
|
|
2013-06-20 18:26:56 -07:00
|
|
|
#[test]
|
|
|
|
fn comm_shared_chan() {
|
Rewrite channels yet again for upgradeability
This, the Nth rewrite of channels, is not a rewrite of the core logic behind
channels, but rather their API usage. In the past, we had the distinction
between oneshot, stream, and shared channels, but the most recent rewrite
dropped oneshots in favor of streams and shared channels.
This distinction of stream vs shared has shown that it's not quite what we'd
like either, and this moves the `std::comm` module in the direction of "one
channel to rule them all". There now remains only one Chan and one Port.
This new channel is actually a hybrid oneshot/stream/shared channel under the
hood in order to optimize for the use cases in question. Additionally, this also
reduces the cognitive burden of having to choose between a Chan or a SharedChan
in an API.
My simple benchmarks show no reduction in efficiency over the existing channels
today, and a 3x improvement in the oneshot case. I sadly don't have a
pre-last-rewrite compiler to test out the old old oneshots, but I would imagine
that the performance is comparable, but slightly slower (due to atomic reference
counting).
This commit also brings the bonus bugfix to channels that the pending queue of
messages are all dropped when a Port disappears rather then when both the Port
and the Chan disappear.
2014-01-08 18:31:48 -08:00
|
|
|
let (port, chan) = Chan::new();
|
2013-12-12 21:38:57 -08:00
|
|
|
chan.send(10);
|
|
|
|
assert!(port.recv() == 10);
|
2013-06-20 18:26:56 -07:00
|
|
|
}
|
|
|
|
|
2013-06-22 01:09:06 -07:00
|
|
|
#[test]
|
|
|
|
fn heap_cycles() {
|
2013-12-30 17:46:48 -08:00
|
|
|
use cell::RefCell;
|
2013-06-22 01:09:06 -07:00
|
|
|
use option::{Option, Some, None};
|
|
|
|
|
2013-12-12 21:38:57 -08:00
|
|
|
struct List {
|
2013-12-30 17:46:48 -08:00
|
|
|
next: Option<@RefCell<List>>,
|
2013-12-12 21:38:57 -08:00
|
|
|
}
|
2013-06-22 01:09:06 -07:00
|
|
|
|
2013-12-30 17:46:48 -08:00
|
|
|
let a = @RefCell::new(List { next: None });
|
|
|
|
let b = @RefCell::new(List { next: Some(a) });
|
2013-06-22 01:09:06 -07:00
|
|
|
|
2013-12-30 17:46:48 -08:00
|
|
|
{
|
|
|
|
let mut a = a.borrow_mut();
|
|
|
|
a.get().next = Some(b);
|
|
|
|
}
|
2013-06-22 01:09:06 -07:00
|
|
|
}
|
2013-10-27 20:12:40 +01:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[should_fail]
|
2013-12-18 09:57:58 -08:00
|
|
|
fn test_begin_unwind() {
|
|
|
|
use rt::unwind::begin_unwind;
|
|
|
|
begin_unwind("cause", file!(), line!())
|
|
|
|
}
|
2013-12-12 18:01:59 -08:00
|
|
|
|
|
|
|
// Task blocking tests
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn block_and_wake() {
|
2013-12-12 21:38:57 -08:00
|
|
|
let task = ~Task::new();
|
|
|
|
let mut task = BlockedTask::block(task).wake().unwrap();
|
|
|
|
task.destroyed = true;
|
2013-12-12 18:01:59 -08:00
|
|
|
}
|
2013-05-08 12:26:34 -07:00
|
|
|
}
|