2013-04-21 18:28:17 -05: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.
|
|
|
|
|
|
|
|
//! Language-level runtime services that should reasonably expected
|
|
|
|
//! to be available 'everywhere'. Local heaps, GC, unwinding,
|
|
|
|
//! local storage, and logging. Even a 'freestanding' Rust would likely want
|
|
|
|
//! to implement this.
|
|
|
|
|
2013-10-11 16:20:34 -05:00
|
|
|
use super::local_heap::LocalHeap;
|
|
|
|
|
|
|
|
use prelude::*;
|
|
|
|
|
2013-06-02 18:16:40 -05:00
|
|
|
use borrow;
|
2013-04-22 19:15:31 -05:00
|
|
|
use cast::transmute;
|
2013-10-11 16:20:34 -05:00
|
|
|
use cell::Cell;
|
2013-06-22 03:09:06 -05:00
|
|
|
use cleanup;
|
2013-10-09 12:34:27 -05:00
|
|
|
use libc::{c_void, uintptr_t, c_char, size_t};
|
2013-10-11 16:20:34 -05:00
|
|
|
use local_data;
|
2013-06-14 01:31:19 -05:00
|
|
|
use option::{Option, Some, None};
|
2013-08-11 20:54:23 -05:00
|
|
|
use rt::borrowck::BorrowRecord;
|
2013-10-11 16:20:34 -05:00
|
|
|
use rt::borrowck;
|
|
|
|
use rt::context::Context;
|
|
|
|
use rt::context;
|
2013-08-05 14:43:33 -05:00
|
|
|
use rt::env;
|
2013-11-11 00:46:32 -06:00
|
|
|
use io::Writer;
|
2013-07-01 22:24:24 -05:00
|
|
|
use rt::kill::Death;
|
2013-05-19 17:45:39 -05:00
|
|
|
use rt::local::Local;
|
2013-05-07 17:57:15 -05:00
|
|
|
use rt::logging::StdErrLogger;
|
2013-06-26 18:41:00 -05:00
|
|
|
use rt::sched::{Scheduler, SchedHandle};
|
|
|
|
use rt::stack::{StackSegment, StackPool};
|
2013-10-05 14:01:58 -05:00
|
|
|
use send_str::SendStr;
|
2013-11-01 13:20:01 -05:00
|
|
|
use task::LinkedFailure;
|
2013-10-11 16:20:34 -05:00
|
|
|
use task::spawn::Taskgroup;
|
|
|
|
use unstable::finally::Finally;
|
2013-04-21 18:28:17 -05:00
|
|
|
|
2013-07-19 16:25:05 -05:00
|
|
|
// The Task struct represents all state associated with a rust
|
|
|
|
// task. There are at this point two primary "subtypes" of task,
|
|
|
|
// however instead of using a subtype we just have a "task_type" field
|
|
|
|
// in the struct. This contains a pointer to another struct that holds
|
|
|
|
// the type-specific state.
|
|
|
|
|
2013-05-19 03:04:01 -05:00
|
|
|
pub struct Task {
|
2013-10-20 00:51:30 -05:00
|
|
|
heap: LocalHeap,
|
|
|
|
priv gc: GarbageCollector,
|
|
|
|
storage: LocalStorage,
|
2013-11-05 13:29:45 -06:00
|
|
|
logger: Option<StdErrLogger>,
|
2013-10-20 00:51:30 -05:00
|
|
|
unwinder: Unwinder,
|
|
|
|
taskgroup: Option<Taskgroup>,
|
|
|
|
death: Death,
|
|
|
|
destroyed: bool,
|
|
|
|
name: Option<SendStr>,
|
|
|
|
coroutine: Option<Coroutine>,
|
|
|
|
sched: Option<~Scheduler>,
|
|
|
|
task_type: TaskType,
|
2013-08-11 20:54:23 -05:00
|
|
|
// Dynamic borrowck debugging info
|
2013-10-24 19:30:36 -05:00
|
|
|
borrow_list: Option<~[BorrowRecord]>,
|
|
|
|
stdout_handle: Option<~Writer>,
|
2013-07-19 16:25:05 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
pub enum TaskType {
|
2013-08-18 00:59:46 -05:00
|
|
|
GreenTask(Option<SchedHome>),
|
2013-07-19 16:25:05 -05:00
|
|
|
SchedTask
|
2013-06-26 18:41:00 -05:00
|
|
|
}
|
|
|
|
|
2013-07-19 16:25:05 -05:00
|
|
|
/// A coroutine is nothing more than a (register context, stack) pair.
|
2013-06-26 18:41:00 -05:00
|
|
|
pub struct Coroutine {
|
|
|
|
/// The segment of stack on which the task is currently running or
|
|
|
|
/// if the task is blocked, on which the task will resume
|
|
|
|
/// execution.
|
2013-10-28 22:28:41 -05:00
|
|
|
///
|
|
|
|
/// Servo needs this to be public in order to tell SpiderMonkey
|
|
|
|
/// about the stack bounds.
|
|
|
|
current_stack_segment: StackSegment,
|
2013-06-26 18:41:00 -05:00
|
|
|
/// Always valid if the task is alive and not running.
|
|
|
|
saved_context: Context
|
|
|
|
}
|
|
|
|
|
2013-08-16 00:41:28 -05:00
|
|
|
/// Some tasks have a dedicated home scheduler that they must run on.
|
2013-06-26 18:41:00 -05:00
|
|
|
pub enum SchedHome {
|
|
|
|
AnySched,
|
|
|
|
Sched(SchedHandle)
|
2013-04-21 18:28:17 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
pub struct GarbageCollector;
|
2013-08-10 22:06:39 -05:00
|
|
|
pub struct LocalStorage(Option<local_data::Map>);
|
2013-04-22 19:15:31 -05:00
|
|
|
|
2013-10-11 16:20:34 -05:00
|
|
|
/// Represents the reason for the current unwinding process
|
|
|
|
pub enum UnwindResult {
|
|
|
|
/// The task is ending successfully
|
|
|
|
Success,
|
|
|
|
|
2013-11-01 13:20:01 -05:00
|
|
|
/// The Task is failing with reason `~Any`
|
|
|
|
Failure(~Any),
|
2013-10-11 16:20:34 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
impl UnwindResult {
|
|
|
|
/// Returns `true` if this `UnwindResult` is a failure
|
|
|
|
#[inline]
|
|
|
|
pub fn is_failure(&self) -> bool {
|
|
|
|
match *self {
|
|
|
|
Success => false,
|
|
|
|
Failure(_) => true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns `true` if this `UnwindResult` is a success
|
|
|
|
#[inline]
|
|
|
|
pub fn is_success(&self) -> bool {
|
|
|
|
match *self {
|
|
|
|
Success => true,
|
|
|
|
Failure(_) => false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-22 19:15:31 -05:00
|
|
|
pub struct Unwinder {
|
|
|
|
unwinding: bool,
|
2013-11-01 13:20:01 -05:00
|
|
|
cause: Option<~Any>
|
2013-10-11 16:20:34 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Unwinder {
|
|
|
|
fn to_unwind_result(&mut self) -> UnwindResult {
|
|
|
|
if self.unwinding {
|
|
|
|
Failure(self.cause.take().unwrap())
|
|
|
|
} else {
|
|
|
|
Success
|
|
|
|
}
|
|
|
|
}
|
2013-04-22 19:15:31 -05:00
|
|
|
}
|
2013-04-21 18:28:17 -05:00
|
|
|
|
2013-05-19 03:04:01 -05:00
|
|
|
impl Task {
|
2013-06-26 18:41:00 -05:00
|
|
|
|
2013-07-19 16:25:05 -05:00
|
|
|
// A helper to build a new task using the dynamically found
|
|
|
|
// scheduler and task. Only works in GreenTask context.
|
2013-08-05 15:10:08 -05:00
|
|
|
pub fn build_homed_child(stack_size: Option<uint>, f: ~fn(), home: SchedHome) -> ~Task {
|
2013-07-19 16:25:05 -05:00
|
|
|
let f = Cell::new(f);
|
|
|
|
let home = Cell::new(home);
|
2013-08-08 13:38:10 -05:00
|
|
|
do Local::borrow |running_task: &mut Task| {
|
2013-07-19 16:25:05 -05:00
|
|
|
let mut sched = running_task.sched.take_unwrap();
|
|
|
|
let new_task = ~running_task.new_child_homed(&mut sched.stack_pool,
|
2013-08-05 15:10:08 -05:00
|
|
|
stack_size,
|
2013-07-19 16:25:05 -05:00
|
|
|
home.take(),
|
|
|
|
f.take());
|
|
|
|
running_task.sched = Some(sched);
|
|
|
|
new_task
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-05 15:10:08 -05:00
|
|
|
pub fn build_child(stack_size: Option<uint>, f: ~fn()) -> ~Task {
|
|
|
|
Task::build_homed_child(stack_size, f, AnySched)
|
2013-07-19 16:25:05 -05:00
|
|
|
}
|
|
|
|
|
2013-08-05 15:10:08 -05:00
|
|
|
pub fn build_homed_root(stack_size: Option<uint>, f: ~fn(), home: SchedHome) -> ~Task {
|
2013-07-19 16:25:05 -05:00
|
|
|
let f = Cell::new(f);
|
|
|
|
let home = Cell::new(home);
|
2013-08-08 13:38:10 -05:00
|
|
|
do Local::borrow |running_task: &mut Task| {
|
2013-07-19 16:25:05 -05:00
|
|
|
let mut sched = running_task.sched.take_unwrap();
|
|
|
|
let new_task = ~Task::new_root_homed(&mut sched.stack_pool,
|
2013-08-05 15:10:08 -05:00
|
|
|
stack_size,
|
|
|
|
home.take(),
|
|
|
|
f.take());
|
2013-07-19 16:25:05 -05:00
|
|
|
running_task.sched = Some(sched);
|
|
|
|
new_task
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-05 15:10:08 -05:00
|
|
|
pub fn build_root(stack_size: Option<uint>, f: ~fn()) -> ~Task {
|
|
|
|
Task::build_homed_root(stack_size, f, AnySched)
|
2013-07-19 16:25:05 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn new_sched_task() -> Task {
|
|
|
|
Task {
|
|
|
|
heap: LocalHeap::new(),
|
|
|
|
gc: GarbageCollector,
|
2013-08-10 22:06:39 -05:00
|
|
|
storage: LocalStorage(None),
|
2013-11-05 13:29:45 -06:00
|
|
|
logger: None,
|
2013-10-11 16:20:34 -05:00
|
|
|
unwinder: Unwinder { unwinding: false, cause: None },
|
2013-07-19 16:25:05 -05:00
|
|
|
taskgroup: None,
|
|
|
|
death: Death::new(),
|
|
|
|
destroyed: false,
|
2013-07-29 15:34:08 -05:00
|
|
|
coroutine: Some(Coroutine::empty()),
|
2013-08-01 17:08:51 -05:00
|
|
|
name: None,
|
2013-07-19 16:25:05 -05:00
|
|
|
sched: None,
|
2013-08-11 20:54:23 -05:00
|
|
|
task_type: SchedTask,
|
2013-10-24 19:30:36 -05:00
|
|
|
borrow_list: None,
|
|
|
|
stdout_handle: None,
|
2013-07-19 16:25:05 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-26 18:41:00 -05:00
|
|
|
pub fn new_root(stack_pool: &mut StackPool,
|
2013-08-05 15:10:08 -05:00
|
|
|
stack_size: Option<uint>,
|
2013-06-26 18:41:00 -05:00
|
|
|
start: ~fn()) -> Task {
|
2013-08-05 15:10:08 -05:00
|
|
|
Task::new_root_homed(stack_pool, stack_size, AnySched, start)
|
2013-06-26 18:41:00 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn new_child(&mut self,
|
|
|
|
stack_pool: &mut StackPool,
|
2013-08-05 15:10:08 -05:00
|
|
|
stack_size: Option<uint>,
|
2013-06-26 18:41:00 -05:00
|
|
|
start: ~fn()) -> Task {
|
2013-08-05 15:10:08 -05:00
|
|
|
self.new_child_homed(stack_pool, stack_size, AnySched, start)
|
2013-06-26 18:41:00 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn new_root_homed(stack_pool: &mut StackPool,
|
2013-08-05 15:10:08 -05:00
|
|
|
stack_size: Option<uint>,
|
2013-06-26 18:41:00 -05:00
|
|
|
home: SchedHome,
|
|
|
|
start: ~fn()) -> Task {
|
2013-05-19 03:04:01 -05:00
|
|
|
Task {
|
2013-04-21 21:03:52 -05:00
|
|
|
heap: LocalHeap::new(),
|
2013-04-21 18:28:17 -05:00
|
|
|
gc: GarbageCollector,
|
2013-08-10 22:06:39 -05:00
|
|
|
storage: LocalStorage(None),
|
2013-11-05 13:29:45 -06:00
|
|
|
logger: None,
|
2013-10-11 16:20:34 -05:00
|
|
|
unwinder: Unwinder { unwinding: false, cause: None },
|
2013-07-12 21:45:19 -05:00
|
|
|
taskgroup: None,
|
2013-07-01 22:24:24 -05:00
|
|
|
death: Death::new(),
|
2013-06-26 18:41:00 -05:00
|
|
|
destroyed: false,
|
2013-07-30 18:20:59 -05:00
|
|
|
name: None,
|
2013-08-05 15:10:08 -05:00
|
|
|
coroutine: Some(Coroutine::new(stack_pool, stack_size, start)),
|
2013-07-19 16:25:05 -05:00
|
|
|
sched: None,
|
2013-08-18 00:59:46 -05:00
|
|
|
task_type: GreenTask(Some(home)),
|
2013-10-24 19:30:36 -05:00
|
|
|
borrow_list: None,
|
|
|
|
stdout_handle: None,
|
2013-04-23 17:11:28 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-26 18:41:00 -05:00
|
|
|
pub fn new_child_homed(&mut self,
|
|
|
|
stack_pool: &mut StackPool,
|
2013-08-05 15:10:08 -05:00
|
|
|
stack_size: Option<uint>,
|
2013-06-26 18:41:00 -05:00
|
|
|
home: SchedHome,
|
|
|
|
start: ~fn()) -> Task {
|
2013-05-19 03:04:01 -05:00
|
|
|
Task {
|
2013-04-23 17:11:28 -05:00
|
|
|
heap: LocalHeap::new(),
|
|
|
|
gc: GarbageCollector,
|
2013-08-10 22:06:39 -05:00
|
|
|
storage: LocalStorage(None),
|
2013-11-05 13:29:45 -06:00
|
|
|
logger: None,
|
2013-10-11 16:20:34 -05:00
|
|
|
unwinder: Unwinder { unwinding: false, cause: None },
|
2013-07-12 21:45:19 -05:00
|
|
|
taskgroup: None,
|
2013-07-01 22:24:24 -05:00
|
|
|
// FIXME(#7544) make watching optional
|
|
|
|
death: self.death.new_child(),
|
2013-06-26 18:41:00 -05:00
|
|
|
destroyed: false,
|
2013-07-30 18:20:59 -05:00
|
|
|
name: None,
|
2013-08-05 15:10:08 -05:00
|
|
|
coroutine: Some(Coroutine::new(stack_pool, stack_size, start)),
|
2013-07-19 16:25:05 -05:00
|
|
|
sched: None,
|
2013-08-18 00:59:46 -05:00
|
|
|
task_type: GreenTask(Some(home)),
|
2013-10-24 19:30:36 -05:00
|
|
|
borrow_list: None,
|
|
|
|
stdout_handle: None,
|
2013-04-21 18:28:17 -05:00
|
|
|
}
|
|
|
|
}
|
2013-04-22 14:54:03 -05:00
|
|
|
|
2013-06-14 14:17:56 -05:00
|
|
|
pub fn give_home(&mut self, new_home: SchedHome) {
|
2013-07-19 16:25:05 -05:00
|
|
|
match self.task_type {
|
|
|
|
GreenTask(ref mut home) => {
|
2013-08-18 00:59:46 -05:00
|
|
|
*home = Some(new_home);
|
2013-07-19 16:25:05 -05:00
|
|
|
}
|
|
|
|
SchedTask => {
|
|
|
|
rtabort!("type error: used SchedTask as GreenTask");
|
|
|
|
}
|
|
|
|
}
|
2013-06-14 14:17:56 -05:00
|
|
|
}
|
|
|
|
|
2013-07-19 16:25:05 -05:00
|
|
|
pub fn take_unwrap_home(&mut self) -> SchedHome {
|
|
|
|
match self.task_type {
|
|
|
|
GreenTask(ref mut home) => {
|
|
|
|
let out = home.take_unwrap();
|
2013-08-18 00:59:46 -05:00
|
|
|
return out;
|
2013-07-19 16:25:05 -05:00
|
|
|
}
|
|
|
|
SchedTask => {
|
|
|
|
rtabort!("type error: used SchedTask as GreenTask");
|
|
|
|
}
|
2013-04-22 19:15:31 -05:00
|
|
|
}
|
2013-07-19 16:25:05 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn run(&mut self, f: &fn()) {
|
2013-09-27 19:02:31 -05:00
|
|
|
rtdebug!("run called on task: {}", borrow::to_uint(self));
|
2013-04-22 19:15:31 -05:00
|
|
|
|
2013-08-03 16:43:16 -05:00
|
|
|
// The only try/catch block in the world. Attempt to run the task's
|
|
|
|
// client-specified code and catch any failures.
|
|
|
|
do self.unwinder.try {
|
2013-06-26 18:41:00 -05:00
|
|
|
|
2013-08-03 16:43:16 -05:00
|
|
|
// Run the task main function, then do some cleanup.
|
|
|
|
do f.finally {
|
2013-10-24 19:30:36 -05:00
|
|
|
|
2013-08-10 22:06:39 -05:00
|
|
|
// First, destroy task-local storage. This may run user dtors.
|
|
|
|
//
|
2013-08-05 02:32:46 -05:00
|
|
|
// FIXME #8302: Dear diary. I'm so tired and confused.
|
|
|
|
// There's some interaction in rustc between the box
|
|
|
|
// annihilator and the TLS dtor by which TLS is
|
|
|
|
// accessed from annihilated box dtors *after* TLS is
|
|
|
|
// destroyed. Somehow setting TLS back to null, as the
|
|
|
|
// old runtime did, makes this work, but I don't currently
|
|
|
|
// understand how. I would expect that, if the annihilator
|
|
|
|
// reinvokes TLS while TLS is uninitialized, that
|
|
|
|
// TLS would be reinitialized but never destroyed,
|
|
|
|
// but somehow this works. I have no idea what's going
|
|
|
|
// on but this seems to make things magically work. FML.
|
2013-08-10 22:06:39 -05:00
|
|
|
//
|
|
|
|
// (added after initial comment) A possible interaction here is
|
|
|
|
// that the destructors for the objects in TLS themselves invoke
|
|
|
|
// TLS, or possibly some destructors for those objects being
|
|
|
|
// annihilated invoke TLS. Sadly these two operations seemed to
|
|
|
|
// be intertwined, and miraculously work for now...
|
|
|
|
self.storage.take();
|
2013-08-05 02:32:46 -05:00
|
|
|
|
2013-08-03 16:43:16 -05:00
|
|
|
// Destroy remaining boxes. Also may run user dtors.
|
|
|
|
unsafe { cleanup::annihilate(); }
|
2013-10-24 19:30:36 -05:00
|
|
|
|
|
|
|
// Finally flush and destroy any output handles which the task
|
|
|
|
// owns. There are no boxes here, and no user destructors should
|
|
|
|
// run after this any more.
|
|
|
|
match self.stdout_handle.take() {
|
|
|
|
Some(handle) => {
|
|
|
|
let mut handle = handle;
|
|
|
|
handle.flush();
|
|
|
|
}
|
|
|
|
None => {}
|
|
|
|
}
|
2013-11-05 13:29:45 -06:00
|
|
|
self.logger.take();
|
2013-04-23 17:16:04 -05:00
|
|
|
}
|
2013-04-22 14:54:03 -05:00
|
|
|
}
|
2013-06-22 03:09:06 -05:00
|
|
|
|
2013-08-11 20:54:23 -05:00
|
|
|
// Cleanup the dynamic borrowck debugging info
|
|
|
|
borrowck::clear_task_borrow_list();
|
|
|
|
|
2013-08-12 13:54:09 -05:00
|
|
|
// NB. We pass the taskgroup into death so that it can be dropped while
|
|
|
|
// the unkillable counter is set. This is necessary for when the
|
|
|
|
// taskgroup destruction code drops references on KillHandles, which
|
|
|
|
// might require using unkillable (to synchronize with an unwrapper).
|
2013-10-11 16:20:34 -05:00
|
|
|
self.death.collect_failure(self.unwinder.to_unwind_result(), self.taskgroup.take());
|
2013-04-22 14:54:03 -05:00
|
|
|
self.destroyed = true;
|
|
|
|
}
|
2013-06-26 18:41:00 -05:00
|
|
|
|
2013-07-19 16:25:05 -05:00
|
|
|
// New utility functions for homes.
|
2013-06-26 18:41:00 -05:00
|
|
|
|
|
|
|
pub fn is_home_no_tls(&self, sched: &~Scheduler) -> bool {
|
2013-07-19 16:25:05 -05:00
|
|
|
match self.task_type {
|
2013-08-18 00:59:46 -05:00
|
|
|
GreenTask(Some(AnySched)) => { false }
|
|
|
|
GreenTask(Some(Sched(SchedHandle { sched_id: ref id, _}))) => {
|
2013-06-26 18:41:00 -05:00
|
|
|
*id == sched.sched_id()
|
|
|
|
}
|
2013-07-19 16:25:05 -05:00
|
|
|
GreenTask(None) => {
|
|
|
|
rtabort!("task without home");
|
|
|
|
}
|
|
|
|
SchedTask => {
|
|
|
|
// Awe yea
|
|
|
|
rtabort!("type error: expected: GreenTask, found: SchedTask");
|
2013-06-26 18:41:00 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn homed(&self) -> bool {
|
2013-07-19 16:25:05 -05:00
|
|
|
match self.task_type {
|
2013-08-18 00:59:46 -05:00
|
|
|
GreenTask(Some(AnySched)) => { false }
|
|
|
|
GreenTask(Some(Sched(SchedHandle { _ }))) => { true }
|
2013-07-19 16:25:05 -05:00
|
|
|
GreenTask(None) => {
|
|
|
|
rtabort!("task without home");
|
|
|
|
}
|
|
|
|
SchedTask => {
|
|
|
|
rtabort!("type error: expected: GreenTask, found: SchedTask");
|
2013-06-26 18:41:00 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-19 16:25:05 -05:00
|
|
|
// Grab both the scheduler and the task from TLS and check if the
|
|
|
|
// task is executing on an appropriate scheduler.
|
|
|
|
pub fn on_appropriate_sched() -> bool {
|
2013-08-08 13:38:10 -05:00
|
|
|
do Local::borrow |task: &mut Task| {
|
2013-07-19 16:25:05 -05:00
|
|
|
let sched_id = task.sched.get_ref().sched_id();
|
|
|
|
let sched_run_anything = task.sched.get_ref().run_anything;
|
|
|
|
match task.task_type {
|
2013-08-18 00:59:46 -05:00
|
|
|
GreenTask(Some(AnySched)) => {
|
2013-07-19 16:25:05 -05:00
|
|
|
rtdebug!("anysched task in sched check ****");
|
|
|
|
sched_run_anything
|
|
|
|
}
|
2013-08-18 00:59:46 -05:00
|
|
|
GreenTask(Some(Sched(SchedHandle { sched_id: ref id, _ }))) => {
|
2013-07-19 16:25:05 -05:00
|
|
|
rtdebug!("homed task in sched check ****");
|
|
|
|
*id == sched_id
|
|
|
|
}
|
|
|
|
GreenTask(None) => {
|
|
|
|
rtabort!("task without home");
|
|
|
|
}
|
|
|
|
SchedTask => {
|
|
|
|
rtabort!("type error: expected: GreenTask, found: SchedTask");
|
|
|
|
}
|
|
|
|
}
|
2013-06-26 18:41:00 -05:00
|
|
|
}
|
|
|
|
}
|
2013-04-22 14:54:03 -05:00
|
|
|
}
|
|
|
|
|
2013-05-19 03:04:01 -05:00
|
|
|
impl Drop for Task {
|
2013-09-16 20:18:07 -05:00
|
|
|
fn drop(&mut self) {
|
2013-09-27 19:02:31 -05:00
|
|
|
rtdebug!("called drop for a task: {}", borrow::to_uint(self));
|
2013-10-24 19:30:36 -05:00
|
|
|
rtassert!(self.destroyed);
|
2013-07-19 16:25:05 -05:00
|
|
|
}
|
2013-04-21 18:28:17 -05:00
|
|
|
}
|
|
|
|
|
2013-06-26 18:41:00 -05:00
|
|
|
// Coroutines represent nothing more than a context and a stack
|
|
|
|
// segment.
|
|
|
|
|
|
|
|
impl Coroutine {
|
|
|
|
|
2013-08-05 15:10:08 -05:00
|
|
|
pub fn new(stack_pool: &mut StackPool, stack_size: Option<uint>, start: ~fn()) -> Coroutine {
|
|
|
|
let stack_size = match stack_size {
|
|
|
|
Some(size) => size,
|
|
|
|
None => env::min_stack()
|
|
|
|
};
|
2013-06-26 18:41:00 -05:00
|
|
|
let start = Coroutine::build_start_wrapper(start);
|
2013-08-05 14:43:33 -05:00
|
|
|
let mut stack = stack_pool.take_segment(stack_size);
|
2013-06-26 18:41:00 -05:00
|
|
|
let initial_context = Context::new(start, &mut stack);
|
|
|
|
Coroutine {
|
|
|
|
current_stack_segment: stack,
|
|
|
|
saved_context: initial_context
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-19 16:25:05 -05:00
|
|
|
pub fn empty() -> Coroutine {
|
|
|
|
Coroutine {
|
|
|
|
current_stack_segment: StackSegment::new(0),
|
|
|
|
saved_context: Context::empty()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-26 18:41:00 -05:00
|
|
|
fn build_start_wrapper(start: ~fn()) -> ~fn() {
|
|
|
|
let start_cell = Cell::new(start);
|
|
|
|
let wrapper: ~fn() = || {
|
|
|
|
// First code after swap to this new context. Run our
|
|
|
|
// cleanup job.
|
|
|
|
unsafe {
|
|
|
|
|
2013-07-19 16:25:05 -05:00
|
|
|
// Again - might work while safe, or it might not.
|
2013-08-08 13:38:10 -05:00
|
|
|
do Local::borrow |sched: &mut Scheduler| {
|
2013-08-15 21:46:23 -05:00
|
|
|
sched.run_cleanup_job();
|
2013-07-19 16:25:05 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// To call the run method on a task we need a direct
|
|
|
|
// reference to it. The task is in TLS, so we can
|
|
|
|
// simply unsafe_borrow it to get this reference. We
|
|
|
|
// need to still have the task in TLS though, so we
|
|
|
|
// need to unsafe_borrow.
|
2013-08-08 13:38:10 -05:00
|
|
|
let task: *mut Task = Local::unsafe_borrow();
|
2013-06-26 18:41:00 -05:00
|
|
|
|
2013-07-19 16:25:05 -05:00
|
|
|
do (*task).run {
|
2013-06-26 18:41:00 -05:00
|
|
|
// N.B. Removing `start` from the start wrapper
|
|
|
|
// closure by emptying a cell is critical for
|
|
|
|
// correctness. The ~Task pointer, and in turn the
|
|
|
|
// closure used to initialize the first call
|
|
|
|
// frame, is destroyed in the scheduler context,
|
|
|
|
// not task context. So any captured closures must
|
|
|
|
// not contain user-definable dtors that expect to
|
|
|
|
// be in task context. By moving `start` out of
|
|
|
|
// the closure, all the user code goes our of
|
|
|
|
// scope while the task is still running.
|
|
|
|
let start = start_cell.take();
|
|
|
|
start();
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2013-07-19 16:25:05 -05:00
|
|
|
// We remove the sched from the Task in TLS right now.
|
2013-08-08 13:38:10 -05:00
|
|
|
let sched: ~Scheduler = Local::take();
|
2013-07-19 16:25:05 -05:00
|
|
|
// ... allowing us to give it away when performing a
|
|
|
|
// scheduling operation.
|
|
|
|
sched.terminate_current_task()
|
2013-06-26 18:41:00 -05:00
|
|
|
};
|
|
|
|
return wrapper;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Destroy coroutine and try to reuse stack segment.
|
2013-07-29 15:34:08 -05:00
|
|
|
pub fn recycle(self, stack_pool: &mut StackPool) {
|
2013-06-26 18:41:00 -05:00
|
|
|
match self {
|
2013-07-29 15:34:08 -05:00
|
|
|
Coroutine { current_stack_segment, _ } => {
|
2013-06-26 18:41:00 -05:00
|
|
|
stack_pool.give_segment(current_stack_segment);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-04-22 19:15:31 -05:00
|
|
|
// Just a sanity check to make sure we are catching a Rust-thrown exception
|
|
|
|
static UNWIND_TOKEN: uintptr_t = 839147;
|
|
|
|
|
|
|
|
impl Unwinder {
|
|
|
|
pub fn try(&mut self, f: &fn()) {
|
2013-07-21 19:20:52 -05:00
|
|
|
use unstable::raw::Closure;
|
2013-04-22 19:15:31 -05:00
|
|
|
|
|
|
|
unsafe {
|
|
|
|
let closure: Closure = transmute(f);
|
|
|
|
let code = transmute(closure.code);
|
|
|
|
let env = transmute(closure.env);
|
|
|
|
|
|
|
|
let token = rust_try(try_fn, code, env);
|
|
|
|
assert!(token == 0 || token == UNWIND_TOKEN);
|
|
|
|
}
|
|
|
|
|
|
|
|
extern fn try_fn(code: *c_void, env: *c_void) {
|
|
|
|
unsafe {
|
|
|
|
let closure: Closure = Closure {
|
|
|
|
code: transmute(code),
|
|
|
|
env: transmute(env),
|
|
|
|
};
|
|
|
|
let closure: &fn() = transmute(closure);
|
|
|
|
closure();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extern {
|
2013-08-21 08:31:02 -05:00
|
|
|
fn rust_try(f: extern "C" fn(*c_void, *c_void),
|
|
|
|
code: *c_void,
|
|
|
|
data: *c_void) -> uintptr_t;
|
2013-04-22 19:15:31 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-01 13:20:01 -05:00
|
|
|
pub fn begin_unwind(&mut self, cause: ~Any) -> ! {
|
2013-04-22 19:15:31 -05:00
|
|
|
self.unwinding = true;
|
2013-10-11 16:20:34 -05:00
|
|
|
self.cause = Some(cause);
|
2013-04-22 19:15:31 -05:00
|
|
|
unsafe {
|
|
|
|
rust_begin_unwind(UNWIND_TOKEN);
|
|
|
|
return transmute(());
|
|
|
|
}
|
|
|
|
extern {
|
|
|
|
fn rust_begin_unwind(token: uintptr_t);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-17 03:40:33 -05:00
|
|
|
/// This function is invoked from rust's current __morestack function. Segmented
|
|
|
|
/// stacks are currently not enabled as segmented stacks, but rather one giant
|
|
|
|
/// stack segment. This means that whenever we run out of stack, we want to
|
|
|
|
/// truly consider it to be stack overflow rather than allocating a new stack.
|
|
|
|
#[no_mangle] // - this is called from C code
|
|
|
|
#[no_split_stack] // - it would be sad for this function to trigger __morestack
|
2013-11-06 17:16:04 -06:00
|
|
|
#[doc(hidden)] // - Function must be `pub` to get exported, but it's
|
|
|
|
// irrelevant for documentation purposes.
|
2013-10-17 03:40:33 -05:00
|
|
|
pub extern "C" fn rust_stack_exhausted() {
|
|
|
|
use rt::in_green_task_context;
|
|
|
|
use rt::task::Task;
|
|
|
|
use rt::local::Local;
|
|
|
|
use unstable::intrinsics;
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
// We're calling this function because the stack just ran out. We need
|
|
|
|
// to call some other rust functions, but if we invoke the functions
|
|
|
|
// right now it'll just trigger this handler being called again. In
|
|
|
|
// order to alleviate this, we move the stack limit to be inside of the
|
|
|
|
// red zone that was allocated for exactly this reason.
|
|
|
|
let limit = context::get_sp_limit();
|
|
|
|
context::record_sp_limit(limit - context::RED_ZONE / 2);
|
|
|
|
|
|
|
|
// This probably isn't the best course of action. Ideally one would want
|
|
|
|
// to unwind the stack here instead of just aborting the entire process.
|
|
|
|
// This is a tricky problem, however. There's a few things which need to
|
|
|
|
// be considered:
|
|
|
|
//
|
|
|
|
// 1. We're here because of a stack overflow, yet unwinding will run
|
|
|
|
// destructors and hence arbitrary code. What if that code overflows
|
|
|
|
// the stack? One possibility is to use the above allocation of an
|
|
|
|
// extra 10k to hope that we don't hit the limit, and if we do then
|
|
|
|
// abort the whole program. Not the best, but kind of hard to deal
|
|
|
|
// with unless we want to switch stacks.
|
|
|
|
//
|
|
|
|
// 2. LLVM will optimize functions based on whether they can unwind or
|
|
|
|
// not. It will flag functions with 'nounwind' if it believes that
|
|
|
|
// the function cannot trigger unwinding, but if we do unwind on
|
|
|
|
// stack overflow then it means that we could unwind in any function
|
|
|
|
// anywhere. We would have to make sure that LLVM only places the
|
|
|
|
// nounwind flag on functions which don't call any other functions.
|
|
|
|
//
|
|
|
|
// 3. The function that overflowed may have owned arguments. These
|
|
|
|
// arguments need to have their destructors run, but we haven't even
|
|
|
|
// begun executing the function yet, so unwinding will not run the
|
|
|
|
// any landing pads for these functions. If this is ignored, then
|
|
|
|
// the arguments will just be leaked.
|
|
|
|
//
|
|
|
|
// Exactly what to do here is a very delicate topic, and is possibly
|
|
|
|
// still up in the air for what exactly to do. Some relevant issues:
|
|
|
|
//
|
|
|
|
// #3555 - out-of-stack failure leaks arguments
|
|
|
|
// #3695 - should there be a stack limit?
|
|
|
|
// #9855 - possible strategies which could be taken
|
|
|
|
// #9854 - unwinding on windows through __morestack has never worked
|
|
|
|
// #2361 - possible implementation of not using landing pads
|
|
|
|
|
|
|
|
if in_green_task_context() {
|
|
|
|
do Local::borrow |task: &mut Task| {
|
|
|
|
let n = task.name.as_ref().map(|n| n.as_slice()).unwrap_or("<unnamed>");
|
|
|
|
|
2013-10-22 11:36:30 -05:00
|
|
|
// See the message below for why this is not emitted to the
|
|
|
|
// task's logger. This has the additional conundrum of the
|
|
|
|
// logger may not be initialized just yet, meaning that an FFI
|
|
|
|
// call would happen to initialized it (calling out to libuv),
|
|
|
|
// and the FFI call needs 2MB of stack when we just ran out.
|
|
|
|
rterrln!("task '{}' has overflowed its stack", n);
|
2013-10-17 03:40:33 -05:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
rterrln!("stack overflow in non-task context");
|
|
|
|
}
|
|
|
|
|
|
|
|
intrinsics::abort();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-09 12:34:27 -05:00
|
|
|
/// This is the entry point of unwinding for things like lang items and such.
|
2013-10-11 16:20:34 -05:00
|
|
|
/// The arguments are normally generated by the compiler, and need to
|
|
|
|
/// have static lifetimes.
|
2013-10-27 14:12:40 -05:00
|
|
|
pub fn begin_unwind_raw(msg: *c_char, file: *c_char, line: size_t) -> ! {
|
2013-10-11 16:20:34 -05:00
|
|
|
use c_str::CString;
|
|
|
|
use cast::transmute;
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn static_char_ptr(p: *c_char) -> &'static str {
|
|
|
|
let s = unsafe { CString::new(p, false) };
|
|
|
|
match s.as_str() {
|
|
|
|
Some(s) => unsafe { transmute::<&str, &'static str>(s) },
|
|
|
|
None => rtabort!("message wasn't utf8?")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let msg = static_char_ptr(msg);
|
|
|
|
let file = static_char_ptr(file);
|
|
|
|
|
2013-10-27 14:12:40 -05:00
|
|
|
begin_unwind(msg, file, line as uint)
|
2013-10-11 16:20:34 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/// This is the entry point of unwinding for fail!() and assert!().
|
2013-10-27 14:12:40 -05:00
|
|
|
pub fn begin_unwind<M: Any + Send>(msg: M, file: &'static str, line: uint) -> ! {
|
2013-11-01 13:20:01 -05:00
|
|
|
use any::AnyRefExt;
|
2013-10-09 12:34:27 -05:00
|
|
|
use rt::in_green_task_context;
|
|
|
|
use rt::local::Local;
|
2013-11-01 13:20:01 -05:00
|
|
|
use rt::task::Task;
|
2013-10-09 12:34:27 -05:00
|
|
|
use str::Str;
|
2013-10-17 19:04:51 -05:00
|
|
|
use unstable::intrinsics;
|
2013-10-09 12:34:27 -05:00
|
|
|
|
|
|
|
unsafe {
|
2013-10-11 16:20:34 -05:00
|
|
|
let task: *mut Task;
|
2013-11-01 13:20:01 -05:00
|
|
|
// Note that this should be the only allocation performed in this block.
|
|
|
|
// Currently this means that fail!() on OOM will invoke this code path,
|
|
|
|
// but then again we're not really ready for failing on OOM anyway. If
|
|
|
|
// we do start doing this, then we should propagate this allocation to
|
|
|
|
// be performed in the parent of this task instead of the task that's
|
|
|
|
// failing.
|
|
|
|
let msg = ~msg as ~Any;
|
2013-10-11 16:20:34 -05:00
|
|
|
|
|
|
|
{
|
2013-11-01 13:20:01 -05:00
|
|
|
//let msg: &Any = msg;
|
|
|
|
let msg_s = match msg.as_ref::<&'static str>() {
|
|
|
|
Some(s) => *s,
|
|
|
|
None => match msg.as_ref::<~str>() {
|
|
|
|
Some(s) => s.as_slice(),
|
|
|
|
None => match msg.as_ref::<LinkedFailure>() {
|
|
|
|
Some(*) => "linked failure",
|
|
|
|
None => "~Any",
|
|
|
|
}
|
|
|
|
}
|
2013-10-11 16:20:34 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
if !in_green_task_context() {
|
|
|
|
rterrln!("failed in non-task context at '{}', {}:{}",
|
2013-10-27 14:12:40 -05:00
|
|
|
msg_s, file, line);
|
2013-10-11 16:20:34 -05:00
|
|
|
intrinsics::abort();
|
2013-10-17 03:40:33 -05:00
|
|
|
}
|
2013-10-09 12:34:27 -05:00
|
|
|
|
2013-10-11 16:20:34 -05:00
|
|
|
task = Local::unsafe_borrow();
|
|
|
|
let n = (*task).name.as_ref().map(|n| n.as_slice()).unwrap_or("<unnamed>");
|
|
|
|
|
|
|
|
// XXX: this should no get forcibly printed to the console, this should
|
|
|
|
// either be sent to the parent task (ideally), or get printed to
|
|
|
|
// the task's logger. Right now the logger is actually a uvio
|
|
|
|
// instance, which uses unkillable blocks internally for various
|
|
|
|
// reasons. This will cause serious trouble if the task is failing
|
|
|
|
// due to mismanagment of its own kill flag, so calling our own
|
|
|
|
// logger in its current state is a bit of a problem.
|
|
|
|
|
2013-10-27 14:12:40 -05:00
|
|
|
rterrln!("task '{}' failed at '{}', {}:{}", n, msg_s, file, line);
|
2013-10-11 16:20:34 -05:00
|
|
|
|
|
|
|
if (*task).unwinder.unwinding {
|
|
|
|
rtabort!("unwinding again");
|
2013-10-17 19:04:51 -05:00
|
|
|
}
|
2013-10-09 12:34:27 -05:00
|
|
|
}
|
2013-10-11 16:20:34 -05:00
|
|
|
|
2013-10-27 14:12:40 -05:00
|
|
|
(*task).unwinder.begin_unwind(msg);
|
2013-10-09 12:34:27 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-21 21:03:52 -05:00
|
|
|
#[cfg(test)]
|
|
|
|
mod test {
|
2013-10-27 14:12:40 -05:00
|
|
|
use super::*;
|
2013-04-21 21:03:52 -05:00
|
|
|
use rt::test::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn local_heap() {
|
|
|
|
do run_in_newsched_task() {
|
|
|
|
let a = @5;
|
|
|
|
let b = a;
|
|
|
|
assert!(*a == 5);
|
|
|
|
assert!(*b == 5);
|
|
|
|
}
|
|
|
|
}
|
2013-04-22 14:54:03 -05:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn tls() {
|
2013-07-11 00:14:40 -05:00
|
|
|
use local_data;
|
2013-04-22 14:54:03 -05:00
|
|
|
do run_in_newsched_task() {
|
2013-09-17 01:34:40 -05:00
|
|
|
local_data_key!(key: @~str)
|
2013-07-12 03:38:44 -05:00
|
|
|
local_data::set(key, @~"data");
|
2013-09-20 01:08:47 -05:00
|
|
|
assert!(*local_data::get(key, |k| k.map(|k| *k)).unwrap() == ~"data");
|
2013-09-17 01:34:40 -05:00
|
|
|
local_data_key!(key2: @~str)
|
2013-07-12 03:38:44 -05:00
|
|
|
local_data::set(key2, @~"data");
|
2013-09-20 01:08:47 -05:00
|
|
|
assert!(*local_data::get(key2, |k| k.map(|k| *k)).unwrap() == ~"data");
|
2013-04-22 14:54:03 -05:00
|
|
|
}
|
|
|
|
}
|
2013-04-22 19:15:31 -05:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn unwind() {
|
|
|
|
do run_in_newsched_task() {
|
2013-04-23 17:11:28 -05:00
|
|
|
let result = spawntask_try(||());
|
2013-06-26 18:41:00 -05:00
|
|
|
rtdebug!("trying first assert");
|
2013-04-22 19:15:31 -05:00
|
|
|
assert!(result.is_ok());
|
2013-10-21 15:08:31 -05:00
|
|
|
let result = spawntask_try(|| fail!());
|
2013-06-26 18:41:00 -05:00
|
|
|
rtdebug!("trying second assert");
|
2013-04-22 19:15:31 -05:00
|
|
|
assert!(result.is_err());
|
|
|
|
}
|
|
|
|
}
|
2013-05-06 20:24:37 -05:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn rng() {
|
2013-10-22 17:00:37 -05:00
|
|
|
do run_in_uv_task() {
|
2013-05-06 20:24:37 -05:00
|
|
|
use rand::{rng, Rng};
|
2013-05-08 14:26:34 -05:00
|
|
|
let mut r = rng();
|
2013-09-21 07:06:50 -05:00
|
|
|
let _ = r.next_u32();
|
2013-05-06 20:24:37 -05:00
|
|
|
}
|
|
|
|
}
|
2013-04-27 20:57:15 -05:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn logging() {
|
2013-10-22 17:00:37 -05:00
|
|
|
do run_in_uv_task() {
|
2013-10-21 15:08:31 -05:00
|
|
|
info!("here i am. logging in a newsched task");
|
2013-04-27 20:57:15 -05:00
|
|
|
}
|
|
|
|
}
|
2013-05-17 01:12:22 -05:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn comm_oneshot() {
|
|
|
|
use comm::*;
|
|
|
|
|
|
|
|
do run_in_newsched_task {
|
|
|
|
let (port, chan) = oneshot();
|
2013-08-01 01:12:20 -05:00
|
|
|
chan.send(10);
|
|
|
|
assert!(port.recv() == 10);
|
2013-05-17 01:12:22 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn comm_stream() {
|
|
|
|
use comm::*;
|
|
|
|
|
|
|
|
do run_in_newsched_task() {
|
2013-05-17 19:47:10 -05:00
|
|
|
let (port, chan) = stream();
|
2013-05-17 01:12:22 -05:00
|
|
|
chan.send(10);
|
|
|
|
assert!(port.recv() == 10);
|
|
|
|
}
|
|
|
|
}
|
2013-06-14 01:31:19 -05:00
|
|
|
|
2013-06-20 20:26:56 -05:00
|
|
|
#[test]
|
|
|
|
fn comm_shared_chan() {
|
|
|
|
use comm::*;
|
|
|
|
|
|
|
|
do run_in_newsched_task() {
|
|
|
|
let (port, chan) = stream();
|
|
|
|
let chan = SharedChan::new(chan);
|
|
|
|
chan.send(10);
|
|
|
|
assert!(port.recv() == 10);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-14 01:31:19 -05:00
|
|
|
#[test]
|
|
|
|
fn linked_failure() {
|
|
|
|
do run_in_newsched_task() {
|
|
|
|
let res = do spawntask_try {
|
2013-10-21 15:08:31 -05:00
|
|
|
spawntask_random(|| fail!());
|
2013-06-14 01:31:19 -05:00
|
|
|
};
|
|
|
|
assert!(res.is_err());
|
|
|
|
}
|
|
|
|
}
|
2013-06-22 03:09:06 -05:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn heap_cycles() {
|
|
|
|
use option::{Option, Some, None};
|
|
|
|
|
|
|
|
do run_in_newsched_task {
|
|
|
|
struct List {
|
|
|
|
next: Option<@mut List>,
|
|
|
|
}
|
|
|
|
|
|
|
|
let a = @mut List { next: None };
|
|
|
|
let b = @mut List { next: Some(a) };
|
|
|
|
|
|
|
|
a.next = Some(b);
|
|
|
|
}
|
|
|
|
}
|
2013-10-27 14:12:40 -05:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[should_fail]
|
|
|
|
fn test_begin_unwind() { begin_unwind("cause", file!(), line!()) }
|
2013-05-08 14:26:34 -05:00
|
|
|
}
|