// 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 or the MIT license // , at your // option. This file may not be copied, modified, or distributed // except according to those terms. /*! * 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())) * ~~~ */ use core::cast::copy_lifetime; use core::cast; use core::either::Either; use core::option; use core::pipes::{recv, oneshot, ChanOne, PortOne, send_one, recv_one}; use core::prelude::*; use core::task; #[doc = "The future type"] pub struct Future { /*priv*/ mut state: FutureState, } // FIXME(#2829) -- futures should not be copyable, because they close // over fn~'s that have pipes and so forth within! impl Future : Drop { fn finalize(&self) {} } priv enum FutureState { Pending(fn~() -> A), Evaluating, Forced(A) } /// Methods on the `future` type impl Future { fn get() -> A { //! Get the value of the future *(self.get_ref()) } } impl Future { pure fn get_ref(&self) -> &self/A { /*! * Executes the future's closure and then returns a borrowed * pointer to the result. The borrowed pointer lasts as long as * the future. */ unsafe { match self.state { Forced(ref mut v) => { return cast::transmute(v); } Evaluating => fail ~"Recursive forcing of future!", Pending(_) => {} } let mut state = Evaluating; self.state <-> state; match move state { Forced(_) | Evaluating => fail ~"Logic error.", Pending(move f) => { self.state = Forced(move f()); self.get_ref() } } } } } pub fn from_value(val: A) -> Future { /*! * Create a future from a value * * The value is immediately available and calling `get` later will * not block. */ Future {state: Forced(move val)} } pub fn from_port(port: PortOne) -> Future { /*! * 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(move port); do from_fn |move port| { let mut port_ = None; port_ <-> *port; let port = option::unwrap(move port_); match recv(move port) { oneshot::send(move data) => move data } } } pub fn from_fn(f: ~fn() -> A) -> Future { /*! * 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 {state: Pending(move f)} } pub fn spawn(blk: fn~() -> A) -> Future { /*! * 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. */ let (chan, port) = oneshot::init(); let chan = ~mut Some(move chan); do task::spawn |move blk, move chan| { let chan = option::swap_unwrap(&mut *chan); send_one(move chan, blk()); } return from_port(move port); } #[allow(non_implicitly_copyable_typarams)] pub mod test { use core::prelude::*; use future::*; use core::pipes::oneshot; use core::task; #[test] pub fn test_from_value() { let f = from_value(~"snail"); assert f.get() == ~"snail"; } #[test] pub fn test_from_port() { let (ch, po) = oneshot::init(); send_one(move ch, ~"whale"); let f = from_port(move po); assert f.get() == ~"whale"; } #[test] pub fn test_from_fn() { let f = from_fn(|| ~"brail"); assert f.get() == ~"brail"; } #[test] pub fn test_interface_get() { let f = from_value(~"fail"); assert f.get() == ~"fail"; } #[test] pub fn test_get_ref_method() { let f = from_value(22); assert *f.get_ref() == 22; } #[test] pub fn test_spawn() { let f = spawn(|| ~"bale"); assert f.get() == ~"bale"; } #[test] #[should_fail] #[ignore(cfg(target_os = "win32"))] pub fn test_futurefail() { let f = spawn(|| fail); let _x: ~str = f.get(); } #[test] pub fn test_sendable_future() { let expected = ~"schlorf"; let f = do spawn |copy expected| { copy expected }; do task::spawn |move f, move expected| { let actual = f.get(); assert actual == expected; } } }