forked from ivmai/bdwgc
-
Notifications
You must be signed in to change notification settings - Fork 0
/
fnlz_mlc.c
128 lines (112 loc) · 4.68 KB
/
fnlz_mlc.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
/*
* Copyright (c) 2011 by Hewlett-Packard Company. All rights reserved.
*
* THIS MATERIAL IS PROVIDED AS IS, WITH ABSOLUTELY NO WARRANTY EXPRESSED
* OR IMPLIED. ANY USE IS AT YOUR OWN RISK.
*
* Permission is hereby granted to use or copy this program
* for any purpose, provided the above notices are retained on all copies.
* Permission to modify the code and to distribute modified code is granted,
* provided the above notices are retained, and a notice that the code was
* modified is included with the above copyright notice.
*
*/
#include "private/gc_priv.h"
#ifdef ENABLE_DISCLAIM
#include "gc/gc_disclaim.h"
#include "private/dbg_mlc.h" /* for oh type */
#if defined(KEEP_BACK_PTRS) || defined(MAKE_BACK_GRAPH)
/* The first bit is already used for a debug purpose. */
# define FINALIZER_CLOSURE_FLAG 0x2
#else
# define FINALIZER_CLOSURE_FLAG 0x1
#endif
STATIC int GC_CALLBACK GC_finalized_disclaim(void *obj)
{
# ifdef AO_HAVE_load
word fc_word = (word)AO_load((volatile AO_t *)obj);
# else
word fc_word = *(word *)obj;
# endif
if ((fc_word & FINALIZER_CLOSURE_FLAG) != 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 is always multiple of 4 (a link to the next */
/* fragment, or NULL). If it is desirable to have a finalizer */
/* which does not use the first word for storing finalization */
/* info, GC_disclaim_and_reclaim() must be extended to clear */
/* fragments so that the assumption holds for the selected word. */
const struct GC_finalizer_closure *fc
= (struct GC_finalizer_closure *)(fc_word
& ~(word)FINALIZER_CLOSURE_FLAG);
GC_ASSERT(!GC_find_leak);
(*fc->proc)((word *)obj + 1, fc->cd);
}
return 0;
}
STATIC void GC_register_disclaim_proc_inner(unsigned kind,
GC_disclaim_proc proc,
GC_bool mark_unconditionally)
{
GC_ASSERT(kind < MAXOBJKINDS);
if (EXPECT(GC_find_leak, FALSE)) return;
GC_obj_kinds[kind].ok_disclaim_proc = proc;
GC_obj_kinds[kind].ok_mark_unconditionally = mark_unconditionally;
}
GC_API void GC_CALL GC_init_finalized_malloc(void)
{
GC_init(); /* In case it's not already done. */
LOCK();
if (GC_finalized_kind != 0) {
UNLOCK();
return;
}
/* 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));
/* And, the pointer to the finalizer closure object itself is */
/* displaced due to baking in this indicator. */
GC_register_displacement_inner(FINALIZER_CLOSURE_FLAG);
GC_register_displacement_inner(sizeof(oh) + FINALIZER_CLOSURE_FLAG);
GC_finalized_kind = GC_new_kind_inner(GC_new_free_list_inner(),
GC_DS_LENGTH, TRUE, TRUE);
GC_ASSERT(GC_finalized_kind != 0);
GC_register_disclaim_proc_inner(GC_finalized_kind, GC_finalized_disclaim,
TRUE);
UNLOCK();
}
GC_API void GC_CALL GC_register_disclaim_proc(int kind, GC_disclaim_proc proc,
int mark_unconditionally)
{
LOCK();
GC_register_disclaim_proc_inner((unsigned)kind, proc,
(GC_bool)mark_unconditionally);
UNLOCK();
}
GC_API GC_ATTR_MALLOC void * GC_CALL GC_finalized_malloc(size_t lb,
const struct GC_finalizer_closure *fclos)
{
void *op;
# ifndef LINT2 /* no data race because the variable is set once */
GC_ASSERT(GC_finalized_kind != 0);
# endif
GC_ASSERT(NONNULL_ARG_NOT_NULL(fclos));
GC_ASSERT((ADDR(fclos) & FINALIZER_CLOSURE_FLAG) == 0);
op = GC_malloc_kind(SIZET_SAT_ADD(lb, sizeof(word)),
(int)GC_finalized_kind);
if (EXPECT(NULL == op, FALSE))
return NULL;
# ifdef AO_HAVE_store
AO_store((volatile AO_t *)op, (AO_t)fclos | FINALIZER_CLOSURE_FLAG);
# else
*(word *)op = (word)fclos | FINALIZER_CLOSURE_FLAG;
# endif
GC_dirty(op);
REACHABLE_AFTER_DIRTY(fclos);
return (word *)op + 1;
}
#endif /* ENABLE_DISCLAIM */