2013-05-30 03:16:33 -07:00
|
|
|
// Copyright 2012-2013 The Rust Project Developers. See the COPYRIGHT
|
2012-12-03 16:48:01 -08: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.
|
|
|
|
|
2012-11-28 16:20:41 -08:00
|
|
|
/*!
|
2013-12-24 17:08:28 +01:00
|
|
|
* Utilities for managing and scheduling tasks
|
2012-11-28 16:20:41 -08:00
|
|
|
*
|
|
|
|
* An executing Rust program consists of a tree of tasks, each with their own
|
|
|
|
* stack, and sole ownership of their allocated heap data. Tasks communicate
|
2013-09-27 23:22:16 +02:00
|
|
|
* with each other using ports and channels (see std::rt::comm for more info
|
|
|
|
* about how communication works).
|
2012-11-28 16:20:41 -08:00
|
|
|
*
|
2013-09-27 23:22:16 +02:00
|
|
|
* Tasks can be spawned in 3 different modes.
|
2012-11-28 16:20:41 -08:00
|
|
|
*
|
2013-09-27 23:22:16 +02:00
|
|
|
* * Bidirectionally linked: This is the default mode and it's what ```spawn``` does.
|
|
|
|
* Failures will be propagated from parent to child and vice versa.
|
|
|
|
*
|
|
|
|
* * Unidirectionally linked (parent->child): This type of task can be created with
|
|
|
|
* ```spawn_supervised```. In this case, failures are propagated from parent to child
|
|
|
|
* but not the other way around.
|
|
|
|
*
|
|
|
|
* * Unlinked: Tasks can be completely unlinked. These tasks can be created by using
|
|
|
|
* ```spawn_unlinked```. In this case failures are not propagated at all.
|
|
|
|
*
|
|
|
|
* Tasks' failure modes can be further configured. For instance, parent tasks can (un)watch
|
|
|
|
* children failures. Please, refer to TaskBuilder's documentation bellow for more information.
|
|
|
|
*
|
|
|
|
* When a (bi|uni)directionally linked task fails, its failure will be propagated to all tasks
|
|
|
|
* linked to it, this will cause such tasks to fail by a `linked failure`.
|
|
|
|
*
|
|
|
|
* Task Scheduling:
|
|
|
|
*
|
|
|
|
* By default, every task is created in the same scheduler as its parent, where it
|
|
|
|
* is scheduled cooperatively with all other tasks in that scheduler. Some specialized
|
|
|
|
* applications may want more control over their scheduling, in which case they can be
|
|
|
|
* spawned into a new scheduler with the specific properties required. See TaskBuilder's
|
|
|
|
* documentation bellow for more information.
|
2012-11-28 16:20:41 -08:00
|
|
|
*
|
|
|
|
* # Example
|
|
|
|
*
|
2013-09-23 17:20:36 -07:00
|
|
|
* ```
|
2014-01-26 22:42:26 -05:00
|
|
|
* spawn(proc() {
|
2012-11-28 16:20:41 -08:00
|
|
|
* log(error, "Hello, World!");
|
2014-01-26 22:42:26 -05:00
|
|
|
* })
|
2013-09-23 17:20:36 -07:00
|
|
|
* ```
|
2012-11-28 16:20:41 -08:00
|
|
|
*/
|
2012-11-18 17:56:50 -08:00
|
|
|
|
2013-05-28 16:35:52 -05:00
|
|
|
#[allow(missing_doc)];
|
|
|
|
|
2013-12-12 18:01:59 -08:00
|
|
|
use any::Any;
|
2013-12-05 18:19:06 -08:00
|
|
|
use comm::{Chan, Port};
|
2014-01-06 10:26:11 -08:00
|
|
|
use io::Writer;
|
2014-01-26 11:42:46 +01:00
|
|
|
use kinds::{Send, marker};
|
2014-01-06 10:26:11 -08:00
|
|
|
use logging::Logger;
|
2013-12-12 18:01:59 -08:00
|
|
|
use option::{None, Some, Option};
|
2013-10-11 23:20:34 +02:00
|
|
|
use result::{Result, Ok, Err};
|
2013-07-08 13:48:57 -04:00
|
|
|
use rt::local::Local;
|
2013-12-12 18:01:59 -08:00
|
|
|
use rt::task::Task;
|
2014-02-07 16:36:59 -08:00
|
|
|
use str::{Str, SendStr, IntoMaybeOwned};
|
2012-11-18 17:56:50 -08:00
|
|
|
|
2013-12-12 21:38:57 -08:00
|
|
|
#[cfg(test)] use any::{AnyOwnExt, AnyRefExt};
|
2013-05-30 03:16:33 -07:00
|
|
|
#[cfg(test)] use ptr;
|
2013-10-11 23:20:34 +02:00
|
|
|
#[cfg(test)] use result;
|
2013-03-26 16:38:07 -04:00
|
|
|
|
2013-10-11 23:20:34 +02:00
|
|
|
/// Indicates the manner in which a task exited.
|
|
|
|
///
|
|
|
|
/// A task that completes without failing is considered to exit successfully.
|
|
|
|
/// Supervised ancestors and linked siblings may yet fail after this task
|
|
|
|
/// succeeds. Also note that in such a case, it may be nondeterministic whether
|
|
|
|
/// linked failure or successful exit happen first.
|
|
|
|
///
|
|
|
|
/// If you wish for this result's delivery to block until all linked and/or
|
|
|
|
/// children tasks complete, recommend using a result future.
|
|
|
|
pub type TaskResult = Result<(), ~Any>;
|
|
|
|
|
2012-11-28 16:20:41 -08:00
|
|
|
/**
|
|
|
|
* Task configuration options
|
|
|
|
*
|
|
|
|
* # Fields
|
|
|
|
*
|
2013-07-16 17:28:46 -04:00
|
|
|
* * watched - Make parent task collect exit status notifications from child
|
|
|
|
* before reporting its own exit status. (This delays the parent
|
|
|
|
* task's death and cleanup until after all transitively watched
|
|
|
|
* children also exit.) True by default.
|
|
|
|
*
|
2012-11-28 16:20:41 -08:00
|
|
|
* * notify_chan - Enable lifecycle notifications on the given channel
|
|
|
|
*
|
2013-07-30 19:20:59 -04:00
|
|
|
* * name - A name for the task-to-be, for identification in failure messages.
|
|
|
|
*
|
2012-11-28 16:20:41 -08:00
|
|
|
* * sched - Specify the configuration of a new scheduler to create the task
|
2013-09-27 23:22:16 +02:00
|
|
|
* in. This is of particular importance for libraries which want to call
|
|
|
|
* into foreign code that blocks. Without doing so in a different
|
|
|
|
* scheduler other tasks will be impeded or even blocked indefinitely.
|
2012-11-28 16:20:41 -08:00
|
|
|
*/
|
2013-01-22 08:12:52 -08:00
|
|
|
pub struct TaskOpts {
|
2013-12-12 18:01:59 -08:00
|
|
|
watched: bool,
|
|
|
|
notify_chan: Option<Chan<TaskResult>>,
|
2013-10-20 08:56:42 +05:30
|
|
|
name: Option<SendStr>,
|
2014-01-06 10:26:11 -08:00
|
|
|
stack_size: Option<uint>,
|
|
|
|
logger: Option<~Logger>,
|
|
|
|
stdout: Option<~Writer>,
|
|
|
|
stderr: Option<~Writer>,
|
2013-01-22 08:12:52 -08:00
|
|
|
}
|
2012-11-28 16:20:41 -08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The task builder type.
|
|
|
|
*
|
|
|
|
* Provides detailed control over the properties and behavior of new tasks.
|
|
|
|
*/
|
|
|
|
// NB: Builders are designed to be single-use because they do stateful
|
|
|
|
// things that get weird when reusing - e.g. if you create a result future
|
|
|
|
// it only applies to a single task, so then you have to maintain Some
|
|
|
|
// potentially tricky state to ensure that everything behaves correctly
|
|
|
|
// when you try to reuse the builder to spawn a new task. We'll just
|
|
|
|
// sidestep that whole issue by making builders uncopyable and making
|
|
|
|
// the run function move them in.
|
2012-12-05 15:06:54 -08:00
|
|
|
pub struct TaskBuilder {
|
2012-11-28 16:20:41 -08:00
|
|
|
opts: TaskOpts,
|
2013-11-18 13:25:09 -08:00
|
|
|
priv gen_body: Option<proc(v: proc()) -> proc()>,
|
2014-01-26 11:42:46 +01:00
|
|
|
priv nopod: Option<marker::NoPod>,
|
2012-12-05 15:06:54 -08:00
|
|
|
}
|
2012-11-28 16:20:41 -08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Generate the base configuration for spawning a task, off of which more
|
|
|
|
* configuration methods can be chained.
|
|
|
|
* For example, task().unlinked().spawn is equivalent to spawn_unlinked.
|
|
|
|
*/
|
|
|
|
pub fn task() -> TaskBuilder {
|
2012-12-05 15:06:54 -08:00
|
|
|
TaskBuilder {
|
2013-12-13 16:26:02 -08:00
|
|
|
opts: TaskOpts::new(),
|
2013-04-18 18:22:04 -07:00
|
|
|
gen_body: None,
|
2014-01-26 11:42:46 +01:00
|
|
|
nopod: None,
|
2012-12-05 15:06:54 -08:00
|
|
|
}
|
2012-11-28 16:20:41 -08:00
|
|
|
}
|
|
|
|
|
2013-05-31 15:17:22 -07:00
|
|
|
impl TaskBuilder {
|
2013-07-16 17:28:46 -04:00
|
|
|
/// Cause the parent task to collect the child's exit status (and that of
|
|
|
|
/// all transitively-watched grandchildren) before reporting its own.
|
|
|
|
pub fn watched(&mut self) {
|
|
|
|
self.opts.watched = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Allow the child task to outlive the parent task, at the possible cost
|
|
|
|
/// of the parent reporting success even if the child task fails later.
|
|
|
|
pub fn unwatched(&mut self) {
|
|
|
|
self.opts.watched = false;
|
|
|
|
}
|
|
|
|
|
2013-10-18 10:38:46 +02:00
|
|
|
/// Get a future representing the exit status of the task.
|
|
|
|
///
|
|
|
|
/// Taking the value of the future will block until the child task
|
|
|
|
/// terminates. The future result return value will be created *before* the task is
|
|
|
|
/// spawned; as such, do not invoke .get() on it directly;
|
|
|
|
/// rather, store it in an outer variable/list for later use.
|
|
|
|
///
|
|
|
|
/// Note that the future returned by this function is only useful for
|
|
|
|
/// obtaining the value of the next task to be spawning with the
|
|
|
|
/// builder. If additional tasks are spawned with the same builder
|
|
|
|
/// then a new result future must be obtained prior to spawning each
|
|
|
|
/// task.
|
|
|
|
///
|
|
|
|
/// # Failure
|
|
|
|
/// Fails if a future_result was already set for this task.
|
2013-12-05 18:19:06 -08:00
|
|
|
pub fn future_result(&mut self) -> Port<TaskResult> {
|
2012-11-28 16:20:41 -08:00
|
|
|
// FIXME (#3725): Once linked failure and notification are
|
|
|
|
// handled in the library, I can imagine implementing this by just
|
|
|
|
// registering an arbitrary number of task::on_exit handlers and
|
|
|
|
// sending out messages.
|
|
|
|
|
|
|
|
if self.opts.notify_chan.is_some() {
|
2013-10-21 13:08:31 -07:00
|
|
|
fail!("Can't set multiple future_results for one task!");
|
2012-11-28 16:20:41 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Construct the future and give it to the caller.
|
2013-12-05 18:19:06 -08:00
|
|
|
let (notify_pipe_po, notify_pipe_ch) = Chan::new();
|
2012-11-28 16:20:41 -08:00
|
|
|
|
|
|
|
// Reconfigure self to use a notify channel.
|
2013-05-03 13:21:33 -07:00
|
|
|
self.opts.notify_chan = Some(notify_pipe_ch);
|
2013-10-18 10:38:46 +02:00
|
|
|
|
2013-12-05 18:19:06 -08:00
|
|
|
notify_pipe_po
|
2012-11-28 16:20:41 -08:00
|
|
|
}
|
2013-05-03 13:21:33 -07:00
|
|
|
|
2013-07-30 19:20:59 -04:00
|
|
|
/// Name the task-to-be. Currently the name is used for identification
|
|
|
|
/// only in failure messages.
|
2014-02-07 16:36:59 -08:00
|
|
|
pub fn name<S: IntoMaybeOwned<'static>>(&mut self, name: S) {
|
|
|
|
self.opts.name = Some(name.into_maybe_owned());
|
2013-07-30 19:20:59 -04:00
|
|
|
}
|
|
|
|
|
2012-11-28 16:20:41 -08:00
|
|
|
/**
|
|
|
|
* Add a wrapper to the body of the spawned task.
|
|
|
|
*
|
|
|
|
* Before the task is spawned it is passed through a 'body generator'
|
|
|
|
* function that may perform local setup operations as well as wrap
|
|
|
|
* the task body in remote setup operations. With this the behavior
|
|
|
|
* of tasks can be extended in simple ways.
|
|
|
|
*
|
|
|
|
* This function augments the current body generator with a new body
|
|
|
|
* generator by applying the task body which results from the
|
|
|
|
* existing body generator to the new body generator.
|
|
|
|
*/
|
2013-11-18 13:25:09 -08:00
|
|
|
pub fn add_wrapper(&mut self, wrapper: proc(v: proc()) -> proc()) {
|
2013-07-17 23:41:50 +02:00
|
|
|
let prev_gen_body = self.gen_body.take();
|
2013-04-18 18:22:04 -07:00
|
|
|
let prev_gen_body = match prev_gen_body {
|
|
|
|
Some(gen) => gen,
|
|
|
|
None => {
|
2013-11-21 23:36:52 -08:00
|
|
|
let f: proc(proc()) -> proc() = proc(body) body;
|
2013-04-18 18:22:04 -07:00
|
|
|
f
|
|
|
|
}
|
|
|
|
};
|
|
|
|
let next_gen_body = {
|
2013-11-21 23:36:52 -08:00
|
|
|
let f: proc(proc()) -> proc() = proc(body) {
|
2013-04-18 18:22:04 -07:00
|
|
|
wrapper(prev_gen_body(body))
|
|
|
|
};
|
|
|
|
f
|
|
|
|
};
|
2013-05-03 13:21:33 -07:00
|
|
|
self.gen_body = Some(next_gen_body);
|
2012-11-28 16:20:41 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates and executes a new child task
|
|
|
|
*
|
|
|
|
* Sets up a new task with its own call stack and schedules it to run
|
|
|
|
* the provided unique closure. The task has the properties and behavior
|
|
|
|
* specified by the task_builder.
|
|
|
|
*
|
|
|
|
* # Failure
|
|
|
|
*
|
|
|
|
* When spawning into a new scheduler, the number of threads requested
|
|
|
|
* must be greater than zero.
|
|
|
|
*/
|
2013-11-21 16:55:40 -08:00
|
|
|
pub fn spawn(mut self, f: proc()) {
|
2013-07-17 23:41:50 +02:00
|
|
|
let gen_body = self.gen_body.take();
|
2013-04-18 18:22:04 -07:00
|
|
|
let f = match gen_body {
|
2014-01-06 10:26:11 -08:00
|
|
|
Some(gen) => gen(f),
|
|
|
|
None => f
|
2013-04-18 18:22:04 -07:00
|
|
|
};
|
2013-12-12 18:01:59 -08:00
|
|
|
let t: ~Task = Local::take();
|
2014-01-06 10:26:11 -08:00
|
|
|
t.spawn_sibling(self.opts, f);
|
2012-11-28 16:20:41 -08:00
|
|
|
}
|
2013-05-03 13:21:33 -07:00
|
|
|
|
2012-11-28 16:20:41 -08:00
|
|
|
/**
|
|
|
|
* Execute a function in another task and return either the return value
|
|
|
|
* of the function or result::err.
|
|
|
|
*
|
|
|
|
* # Return value
|
|
|
|
*
|
|
|
|
* If the function executed successfully then try returns result::ok
|
|
|
|
* containing the value returned by the function. If the function fails
|
|
|
|
* then try returns result::err containing nil.
|
|
|
|
*
|
|
|
|
* # Failure
|
|
|
|
* Fails if a future_result was already set for this task.
|
|
|
|
*/
|
2013-11-21 16:55:40 -08:00
|
|
|
pub fn try<T:Send>(mut self, f: proc() -> T) -> Result<T, ~Any> {
|
2013-12-05 18:19:06 -08:00
|
|
|
let (po, ch) = Chan::new();
|
2012-11-28 16:20:41 -08:00
|
|
|
|
2013-10-18 10:38:46 +02:00
|
|
|
let result = self.future_result();
|
2013-05-03 13:21:33 -07:00
|
|
|
|
2014-01-26 22:42:26 -05:00
|
|
|
self.spawn(proc() {
|
2013-01-22 12:38:08 -08:00
|
|
|
ch.send(f());
|
2014-01-26 22:42:26 -05:00
|
|
|
});
|
2013-05-03 13:21:33 -07:00
|
|
|
|
2013-10-18 10:38:46 +02:00
|
|
|
match result.recv() {
|
2013-10-11 23:20:34 +02:00
|
|
|
Ok(()) => Ok(po.recv()),
|
|
|
|
Err(cause) => Err(cause)
|
2012-11-28 16:20:41 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Task construction */
|
|
|
|
|
2013-12-13 16:26:02 -08:00
|
|
|
impl TaskOpts {
|
|
|
|
pub fn new() -> TaskOpts {
|
|
|
|
/*!
|
|
|
|
* The default task options
|
|
|
|
*
|
|
|
|
* By default all tasks are supervised by their parent, are spawned
|
|
|
|
* into the same scheduler, and do not post lifecycle notifications.
|
|
|
|
*/
|
2012-11-28 16:20:41 -08:00
|
|
|
|
2013-12-13 16:26:02 -08:00
|
|
|
TaskOpts {
|
|
|
|
watched: true,
|
|
|
|
notify_chan: None,
|
|
|
|
name: None,
|
2014-01-06 10:26:11 -08:00
|
|
|
stack_size: None,
|
|
|
|
logger: None,
|
|
|
|
stdout: None,
|
|
|
|
stderr: None,
|
2013-12-13 16:26:02 -08:00
|
|
|
}
|
2012-11-28 16:20:41 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Spawn convenience functions */
|
|
|
|
|
2013-05-03 13:21:33 -07:00
|
|
|
/// Creates and executes a new child task
|
|
|
|
///
|
|
|
|
/// Sets up a new task with its own call stack and schedules it to run
|
|
|
|
/// the provided unique closure.
|
|
|
|
///
|
|
|
|
/// This function is equivalent to `task().spawn(f)`.
|
2013-11-18 13:25:09 -08:00
|
|
|
pub fn spawn(f: proc()) {
|
2013-11-21 16:55:40 -08:00
|
|
|
let task = task();
|
2013-05-03 13:21:33 -07:00
|
|
|
task.spawn(f)
|
2012-11-28 16:20:41 -08:00
|
|
|
}
|
|
|
|
|
2013-11-18 13:25:09 -08:00
|
|
|
pub fn try<T:Send>(f: proc() -> T) -> Result<T, ~Any> {
|
2012-11-28 16:20:41 -08:00
|
|
|
/*!
|
|
|
|
* Execute a function in another task and return either the return value
|
|
|
|
* of the function or result::err.
|
|
|
|
*
|
|
|
|
* This is equivalent to task().supervised().try.
|
|
|
|
*/
|
|
|
|
|
2013-11-21 16:55:40 -08:00
|
|
|
let task = task();
|
2013-05-03 13:21:33 -07:00
|
|
|
task.try(f)
|
2012-11-28 16:20:41 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Lifecycle functions */
|
|
|
|
|
2013-07-30 19:20:59 -04:00
|
|
|
/// Read the name of the current task.
|
2013-11-18 21:15:42 -08:00
|
|
|
pub fn with_task_name<U>(blk: |Option<&str>| -> U) -> U {
|
2013-07-30 19:20:59 -04:00
|
|
|
use rt::task::Task;
|
|
|
|
|
2013-12-12 18:01:59 -08:00
|
|
|
let mut task = Local::borrow(None::<Task>);
|
|
|
|
match task.get().name {
|
|
|
|
Some(ref name) => blk(Some(name.as_slice())),
|
|
|
|
None => blk(None)
|
2013-07-30 19:20:59 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-16 12:49:40 -07:00
|
|
|
pub fn deschedule() {
|
2012-11-28 16:20:41 -08:00
|
|
|
//! Yield control to the task scheduler
|
|
|
|
|
2013-07-08 18:06:17 -07:00
|
|
|
use rt::local::Local;
|
|
|
|
|
2013-07-31 23:12:20 -07:00
|
|
|
// FIXME(#7544): Optimize this, since we know we won't block.
|
2013-12-12 18:01:59 -08:00
|
|
|
let task: ~Task = Local::take();
|
|
|
|
task.yield_now();
|
2012-11-28 16:20:41 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn failing() -> bool {
|
|
|
|
//! True if the running task has failed
|
|
|
|
|
2013-05-19 16:50:21 -07:00
|
|
|
use rt::task::Task;
|
2013-04-22 17:15:31 -07:00
|
|
|
|
2013-12-03 19:18:58 -08:00
|
|
|
let mut local = Local::borrow(None::<Task>);
|
2013-12-12 18:01:59 -08:00
|
|
|
local.get().unwinder.unwinding()
|
2012-11-28 16:20:41 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
// The following 8 tests test the following 2^3 combinations:
|
|
|
|
// {un,}linked {un,}supervised failure propagation {up,down}wards.
|
|
|
|
|
|
|
|
// !!! These tests are dangerous. If Something is buggy, they will hang, !!!
|
|
|
|
// !!! instead of exiting cleanly. This might wedge the buildbots. !!!
|
|
|
|
|
2013-07-30 19:20:59 -04:00
|
|
|
#[test]
|
|
|
|
fn test_unnamed_task() {
|
2014-01-26 22:42:26 -05:00
|
|
|
spawn(proc() {
|
2013-12-12 21:38:57 -08:00
|
|
|
with_task_name(|name| {
|
|
|
|
assert!(name.is_none());
|
|
|
|
})
|
2014-01-26 22:42:26 -05:00
|
|
|
})
|
2013-07-30 19:20:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2013-10-05 21:01:58 +02:00
|
|
|
fn test_owned_named_task() {
|
2013-12-12 21:38:57 -08:00
|
|
|
let mut t = task();
|
|
|
|
t.name(~"ada lovelace");
|
2014-01-26 22:42:26 -05:00
|
|
|
t.spawn(proc() {
|
2013-12-12 21:38:57 -08:00
|
|
|
with_task_name(|name| {
|
|
|
|
assert!(name.unwrap() == "ada lovelace");
|
|
|
|
})
|
2014-01-26 22:42:26 -05:00
|
|
|
})
|
2013-07-30 19:20:59 -04:00
|
|
|
}
|
|
|
|
|
2013-10-05 21:01:58 +02:00
|
|
|
#[test]
|
|
|
|
fn test_static_named_task() {
|
2013-12-12 21:38:57 -08:00
|
|
|
let mut t = task();
|
|
|
|
t.name("ada lovelace");
|
2014-01-26 22:42:26 -05:00
|
|
|
t.spawn(proc() {
|
2013-12-12 21:38:57 -08:00
|
|
|
with_task_name(|name| {
|
|
|
|
assert!(name.unwrap() == "ada lovelace");
|
|
|
|
})
|
2014-01-26 22:42:26 -05:00
|
|
|
})
|
2013-10-05 21:01:58 +02:00
|
|
|
}
|
|
|
|
|
2013-10-05 21:23:24 +02:00
|
|
|
#[test]
|
|
|
|
fn test_send_named_task() {
|
2013-12-12 21:38:57 -08:00
|
|
|
let mut t = task();
|
2014-02-07 16:36:59 -08:00
|
|
|
t.name("ada lovelace".into_maybe_owned());
|
2014-01-26 22:42:26 -05:00
|
|
|
t.spawn(proc() {
|
2013-12-12 21:38:57 -08:00
|
|
|
with_task_name(|name| {
|
|
|
|
assert!(name.unwrap() == "ada lovelace");
|
|
|
|
})
|
2014-01-26 22:42:26 -05:00
|
|
|
})
|
2013-10-05 21:23:24 +02:00
|
|
|
}
|
|
|
|
|
2012-11-28 16:20:41 -08:00
|
|
|
#[test]
|
|
|
|
fn test_run_basic() {
|
2013-12-05 18:19:06 -08:00
|
|
|
let (po, ch) = Chan::new();
|
2014-01-26 22:42:26 -05:00
|
|
|
task().spawn(proc() {
|
2013-01-22 12:38:08 -08:00
|
|
|
ch.send(());
|
2014-01-26 22:42:26 -05:00
|
|
|
});
|
2013-01-22 12:38:08 -08:00
|
|
|
po.recv();
|
2012-11-28 16:20:41 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_add_wrapper() {
|
2013-12-05 18:19:06 -08:00
|
|
|
let (po, ch) = Chan::new();
|
2013-05-07 17:57:58 -07:00
|
|
|
let mut b0 = task();
|
2014-01-26 22:42:26 -05:00
|
|
|
b0.add_wrapper(proc(body) {
|
2013-12-03 16:44:16 -08:00
|
|
|
let ch = ch;
|
2013-11-21 23:36:52 -08:00
|
|
|
let result: proc() = proc() {
|
2012-11-28 16:20:41 -08:00
|
|
|
body();
|
2013-01-22 12:38:08 -08:00
|
|
|
ch.send(());
|
2013-03-01 14:32:37 -08:00
|
|
|
};
|
|
|
|
result
|
2014-01-26 22:42:26 -05:00
|
|
|
});
|
|
|
|
b0.spawn(proc() { });
|
2013-01-22 12:38:08 -08:00
|
|
|
po.recv();
|
2012-11-28 16:20:41 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_future_result() {
|
2013-05-07 17:57:58 -07:00
|
|
|
let mut builder = task();
|
2013-10-18 10:38:46 +02:00
|
|
|
let result = builder.future_result();
|
2014-01-26 22:42:26 -05:00
|
|
|
builder.spawn(proc() {});
|
2013-10-11 23:20:34 +02:00
|
|
|
assert!(result.recv().is_ok());
|
2012-11-28 16:20:41 -08:00
|
|
|
|
2013-05-07 17:57:58 -07:00
|
|
|
let mut builder = task();
|
2013-10-18 10:38:46 +02:00
|
|
|
let result = builder.future_result();
|
2014-01-26 22:42:26 -05:00
|
|
|
builder.spawn(proc() {
|
2013-10-21 13:08:31 -07:00
|
|
|
fail!();
|
2014-01-26 22:42:26 -05:00
|
|
|
});
|
2013-10-11 23:20:34 +02:00
|
|
|
assert!(result.recv().is_err());
|
2012-11-28 16:20:41 -08:00
|
|
|
}
|
|
|
|
|
2013-08-19 15:40:37 -07:00
|
|
|
#[test] #[should_fail]
|
2012-11-28 16:20:41 -08:00
|
|
|
fn test_back_to_the_future_result() {
|
2013-05-07 17:57:58 -07:00
|
|
|
let mut builder = task();
|
2013-10-18 10:38:46 +02:00
|
|
|
builder.future_result();
|
|
|
|
builder.future_result();
|
2012-11-28 16:20:41 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_try_success() {
|
2014-01-26 22:42:26 -05:00
|
|
|
match try(proc() {
|
2012-11-28 16:20:41 -08:00
|
|
|
~"Success!"
|
2014-01-26 22:42:26 -05:00
|
|
|
}) {
|
2012-11-28 16:20:41 -08:00
|
|
|
result::Ok(~"Success!") => (),
|
2013-10-21 13:08:31 -07:00
|
|
|
_ => fail!()
|
2012-11-28 16:20:41 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_try_fail() {
|
2014-01-26 22:42:26 -05:00
|
|
|
match try(proc() {
|
2013-10-21 13:08:31 -07:00
|
|
|
fail!()
|
2014-01-26 22:42:26 -05:00
|
|
|
}) {
|
2013-10-11 23:20:34 +02:00
|
|
|
result::Err(_) => (),
|
2013-10-21 13:08:31 -07:00
|
|
|
result::Ok(()) => fail!()
|
2012-11-28 16:20:41 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_spawn_sched() {
|
2013-12-12 21:38:57 -08:00
|
|
|
use clone::Clone;
|
|
|
|
|
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 (po, ch) = Chan::new();
|
2012-11-28 16:20:41 -08:00
|
|
|
|
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
|
|
|
fn f(i: int, ch: Chan<()>) {
|
2013-12-12 21:38:57 -08:00
|
|
|
let ch = ch.clone();
|
2014-01-26 22:42:26 -05:00
|
|
|
spawn(proc() {
|
2014-01-19 19:21:14 +11:00
|
|
|
if i == 0 {
|
2013-01-22 12:38:08 -08:00
|
|
|
ch.send(());
|
2012-11-28 16:20:41 -08:00
|
|
|
} else {
|
2013-12-12 21:38:57 -08:00
|
|
|
f(i - 1, ch);
|
2012-11-28 16:20:41 -08:00
|
|
|
}
|
2014-01-26 22:42:26 -05:00
|
|
|
});
|
2012-11-28 16:20:41 -08:00
|
|
|
|
|
|
|
}
|
|
|
|
f(10, ch);
|
2013-01-22 12:38:08 -08:00
|
|
|
po.recv();
|
2012-11-28 16:20:41 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2013-01-08 19:46:12 -08:00
|
|
|
fn test_spawn_sched_childs_on_default_sched() {
|
2013-12-05 18:19:06 -08:00
|
|
|
let (po, ch) = Chan::new();
|
2012-11-28 16:20:41 -08:00
|
|
|
|
2014-01-26 22:42:26 -05:00
|
|
|
spawn(proc() {
|
2013-12-03 16:44:16 -08:00
|
|
|
let ch = ch;
|
2014-01-26 22:42:26 -05:00
|
|
|
spawn(proc() {
|
2013-01-22 12:38:08 -08:00
|
|
|
ch.send(());
|
2014-01-26 22:42:26 -05:00
|
|
|
});
|
|
|
|
});
|
2012-11-28 16:20:41 -08:00
|
|
|
|
2013-01-22 12:38:08 -08:00
|
|
|
po.recv();
|
2012-11-28 16:20:41 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
2013-11-18 21:15:42 -08:00
|
|
|
fn avoid_copying_the_body(spawnfn: |v: proc()|) {
|
2013-12-05 18:19:06 -08:00
|
|
|
let (p, ch) = Chan::<uint>::new();
|
2012-11-28 16:20:41 -08:00
|
|
|
|
|
|
|
let x = ~1;
|
2013-04-22 14:27:30 -07:00
|
|
|
let x_in_parent = ptr::to_unsafe_ptr(&*x) as uint;
|
2012-11-28 16:20:41 -08:00
|
|
|
|
2014-01-26 22:42:26 -05:00
|
|
|
spawnfn(proc() {
|
2013-04-22 14:27:30 -07:00
|
|
|
let x_in_child = ptr::to_unsafe_ptr(&*x) as uint;
|
2013-01-22 12:38:08 -08:00
|
|
|
ch.send(x_in_child);
|
2014-01-26 22:42:26 -05:00
|
|
|
});
|
2012-11-28 16:20:41 -08:00
|
|
|
|
2013-01-22 12:38:08 -08:00
|
|
|
let x_in_child = p.recv();
|
2013-05-18 22:02:45 -04:00
|
|
|
assert_eq!(x_in_parent, x_in_child);
|
2012-11-28 16:20:41 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_avoid_copying_the_body_spawn() {
|
|
|
|
avoid_copying_the_body(spawn);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_avoid_copying_the_body_task_spawn() {
|
2013-11-21 17:23:21 -08:00
|
|
|
avoid_copying_the_body(|f| {
|
2013-11-21 16:55:40 -08:00
|
|
|
let builder = task();
|
2014-01-26 22:42:26 -05:00
|
|
|
builder.spawn(proc() {
|
2012-11-28 16:20:41 -08:00
|
|
|
f();
|
2014-01-26 22:42:26 -05:00
|
|
|
});
|
2013-11-21 17:23:21 -08:00
|
|
|
})
|
2012-11-28 16:20:41 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_avoid_copying_the_body_try() {
|
2013-11-21 17:23:21 -08:00
|
|
|
avoid_copying_the_body(|f| {
|
2014-01-30 14:10:53 -08:00
|
|
|
let _ = try(proc() {
|
2012-11-28 16:20:41 -08:00
|
|
|
f()
|
2014-01-26 22:42:26 -05:00
|
|
|
});
|
2013-11-21 17:23:21 -08:00
|
|
|
})
|
2012-11-28 16:20:41 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_child_doesnt_ref_parent() {
|
|
|
|
// If the child refcounts the parent task, this will stack overflow when
|
|
|
|
// climbing the task tree to dereference each ancestor. (See #1789)
|
|
|
|
// (well, it would if the constant were 8000+ - I lowered it to be more
|
|
|
|
// valgrind-friendly. try this at home, instead..!)
|
2013-03-22 14:00:15 -07:00
|
|
|
static generations: uint = 16;
|
2013-11-18 13:25:09 -08:00
|
|
|
fn child_no(x: uint) -> proc() {
|
2013-11-21 23:36:52 -08:00
|
|
|
return proc() {
|
2012-11-28 16:20:41 -08:00
|
|
|
if x < generations {
|
2013-07-26 17:20:26 -04:00
|
|
|
let mut t = task();
|
|
|
|
t.unwatched();
|
|
|
|
t.spawn(child_no(x+1));
|
2012-11-28 16:20:41 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-07-26 17:20:26 -04:00
|
|
|
let mut t = task();
|
|
|
|
t.unwatched();
|
|
|
|
t.spawn(child_no(0));
|
2012-11-28 16:20:41 -08:00
|
|
|
}
|
|
|
|
|
2013-04-18 18:38:12 -07:00
|
|
|
#[test]
|
|
|
|
fn test_simple_newsched_spawn() {
|
2013-12-12 21:38:57 -08:00
|
|
|
spawn(proc()())
|
2013-04-18 18:38:12 -07:00
|
|
|
}
|
2013-06-26 16:41:00 -07:00
|
|
|
|
2013-10-11 23:20:34 +02:00
|
|
|
#[test]
|
2013-10-27 20:12:40 +01:00
|
|
|
fn test_try_fail_message_static_str() {
|
2014-01-26 22:42:26 -05:00
|
|
|
match try(proc() {
|
2013-10-11 23:20:34 +02:00
|
|
|
fail!("static string");
|
2014-01-26 22:42:26 -05:00
|
|
|
}) {
|
2013-10-27 20:12:40 +01:00
|
|
|
Err(e) => {
|
|
|
|
type T = &'static str;
|
|
|
|
assert!(e.is::<T>());
|
|
|
|
assert_eq!(*e.move::<T>().unwrap(), "static string");
|
|
|
|
}
|
|
|
|
Ok(()) => fail!()
|
2013-10-11 23:20:34 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2013-10-27 20:12:40 +01:00
|
|
|
fn test_try_fail_message_owned_str() {
|
2014-01-26 22:42:26 -05:00
|
|
|
match try(proc() {
|
2013-10-11 23:20:34 +02:00
|
|
|
fail!(~"owned string");
|
2014-01-26 22:42:26 -05:00
|
|
|
}) {
|
2013-10-27 20:12:40 +01:00
|
|
|
Err(e) => {
|
|
|
|
type T = ~str;
|
|
|
|
assert!(e.is::<T>());
|
|
|
|
assert_eq!(*e.move::<T>().unwrap(), ~"owned string");
|
|
|
|
}
|
|
|
|
Ok(()) => fail!()
|
2013-10-11 23:20:34 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2013-10-27 20:12:40 +01:00
|
|
|
fn test_try_fail_message_any() {
|
2014-01-26 22:42:26 -05:00
|
|
|
match try(proc() {
|
2013-10-11 23:20:34 +02:00
|
|
|
fail!(~413u16 as ~Any);
|
2014-01-26 22:42:26 -05:00
|
|
|
}) {
|
2013-10-27 20:12:40 +01:00
|
|
|
Err(e) => {
|
|
|
|
type T = ~Any;
|
|
|
|
assert!(e.is::<T>());
|
|
|
|
let any = e.move::<T>().unwrap();
|
|
|
|
assert!(any.is::<u16>());
|
|
|
|
assert_eq!(*any.move::<u16>().unwrap(), 413u16);
|
|
|
|
}
|
|
|
|
Ok(()) => fail!()
|
2013-10-11 23:20:34 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2013-10-27 20:12:40 +01:00
|
|
|
fn test_try_fail_message_unit_struct() {
|
2013-10-11 23:20:34 +02:00
|
|
|
struct Juju;
|
|
|
|
|
2014-01-26 22:42:26 -05:00
|
|
|
match try(proc() {
|
2013-10-27 20:12:40 +01:00
|
|
|
fail!(Juju)
|
2014-01-26 22:42:26 -05:00
|
|
|
}) {
|
2013-10-11 23:20:34 +02:00
|
|
|
Err(ref e) if e.is::<Juju>() => {}
|
|
|
|
Err(_) | Ok(()) => fail!()
|
|
|
|
}
|
|
|
|
}
|