rust/src/libstd/rt/mod.rs

376 lines
11 KiB
Rust
Raw Normal View History

// 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-06 23:11:02 -07: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 01:04:01 -07:00
All other runtime features are task-local, including the local heap,
2013-05-06 23:11:02 -07: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 19:21:37 -07:00
#[doc(hidden)];
#[deny(unused_imports)];
#[deny(unused_mut)];
#[deny(unused_variable)];
2013-06-23 14:15:39 -07:00
#[deny(unused_unsafe)];
use cell::Cell;
use clone::Clone;
use container::Container;
use from_str::FromStr;
use iter::Times;
use iterator::IteratorUtil;
use option::{Some, None};
use os;
use ptr::RawPtr;
use rt::sched::{Scheduler, Coroutine, Shutdown};
use rt::sleeper_list::SleeperList;
use rt::task::Task;
use rt::thread::Thread;
use rt::work_queue::WorkQueue;
use rt::uv::uvio::UvEventLoop;
2013-06-19 00:39:10 -07:00
use unstable::atomics::{AtomicInt, SeqCst};
use unstable::sync::UnsafeAtomicRcBox;
use vec::{OwnedVector, MutableVector};
2013-03-12 13:05:45 -07:00
/// The global (exchange) heap.
pub mod global_heap;
2013-05-19 01:04:01 -07:00
/// Implementations of language-critical runtime features like @.
pub mod task;
/// The coroutine task scheduler, built on the `io` event loop.
mod sched;
2013-04-26 23:21:58 -07:00
2013-05-06 23:11:02 -07:00
/// Synchronous I/O.
2013-04-26 23:21:58 -07:00
#[path = "io/mod.rs"]
pub mod io;
2013-05-06 23:11:02 -07:00
/// The EventLoop and internal synchronous I/O interface.
mod rtio;
2013-04-26 23:21:58 -07:00
2013-05-06 23:11:02 -07:00
/// libuv and default rtio implementation.
#[path = "uv/mod.rs"]
pub mod uv;
2013-04-26 23:21:58 -07:00
2013-05-19 01:13:53 -07:00
/// The Local trait for types that are accessible via thread-local
/// or task-local storage.
pub mod local;
/// A parallel work-stealing deque.
mod work_queue;
2013-04-26 23:21:58 -07:00
/// A parallel queue.
mod message_queue;
/// A parallel data structure for tracking sleeping schedulers.
mod sleeper_list;
2013-05-06 23:11:02 -07:00
/// Stack segments and caching.
mod stack;
2013-04-26 23:21:58 -07:00
2013-05-06 23:11:02 -07:00
/// CPU context swapping.
mod context;
2013-04-26 23:21:58 -07:00
2013-05-06 23:11:02 -07:00
/// Bindings to system threading libraries.
mod thread;
2013-04-26 23:21:58 -07:00
/// The runtime configuration, read from environment variables
pub mod env;
2013-04-26 23:21:58 -07:00
/// The local, managed heap
pub mod local_heap;
/// The Logger trait and implementations
pub mod logging;
/// Tools for testing the runtime
pub mod test;
/// Reference counting
pub mod rc;
/// A simple single-threaded channel type for passing buffered data between
/// scheduler and task context
pub mod tube;
/// Simple reimplementation of core::comm
pub mod comm;
// 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;
/// For waiting on child tasks.
pub mod join_latch;
pub mod metrics;
// FIXME #5248 shouldn't be pub
/// Just stuff
pub mod util;
2013-06-21 01:28:23 -07:00
// Global command line argument storage
pub mod args;
// Support for dynamic borrowck
pub mod borrowck;
2013-04-26 23:21:58 -07: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 01:28:23 -07:00
pub fn start(argc: int, argv: **u8, crate_map: *u8, main: ~fn()) -> int {
2013-06-21 01:28:23 -07:00
init(argc, argv, crate_map);
2013-06-19 00:39:10 -07:00
let exit_code = run(main);
2013-06-17 23:18:20 -07:00
cleanup();
2013-06-19 00:39:10 -07:00
return exit_code;
}
2013-06-21 01:28:23 -07: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);
logging::init(crate_map);
rust_update_gc_metadata(crate_map);
}
2013-06-19 16:08:07 -07:00
extern {
fn rust_update_gc_metadata(crate_map: *u8);
}
2013-05-08 16:53:40 -07:00
}
/// One-time runtime cleanup.
2013-06-17 23:18:20 -07:00
pub fn cleanup() {
2013-06-21 01:28:23 -07:00
args::cleanup();
2013-06-17 23:18:20 -07:00
global_heap::cleanup();
}
/// 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 00:39:10 -07:00
pub fn run(main: ~fn()) -> int {
2013-06-19 00:39:10 -07:00
static DEFAULT_ERROR_CODE: int = 101;
let nthreads = match os::getenv("RUST_THREADS") {
Some(nstr) => FromStr::from_str(nstr).get(),
2013-06-23 14:15:39 -07:00
None => util::num_cpus()
};
// The shared list of sleeping schedulers. Schedulers wake each other
// occassionally to do new work.
let sleepers = SleeperList::new();
// The shared work queue. Temporary until work stealing is implemented.
let work_queue = WorkQueue::new();
// The schedulers.
let mut scheds = ~[];
// Handles to the schedulers. When the main task ends these will be
// sent the Shutdown message to terminate the schedulers.
let mut handles = ~[];
for nthreads.times {
// Every scheduler is driven by an I/O event loop.
let loop_ = ~UvEventLoop::new();
let mut sched = ~Scheduler::new(loop_, work_queue.clone(), sleepers.clone());
let handle = sched.make_handle();
scheds.push(sched);
handles.push(handle);
}
// Create a shared cell for transmitting the process exit
// code from the main task to this function.
2013-06-19 00:39:10 -07:00
let exit_code = UnsafeAtomicRcBox::new(AtomicInt::new(0));
let exit_code_clone = exit_code.clone();
// When the main task exits, after all the tasks in the main
// task tree, shut down the schedulers and set the exit code.
let handles = Cell::new(handles);
2013-06-19 00:39:10 -07:00
let on_exit: ~fn(bool) = |exit_success| {
let mut handles = handles.take();
for handles.mut_iter().advance |handle| {
handle.send(Shutdown);
}
2013-06-19 00:39:10 -07:00
unsafe {
let exit_code = if exit_success { 0 } else { DEFAULT_ERROR_CODE };
(*exit_code_clone.get()).store(exit_code, SeqCst);
}
};
// Create and enqueue the main task.
let main_cell = Cell::new(main);
let mut new_task = ~Task::new_root();
new_task.on_exit = Some(on_exit);
let main_task = ~Coroutine::with_task(&mut scheds[0].stack_pool,
new_task, main_cell.take());
scheds[0].enqueue_task(main_task);
// Run each scheduler in a thread.
let mut threads = ~[];
while !scheds.is_empty() {
let sched = scheds.pop();
let sched_cell = Cell::new(sched);
let thread = do Thread::start {
let sched = sched_cell.take();
sched.run();
};
threads.push(thread);
}
// Wait for schedulers
{ let _threads = threads; }
2013-06-19 00:39:10 -07:00
// Return the exit code
2013-06-19 00:39:10 -07:00
unsafe {
(*exit_code.get()).load(SeqCst)
}
}
/// Possible contexts in which Rust code may be executing.
/// Different runtime services are available depending on context.
2013-04-26 23:21:58 -07:00
/// Mostly used for determining if we're using the new scheduler
/// or the old scheduler.
#[deriving(Eq)]
pub enum RuntimeContext {
// Only the exchange heap is available
GlobalContext,
// The scheduler may be accessed
SchedulerContext,
// Full task services, e.g. local heap, unwinding
TaskContext,
// Running in an old-style task
OldTaskContext
}
2013-04-26 23:21:58 -07:00
/// Determine the current RuntimeContext
pub fn context() -> RuntimeContext {
use task::rt::rust_task;
use self::local::Local;
2013-05-19 16:10:44 -07:00
use self::sched::Scheduler;
// XXX: Hitting TLS twice to check if the scheduler exists
// then to check for the task is not good for perf
if unsafe { rust_try_get_task().is_not_null() } {
return OldTaskContext;
} else {
if Local::exists::<Scheduler>() {
let context = Cell::new_empty();
do Local::borrow::<Scheduler, ()> |sched| {
if sched.in_task_context() {
context.put_back(TaskContext);
} else {
context.put_back(SchedulerContext);
}
}
return context.take();
} else {
return GlobalContext;
}
}
pub extern {
#[rust_stack]
fn rust_try_get_task() -> *rust_task;
}
}
#[test]
fn test_context() {
use unstable::run_in_bare_thread;
2013-05-19 16:10:44 -07:00
use self::sched::{Scheduler, Coroutine};
use rt::local::Local;
use rt::test::new_test_uv_sched;
assert_eq!(context(), OldTaskContext);
do run_in_bare_thread {
assert_eq!(context(), GlobalContext);
let mut sched = ~new_test_uv_sched();
let task = ~do Coroutine::new_root(&mut sched.stack_pool) {
assert_eq!(context(), TaskContext);
let sched = Local::take::<Scheduler>();
do sched.deschedule_running_task_and_then() |sched, task| {
assert_eq!(context(), SchedulerContext);
sched.enqueue_task(task);
}
};
sched.enqueue_task(task);
sched.run();
}
}