Листинг 26.3. Использование вызова waitpid() для получения статуса дочернего процесса

procexec/child_status.c

#include

#include "print_wait_status.h" /* Объявление printWaitStatus() */

#include "tlpi_hdr.h"

int

main(int argc, char *argv[])

{

int status;

pid_t childPid;

if (argc > 1 && strcmp(argv[1], "-help") == 0)

usageErr("%s [exit-status]\n", argv[0]);

switch (fork()) {

case –1: errExit("fork");

case 0: /* Потомок либо немедленно завершается с заданным кодом,

либо входит в цикл ожидания сигналов */

printf("Child started with PID = %ld\n", (long) getpid());

if (argc > 1) /* Был ли предоставлен статус в командной строке? */

exit(getInt(argv[1], 0, "exit-status"));

else /* Если нет, ждем сигналов */

for (;;)

pause();

exit(EXIT_FAILURE); /* В нашем случае никогда не выполнится,

но считается хорошим тоном */

default: /* Родитель ждет в цикле, пока потомок не будет

завершен — либо штатно, либо по сигналу */

for (;;) {

childPid = waitpid(–1, &status, WUNTRACED

#ifdef WCONTINUED /* Отсутствует в старых версиях Linux */

| WCONTINUED

#endif

);

if (childPid == –1)

errExit("waitpid");

/* Выводит статус как восьмеричное число

и десятичные значения отдельных байтов */

printf("waitpid() returned: PID=%ld; status=0x%04x (%d,%d)\n",

(long) childPid,

(unsigned int) status, status >> 8, status & 0xff);

printWaitStatus(NULL, status);

if (WIFEXITED(status) || WIFSIGNALED(status))

exit(EXIT_SUCCESS);

}

}

}

procexec/child_status.c

26.1.4. Завершение процесса из обработчика сигнала

Как показано в табл. 20.1, некоторые сигналы по умолчанию завершают процесс. В определенных обстоятельствах перед завершением работы возникает необходимость в освобождении ресурсов. Для этого можно предусмотреть обработчик, который будет перехватывать такие сигналы, освобождать ресурсы и только потом завершать процесс. Делая это, мы должны иметь в виду, что код завершения процесса доступен его родителю посредством вызовов wait() и waitpid(). Например, если сделать из обработчика сигнала вызов _exit(EXIT_SUCCESS), родитель будет знать, что потомок завершился успешно.

Если потомку нужно сообщить о том, что он завершается из-за сигнала, его обработчик должен сначала себя отключить, и затем еще раз послать тот же сигнал, которому уже ничто не помешает завершить работу процесса. В этом случае обработчик сигнала будет содержать примерно такой код:

void

handler(int sig)

{

/* Освобождаем ресурсы */

signal(sig, SIG_DFL); /* Отключаем обработчик */

raise(sig); /* Генерируем сигнал повторно */

}

26.1.5. Системный вызов waitid()

Как и waitpid(), вызов waitid() возвращает статус дочернего процесса. Однако, кроме этого, он предоставляет дополнительные возможности, недоступные в waitpid(). Этот системный вызов происходит из System V, хотя и не является частью стандарта SUSv3. Он поддерживается в ядре Linux с версии 2.6.9.

#include

int waitid(idtype_t idtype, id_t id, siginfo_t *infop, int options);

Возвращает 0 при успешном завершении или если был указан флаг WNOHANG и больше не осталось потомков для ожидания; возвращает –1 при ошибке

Аргументы idtype и id определяют, какого из потомков нужно ждать:

• если idtype равен P_ALL, ждем любого потомка; id игнорируется;

• если idtype равен P_PID, ждем потомка, чей идентификатор равен id;

• если idtype равен P_PGID, ждем любого потомка, идентификатор группы процессов которого равен id.

Стоит отметить, что waitid(), в отличие от waitpid(), не позволяет охватить всех потомков в группе вызывающего процесса, указав для id значение 0. Вместо этого следует явно задать идентификатор группы вызывающего процесса, используя значение, возвращенное вызовом getpgrp().

Главным отличием вызова waitid() от waitpid() является то, что он предоставляет более тонкий контроль за событиями ожидаемого потомка. Это обеспечивается за счет задания внутри параметра options одного или нескольких флагов, скомбинированных с помощью побитового ИЛИ.

• WEXITED — позволяет ждать потомка, который завершился штатным образом или по сигналу.

• WSTOPPED — позволяет ждать потомка, который был остановлен по сигналу.

• WCONTINUED — позволяет ждать потомка, который возобновил работу, получив сигнал SIGCONT.

Параметр options также может содержать следующие флаги, разделенные побитовым ИЛИ.

• WNOHANG — означает то же самое, что и в случае с вызовом waitpid(). Если ни один из потомков, указанных в id, все еще не изменил свое состояние, вызов немедленно возвращается, не переходя к блокированию (то есть «опрашивает» потомков). В этом случае возвращаемое значение waitid() равно 0. Если у родителя не оказывается дочерних процессов, соответствующих значению id, waitid() возвращает ошибку ECHILD.

Перейти на страницу:

Похожие книги