Можно сделать прототип функции signal() более понятным, применяя следующее определение типа для указателя на функцию обработчика сигнала:
typedef void (*sighandler_t)(int)
Это позволяет переписать прототип функции signal() таким образом:
sighandler_t signal(int sig, sighandler_t handler);
Если определен макрос проверки возможностей _GNU_SOURCE, то библиотека glibc представляет нестандартный тип данных sighandler_t в заголовочном файле
Вместо указания адреса функции в виде аргумента handler функции signal() можно указать одно из следующих значений.
• SIG_DFL — сбросить диспозицию сигнала до значения по умолчанию (см. табл. 20.1). Применяется для отмены эффекта предыдущего вызова функции signal(), изменившего диспозицию сигнала.
• SIG_IGN — игнорировать сигнал. При генерации сигнала для процесса ядро незаметно удаляет сгенерированный сигнал. Процесс никогда не узнает, что сигнал был сгенерирован.
Успешный вызов функции signal() возвращает предыдущее значение диспозиции сигнала, которое может представлять собой адрес ранее установленной функции обработчика либо одной из двух констант: SIG_DFL или SIG_IGN. При возникновении ошибки функция signal() возвращает значение SIG_ERR.
Активация обработчика сигнала может в любое время прервать ход выполнения программы. Ядро осуществляет вызов обработчика от имени процесса, а по возвращении из обработчика выполнение программы возобновляется с той же точки, в которой оно было прервано. Эта последовательность проиллюстрирована на рис. 20.1.
Хотя обработчики сигнала могут выполнять практически любое действие, в целом они должны быть максимально простыми. Мы вернемся к этой теме в разделе 21.1.
В листинге 20.1 показан простой пример функции обработчика сигнала, а также основной программы, устанавливающей эту функцию в качестве обработчика сигнала SIGINT. (Драйвер терминала генерирует этот сигнал при вводе с клавиатуры символа
Рис. 20.1.
Листинг 20.1. Установка обработчика для SIGINT
signals/ouch.c
#include
#include "tlpi_hdr.h"
static void
sigHandler(int sig)
{
printf("Ouch!\n"); /* НЕБЕЗОПАСНО (см. подраздел 21.1.2) */
}
int
main(int argc, char *argv[])
{
int j;
if (signal(SIGINT, sigHandler) == SIG_ERR)
errExit("signal");
for (j = 0;; j++) {
printf("%d\n", j);
sleep(3); /* Медленный цикл… */
}
}
signals/ouch.c
Основная программа непрерывно выполняет цикл. При каждом проходе по циклу она увеличивает значение счетчика и распечатывает его значение, а затем «засыпает» на несколько секунд. (Для этого мы используем функцию sleep(), приостанавливающую выполнение вызывавшего ее кода на указанное количество секунд. Описание этой функции дано в подразделе 23.4.1.)
При запуске программы из листинга 20.1 на экране мы увидим примерно следующее:
$ ./ouch
0
Ouch!
1
2
Ouch!
3
Quit (core dumped)
После активации ядром обработчика сигнала, номер сигнала, ставшего причиной активации, передается в обработчик в виде целочисленного аргумента. (В листинге 20.1 — это аргумент sig.) Если перехватывается только один тип сигнала, то толку от этого аргумента мало. Однако мы можем установить один обработчик на перехват нескольких типов сигналов и использовать этот аргумент для определения того, какой именно сигнал стал причиной активации обработчика.
В листинге 20.2 приводится программа, устанавливающая один и тот же обработчик для сигналов SIGINT и SIGQUIT. (Сигнал SIGQUIT генерируется драйвером терминала, когда мы вводим символ
Следующий журнал сессии оболочки демонстрирует применение этой программы:
$ ./intquit
Caught SIGINT (1)
Caught SIGINT (2)
Caught SIGINT (3)
Caught SIGQUIT — that's all folks!