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.
|
|
|
|
|
|
|
|
use option::*;
|
|
|
|
use sys;
|
|
|
|
use cast::transmute;
|
|
|
|
use libc::c_void;
|
|
|
|
use ptr::mut_null;
|
|
|
|
|
|
|
|
use super::work_queue::WorkQueue;
|
|
|
|
use super::stack::{StackPool, StackSegment};
|
2013-03-13 20:21:47 -05:00
|
|
|
use super::rtio::{EventLoop, EventLoopObject};
|
2013-02-03 20:15:43 -06:00
|
|
|
use super::context::Context;
|
|
|
|
use tls = super::thread_local_storage;
|
|
|
|
|
|
|
|
#[cfg(test)] use super::uvio::UvEventLoop;
|
|
|
|
#[cfg(test)] use unstable::run_in_bare_thread;
|
|
|
|
#[cfg(test)] use int;
|
|
|
|
|
|
|
|
/// The Scheduler is responsible for coordinating execution of Tasks
|
|
|
|
/// on a single thread. When the scheduler is running it is owned by
|
|
|
|
/// thread local storage and the running task is owned by the
|
|
|
|
/// scheduler.
|
|
|
|
pub struct Scheduler {
|
|
|
|
task_queue: WorkQueue<~Task>,
|
|
|
|
stack_pool: StackPool,
|
|
|
|
/// The event loop used to drive the scheduler and perform I/O
|
|
|
|
event_loop: ~EventLoopObject,
|
|
|
|
/// The scheduler's saved context.
|
|
|
|
/// Always valid when a task is executing, otherwise not
|
|
|
|
priv saved_context: Context,
|
|
|
|
/// The currently executing task
|
|
|
|
priv current_task: Option<~Task>,
|
|
|
|
/// A queue of jobs to perform immediately upon return from task
|
|
|
|
/// context to scheduler context.
|
|
|
|
/// XXX: This probably should be a single cleanup action and it
|
|
|
|
/// should run after a context switch, not on return from the
|
|
|
|
/// scheduler
|
|
|
|
priv cleanup_jobs: ~[CleanupJob]
|
|
|
|
}
|
|
|
|
|
|
|
|
// XXX: Some hacks to put a &fn in Scheduler without borrowck
|
|
|
|
// complaining
|
|
|
|
type UnsafeTaskReceiver = sys::Closure;
|
|
|
|
trait HackAroundBorrowCk {
|
2013-03-21 21:07:54 -05:00
|
|
|
fn from_fn(&fn(&mut Scheduler, ~Task)) -> Self;
|
2013-02-03 20:15:43 -06:00
|
|
|
fn to_fn(self) -> &fn(&mut Scheduler, ~Task);
|
|
|
|
}
|
|
|
|
impl HackAroundBorrowCk for UnsafeTaskReceiver {
|
2013-03-21 21:07:54 -05:00
|
|
|
fn from_fn(f: &fn(&mut Scheduler, ~Task)) -> UnsafeTaskReceiver {
|
2013-02-03 20:15:43 -06:00
|
|
|
unsafe { transmute(f) }
|
|
|
|
}
|
|
|
|
fn to_fn(self) -> &fn(&mut Scheduler, ~Task) {
|
|
|
|
unsafe { transmute(self) }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
enum CleanupJob {
|
|
|
|
RescheduleTask(~Task),
|
|
|
|
RecycleTask(~Task),
|
|
|
|
GiveTask(~Task, UnsafeTaskReceiver)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub impl Scheduler {
|
|
|
|
|
2013-03-24 17:37:59 -05:00
|
|
|
fn new(event_loop: ~EventLoopObject) -> Scheduler {
|
2013-03-15 20:35:33 -05:00
|
|
|
|
|
|
|
// Lazily initialize the global state, currently the scheduler TLS key
|
|
|
|
unsafe { rust_initialize_global_state(); }
|
|
|
|
extern {
|
|
|
|
fn rust_initialize_global_state();
|
|
|
|
}
|
|
|
|
|
2013-02-03 20:15:43 -06:00
|
|
|
Scheduler {
|
|
|
|
event_loop: event_loop,
|
|
|
|
task_queue: WorkQueue::new(),
|
|
|
|
stack_pool: StackPool::new(),
|
|
|
|
saved_context: Context::empty(),
|
|
|
|
current_task: None,
|
|
|
|
cleanup_jobs: ~[]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// XXX: This may eventually need to be refactored so that
|
|
|
|
// the scheduler itself doesn't have to call event_loop.run.
|
|
|
|
// That will be important for embedding the runtime into external
|
|
|
|
// event loops.
|
|
|
|
fn run(~self) -> ~Scheduler {
|
2013-03-28 20:39:09 -05:00
|
|
|
assert!(!self.in_task_context());
|
2013-02-03 20:15:43 -06:00
|
|
|
|
|
|
|
// Give ownership of the scheduler (self) to the thread
|
|
|
|
do self.install |scheduler| {
|
|
|
|
fn run_scheduler_once() {
|
|
|
|
do Scheduler::local |scheduler| {
|
|
|
|
if scheduler.resume_task_from_queue() {
|
|
|
|
// Ok, a task ran. Nice! We'll do it again later
|
|
|
|
scheduler.event_loop.callback(run_scheduler_once);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
scheduler.event_loop.callback(run_scheduler_once);
|
|
|
|
scheduler.event_loop.run();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn install(~self, f: &fn(&mut Scheduler)) -> ~Scheduler {
|
|
|
|
let mut tlsched = ThreadLocalScheduler::new();
|
|
|
|
tlsched.put_scheduler(self);
|
|
|
|
{
|
|
|
|
let sched = tlsched.get_scheduler();
|
|
|
|
f(sched);
|
|
|
|
}
|
|
|
|
return tlsched.take_scheduler();
|
|
|
|
}
|
|
|
|
|
2013-03-21 21:07:54 -05:00
|
|
|
fn local(f: &fn(&mut Scheduler)) {
|
2013-02-03 20:15:43 -06:00
|
|
|
let mut tlsched = ThreadLocalScheduler::new();
|
|
|
|
f(tlsched.get_scheduler());
|
|
|
|
}
|
|
|
|
|
|
|
|
// * Scheduler-context operations
|
|
|
|
|
|
|
|
fn resume_task_from_queue(&mut self) -> bool {
|
2013-03-28 20:39:09 -05:00
|
|
|
assert!(!self.in_task_context());
|
2013-02-03 20:15:43 -06:00
|
|
|
|
|
|
|
let mut self = self;
|
|
|
|
match self.task_queue.pop_front() {
|
|
|
|
Some(task) => {
|
|
|
|
self.resume_task_immediately(task);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
None => {
|
|
|
|
rtdebug!("no tasks in queue");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn resume_task_immediately(&mut self, task: ~Task) {
|
2013-03-28 20:39:09 -05:00
|
|
|
assert!(!self.in_task_context());
|
2013-02-03 20:15:43 -06:00
|
|
|
|
|
|
|
rtdebug!("scheduling a task");
|
|
|
|
|
|
|
|
// Store the task in the scheduler so it can be grabbed later
|
|
|
|
self.current_task = Some(task);
|
2013-04-11 19:34:52 -05:00
|
|
|
|
|
|
|
// Take pointers to both the task and scheduler's saved registers.
|
|
|
|
{
|
|
|
|
let (sched_context, _, next_task_context) = self.get_contexts();
|
|
|
|
let next_task_context = next_task_context.unwrap();
|
|
|
|
// Context switch to the task, restoring it's registers
|
|
|
|
// and saving the scheduler's
|
|
|
|
Context::swap(sched_context, next_task_context);
|
|
|
|
}
|
|
|
|
|
2013-02-03 20:15:43 -06:00
|
|
|
// The running task should have passed ownership elsewhere
|
2013-03-28 20:39:09 -05:00
|
|
|
assert!(self.current_task.is_none());
|
2013-02-03 20:15:43 -06:00
|
|
|
|
|
|
|
// Running tasks may have asked us to do some cleanup
|
|
|
|
self.run_cleanup_jobs();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// * Task-context operations
|
|
|
|
|
|
|
|
/// Called by a running task to end execution, after which it will
|
|
|
|
/// be recycled by the scheduler for reuse in a new task.
|
|
|
|
fn terminate_current_task(&mut self) {
|
2013-03-28 20:39:09 -05:00
|
|
|
assert!(self.in_task_context());
|
2013-02-03 20:15:43 -06:00
|
|
|
|
|
|
|
rtdebug!("ending running task");
|
|
|
|
|
|
|
|
let dead_task = self.current_task.swap_unwrap();
|
|
|
|
self.enqueue_cleanup_job(RecycleTask(dead_task));
|
2013-04-11 19:34:52 -05:00
|
|
|
{
|
|
|
|
let (sched_context, last_task_context, _) = self.get_contexts();
|
|
|
|
let last_task_context = last_task_context.unwrap();
|
|
|
|
Context::swap(last_task_context, sched_context);
|
|
|
|
}
|
2013-02-03 20:15:43 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Block a running task, context switch to the scheduler, then pass the
|
|
|
|
/// blocked task to a closure.
|
|
|
|
///
|
|
|
|
/// # Safety note
|
|
|
|
///
|
|
|
|
/// The closure here is a *stack* closure that lives in the
|
|
|
|
/// running task. It gets transmuted to the scheduler's lifetime
|
|
|
|
/// and called while the task is blocked.
|
2013-04-14 18:12:30 -05:00
|
|
|
fn deschedule_running_task_and_then(&mut self, f: &fn(&mut Scheduler, ~Task)) {
|
2013-03-28 20:39:09 -05:00
|
|
|
assert!(self.in_task_context());
|
2013-02-03 20:15:43 -06:00
|
|
|
|
|
|
|
rtdebug!("blocking task");
|
|
|
|
|
|
|
|
let blocked_task = self.current_task.swap_unwrap();
|
|
|
|
let f_fake_region = unsafe {
|
2013-03-13 19:58:23 -05:00
|
|
|
transmute::<&fn(&mut Scheduler, ~Task), &fn(&mut Scheduler, ~Task)>(f)
|
2013-02-03 20:15:43 -06:00
|
|
|
};
|
|
|
|
let f_opaque = HackAroundBorrowCk::from_fn(f_fake_region);
|
|
|
|
self.enqueue_cleanup_job(GiveTask(blocked_task, f_opaque));
|
2013-04-11 19:34:52 -05:00
|
|
|
{
|
|
|
|
let (sched_context, last_task_context, _) = self.get_contexts();
|
|
|
|
let last_task_context = last_task_context.unwrap();
|
|
|
|
Context::swap(last_task_context, sched_context);
|
|
|
|
}
|
2013-02-03 20:15:43 -06:00
|
|
|
|
2013-04-14 18:25:33 -05:00
|
|
|
self.run_cleanup_jobs();
|
2013-02-03 20:15:43 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Switch directly to another task, without going through the scheduler.
|
|
|
|
/// You would want to think hard about doing this, e.g. if there are
|
|
|
|
/// pending I/O events it would be a bad idea.
|
|
|
|
fn resume_task_from_running_task_direct(&mut self, next_task: ~Task) {
|
2013-03-28 20:39:09 -05:00
|
|
|
assert!(self.in_task_context());
|
2013-02-03 20:15:43 -06:00
|
|
|
|
|
|
|
rtdebug!("switching tasks");
|
|
|
|
|
|
|
|
let old_running_task = self.current_task.swap_unwrap();
|
|
|
|
self.enqueue_cleanup_job(RescheduleTask(old_running_task));
|
|
|
|
self.current_task = Some(next_task);
|
2013-04-11 19:34:52 -05:00
|
|
|
{
|
|
|
|
let (_, last_task_context, next_task_context) = self.get_contexts();
|
|
|
|
let last_task_context = last_task_context.unwrap();
|
|
|
|
let next_task_context = next_task_context.unwrap();
|
|
|
|
Context::swap(last_task_context, next_task_context);
|
|
|
|
}
|
2013-02-03 20:15:43 -06:00
|
|
|
|
2013-04-14 18:25:33 -05:00
|
|
|
self.run_cleanup_jobs();
|
2013-02-03 20:15:43 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// * Other stuff
|
|
|
|
|
|
|
|
fn in_task_context(&self) -> bool { self.current_task.is_some() }
|
|
|
|
|
|
|
|
fn enqueue_cleanup_job(&mut self, job: CleanupJob) {
|
|
|
|
self.cleanup_jobs.unshift(job);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn run_cleanup_jobs(&mut self) {
|
|
|
|
rtdebug!("running cleanup jobs");
|
|
|
|
|
|
|
|
while !self.cleanup_jobs.is_empty() {
|
|
|
|
match self.cleanup_jobs.pop() {
|
|
|
|
RescheduleTask(task) => {
|
|
|
|
// NB: Pushing to the *front* of the queue
|
|
|
|
self.task_queue.push_front(task);
|
|
|
|
}
|
|
|
|
RecycleTask(task) => task.recycle(&mut self.stack_pool),
|
|
|
|
GiveTask(task, f) => (f.to_fn())(self, task)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-11 19:34:52 -05:00
|
|
|
/// Get mutable references to all the contexts that may be involved in a
|
|
|
|
/// context switch.
|
|
|
|
///
|
|
|
|
/// Returns (the scheduler context, the optional context of the
|
|
|
|
/// task in the cleanup list, the optional context of the task in
|
|
|
|
/// the current task slot). When context switching to a task,
|
|
|
|
/// callers should first arrange for that task to be located in the
|
|
|
|
/// Scheduler's current_task slot and set up the
|
|
|
|
/// post-context-switch cleanup job.
|
|
|
|
fn get_contexts(&mut self) -> (&'self mut Context,
|
|
|
|
Option<&'self mut Context>,
|
|
|
|
Option<&'self mut Context>) {
|
|
|
|
let last_task = if !self.cleanup_jobs.is_empty() {
|
|
|
|
let last_job: &'self mut CleanupJob = &mut self.cleanup_jobs[0];
|
|
|
|
let last_task: &'self Task = match last_job {
|
|
|
|
&RescheduleTask(~ref task) => task,
|
|
|
|
&RecycleTask(~ref task) => task,
|
|
|
|
&GiveTask(~ref task, _) => task,
|
|
|
|
};
|
|
|
|
Some(last_task)
|
|
|
|
} else {
|
|
|
|
None
|
2013-02-03 20:15:43 -06:00
|
|
|
};
|
|
|
|
// XXX: Pattern matching mutable pointers above doesn't work
|
|
|
|
// because borrowck thinks the three patterns are conflicting
|
|
|
|
// borrows
|
2013-04-11 19:34:52 -05:00
|
|
|
let last_task = unsafe { transmute::<Option<&Task>, Option<&mut Task>>(last_task) };
|
|
|
|
let last_task_context = match last_task {
|
|
|
|
Some(ref t) => Some(&mut t.saved_context), None => None
|
|
|
|
};
|
|
|
|
let next_task_context = match self.current_task {
|
|
|
|
Some(ref mut t) => Some(&mut t.saved_context), None => None
|
|
|
|
};
|
|
|
|
return (&mut self.saved_context,
|
|
|
|
last_task_context,
|
|
|
|
next_task_context);
|
2013-02-03 20:15:43 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-22 16:00:15 -05:00
|
|
|
static TASK_MIN_STACK_SIZE: uint = 10000000; // XXX: Too much stack
|
2013-02-03 20:15:43 -06:00
|
|
|
|
|
|
|
pub struct Task {
|
|
|
|
/// The segment of stack on which the task is currently running or,
|
|
|
|
/// if the task is blocked, on which the task will resume execution
|
|
|
|
priv current_stack_segment: StackSegment,
|
|
|
|
/// These are always valid when the task is not running, unless
|
|
|
|
/// the task is dead
|
|
|
|
priv saved_context: Context,
|
|
|
|
}
|
|
|
|
|
2013-03-24 17:37:59 -05:00
|
|
|
pub impl Task {
|
|
|
|
fn new(stack_pool: &mut StackPool, start: ~fn()) -> Task {
|
2013-03-12 02:48:41 -05:00
|
|
|
let start = Task::build_start_wrapper(start);
|
2013-02-03 20:15:43 -06:00
|
|
|
let mut stack = stack_pool.take_segment(TASK_MIN_STACK_SIZE);
|
|
|
|
// NB: Context holds a pointer to that ~fn
|
2013-03-12 02:48:41 -05:00
|
|
|
let initial_context = Context::new(start, &mut stack);
|
2013-02-03 20:15:43 -06:00
|
|
|
return Task {
|
|
|
|
current_stack_segment: stack,
|
|
|
|
saved_context: initial_context,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2013-03-22 14:56:10 -05:00
|
|
|
priv fn build_start_wrapper(start: ~fn()) -> ~fn() {
|
2013-02-03 20:15:43 -06:00
|
|
|
// XXX: The old code didn't have this extra allocation
|
|
|
|
let wrapper: ~fn() = || {
|
2013-04-14 18:25:33 -05:00
|
|
|
// This is the first code to execute after the initial
|
|
|
|
// context switch to the task. The previous context may
|
|
|
|
// have asked us to do some cleanup.
|
|
|
|
let mut sched = ThreadLocalScheduler::new();
|
|
|
|
let sched = sched.get_scheduler();
|
|
|
|
sched.run_cleanup_jobs();
|
2013-04-11 19:34:52 -05:00
|
|
|
|
2013-02-03 20:15:43 -06:00
|
|
|
start();
|
|
|
|
|
|
|
|
let mut sched = ThreadLocalScheduler::new();
|
|
|
|
let sched = sched.get_scheduler();
|
|
|
|
sched.terminate_current_task();
|
|
|
|
};
|
|
|
|
return wrapper;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Destroy the task and try to reuse its components
|
|
|
|
fn recycle(~self, stack_pool: &mut StackPool) {
|
|
|
|
match self {
|
|
|
|
~Task {current_stack_segment, _} => {
|
|
|
|
stack_pool.give_segment(current_stack_segment);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// NB: This is a type so we can use make use of the &self region.
|
|
|
|
struct ThreadLocalScheduler(tls::Key);
|
|
|
|
|
|
|
|
impl ThreadLocalScheduler {
|
2013-03-21 21:07:54 -05:00
|
|
|
fn new() -> ThreadLocalScheduler {
|
2013-02-03 20:15:43 -06:00
|
|
|
unsafe {
|
|
|
|
// NB: This assumes that the TLS key has been created prior.
|
|
|
|
// Currently done in rust_start.
|
|
|
|
let key: *mut c_void = rust_get_sched_tls_key();
|
|
|
|
let key: &mut tls::Key = transmute(key);
|
|
|
|
ThreadLocalScheduler(*key)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn put_scheduler(&mut self, scheduler: ~Scheduler) {
|
|
|
|
unsafe {
|
|
|
|
let key = match self { &ThreadLocalScheduler(key) => key };
|
2013-03-13 19:58:23 -05:00
|
|
|
let value: *mut c_void = transmute::<~Scheduler, *mut c_void>(scheduler);
|
2013-02-03 20:15:43 -06:00
|
|
|
tls::set(key, value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-14 13:22:51 -05:00
|
|
|
fn get_scheduler(&mut self) -> &'self mut Scheduler {
|
2013-02-03 20:15:43 -06:00
|
|
|
unsafe {
|
|
|
|
let key = match self { &ThreadLocalScheduler(key) => key };
|
|
|
|
let mut value: *mut c_void = tls::get(key);
|
2013-03-28 20:39:09 -05:00
|
|
|
assert!(value.is_not_null());
|
2013-02-03 20:15:43 -06:00
|
|
|
{
|
|
|
|
let value_ptr = &mut value;
|
2013-03-13 19:58:23 -05:00
|
|
|
let sched: &mut ~Scheduler = {
|
|
|
|
transmute::<&mut *mut c_void, &mut ~Scheduler>(value_ptr)
|
|
|
|
};
|
2013-02-03 20:15:43 -06:00
|
|
|
let sched: &mut Scheduler = &mut **sched;
|
|
|
|
return sched;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn take_scheduler(&mut self) -> ~Scheduler {
|
|
|
|
unsafe {
|
|
|
|
let key = match self { &ThreadLocalScheduler(key) => key };
|
|
|
|
let value: *mut c_void = tls::get(key);
|
2013-03-28 20:39:09 -05:00
|
|
|
assert!(value.is_not_null());
|
2013-02-03 20:15:43 -06:00
|
|
|
let sched = transmute(value);
|
|
|
|
tls::set(key, mut_null());
|
|
|
|
return sched;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extern {
|
|
|
|
fn rust_get_sched_tls_key() -> *mut c_void;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn thread_local_scheduler_smoke_test() {
|
|
|
|
let scheduler = ~UvEventLoop::new_scheduler();
|
|
|
|
let mut tls_scheduler = ThreadLocalScheduler::new();
|
|
|
|
tls_scheduler.put_scheduler(scheduler);
|
|
|
|
{
|
|
|
|
let _scheduler = tls_scheduler.get_scheduler();
|
|
|
|
}
|
|
|
|
let _scheduler = tls_scheduler.take_scheduler();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn thread_local_scheduler_two_instances() {
|
|
|
|
let scheduler = ~UvEventLoop::new_scheduler();
|
|
|
|
let mut tls_scheduler = ThreadLocalScheduler::new();
|
|
|
|
tls_scheduler.put_scheduler(scheduler);
|
|
|
|
{
|
|
|
|
|
|
|
|
let _scheduler = tls_scheduler.get_scheduler();
|
|
|
|
}
|
|
|
|
{
|
|
|
|
let scheduler = tls_scheduler.take_scheduler();
|
|
|
|
tls_scheduler.put_scheduler(scheduler);
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut tls_scheduler = ThreadLocalScheduler::new();
|
|
|
|
{
|
|
|
|
let _scheduler = tls_scheduler.get_scheduler();
|
|
|
|
}
|
|
|
|
let _scheduler = tls_scheduler.take_scheduler();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_simple_scheduling() {
|
|
|
|
do run_in_bare_thread {
|
|
|
|
let mut task_ran = false;
|
|
|
|
let task_ran_ptr: *mut bool = &mut task_ran;
|
|
|
|
|
|
|
|
let mut sched = ~UvEventLoop::new_scheduler();
|
|
|
|
let task = ~do Task::new(&mut sched.stack_pool) {
|
|
|
|
unsafe { *task_ran_ptr = true; }
|
|
|
|
};
|
|
|
|
sched.task_queue.push_back(task);
|
|
|
|
sched.run();
|
2013-03-28 20:39:09 -05:00
|
|
|
assert!(task_ran);
|
2013-02-03 20:15:43 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_several_tasks() {
|
|
|
|
do run_in_bare_thread {
|
|
|
|
let total = 10;
|
|
|
|
let mut task_count = 0;
|
|
|
|
let task_count_ptr: *mut int = &mut task_count;
|
|
|
|
|
|
|
|
let mut sched = ~UvEventLoop::new_scheduler();
|
|
|
|
for int::range(0, total) |_| {
|
|
|
|
let task = ~do Task::new(&mut sched.stack_pool) {
|
|
|
|
unsafe { *task_count_ptr = *task_count_ptr + 1; }
|
|
|
|
};
|
|
|
|
sched.task_queue.push_back(task);
|
|
|
|
}
|
|
|
|
sched.run();
|
2013-03-28 20:39:09 -05:00
|
|
|
assert!(task_count == total);
|
2013-02-03 20:15:43 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_swap_tasks() {
|
|
|
|
do run_in_bare_thread {
|
|
|
|
let mut count = 0;
|
|
|
|
let count_ptr: *mut int = &mut count;
|
|
|
|
|
|
|
|
let mut sched = ~UvEventLoop::new_scheduler();
|
|
|
|
let task1 = ~do Task::new(&mut sched.stack_pool) {
|
|
|
|
unsafe { *count_ptr = *count_ptr + 1; }
|
|
|
|
do Scheduler::local |sched| {
|
|
|
|
let task2 = ~do Task::new(&mut sched.stack_pool) {
|
|
|
|
unsafe { *count_ptr = *count_ptr + 1; }
|
|
|
|
};
|
|
|
|
// Context switch directly to the new task
|
|
|
|
sched.resume_task_from_running_task_direct(task2);
|
|
|
|
}
|
|
|
|
unsafe { *count_ptr = *count_ptr + 1; }
|
|
|
|
};
|
|
|
|
sched.task_queue.push_back(task1);
|
|
|
|
sched.run();
|
2013-03-28 20:39:09 -05:00
|
|
|
assert!(count == 3);
|
2013-02-03 20:15:43 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[bench] #[test] #[ignore(reason = "long test")]
|
|
|
|
fn test_run_a_lot_of_tasks_queued() {
|
|
|
|
do run_in_bare_thread {
|
2013-03-22 16:00:15 -05:00
|
|
|
static MAX: int = 1000000;
|
2013-02-03 20:15:43 -06:00
|
|
|
let mut count = 0;
|
|
|
|
let count_ptr: *mut int = &mut count;
|
|
|
|
|
|
|
|
let mut sched = ~UvEventLoop::new_scheduler();
|
|
|
|
|
|
|
|
let start_task = ~do Task::new(&mut sched.stack_pool) {
|
|
|
|
run_task(count_ptr);
|
|
|
|
};
|
|
|
|
sched.task_queue.push_back(start_task);
|
|
|
|
sched.run();
|
|
|
|
|
2013-03-28 20:39:09 -05:00
|
|
|
assert!(count == MAX);
|
2013-02-03 20:15:43 -06:00
|
|
|
|
|
|
|
fn run_task(count_ptr: *mut int) {
|
|
|
|
do Scheduler::local |sched| {
|
|
|
|
let task = ~do Task::new(&mut sched.stack_pool) {
|
|
|
|
unsafe {
|
|
|
|
*count_ptr = *count_ptr + 1;
|
|
|
|
if *count_ptr != MAX {
|
|
|
|
run_task(count_ptr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
sched.task_queue.push_back(task);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[bench] #[test] #[ignore(reason = "too much stack allocation")]
|
|
|
|
fn test_run_a_lot_of_tasks_direct() {
|
|
|
|
do run_in_bare_thread {
|
2013-03-22 16:00:15 -05:00
|
|
|
static MAX: int = 100000;
|
2013-02-03 20:15:43 -06:00
|
|
|
let mut count = 0;
|
|
|
|
let count_ptr: *mut int = &mut count;
|
|
|
|
|
|
|
|
let mut sched = ~UvEventLoop::new_scheduler();
|
|
|
|
|
|
|
|
let start_task = ~do Task::new(&mut sched.stack_pool) {
|
|
|
|
run_task(count_ptr);
|
|
|
|
};
|
|
|
|
sched.task_queue.push_back(start_task);
|
|
|
|
sched.run();
|
|
|
|
|
2013-03-28 20:39:09 -05:00
|
|
|
assert!(count == MAX);
|
2013-02-03 20:15:43 -06:00
|
|
|
|
|
|
|
fn run_task(count_ptr: *mut int) {
|
|
|
|
do Scheduler::local |sched| {
|
|
|
|
let task = ~do Task::new(&mut sched.stack_pool) {
|
|
|
|
unsafe {
|
|
|
|
*count_ptr = *count_ptr + 1;
|
|
|
|
if *count_ptr != MAX {
|
|
|
|
run_task(count_ptr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
// Context switch directly to the new task
|
|
|
|
sched.resume_task_from_running_task_direct(task);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_block_task() {
|
|
|
|
do run_in_bare_thread {
|
|
|
|
let mut sched = ~UvEventLoop::new_scheduler();
|
|
|
|
let task = ~do Task::new(&mut sched.stack_pool) {
|
|
|
|
do Scheduler::local |sched| {
|
2013-03-28 20:39:09 -05:00
|
|
|
assert!(sched.in_task_context());
|
2013-04-14 18:12:30 -05:00
|
|
|
do sched.deschedule_running_task_and_then() |sched, task| {
|
2013-03-28 20:39:09 -05:00
|
|
|
assert!(!sched.in_task_context());
|
2013-02-03 20:15:43 -06:00
|
|
|
sched.task_queue.push_back(task);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
sched.task_queue.push_back(task);
|
|
|
|
sched.run();
|
|
|
|
}
|
|
|
|
}
|