diff options
author | Simon Marlow <marlowsd@gmail.com> | 2012-03-28 11:45:39 +0100 |
---|---|---|
committer | Simon Marlow <marlowsd@gmail.com> | 2012-03-28 14:45:14 +0100 |
commit | 734f1d484b12b785795b249b4116cfb4817ab55d (patch) | |
tree | 518e9027a151295be28bf4f3b04d221920bead72 | |
parent | 59d6d62bc13dd0bca87cb30c96bf9d526dc88bfe (diff) | |
download | haskell-734f1d484b12b785795b249b4116cfb4817ab55d.tar.gz |
Fix a bug in threadStackOverflow() (maybe #5214)
If we overflow the current stack chunk and copy its entire contents
into the next stack chunk, we could end up with two UNDERFLOW_FRAMEs.
We had a special case to catch this in the case when the old stack
chunk was the last one (ending in STOP_FRAME), but it went wrong for
other chunks.
I found this bug while poking around in the core dump attached to
options and running the nofib suite: imaginary/wheel_sieve2 crashed
with +RTS -kc600 -kb300.
I don't know if this is the cause of all the symptoms reported in
-rw-r--r-- | rts/Threads.c | 35 |
1 files changed, 22 insertions, 13 deletions
diff --git a/rts/Threads.c b/rts/Threads.c index 5488a320fe..fc520c6a93 100644 --- a/rts/Threads.c +++ b/rts/Threads.c @@ -579,11 +579,6 @@ threadStackOverflow (Capability *cap, StgTSO *tso) tso->tot_stack_size += new_stack->stack_size; - new_stack->sp -= sizeofW(StgUnderflowFrame); - frame = (StgUnderflowFrame*)new_stack->sp; - frame->info = &stg_stack_underflow_frame_info; - frame->next_chunk = old_stack; - { StgWord *sp; nat chunk_words, size; @@ -611,6 +606,28 @@ threadStackOverflow (Capability *cap, StgTSO *tso) sp += size; } + if (sp == old_stack->stack + old_stack->stack_size) { + // + // the old stack chunk is now empty, so we do *not* insert + // an underflow frame pointing back to it. There are two + // cases: either the old stack chunk was the last one, in + // which case it ends with a STOP_FRAME, or it is not the + // last one, and it already ends with an UNDERFLOW_FRAME + // pointing to the previous chunk. In the latter case, we + // will copy the UNDERFLOW_FRAME into the new stack chunk. + // In both cases, the old chunk will be subsequently GC'd. + // + // With the default settings, -ki1k -kb1k, this means the + // first stack chunk will be discarded after the first + // overflow, being replaced by a non-moving 32k chunk. + // + } else { + new_stack->sp -= sizeofW(StgUnderflowFrame); + frame = (StgUnderflowFrame*)new_stack->sp; + frame->info = &stg_stack_underflow_frame_info; + frame->next_chunk = old_stack; + } + // copy the stack chunk between tso->sp and sp to // new_tso->sp + (tso->sp - sp) chunk_words = sp - old_stack->sp; @@ -623,14 +640,6 @@ threadStackOverflow (Capability *cap, StgTSO *tso) new_stack->sp -= chunk_words; } - // if the old stack chunk is now empty, discard it. With the - // default settings, -ki1k -kb1k, this means the first stack chunk - // will be discarded after the first overflow, being replaced by a - // non-moving 32k chunk. - if (old_stack->sp == old_stack->stack + old_stack->stack_size) { - frame->next_chunk = (StgStack*)END_TSO_QUEUE; // dummy - } - tso->stackobj = new_stack; // we're about to run it, better mark it dirty |