diff --git a/am/bootstrap.am b/am/bootstrap.am
index 345b10ffc..66d91a165 100644
--- a/am/bootstrap.am
+++ b/am/bootstrap.am
@@ -147,6 +147,7 @@ SOURCES = \
ice-9/futures.scm \
ice-9/gap-buffer.scm \
ice-9/getopt-long.scm \
+ ice-9/guardians.scm \
ice-9/hash-table.scm \
ice-9/hcons.scm \
ice-9/history.scm \
diff --git a/libguile.h b/libguile.h
index 6b76abe29..75554c0c1 100644
--- a/libguile.h
+++ b/libguile.h
@@ -1,7 +1,7 @@
#ifndef 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.
This file is part of Guile.
@@ -61,7 +61,6 @@ extern "C" {
#include "libguile/generalized-vectors.h"
#include "libguile/goops.h"
#include "libguile/gsubr.h"
-#include "libguile/guardians.h"
#include "libguile/hash.h"
#include "libguile/hashtab.h"
#include "libguile/hooks.h"
diff --git a/libguile/Makefile.am b/libguile/Makefile.am
index e8be11227..03e204941 100644
--- a/libguile/Makefile.am
+++ b/libguile/Makefile.am
@@ -173,7 +173,6 @@ libguile_@GUILE_EFFECTIVE_VERSION@_la_SOURCES = \
generalized-vectors.c \
goops.c \
gsubr.c \
- guardians.c \
hash.c \
hashtab.c \
hooks.c \
@@ -292,7 +291,6 @@ DOT_X_FILES = \
generalized-vectors.x \
goops.x \
gsubr.x \
- guardians.x \
hash.x \
hashtab.x \
hooks.x \
@@ -397,7 +395,6 @@ DOT_DOC_FILES = \
generalized-vectors.doc \
goops.doc \
gsubr.doc \
- guardians.doc \
hash.doc \
hashtab.doc \
hooks.doc \
@@ -644,7 +641,6 @@ modinclude_HEADERS = \
generalized-vectors.h \
goops.h \
gsubr.h \
- guardians.h \
hash.h \
hashtab.h \
hooks.h \
diff --git a/libguile/deprecated.c b/libguile/deprecated.c
index 7329e3b5e..c3ba0a9eb 100644
--- a/libguile/deprecated.c
+++ b/libguile/deprecated.c
@@ -23,6 +23,13 @@
#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"
#if (SCM_ENABLE_DEPRECATED == 1)
@@ -30,6 +37,26 @@
/* 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));
+}
diff --git a/libguile/deprecated.h b/libguile/deprecated.h
index f1a76baa2..dbe2e4ce2 100644
--- a/libguile/deprecated.h
+++ b/libguile/deprecated.h
@@ -24,6 +24,7 @@
#if (SCM_ENABLE_DEPRECATED == 1)
+SCM_DEPRECATED SCM scm_make_guardian (void);
/* Deprecated declarations go here. */
void scm_i_init_deprecated (void);
diff --git a/libguile/finalizers.c b/libguile/finalizers.c
index 07335a970..44648fe13 100644
--- a/libguile/finalizers.c
+++ b/libguile/finalizers.c
@@ -39,7 +39,6 @@
#include "foreign.h"
#include "gc-internal.h"
#include "gsubr.h"
-#include "guardians.h"
#include "init.h"
#include "numbers.h"
#include "ports.h"
diff --git a/libguile/guardians.c b/libguile/guardians.c
deleted file mode 100644
index fa8c8b8f7..000000000
--- a/libguile/guardians.c
+++ /dev/null
@@ -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
- . */
-
-
-/* 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
-#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 ("#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"
-}
diff --git a/libguile/guardians.h b/libguile/guardians.h
deleted file mode 100644
index 7a081bd5c..000000000
--- a/libguile/guardians.h
+++ /dev/null
@@ -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
- . */
-
-
-
-#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 */
diff --git a/libguile/init.c b/libguile/init.c
index 32ff25cc6..592024d01 100644
--- a/libguile/init.c
+++ b/libguile/init.c
@@ -80,7 +80,6 @@
#include "gettext.h"
#include "goops.h"
#include "gsubr.h"
-#include "guardians.h"
#include "hash.h"
#include "hashtab.h"
#include "hooks.h"
@@ -460,7 +459,6 @@ scm_i_init_guile (struct gc_stack_addr base)
scm_init_weak_set ();
scm_init_weak_table ();
scm_init_weak_vectors ();
- scm_init_guardians (); /* requires smob_prehistory */
scm_init_standard_ports (); /* Requires fports */
scm_init_expand (); /* Requires structs */
scm_init_memoize (); /* Requires smob_prehistory */
diff --git a/module/ice-9/deprecated.scm b/module/ice-9/deprecated.scm
index 4c4a484ca..4d87e8fb3 100644
--- a/module/ice-9/deprecated.scm
+++ b/module/ice-9/deprecated.scm
@@ -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
;;;; modify it under the terms of the GNU Lesser General Public
@@ -16,9 +16,10 @@
;;;;
(define-module (ice-9 deprecated)
- #:use-module (ice-9 copy-tree)
- #:export ((copy-tree* . copy-tree)))
+ #:use-module (ice-9 guardians)
+ #:export ((make-guardian* . make-guardian)))
+#;
(define-syntax-rule (define-deprecated name message exp)
(begin
(define-syntax rule
@@ -28,14 +29,8 @@
exp)))
(export rule)))
-(define %allow-legacy-syntax-objects? (make-parameter #f))
-(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)
+(define (make-guardian*)
(issue-deprecation-warning
- "copy-tree in the default environment is deprecated. Import it
-from (ice-9 copy-tree) instead.")
- (copy-tree x))
+ "make-guardian in the default environment is deprecated. Import it
+from (ice-9 guardians) instead.")
+ (make-guardian))
diff --git a/module/ice-9/guardians.scm b/module/ice-9/guardians.scm
new file mode 100644
index 000000000..2531c6e8f
--- /dev/null
+++ b/module/ice-9/guardians.scm
@@ -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
+;;; .
+
+;;; 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)
diff --git a/module/ice-9/popen.scm b/module/ice-9/popen.scm
index 957cde0aa..befa528ab 100644
--- a/module/ice-9/popen.scm
+++ b/module/ice-9/popen.scm
@@ -1,6 +1,6 @@
;; 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.
;;;;
;;;; This library is free software; you can redistribute it and/or
@@ -20,6 +20,7 @@
(define-module (ice-9 popen)
#:use-module (ice-9 binary-ports)
+ #:use-module (ice-9 guardians)
#:use-module (ice-9 threads)
#:use-module (srfi srfi-1)
#:use-module (srfi srfi-9)
diff --git a/module/oop/goops.scm b/module/oop/goops.scm
index 1d05225e9..caa4bc3fe 100644
--- a/module/oop/goops.scm
+++ b/module/oop/goops.scm
@@ -85,7 +85,7 @@
-
+
;; Modules.
@@ -3537,7 +3537,6 @@ var{initargs}."
(define (find-subclass '))
(define (find-subclass '))
(define (find-subclass '))
-(define (find-subclass '))
(define (find-subclass '))
;; used to be a SMOB type, albeit not exported even to
diff --git a/test-suite/tests/eval.test b/test-suite/tests/eval.test
index 5bd2ac21a..968c78d12 100644
--- a/test-suite/tests/eval.test
+++ b/test-suite/tests/eval.test
@@ -1,5 +1,5 @@
;;;; 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.
;;;;
;;;; 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 (ice-9 documentation)
:use-module (ice-9 exceptions)
+ :use-module (ice-9 guardians)
:use-module (ice-9 local-eval))
diff --git a/test-suite/tests/fluids.test b/test-suite/tests/fluids.test
index 5026c2f30..9269c48bd 100644
--- a/test-suite/tests/fluids.test
+++ b/test-suite/tests/fluids.test
@@ -1,7 +1,7 @@
;;;; -*- scheme -*-
;;;; 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
;;;; 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
(define-module (test-suite test-fluids)
+ #:use-module (ice-9 guardians)
#:use-module (ice-9 threads)
#:use-module (test-suite lib)
#:use-module (system base compile))
diff --git a/test-suite/tests/gc.test b/test-suite/tests/gc.test
index 1827599a4..ab710f28d 100644
--- a/test-suite/tests/gc.test
+++ b/test-suite/tests/gc.test
@@ -1,6 +1,6 @@
;;;; gc.test --- test guile's garbage collection -*- scheme -*-
;;;; 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
;;;; modify it under the terms of the GNU Lesser General Public
@@ -18,6 +18,7 @@
(define-module (tests gc)
#:use-module (ice-9 documentation)
+ #:use-module (ice-9 guardians)
#:use-module (test-suite lib)
#:use-module ((system base compile) #:select (compile)))
@@ -93,6 +94,7 @@
(pass-if "Lexical vars are collectable"
(let ((l (compile
'(begin
+ (use-modules (ice-9 guardians))
(define guardian (make-guardian))
(let ((f (list 'foo)))
(guardian f))
diff --git a/test-suite/tests/guardians.test b/test-suite/tests/guardians.test
index fc4c9d861..002f3829d 100644
--- a/test-suite/tests/guardians.test
+++ b/test-suite/tests/guardians.test
@@ -1,7 +1,7 @@
;;;; guardians.test --- test suite for Guile Guardians -*- scheme -*-
;;;; Jim Blandy --- 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
;;;; modify it under the terms of the GNU Lesser General Public
@@ -37,6 +37,7 @@
(define-module (test-guardians)
:use-module (test-suite lib)
:use-module (ice-9 documentation)
+ :use-module (ice-9 guardians)
:use-module (ice-9 weak-vector))
diff --git a/test-suite/tests/threads.test b/test-suite/tests/threads.test
index fa89deeb2..d0a7412b2 100644
--- a/test-suite/tests/threads.test
+++ b/test-suite/tests/threads.test
@@ -1,7 +1,7 @@
;;;; threads.test --- Tests for Guile threading. -*- scheme -*-
;;;;
;;;; 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
;;;; 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
(define-module (test-threads)
+ #:use-module (ice-9 guardians)
#:use-module (ice-9 threads)
#:use-module (system base compile)
#:use-module (test-suite lib))
diff --git a/test-suite/tests/types.test b/test-suite/tests/types.test
index 9a9cdf73d..03dbea83a 100644
--- a/test-suite/tests/types.test
+++ b/test-suite/tests/types.test
@@ -1,6 +1,6 @@
;;;; 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.
;;;;
@@ -97,7 +97,6 @@
(with-test-prefix "opaque objects"
(test-inferior-objects
- ((make-guardian) smob (? integer?))
((%make-void-port "w") port (? inferior-object?))
((open-input-string "hello") port (? inferior-object?))
((lambda () #t) program _)