Skip to content

Commit 8643a0d

Browse files
committed
green: Prevent runtime corruption on spawn failure
Like with libnative, when a green task failed to spawn it would leave the world in a corrupt state where the local scheduler had been dropped as well as the local task. Also like libnative, this patch sets up a "bomb" which when it goes off will restore the state of the world.
1 parent 355c798 commit 8643a0d

File tree

4 files changed

+68
-6
lines changed

4 files changed

+68
-6
lines changed

src/libgreen/task.rs

+18-3
Original file line numberDiff line numberDiff line change
@@ -442,15 +442,30 @@ impl Runtime for GreenTask {
442442
f: proc():Send) {
443443
self.put_task(cur_task);
444444

445+
// First, set up a bomb which when it goes off will restore the local
446+
// task unless its disarmed. This will allow us to gracefully fail from
447+
// inside of `configure` which allocates a new task.
448+
struct Bomb { inner: Option<Box<GreenTask>> }
449+
impl Drop for Bomb {
450+
fn drop(&mut self) {
451+
let _ = self.inner.take().map(|task| task.put());
452+
}
453+
}
454+
let mut bomb = Bomb { inner: Some(self) };
455+
445456
// Spawns a task into the current scheduler. We allocate the new task's
446457
// stack from the scheduler's stack pool, and then configure it
447458
// accordingly to `opts`. Afterwards we bootstrap it immediately by
448459
// switching to it.
449460
//
450461
// Upon returning, our task is back in TLS and we're good to return.
451-
let mut sched = self.sched.take_unwrap();
452-
let sibling = GreenTask::configure(&mut sched.stack_pool, opts, f);
453-
sched.run_task(self, sibling)
462+
let sibling = {
463+
let sched = bomb.inner.get_mut_ref().sched.get_mut_ref();
464+
GreenTask::configure(&mut sched.stack_pool, opts, f)
465+
};
466+
let mut me = bomb.inner.take().unwrap();
467+
let sched = me.sched.take().unwrap();
468+
sched.run_task(me, sibling)
454469
}
455470

456471
// Local I/O is provided by the scheduler's event loop

src/librustrt/bookkeeping.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,7 @@ use mutex::{StaticNativeMutex, NATIVE_MUTEX_INIT};
2626
static mut TASK_COUNT: atomics::AtomicUint = atomics::INIT_ATOMIC_UINT;
2727
static mut TASK_LOCK: StaticNativeMutex = NATIVE_MUTEX_INIT;
2828

29-
pub struct Token(());
29+
pub struct Token { _private: () }
3030

3131
impl Drop for Token {
3232
fn drop(&mut self) { decrement() }
@@ -36,7 +36,7 @@ impl Drop for Token {
3636
/// the count when dropped.
3737
pub fn increment() -> Token {
3838
let _ = unsafe { TASK_COUNT.fetch_add(1, atomics::SeqCst) };
39-
Token(())
39+
Token { _private: () }
4040
}
4141

4242
pub fn decrement() {

src/librustrt/task.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -663,6 +663,6 @@ mod test {
663663
fn block_and_wake() {
664664
let task = box Task::new();
665665
let mut task = BlockedTask::block(task).wake().unwrap();
666-
task.destroy();
666+
task.drop();
667667
}
668668
}
+47
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
// Copyright 2014 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
// ignore-macos apparently gargantuan mmap requests are ok?
12+
13+
#![feature(phase)]
14+
15+
#[phase(plugin)]
16+
extern crate green;
17+
extern crate native;
18+
19+
use std::task::TaskBuilder;
20+
use native::NativeTaskBuilder;
21+
22+
green_start!(main)
23+
24+
fn main() {
25+
test();
26+
27+
let (tx, rx) = channel();
28+
TaskBuilder::new().native().spawn(proc() {
29+
tx.send(test());
30+
});
31+
rx.recv();
32+
}
33+
34+
#[cfg(not(target_word_size = "64"))]
35+
fn test() {}
36+
37+
#[cfg(target_word_size = "64")]
38+
fn test() {
39+
let (tx, rx) = channel();
40+
spawn(proc() {
41+
TaskBuilder::new().stack_size(1024 * 1024 * 1024 * 64).spawn(proc() {
42+
});
43+
tx.send(());
44+
});
45+
46+
assert!(rx.recv_opt().is_err());
47+
}

0 commit comments

Comments
 (0)