In computing, a parent process is a process that has created one or more child processes.
In the Unix operating system, every process except process 0 (the swapper) is created when another process executes the fork() system call. The process that invoked fork is the parent process and the newly created process is the child process. Every process (except process 0) has one parent process, but can have many child processes.
The operating system kernel identifies each process by its process identifier. Process 0 is a special process that is created when the system boots; after forking a child process (process 1), process 0becomes the swapper process (sometimes also known as the “idle task“). Process 1, known as init, is the ancestor of every other process in the system.
The operating system maintains a table that associates every process, by means of its process identifier (generally referred to as “pid“) to the data necessary for its functioning. During a process’s lifetime, such data might include memory segments designated to the process, the arguments it’s been invoked with, environment variables, counters about resource usage, user-id, group-id and group set, and maybe other types of information.
When a process terminates its execution, either by calling exit (even if implicitly, by executing a returncommand from the main function) or by receiving a signal that causes it to terminate abruptly, the operating system releases most of the resources and information related to that process, but still keeps the data about resource utilization and the termination status code, because a parent process might be interested in knowing if that child executed successfully (by using standard functions to decode the termination status code) and the amount of system resources it consumed during its execution.
By default, the system assumes that the parent process is indeed interested in such information at the time of the child’s termination, and thus sends the parent the signal SIGCHLD to alert that there is some data about a child to be collected. Such collection is done by calling a function of the wait family (eitherwait itself or one of its relatives, such as waitpid, waitid or wait4). As soon as this collection is made, the system releases those last bits of information about the child process and removes its pid from the process table. However, if the parent process lingers in collecting the child’s data (or fails to do it at all), the system has no option but keep the child’s pid and termination data in the process table indefinitely.
Such a terminated process whose data has not been collected is called a zombie process, or simply azombie, in the UNIX parlance. The name is a humorous analogy due to considering terminated process as “no longer alive” or “dead”—since it has really ceased functioning—and a lingering dead process still “incarnated” in the “world of the living” processes—the process table—which is therefore actually “undead”, or “zombie”.
Zombie processes might pose problems on systems with limited resources or that have limited-size process tables, as the creation of new, active processes might be prevented by the lack of resources still used by long lasting zombies.
It is, therefore, a good programming practice in any program that might spawn child processes to have code to prevent the formation of long lasting zombies from its original children. The most obvious approach is to have code that calls wait or one of its relatives somewhere after having created a new process. If the program is expected to create many child processes that may execute asynchronously and terminate in an unpredictable order, it is generally good to create a handler for the SIGCHLD signal, calling one of the wait-family function in a loop, until no uncollected child data remains. It is possible for the parent process to completely ignore the termination of its children and still not create zombies, but this requires the explicit definition of a handler for SIGCHLD through a call to sigaction with the special option flag SA_NOCLDWAIT.
Orphan processes is kind of the opposite situation of zombie processes, since it refers to the case where a parent process terminates before its child processes, in which case these children are said to become “orphaned”.
Unlike the asynchronous child-to-parent notification that happens when a child process terminates (via the SIGCHLD signal), child processes are not notified immediately when their parent finishes. Instead, the system simply redefines the “parent-pid” field in the child process’s data to be the process that is the “ancestor” of every other process in the system, whose pid generally has the value 1 (one), and whose name is traditionally “init“. It is thus said that “init ‘adopts’ every orphan process on the system”.
A somewhat common assumption by programmers new to UNIX is that the child processes of a terminating process will be adopted this process’s immediate parent process (hence those child processes’ “grandparent”). Such assumption is incorrect—unless, of course, that “grandparent” is inititself.
In the Linux kernel, in which there is a very slim difference between processes and POSIX threads, there are two kinds of parent processes, namely real parent and parent. Parent is the process that receives the SIGCHLD signal on child’s termination, whereas real parent is the thread that actually created this child process in a multithreaded environment. For a normal process, both these two values are same, but for a POSIX thread which acts as a process, these two values may be different.
- Srinivasan, Sundar (2010-09-01). “An Engineer’s Options & Futures: A Sneak-Peek into Linux Kernel – Chapter 2: Process Creation”. Sunnyeves.blogspot.com. Retrieved 2014-04-30.