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:
parent
18064e19f6
commit
062223a5df
@ -350,6 +350,8 @@ svc_exec(const char *arg1, const char *arg2)
|
||||
size_t bytes;
|
||||
bool prefixed = false;
|
||||
int slave_tty;
|
||||
sigset_t sigchldmask;
|
||||
sigset_t oldmask;
|
||||
|
||||
/* Setup our signal pipe */
|
||||
if (pipe(signal_pipe) == -1)
|
||||
@ -439,10 +441,17 @@ svc_exec(const char *arg1, const char *arg2)
|
||||
}
|
||||
|
||||
free(buffer);
|
||||
|
||||
sigemptyset (&sigchldmask);
|
||||
sigaddset (&sigchldmask, SIGCHLD);
|
||||
sigprocmask (SIG_BLOCK, &sigchldmask, &oldmask);
|
||||
|
||||
close(signal_pipe[0]);
|
||||
close(signal_pipe[1]);
|
||||
signal_pipe[0] = signal_pipe[1] = -1;
|
||||
|
||||
sigprocmask (SIG_SETMASK, &oldmask, NULL);
|
||||
|
||||
if (master_tty >= 0) {
|
||||
/* Why did we do this? */
|
||||
/* signal (SIGWINCH, SIG_IGN); */
|
||||
|
Loading…
Reference in New Issue
Block a user