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