1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72
|
/* Handling of signal SIGPIPE. */
/* -------------------- Specification --------------------------------- */
#if defined(HAVE_SIGNALS) && defined(SIGPIPE)
#if !defined(MULTITHREAD)
/* Variable to be set ONLY during IO calls to pipes directed to subprocesses. */
modexp bool writing_to_subprocess = false;
#endif
/* Install the signal handler for SIGPIPE. */
local void install_sigpipe_handler (void);
#if !defined(RELY_ON_WRITING_TO_SUBPROCESS)
/* for clx/new-clx */
global void disable_sigpipe (void);
#endif
#endif
/* ---------------------- Implementation -------------------------------- */
#if defined(HAVE_SIGNALS) && defined(SIGPIPE)
/* SIGPIPE is the signal sent to a process calling write() on a pipe with no
readers. Together with the signal, the write() call is terminated with
return value -1, errno = EPIPE. Note that SIGPIPE is a *synchronous* signal:
it occurs only during write(), without delay (unless blocked).
The default action of SIGPIPE signals is to terminate the process without
core dump. This is fine for pipes that did exist prior to ourselves, but is
not what we want when we write to subprocesses we have spawned using
MAKE-PIPE-OUTPUT-STREAM or MAKE-PIPE-IO-STREAM. In this case we want a normal
error message, and not die of the signal. */
/* Signal handler for SIGPIPE: */
local void sigpipe_handler (int sig)
{
signal_acknowledge(SIGPIPE,&sigpipe_handler);
if (writing_to_subprocess)
/* Ignore the signal, the write() error code is sufficient. */
return;
/* Revert to the default handler and re-raise the signal.
This should be sufficient to kill us. */
SIGNAL(SIGPIPE,SIG_DFL);
raise(SIGPIPE);
}
#if !defined(RELY_ON_WRITING_TO_SUBPROCESS)
/* If X Server decides to not longer talk to us, the pipe or socket gets
broken; If SIGPIPE is not blocked, we receive it and do not handle it,
which causes the CLISP to quit. If we block it the system calls return
with EPIPE instead, enabling libX to handle it. [Raising the XIOError.] */
global void disable_sigpipe (void) {
begin_system_call();
/* 'signalblock_on' and 'signalblock_off' should be used in pairs.
Since we do not want to unblock the SIGPIPE ever; I do this 'skip_that'
klugde here. [signalblock_on alone may produces a syntax error.] */
signalblock_on (SIGPIPE);
goto skip_that;
signalblock_off (SIGPIPE);
skip_that:;
end_system_call();
}
#endif
#define install_sigpipe_handler() \
SIGNAL(SIGPIPE,&sigpipe_handler);
#endif
|