Don't register disappearing links for non-heap objects.
[bpt/guile.git] / libguile / weaks.c
CommitLineData
dbe4d258 1/* Copyright (C) 1995,1996,1998,2000,2001, 2003, 2006, 2008, 2009 Free Software Foundation, Inc.
0f2d19dd 2 *
d3cf93bc
NJ
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.
0f2d19dd 7 *
d3cf93bc
NJ
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.
0f2d19dd 12 *
d3cf93bc
NJ
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 */
1bbd0b84 17
1bbd0b84 18
0f2d19dd 19\f
dbb605f5
LC
20#ifdef HAVE_CONFIG_H
21# include <config.h>
22#endif
592996c9 23
06c1d900
MV
24#include <stdio.h>
25
a0599745
MD
26#include "libguile/_scm.h"
27#include "libguile/vectors.h"
c96d76b8 28#include "libguile/lang.h"
f59a096e 29#include "libguile/hashtab.h"
0f2d19dd 30
a0599745
MD
31#include "libguile/validate.h"
32#include "libguile/weaks.h"
0f2d19dd 33
e7bca227 34#include "libguile/boehm-gc.h"
986ec822
LC
35#include <gc/gc_typed.h>
36
37
38\f
39/* Weak pairs for use in weak alist vectors and weak hash tables.
40
41 We have weal-car pairs, weak-cdr pairs, and doubly weak pairs. In weak
42 pairs, the weak component(s) are not scanned for pointers and are
43 registered as disapperaring links; therefore, the weak component may be
44 set to NULL by the garbage collector when no other reference to that word
45 exist. Thus, users should only access weak pairs via the
46 `SCM_WEAK_PAIR_C[AD]R ()' macros. See also `scm_fixup_weak_alist ()' in
47 `hashtab.c'. */
48
49/* Type descriptors for weak-c[ad]r pairs. */
50static GC_descr wcar_pair_descr, wcdr_pair_descr;
51
52
53SCM
54scm_weak_car_pair (SCM car, SCM cdr)
55{
56 scm_t_cell *cell;
57
58 cell = (scm_t_cell *)GC_malloc_explicitly_typed (sizeof (*cell),
59 wcar_pair_descr);
60
61 cell->word_0 = car;
62 cell->word_1 = cdr;
63
64 if (SCM_NIMP (car))
65 {
66 /* Weak car cells make sense iff the car is non-immediate. */
dbe4d258
LC
67 SCM_I_REGISTER_DISAPPEARING_LINK ((GC_PTR) &cell->word_0,
68 (GC_PTR) SCM_UNPACK (car));
986ec822
LC
69 }
70
71 return (SCM_PACK (cell));
72}
73
74SCM
75scm_weak_cdr_pair (SCM car, SCM cdr)
76{
77 scm_t_cell *cell;
78
79 cell = (scm_t_cell *)GC_malloc_explicitly_typed (sizeof (*cell),
80 wcdr_pair_descr);
81
82 cell->word_0 = car;
83 cell->word_1 = cdr;
84
85 if (SCM_NIMP (cdr))
86 {
87 /* Weak cdr cells make sense iff the cdr is non-immediate. */
dbe4d258
LC
88 SCM_I_REGISTER_DISAPPEARING_LINK ((GC_PTR) &cell->word_1,
89 (GC_PTR) SCM_UNPACK (cdr));
986ec822
LC
90 }
91
92 return (SCM_PACK (cell));
93}
94
95SCM
96scm_doubly_weak_pair (SCM car, SCM cdr)
97{
98 /* Doubly weak cells shall not be scanned at all for pointers. */
99 scm_t_cell *cell = (scm_t_cell *)scm_gc_malloc_pointerless (sizeof (*cell),
100 "weak cell");
101
102 cell->word_0 = car;
103 cell->word_1 = cdr;
104
105 if (SCM_NIMP (car))
106 {
dbe4d258
LC
107 SCM_I_REGISTER_DISAPPEARING_LINK ((GC_PTR) &cell->word_0,
108 (GC_PTR) SCM_UNPACK (car));
986ec822
LC
109 }
110 if (SCM_NIMP (cdr))
111 {
dbe4d258
LC
112 SCM_I_REGISTER_DISAPPEARING_LINK ((GC_PTR) &cell->word_1,
113 (GC_PTR) SCM_UNPACK (cdr));
986ec822
LC
114 }
115
116 return (SCM_PACK (cell));
117}
118
119
592996c9 120\f
0f2d19dd 121
c35738c1
MD
122/* 1. The current hash table implementation in hashtab.c uses weak alist
123 * vectors (formerly called weak hash tables) internally.
124 *
125 * 2. All hash table operations still work on alist vectors.
126 *
127 * 3. The weak vector and alist vector Scheme API is accessed through
128 * the module (ice-9 weak-vector).
129 */
130
131
0f2d19dd
JB
132/* {Weak Vectors}
133 */
134
135
3b3b36dd 136SCM_DEFINE (scm_make_weak_vector, "make-weak-vector", 1, 1, 0,
1e6808ea 137 (SCM size, SCM fill),
b380b885 138 "Return a weak vector with @var{size} elements. If the optional\n"
1e6808ea
MG
139 "argument @var{fill} is given, all entries in the vector will be\n"
140 "set to @var{fill}. The default value for @var{fill} is the\n"
141 "empty list.")
1bbd0b84 142#define FUNC_NAME s_scm_make_weak_vector
0f2d19dd 143{
d525e4f9 144 return scm_i_make_weak_vector (0, size, fill);
0f2d19dd 145}
1bbd0b84 146#undef FUNC_NAME
0f2d19dd
JB
147
148
1bbd0b84 149SCM_REGISTER_PROC(s_list_to_weak_vector, "list->weak-vector", 1, 0, 0, scm_weak_vector);
1cc91f1b 150
3b3b36dd 151SCM_DEFINE (scm_weak_vector, "weak-vector", 0, 0, 1,
1bbd0b84 152 (SCM l),
8f85c0c6 153 "@deffnx {Scheme Procedure} list->weak-vector l\n"
1e6808ea
MG
154 "Construct a weak vector from a list: @code{weak-vector} uses\n"
155 "the list of its arguments while @code{list->weak-vector} uses\n"
156 "its only argument @var{l} (a list) to construct a weak vector\n"
157 "the same way @code{list->vector} would.")
1bbd0b84 158#define FUNC_NAME s_scm_weak_vector
0f2d19dd 159{
d525e4f9 160 return scm_i_make_weak_vector_from_list (0, l);
0f2d19dd 161}
1bbd0b84 162#undef FUNC_NAME
0f2d19dd
JB
163
164
3b3b36dd 165SCM_DEFINE (scm_weak_vector_p, "weak-vector?", 1, 0, 0,
1e6808ea 166 (SCM obj),
5352393c
MG
167 "Return @code{#t} if @var{obj} is a weak vector. Note that all\n"
168 "weak hashes are also weak vectors.")
1bbd0b84 169#define FUNC_NAME s_scm_weak_vector_p
0f2d19dd 170{
6e708ef2 171 return scm_from_bool (SCM_I_WVECTP (obj) && !SCM_IS_WHVEC (obj));
0f2d19dd 172}
1bbd0b84 173#undef FUNC_NAME
0f2d19dd 174
0f2d19dd 175\f
3a2de079
LC
176/* Weak alist vectors, i.e., vectors of alists.
177
178 The alist vector themselves are _not_ weak. The `car' (or `cdr', or both)
179 of the pairs within it are weak. See `hashtab.c' for details. */
0f2d19dd 180
4650cdd2
LC
181
182/* FIXME: We used to have two implementations of weak hash tables: the one in
183 here and the one in `hashtab.c'. The difference is that weak alist
184 vectors could be used as vectors while (weak) hash tables can't. We need
185 to unify that. */
186
c35738c1 187SCM_DEFINE (scm_make_weak_key_alist_vector, "make-weak-key-alist-vector", 0, 1, 0,
1e6808ea 188 (SCM size),
c35738c1
MD
189 "@deffnx {Scheme Procedure} make-weak-value-alist-vector size\n"
190 "@deffnx {Scheme Procedure} make-doubly-weak-alist-vector size\n"
1e6808ea
MG
191 "Return a weak hash table with @var{size} buckets. As with any\n"
192 "hash table, choosing a good size for the table requires some\n"
193 "caution.\n"
194 "\n"
195 "You can modify weak hash tables in exactly the same way you\n"
196 "would modify regular hash tables. (@pxref{Hash Tables})")
c35738c1 197#define FUNC_NAME s_scm_make_weak_key_alist_vector
0f2d19dd 198{
4650cdd2 199 return scm_make_weak_key_hash_table (size);
0f2d19dd 200}
1bbd0b84 201#undef FUNC_NAME
0f2d19dd
JB
202
203
c35738c1 204SCM_DEFINE (scm_make_weak_value_alist_vector, "make-weak-value-alist-vector", 0, 1, 0,
1e6808ea 205 (SCM size),
e3239868
DH
206 "Return a hash table with weak values with @var{size} buckets.\n"
207 "(@pxref{Hash Tables})")
c35738c1 208#define FUNC_NAME s_scm_make_weak_value_alist_vector
0f2d19dd 209{
4650cdd2 210 return scm_make_weak_value_hash_table (size);
0f2d19dd 211}
1bbd0b84 212#undef FUNC_NAME
0f2d19dd
JB
213
214
c35738c1 215SCM_DEFINE (scm_make_doubly_weak_alist_vector, "make-doubly-weak-alist-vector", 1, 0, 0,
1e6808ea 216 (SCM size),
e3239868
DH
217 "Return a hash table with weak keys and values with @var{size}\n"
218 "buckets. (@pxref{Hash Tables})")
c35738c1 219#define FUNC_NAME s_scm_make_doubly_weak_alist_vector
0f2d19dd 220{
b6ed39c4 221 return scm_make_doubly_weak_hash_table (size);
0f2d19dd 222}
1bbd0b84 223#undef FUNC_NAME
0f2d19dd 224
592996c9 225
c35738c1 226SCM_DEFINE (scm_weak_key_alist_vector_p, "weak-key-alist-vector?", 1, 0, 0,
1e6808ea 227 (SCM obj),
c35738c1
MD
228 "@deffnx {Scheme Procedure} weak-value-alist-vector? obj\n"
229 "@deffnx {Scheme Procedure} doubly-weak-alist-vector? obj\n"
5352393c
MG
230 "Return @code{#t} if @var{obj} is the specified weak hash\n"
231 "table. Note that a doubly weak hash table is neither a weak key\n"
232 "nor a weak value hash table.")
c35738c1 233#define FUNC_NAME s_scm_weak_key_alist_vector_p
0f2d19dd 234{
6e708ef2 235 return scm_from_bool (SCM_I_WVECTP (obj) && SCM_IS_WHVEC (obj));
0f2d19dd 236}
1bbd0b84 237#undef FUNC_NAME
0f2d19dd
JB
238
239
c35738c1 240SCM_DEFINE (scm_weak_value_alist_vector_p, "weak-value-alist-vector?", 1, 0, 0,
1e6808ea
MG
241 (SCM obj),
242 "Return @code{#t} if @var{obj} is a weak value hash table.")
c35738c1 243#define FUNC_NAME s_scm_weak_value_alist_vector_p
0f2d19dd 244{
6e708ef2 245 return scm_from_bool (SCM_I_WVECTP (obj) && SCM_IS_WHVEC_V (obj));
0f2d19dd 246}
1bbd0b84 247#undef FUNC_NAME
0f2d19dd
JB
248
249
c35738c1 250SCM_DEFINE (scm_doubly_weak_alist_vector_p, "doubly-weak-alist-vector?", 1, 0, 0,
1e6808ea
MG
251 (SCM obj),
252 "Return @code{#t} if @var{obj} is a doubly weak hash table.")
c35738c1 253#define FUNC_NAME s_scm_doubly_weak_alist_vector_p
0f2d19dd 254{
6e708ef2 255 return scm_from_bool (SCM_I_WVECTP (obj) && SCM_IS_WHVEC_B (obj));
0f2d19dd 256}
1bbd0b84 257#undef FUNC_NAME
0f2d19dd 258
592996c9 259
592996c9 260
06c1d900 261\f
c35738c1
MD
262SCM
263scm_init_weaks_builtins ()
264{
265#include "libguile/weaks.x"
266 return SCM_UNSPECIFIED;
267}
268
986ec822
LC
269void
270scm_weaks_prehistory ()
271{
272 /* Initialize weak pairs. */
273 GC_word wcar_pair_bitmap[GC_BITMAP_SIZE (scm_t_cell)] = { 0 };
274 GC_word wcdr_pair_bitmap[GC_BITMAP_SIZE (scm_t_cell)] = { 0 };
275
276 /* In a weak-car pair, only the second word must be scanned for
277 pointers. */
278 GC_set_bit (wcar_pair_bitmap, GC_WORD_OFFSET (scm_t_cell, word_1));
279 wcar_pair_descr = GC_make_descriptor (wcar_pair_bitmap,
280 GC_WORD_LEN (scm_t_cell));
281
282 /* Conversely, in a weak-cdr pair, only the first word must be scanned for
283 pointers. */
284 GC_set_bit (wcdr_pair_bitmap, GC_WORD_OFFSET (scm_t_cell, word_0));
285 wcdr_pair_descr = GC_make_descriptor (wcdr_pair_bitmap,
286 GC_WORD_LEN (scm_t_cell));
287
288}
289
0f2d19dd
JB
290void
291scm_init_weaks ()
0f2d19dd 292{
c35738c1
MD
293 scm_c_define_gsubr ("%init-weaks-builtins", 0, 0, 0,
294 scm_init_weaks_builtins);
0f2d19dd
JB
295}
296
89e00824
ML
297
298/*
299 Local Variables:
300 c-file-style: "gnu"
301 End:
302*/