gc/gc-7.4.0-disclaim_test-fail.patch

119 lines
4.6 KiB
Diff
Raw Normal View History

From 33535060b4898c7883c3d98c4591bfc964541830 Mon Sep 17 00:00:00 2001
From: Petter Urkedal <paurkedal@gmail.com>
Date: Sun, 25 May 2014 15:57:03 +0200
Subject: [PATCH 1/3] Fix GC_finalized_malloc failure on disclaim_test (revert
part of commit 63fd11d)
The finalizer closure is placed in the first word in order to use the
lower bits to distinguish live objects from objects on the free list.
The downside of this is that we need one-word offset interior pointers,
and that GC_base does not return the start of the user region.
* fnlz_mlc.c (GC_finalized_disclaim): Move finalizer closure for
finalized object kinds back to the start of objects.
* fnlz_mlc.c (GC_init_finalized_malloc): Call
GC_register_displacement_inner() as GC_finalized_malloc returns
displaced pointer.
* fnlz_mlc.c (GC_core_finalized_malloc, GC_finalized_malloc): Store
fclos in first word of created object (instead of the last word);
return pointer to object displaced by 1 word.
---
diff --git a/fnlz_mlc.c b/fnlz_mlc.c
old mode 100755
new mode 100644
index 0dad28a..b8478d2
--- a/fnlz_mlc.c
+++ b/fnlz_mlc.c
@@ -28,13 +28,10 @@ STATIC int GC_finalized_kind = 0;
STATIC int GC_CALLBACK GC_finalized_disclaim(void *obj)
{
- void **fc_addr;
- const struct GC_finalizer_closure *fc;
+ word fc_word = *(word *)obj;
- fc_addr = &((void **)obj)[GC_size(obj) / sizeof(void *) - 1];
- fc = *fc_addr;
- if (fc != NULL) {
- /* [1] The disclaim function may be passed fragments from the */
+ if ((fc_word & 1) != 0) {
+ /* The disclaim function may be passed fragments from the */
/* free-list, on which it should not run finalization. */
/* To recognize this case, we use the fact that the first word */
/* on such fragments are always even (a link to the next */
@@ -42,8 +39,8 @@ STATIC int GC_CALLBACK GC_finalized_disclaim(void *obj)
/* which does not use the first word for storing finalization */
/* info, GC_reclaim_with_finalization must be extended to clear */
/* fragments so that the assumption holds for the selected word. */
- (*fc->proc)(obj, fc->cd);
- *fc_addr = NULL;
+ const struct GC_finalizer_closure *fc = (void *)(fc_word & ~(word)1);
+ (*fc->proc)((word *)obj + 1, fc->cd);
}
return 0;
}
@@ -62,6 +59,13 @@ GC_API void GC_CALL GC_init_finalized_malloc(void)
}
done_init = TRUE;
+ /* The finalizer closure is placed in the first word in order to */
+ /* use the lower bits to distinguish live objects from objects on */
+ /* the free list. The downside of this is that we need one-word */
+ /* offset interior pointers, and that GC_base does not return the */
+ /* start of the user region. */
+ GC_register_displacement_inner(sizeof(word));
+
GC_finalized_objfreelist = (ptr_t *)GC_new_free_list_inner();
GC_finalized_kind = GC_new_kind_inner((void **)GC_finalized_objfreelist,
GC_DS_LENGTH, TRUE, TRUE);
@@ -90,7 +94,7 @@ GC_API void GC_CALL GC_register_disclaim_proc(int kind, GC_disclaim_proc proc,
word lg;
DCL_LOCK_STATE;
- lb += sizeof(void *);
+ lb += sizeof(word);
GC_ASSERT(done_init);
if (SMALL_OBJ(lb)) {
GC_DBG_COLLECT_AT_MALLOC(lb);
@@ -112,7 +116,6 @@ GC_API void GC_CALL GC_register_disclaim_proc(int kind, GC_disclaim_proc proc,
UNLOCK();
}
GC_ASSERT(lg > 0);
- ((const void **)op)[GRANULES_TO_WORDS(lg) - 1] = fclos;
} else {
size_t op_sz;
@@ -121,16 +124,16 @@ GC_API void GC_CALL GC_register_disclaim_proc(int kind, GC_disclaim_proc proc,
return NULL;
op_sz = GC_size(op);
GC_ASSERT(op_sz >= lb);
- ((const void **)op)[op_sz / sizeof(void *) - 1] = fclos;
}
- return GC_clear_stack(op);
+ *(word *)op = (word)fclos | 1;
+ return GC_clear_stack((word *)op + 1);
}
#ifdef THREAD_LOCAL_ALLOC
GC_API void * GC_CALL GC_finalized_malloc(size_t client_lb,
const struct GC_finalizer_closure *fclos)
{
- size_t lb = client_lb + sizeof(void *);
+ size_t lb = client_lb + sizeof(word);
size_t lg = ROUNDED_UP_GRANULES(lb);
GC_tlfs tsd;
void *result;
@@ -163,9 +166,9 @@ GC_API void GC_CALL GC_register_disclaim_proc(int kind, GC_disclaim_proc proc,
result = (void *)my_entry;
*my_fl = next;
obj_link(result) = 0;
- ((const void **)result)[GRANULES_TO_WORDS(lg) - 1] = fclos;
+ *(word *)result = (word)fclos | 1;
PREFETCH_FOR_WRITE(next);
- return result;
+ return (word *)result + 1;
}
#endif /* THREAD_LOCAL_ALLOC */