diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2015-09-09 07:51:16 -0700 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2015-09-09 07:51:16 -0700 |
commit | d682e4e1b074ffa4315b67a56109ef648931883c (patch) | |
tree | c582a6927b479cba581966330ffbfe19b71f7c4f /gc.c | |
parent | aaef40301fcb91506898678c39cb62dc6fe0a285 (diff) | |
download | txr-d682e4e1b074ffa4315b67a56109ef648931883c.tar.gz txr-d682e4e1b074ffa4315b67a56109ef648931883c.tar.bz2 txr-d682e4e1b074ffa4315b67a56109ef648931883c.zip |
Finalization fix: incorrect for generational gc.
Another bug found in the finalization hooks. The reachability
test being applied is not correct for generational GC,
relying only on the REACHABLE flag. This bug means that we
call finalization hooks on mature objects, while they remain
reachable. It must use the gc_is_reachable logic, just like
weak hash table processing does, because under an incremental
garbage collection pass, all gen 1 objects are considered
reachable (though they don't have the REACHABLE flag set).
* gc.c (struct fin_reg): Remving obj_type member, replacing
with reachable flag.
(is_reachable): New static function based on gc_is_reachable,
minus the is_ptr test. (GC code which knows it is dealing with
a pointer doesn't need the test).
(prepare_finals): In the first pass, instead of memorizing the
prior object type, simply calculate the reachable flag,
correctly taking into account generational GC.
(call_finals): Rely on the new reachable flag rather than
the REACHABLE flag in the saved obj_type.
(gc_is_reachable): Becomes a public wrapper for is_reachable
which adds the pointer test.
(gc_finalize): Initialize the reachable member of fin_reg.
Remove initialization for obj_type.
Diffstat (limited to 'gc.c')
-rw-r--r-- | gc.c | 36 |
1 files changed, 19 insertions, 17 deletions
@@ -87,7 +87,7 @@ static struct fin_reg { struct fin_reg *next; val obj; val fun; - type_t obj_type; + int reachable; } *final_list, **final_tail = &final_list; #if CONFIG_GEN_GC @@ -601,6 +601,20 @@ static int_ptr_t sweep(void) return free_count; } +static int is_reachable(val obj) +{ + type_t t; + +#if CONFIG_GEN_GC + if (!full_gc && obj->t.gen > 0) + return 1; +#endif + + t = obj->t.type; + + return (t & REACHABLE) != 0; +} + static void prepare_finals(void) { struct fin_reg *f; @@ -613,7 +627,7 @@ static void prepare_finals(void) #endif for (f = final_list; f; f = f->next) - f->obj_type = f->obj->t.type; + f->reachable = is_reachable(f->obj); for (f = final_list; f; f = f->next) { mark_obj(f->obj); @@ -639,7 +653,7 @@ static void call_finals(void) for (f = old_list; f; f = next) { next = f->next; - if ((f->obj_type & REACHABLE) == 0) { + if (!f->reachable) { funcall1(f->fun, f->obj); free(f); } else { @@ -725,19 +739,7 @@ void gc_conservative_mark(val maybe_obj) int gc_is_reachable(val obj) { - type_t t; - - if (!is_ptr(obj)) - return 1; - -#if CONFIG_GEN_GC - if (!full_gc && obj->t.gen > 0) - return 1; -#endif - - t = obj->t.type; - - return (t & REACHABLE) != 0; + return is_ptr(obj) ? is_reachable(obj) : 1; } #if CONFIG_GEN_GC @@ -815,7 +817,7 @@ val gc_finalize(val obj, val fun) struct fin_reg *f = coerce(struct fin_reg *, chk_malloc(sizeof *f)); f->obj = obj; f->fun = fun; - f->obj_type = NIL; + f->reachable = 0; f->next = 0; *final_tail = f; final_tail = &f->next; |