As its name implies, the state field of
the process descriptor describes what is currently happening to the
process. It consists of an array of flags, each of which describes a
possible process state. In the current Linux version, these states are
mutually exclusive, and hence exactly one flag of state always is set;
the remaining flags are cleared. The following are the possible process
states:
TASK_RUNNING
The process is either executing on a CPU or waiting to be executed.
TASK_INTERRUPTIBLE
The process is suspended (sleeping)
until some condition becomes true. Raising a hardware interrupt,
releasing a system resource the process is waiting for, or delivering a
signal are examples of conditions that might wake up the process (put
its state back to TASK_RUNNING).
TASK_UNINTERRUPTIBLE
Like TASK_INTERRUPTIBLE, except that
delivering a signal to the sleeping process leaves its state unchanged.
This process state is seldom used. It is valuable, however, under
certain specific conditions in which a process must wait until a given
event occurs without being interrupted. For instance, this state may be
used when a process opens a device file and the corresponding device
driver starts probing for a corresponding hardware device. The device
driver must not be interrupted until the probing is complete, or the
hardware device could be left in an unpredictable state.
TASK_STOPPED
Process execution has been stopped;
the process enters this state after receiving a SIGSTOP, SIGTSTP,
SIGTTIN, or SIGTTOU signal.
TASK_TRACED
Process execution has been stopped
by a debugger. When a process is being monitored by another (such as
when a debugger executes a ptrace( ) system call to monitor a test
program), each signal may put the process in the TASK_TRACED state.
-----------------------------------------------------
Two additional states of the process can be stored both in the
state field and in the exit_state field of the process descriptor; as
the field name suggests, a process reaches one of these two states only
when its execution is terminated:
EXIT_ZOMBIE
Process execution is terminated, but
the parent process has not yet issued a wait4( ) or waitpid( ) system
call to return information about the dead process. [*]Before
the wait( )-like call is issued, the kernel cannot discard the data
contained in the dead process descriptor because the parent might need
it.
[*] There are other wait( ) -like library functions, such as wait3(
) and wait( ), but in Linux they are implemented by means of the wait4(
) and waitpid( ) system calls.
EXIT_DEAD
The final state: the process is
being removed by the system because the parent process has just issued
a wait4( ) or waitpid( ) system call for it. Changing
its state from EXIT_ZOMBIE to EXIT_DEAD avoids race conditions due to
other threads of execution that execute wait( )-like calls on the same
process.
The value of the state field is usually set with a simple assignment. For instance:
p->state = TASK_RUNNING;
The kernel also uses the set_task_state and set_current_state
macros: they set the state of a specified process and of the process
currently executed, respectively. Moreover, these macros ensure that
the assignment operation is not mixed with other instructions by the
compiler or the CPU control unit. Mixing the instruction order may
sometimes lead to catastrophic results.
阅读(2068) | 评论(0) | 转发(0) |