Можно сделать прототип функции 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.4. Введение в обработчики сигналов

Обработчик сигнала (также называемый перехватчиком сигнала) — это функция, вызываемая при получении указанного сигнала процессом. В этом разделе приводятся базовые сведения об обработчиках сигналов, а в главе 21 — уже более подробная информация.

Активация обработчика сигнала может в любое время прервать ход выполнения программы. Ядро осуществляет вызов обработчика от имени процесса, а по возвращении из обработчика выполнение программы возобновляется с той же точки, в которой оно было прервано. Эта последовательность проиллюстрирована на рис. 20.1.

Хотя обработчики сигнала могут выполнять практически любое действие, в целом они должны быть максимально простыми. Мы вернемся к этой теме в разделе 21.1.

В листинге 20.1 показан простой пример функции обработчика сигнала, а также основной программы, устанавливающей эту функцию в качестве обработчика сигнала SIGINT. (Драйвер терминала генерирует этот сигнал при вводе с клавиатуры символа прерывания, обычно это сочетание клавиш Ctrl+C.) Обработчик просто выводит на печать сообщение — и возвращает управление в основную программу.

Рис. 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

Основная программа выполняет цикл, отображая увеличивающиеся числа

Нажмите Ctrl-C

Ouch! Обработчик сигнала выполнен

Управление возвращено в основную программу

2

Снова нажмите Ctrl-C

Ouch!

3

Нажмите Ctrl-\ (символ выхода терминала)

Quit (core dumped)

После активации ядром обработчика сигнала, номер сигнала, ставшего причиной активации, передается в обработчик в виде целочисленного аргумента. (В листинге 20.1 — это аргумент sig.) Если перехватывается только один тип сигнала, то толку от этого аргумента мало. Однако мы можем установить один обработчик на перехват нескольких типов сигналов и использовать этот аргумент для определения того, какой именно сигнал стал причиной активации обработчика.

В листинге 20.2 приводится программа, устанавливающая один и тот же обработчик для сигналов SIGINT и SIGQUIT. (Сигнал SIGQUIT генерируется драйвером терминала, когда мы вводим символ выхода, обычно Ctrl+\.) Код обработчика различает два сигнала путем осмотра аргумента sig — и выполняет различные действия для каждого. В функции main() мы используем функцию pause() (см. раздел 20.14) для блокирования процесса до перехвата сигнала.

Следующий журнал сессии оболочки демонстрирует применение этой программы:

$ ./intquit

Нажмите Ctrl-C

Caught SIGINT (1)

Нажмите Ctrl-C снова

Caught SIGINT (2)

и еще раз

Caught SIGINT (3)

Нажмите Ctrl-\

Caught SIGQUIT — that's all folks!

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

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