Switch to wait_by_process when waitpid without WNOHANG returns nothing

By exclusively waiting by pgrp, we can fail to reap processes that
change their own pgrp then either crash or close their fds. If we wind
up in a situation where `waitpid(2)` returns 0 or ECHLD even though we
did not specify `WNOHANG` but we still have unreaped child processes,
wait on them by pid.

Closes #5596.
This commit is contained in:
Mahmoud Al-Qudsi 2019-01-28 22:25:55 -06:00
parent 93c0d3f4a5
commit 4dfaa33d95

View file

@ -511,6 +511,7 @@ static bool process_mark_finished_children(bool block_on_fg) {
if (pid > 0) { if (pid > 0) {
// A child process has been reaped // A child process has been reaped
debug(4, "Reaped PID %d", pid);
handle_child_status(pid, status); handle_child_status(pid, status);
// Always set WNOHANG (that is, don't hang). Otherwise we might wait on a non-stopped job // Always set WNOHANG (that is, don't hang). Otherwise we might wait on a non-stopped job
@ -520,15 +521,22 @@ static bool process_mark_finished_children(bool block_on_fg) {
} else if (pid == 0 || errno == ECHILD) { } else if (pid == 0 || errno == ECHILD) {
// No killed/dead children in this particular process group // No killed/dead children in this particular process group
if (!wait_by_process) { if (!wait_by_process) {
if ((options & WNOHANG) == 0) {
// This normally implies that the job has completed, but if we try to wait
// on a job that includes a process that changed its own group before we
// enter `waitpid`, we will be waiting forever. See #5596 for such a case.
wait_by_process = true;
continue;
}
break; break;
} }
} else { } else {
// pid < 0 indicates an error. One likely failure is ECHILD (no children), which is // pid < 0 indicates an error. One likely failure is ECHILD (no children), which is
// not an error and is ignored. The other likely failure is EINTR, which means we // not an error and is ignored in the branch above. The other likely failure is
// got a signal, which is considered an error. We absolutely do not break or return // EINTR, which means we got a signal, which is considered an error. We absolutely
// on error, as we need to iterate over all constructed jobs but we only call // do not break or return on error, as we need to iterate over all constructed jobs
// waitpid for one pgrp at a time. We do bypass future waits in case of error, // but we only call waitpid for one pgrp at a time. We do bypass future waits in
// however. // case of error, however.
has_error = true; has_error = true;
// Do not audibly complain on interrupt (see #5293) // Do not audibly complain on interrupt (see #5293)