sigfillset(&blockingMask);
if (sigprocmask(SIG_SETMASK, &blockingMask, NULL) == –1)
errExit("sigprocmask");
printf("%s: sleeping for %d seconds\n", argv[0], numSecs);
sleep(numSecs);
if (sigpending(&pendingMask) == –1)
errExit("sigpending");
printf("%s: pending signals are: \n", argv[0]);
printSigset(stdout, "\t\t", &pendingMask);
sigemptyset(&emptyMask); /* Разблокировать все сигналы */
if (sigprocmask(SIG_SETMASK, &emptyMask, NULL) == –1)
errExit("sigprocmask");
}
continue;
if (sigCnt[n]!= 0)
printf("%s: signal %d caught %d time%s\n", argv[0], n,
sigCnt[n], (sigCnt[n] == 1)? "": "s");
exit(EXIT_SUCCESS);
}
signals/sig_receiver.c
Системный вызов sigaction() является альтернативой функции signal() в части установки диспозиции сигнала. Несмотря на то что функция sigaction() в той или иной степени сложнее в использовании, чем функция signal(), взамен она предоставляет большую гибкость. В частности, функция sigaction() позволяет получать текущую диспозицию сигнала без ее изменения, а также устанавливать различные атрибуты для точного управления тем, что происходит при активации обработчика сигнала. Кроме того, как мы выясним после тщательного анализа в разделе 22.7, функция sigaction() обладает куда большими свойствами переносимости при установке обработчика сигнала, по сравнению с функцией signal().
#include
int sigaction(int
Возвращает 0 при успешном завершении или –1 при ошибке
Аргумент sig означает сигнал, диспозицию которого мы хотим получить или изменить. Этим аргументом может быть любой сигнал, за исключением SIGKILL или SIGSTOP.
Аргумент act — это указатель на структуру, устанавливающую новую диспозицию сигнала. Если нам необходимо только лишь выяснить текущую диспозицию, то мы можем указать для этого аргумента значение NULL. Аргумент oldact — это указатель на структуру такого же типа, он используется для возврата информации о предыдущей диспозиции сигнала. Если нам это неинтересно, то мы можем задать для этого аргумента значение NULL. Структура, на которую указывают аргументы act и oldact, имеет следующий тип:
struct sigaction {
void (*sa_handler)(int); /* Адрес обработчика */
sigset_t sa_mask; /* Сигналы, блокируемые во время вызова обработчика */
int sa_flags; /* Флаги, контролирующие активацию обработчика */
void (*sa_restorer)(void); /* Не для использования в приложениях */
};
На самом деле структура sigaction несколько более сложна, чем показано выше. Мы обсудим это более детально в разделе 21.4.
Поле sa_handler соотносится с аргументом handler, передаваемым функции signal(). В данном поле указывается адрес обработчика сигнала или одна из констант — SIG_IGN или SIG_DFL. Поля sa_mask и sa_flags, которые мы обсудим в ближайшее время, интерпретируются только в том случае, если поле sa_handler содержит адрес обработчика сигнала, иными словами — значение, отличное от констант SIG_IGN и SIG_DFL. Оставшееся поле, sa_restorer, не предназначено для использования в приложениях и в стандарте SUSv3 не устанавливается.
Поле sa_restorer задействуется внутренне для гарантии того, что по завершении работы обработчика сигнала осуществляется системный вызов специального назначения sigreturn(), восстанавливающий контекст выполнения процесса в той точке, на которой оно было прервано. Пример такого использования данной функции может быть найден в файле исходного кода glibc: sysdeps/unix/sysv/linux/i386/sigaction.c.
Поле sa_mask определяет список сигналов, блокируемых во время активации обработчика, определенного аргументом sa_handler. После активации обработчика все сигналы из этого набора, не являющиеся в данный момент частью сигнальной маски процесса, автоматически добавляются в маску до вызова обработчика. Они остаются в сигнальной маске процесса до тех пор, пока не происходит возврат из обработчика сигнала, после чего они автоматически удаляются. Поле sa_mask позволяет указать набор сигналов, которым не разрешено прерывать выполнение данного обработчика. Кроме того, сигнал, активировавший обработчик, также автоматически добавляется в маску процесса. Это означает, что обработчик сигнала рекурсивно не прервет сам себя в случае доставки второго экземпляра того же сигнала во время выполнения кода обработчика. Так как заблокированные сигналы не ставятся в очередь, если любой из них генерируется повторно во время выполнения обработчика, то этот сигнал будет (позже) доставлен в процесс только однажды.
Поле sa_flags — битовая маска, устанавливающая разные параметры, контролирующие обработку сигнала. Следующие биты могут быть объединены в этом поле битовой операцией ИЛИ (|).
• SA_NOCLDSTOP — если аргумент sig равен SIGCHILD, не генерировать сигнал, если дочерний процесс остановлен или возобновлен в результате получения сигнала.