An operating system consists of processes. These processes are executed in a specific order and are related to each other. There are two categories of processes, those focused on the user environment and those focused on the hardware environment.
When a program runs, the system will create a process by placing the program data and code in memory and creating a runtime stack. A process is therefore an instance of a program with an associated processor environment (ordinal counter, registers, etc...) and memory environment.
Each process has:
a PID: Process IDentifier, a unique process identifier
a PPID: Parent Process IDentifier, unique identifier of parent process
By successive filiations, the init process is the father of all processes.
A process is always created by a parent process
A parent process can have multiple child processes
There is a parent/child relationship between processes. A child process is the result of the parent process calling the fork() primitive and duplicating its own code to create a child. The PID of the child is returned to the parent process so that it can talk to it. Each child has its parent's identifier, the PPID.
The PID number represents the process at the time of execution. When the process finishes, the number is available again for another process. Running the same command several times will produce a different PID each time.
Note
Processes are not to be confused with threads. Each process has its own memory context (resources and address space), while threads from the same process share this same context.
When a command is executed, the user's credentials are passed to the created process.
By default, the actual UID and GID (of the process) are therefore identical to the actualUID and GID (the UID and GID of the user who executed the command).
When a SUID (and/or SGID) is set on a command, the actual UID (and/or GID) becomes that of the owner (and/or owner group) of the command and no longer that of the user or user group that issued the command. Effective and real UIDs are therefore different.
Each time a file is accessed, the system checks the rights of the process according to its effective identifiers.
A process cannot be run indefinitely, as this would be to the detriment of other running processes and would prevent multitasking.
The total processing time available is therefore divided into small ranges, and each process (with a priority) accesses the processor in a sequenced manner. The process will take several states during its life among the states:
ready: waiting for the availability of the process
in execution: accesses the processor
suspended: waiting for an I/O (input/output)
stopped: waiting for a signal from another process
zombie: request for destruction
dead: the parent process ends the child process
The end-of-process sequencing is as follows:
Closing of the open files
Release of the used memory
Sending a signal to the parent and child processes
When a parent process dies, its children are said to be orphans. They are then adopted by the init process which will destroy them.
By pressing the CTRL + Z keys simultaneously, the synchronous process is temporarily suspended. Access to the prompt is restored after displaying the number of the process that has just been suspended.
The & statement executes the command asynchronously (the command is then called job) and displays the number of job. Access to the prompt is then returned.
Example:
$ time ls -lR / > list.ls 2> /dev/null &
[1] 15430
$
The job number is obtained during background processing and is displayed in square brackets, followed by the PID number.
The fg command puts the process in the foreground:
$ time ls -lR / > list.ls 2>/dev/null &
$ fg 1
time ls -lR / > list.ls 2/dev/null
while the command bg places it in the background:
[CTRL]+[Z]
^Z
[1]+ Stopped
$ bg 1
[1] 15430
$
Whether it was put in the background when it was created with the & argument or later with the CTRL +Z keys, a process can be brought back to the foreground with the fg command and its job number.
The command nice allows the execution of a command by specifying its priority.
nice priority command
Example:
$ nice -n+15 find / -name "file"
Unlike root, a standard user can only reduce the priority of a process. Only values between +0 and +19 will be accepted.
Tip
This last limitation can be lifted on a per-user or per-group basis by modifying the /etc/security/limits.conf file.
The renice command allows you to change the priority of a running process.
renice priority [-g GID] [-p PID] [-u UID]
Example:
$ renice +15 -p 1664
| Option | Description |
|--------|-----------------------------------|
| -g | GID of the process owner group. |
| -p | PID of the process. |
| -u | UID of the process owner. |
The renice command acts on processes already running. It is therefore possible to change the priority of a specific process, but also of several processes belonging to a user or a group.
Tip
The pidof command, coupled with the xargs command (see the Advanced Commands course), allows a new priority to be applied in a single command: