2013-04-20 00:33:49 -07: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-15 17:20:48 -07:00
|
|
|
use uint;
|
|
|
|
use option::*;
|
2013-04-23 15:11:28 -07:00
|
|
|
use cell::Cell;
|
2013-04-22 17:15:31 -07:00
|
|
|
use result::{Result, Ok, Err};
|
2013-04-20 02:41:30 -07:00
|
|
|
use super::io::net::ip::{IpAddr, Ipv4};
|
2013-05-19 01:04:01 -07:00
|
|
|
use rt::task::Task;
|
2013-05-15 17:20:48 -07:00
|
|
|
use rt::thread::Thread;
|
2013-05-19 15:45:39 -07:00
|
|
|
use rt::local::Local;
|
2013-04-20 02:41:30 -07:00
|
|
|
|
2013-04-20 15:55:07 -07:00
|
|
|
/// Creates a new scheduler in a new thread and runs a task in it,
|
2013-04-23 15:11:28 -07:00
|
|
|
/// then waits for the scheduler to exit. Failure of the task
|
|
|
|
/// will abort the process.
|
2013-04-20 00:33:49 -07:00
|
|
|
pub fn run_in_newsched_task(f: ~fn()) {
|
2013-05-15 17:20:48 -07:00
|
|
|
use super::sched::*;
|
2013-04-20 00:33:49 -07:00
|
|
|
use unstable::run_in_bare_thread;
|
2013-04-27 00:09:27 -07:00
|
|
|
use rt::uv::uvio::UvEventLoop;
|
2013-04-20 00:33:49 -07:00
|
|
|
|
2013-06-04 12:03:58 +02:00
|
|
|
let f = Cell::new(f);
|
2013-04-20 00:33:49 -07:00
|
|
|
|
|
|
|
do run_in_bare_thread {
|
|
|
|
let mut sched = ~UvEventLoop::new_scheduler();
|
2013-05-19 01:04:01 -07:00
|
|
|
let task = ~Coroutine::with_task(&mut sched.stack_pool,
|
2013-05-19 14:05:20 -07:00
|
|
|
~Task::without_unwinding(),
|
2013-05-19 01:04:01 -07:00
|
|
|
f.take());
|
2013-05-11 00:42:16 -07:00
|
|
|
sched.enqueue_task(task);
|
2013-04-20 00:33:49 -07:00
|
|
|
sched.run();
|
|
|
|
}
|
|
|
|
}
|
2013-04-20 01:16:06 -07:00
|
|
|
|
2013-04-23 15:11:28 -07:00
|
|
|
/// Test tasks will abort on failure instead of unwinding
|
|
|
|
pub fn spawntask(f: ~fn()) {
|
|
|
|
use super::sched::*;
|
|
|
|
|
2013-05-19 15:45:39 -07:00
|
|
|
let mut sched = Local::take::<Scheduler>();
|
2013-05-19 01:04:01 -07:00
|
|
|
let task = ~Coroutine::with_task(&mut sched.stack_pool,
|
2013-05-19 14:05:20 -07:00
|
|
|
~Task::without_unwinding(),
|
2013-05-19 01:04:01 -07:00
|
|
|
f);
|
2013-04-23 15:11:28 -07:00
|
|
|
do sched.switch_running_tasks_and_then(task) |task| {
|
2013-06-04 12:03:58 +02:00
|
|
|
let task = Cell::new(task);
|
2013-05-19 15:45:39 -07:00
|
|
|
let sched = Local::take::<Scheduler>();
|
2013-04-23 15:11:28 -07:00
|
|
|
sched.schedule_new_task(task.take());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Create a new task and run it right now. Aborts on failure
|
|
|
|
pub fn spawntask_immediately(f: ~fn()) {
|
2013-04-20 01:55:10 -07:00
|
|
|
use super::sched::*;
|
|
|
|
|
2013-05-19 15:45:39 -07:00
|
|
|
let mut sched = Local::take::<Scheduler>();
|
2013-05-19 01:04:01 -07:00
|
|
|
let task = ~Coroutine::with_task(&mut sched.stack_pool,
|
2013-05-19 14:05:20 -07:00
|
|
|
~Task::without_unwinding(),
|
2013-05-19 01:04:01 -07:00
|
|
|
f);
|
2013-04-20 01:55:10 -07:00
|
|
|
do sched.switch_running_tasks_and_then(task) |task| {
|
2013-06-04 12:03:58 +02:00
|
|
|
let task = Cell::new(task);
|
2013-05-19 15:45:39 -07:00
|
|
|
do Local::borrow::<Scheduler> |sched| {
|
2013-05-11 00:42:16 -07:00
|
|
|
sched.enqueue_task(task.take());
|
2013-04-20 01:55:10 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-06 14:28:16 -07:00
|
|
|
/// Create a new task and run it right now. Aborts on failure
|
|
|
|
pub fn spawntask_later(f: ~fn()) {
|
|
|
|
use super::sched::*;
|
|
|
|
|
2013-05-19 15:45:39 -07:00
|
|
|
let mut sched = Local::take::<Scheduler>();
|
2013-05-19 01:04:01 -07:00
|
|
|
let task = ~Coroutine::with_task(&mut sched.stack_pool,
|
2013-05-19 14:05:20 -07:00
|
|
|
~Task::without_unwinding(),
|
2013-05-19 01:04:01 -07:00
|
|
|
f);
|
2013-05-06 14:28:16 -07:00
|
|
|
|
2013-05-11 00:42:16 -07:00
|
|
|
sched.enqueue_task(task);
|
2013-05-19 15:45:39 -07:00
|
|
|
Local::put(sched);
|
2013-05-06 14:28:16 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Spawn a task and either run it immediately or run it later
|
|
|
|
pub fn spawntask_random(f: ~fn()) {
|
|
|
|
use super::sched::*;
|
|
|
|
use rand::{Rand, rng};
|
|
|
|
|
|
|
|
let mut rng = rng();
|
|
|
|
let run_now: bool = Rand::rand(&mut rng);
|
|
|
|
|
2013-05-19 15:45:39 -07:00
|
|
|
let mut sched = Local::take::<Scheduler>();
|
2013-05-19 01:04:01 -07:00
|
|
|
let task = ~Coroutine::with_task(&mut sched.stack_pool,
|
2013-05-19 14:05:20 -07:00
|
|
|
~Task::without_unwinding(),
|
2013-05-19 01:04:01 -07:00
|
|
|
f);
|
2013-05-06 14:28:16 -07:00
|
|
|
|
|
|
|
if run_now {
|
|
|
|
do sched.switch_running_tasks_and_then(task) |task| {
|
2013-06-04 12:03:58 +02:00
|
|
|
let task = Cell::new(task);
|
2013-05-19 15:45:39 -07:00
|
|
|
do Local::borrow::<Scheduler> |sched| {
|
2013-05-11 00:42:16 -07:00
|
|
|
sched.enqueue_task(task.take());
|
2013-05-06 14:28:16 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
2013-05-11 00:42:16 -07:00
|
|
|
sched.enqueue_task(task);
|
2013-05-19 15:45:39 -07:00
|
|
|
Local::put(sched);
|
2013-05-06 14:28:16 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-04-22 17:15:31 -07:00
|
|
|
/// Spawn a task and wait for it to finish, returning whether it completed successfully or failed
|
2013-04-23 15:11:28 -07:00
|
|
|
pub fn spawntask_try(f: ~fn()) -> Result<(), ()> {
|
2013-04-22 17:15:31 -07:00
|
|
|
use cell::Cell;
|
|
|
|
use super::sched::*;
|
|
|
|
use task;
|
|
|
|
use unstable::finally::Finally;
|
|
|
|
|
|
|
|
// Our status variables will be filled in from the scheduler context
|
|
|
|
let mut failed = false;
|
|
|
|
let failed_ptr: *mut bool = &mut failed;
|
|
|
|
|
|
|
|
// Switch to the scheduler
|
2013-06-04 12:03:58 +02:00
|
|
|
let f = Cell::new(Cell::new(f));
|
2013-05-19 15:45:39 -07:00
|
|
|
let sched = Local::take::<Scheduler>();
|
2013-04-22 17:15:31 -07:00
|
|
|
do sched.deschedule_running_task_and_then() |old_task| {
|
2013-06-04 12:03:58 +02:00
|
|
|
let old_task = Cell::new(old_task);
|
2013-04-22 17:15:31 -07:00
|
|
|
let f = f.take();
|
2013-05-19 15:45:39 -07:00
|
|
|
let mut sched = Local::take::<Scheduler>();
|
2013-05-12 15:26:19 -07:00
|
|
|
let new_task = ~do Coroutine::new(&mut sched.stack_pool) {
|
2013-04-22 17:15:31 -07:00
|
|
|
do (|| {
|
|
|
|
(f.take())()
|
|
|
|
}).finally {
|
|
|
|
// Check for failure then resume the parent task
|
|
|
|
unsafe { *failed_ptr = task::failing(); }
|
2013-05-19 15:45:39 -07:00
|
|
|
let sched = Local::take::<Scheduler>();
|
2013-04-22 17:15:31 -07:00
|
|
|
do sched.switch_running_tasks_and_then(old_task.take()) |new_task| {
|
2013-06-04 12:03:58 +02:00
|
|
|
let new_task = Cell::new(new_task);
|
2013-05-19 15:45:39 -07:00
|
|
|
do Local::borrow::<Scheduler> |sched| {
|
2013-05-11 00:42:16 -07:00
|
|
|
sched.enqueue_task(new_task.take());
|
2013-04-22 17:15:31 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
sched.resume_task_immediately(new_task);
|
|
|
|
}
|
|
|
|
|
|
|
|
if !failed { Ok(()) } else { Err(()) }
|
|
|
|
}
|
|
|
|
|
2013-05-15 17:20:48 -07:00
|
|
|
// Spawn a new task in a new scheduler and return a thread handle.
|
|
|
|
pub fn spawntask_thread(f: ~fn()) -> Thread {
|
|
|
|
use rt::sched::*;
|
|
|
|
use rt::uv::uvio::UvEventLoop;
|
|
|
|
|
2013-06-04 12:03:58 +02:00
|
|
|
let f = Cell::new(f);
|
2013-05-15 17:20:48 -07:00
|
|
|
let thread = do Thread::start {
|
|
|
|
let mut sched = ~UvEventLoop::new_scheduler();
|
2013-05-19 01:04:01 -07:00
|
|
|
let task = ~Coroutine::with_task(&mut sched.stack_pool,
|
2013-05-19 14:05:20 -07:00
|
|
|
~Task::without_unwinding(),
|
2013-05-19 01:04:01 -07:00
|
|
|
f.take());
|
2013-05-15 17:20:48 -07:00
|
|
|
sched.enqueue_task(task);
|
|
|
|
sched.run();
|
|
|
|
};
|
|
|
|
return thread;
|
|
|
|
}
|
|
|
|
|
2013-04-20 01:16:06 -07:00
|
|
|
/// Get a port number, starting at 9600, for use in tests
|
|
|
|
pub fn next_test_port() -> u16 {
|
|
|
|
unsafe {
|
|
|
|
return rust_dbg_next_port() as u16;
|
|
|
|
}
|
|
|
|
extern {
|
|
|
|
fn rust_dbg_next_port() -> ::libc::uintptr_t;
|
|
|
|
}
|
|
|
|
}
|
2013-04-20 02:41:30 -07:00
|
|
|
|
|
|
|
/// Get a unique localhost:port pair starting at 9600
|
|
|
|
pub fn next_test_ip4() -> IpAddr {
|
|
|
|
Ipv4(127, 0, 0, 1, next_test_port())
|
|
|
|
}
|
2013-05-15 17:20:48 -07:00
|
|
|
|
|
|
|
/// Get a constant that represents the number of times to repeat stress tests. Default 1.
|
|
|
|
pub fn stress_factor() -> uint {
|
|
|
|
use os::getenv;
|
|
|
|
|
|
|
|
match getenv("RUST_RT_STRESS") {
|
|
|
|
Some(val) => uint::from_str(val).get(),
|
|
|
|
None => 1
|
|
|
|
}
|
|
|
|
}
|