#ifdef ENABLE_DISCLAIM
#include "gc_disclaim.h"
-
-#ifdef THREAD_LOCAL_ALLOC
-# include "private/thread_local_alloc.h"
-#else
- STATIC ptr_t * GC_finalized_objfreelist = NULL;
-#endif /* !THREAD_LOCAL_ALLOC */
+#include "gc_inline.h" /* for GC_malloc_kind */
STATIC int GC_finalized_kind = 0;
/* 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_finalized_kind = GC_new_kind_inner(GC_new_free_list_inner(),
GC_DS_LENGTH, TRUE, TRUE);
GC_register_disclaim_proc(GC_finalized_kind, GC_finalized_disclaim, TRUE);
UNLOCK();
GC_obj_kinds[kind].ok_mark_unconditionally = (GC_bool)mark_unconditionally;
}
-#ifdef THREAD_LOCAL_ALLOC
- STATIC void * GC_core_finalized_malloc(size_t lb,
- const struct GC_finalizer_closure *fclos)
-#else
- GC_API GC_ATTR_MALLOC void * GC_CALL GC_finalized_malloc(size_t lb,
+GC_API GC_ATTR_MALLOC void * GC_CALL GC_finalized_malloc(size_t lb,
const struct GC_finalizer_closure *fclos)
-#endif
{
- ptr_t op;
- word lg;
- DCL_LOCK_STATE;
+ word *op;
- lb += sizeof(word);
GC_ASSERT(done_init);
- if (SMALL_OBJ(lb)) {
- GC_DBG_COLLECT_AT_MALLOC(lb);
- lg = GC_size_map[lb];
- LOCK();
- op = GC_finalized_objfreelist[lg];
- if (EXPECT(0 == op, FALSE)) {
- UNLOCK();
- op = GC_generic_malloc(lb, GC_finalized_kind);
- if (NULL == op)
- return NULL;
- /* GC_generic_malloc has extended the size map for us. */
- lg = GC_size_map[lb];
- } else {
- GC_finalized_objfreelist[lg] = obj_link(op);
- obj_link(op) = 0;
- GC_bytes_allocd += GRANULES_TO_BYTES(lg);
- UNLOCK();
- }
- GC_ASSERT(lg > 0);
- } else {
- op = GC_generic_malloc(lb, GC_finalized_kind);
- if (NULL == op)
- return NULL;
- GC_ASSERT(GC_size(op) >= lb);
- }
- *(word *)op = (word)fclos | 1;
- return GC_clear_stack((word *)op + 1);
+ op = GC_malloc_kind(lb + sizeof(word), GC_finalized_kind);
+ if (EXPECT(NULL == op, FALSE))
+ return NULL;
+ *op = (word)fclos | 1;
+ return op + 1;
}
-#ifdef THREAD_LOCAL_ALLOC
- GC_API GC_ATTR_MALLOC void * GC_CALL GC_finalized_malloc(size_t client_lb,
- const struct GC_finalizer_closure *fclos)
- {
- size_t lb = client_lb + sizeof(word);
- size_t lg = ROUNDED_UP_GRANULES(lb);
- GC_tlfs tsd;
- void *result;
- void **tiny_fl, **my_fl, *my_entry;
- void *next;
-
- if (EXPECT(lg >= GC_TINY_FREELISTS, FALSE))
- return GC_core_finalized_malloc(client_lb, fclos);
-
- tsd = GC_getspecific(GC_thread_key);
- tiny_fl = tsd->finalized_freelists;
- my_fl = tiny_fl + lg;
- my_entry = *my_fl;
- while (EXPECT((word)my_entry
- <= DIRECT_GRANULES + GC_TINY_FREELISTS + 1, FALSE)) {
- if ((word)my_entry - 1 < DIRECT_GRANULES) {
- *my_fl = (ptr_t)my_entry + lg + 1;
- return GC_core_finalized_malloc(client_lb, fclos);
- } else {
- GC_generic_malloc_many(GC_RAW_BYTES_FROM_INDEX(lg),
- GC_finalized_kind, my_fl);
- my_entry = *my_fl;
- if (my_entry == 0) {
- return (*GC_get_oom_fn())(lb);
- }
- }
- }
-
- next = obj_link(my_entry);
- result = (void *)my_entry;
- *my_fl = next;
- obj_link(result) = 0;
- *(word *)result = (word)fclos | 1;
- GC_PREFETCH_FOR_WRITE(next);
- return (word *)result + 1;
- }
-#endif /* THREAD_LOCAL_ALLOC */
-
#endif /* ENABLE_DISCLAIM */
#include <stdlib.h>
#ifndef THREAD_FREELISTS_KINDS
-# define THREAD_FREELISTS_KINDS (NORMAL+1)
-#endif
+# ifdef ENABLE_DISCLAIM
+# define THREAD_FREELISTS_KINDS (NORMAL+2)
+# else
+# define THREAD_FREELISTS_KINDS (NORMAL+1)
+# endif
+#endif /* !THREAD_FREELISTS_KINDS */
/* One of these should be declared as the tlfs field in the */
/* structure pointed to by a GC_thread. */
# define ERROR_FL ((void *)(word)-1)
/* Value used for gcj_freelists[-1]; allocation is */
/* erroneous. */
-# endif
-# ifdef ENABLE_DISCLAIM
- void * finalized_freelists[TINY_FREELISTS];
# endif
/* Free lists contain either a pointer or a small count */
/* reflecting the number of granules allocated at that */
/* we take care of an individual thread freelist structure. */
GC_INNER void GC_mark_thread_local_fls_for(GC_tlfs p);
-#ifdef ENABLE_DISCLAIM
- GC_EXTERN ptr_t * GC_finalized_objfreelist;
-#endif
-
#ifndef GC_ATTR_TLS_FAST
# define GC_ATTR_TLS_FAST /* empty */
#endif
static GC_bool keys_initialized;
-#ifdef ENABLE_DISCLAIM
- GC_INNER ptr_t * GC_finalized_objfreelist = NULL;
- /* This variable is declared here to prevent linking of */
- /* fnlz_mlc module unless the client uses the latter one. */
-#endif
-
/* Return a single nonempty freelist fl to the global one pointed to */
/* by gfl. */
}
# ifdef GC_GCJ_SUPPORT
p -> gcj_freelists[j] = (void *)(word)1;
-# endif
-# ifdef ENABLE_DISCLAIM
- p -> finalized_freelists[j] = (void *)(word)1;
# endif
}
/* The size 0 free lists are handled like the regular free lists, */
# ifdef GC_GCJ_SUPPORT
return_freelists(p -> gcj_freelists, (void **)GC_gcjobjfreelist);
# endif
-# ifdef ENABLE_DISCLAIM
- return_freelists(p -> finalized_freelists,
- (void **)GC_finalized_objfreelist);
-# endif
}
#ifdef GC_ASSERTIONS
if ((word)q > HBLKSIZE)
GC_set_fl_marks(q);
}
-# endif
-# ifdef ENABLE_DISCLAIM
- q = p -> finalized_freelists[j];
- if ((word)q > HBLKSIZE)
- GC_set_fl_marks(q);
# endif
}
}
}
# ifdef GC_GCJ_SUPPORT
GC_check_fl_marks(&p->gcj_freelists[j]);
-# endif
-# ifdef ENABLE_DISCLAIM
- GC_check_fl_marks(&p->finalized_freelists[j]);
# endif
}
}