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