guardians.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385
  1. /* Copyright (C) 1998-2001, 2006, 2008, 2009, 2011-2013, 2019
  2. * 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. /* 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. */
  42. /* Uncomment the following line to debug guardian finalization. */
  43. /* #define DEBUG_GUARDIANS 1 */
  44. #ifdef HAVE_CONFIG_H
  45. # include <config.h>
  46. #endif
  47. #include "libguile/_scm.h"
  48. #include "libguile/ports.h"
  49. #include "libguile/print.h"
  50. #include "libguile/smob.h"
  51. #include "libguile/validate.h"
  52. #include "libguile/hashtab.h"
  53. #include "libguile/deprecation.h"
  54. #include "libguile/eval.h"
  55. #include "libguile/guardians.h"
  56. #include "libguile/bdw-gc.h"
  57. static scm_t_bits tc16_guardian;
  58. typedef struct t_guardian
  59. {
  60. scm_i_pthread_mutex_t mutex;
  61. unsigned long live;
  62. SCM zombies;
  63. struct t_guardian *next;
  64. } t_guardian;
  65. #define GUARDIAN_P(x) SCM_SMOB_PREDICATE(tc16_guardian, x)
  66. #define GUARDIAN_DATA(x) ((t_guardian *) SCM_SMOB_DATA_1 (x))
  67. static int
  68. guardian_print (SCM guardian, SCM port, scm_print_state *pstate SCM_UNUSED)
  69. {
  70. t_guardian *g = GUARDIAN_DATA (guardian);
  71. scm_puts ("#<guardian ", port);
  72. scm_uintprint ((scm_t_bits) g, 16, port);
  73. scm_puts (" (reachable: ", port);
  74. scm_display (scm_from_ulong (g->live), port);
  75. scm_puts (" unreachable: ", port);
  76. scm_display (scm_length (g->zombies), port);
  77. scm_puts (")", port);
  78. scm_puts (">", port);
  79. return 1;
  80. }
  81. /* Handle finalization of OBJ which is guarded by the guardians listed in
  82. GUARDIAN_LIST. */
  83. static void
  84. finalize_guarded (void *ptr, void *finalizer_data)
  85. {
  86. SCM cell_pool;
  87. SCM obj, guardian_list, proxied_finalizer;
  88. obj = SCM_PACK_POINTER (ptr);
  89. guardian_list = SCM_CDR (SCM_PACK_POINTER (finalizer_data));
  90. proxied_finalizer = SCM_CAR (SCM_PACK_POINTER (finalizer_data));
  91. #ifdef DEBUG_GUARDIANS
  92. printf ("finalizing guarded %p (%u guardians)\n",
  93. ptr, scm_to_uint (scm_length (guardian_list)));
  94. #endif
  95. /* Preallocate a bunch of cells so that we can make sure that no garbage
  96. collection (and, thus, nested calls to `finalize_guarded ()') occurs
  97. while executing the following loop. This is quite inefficient (call to
  98. `scm_length ()') but that shouldn't be a problem in most cases. */
  99. cell_pool = scm_make_list (scm_length (guardian_list), SCM_UNSPECIFIED);
  100. /* Tell each guardian interested in OBJ that OBJ is no longer
  101. reachable. */
  102. for (;
  103. !scm_is_null (guardian_list);
  104. guardian_list = SCM_CDR (guardian_list))
  105. {
  106. SCM zombies;
  107. SCM guardian;
  108. t_guardian *g;
  109. guardian = scm_c_weak_vector_ref (scm_car (guardian_list), 0);
  110. if (scm_is_false (guardian))
  111. {
  112. /* The guardian itself vanished in the meantime. */
  113. #ifdef DEBUG_GUARDIANS
  114. printf (" guardian for %p vanished\n", ptr);
  115. #endif
  116. continue;
  117. }
  118. g = GUARDIAN_DATA (guardian);
  119. scm_i_pthread_mutex_lock (&g->mutex);
  120. if (g->live == 0)
  121. abort ();
  122. /* Get a fresh cell from CELL_POOL. */
  123. zombies = cell_pool;
  124. cell_pool = SCM_CDR (cell_pool);
  125. /* Compute and update G's zombie list. */
  126. SCM_SETCAR (zombies, obj);
  127. SCM_SETCDR (zombies, g->zombies);
  128. g->zombies = zombies;
  129. g->live--;
  130. scm_i_pthread_mutex_unlock (&g->mutex);
  131. }
  132. if (scm_is_true (proxied_finalizer))
  133. {
  134. /* Re-register the finalizer that was in place before we installed this
  135. one. */
  136. GC_finalization_proc finalizer, prev_finalizer;
  137. void *finalizer_data, *prev_finalizer_data;
  138. finalizer = (GC_finalization_proc) SCM_UNPACK_POINTER (SCM_CAR (proxied_finalizer));
  139. finalizer_data = SCM_UNPACK_POINTER (SCM_CDR (proxied_finalizer));
  140. if (finalizer == NULL)
  141. abort ();
  142. GC_REGISTER_FINALIZER_NO_ORDER (ptr, finalizer, finalizer_data,
  143. &prev_finalizer, &prev_finalizer_data);
  144. #ifdef DEBUG_GUARDIANS
  145. printf (" reinstalled proxied finalizer %p for %p\n", finalizer, ptr);
  146. #endif
  147. }
  148. #ifdef DEBUG_GUARDIANS
  149. printf ("end of finalize (%p)\n", ptr);
  150. #endif
  151. }
  152. /* Add OBJ as a guarded object of GUARDIAN. */
  153. static void
  154. scm_i_guard (SCM guardian, SCM obj)
  155. {
  156. t_guardian *g = GUARDIAN_DATA (guardian);
  157. if (SCM_HEAP_OBJECT_P (obj))
  158. {
  159. /* Register a finalizer and pass a pair as the ``client data''
  160. argument. The pair contains in its car `#f' or a pair describing a
  161. ``proxied'' finalizer (see below); its cdr contains a list of
  162. guardians interested in OBJ.
  163. A ``proxied'' finalizer is a finalizer that was registered for OBJ
  164. before OBJ became guarded (e.g., a SMOB `free' function). We are
  165. assuming here that finalizers are only used internally, either at
  166. the very beginning of an object's lifetime (e.g., see `SCM_NEWSMOB')
  167. or by this function. */
  168. GC_finalization_proc prev_finalizer;
  169. void *prev_data;
  170. SCM guardians_for_obj, finalizer_data;
  171. scm_i_pthread_mutex_lock (&g->mutex);
  172. g->live++;
  173. /* Note: GUARDIANS_FOR_OBJ holds weak references to guardians so
  174. that a guardian can be collected before the objects it guards
  175. (see `guardians.test'). */
  176. guardians_for_obj = scm_cons (scm_make_weak_vector (SCM_INUM1, guardian),
  177. SCM_EOL);
  178. finalizer_data = scm_cons (SCM_BOOL_F, guardians_for_obj);
  179. GC_REGISTER_FINALIZER_NO_ORDER (SCM_UNPACK_POINTER (obj), finalize_guarded,
  180. SCM_UNPACK_POINTER (finalizer_data),
  181. &prev_finalizer, &prev_data);
  182. if (prev_finalizer == finalize_guarded)
  183. {
  184. /* OBJ is already guarded by another guardian: add GUARDIAN to its
  185. list of guardians. */
  186. SCM prev_guardian_list, prev_finalizer_data;
  187. if (prev_data == NULL)
  188. abort ();
  189. prev_finalizer_data = SCM_PACK_POINTER (prev_data);
  190. if (!scm_is_pair (prev_finalizer_data))
  191. abort ();
  192. prev_guardian_list = SCM_CDR (prev_finalizer_data);
  193. SCM_SETCDR (guardians_for_obj, prev_guardian_list);
  194. /* Also copy information about proxied finalizers. */
  195. SCM_SETCAR (finalizer_data, SCM_CAR (prev_finalizer_data));
  196. }
  197. else if (prev_finalizer != NULL)
  198. {
  199. /* There was already a finalizer registered for OBJ so we will
  200. ``proxy'' it, i.e., record it so that we can re-register it once
  201. `finalize_guarded ()' has finished. */
  202. SCM proxied_finalizer;
  203. proxied_finalizer = scm_cons (SCM_PACK_POINTER (prev_finalizer),
  204. SCM_PACK_POINTER (prev_data));
  205. SCM_SETCAR (finalizer_data, proxied_finalizer);
  206. }
  207. scm_i_pthread_mutex_unlock (&g->mutex);
  208. }
  209. }
  210. static SCM
  211. scm_i_get_one_zombie (SCM guardian)
  212. {
  213. t_guardian *g = GUARDIAN_DATA (guardian);
  214. SCM res = SCM_BOOL_F;
  215. scm_i_pthread_mutex_lock (&g->mutex);
  216. if (!scm_is_null (g->zombies))
  217. {
  218. /* Note: We return zombies in reverse order. */
  219. res = SCM_CAR (g->zombies);
  220. g->zombies = SCM_CDR (g->zombies);
  221. }
  222. scm_i_pthread_mutex_unlock (&g->mutex);
  223. return res;
  224. }
  225. /* This is the Scheme entry point for each guardian: If OBJ is an
  226. * object, it's added to the guardian's live list. If OBJ is unbound,
  227. * the next available unreachable object (or #f if none) is returned.
  228. *
  229. * If the second optional argument THROW_P is true (the default), then
  230. * an error is raised if GUARDIAN is greedy and OBJ is already greedily
  231. * guarded. If THROW_P is false, #f is returned instead of raising the
  232. * error, and #t is returned if everything is fine.
  233. */
  234. static SCM
  235. guardian_apply (SCM guardian, SCM obj, SCM throw_p)
  236. {
  237. if (!SCM_UNBNDP (obj))
  238. {
  239. scm_i_guard (guardian, obj);
  240. return SCM_UNSPECIFIED;
  241. }
  242. else
  243. return scm_i_get_one_zombie (guardian);
  244. }
  245. SCM_DEFINE (scm_make_guardian, "make-guardian", 0, 0, 0,
  246. (),
  247. "Create a new guardian. A guardian protects a set of objects from\n"
  248. "garbage collection, allowing a program to apply cleanup or other\n"
  249. "actions.\n"
  250. "\n"
  251. "@code{make-guardian} returns a procedure representing the guardian.\n"
  252. "Calling the guardian procedure with an argument adds the argument to\n"
  253. "the guardian's set of protected objects. Calling the guardian\n"
  254. "procedure without an argument returns one of the protected objects\n"
  255. "which are ready for garbage collection, or @code{#f} if no such object\n"
  256. "is available. Objects which are returned in this way are removed from\n"
  257. "the guardian.\n"
  258. "\n"
  259. "You can put a single object into a guardian more than once and you can\n"
  260. "put a single object into more than one guardian. The object will then\n"
  261. "be returned multiple times by the guardian procedures.\n"
  262. "\n"
  263. "An object is eligible to be returned from a guardian when it is no\n"
  264. "longer referenced from outside any guardian.\n"
  265. "\n"
  266. "There is no guarantee about the order in which objects are returned\n"
  267. "from a guardian. If you want to impose an order on finalization\n"
  268. "actions, for example, you can do that by keeping objects alive in some\n"
  269. "global data structure until they are no longer needed for finalizing\n"
  270. "other objects.\n"
  271. "\n"
  272. "Being an element in a weak vector, a key in a hash table with weak\n"
  273. "keys, or a value in a hash table with weak value does not prevent an\n"
  274. "object from being returned by a guardian. But as long as an object\n"
  275. "can be returned from a guardian it will not be removed from such a\n"
  276. "weak vector or hash table. In other words, a weak link does not\n"
  277. "prevent an object from being considered collectable, but being inside\n"
  278. "a guardian prevents a weak link from being broken.\n"
  279. "\n"
  280. "A key in a weak key hash table can be though of as having a strong\n"
  281. "reference to its associated value as long as the key is accessible.\n"
  282. "Consequently, when the key only accessible from within a guardian, the\n"
  283. "reference from the key to the value is also considered to be coming\n"
  284. "from within a guardian. Thus, if there is no other reference to the\n"
  285. "value, it is eligible to be returned from a guardian.\n")
  286. #define FUNC_NAME s_scm_make_guardian
  287. {
  288. t_guardian *g = scm_gc_malloc (sizeof (t_guardian), "guardian");
  289. SCM z;
  290. scm_i_pthread_mutex_init (&g->mutex, NULL);
  291. /* A tconc starts out with one tail pair. */
  292. g->live = 0;
  293. g->zombies = SCM_EOL;
  294. g->next = NULL;
  295. SCM_NEWSMOB (z, tc16_guardian, g);
  296. return z;
  297. }
  298. #undef FUNC_NAME
  299. void
  300. scm_init_guardians ()
  301. {
  302. /* We use unordered finalization `a la Java. */
  303. GC_set_java_finalization (1);
  304. tc16_guardian = scm_make_smob_type ("guardian", 0);
  305. scm_set_smob_print (tc16_guardian, guardian_print);
  306. scm_set_smob_apply (tc16_guardian, guardian_apply, 0, 1, 0);
  307. #include "libguile/guardians.x"
  308. }
  309. /*
  310. Local Variables:
  311. c-file-style: "gnu"
  312. End:
  313. */