weaks.c 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295
  1. /* Copyright (C) 1995,1996,1998,2000,2001, 2003, 2006, 2008, 2009, 2010, 2011 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. #ifdef HAVE_CONFIG_H
  19. # include <config.h>
  20. #endif
  21. #include <stdio.h>
  22. #include "libguile/_scm.h"
  23. #include "libguile/vectors.h"
  24. #include "libguile/hashtab.h"
  25. #include "libguile/validate.h"
  26. #include "libguile/weaks.h"
  27. #include "libguile/bdw-gc.h"
  28. #include <gc/gc_typed.h>
  29. /* Weak pairs for use in weak alist vectors and weak hash tables.
  30. We have weal-car pairs, weak-cdr pairs, and doubly weak pairs. In weak
  31. pairs, the weak component(s) are not scanned for pointers and are
  32. registered as disapperaring links; therefore, the weak component may be
  33. set to NULL by the garbage collector when no other reference to that word
  34. exist. Thus, users should only access weak pairs via the
  35. `SCM_WEAK_PAIR_C[AD]R ()' macros. See also `scm_fixup_weak_alist ()' in
  36. `hashtab.c'. */
  37. /* Type descriptors for weak-c[ad]r pairs. */
  38. static GC_descr wcar_pair_descr, wcdr_pair_descr;
  39. SCM
  40. scm_weak_car_pair (SCM car, SCM cdr)
  41. {
  42. scm_t_cell *cell;
  43. cell = (scm_t_cell *)GC_malloc_explicitly_typed (sizeof (*cell),
  44. wcar_pair_descr);
  45. cell->word_0 = car;
  46. cell->word_1 = cdr;
  47. if (SCM_NIMP (car))
  48. /* Weak car cells make sense iff the car is non-immediate. */
  49. SCM_I_REGISTER_DISAPPEARING_LINK ((GC_PTR) &cell->word_0,
  50. (GC_PTR) SCM2PTR (car));
  51. return (SCM_PACK (cell));
  52. }
  53. SCM
  54. scm_weak_cdr_pair (SCM car, SCM cdr)
  55. {
  56. scm_t_cell *cell;
  57. cell = (scm_t_cell *)GC_malloc_explicitly_typed (sizeof (*cell),
  58. wcdr_pair_descr);
  59. cell->word_0 = car;
  60. cell->word_1 = cdr;
  61. if (SCM_NIMP (cdr))
  62. /* Weak cdr cells make sense iff the cdr is non-immediate. */
  63. SCM_I_REGISTER_DISAPPEARING_LINK ((GC_PTR) &cell->word_1,
  64. (GC_PTR) SCM2PTR (cdr));
  65. return (SCM_PACK (cell));
  66. }
  67. SCM
  68. scm_doubly_weak_pair (SCM car, SCM cdr)
  69. {
  70. /* Doubly weak cells shall not be scanned at all for pointers. */
  71. scm_t_cell *cell = (scm_t_cell *)scm_gc_malloc_pointerless (sizeof (*cell),
  72. "weak cell");
  73. cell->word_0 = car;
  74. cell->word_1 = cdr;
  75. if (SCM_NIMP (car))
  76. SCM_I_REGISTER_DISAPPEARING_LINK ((GC_PTR) &cell->word_0,
  77. (GC_PTR) SCM2PTR (car));
  78. if (SCM_NIMP (cdr))
  79. SCM_I_REGISTER_DISAPPEARING_LINK ((GC_PTR) &cell->word_1,
  80. (GC_PTR) SCM2PTR (cdr));
  81. return (SCM_PACK (cell));
  82. }
  83. /* 1. The current hash table implementation in hashtab.c uses weak alist
  84. * vectors (formerly called weak hash tables) internally.
  85. *
  86. * 2. All hash table operations still work on alist vectors.
  87. *
  88. * 3. The weak vector and alist vector Scheme API is accessed through
  89. * the module (ice-9 weak-vector).
  90. */
  91. /* {Weak Vectors}
  92. */
  93. SCM_DEFINE (scm_make_weak_vector, "make-weak-vector", 1, 1, 0,
  94. (SCM size, SCM fill),
  95. "Return a weak vector with @var{size} elements. If the optional\n"
  96. "argument @var{fill} is given, all entries in the vector will be\n"
  97. "set to @var{fill}. The default value for @var{fill} is the\n"
  98. "empty list.")
  99. #define FUNC_NAME s_scm_make_weak_vector
  100. {
  101. return scm_i_make_weak_vector (0, size, fill);
  102. }
  103. #undef FUNC_NAME
  104. SCM_REGISTER_PROC(s_list_to_weak_vector, "list->weak-vector", 1, 0, 0, scm_weak_vector);
  105. SCM_DEFINE (scm_weak_vector, "weak-vector", 0, 0, 1,
  106. (SCM l),
  107. "@deffnx {Scheme Procedure} list->weak-vector l\n"
  108. "Construct a weak vector from a list: @code{weak-vector} uses\n"
  109. "the list of its arguments while @code{list->weak-vector} uses\n"
  110. "its only argument @var{l} (a list) to construct a weak vector\n"
  111. "the same way @code{list->vector} would.")
  112. #define FUNC_NAME s_scm_weak_vector
  113. {
  114. return scm_i_make_weak_vector_from_list (0, l);
  115. }
  116. #undef FUNC_NAME
  117. SCM_DEFINE (scm_weak_vector_p, "weak-vector?", 1, 0, 0,
  118. (SCM obj),
  119. "Return @code{#t} if @var{obj} is a weak vector. Note that all\n"
  120. "weak hashes are also weak vectors.")
  121. #define FUNC_NAME s_scm_weak_vector_p
  122. {
  123. return scm_from_bool (SCM_I_WVECTP (obj) && !SCM_IS_WHVEC (obj));
  124. }
  125. #undef FUNC_NAME
  126. /* Weak alist vectors, i.e., vectors of alists.
  127. The alist vector themselves are _not_ weak. The `car' (or `cdr', or both)
  128. of the pairs within it are weak. See `hashtab.c' for details. */
  129. /* FIXME: We used to have two implementations of weak hash tables: the one in
  130. here and the one in `hashtab.c'. The difference is that weak alist
  131. vectors could be used as vectors while (weak) hash tables can't. We need
  132. to unify that. */
  133. SCM_DEFINE (scm_make_weak_key_alist_vector, "make-weak-key-alist-vector", 0, 1, 0,
  134. (SCM size),
  135. "@deffnx {Scheme Procedure} make-weak-value-alist-vector size\n"
  136. "@deffnx {Scheme Procedure} make-doubly-weak-alist-vector size\n"
  137. "Return a weak hash table with @var{size} buckets. As with any\n"
  138. "hash table, choosing a good size for the table requires some\n"
  139. "caution.\n"
  140. "\n"
  141. "You can modify weak hash tables in exactly the same way you\n"
  142. "would modify regular hash tables. (@pxref{Hash Tables})")
  143. #define FUNC_NAME s_scm_make_weak_key_alist_vector
  144. {
  145. return scm_make_weak_key_hash_table (size);
  146. }
  147. #undef FUNC_NAME
  148. SCM_DEFINE (scm_make_weak_value_alist_vector, "make-weak-value-alist-vector", 0, 1, 0,
  149. (SCM size),
  150. "Return a hash table with weak values with @var{size} buckets.\n"
  151. "(@pxref{Hash Tables})")
  152. #define FUNC_NAME s_scm_make_weak_value_alist_vector
  153. {
  154. return scm_make_weak_value_hash_table (size);
  155. }
  156. #undef FUNC_NAME
  157. SCM_DEFINE (scm_make_doubly_weak_alist_vector, "make-doubly-weak-alist-vector", 1, 0, 0,
  158. (SCM size),
  159. "Return a hash table with weak keys and values with @var{size}\n"
  160. "buckets. (@pxref{Hash Tables})")
  161. #define FUNC_NAME s_scm_make_doubly_weak_alist_vector
  162. {
  163. return scm_make_doubly_weak_hash_table (size);
  164. }
  165. #undef FUNC_NAME
  166. SCM_DEFINE (scm_weak_key_alist_vector_p, "weak-key-alist-vector?", 1, 0, 0,
  167. (SCM obj),
  168. "@deffnx {Scheme Procedure} weak-value-alist-vector? obj\n"
  169. "@deffnx {Scheme Procedure} doubly-weak-alist-vector? obj\n"
  170. "Return @code{#t} if @var{obj} is the specified weak hash\n"
  171. "table. Note that a doubly weak hash table is neither a weak key\n"
  172. "nor a weak value hash table.")
  173. #define FUNC_NAME s_scm_weak_key_alist_vector_p
  174. {
  175. return scm_from_bool (SCM_I_WVECTP (obj) && SCM_IS_WHVEC (obj));
  176. }
  177. #undef FUNC_NAME
  178. SCM_DEFINE (scm_weak_value_alist_vector_p, "weak-value-alist-vector?", 1, 0, 0,
  179. (SCM obj),
  180. "Return @code{#t} if @var{obj} is a weak value hash table.")
  181. #define FUNC_NAME s_scm_weak_value_alist_vector_p
  182. {
  183. return scm_from_bool (SCM_I_WVECTP (obj) && SCM_IS_WHVEC_V (obj));
  184. }
  185. #undef FUNC_NAME
  186. SCM_DEFINE (scm_doubly_weak_alist_vector_p, "doubly-weak-alist-vector?", 1, 0, 0,
  187. (SCM obj),
  188. "Return @code{#t} if @var{obj} is a doubly weak hash table.")
  189. #define FUNC_NAME s_scm_doubly_weak_alist_vector_p
  190. {
  191. return scm_from_bool (SCM_I_WVECTP (obj) && SCM_IS_WHVEC_B (obj));
  192. }
  193. #undef FUNC_NAME
  194. SCM
  195. scm_init_weaks_builtins ()
  196. {
  197. #include "libguile/weaks.x"
  198. return SCM_UNSPECIFIED;
  199. }
  200. void
  201. scm_weaks_prehistory ()
  202. {
  203. /* Initialize weak pairs. */
  204. GC_word wcar_pair_bitmap[GC_BITMAP_SIZE (scm_t_cell)] = { 0 };
  205. GC_word wcdr_pair_bitmap[GC_BITMAP_SIZE (scm_t_cell)] = { 0 };
  206. /* In a weak-car pair, only the second word must be scanned for
  207. pointers. */
  208. GC_set_bit (wcar_pair_bitmap, GC_WORD_OFFSET (scm_t_cell, word_1));
  209. wcar_pair_descr = GC_make_descriptor (wcar_pair_bitmap,
  210. GC_WORD_LEN (scm_t_cell));
  211. /* Conversely, in a weak-cdr pair, only the first word must be scanned for
  212. pointers. */
  213. GC_set_bit (wcdr_pair_bitmap, GC_WORD_OFFSET (scm_t_cell, word_0));
  214. wcdr_pair_descr = GC_make_descriptor (wcdr_pair_bitmap,
  215. GC_WORD_LEN (scm_t_cell));
  216. }
  217. void
  218. scm_init_weaks ()
  219. {
  220. scm_c_define_gsubr ("%init-weaks-builtins", 0, 0, 0,
  221. scm_init_weaks_builtins);
  222. }
  223. /*
  224. Local Variables:
  225. c-file-style: "gnu"
  226. End:
  227. */