Второй член структуры — rlim_max
, указывает жесткое ограничение лимита, переданного в параметре resource
, a rlim_cur
— мягкое ограничение. Это те же наборы лимитов, которыми манипулируют команды ulimit
и limit
, одна из которых встроена в большинство командных оболочек.
10.4. Примитивы процессов
Несмотря на относительно длинную дискуссию, необходимую для описания процесса, создание и уничтожение процессов в Linux достаточно просто.
10.4.1. Создание дочерних процессов
В Linux предусмотрены два системных вызова, которые создают новые процессы: fork()
и clone()
. Как упоминалось ранее, clone()
используется для создания потоков, и этот вызов будет кратко описан далее. А сейчас мы сосредоточимся на fork()
— наиболее популярном методе создания процессов.
#include
pid_t fork(void);
Этот системный вызов имеет уникальное свойство возвращать управление не один раз, а дважды: один раз в родительском процессе и другой — в дочернем. Обратите внимание, что мы не говорим "первый — в родительском" — написание кода, который делает какие-то предположения относительно предопределенного порядка — очень плохая идея.
Каждый из двух возвратов системного вызова fork()
имеет разные значения. В родительский процесс этот системный вызов возвращает pid вновь созданного дочернего процесса, а в дочернем он возвращает 0.
Разница возвращаемых значений — это единственное отличие, видимое процессам. Оба имеют одинаковые образы памяти, права доступа, открытые файлы и обработчики сигналов[19]. Рассмотрим простой пример программы, порождающей дочерний процесс.
#include
#include
#include
int main(void) {
pid_t child;
if (!(child = fork())) {
printf("в дочернем\n");
exit (0);
}
printf("в родительском - дочерний: %d\n", child);
return 0;
}
10.4.2. Наблюдение за уничтожением дочерних процессов
Сбор состояний возврата дочернего процесса называется ожиданием процесса. Это можно делать четырьмя способами, хотя только один из вызовов предоставляется ядром. Остальные три метода реализованы в стандартной библиотеке С. Поскольку системный вызов ядра принимает четыре аргумента, он называется wait4()
.
pid_t wait4(pid_t pid, int *status, int options, struct rusage *rusage);
Первый аргумент, pid
, представляет собой процесс, код возврата которого должен быть возвращен. Он может принимать ряд специальных значений.
pid < -1
Ожидать завершения любого дочернего процесса, чей pgid равен абсолютному значению pid
.
pid = -1
Ожидать прерывания любого дочернего процесса.
pid = 0
Ожидать завершения дочернего из той же группы процессов, что и текущий[20].
pid > 0
Ожидать выхода процесса pid
.
Второй параметр — это указатель на целое, которое устанавливается в значение, равное соду возврата того процесса, который заставляет wait4()
вернуть управление (мы будем зазывать его "проверяемым" процессом). Формат возвращенного состояния довольно закрученный, и для того, чтобы сделать его осмысленным, существует набор макросов.
Три события заставляют wait4()
вернуть состояние проверяемого процесса. Процесс может завершиться, он может быть прерван вызовом kill()
(получит фатальный сигнал) либо он может быть остановлен по какой-либо причине[21]. Вы можете узнать, что именно случилось, с помощью описанных ниже макросов, каждый из которых принимает возвращаемое состояние wait4()
в качестве единственного параметра.