Avoid race condition in runscript (bug #319865)

Under normal conditions, runscript creates one child and waits for its
termination, which is signaled by a pipe write from the SIGCHLD
sighandler.

When running killprocs however more than one SIGHCLD signal is generated, at
least on all of my amd64 boxes running on real hardware and in vmware.

When the first SIGCHLD occurs svc_exec leaves the loop and closes the pipe.
Subsequent SIGCHLDs during the close can lead to a race condition and create an
EBADF error in the pipe write (pipe is closed but the file handle is still !=
-1).

We avoid this by blocking SIGHCHLD during the pipe close.
This commit is contained in:
Thomas Pfaff 2010-12-03 14:01:28 -06:00 committed by William Hubbs
parent 18064e19f6
commit 062223a5df

View File

@ -350,6 +350,8 @@ svc_exec(const char *arg1, const char *arg2)
size_t bytes; size_t bytes;
bool prefixed = false; bool prefixed = false;
int slave_tty; int slave_tty;
sigset_t sigchldmask;
sigset_t oldmask;
/* Setup our signal pipe */ /* Setup our signal pipe */
if (pipe(signal_pipe) == -1) if (pipe(signal_pipe) == -1)
@ -439,10 +441,17 @@ svc_exec(const char *arg1, const char *arg2)
} }
free(buffer); free(buffer);
sigemptyset (&sigchldmask);
sigaddset (&sigchldmask, SIGCHLD);
sigprocmask (SIG_BLOCK, &sigchldmask, &oldmask);
close(signal_pipe[0]); close(signal_pipe[0]);
close(signal_pipe[1]); close(signal_pipe[1]);
signal_pipe[0] = signal_pipe[1] = -1; signal_pipe[0] = signal_pipe[1] = -1;
sigprocmask (SIG_SETMASK, &oldmask, NULL);
if (master_tty >= 0) { if (master_tty >= 0) {
/* Why did we do this? */ /* Why did we do this? */
/* signal (SIGWINCH, SIG_IGN); */ /* signal (SIGWINCH, SIG_IGN); */