1
Fork 0
mirror of https://git.savannah.gnu.org/git/guile.git synced 2025-06-06 04:00:26 +02:00

Reimplement guardians in Scheme

Also, shunt them off to a module, and deprecate the C interface.

* module/ice-9/guardians.scm: New file.
* am/bootstrap.am (SOURCES): Add new file.

* libguile.h: Remove guardians.h include.
* libguile/Makefile.am (libguile_@GUILE_EFFECTIVE_VERSION@_la_SOURCES):
(DOT_X_FILES):
(DOT_DOC_FILES):
(modinclude_HEADERS): Remove guardians.[ch] files.
* libguile/init.c (scm_i_init_guile): Remove guardians.

* libguile/deprecated.h:
* libguile/deprecated.c (scm_make_guardian): Deprecate.
* module/ice-9/deprecated.scm: Add make-guardian shim.

* module/oop/goops.scm (<guardian>): Remove class; they no longer have a
tc16.

* module/ice-9/popen.scm: Add guardians module.
* test-suite/tests/eval.test:
* test-suite/tests/fluids.test:
* test-suite/tests/gc.test:
* test-suite/tests/guardians.test:
* test-suite/tests/threads.test:
* test-suite/tests/types.test: Use the guardians module.
This commit is contained in:
Andy Wingo 2025-05-04 20:41:16 +02:00
parent b1d7d3538a
commit 66f9815c3f
19 changed files with 156 additions and 448 deletions

View file

@ -147,6 +147,7 @@ SOURCES = \
ice-9/futures.scm \ ice-9/futures.scm \
ice-9/gap-buffer.scm \ ice-9/gap-buffer.scm \
ice-9/getopt-long.scm \ ice-9/getopt-long.scm \
ice-9/guardians.scm \
ice-9/hash-table.scm \ ice-9/hash-table.scm \
ice-9/hcons.scm \ ice-9/hcons.scm \
ice-9/history.scm \ ice-9/history.scm \

View file

@ -1,7 +1,7 @@
#ifndef SCM_LIBGUILE_H #ifndef SCM_LIBGUILE_H
#define SCM_LIBGUILE_H #define SCM_LIBGUILE_H
/* Copyright 1995-1998,2000-2004,2006,2008-2014,2018,2020-2021 /* Copyright 1995-1998,2000-2004,2006,2008-2014,2018,2020-2021,2025
Free Software Foundation, Inc. Free Software Foundation, Inc.
This file is part of Guile. This file is part of Guile.
@ -61,7 +61,6 @@ extern "C" {
#include "libguile/generalized-vectors.h" #include "libguile/generalized-vectors.h"
#include "libguile/goops.h" #include "libguile/goops.h"
#include "libguile/gsubr.h" #include "libguile/gsubr.h"
#include "libguile/guardians.h"
#include "libguile/hash.h" #include "libguile/hash.h"
#include "libguile/hashtab.h" #include "libguile/hashtab.h"
#include "libguile/hooks.h" #include "libguile/hooks.h"

View file

@ -173,7 +173,6 @@ libguile_@GUILE_EFFECTIVE_VERSION@_la_SOURCES = \
generalized-vectors.c \ generalized-vectors.c \
goops.c \ goops.c \
gsubr.c \ gsubr.c \
guardians.c \
hash.c \ hash.c \
hashtab.c \ hashtab.c \
hooks.c \ hooks.c \
@ -292,7 +291,6 @@ DOT_X_FILES = \
generalized-vectors.x \ generalized-vectors.x \
goops.x \ goops.x \
gsubr.x \ gsubr.x \
guardians.x \
hash.x \ hash.x \
hashtab.x \ hashtab.x \
hooks.x \ hooks.x \
@ -397,7 +395,6 @@ DOT_DOC_FILES = \
generalized-vectors.doc \ generalized-vectors.doc \
goops.doc \ goops.doc \
gsubr.doc \ gsubr.doc \
guardians.doc \
hash.doc \ hash.doc \
hashtab.doc \ hashtab.doc \
hooks.doc \ hooks.doc \
@ -644,7 +641,6 @@ modinclude_HEADERS = \
generalized-vectors.h \ generalized-vectors.h \
goops.h \ goops.h \
gsubr.h \ gsubr.h \
guardians.h \
hash.h \ hash.h \
hashtab.h \ hashtab.h \
hooks.h \ hooks.h \

View file

@ -23,6 +23,13 @@
#define SCM_BUILDING_DEPRECATED_CODE #define SCM_BUILDING_DEPRECATED_CODE
#include "deprecation.h"
#include "eval.h"
#include "gsubr.h"
#include "modules.h"
#include "threads.h"
#include "variable.h"
#include "deprecated.h" #include "deprecated.h"
#if (SCM_ENABLE_DEPRECATED == 1) #if (SCM_ENABLE_DEPRECATED == 1)
@ -30,6 +37,26 @@
/* Deprecated functions go here. */ /* Deprecated functions go here. */
static SCM make_guardian_var;
static void
init_make_guardian_var (void)
{
make_guardian_var = scm_c_public_lookup ("ice-9 guardians", "make-guardian");
}
SCM
scm_make_guardian (void)
{
static scm_i_pthread_once_t once = SCM_I_PTHREAD_ONCE_INIT;
scm_i_pthread_once (&once, init_make_guardian_var);
scm_c_issue_deprecation_warning
("The scm_make_guardian C interface is deprecated. Invoke the Scheme "
"make-guardian procedure from (ice-9 guardians) instead.");
return scm_call_0 (scm_variable_ref (make_guardian_var));
}

View file

@ -24,6 +24,7 @@
#if (SCM_ENABLE_DEPRECATED == 1) #if (SCM_ENABLE_DEPRECATED == 1)
SCM_DEPRECATED SCM scm_make_guardian (void);
/* Deprecated declarations go here. */ /* Deprecated declarations go here. */
void scm_i_init_deprecated (void); void scm_i_init_deprecated (void);

View file

@ -39,7 +39,6 @@
#include "foreign.h" #include "foreign.h"
#include "gc-internal.h" #include "gc-internal.h"
#include "gsubr.h" #include "gsubr.h"
#include "guardians.h"
#include "init.h" #include "init.h"
#include "numbers.h" #include "numbers.h"
#include "ports.h" #include "ports.h"

View file

@ -1,381 +0,0 @@
/* Copyright 1998-2001,2006,2008-2009,2011-2013,2018-2019
Free Software Foundation, Inc.
This file is part of Guile.
Guile is free software: you can redistribute it and/or modify it
under the terms of the GNU Lesser General Public License as published
by the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
Guile is distributed in the hope that it will be useful, but WITHOUT
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public
License for more details.
You should have received a copy of the GNU Lesser General Public
License along with Guile. If not, see
<https://www.gnu.org/licenses/>. */
/* This is an implementation of guardians as described in
* R. Kent Dybvig, Carl Bruggeman, and David Eby (1993) "Guardians in
* a Generation-Based Garbage Collector" ACM SIGPLAN Conference on
* Programming Language Design and Implementation, June 1993
* ftp://ftp.cs.indiana.edu/pub/scheme-repository/doc/pubs/guardians.ps.gz
*
* Original design: Mikael Djurfeldt
* Original implementation: Michael Livshin
* Hacked on since by: everybody
*
* By this point, the semantics are actually quite different from
* those described in the abovementioned paper. The semantic changes
* are there to improve safety and intuitiveness. The interface is
* still (mostly) the one described by the paper, however.
*
* Boiled down again: Marius Vollmer
*
* Now they should again behave like those described in the paper.
* Scheme guardians should be simple and friendly, not like the greedy
* monsters we had...
*
* Rewritten for the Boehm-Demers-Weiser GC by Ludovic Courtès.
*/
/* Uncomment the following line to debug guardian finalization. */
/* #define DEBUG_GUARDIANS 1 */
#ifdef HAVE_CONFIG_H
# include <config.h>
#endif
#include "bdw-gc.h"
#include "boolean.h"
#include "deprecation.h"
#include "eval.h"
#include "gsubr.h"
#include "hashtab.h"
#include "list.h"
#include "numbers.h"
#include "pairs.h"
#include "ports.h"
#include "print.h"
#include "smob.h"
#include "threads.h"
#include "weak-vector.h"
#include "guardians.h"
static scm_t_bits tc16_guardian;
typedef struct t_guardian
{
scm_i_pthread_mutex_t mutex;
unsigned long live;
SCM zombies;
struct t_guardian *next;
} t_guardian;
#define GUARDIAN_P(x) SCM_SMOB_PREDICATE(tc16_guardian, x)
#define GUARDIAN_DATA(x) ((t_guardian *) SCM_SMOB_DATA_1 (x))
static int
guardian_print (SCM guardian, SCM port, scm_print_state *pstate SCM_UNUSED)
{
t_guardian *g = GUARDIAN_DATA (guardian);
scm_puts ("#<guardian ", port);
scm_uintprint ((scm_t_bits) g, 16, port);
scm_puts (" (reachable: ", port);
scm_display (scm_from_ulong (g->live), port);
scm_puts (" unreachable: ", port);
scm_display (scm_length (g->zombies), port);
scm_puts (")", port);
scm_puts (">", port);
return 1;
}
/* Handle finalization of OBJ which is guarded by the guardians listed in
GUARDIAN_LIST. */
static void
finalize_guarded (void *ptr, void *finalizer_data)
{
SCM cell_pool;
SCM obj, guardian_list, proxied_finalizer;
obj = SCM_PACK_POINTER (ptr);
guardian_list = SCM_CDR (SCM_PACK_POINTER (finalizer_data));
proxied_finalizer = SCM_CAR (SCM_PACK_POINTER (finalizer_data));
#ifdef DEBUG_GUARDIANS
printf ("finalizing guarded %p (%u guardians)\n",
ptr, scm_to_uint (scm_length (guardian_list)));
#endif
/* Preallocate a bunch of cells so that we can make sure that no garbage
collection (and, thus, nested calls to `finalize_guarded ()') occurs
while executing the following loop. This is quite inefficient (call to
`scm_length ()') but that shouldn't be a problem in most cases. */
cell_pool = scm_make_list (scm_length (guardian_list), SCM_UNSPECIFIED);
/* Tell each guardian interested in OBJ that OBJ is no longer
reachable. */
for (;
!scm_is_null (guardian_list);
guardian_list = SCM_CDR (guardian_list))
{
SCM zombies;
SCM guardian;
t_guardian *g;
guardian = scm_c_weak_vector_ref (scm_car (guardian_list), 0);
if (scm_is_false (guardian))
{
/* The guardian itself vanished in the meantime. */
#ifdef DEBUG_GUARDIANS
printf (" guardian for %p vanished\n", ptr);
#endif
continue;
}
g = GUARDIAN_DATA (guardian);
scm_i_pthread_mutex_lock (&g->mutex);
if (g->live == 0)
abort ();
/* Get a fresh cell from CELL_POOL. */
zombies = cell_pool;
cell_pool = SCM_CDR (cell_pool);
/* Compute and update G's zombie list. */
SCM_SETCAR (zombies, obj);
SCM_SETCDR (zombies, g->zombies);
g->zombies = zombies;
g->live--;
scm_i_pthread_mutex_unlock (&g->mutex);
}
if (scm_is_true (proxied_finalizer))
{
/* Re-register the finalizer that was in place before we installed this
one. */
GC_finalization_proc finalizer, prev_finalizer;
void *finalizer_data, *prev_finalizer_data;
finalizer = (GC_finalization_proc) SCM_UNPACK_POINTER (SCM_CAR (proxied_finalizer));
finalizer_data = SCM_UNPACK_POINTER (SCM_CDR (proxied_finalizer));
if (finalizer == NULL)
abort ();
GC_REGISTER_FINALIZER_NO_ORDER (ptr, finalizer, finalizer_data,
&prev_finalizer, &prev_finalizer_data);
#ifdef DEBUG_GUARDIANS
printf (" reinstalled proxied finalizer %p for %p\n", finalizer, ptr);
#endif
}
#ifdef DEBUG_GUARDIANS
printf ("end of finalize (%p)\n", ptr);
#endif
}
/* Add OBJ as a guarded object of GUARDIAN. */
static void
scm_i_guard (SCM guardian, SCM obj)
{
t_guardian *g = GUARDIAN_DATA (guardian);
if (SCM_HEAP_OBJECT_P (obj))
{
/* Register a finalizer and pass a pair as the ``client data''
argument. The pair contains in its car `#f' or a pair describing a
``proxied'' finalizer (see below); its cdr contains a list of
guardians interested in OBJ.
A ``proxied'' finalizer is a finalizer that was registered for OBJ
before OBJ became guarded (e.g., a SMOB `free' function). We are
assuming here that finalizers are only used internally, either at
the very beginning of an object's lifetime (e.g., see `SCM_NEWSMOB')
or by this function. */
GC_finalization_proc prev_finalizer;
void *prev_data;
SCM guardians_for_obj, finalizer_data;
scm_i_pthread_mutex_lock (&g->mutex);
g->live++;
/* Note: GUARDIANS_FOR_OBJ holds weak references to guardians so
that a guardian can be collected before the objects it guards
(see `guardians.test'). */
guardians_for_obj = scm_cons (scm_make_weak_vector (SCM_INUM1, guardian),
SCM_EOL);
finalizer_data = scm_cons (SCM_BOOL_F, guardians_for_obj);
GC_REGISTER_FINALIZER_NO_ORDER (SCM_UNPACK_POINTER (obj), finalize_guarded,
SCM_UNPACK_POINTER (finalizer_data),
&prev_finalizer, &prev_data);
if (prev_finalizer == finalize_guarded)
{
/* OBJ is already guarded by another guardian: add GUARDIAN to its
list of guardians. */
SCM prev_guardian_list, prev_finalizer_data;
if (prev_data == NULL)
abort ();
prev_finalizer_data = SCM_PACK_POINTER (prev_data);
if (!scm_is_pair (prev_finalizer_data))
abort ();
prev_guardian_list = SCM_CDR (prev_finalizer_data);
SCM_SETCDR (guardians_for_obj, prev_guardian_list);
/* Also copy information about proxied finalizers. */
SCM_SETCAR (finalizer_data, SCM_CAR (prev_finalizer_data));
}
else if (prev_finalizer != NULL)
{
/* There was already a finalizer registered for OBJ so we will
``proxy'' it, i.e., record it so that we can re-register it once
`finalize_guarded ()' has finished. */
SCM proxied_finalizer;
proxied_finalizer = scm_cons (SCM_PACK_POINTER (prev_finalizer),
SCM_PACK_POINTER (prev_data));
SCM_SETCAR (finalizer_data, proxied_finalizer);
}
scm_i_pthread_mutex_unlock (&g->mutex);
}
}
static SCM
scm_i_get_one_zombie (SCM guardian)
{
t_guardian *g = GUARDIAN_DATA (guardian);
SCM res = SCM_BOOL_F;
scm_i_pthread_mutex_lock (&g->mutex);
if (!scm_is_null (g->zombies))
{
/* Note: We return zombies in reverse order. */
res = SCM_CAR (g->zombies);
g->zombies = SCM_CDR (g->zombies);
}
scm_i_pthread_mutex_unlock (&g->mutex);
return res;
}
/* This is the Scheme entry point for each guardian: If OBJ is an
* object, it's added to the guardian's live list. If OBJ is unbound,
* the next available unreachable object (or #f if none) is returned.
*
* If the second optional argument THROW_P is true (the default), then
* an error is raised if GUARDIAN is greedy and OBJ is already greedily
* guarded. If THROW_P is false, #f is returned instead of raising the
* error, and #t is returned if everything is fine.
*/
static SCM
guardian_apply (SCM guardian, SCM obj, SCM throw_p)
{
if (!SCM_UNBNDP (obj))
{
scm_i_guard (guardian, obj);
return SCM_UNSPECIFIED;
}
else
return scm_i_get_one_zombie (guardian);
}
SCM_DEFINE (scm_make_guardian, "make-guardian", 0, 0, 0,
(),
"Create a new guardian. A guardian protects a set of objects from\n"
"garbage collection, allowing a program to apply cleanup or other\n"
"actions.\n"
"\n"
"@code{make-guardian} returns a procedure representing the guardian.\n"
"Calling the guardian procedure with an argument adds the argument to\n"
"the guardian's set of protected objects. Calling the guardian\n"
"procedure without an argument returns one of the protected objects\n"
"which are ready for garbage collection, or @code{#f} if no such object\n"
"is available. Objects which are returned in this way are removed from\n"
"the guardian.\n"
"\n"
"You can put a single object into a guardian more than once and you can\n"
"put a single object into more than one guardian. The object will then\n"
"be returned multiple times by the guardian procedures.\n"
"\n"
"An object is eligible to be returned from a guardian when it is no\n"
"longer referenced from outside any guardian.\n"
"\n"
"There is no guarantee about the order in which objects are returned\n"
"from a guardian. If you want to impose an order on finalization\n"
"actions, for example, you can do that by keeping objects alive in some\n"
"global data structure until they are no longer needed for finalizing\n"
"other objects.\n"
"\n"
"Being an element in a weak vector, a key in a hash table with weak\n"
"keys, or a value in a hash table with weak value does not prevent an\n"
"object from being returned by a guardian. But as long as an object\n"
"can be returned from a guardian it will not be removed from such a\n"
"weak vector or hash table. In other words, a weak link does not\n"
"prevent an object from being considered collectable, but being inside\n"
"a guardian prevents a weak link from being broken.\n"
"\n"
"A key in a weak key hash table can be though of as having a strong\n"
"reference to its associated value as long as the key is accessible.\n"
"Consequently, when the key only accessible from within a guardian, the\n"
"reference from the key to the value is also considered to be coming\n"
"from within a guardian. Thus, if there is no other reference to the\n"
"value, it is eligible to be returned from a guardian.\n")
#define FUNC_NAME s_scm_make_guardian
{
t_guardian *g = scm_gc_malloc (sizeof (t_guardian), "guardian");
SCM z;
scm_i_pthread_mutex_init (&g->mutex, NULL);
/* A tconc starts out with one tail pair. */
g->live = 0;
g->zombies = SCM_EOL;
g->next = NULL;
SCM_NEWSMOB (z, tc16_guardian, g);
return z;
}
#undef FUNC_NAME
void
scm_init_guardians ()
{
/* We use unordered finalization `a la Java. */
GC_set_java_finalization (1);
tc16_guardian = scm_make_smob_type ("guardian", 0);
scm_set_smob_print (tc16_guardian, guardian_print);
scm_set_smob_apply (tc16_guardian, guardian_apply, 0, 1, 0);
#include "guardians.x"
}

View file

@ -1,35 +0,0 @@
#ifndef SCM_GUARDIANS_H
#define SCM_GUARDIANS_H
/* Copyright 1998,2000-2001,2006,2008,2018
Free Software Foundation, Inc.
This file is part of Guile.
Guile is free software: you can redistribute it and/or modify it
under the terms of the GNU Lesser General Public License as published
by the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
Guile is distributed in the hope that it will be useful, but WITHOUT
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public
License for more details.
You should have received a copy of the GNU Lesser General Public
License along with Guile. If not, see
<https://www.gnu.org/licenses/>. */
#include "libguile/scm.h"
SCM_API SCM scm_make_guardian (void);
SCM_INTERNAL void scm_i_init_guardians_for_gc (void);
SCM_INTERNAL void scm_i_identify_inaccessible_guardeds (void);
SCM_INTERNAL int scm_i_mark_inaccessible_guardeds (void);
SCM_INTERNAL void scm_init_guardians (void);
#endif /* SCM_GUARDIANS_H */

View file

@ -80,7 +80,6 @@
#include "gettext.h" #include "gettext.h"
#include "goops.h" #include "goops.h"
#include "gsubr.h" #include "gsubr.h"
#include "guardians.h"
#include "hash.h" #include "hash.h"
#include "hashtab.h" #include "hashtab.h"
#include "hooks.h" #include "hooks.h"
@ -460,7 +459,6 @@ scm_i_init_guile (struct gc_stack_addr base)
scm_init_weak_set (); scm_init_weak_set ();
scm_init_weak_table (); scm_init_weak_table ();
scm_init_weak_vectors (); scm_init_weak_vectors ();
scm_init_guardians (); /* requires smob_prehistory */
scm_init_standard_ports (); /* Requires fports */ scm_init_standard_ports (); /* Requires fports */
scm_init_expand (); /* Requires structs */ scm_init_expand (); /* Requires structs */
scm_init_memoize (); /* Requires smob_prehistory */ scm_init_memoize (); /* Requires smob_prehistory */

View file

@ -1,4 +1,4 @@
;;;; Copyright (C) 2017, 2020 Free Software Foundation, Inc. ;;;; Copyright (C) 2025 Free Software Foundation, Inc.
;;;; ;;;;
;;;; This library is free software; you can redistribute it and/or ;;;; This library is free software; you can redistribute it and/or
;;;; modify it under the terms of the GNU Lesser General Public ;;;; modify it under the terms of the GNU Lesser General Public
@ -16,9 +16,10 @@
;;;; ;;;;
(define-module (ice-9 deprecated) (define-module (ice-9 deprecated)
#:use-module (ice-9 copy-tree) #:use-module (ice-9 guardians)
#:export ((copy-tree* . copy-tree))) #:export ((make-guardian* . make-guardian)))
#;
(define-syntax-rule (define-deprecated name message exp) (define-syntax-rule (define-deprecated name message exp)
(begin (begin
(define-syntax rule (define-syntax rule
@ -28,14 +29,8 @@
exp))) exp)))
(export rule))) (export rule)))
(define %allow-legacy-syntax-objects? (make-parameter #f)) (define (make-guardian*)
(define-deprecated allow-legacy-syntax-objects?
"allow-legacy-syntax-objects? is deprecated and has no effect. Guile
3.0 has no legacy syntax objects."
%allow-legacy-syntax-objects?)
(define (copy-tree* x)
(issue-deprecation-warning (issue-deprecation-warning
"copy-tree in the default environment is deprecated. Import it "make-guardian in the default environment is deprecated. Import it
from (ice-9 copy-tree) instead.") from (ice-9 guardians) instead.")
(copy-tree x)) (make-guardian))

103
module/ice-9/guardians.scm Normal file
View file

@ -0,0 +1,103 @@
;;; Copyright (C) 2025 Free Software Foundation, Inc.
;;;
;;; This library is free software: you can redistribute it and/or modify
;;; it under the terms of the GNU Lesser General Public License as
;;; published by the Free Software Foundation, either version 3 of the
;;; License, or (at your option) any later version.
;;;
;;; This library is distributed in the hope that it will be useful, but
;;; WITHOUT ANY WARRANTY; without even the implied warranty of
;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
;;; Lesser General Public License for more details.
;;;
;;; You should have received a copy of the GNU Lesser General Public
;;; License along with this program. If not, see
;;; <http://www.gnu.org/licenses/>.
;;; Commentary:
;;;
;;; This is an implementation of guardians as described in:
;;;
;;; R. Kent Dybvig, Carl Bruggeman, and David Eby. "Guardians in a
;;; Generation-Based Garbage Collector." PLDI 1993.
;;; https://dl.acm.org/doi/abs/10.1145/173262.155110
;;;
;;; Our implementation is terms of Whippet's multi-priority finalizers:
;;; https://wingolog.org/archives/2024/07/22/finalizers-guardians-phantom-references-et-cetera
;;;
;;; Specifically, all guardian finalizers will run before any "normal"
;;; finalizer runs, so guarded objects that are returned to Scheme
;;; aren't finalized yet.
;;;
;;; Code:
(define-module (ice-9 guardians)
#:use-module (srfi srfi-9)
#:use-module (system finalizers)
#:use-module (ice-9 atomic)
#:use-module (ice-9 match)
#:replace (make-guardian))
(define (immediate? x)
(cond
((exact-integer? x) (<= most-negative-fixnum x most-positive-fixnum))
((char? x) #t)
((eq? x #f) #t)
((eq? x #nil) #t)
((eq? x '()) #t)
((eq? x #t) #t)
((unspecified? x) #t)
((eof-object? x) #t)
(else #f)))
(define (heap-object? x)
(not (immediate? x)))
(define (make-atomic-fifo)
(define inbox (make-atomic-box '()))
(define outbox (make-atomic-box '()))
(define (push! x)
(let lp ((in (atomic-box-ref inbox)))
(let ((prev (atomic-box-compare-and-swap! inbox in (cons x in))))
(if (eq? prev in)
(values)
(lp prev)))))
(define (transfer! in out)
(match in
(() (values))
((x . in*)
(let* ((out* (cons x out))
(out** (atomic-box-compare-and-swap! outbox out out*)))
(if (eq? out out**)
(transfer! in* out*)
(transfer! in out**))))))
(define (pop!)
(let lp ((out (atomic-box-ref outbox)))
(match out
((head . tail)
(let ((prev (atomic-box-compare-and-swap! outbox out tail)))
(if (eq? prev out)
head
(lp prev))))
(()
(match (atomic-box-swap! inbox '())
(() #f)
(in
(transfer! in '())
(pop!)))))))
(values push! pop!))
(define (make-guardian)
(define-values (push! pop!) (make-atomic-fifo))
(define (guard! obj)
(when (heap-object? obj)
(add-finalizer! obj push!)))
(define guardian
(case-lambda
(()
(pop!))
((obj)
(guard! obj)
(values))))
guardian)

View file

@ -1,6 +1,6 @@
;; popen emulation, for non-stdio based ports. ;; popen emulation, for non-stdio based ports.
;;;; Copyright (C) 1998-2001, 2003, 2006, 2010-2013, 2019 ;;;; Copyright (C) 1998-2001, 2003, 2006, 2010-2013, 2019, 2025
;;;; Free Software Foundation, Inc. ;;;; Free Software Foundation, Inc.
;;;; ;;;;
;;;; This library is free software; you can redistribute it and/or ;;;; This library is free software; you can redistribute it and/or
@ -20,6 +20,7 @@
(define-module (ice-9 popen) (define-module (ice-9 popen)
#:use-module (ice-9 binary-ports) #:use-module (ice-9 binary-ports)
#:use-module (ice-9 guardians)
#:use-module (ice-9 threads) #:use-module (ice-9 threads)
#:use-module (srfi srfi-1) #:use-module (srfi srfi-1)
#:use-module (srfi srfi-9) #:use-module (srfi srfi-9)

View file

@ -85,7 +85,7 @@
<promise> <mutex> <condition-variable> <promise> <mutex> <condition-variable>
<regexp> <hook> <random-state> <regexp> <hook> <random-state>
<directory> <array> <character-set> <directory> <array> <character-set>
<dynamic-object> <guardian> <macro> <dynamic-object> <macro>
;; Modules. ;; Modules.
<module> <module>
@ -3537,7 +3537,6 @@ var{initargs}."
(define <directory> (find-subclass <top> '<directory>)) (define <directory> (find-subclass <top> '<directory>))
(define <array> (find-subclass <top> '<array>)) (define <array> (find-subclass <top> '<array>))
(define <character-set> (find-subclass <top> '<character-set>)) (define <character-set> (find-subclass <top> '<character-set>))
(define <guardian> (find-subclass <applicable> '<guardian>))
(define <macro> (find-subclass <top> '<macro>)) (define <macro> (find-subclass <top> '<macro>))
;; <dynamic-object> used to be a SMOB type, albeit not exported even to ;; <dynamic-object> used to be a SMOB type, albeit not exported even to

View file

@ -1,5 +1,5 @@
;;;; eval.test --- tests guile's evaluator -*- scheme -*- ;;;; eval.test --- tests guile's evaluator -*- scheme -*-
;;;; Copyright (C) 2000-2001,2003-2015,2017,2019,2020 ;;;; Copyright (C) 2000-2001,2003-2015,2017,2019,2020,2025
;;;; Free Software Foundation, Inc. ;;;; Free Software Foundation, Inc.
;;;; ;;;;
;;;; This library is free software; you can redistribute it and/or ;;;; This library is free software; you can redistribute it and/or
@ -23,6 +23,7 @@
:use-module ((system vm frame) :select (frame-call-representation)) :use-module ((system vm frame) :select (frame-call-representation))
:use-module (ice-9 documentation) :use-module (ice-9 documentation)
:use-module (ice-9 exceptions) :use-module (ice-9 exceptions)
:use-module (ice-9 guardians)
:use-module (ice-9 local-eval)) :use-module (ice-9 local-eval))

View file

@ -1,7 +1,7 @@
;;;; -*- scheme -*- ;;;; -*- scheme -*-
;;;; fluids.test --- test suite for fluid values ;;;; fluids.test --- test suite for fluid values
;;;; ;;;;
;;;; Copyright (C) 2010 Free Software Foundation, Inc. ;;;; Copyright (C) 2010, 2025 Free Software Foundation, Inc.
;;;; ;;;;
;;;; This library is free software; you can redistribute it and/or ;;;; This library is free software; you can redistribute it and/or
;;;; modify it under the terms of the GNU Lesser General Public ;;;; modify it under the terms of the GNU Lesser General Public
@ -18,6 +18,7 @@
;;;; Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA ;;;; Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
(define-module (test-suite test-fluids) (define-module (test-suite test-fluids)
#:use-module (ice-9 guardians)
#:use-module (ice-9 threads) #:use-module (ice-9 threads)
#:use-module (test-suite lib) #:use-module (test-suite lib)
#:use-module (system base compile)) #:use-module (system base compile))

View file

@ -1,6 +1,6 @@
;;;; gc.test --- test guile's garbage collection -*- scheme -*- ;;;; gc.test --- test guile's garbage collection -*- scheme -*-
;;;; Copyright (C) 2000, 2001, 2004, 2006, 2007, 2008, 2009, ;;;; Copyright (C) 2000, 2001, 2004, 2006, 2007, 2008, 2009,
;;;; 2011, 2012, 2013 Free Software Foundation, Inc. ;;;; 2011, 2012, 2013, 2025 Free Software Foundation, Inc.
;;;; ;;;;
;;;; This library is free software; you can redistribute it and/or ;;;; This library is free software; you can redistribute it and/or
;;;; modify it under the terms of the GNU Lesser General Public ;;;; modify it under the terms of the GNU Lesser General Public
@ -18,6 +18,7 @@
(define-module (tests gc) (define-module (tests gc)
#:use-module (ice-9 documentation) #:use-module (ice-9 documentation)
#:use-module (ice-9 guardians)
#:use-module (test-suite lib) #:use-module (test-suite lib)
#:use-module ((system base compile) #:select (compile))) #:use-module ((system base compile) #:select (compile)))
@ -93,6 +94,7 @@
(pass-if "Lexical vars are collectable" (pass-if "Lexical vars are collectable"
(let ((l (compile (let ((l (compile
'(begin '(begin
(use-modules (ice-9 guardians))
(define guardian (make-guardian)) (define guardian (make-guardian))
(let ((f (list 'foo))) (let ((f (list 'foo)))
(guardian f)) (guardian f))

View file

@ -1,7 +1,7 @@
;;;; guardians.test --- test suite for Guile Guardians -*- scheme -*- ;;;; guardians.test --- test suite for Guile Guardians -*- scheme -*-
;;;; Jim Blandy <jimb@red-bean.com> --- July 1999 ;;;; Jim Blandy <jimb@red-bean.com> --- July 1999
;;;; ;;;;
;;;; Copyright (C) 1999, 2001, 2006, 2014 Free Software Foundation, Inc. ;;;; Copyright (C) 1999, 2001, 2006, 2014, 2025 Free Software Foundation, Inc.
;;;; ;;;;
;;;; This library is free software; you can redistribute it and/or ;;;; This library is free software; you can redistribute it and/or
;;;; modify it under the terms of the GNU Lesser General Public ;;;; modify it under the terms of the GNU Lesser General Public
@ -37,6 +37,7 @@
(define-module (test-guardians) (define-module (test-guardians)
:use-module (test-suite lib) :use-module (test-suite lib)
:use-module (ice-9 documentation) :use-module (ice-9 documentation)
:use-module (ice-9 guardians)
:use-module (ice-9 weak-vector)) :use-module (ice-9 weak-vector))

View file

@ -1,7 +1,7 @@
;;;; threads.test --- Tests for Guile threading. -*- scheme -*- ;;;; threads.test --- Tests for Guile threading. -*- scheme -*-
;;;; ;;;;
;;;; Copyright 2003, 2006, 2007, 2009, 2010, 2011, 2012, 2013, ;;;; Copyright 2003, 2006, 2007, 2009, 2010, 2011, 2012, 2013,
;;;; 2014 Free Software Foundation, Inc. ;;;; 2014, 2025 Free Software Foundation, Inc.
;;;; ;;;;
;;;; This library is free software; you can redistribute it and/or ;;;; This library is free software; you can redistribute it and/or
;;;; modify it under the terms of the GNU Lesser General Public ;;;; modify it under the terms of the GNU Lesser General Public
@ -18,6 +18,7 @@
;;;; Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA ;;;; Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
(define-module (test-threads) (define-module (test-threads)
#:use-module (ice-9 guardians)
#:use-module (ice-9 threads) #:use-module (ice-9 threads)
#:use-module (system base compile) #:use-module (system base compile)
#:use-module (test-suite lib)) #:use-module (test-suite lib))

View file

@ -1,6 +1,6 @@
;;;; types.test --- Type tag decoding. -*- mode: scheme; coding: utf-8; -*- ;;;; types.test --- Type tag decoding. -*- mode: scheme; coding: utf-8; -*-
;;;; ;;;;
;;;; Copyright (C) 2014, 2015, 2018 Free Software Foundation, Inc. ;;;; Copyright (C) 2014, 2015, 2018, 2025 Free Software Foundation, Inc.
;;;; ;;;;
;;;; This file is part of GNU Guile. ;;;; This file is part of GNU Guile.
;;;; ;;;;
@ -97,7 +97,6 @@
(with-test-prefix "opaque objects" (with-test-prefix "opaque objects"
(test-inferior-objects (test-inferior-objects
((make-guardian) smob (? integer?))
((%make-void-port "w") port (? inferior-object?)) ((%make-void-port "w") port (? inferior-object?))
((open-input-string "hello") port (? inferior-object?)) ((open-input-string "hello") port (? inferior-object?))
((lambda () #t) program _) ((lambda () #t) program _)