From 7036fde9df61b6eae9719c7f6c656778c756bec9 Mon Sep 17 00:00:00 2001 From: Simon Marlow Date: Fri, 29 Jul 2016 14:11:03 +0100 Subject: Overhaul of Compact Regions (#12455) Summary: This commit makes various improvements and addresses some issues with Compact Regions (aka Compact Normal Forms). This was the most important thing I wanted to fix. Compaction previously prevented GC from running until it was complete, which would be a problem in a multicore setting. Now, we compact using a hand-written Cmm routine that can be interrupted at any point. When a GC is triggered during a sharing-enabled compaction, the GC has to traverse and update the hash table, so this hash table is now stored in the StgCompactNFData object. Previously, compaction consisted of a deepseq using the NFData class, followed by a traversal in C code to copy the data. This is now done in a single pass with hand-written Cmm (see rts/Compact.cmm). We no longer use the NFData instances, instead the Cmm routine evaluates components directly as it compacts. The new compaction is about 50% faster than the old one with no sharing, and a little faster on average with sharing (the cost of the hash table dominates when we're doing sharing). Static objects that don't (transitively) refer to any CAFs don't need to be copied into the compact region. In particular this means we often avoid copying Char values and small Int values, because these are static closures in the runtime. Each Compact# object can support a single compactAdd# operation at any given time, so the Data.Compact library now enforces mutual exclusion using an MVar stored in the Compact object. We now get exceptions rather than killing everything with a barf() when we encounter an object that cannot be compacted (a function, or a mutable object). We now also detect pinned objects, which can't be compacted either. The Data.Compact API has been refactored and cleaned up. A new compactSize operation returns the size (in bytes) of the compact object. Most of the documentation is in the Haddock docs for the compact library, which I've expanded and improved here. Various comments in the code have been improved, especially the main Note [Compact Normal Forms] in rts/sm/CNF.c. I've added a few tests, and expanded a few of the tests that were there. We now also run the tests with GHCi, and in a new test way that enables sanity checking (+RTS -DS). There's a benchmark in libraries/compact/tests/compact_bench.hs for measuring compaction speed and comparing sharing vs. no sharing. The field totalDataW in StgCompactNFData was unnecessary. Test Plan: * new unit tests * validate * tested manually that we can compact Data.Aeson data Reviewers: gcampax, bgamari, ezyang, austin, niteria, hvr, erikd Subscribers: thomie, simonpj Differential Revision: https://phabricator.haskell.org/D2751 GHC Trac Issues: #12455 --- rts/sm/Scav.c | 64 +++++++++++++++++++++++++++++++++++++++++++++++++---------- 1 file changed, 54 insertions(+), 10 deletions(-) (limited to 'rts/sm/Scav.c') diff --git a/rts/sm/Scav.c b/rts/sm/Scav.c index 940f11fea4..10ce1e46a8 100644 --- a/rts/sm/Scav.c +++ b/rts/sm/Scav.c @@ -27,6 +27,7 @@ #include "Sanity.h" #include "Capability.h" #include "LdvProfile.h" +#include "Hash.h" #include "sm/MarkWeak.h" @@ -100,6 +101,45 @@ scavengeTSO (StgTSO *tso) gct->eager_promotion = saved_eager; } +/* ---------------------------------------------------------------------------- + Scavenging compact objects + ------------------------------------------------------------------------- */ + +static void +evacuate_hash_entry(HashTable *newHash, StgWord key, const void *value) +{ + StgClosure *p = (StgClosure*)key; + + evacuate(&p); + insertHashTable(newHash, (StgWord)p, value); +} + +static void +scavenge_compact(StgCompactNFData *str) +{ + bool saved_eager; + saved_eager = gct->eager_promotion; + gct->eager_promotion = false; + + if (str->hash) { + HashTable *newHash = allocHashTable(); + mapHashTable(str->hash, (void*)newHash, (MapHashFn)evacuate_hash_entry); + freeHashTable(str->hash, NULL); + str->hash = newHash; + } + + debugTrace(DEBUG_compact, + "compact alive @%p, gen %d, %" FMT_Word " bytes", + str, Bdescr((P_)str)->gen_no, str->totalW * sizeof(W_)) + + gct->eager_promotion = saved_eager; + if (gct->failed_to_evac) { + ((StgClosure *)str)->header.info = &stg_COMPACT_NFDATA_DIRTY_info; + } else { + ((StgClosure *)str)->header.info = &stg_COMPACT_NFDATA_CLEAN_info; + } +} + /* ----------------------------------------------------------------------------- Mutable arrays of pointers -------------------------------------------------------------------------- */ @@ -796,13 +836,6 @@ scavenge_block (bdescr *bd) break; } - case COMPACT_NFDATA: - // CompactNFData blocks live in compact lists, which we don't - // scavenge, because there nothing to scavenge in them - // so we should never ever see them - barf("scavenge: found unexpected Compact structure"); - break; - default: barf("scavenge: unimplemented/strange closure type %d @ %p", info->type, p); @@ -1557,6 +1590,10 @@ scavenge_one(StgPtr p) #endif break; + case COMPACT_NFDATA: + scavenge_compact((StgCompactNFData*)p); + break; + default: barf("scavenge_one: strange object %d", (int)(info->type)); } @@ -1974,11 +2011,18 @@ scavenge_large (gen_workspace *ws) ws->todo_large_objects = bd->link; ACQUIRE_SPIN_LOCK(&ws->gen->sync); - dbl_link_onto(bd, &ws->gen->scavenged_large_objects); - ws->gen->n_scavenged_large_blocks += bd->blocks; + if (bd->flags & BF_COMPACT) { + dbl_link_onto(bd, &ws->gen->live_compact_objects); + StgCompactNFData *str = ((StgCompactNFDataBlock*)bd->start)->owner; + ws->gen->n_live_compact_blocks += str->totalW / BLOCK_SIZE_W; + p = (StgPtr)str; + } else { + dbl_link_onto(bd, &ws->gen->scavenged_large_objects); + ws->gen->n_scavenged_large_blocks += bd->blocks; + p = bd->start; + } RELEASE_SPIN_LOCK(&ws->gen->sync); - p = bd->start; if (scavenge_one(p)) { if (ws->gen->no > 0) { recordMutableGen_GC((StgClosure *)p, ws->gen->no); -- cgit v1.2.1