diff options
author | GHC GitLab CI <ghc-ci@gitlab-haskell.org> | 2020-12-07 21:43:31 +0000 |
---|---|---|
committer | Ben Gamari <ben@well-typed.com> | 2020-12-20 21:05:13 -0500 |
commit | cde7499476f66f1369bd55cc046e4bfdb5d4a2a6 (patch) | |
tree | 95cd67fc6063540dcc112d47d12d9cdca70e0b0c | |
parent | a5b2fded602ee8110dc75de2ec9910c36980ca24 (diff) | |
download | haskell-cde7499476f66f1369bd55cc046e4bfdb5d4a2a6.tar.gz |
nonmoving: Ensure deadlock detection promotion works
Previously the deadlock-detection promotion logic in alloc_for_copy was
just plain wrong: it failed to fire when gct->evac_gen_no !=
oldest_gen->gen_no. The fix is simple: move the
-rw-r--r-- | rts/sm/Evac.c | 40 |
1 files changed, 22 insertions, 18 deletions
diff --git a/rts/sm/Evac.c b/rts/sm/Evac.c index e9b9c84324..c37251d9b3 100644 --- a/rts/sm/Evac.c +++ b/rts/sm/Evac.c @@ -73,24 +73,14 @@ alloc_for_copy (uint32_t size, uint32_t gen_no) StgPtr to; gen_workspace *ws; - /* Find out where we're going, using the handy "to" pointer in - * the gen of the source object. If it turns out we need to - * evacuate to an older generation, adjust it here (see comment - * by evacuate()). - */ - if (gen_no < gct->evac_gen_no) { - if (gct->eager_promotion) { - gen_no = gct->evac_gen_no; - } else if (RTS_UNLIKELY(RtsFlags.GcFlags.useNonmoving) && deadlock_detect_gc) { - /* See Note [Deadlock detection under nonmoving collector]. */ - gen_no = oldest_gen->no; - } else { - gct->failed_to_evac = true; + if (RTS_UNLIKELY(RtsFlags.GcFlags.useNonmoving)) { + /* See Note [Deadlock detection under nonmoving collector]. */ + const uint32_t oldest_gen_no = oldest_gen->no; + if (deadlock_detect_gc) { + gen_no = oldest_gen_no; } - } - if (RTS_UNLIKELY(RtsFlags.GcFlags.useNonmoving)) { - if (gen_no == oldest_gen->no) { + if (gen_no == oldest_gen_no) { gct->copied += size; to = nonmovingAllocate(gct->cap, size); @@ -98,7 +88,7 @@ alloc_for_copy (uint32_t size, uint32_t gen_no) // current->todo_link == NULL means not in todo list struct NonmovingSegment *seg = nonmovingGetSegment(to); if (!seg->todo_link) { - gen_workspace *ws = &gct->gens[oldest_gen->no]; + gen_workspace *ws = &gct->gens[oldest_gen_no]; seg->todo_link = ws->todo_seg; ws->todo_seg = seg; } @@ -111,12 +101,26 @@ alloc_for_copy (uint32_t size, uint32_t gen_no) // so there is no need. // // See Note [Non-moving GC: Marking evacuated objects]. - if (major_gc && !deadlock_detect_gc) + if (major_gc && !deadlock_detect_gc) { markQueuePushClosureGC(&gct->cap->upd_rem_set.queue, (StgClosure *) to); + } return to; } } + /* Find out where we're going, using the handy "to" pointer in + * the gen of the source object. If it turns out we need to + * evacuate to an older generation, adjust it here (see comment + * by evacuate()). + */ + if (gen_no < gct->evac_gen_no) { + if (gct->eager_promotion) { + gen_no = gct->evac_gen_no; + } else { + gct->failed_to_evac = true; + } + } + ws = &gct->gens[gen_no]; // zero memory references here /* chain a new block onto the to-space for the destination gen if |