printf("%s: exiting\n", argv[0]);
exit(EXIT_SUCCESS);
}
signals/sig_sender.c
Программа из листинга 20.7 предназначена для перехвата и передачи информации о сигналах, отправленных программой из листинга 20.6. Она выполняет такие шаги.
• Устанавливает обработчик для перехвата всех сигналов
• Если в программу был передан аргумент командной строки, то программа блокирует все сигналы на протяжении количества секунд, установленных этим аргументом, а затем, перед разблокированием сигналов, отображает набор ожидающих сигналов
• Программа выполняет цикл while, потребляющий процессорное время до тех пор, пока не установлен флаг gotSigint
• После выхода из цикла while программа отображает количество всех полученных сигналов
Мы сначала используем эти две программы для иллюстрации того, что заблокированный сигнал доставляется только однажды, вне зависимости от того, сколько раз он был сгенерирован. Это делается путем установки времени сна получателя и отправки всех сигналов до окончания установленного временного интервала.
$ ./sig_receiver 15 &
[1] 5368
./sig_receiver: PID is 5386
./sig_receiver: sleeping for 15 seconds
$ ./sig_sender 5368 1000000 10 2
./sig_sender: sending signal 10 to process 5368 1000000 times
./sig_sender: exiting
./sig_receiver: pending signals are:
2 (Interrupt)
10 (User defined signal 1)
./sig_receiver: signal 10 caught 1 time
[1]+ Done./sig_receiver 15
В аргументах командной строки для отправляющей программы были указаны сигналы SIGUSR1 и SIGINT, являющиеся в Linux/ х86 сигналами 10 и 2 соответственно.
Из программного вывода, приведенного выше, мы видим, что из миллиона отправленных нами сигналов получателю доставлен был только один.
Даже если процесс не блокирует сигналы, он может получить сигналов меньше, чем было отправлено. Это может произойти, если сигналы отправляются настолько быстро, что они прибывают до того, как выполнение получающего сигналы процесса может быть запланировано ядром. В результате, множественные сигналы записываются в набор ожидающих сигналов процесса лишь однажды. Если мы запустим программу из листинга 20.7 без аргументов командной строки (так, чтобы программа не блокировала сигналы и не переходила в режим сна), на экране мы увидим следующее:
$ ./sig_receiver &
[1] 5393
./sig_receiver: PID is 5393
$ ./sig_sender 5393 1000000 10 2
./sig_sender: sending signal 10 to process 5393 1000000 times
./sig_sender: exiting
./sig_receiver: signal 10 caught 52 times
[1]+ Done./sig_receiver
Из миллиона отправленных сигналов только 52 были перехвачены процессом-получателем. (Точное количество перехваченных сигналов будет варьироваться в зависимости от причуд алгоритма планирования ядра.) Причина этого заключается в том, что каждый раз, когда программа-отправитель запускается по плану, она отправляет получателю несколько сигналов. Однако только один из этих сигналов помечается как ожидающий и доставляется лишь тогда, когда у получателя есть возможность запуститься.
Листинг 20.7. Перехват и подсчет сигналов
signals/sig_receiver.c
#define _GNU_SOURCE
#include
#include "signal_functions.h" /* Объявление printSigset() */
#include "tlpi_hdr.h"
static int sigCnt[NSIG]; /* Считает количество доставок каждого сигнала */
static volatile sig_atomic_t gotSigint = 0;
/* Устанавливается ненулевое значение в случае доставки SIGINT */
static void
{
if (sig == SIGINT)
gotSigint = 1;
else
sigCnt[sig]++;
}
int
main(int argc, char *argv[])
{
int n, numSecs;
sigset_t pendingMask, blockingMask, emptyMask;
printf("%s: PID is %ld\n", argv[0], (long) getpid());
(void) signal(n, handler); /* Игнорируем ошибки */
/* Если указано время сна, временно заблокировать все сигналы, спать (пока другой
процесс посылает нам сигналы), а затем отобразить маску ожидающих сигналов
и разблокировать все сигналы */
numSecs = getInt(argv[1], GN_GT_0, NULL);