I have a while loop that reads data from a child process using blocking I/O by redirecting stdout
of the child process to the parent process. Normally, as soon as the child process exits, a blocking read()
in this case will return since the pipe that is read from is closed by the child process.
Now I have a case where the read()
call does not exit for a child process that ends up in a zombie state. So the operating system is waiting for my code to reap it, but instead my code is blocking on the read()
call.
The child process itself does not have any child processes running at the time of the hang, and I do not see any file descriptors listed when looking in /proc/<child process PID>/fd
. The child process did however fork two daemon processes, whose purpose seems to be to monitor the child process (the child process is a proprietary application I do not have any control over, so it is hard to say for sure).
When run from a terminal, the child process I try to read()
from exits automatically, and in turn the daemon processes it forked terminate as well.
Linux version is 4.19.2.
What could be the reason of read()
not returning in this case?
Follow-up: How to avoid `read()` hanging in the following situation?
The child process did however fork two daemon processes ... What could be the reason of read()
not returning in this case?
Forked processes still have the file descriptor open when the child terminates. Hence read
call never returns 0.
Those daemon processes should close all file descriptors and open files for logging.
A possible reason (the most common) for read(2)
blocking on a pipe with a dead child, is that the parent has not closed the writing side of the pipe, so there's still an open (for writing) descriptor for that pipe. Close the writing side of the pipe in the parent process before reading from it. The child is dead (you said zombie) so it cannot be the process with the writing side of the pipe open. And don't forget to wait(2)
for the child in the parent, or you'll get a system full of zombies :)
Remember, you have to do two closes in your code:
One in the parent process, to close the writing side of the pipe, leaving the parent process with only a reading descriptor.
One in the child process (just before exec(2)
ing) closing the reading side of the pipe, leaving the child process only with a writing descriptor.
In case you want to use the pipe(2)
to send information to the child, change the reading for writing and viceversa in the above two points.