rust/src/libcore/future.rs
2012-08-15 17:46:05 -07:00

195 lines
3.8 KiB
Rust

// NB: transitionary, de-mode-ing.
#[forbid(deprecated_mode)];
#[forbid(deprecated_pattern)];
/*!
* A type representing values that may be computed concurrently and
* operations for working with them.
*
* # Example
*
* ~~~
* let delayed_fib = future::spawn {|| fib(5000) };
* make_a_sandwich();
* io::println(fmt!{"fib(5000) = %?", delayed_fib.get()})
* ~~~
*/
import either::Either;
import pipes::recv;
export Future;
export extensions;
export from_value;
export from_port;
export from_fn;
export get;
export with;
export spawn;
// for task.rs
export future_pipe;
#[doc = "The future type"]
enum Future<A> = {
mut v: Either<@A, fn@() -> A>
};
/// Methods on the `future` type
impl<A:copy send> Future<A> {
fn get() -> A {
//! Get the value of the future
get(&self)
}
fn with<B>(blk: fn((&A)) -> B) -> B {
//! Work with the value without copying it
with(&self, blk)
}
}
fn from_value<A>(+val: A) -> Future<A> {
/*!
* Create a future from a value
*
* The value is immediately available and calling `get` later will
* not block.
*/
Future({
mut v: either::Left(@val)
})
}
macro_rules! move_it {
{$x:expr} => { unsafe { let y <- *ptr::addr_of($x); y } }
}
fn from_port<A:send>(+port: future_pipe::client::waiting<A>) -> Future<A> {
#[doc = "
Create a future from a port
The first time that the value is requested the task will block
waiting for the result to be received on the port.
"];
let port = ~mut some(port);
do from_fn |move port| {
let mut port_ = none;
port_ <-> *port;
let port = option::unwrap(port_);
match recv(port) {
future_pipe::completed(data) => move_it!{data}
}
}
}
fn from_fn<A>(f: fn@() -> A) -> Future<A> {
/*!
* Create a future from a function.
*
* The first time that the value is requested it will be retreived by
* calling the function. Note that this function is a local
* function. It is not spawned into another task.
*/
Future({
mut v: either::Right(f)
})
}
fn spawn<A:send>(+blk: fn~() -> A) -> Future<A> {
/*!
* Create a future from a unique closure.
*
* The closure will be run in a new task and its result used as the
* value of the future.
*/
from_port(pipes::spawn_service_recv(future_pipe::init, |ch| {
future_pipe::server::completed(ch, blk());
}))
}
fn get<A:copy>(future: &Future<A>) -> A {
//! Get the value of the future
do with(future) |v| { *v }
}
fn with<A,B>(future: &Future<A>, blk: fn((&A)) -> B) -> B {
//! Work with the value without copying it
let v = match copy future.v {
either::Left(v) => v,
either::Right(f) => {
let v = @f();
future.v = either::Left(v);
v
}
};
blk(v)
}
proto! future_pipe {
waiting:recv<T:send> {
completed(T) -> !
}
}
#[test]
fn test_from_value() {
let f = from_value(~"snail");
assert get(&f) == ~"snail";
}
#[test]
fn test_from_port() {
let (po, ch) = future_pipe::init();
future_pipe::server::completed(ch, ~"whale");
let f = from_port(po);
assert get(&f) == ~"whale";
}
#[test]
fn test_from_fn() {
let f = fn@() -> ~str { ~"brail" };
let f = from_fn(f);
assert get(&f) == ~"brail";
}
#[test]
fn test_interface_get() {
let f = from_value(~"fail");
assert f.get() == ~"fail";
}
#[test]
fn test_with() {
let f = from_value(~"nail");
assert with(&f, |v| *v) == ~"nail";
}
#[test]
fn test_interface_with() {
let f = from_value(~"kale");
assert f.with(|v| *v) == ~"kale";
}
#[test]
fn test_spawn() {
let f = spawn(|| ~"bale");
assert get(&f) == ~"bale";
}
#[test]
#[should_fail]
#[ignore(cfg(target_os = "win32"))]
fn test_futurefail() {
let f = spawn(|| fail);
let _x: ~str = get(&f);
}