Merge branch 'master' into boehm-demers-weiser-gc
[bpt/guile.git] / libguile / guardians.c
1 /* Copyright (C) 1998,1999,2000,2001, 2006, 2008 Free Software Foundation, Inc.
2 *
3 * This library is free software; you can redistribute it and/or
4 * modify it under the terms of the GNU Lesser General Public License
5 * as published by the Free Software Foundation; either version 3 of
6 * the License, or (at your option) any later version.
7 *
8 * This library is distributed in the hope that it will be useful, but
9 * WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
11 * Lesser General Public License for more details.
12 *
13 * You should have received a copy of the GNU Lesser General Public
14 * License along with this library; if not, write to the Free Software
15 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
16 * 02110-1301 USA
17 */
18
19 \f
20 /* This is an implementation of guardians as described in
21 * R. Kent Dybvig, Carl Bruggeman, and David Eby (1993) "Guardians in
22 * a Generation-Based Garbage Collector" ACM SIGPLAN Conference on
23 * Programming Language Design and Implementation, June 1993
24 * ftp://ftp.cs.indiana.edu/pub/scheme-repository/doc/pubs/guardians.ps.gz
25 *
26 * Original design: Mikael Djurfeldt
27 * Original implementation: Michael Livshin
28 * Hacked on since by: everybody
29 *
30 * By this point, the semantics are actually quite different from
31 * those described in the abovementioned paper. The semantic changes
32 * are there to improve safety and intuitiveness. The interface is
33 * still (mostly) the one described by the paper, however.
34 *
35 * Boiled down again: Marius Vollmer
36 *
37 * Now they should again behave like those described in the paper.
38 * Scheme guardians should be simple and friendly, not like the greedy
39 * monsters we had...
40 *
41 * Rewritten for the Boehm-Demers-Weiser GC by Ludovic Courtès.
42 * FIXME: This is currently not thread-safe.
43 */
44
45 /* Uncomment the following line to debug guardian finalization. */
46 /* #define DEBUG_GUARDIANS 1 */
47
48 #ifdef HAVE_CONFIG_H
49 # include <config.h>
50 #endif
51
52 #include "libguile/_scm.h"
53 #include "libguile/async.h"
54 #include "libguile/ports.h"
55 #include "libguile/print.h"
56 #include "libguile/smob.h"
57 #include "libguile/validate.h"
58 #include "libguile/root.h"
59 #include "libguile/hashtab.h"
60 #include "libguile/weaks.h"
61 #include "libguile/deprecation.h"
62 #include "libguile/eval.h"
63
64 #include "libguile/guardians.h"
65 #include "libguile/boehm-gc.h"
66
67
68
69
70 static scm_t_bits tc16_guardian;
71
72 typedef struct t_guardian
73 {
74 unsigned long live;
75 SCM zombies;
76 struct t_guardian *next;
77 } t_guardian;
78
79 #define GUARDIAN_P(x) SCM_SMOB_PREDICATE(tc16_guardian, x)
80 #define GUARDIAN_DATA(x) ((t_guardian *) SCM_CELL_WORD_1 (x))
81
82
83
84
85 static int
86 guardian_print (SCM guardian, SCM port, scm_print_state *pstate SCM_UNUSED)
87 {
88 t_guardian *g = GUARDIAN_DATA (guardian);
89
90 scm_puts ("#<guardian ", port);
91 scm_uintprint ((scm_t_bits) g, 16, port);
92
93 scm_puts (" (reachable: ", port);
94 scm_display (scm_from_uint (g->live), port);
95 scm_puts (" unreachable: ", port);
96 scm_display (scm_length (g->zombies), port);
97 scm_puts (")", port);
98
99 scm_puts (">", port);
100
101 return 1;
102 }
103
104 /* Handle finalization of OBJ which is guarded by the guardians listed in
105 GUARDIAN_LIST. */
106 static void
107 finalize_guarded (GC_PTR ptr, GC_PTR finalizer_data)
108 {
109 SCM cell_pool;
110 SCM obj, guardian_list, proxied_finalizer;
111
112 obj = PTR2SCM (ptr);
113 guardian_list = SCM_CDR (PTR2SCM (finalizer_data));
114 proxied_finalizer = SCM_CAR (PTR2SCM (finalizer_data));
115
116 #ifdef DEBUG_GUARDIANS
117 printf ("finalizing guarded %p (%u guardians)\n",
118 ptr, scm_to_uint (scm_length (guardian_list)));
119 #endif
120
121 /* Preallocate a bunch of cells so that we can make sure that no garbage
122 collection (and, thus, nested calls to `finalize_guarded ()') occurs
123 while executing the following loop. This is quite inefficient (call to
124 `scm_length ()') but that shouldn't be a problem in most cases. */
125 cell_pool = scm_make_list (scm_length (guardian_list), SCM_UNSPECIFIED);
126
127 /* Tell each guardian interested in OBJ that OBJ is no longer
128 reachable. */
129 for (;
130 guardian_list != SCM_EOL;
131 guardian_list = SCM_CDR (guardian_list))
132 {
133 SCM zombies;
134 t_guardian *g;
135
136 if (SCM_WEAK_PAIR_CAR_DELETED_P (guardian_list))
137 {
138 /* The guardian itself vanished in the meantime. */
139 #ifdef DEBUG_GUARDIANS
140 printf (" guardian for %p vanished\n", ptr);
141 #endif
142 continue;
143 }
144
145 g = GUARDIAN_DATA (SCM_CAR (guardian_list));
146 if (g->live == 0)
147 abort ();
148
149 /* Get a fresh cell from CELL_POOL. */
150 zombies = cell_pool;
151 cell_pool = SCM_CDR (cell_pool);
152
153 /* Compute and update G's zombie list. */
154 SCM_SETCAR (zombies, SCM_PACK (obj));
155 SCM_SETCDR (zombies, g->zombies);
156 g->zombies = zombies;
157
158 g->live--;
159 g->zombies = zombies;
160 }
161
162 if (proxied_finalizer != SCM_BOOL_F)
163 {
164 /* Re-register the finalizer that was in place before we installed this
165 one. */
166 GC_finalization_proc finalizer, prev_finalizer;
167 GC_PTR finalizer_data, prev_finalizer_data;
168
169 finalizer = (GC_finalization_proc) SCM2PTR (SCM_CAR (proxied_finalizer));
170 finalizer_data = SCM2PTR (SCM_CDR (proxied_finalizer));
171
172 if (finalizer == NULL)
173 abort ();
174
175 GC_REGISTER_FINALIZER_NO_ORDER (ptr, finalizer, finalizer_data,
176 &prev_finalizer, &prev_finalizer_data);
177
178 #ifdef DEBUG_GUARDIANS
179 printf (" reinstalled proxied finalizer %p for %p\n", finalizer, ptr);
180 #endif
181 }
182
183 #ifdef DEBUG_GUARDIANS
184 printf ("end of finalize (%p)\n", ptr);
185 #endif
186 }
187
188 /* Add OBJ as a guarded object of GUARDIAN. */
189 static void
190 scm_i_guard (SCM guardian, SCM obj)
191 {
192 t_guardian *g = GUARDIAN_DATA (guardian);
193
194 if (SCM_NIMP (obj))
195 {
196 /* Register a finalizer and pass a pair as the ``client data''
197 argument. The pair contains in its car `#f' or a pair describing a
198 ``proxied'' finalizer (see below); its cdr contains a list of
199 guardians interested in OBJ.
200
201 A ``proxied'' finalizer is a finalizer that was registered for OBJ
202 before OBJ became guarded (e.g., a SMOB `free' function). We are
203 assuming here that finalizers are only used internally, either at
204 the very beginning of an object's lifetime (e.g., see `SCM_NEWSMOB')
205 or by this function. */
206 GC_finalization_proc prev_finalizer;
207 GC_PTR prev_data;
208 SCM guardians_for_obj, finalizer_data;
209
210 g->live++;
211
212 /* Note: GUARDIANS_FOR_OBJ is a weak list so that a guardian can be
213 collected before the objects it guards (see `guardians.test'). */
214 guardians_for_obj = scm_weak_car_pair (guardian, SCM_EOL);
215 finalizer_data = scm_cons (SCM_BOOL_F, guardians_for_obj);
216
217 GC_REGISTER_FINALIZER_NO_ORDER (SCM2PTR (obj), finalize_guarded,
218 SCM2PTR (finalizer_data),
219 &prev_finalizer, &prev_data);
220
221 if (prev_finalizer == finalize_guarded)
222 {
223 /* OBJ is already guarded by another guardian: add GUARDIAN to its
224 list of guardians. */
225 SCM prev_guardian_list, prev_finalizer_data;
226
227 if (prev_data == NULL)
228 abort ();
229
230 prev_finalizer_data = PTR2SCM (prev_data);
231 if (!scm_is_pair (prev_finalizer_data))
232 abort ();
233
234 prev_guardian_list = SCM_CDR (prev_finalizer_data);
235 SCM_SETCDR (guardians_for_obj, prev_guardian_list);
236
237 /* Also copy information about proxied finalizers. */
238 SCM_SETCAR (finalizer_data, SCM_CAR (prev_finalizer_data));
239 }
240 else if (prev_finalizer != NULL)
241 {
242 /* There was already a finalizer registered for OBJ so we will
243 ``proxy'' it, i.e., record it so that we can re-register it once
244 `finalize_guarded ()' has finished. */
245 SCM proxied_finalizer;
246
247 proxied_finalizer = scm_cons (PTR2SCM (prev_finalizer),
248 PTR2SCM (prev_data));
249 SCM_SETCAR (finalizer_data, proxied_finalizer);
250 }
251 }
252 }
253
254 static SCM
255 scm_i_get_one_zombie (SCM guardian)
256 {
257 t_guardian *g = GUARDIAN_DATA (guardian);
258 SCM res = SCM_BOOL_F;
259
260 if (g->zombies != SCM_EOL)
261 {
262 /* Note: We return zombies in reverse order. */
263 res = SCM_CAR (g->zombies);
264 g->zombies = SCM_CDR (g->zombies);
265 }
266
267 return res;
268 }
269
270 /* This is the Scheme entry point for each guardian: If OBJ is an
271 * object, it's added to the guardian's live list. If OBJ is unbound,
272 * the next available unreachable object (or #f if none) is returned.
273 *
274 * If the second optional argument THROW_P is true (the default), then
275 * an error is raised if GUARDIAN is greedy and OBJ is already greedily
276 * guarded. If THROW_P is false, #f is returned instead of raising the
277 * error, and #t is returned if everything is fine.
278 */
279 static SCM
280 guardian_apply (SCM guardian, SCM obj, SCM throw_p)
281 {
282 #if ENABLE_DEPRECATED
283 if (!SCM_UNBNDP (throw_p))
284 scm_c_issue_deprecation_warning
285 ("Using the 'throw?' argument of a guardian is deprecated "
286 "and ineffective.");
287 #endif
288
289 if (!SCM_UNBNDP (obj))
290 {
291 scm_i_guard (guardian, obj);
292 return SCM_UNSPECIFIED;
293 }
294 else
295 return scm_i_get_one_zombie (guardian);
296 }
297
298 SCM_DEFINE (scm_make_guardian, "make-guardian", 0, 0, 0,
299 (),
300 "Create a new guardian. A guardian protects a set of objects from\n"
301 "garbage collection, allowing a program to apply cleanup or other\n"
302 "actions.\n"
303 "\n"
304 "@code{make-guardian} returns a procedure representing the guardian.\n"
305 "Calling the guardian procedure with an argument adds the argument to\n"
306 "the guardian's set of protected objects. Calling the guardian\n"
307 "procedure without an argument returns one of the protected objects\n"
308 "which are ready for garbage collection, or @code{#f} if no such object\n"
309 "is available. Objects which are returned in this way are removed from\n"
310 "the guardian.\n"
311 "\n"
312 "You can put a single object into a guardian more than once and you can\n"
313 "put a single object into more than one guardian. The object will then\n"
314 "be returned multiple times by the guardian procedures.\n"
315 "\n"
316 "An object is eligible to be returned from a guardian when it is no\n"
317 "longer referenced from outside any guardian.\n"
318 "\n"
319 "There is no guarantee about the order in which objects are returned\n"
320 "from a guardian. If you want to impose an order on finalization\n"
321 "actions, for example, you can do that by keeping objects alive in some\n"
322 "global data structure until they are no longer needed for finalizing\n"
323 "other objects.\n"
324 "\n"
325 "Being an element in a weak vector, a key in a hash table with weak\n"
326 "keys, or a value in a hash table with weak value does not prevent an\n"
327 "object from being returned by a guardian. But as long as an object\n"
328 "can be returned from a guardian it will not be removed from such a\n"
329 "weak vector or hash table. In other words, a weak link does not\n"
330 "prevent an object from being considered collectable, but being inside\n"
331 "a guardian prevents a weak link from being broken.\n"
332 "\n"
333 "A key in a weak key hash table can be though of as having a strong\n"
334 "reference to its associated value as long as the key is accessible.\n"
335 "Consequently, when the key only accessible from within a guardian, the\n"
336 "reference from the key to the value is also considered to be coming\n"
337 "from within a guardian. Thus, if there is no other reference to the\n"
338 "value, it is eligible to be returned from a guardian.\n")
339 #define FUNC_NAME s_scm_make_guardian
340 {
341 t_guardian *g = scm_gc_malloc (sizeof (t_guardian), "guardian");
342 SCM z;
343
344 /* A tconc starts out with one tail pair. */
345 g->live = 0;
346 g->zombies = SCM_EOL;
347
348 g->next = NULL;
349
350 SCM_NEWSMOB (z, tc16_guardian, g);
351
352 return z;
353 }
354 #undef FUNC_NAME
355
356 void
357 scm_init_guardians ()
358 {
359 /* We use unordered finalization `a la Java. */
360 GC_java_finalization = 1;
361
362 tc16_guardian = scm_make_smob_type ("guardian", 0);
363
364 scm_set_smob_print (tc16_guardian, guardian_print);
365 #if ENABLE_DEPRECATED
366 scm_set_smob_apply (tc16_guardian, guardian_apply, 0, 2, 0);
367 #else
368 scm_set_smob_apply (tc16_guardian, guardian_apply, 0, 1, 0);
369 #endif
370
371 #include "libguile/guardians.x"
372 }
373
374 /*
375 Local Variables:
376 c-file-style: "gnu"
377 End:
378 */