1
Fork 0
mirror of https://git.savannah.gnu.org/git/guile.git synced 2025-04-30 03:40:34 +02:00

Add file descriptor finalizers

* doc/ref/posix.texi (Ports and File Descriptors): Document new
  interfaces.
* libguile/filesys.c (scm_close, scm_close_fdes)
* libguile/fports.c (fport_close):
* libguile/ioext.c (scm_primitive_move_to_fdes): Call
  scm_run_fdes_finalizers.
* module/ice-9/fdes-finalizers.scm:
* test-suite/tests/fdes-finalizers.test:
* libguile/fdes-finalizers.h:
* libguile/fdes-finalizers.c: New files.
* module/Makefile.am:
* test-suite/Makefile.am:
* libguile/Makefile.am:
* libguile.h:
* libguile/init.c: Wire up new files.
This commit is contained in:
Andy Wingo 2016-08-30 23:35:10 +02:00
parent 4256e0655f
commit 2fa2e50a0f
13 changed files with 314 additions and 0 deletions

View file

@ -559,6 +559,51 @@ Duplicates in the input vectors appear only once in output.
An additional @code{select!} interface is provided. An additional @code{select!} interface is provided.
@end deffn @end deffn
While it is sometimes necessary to operate at the level of file
descriptors, this is an operation whose correctness can only be
considered as part of a whole program. So for example while the effects
of @code{(string-set! x 34 #\y)} are limited to the bits of code that
can access @var{x}, @code{(close-fdes 34)} mutates the state of the
entire process. In particular if another thread is using file
descriptor 34 then their state might be corrupted; and another thread
which opens a file might cause file descriptor 34 to be re-used, so that
corruption could manifest itself in a strange way.
@cindex fdes finalizers
@cindex file descriptor finalizers
@cindex finalizers, file descriptor
However when working with file descriptors, it's common to want to
associate information with the file descriptor, perhaps in a side table.
To support this use case and to allow user code to remove an association
when a file descriptor is closed, Guile offers @dfn{fdes finalizers}.
As the name indicates, fdes finalizers are finalizers -- they can run in
response to garbage collection, and they can also run in response to
explicit calls to @code{close-port}, @code{close-fdes}, or the like. As
such they inherit many of the pitfalls of finalizers: they may be
invoked from concurrent threads, or not at all. @xref{Foreign Object
Memory Management}, for more on finalizers.
To use fdes finalizers, import their module;
@example
(use-modules (ice-9 fdes-finalizers))
@end example
@deffn {Scheme Procedure} add-fdes-finalizer! fdes finalizer
@deffnx {Scheme Procedure} remove-fdes-finalizer! fdes finalizer
Add or remove a finalizer for @var{fdes}. A finalizer is a procedure
that is called by Guile when a file descriptor is closed. The file
descriptor being closed is passed as the one argument to the finalizer.
If a finalizer has been added multiple times to a file descriptor, to
remove it would require that number of calls to
@code{remove-fdes-finalizer!}.
The finalizers added to a file descriptor are called by Guile in an
unspecified order, and their return values are ignored.
@end deffn
@node File System @node File System
@subsection File System @subsection File System
@cindex file system @cindex file system

View file

@ -47,6 +47,7 @@ extern "C" {
#include "libguile/eval.h" #include "libguile/eval.h"
#include "libguile/evalext.h" #include "libguile/evalext.h"
#include "libguile/extensions.h" #include "libguile/extensions.h"
#include "libguile/fdes-finalizers.h"
#include "libguile/feature.h" #include "libguile/feature.h"
#include "libguile/filesys.h" #include "libguile/filesys.h"
#include "libguile/finalizers.h" #include "libguile/finalizers.h"

View file

@ -143,6 +143,7 @@ libguile_@GUILE_EFFECTIVE_VERSION@_la_SOURCES = \
evalext.c \ evalext.c \
expand.c \ expand.c \
extensions.c \ extensions.c \
fdes-finalizers.c \
feature.c \ feature.c \
filesys.c \ filesys.c \
finalizers.c \ finalizers.c \
@ -252,6 +253,7 @@ DOT_X_FILES = \
evalext.x \ evalext.x \
expand.x \ expand.x \
extensions.x \ extensions.x \
fdes-finalizers.x \
feature.x \ feature.x \
filesys.x \ filesys.x \
fluids.x \ fluids.x \
@ -358,6 +360,7 @@ DOT_DOC_FILES = \
evalext.doc \ evalext.doc \
expand.doc \ expand.doc \
extensions.doc \ extensions.doc \
fdes-finalizers.doc \
feature.doc \ feature.doc \
filesys.doc \ filesys.doc \
fluids.doc \ fluids.doc \
@ -586,6 +589,7 @@ modinclude_HEADERS = \
evalext.h \ evalext.h \
expand.h \ expand.h \
extensions.h \ extensions.h \
fdes-finalizers.h \
feature.h \ feature.h \
finalizers.h \ finalizers.h \
filesys.h \ filesys.h \

129
libguile/fdes-finalizers.c Normal file
View file

@ -0,0 +1,129 @@
/* Copyright (C) 2016 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 library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
* 02110-1301 USA
*/
#ifdef HAVE_CONFIG_H
# include <config.h>
#endif
#include "libguile/_scm.h"
#include "libguile/hashtab.h"
#include "libguile/numbers.h"
#include "libguile/fdes-finalizers.h"
/* Table of fdes finalizers and associated lock. */
static scm_i_pthread_mutex_t fdes_finalizers_lock =
SCM_I_PTHREAD_MUTEX_INITIALIZER;
static SCM fdes_finalizers;
SCM_DEFINE (scm_add_fdes_finalizer_x, "add-fdes-finalizer!", 2, 0, 0,
(SCM fd, SCM finalizer),
"Add a finalizer that will be called when @var{fd} is closed.")
#define FUNC_NAME s_scm_add_fdes_finalizer_x
{
SCM h;
/* Check type. */
scm_to_uint (fd);
scm_i_pthread_mutex_lock (&fdes_finalizers_lock);
h = scm_hashv_create_handle_x (fdes_finalizers, fd, SCM_EOL);
scm_set_cdr_x (h, scm_cons (finalizer, scm_cdr (h)));
scm_i_pthread_mutex_unlock (&fdes_finalizers_lock);
return SCM_UNSPECIFIED;
}
#undef FUNC_NAME
SCM_DEFINE (scm_remove_fdes_finalizer_x, "remove-fdes-finalizer!", 2, 0, 0,
(SCM fd, SCM finalizer),
"Remove a finalizer that was previously added to the file\n"
"descriptor @var{fd}.")
#define FUNC_NAME s_scm_remove_fdes_finalizer_x
{
SCM h;
/* Check type. */
scm_to_uint (fd);
scm_i_pthread_mutex_lock (&fdes_finalizers_lock);
h = scm_hashv_get_handle (fdes_finalizers, fd);
if (scm_is_true (h))
scm_set_cdr_x (h, scm_delq1_x (finalizer, scm_cdr (h)));
scm_i_pthread_mutex_unlock (&fdes_finalizers_lock);
return SCM_UNSPECIFIED;
}
#undef FUNC_NAME
struct fdes_finalizer_data
{
SCM finalizer;
SCM fd;
};
static SCM
do_run_finalizer (void *data)
{
struct fdes_finalizer_data *fdata = data;
return scm_call_1 (fdata->finalizer, fdata->fd);
}
void
scm_run_fdes_finalizers (int fd)
{
SCM finalizers;
struct fdes_finalizer_data data;
data.fd = scm_from_int (fd);
scm_i_pthread_mutex_lock (&fdes_finalizers_lock);
finalizers = scm_hashv_ref (fdes_finalizers, data.fd, SCM_EOL);
if (!scm_is_null (finalizers))
scm_hashv_remove_x (fdes_finalizers, data.fd);
scm_i_pthread_mutex_unlock (&fdes_finalizers_lock);
for (; !scm_is_null (finalizers); finalizers = scm_cdr (finalizers))
{
data.finalizer = scm_car (finalizers);
scm_internal_catch (SCM_BOOL_T, do_run_finalizer, &data,
scm_handle_by_message_noexit, NULL);
}
}
static void
scm_init_fdes_finalizers (void)
{
#include "libguile/fdes-finalizers.x"
}
void
scm_register_fdes_finalizers ()
{
fdes_finalizers = scm_c_make_hash_table (0);
scm_c_register_extension ("libguile-" SCM_EFFECTIVE_VERSION,
"scm_init_fdes_finalizers",
(scm_t_extension_init_func) scm_init_fdes_finalizers,
NULL);
}

View file

@ -0,0 +1,34 @@
#ifndef SCM_FDES_FINALIZERS_H
#define SCM_FDES_FINALIZERS_H
/* Copyright (C) 2016 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 library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
* 02110-1301 USA
*/
#include "libguile/__scm.h"
SCM_INTERNAL SCM scm_add_fdes_finalizer_x (SCM fd, SCM finalizer);
SCM_INTERNAL SCM scm_remove_fdes_finalizer_x (SCM fd, SCM finalizer);
SCM_INTERNAL void scm_run_fdes_finalizers (int fd);
SCM_INTERNAL void scm_register_fdes_finalizers (void);
#endif /* SCM_FDES_FINALIZERS_H */

View file

@ -43,6 +43,7 @@
#include "libguile/_scm.h" #include "libguile/_scm.h"
#include "libguile/smob.h" #include "libguile/smob.h"
#include "libguile/fdes-finalizers.h"
#include "libguile/feature.h" #include "libguile/feature.h"
#include "libguile/fports.h" #include "libguile/fports.h"
#include "libguile/strings.h" #include "libguile/strings.h"
@ -290,6 +291,7 @@ SCM_DEFINE (scm_close, "close", 1, 0, 0,
return scm_close_port (fd_or_port); return scm_close_port (fd_or_port);
fd = scm_to_int (fd_or_port); fd = scm_to_int (fd_or_port);
scm_evict_ports (fd); /* see scsh manual. */ scm_evict_ports (fd); /* see scsh manual. */
scm_run_fdes_finalizers (fd);
SCM_SYSCALL (rv = close (fd)); SCM_SYSCALL (rv = close (fd));
/* following scsh, closing an already closed file descriptor is /* following scsh, closing an already closed file descriptor is
not an error. */ not an error. */
@ -312,6 +314,7 @@ SCM_DEFINE (scm_close_fdes, "close-fdes", 1, 0, 0,
int rv; int rv;
c_fd = scm_to_int (fd); c_fd = scm_to_int (fd);
scm_run_fdes_finalizers (c_fd);
SCM_SYSCALL (rv = close (c_fd)); SCM_SYSCALL (rv = close (c_fd));
if (rv < 0) if (rv < 0)
SCM_SYSERROR; SCM_SYSERROR;

View file

@ -49,6 +49,7 @@
#include <full-write.h> #include <full-write.h>
#include "libguile/_scm.h" #include "libguile/_scm.h"
#include "libguile/fdes-finalizers.h"
#include "libguile/strings.h" #include "libguile/strings.h"
#include "libguile/validate.h" #include "libguile/validate.h"
#include "libguile/gc.h" #include "libguile/gc.h"
@ -656,6 +657,7 @@ fport_close (SCM port)
{ {
scm_t_fport *fp = SCM_FSTREAM (port); scm_t_fport *fp = SCM_FSTREAM (port);
scm_run_fdes_finalizers (fp->fdes);
if (close (fp->fdes) != 0) if (close (fp->fdes) != 0)
/* It's not useful to retry after EINTR, as the file descriptor is /* It's not useful to retry after EINTR, as the file descriptor is
in an undefined state. See http://lwn.net/Articles/365294/. in an undefined state. See http://lwn.net/Articles/365294/.

View file

@ -56,6 +56,7 @@
#include "libguile/eval.h" #include "libguile/eval.h"
#include "libguile/evalext.h" #include "libguile/evalext.h"
#include "libguile/expand.h" #include "libguile/expand.h"
#include "libguile/fdes-finalizers.h"
#include "libguile/feature.h" #include "libguile/feature.h"
#include "libguile/filesys.h" #include "libguile/filesys.h"
#include "libguile/finalizers.h" #include "libguile/finalizers.h"
@ -398,6 +399,7 @@ scm_i_init_guile (void *base)
scm_bootstrap_programs (); scm_bootstrap_programs ();
scm_bootstrap_vm (); scm_bootstrap_vm ();
scm_register_r6rs_ports (); scm_register_r6rs_ports ();
scm_register_fdes_finalizers ();
scm_register_foreign (); scm_register_foreign ();
scm_register_foreign_object (); scm_register_foreign_object ();
scm_register_srfi_1 (); scm_register_srfi_1 ();

View file

@ -29,6 +29,7 @@
#include "libguile/_scm.h" #include "libguile/_scm.h"
#include "libguile/dynwind.h" #include "libguile/dynwind.h"
#include "libguile/fdes-finalizers.h"
#include "libguile/feature.h" #include "libguile/feature.h"
#include "libguile/fports.h" #include "libguile/fports.h"
#include "libguile/hashtab.h" #include "libguile/hashtab.h"
@ -266,6 +267,7 @@ SCM_DEFINE (scm_primitive_move_to_fdes, "primitive-move->fdes", 2, 0, 0,
if (rv == -1) if (rv == -1)
SCM_SYSERROR; SCM_SYSERROR;
stream->fdes = new_fd; stream->fdes = new_fd;
scm_run_fdes_finalizers (old_fd);
SCM_SYSCALL (close (old_fd)); SCM_SYSCALL (close (old_fd));
return SCM_BOOL_T; return SCM_BOOL_T;
} }

View file

@ -59,6 +59,7 @@ SOURCES = \
ice-9/eval-string.scm \ ice-9/eval-string.scm \
ice-9/eval.scm \ ice-9/eval.scm \
ice-9/expect.scm \ ice-9/expect.scm \
ice-9/fdes-finalizers.scm \
ice-9/format.scm \ ice-9/format.scm \
ice-9/ftw.scm \ ice-9/ftw.scm \
ice-9/futures.scm \ ice-9/futures.scm \

View file

@ -0,0 +1,25 @@
;;;; Copyright (C) 2016 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 library; if not, write to the Free Software
;;;; Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
;;;;
(define-module (ice-9 fdes-finalizers)
#:export (add-fdes-finalizer!
remove-fdes-finalizer!))
(eval-when (expand load eval)
(load-extension (string-append "libguile-" (effective-version))
"scm_init_fdes_finalizers"))

View file

@ -54,6 +54,7 @@ SCM_TESTS = tests/00-initial-env.test \
tests/eval.test \ tests/eval.test \
tests/eval-string.test \ tests/eval-string.test \
tests/exceptions.test \ tests/exceptions.test \
tests/fdes-finalizers.test \
tests/filesys.test \ tests/filesys.test \
tests/fluids.test \ tests/fluids.test \
tests/foreign.test \ tests/foreign.test \

View file

@ -0,0 +1,65 @@
;;;; Copyright (C) 2016 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 library; if not, write to the Free Software
;;;; Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
(define-module (test-suite test-fdes-finalizers)
#:use-module (test-suite lib)
#:use-module (test-suite guile-test)
#:use-module (ice-9 fdes-finalizers))
(define (test-file suffix)
(data-file-name (string-append "ports-test.tmp" suffix)))
(close-port (open-output-file (test-file ".1")))
(close-port (open-output-file (test-file ".2")))
(with-test-prefix "simple"
(let* ((call-count 0)
(f (lambda (fdes) (set! call-count (1+ call-count))))
(p (open-input-file (test-file ".1")))
(q (open-input-file (test-file ".2"))))
(pass-if-equal 0 call-count)
(add-fdes-finalizer! (fileno p) f)
(pass-if-equal 0 call-count)
(close-port q)
(pass-if-equal 0 call-count)
(close-port p)
(pass-if-equal 1 call-count)))
(with-test-prefix "multiple"
(let* ((call-count 0)
(f (lambda (fdes) (set! call-count (1+ call-count))))
(p (open-input-file (test-file ".1"))))
(pass-if-equal 0 call-count)
(add-fdes-finalizer! (fileno p) f)
(add-fdes-finalizer! (fileno p) f)
(pass-if-equal 0 call-count)
(close-port p)
(pass-if-equal 2 call-count)))
(with-test-prefix "with removal"
(let* ((call-count 0)
(f (lambda (fdes) (set! call-count (1+ call-count))))
(p (open-input-file (test-file ".1"))))
(pass-if-equal 0 call-count)
(add-fdes-finalizer! (fileno p) f)
(add-fdes-finalizer! (fileno p) f)
(remove-fdes-finalizer! (fileno p) f)
(pass-if-equal 0 call-count)
(close-port p)
(pass-if-equal 1 call-count)))
(delete-file (test-file ".1"))
(delete-file (test-file ".2"))