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:
parent
b1d7d3538a
commit
66f9815c3f
19 changed files with 156 additions and 448 deletions
|
@ -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 \
|
||||||
|
|
|
@ -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"
|
||||||
|
|
|
@ -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 \
|
||||||
|
|
|
@ -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));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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"
|
||||||
|
|
|
@ -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"
|
|
||||||
}
|
|
|
@ -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 */
|
|
|
@ -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 */
|
||||||
|
|
|
@ -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
103
module/ice-9/guardians.scm
Normal 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)
|
|
@ -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)
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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))
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -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))
|
||||||
|
|
|
@ -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))
|
||||||
|
|
|
@ -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))
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -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))
|
||||||
|
|
|
@ -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 _)
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue