Skip to content

Commit eda5e40

Browse files
committed
auto merge of #7111 : brson/rust/stack, r=brson
... through yields This avoids the following pathological scenario that makes threadring OOM: 1) task calls C using fast_ffi, borrowing a big stack from the scheduler. 2) task returns from C and places the big stack on the task-local stack segment list 3) task calls further Rust functions that require growing the stack, and for this reuses the big stack 4) task yields, failing to return the big stack to the scheduler. 5) repeat 500+ times and OOM (reopening after incoming fallout. *do not r+*. broken)
2 parents 36d7f60 + 8918461 commit eda5e40

File tree

4 files changed

+40
-56
lines changed

4 files changed

+40
-56
lines changed

src/rt/rust_sched_loop.h

+2-1
Original file line numberDiff line numberDiff line change
@@ -211,12 +211,13 @@ rust_sched_loop::return_c_stack(stk_seg *stack) {
211211
// NB: Runs on the Rust stack. Might return NULL!
212212
inline stk_seg *
213213
rust_sched_loop::borrow_big_stack() {
214-
assert(cached_big_stack);
215214
stk_seg *your_stack;
216215
if (extra_big_stack) {
217216
your_stack = extra_big_stack;
218217
extra_big_stack = NULL;
219218
} else {
219+
// NB: This may be null if we're asking for a *second*
220+
// big stack, in which case the caller will fall back to a slow path
220221
your_stack = cached_big_stack;
221222
cached_big_stack = NULL;
222223
}

src/rt/rust_task.cpp

+14-42
Original file line numberDiff line numberDiff line change
@@ -54,8 +54,7 @@ rust_task::rust_task(rust_sched_loop *sched_loop, rust_task_state state,
5454
disallow_yield(0),
5555
c_stack(NULL),
5656
next_c_sp(0),
57-
next_rust_sp(0),
58-
big_stack(NULL)
57+
next_rust_sp(0)
5958
{
6059
LOGPTR(sched_loop, "new task", (uintptr_t)this);
6160
DLOG(sched_loop, task, "sizeof(task) = %d (0x%x)",
@@ -566,14 +565,8 @@ rust_task::cleanup_after_turn() {
566565

567566
while (stk->next) {
568567
stk_seg *new_next = stk->next->next;
569-
570-
if (stk->next->is_big) {
571-
assert (big_stack == stk->next);
572-
sched_loop->return_big_stack(big_stack);
573-
big_stack = NULL;
574-
} else {
575-
free_stack(stk->next);
576-
}
568+
assert (!stk->next->is_big);
569+
free_stack(stk->next);
577570

578571
stk->next = new_next;
579572
}
@@ -584,38 +577,20 @@ rust_task::cleanup_after_turn() {
584577
bool
585578
rust_task::new_big_stack() {
586579
assert(stk);
587-
// If we have a cached big stack segment, use it.
588-
if (big_stack) {
589-
// Check to see if we're already on the big stack.
590-
stk_seg *ss = stk;
591-
while (ss != NULL) {
592-
if (ss == big_stack)
593-
return false;
594-
ss = ss->prev;
595-
}
596580

597-
// Unlink the big stack.
598-
if (big_stack->next)
599-
big_stack->next->prev = big_stack->prev;
600-
if (big_stack->prev)
601-
big_stack->prev->next = big_stack->next;
602-
} else {
603-
stk_seg *borrowed_big_stack = sched_loop->borrow_big_stack();
604-
if (!borrowed_big_stack) {
605-
abort();
606-
} else {
607-
big_stack = borrowed_big_stack;
608-
}
581+
stk_seg *borrowed_big_stack = sched_loop->borrow_big_stack();
582+
if (!borrowed_big_stack) {
583+
return false;
609584
}
610585

611-
big_stack->task = this;
612-
big_stack->next = stk->next;
613-
if (big_stack->next)
614-
big_stack->next->prev = big_stack;
615-
big_stack->prev = stk;
616-
stk->next = big_stack;
586+
borrowed_big_stack->task = this;
587+
borrowed_big_stack->next = stk->next;
588+
if (borrowed_big_stack->next)
589+
borrowed_big_stack->next->prev = borrowed_big_stack;
590+
borrowed_big_stack->prev = stk;
591+
stk->next = borrowed_big_stack;
617592

618-
stk = big_stack;
593+
stk = borrowed_big_stack;
619594

620595
return true;
621596
}
@@ -640,10 +615,9 @@ void
640615
rust_task::reset_stack_limit() {
641616
uintptr_t sp = get_sp();
642617
while (!sp_in_stk_seg(sp, stk)) {
643-
stk = stk->prev;
618+
prev_stack();
644619
assert(stk != NULL && "Failed to find the current stack");
645620
}
646-
record_stack_limit();
647621
}
648622

649623
void
@@ -667,8 +641,6 @@ rust_task::delete_all_stacks() {
667641

668642
stk = prev;
669643
}
670-
671-
big_stack = NULL;
672644
}
673645

674646
/*

src/rt/rust_task.h

+15-4
Original file line numberDiff line numberDiff line change
@@ -278,9 +278,6 @@ rust_task : public kernel_owned<rust_task>
278278
uintptr_t next_c_sp;
279279
uintptr_t next_rust_sp;
280280

281-
// The big stack.
282-
stk_seg *big_stack;
283-
284281
// Called when the atomic refcount reaches zero
285282
void delete_this();
286283

@@ -607,7 +604,21 @@ rust_task::prev_stack() {
607604
// require switching to the C stack and be costly. Instead we'll just move
608605
// up the link list and clean up later, either in new_stack or after our
609606
// turn ends on the scheduler.
610-
stk = stk->prev;
607+
if (stk->is_big) {
608+
stk_seg *ss = stk;
609+
stk = stk->prev;
610+
611+
// Unlink the big stack.
612+
if (ss->next)
613+
ss->next->prev = ss->prev;
614+
if (ss->prev)
615+
ss->prev->next = ss->next;
616+
617+
sched_loop->return_big_stack(ss);
618+
} else {
619+
stk = stk->prev;
620+
}
621+
611622
record_stack_limit();
612623
}
613624

src/test/bench/shootout-threadring.rs

+9-9
Original file line numberDiff line numberDiff line change
@@ -10,35 +10,35 @@
1010

1111
// Based on threadring.erlang by Jira Isa
1212

13-
// xfail-test FIXME #5985 OOM's on the mac bot
13+
use std::os;
1414

1515
fn start(n_tasks: int, token: int) {
16-
let mut (p, ch1) = comm::stream();
16+
let mut (p, ch1) = stream();
1717
ch1.send(token);
1818
// XXX could not get this to work with a range closure
1919
let mut i = 2;
2020
while i <= n_tasks {
21-
let (next_p, ch) = comm::stream();
21+
let (next_p, ch) = stream();
2222
let imm_i = i;
2323
let imm_p = p;
24-
do task::spawn {
24+
do spawn {
2525
roundtrip(imm_i, n_tasks, &imm_p, &ch);
2626
};
2727
p = next_p;
2828
i += 1;
2929
}
3030
let imm_p = p;
3131
let imm_ch = ch1;
32-
do task::spawn {
32+
do spawn {
3333
roundtrip(1, n_tasks, &imm_p, &imm_ch);
3434
}
3535
}
3636

37-
fn roundtrip(id: int, n_tasks: int, p: &comm::Port<int>, ch: &comm::Chan<int>) {
37+
fn roundtrip(id: int, n_tasks: int, p: &Port<int>, ch: &Chan<int>) {
3838
while (true) {
3939
match p.recv() {
4040
1 => {
41-
io::println(fmt!("%d\n", id));
41+
println(fmt!("%d\n", id));
4242
return;
4343
}
4444
token => {
@@ -60,13 +60,13 @@ fn main() {
6060
os::args()
6161
};
6262
let token = if args.len() > 1u {
63-
int::from_str(args[1]).get()
63+
FromStr::from_str(args[1]).get()
6464
}
6565
else {
6666
1000
6767
};
6868
let n_tasks = if args.len() > 2u {
69-
int::from_str(args[2]).get()
69+
FromStr::from_str(args[2]).get()
7070
}
7171
else {
7272
503

0 commit comments

Comments
 (0)