summaryrefslogtreecommitdiffstats
path: root/gc.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2015-09-09 07:51:16 -0700
committerKaz Kylheku <kaz@kylheku.com>2015-09-09 07:51:16 -0700
commitd682e4e1b074ffa4315b67a56109ef648931883c (patch)
treec582a6927b479cba581966330ffbfe19b71f7c4f /gc.c
parentaaef40301fcb91506898678c39cb62dc6fe0a285 (diff)
downloadtxr-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.c36
1 files changed, 19 insertions, 17 deletions
diff --git a/gc.c b/gc.c
index f17a4b47..702ef812 100644
--- a/gc.c
+++ b/gc.c
@@ -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;