2013-02-03 20:15:43 -06: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.
|
|
|
|
|
2013-05-07 01:11:02 -05:00
|
|
|
/*! The Rust Runtime, including the task scheduler and I/O
|
|
|
|
|
|
|
|
The `rt` module provides the private runtime infrastructure necessary
|
|
|
|
to support core language features like the exchange and local heap,
|
|
|
|
the garbage collector, logging, local data and unwinding. It also
|
|
|
|
implements the default task scheduler and task model. Initialization
|
|
|
|
routines are provided for setting up runtime resources in common
|
|
|
|
configurations, including that used by `rustc` when generating
|
|
|
|
executables.
|
|
|
|
|
|
|
|
It is intended that the features provided by `rt` can be factored in a
|
|
|
|
way such that the core library can be built with different 'profiles'
|
|
|
|
for different use cases, e.g. excluding the task scheduler. A number
|
|
|
|
of runtime features though are critical to the functioning of the
|
|
|
|
language and an implementation must be provided regardless of the
|
|
|
|
execution environment.
|
|
|
|
|
|
|
|
Of foremost importance is the global exchange heap, in the module
|
|
|
|
`global_heap`. Very little practical Rust code can be written without
|
|
|
|
access to the global heap. Unlike most of `rt` the global heap is
|
|
|
|
truly a global resource and generally operates independently of the
|
|
|
|
rest of the runtime.
|
|
|
|
|
2013-05-19 03:04:01 -05:00
|
|
|
All other runtime features are task-local, including the local heap,
|
2013-05-07 01:11:02 -05:00
|
|
|
the garbage collector, local storage, logging and the stack unwinder.
|
|
|
|
|
|
|
|
The relationship between `rt` and the rest of the core library is
|
|
|
|
not entirely clear yet and some modules will be moving into or
|
|
|
|
out of `rt` as development proceeds.
|
|
|
|
|
|
|
|
Several modules in `core` are clients of `rt`:
|
|
|
|
|
|
|
|
* `core::task` - The user-facing interface to the Rust task model.
|
|
|
|
* `core::task::local_data` - The interface to local data.
|
|
|
|
* `core::gc` - The garbage collector.
|
|
|
|
* `core::unstable::lang` - Miscellaneous lang items, some of which rely on `core::rt`.
|
|
|
|
* `core::condition` - Uses local data.
|
|
|
|
* `core::cleanup` - Local heap destruction.
|
|
|
|
* `core::io` - In the future `core::io` will use an `rt` implementation.
|
|
|
|
* `core::logging`
|
|
|
|
* `core::pipes`
|
|
|
|
* `core::comm`
|
|
|
|
* `core::stackwalk`
|
|
|
|
|
|
|
|
*/
|
2013-04-23 21:21:37 -05:00
|
|
|
|
2013-03-24 20:59:04 -05:00
|
|
|
#[doc(hidden)];
|
2013-05-30 15:20:17 -05:00
|
|
|
#[deny(unused_imports)];
|
|
|
|
#[deny(unused_mut)];
|
|
|
|
#[deny(unused_variable)];
|
2013-06-23 16:15:39 -05:00
|
|
|
#[deny(unused_unsafe)];
|
2013-03-24 20:59:04 -05:00
|
|
|
|
2013-06-16 04:03:37 -05:00
|
|
|
use cell::Cell;
|
2013-06-18 02:17:14 -05:00
|
|
|
use clone::Clone;
|
|
|
|
use container::Container;
|
2013-08-05 15:06:24 -05:00
|
|
|
use iterator::{Iterator, IteratorUtil, range};
|
2013-07-17 18:48:46 -05:00
|
|
|
use option::{Some, None};
|
2013-06-03 12:50:29 -05:00
|
|
|
use ptr::RawPtr;
|
2013-07-19 16:25:05 -05:00
|
|
|
use rt::local::Local;
|
2013-06-26 18:41:00 -05:00
|
|
|
use rt::sched::{Scheduler, Shutdown};
|
2013-06-18 02:17:14 -05:00
|
|
|
use rt::sleeper_list::SleeperList;
|
2013-07-29 15:34:08 -05:00
|
|
|
use rt::task::{Task, SchedTask, GreenTask, Sched};
|
2013-06-18 02:17:14 -05:00
|
|
|
use rt::thread::Thread;
|
|
|
|
use rt::work_queue::WorkQueue;
|
|
|
|
use rt::uv::uvio::UvEventLoop;
|
2013-06-19 02:39:10 -05:00
|
|
|
use unstable::atomics::{AtomicInt, SeqCst};
|
|
|
|
use unstable::sync::UnsafeAtomicRcBox;
|
2013-06-18 02:17:14 -05:00
|
|
|
use vec::{OwnedVector, MutableVector};
|
2013-03-12 15:05:45 -05:00
|
|
|
|
2013-05-07 17:54:06 -05:00
|
|
|
/// The global (exchange) heap.
|
|
|
|
pub mod global_heap;
|
|
|
|
|
2013-05-19 03:04:01 -05:00
|
|
|
/// Implementations of language-critical runtime features like @.
|
|
|
|
pub mod task;
|
|
|
|
|
2013-07-02 20:15:34 -05:00
|
|
|
/// Facilities related to task failure, killing, and death.
|
|
|
|
mod kill;
|
|
|
|
|
2013-05-19 03:04:01 -05:00
|
|
|
/// The coroutine task scheduler, built on the `io` event loop.
|
2013-02-03 20:15:43 -06:00
|
|
|
mod sched;
|
2013-04-27 01:21:58 -05:00
|
|
|
|
2013-05-07 01:11:02 -05:00
|
|
|
/// Synchronous I/O.
|
2013-04-27 01:21:58 -05:00
|
|
|
pub mod io;
|
|
|
|
|
2013-05-07 01:11:02 -05:00
|
|
|
/// The EventLoop and internal synchronous I/O interface.
|
2013-03-13 20:21:47 -05:00
|
|
|
mod rtio;
|
2013-04-27 01:21:58 -05:00
|
|
|
|
2013-05-07 01:11:02 -05:00
|
|
|
/// libuv and default rtio implementation.
|
2013-04-27 02:09:27 -05:00
|
|
|
pub mod uv;
|
2013-04-27 01:21:58 -05:00
|
|
|
|
2013-05-19 03:13:53 -05:00
|
|
|
/// The Local trait for types that are accessible via thread-local
|
|
|
|
/// or task-local storage.
|
|
|
|
pub mod local;
|
|
|
|
|
2013-05-18 03:38:44 -05:00
|
|
|
/// A parallel work-stealing deque.
|
2013-02-03 20:15:43 -06:00
|
|
|
mod work_queue;
|
2013-04-27 01:21:58 -05:00
|
|
|
|
2013-05-18 03:38:44 -05:00
|
|
|
/// A parallel queue.
|
|
|
|
mod message_queue;
|
|
|
|
|
2013-05-28 20:39:52 -05:00
|
|
|
/// A parallel data structure for tracking sleeping schedulers.
|
|
|
|
mod sleeper_list;
|
|
|
|
|
2013-05-07 01:11:02 -05:00
|
|
|
/// Stack segments and caching.
|
2013-02-03 20:15:43 -06:00
|
|
|
mod stack;
|
2013-04-27 01:21:58 -05:00
|
|
|
|
2013-05-07 01:11:02 -05:00
|
|
|
/// CPU context swapping.
|
2013-02-03 20:15:43 -06:00
|
|
|
mod context;
|
2013-04-27 01:21:58 -05:00
|
|
|
|
2013-05-07 01:11:02 -05:00
|
|
|
/// Bindings to system threading libraries.
|
2013-02-03 20:15:43 -06:00
|
|
|
mod thread;
|
2013-04-27 01:21:58 -05:00
|
|
|
|
|
|
|
/// The runtime configuration, read from environment variables
|
2013-03-12 14:23:24 -05:00
|
|
|
pub mod env;
|
2013-04-27 01:21:58 -05:00
|
|
|
|
|
|
|
/// The local, managed heap
|
2013-06-22 03:09:06 -05:00
|
|
|
pub mod local_heap;
|
2013-03-15 20:06:19 -05:00
|
|
|
|
2013-04-27 20:57:15 -05:00
|
|
|
/// The Logger trait and implementations
|
|
|
|
pub mod logging;
|
|
|
|
|
2013-04-20 02:33:49 -05:00
|
|
|
/// Tools for testing the runtime
|
|
|
|
pub mod test;
|
|
|
|
|
2013-05-03 18:58:20 -05:00
|
|
|
/// Reference counting
|
|
|
|
pub mod rc;
|
|
|
|
|
2013-05-04 19:30:31 -05:00
|
|
|
/// A simple single-threaded channel type for passing buffered data between
|
|
|
|
/// scheduler and task context
|
|
|
|
pub mod tube;
|
|
|
|
|
2013-05-15 19:20:48 -05:00
|
|
|
/// Simple reimplementation of core::comm
|
|
|
|
pub mod comm;
|
|
|
|
|
2013-07-18 19:34:42 -05:00
|
|
|
/// Routines for select()ing on pipes.
|
|
|
|
pub mod select;
|
|
|
|
|
2013-05-19 16:39:46 -05:00
|
|
|
// FIXME #5248 shouldn't be pub
|
|
|
|
/// The runtime needs to be able to put a pointer into thread-local storage.
|
|
|
|
pub mod local_ptr;
|
|
|
|
|
|
|
|
// FIXME #5248: The import in `sched` doesn't resolve unless this is pub!
|
|
|
|
/// Bindings to pthread/windows thread-local storage.
|
|
|
|
pub mod thread_local_storage;
|
|
|
|
|
2013-05-30 00:38:15 -05:00
|
|
|
pub mod metrics;
|
|
|
|
|
2013-06-18 00:17:51 -05:00
|
|
|
// FIXME #5248 shouldn't be pub
|
|
|
|
/// Just stuff
|
|
|
|
pub mod util;
|
2013-05-19 16:39:46 -05:00
|
|
|
|
2013-06-21 03:28:23 -05:00
|
|
|
// Global command line argument storage
|
|
|
|
pub mod args;
|
|
|
|
|
2013-06-22 18:52:40 -05:00
|
|
|
// Support for dynamic borrowck
|
|
|
|
pub mod borrowck;
|
|
|
|
|
2013-04-27 01:21:58 -05:00
|
|
|
/// Set up a default runtime configuration, given compiler-supplied arguments.
|
|
|
|
///
|
|
|
|
/// This is invoked by the `start` _language item_ (unstable::lang) to
|
|
|
|
/// run a Rust executable.
|
|
|
|
///
|
|
|
|
/// # Arguments
|
|
|
|
///
|
|
|
|
/// * `argc` & `argv` - The argument vector. On Unix this information is used
|
|
|
|
/// by os::args.
|
|
|
|
/// * `crate_map` - Runtime information about the executing crate, mostly for logging
|
|
|
|
///
|
|
|
|
/// # Return value
|
|
|
|
///
|
|
|
|
/// The return value is used as the process return code. 0 on success, 101 on error.
|
2013-06-21 03:28:23 -05:00
|
|
|
pub fn start(argc: int, argv: **u8, crate_map: *u8, main: ~fn()) -> int {
|
2013-05-08 01:01:02 -05:00
|
|
|
|
2013-06-21 03:28:23 -05:00
|
|
|
init(argc, argv, crate_map);
|
2013-06-19 02:39:10 -05:00
|
|
|
let exit_code = run(main);
|
2013-06-18 01:18:20 -05:00
|
|
|
cleanup();
|
|
|
|
|
2013-06-19 02:39:10 -05:00
|
|
|
return exit_code;
|
2013-03-30 21:59:21 -05:00
|
|
|
}
|
2013-03-27 17:24:50 -05:00
|
|
|
|
2013-07-17 18:48:46 -05:00
|
|
|
/// Like `start` but creates an additional scheduler on the current thread,
|
|
|
|
/// which in most cases will be the 'main' thread, and pins the main task to it.
|
|
|
|
///
|
|
|
|
/// This is appropriate for running code that must execute on the main thread,
|
|
|
|
/// such as the platform event loop and GUI.
|
|
|
|
pub fn start_on_main_thread(argc: int, argv: **u8, crate_map: *u8, main: ~fn()) -> int {
|
|
|
|
init(argc, argv, crate_map);
|
|
|
|
let exit_code = run_on_main_thread(main);
|
|
|
|
cleanup();
|
|
|
|
|
|
|
|
return exit_code;
|
|
|
|
}
|
|
|
|
|
2013-06-21 03:28:23 -05:00
|
|
|
/// One-time runtime initialization.
|
|
|
|
///
|
|
|
|
/// Initializes global state, including frobbing
|
|
|
|
/// the crate's logging flags, registering GC
|
|
|
|
/// metadata, and storing the process arguments.
|
|
|
|
pub fn init(argc: int, argv: **u8, crate_map: *u8) {
|
|
|
|
// XXX: Derefing these pointers is not safe.
|
|
|
|
// Need to propagate the unsafety to `start`.
|
|
|
|
unsafe {
|
|
|
|
args::init(argc, argv);
|
2013-08-05 14:43:33 -05:00
|
|
|
env::init();
|
2013-06-21 03:28:23 -05:00
|
|
|
logging::init(crate_map);
|
|
|
|
rust_update_gc_metadata(crate_map);
|
|
|
|
}
|
2013-06-19 18:08:07 -05:00
|
|
|
|
|
|
|
extern {
|
|
|
|
fn rust_update_gc_metadata(crate_map: *u8);
|
|
|
|
}
|
2013-05-08 18:53:40 -05:00
|
|
|
}
|
|
|
|
|
2013-06-19 03:08:47 -05:00
|
|
|
/// One-time runtime cleanup.
|
2013-06-18 01:18:20 -05:00
|
|
|
pub fn cleanup() {
|
2013-06-21 03:28:23 -05:00
|
|
|
args::cleanup();
|
2013-06-18 01:18:20 -05:00
|
|
|
}
|
|
|
|
|
2013-06-19 03:08:47 -05:00
|
|
|
/// Execute the main function in a scheduler.
|
|
|
|
///
|
|
|
|
/// Configures the runtime according to the environment, by default
|
|
|
|
/// using a task scheduler with the same number of threads as cores.
|
|
|
|
/// Returns a process exit code.
|
2013-06-19 02:39:10 -05:00
|
|
|
pub fn run(main: ~fn()) -> int {
|
2013-07-17 18:48:46 -05:00
|
|
|
run_(main, false)
|
|
|
|
}
|
2013-06-19 03:08:47 -05:00
|
|
|
|
2013-07-17 18:48:46 -05:00
|
|
|
pub fn run_on_main_thread(main: ~fn()) -> int {
|
|
|
|
run_(main, true)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn run_(main: ~fn(), use_main_sched: bool) -> int {
|
2013-06-19 02:39:10 -05:00
|
|
|
static DEFAULT_ERROR_CODE: int = 101;
|
|
|
|
|
2013-07-17 18:48:46 -05:00
|
|
|
let nscheds = util::default_sched_threads();
|
2013-06-18 02:17:14 -05:00
|
|
|
|
2013-07-29 15:34:08 -05:00
|
|
|
let main = Cell::new(main);
|
|
|
|
|
2013-08-05 15:06:24 -05:00
|
|
|
// The shared list of sleeping schedulers.
|
2013-06-18 02:17:14 -05:00
|
|
|
let sleepers = SleeperList::new();
|
2013-08-05 15:06:24 -05:00
|
|
|
|
|
|
|
// Create a work queue for each scheduler, ntimes. Create an extra
|
|
|
|
// for the main thread if that flag is set. We won't steal from it.
|
|
|
|
let mut work_queues = ~[];
|
|
|
|
for _ in range(0u, nscheds) {
|
|
|
|
let work_queue: WorkQueue<~Task> = WorkQueue::new();
|
|
|
|
work_queues.push(work_queue);
|
|
|
|
}
|
2013-06-18 02:17:14 -05:00
|
|
|
|
2013-06-19 03:08:47 -05:00
|
|
|
// The schedulers.
|
2013-06-18 02:17:14 -05:00
|
|
|
let mut scheds = ~[];
|
2013-06-19 03:08:47 -05:00
|
|
|
// Handles to the schedulers. When the main task ends these will be
|
|
|
|
// sent the Shutdown message to terminate the schedulers.
|
|
|
|
let mut handles = ~[];
|
2013-06-18 02:17:14 -05:00
|
|
|
|
2013-08-05 15:06:24 -05:00
|
|
|
for i in range(0u, nscheds) {
|
2013-07-31 15:52:22 -05:00
|
|
|
rtdebug!("inserting a regular scheduler");
|
|
|
|
|
2013-06-19 03:08:47 -05:00
|
|
|
// Every scheduler is driven by an I/O event loop.
|
2013-06-18 02:17:14 -05:00
|
|
|
let loop_ = ~UvEventLoop::new();
|
2013-08-05 15:06:24 -05:00
|
|
|
let mut sched = ~Scheduler::new(loop_,
|
|
|
|
work_queues[i].clone(),
|
|
|
|
work_queues.clone(),
|
|
|
|
sleepers.clone());
|
2013-06-18 02:17:14 -05:00
|
|
|
let handle = sched.make_handle();
|
|
|
|
|
|
|
|
scheds.push(sched);
|
2013-06-19 03:08:47 -05:00
|
|
|
handles.push(handle);
|
2013-06-18 02:17:14 -05:00
|
|
|
}
|
|
|
|
|
2013-07-17 18:48:46 -05:00
|
|
|
// If we need a main-thread task then create a main thread scheduler
|
|
|
|
// that will reject any task that isn't pinned to it
|
2013-07-29 15:34:08 -05:00
|
|
|
let main_sched = if use_main_sched {
|
|
|
|
|
|
|
|
// Create a friend handle.
|
|
|
|
let mut friend_sched = scheds.pop();
|
|
|
|
let friend_handle = friend_sched.make_handle();
|
|
|
|
scheds.push(friend_sched);
|
|
|
|
|
2013-08-05 15:06:24 -05:00
|
|
|
// This scheduler needs a queue that isn't part of the stealee
|
|
|
|
// set.
|
|
|
|
let work_queue = WorkQueue::new();
|
|
|
|
|
2013-07-17 18:48:46 -05:00
|
|
|
let main_loop = ~UvEventLoop::new();
|
|
|
|
let mut main_sched = ~Scheduler::new_special(main_loop,
|
2013-08-05 15:06:24 -05:00
|
|
|
work_queue,
|
|
|
|
work_queues.clone(),
|
2013-07-17 18:48:46 -05:00
|
|
|
sleepers.clone(),
|
2013-07-29 15:34:08 -05:00
|
|
|
false,
|
|
|
|
Some(friend_handle));
|
2013-07-17 18:48:46 -05:00
|
|
|
let main_handle = main_sched.make_handle();
|
|
|
|
handles.push(main_handle);
|
|
|
|
Some(main_sched)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
};
|
|
|
|
|
2013-06-19 03:08:47 -05:00
|
|
|
// Create a shared cell for transmitting the process exit
|
|
|
|
// code from the main task to this function.
|
2013-06-19 02:39:10 -05:00
|
|
|
let exit_code = UnsafeAtomicRcBox::new(AtomicInt::new(0));
|
|
|
|
let exit_code_clone = exit_code.clone();
|
|
|
|
|
2013-06-19 03:08:47 -05:00
|
|
|
// When the main task exits, after all the tasks in the main
|
|
|
|
// task tree, shut down the schedulers and set the exit code.
|
2013-06-18 02:17:14 -05:00
|
|
|
let handles = Cell::new(handles);
|
2013-06-19 02:39:10 -05:00
|
|
|
let on_exit: ~fn(bool) = |exit_success| {
|
2013-06-18 02:17:14 -05:00
|
|
|
|
|
|
|
let mut handles = handles.take();
|
2013-08-03 11:45:23 -05:00
|
|
|
for handle in handles.mut_iter() {
|
2013-06-18 02:17:14 -05:00
|
|
|
handle.send(Shutdown);
|
|
|
|
}
|
|
|
|
|
2013-06-19 02:39:10 -05:00
|
|
|
unsafe {
|
2013-07-09 15:29:05 -05:00
|
|
|
let exit_code = if exit_success {
|
|
|
|
use rt::util;
|
|
|
|
|
|
|
|
// If we're exiting successfully, then return the global
|
|
|
|
// exit status, which can be set programmatically.
|
|
|
|
util::get_exit_status()
|
|
|
|
} else {
|
|
|
|
DEFAULT_ERROR_CODE
|
|
|
|
};
|
2013-06-19 02:39:10 -05:00
|
|
|
(*exit_code_clone.get()).store(exit_code, SeqCst);
|
|
|
|
}
|
2013-06-18 02:17:14 -05:00
|
|
|
};
|
2013-06-19 03:08:47 -05:00
|
|
|
|
2013-06-18 02:17:14 -05:00
|
|
|
let mut threads = ~[];
|
2013-07-19 16:25:05 -05:00
|
|
|
|
2013-07-29 15:34:08 -05:00
|
|
|
let on_exit = Cell::new(on_exit);
|
|
|
|
|
2013-07-19 16:25:05 -05:00
|
|
|
if !use_main_sched {
|
|
|
|
|
|
|
|
// In the case where we do not use a main_thread scheduler we
|
|
|
|
// run the main task in one of our threads.
|
2013-07-29 15:34:08 -05:00
|
|
|
|
2013-08-05 15:10:08 -05:00
|
|
|
let mut main_task = ~Task::new_root(&mut scheds[0].stack_pool, None, main.take());
|
2013-07-29 15:34:08 -05:00
|
|
|
main_task.death.on_exit = Some(on_exit.take());
|
2013-07-19 16:25:05 -05:00
|
|
|
let main_task_cell = Cell::new(main_task);
|
|
|
|
|
2013-06-18 02:17:14 -05:00
|
|
|
let sched = scheds.pop();
|
|
|
|
let sched_cell = Cell::new(sched);
|
|
|
|
let thread = do Thread::start {
|
|
|
|
let sched = sched_cell.take();
|
2013-07-19 16:25:05 -05:00
|
|
|
sched.bootstrap(main_task_cell.take());
|
2013-06-18 02:17:14 -05:00
|
|
|
};
|
|
|
|
threads.push(thread);
|
|
|
|
}
|
|
|
|
|
2013-07-19 16:25:05 -05:00
|
|
|
// Run each remaining scheduler in a thread.
|
|
|
|
while !scheds.is_empty() {
|
2013-07-31 15:52:22 -05:00
|
|
|
rtdebug!("creating regular schedulers");
|
2013-07-19 16:25:05 -05:00
|
|
|
let sched = scheds.pop();
|
|
|
|
let sched_cell = Cell::new(sched);
|
|
|
|
let thread = do Thread::start {
|
|
|
|
let mut sched = sched_cell.take();
|
2013-08-05 15:10:08 -05:00
|
|
|
let bootstrap_task = ~do Task::new_root(&mut sched.stack_pool, None) || {
|
2013-07-19 16:25:05 -05:00
|
|
|
rtdebug!("boostraping a non-primary scheduler");
|
|
|
|
};
|
|
|
|
sched.bootstrap(bootstrap_task);
|
|
|
|
};
|
|
|
|
threads.push(thread);
|
2013-07-17 18:48:46 -05:00
|
|
|
}
|
|
|
|
|
2013-07-19 16:25:05 -05:00
|
|
|
// If we do have a main thread scheduler, run it now.
|
2013-07-29 15:34:08 -05:00
|
|
|
|
2013-07-19 16:25:05 -05:00
|
|
|
if use_main_sched {
|
2013-07-29 15:34:08 -05:00
|
|
|
|
2013-07-31 15:52:22 -05:00
|
|
|
rtdebug!("about to create the main scheduler task");
|
|
|
|
|
2013-08-03 18:59:24 -05:00
|
|
|
let mut main_sched = main_sched.unwrap();
|
2013-07-29 15:34:08 -05:00
|
|
|
|
2013-07-19 16:25:05 -05:00
|
|
|
let home = Sched(main_sched.make_handle());
|
2013-08-05 15:10:08 -05:00
|
|
|
let mut main_task = ~Task::new_root_homed(&mut main_sched.stack_pool, None,
|
2013-07-29 15:34:08 -05:00
|
|
|
home, main.take());
|
|
|
|
main_task.death.on_exit = Some(on_exit.take());
|
2013-08-05 15:06:24 -05:00
|
|
|
rtdebug!("bootstrapping main_task");
|
2013-07-31 15:52:22 -05:00
|
|
|
|
2013-07-29 15:34:08 -05:00
|
|
|
main_sched.bootstrap(main_task);
|
2013-07-19 16:25:05 -05:00
|
|
|
}
|
2013-07-29 15:34:08 -05:00
|
|
|
|
2013-07-31 15:52:22 -05:00
|
|
|
rtdebug!("waiting for threads");
|
|
|
|
|
2013-06-18 02:17:14 -05:00
|
|
|
// Wait for schedulers
|
2013-08-03 11:45:23 -05:00
|
|
|
for thread in threads.consume_iter() {
|
2013-07-27 14:05:15 -05:00
|
|
|
thread.join();
|
|
|
|
}
|
2013-06-19 02:39:10 -05:00
|
|
|
|
2013-06-19 03:08:47 -05:00
|
|
|
// Return the exit code
|
2013-06-19 02:39:10 -05:00
|
|
|
unsafe {
|
|
|
|
(*exit_code.get()).load(SeqCst)
|
|
|
|
}
|
2013-06-18 02:17:14 -05:00
|
|
|
}
|
|
|
|
|
2013-03-27 17:24:50 -05:00
|
|
|
/// Possible contexts in which Rust code may be executing.
|
|
|
|
/// Different runtime services are available depending on context.
|
2013-04-27 01:21:58 -05:00
|
|
|
/// Mostly used for determining if we're using the new scheduler
|
|
|
|
/// or the old scheduler.
|
2013-03-27 17:24:50 -05:00
|
|
|
#[deriving(Eq)]
|
|
|
|
pub enum RuntimeContext {
|
|
|
|
// Running in an old-style task
|
2013-08-09 03:15:31 -05:00
|
|
|
OldTaskContext,
|
|
|
|
// Not old task context
|
|
|
|
NewRtContext
|
2013-03-27 17:24:50 -05:00
|
|
|
}
|
|
|
|
|
2013-04-27 01:21:58 -05:00
|
|
|
/// Determine the current RuntimeContext
|
2013-03-27 17:24:50 -05:00
|
|
|
pub fn context() -> RuntimeContext {
|
|
|
|
|
|
|
|
use task::rt::rust_task;
|
|
|
|
|
|
|
|
if unsafe { rust_try_get_task().is_not_null() } {
|
|
|
|
return OldTaskContext;
|
2013-07-19 16:25:05 -05:00
|
|
|
} else {
|
2013-08-09 03:15:31 -05:00
|
|
|
return NewRtContext;
|
2013-03-27 17:24:50 -05:00
|
|
|
}
|
|
|
|
|
2013-07-18 21:08:57 -05:00
|
|
|
extern {
|
2013-03-27 17:24:50 -05:00
|
|
|
#[rust_stack]
|
2013-07-18 21:08:57 -05:00
|
|
|
pub fn rust_try_get_task() -> *rust_task;
|
2013-03-27 17:24:50 -05:00
|
|
|
}
|
|
|
|
}
|
2013-08-09 03:15:31 -05:00
|
|
|
|
|
|
|
pub fn in_sched_context() -> bool {
|
|
|
|
unsafe {
|
|
|
|
match Local::try_unsafe_borrow::<Task>() {
|
|
|
|
Some(task) => {
|
|
|
|
match (*task).task_type {
|
|
|
|
SchedTask => true,
|
|
|
|
_ => false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
None => false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn in_green_task_context() -> bool {
|
|
|
|
unsafe {
|
|
|
|
match Local::try_unsafe_borrow::<Task>() {
|
|
|
|
Some(task) => {
|
|
|
|
match (*task).task_type {
|
|
|
|
GreenTask(_) => true,
|
|
|
|
_ => false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
None => false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|