someString
.
void handleHup(int signum) {
free(someString);
someString = strdup('другая строка');
}
В реальных программах новое значение someString
вероятно, будет читаться из внешнего источника (такого как FIFO), но некоторые концепции актуальны и так. Теперь предположим, что основная часть программы копирует строку (этот код аналогичен реализации strcpy()
, хотя и не очень оптимизирован), когда поступает сигнал SIGHUP
.
src = someString;
while(*src)
*dest++ = *src++;
Когда главная часть программы возобновит выполнение, src
будет указывать на память, которая была освобождена обработчиком сигналов. Излишне говорить, что это очень плохая идея[62].
Чтобы решать проблемы такого типа, программный интерфейс сигналов POSIX позволяет процессу блокировать доставку процессу произвольного набора сигналов. При этом сигналы не отбрасываются, просто их доставка задерживается до тех пор, пока процесс не обозначит свою готовность обработать эти сигналы, разблокировав их. Чтобы правильно выполнить показанное выше копирование строки, программа должна блокировать SIGHUP
перед выполнением копирования и разблокировать его после. Обсудив интерфейс манипулирования масками сигналов, далее мы представим соответствующую версию кода.
Набор сигналов, которые процесс блокирует, часто называют маской сигнала
этого процесса. Маска сигналов процесса задается типом sigset_t
и содержит сигналы, заблокированные в данный момент. Функция sigprocmask()
позволяет процессу управлять его текущей маской сигналов.
#include <signal.h>
int sigprocmask(int what, sigset_t *set, sigset_t *oldest);
Первый параметр, what
, описывает, как должна выполняться манипуляция. Если set
равно NULL
, то what
игнорируется.
SIG_BLOCK | Сигналы в set добавляются к текущей маске сигналов. |
SIG_UNBLOCK | Сигналы в set исключаются из текущей маски сигналов. |
SIG_SETMASK | Блокируются сигналы из набора set , остальные разблокируются. |
Во всех трех случаях параметр oldset
типа sigset
_t указывает на исходную маску сигналов, если только он не равен NULL
— в этом случае oldset
игнорируется. Следующий вызов ищет текущую маску сигналов для запущенных процессов.
sigprocmask(SIG_BLOCK, NULL, ¤tSet);
Системный вызов sigprocmask
позволяет исправить код, представленный выше, который мог вызвать состояние состязаний. Все, что потребуется сделать — это блокировать SIGHUP
перед копированием строки и разблокировать после копирования. Следующее усовершенствование делает код более безопасным.
sigset_t hup;
sigemptyset(&hup);
sigaddset(&hup, SIGHUP);
sigprocmask(SIG_BLOCK, &hyp, NULL);
src = someString;
while(*src)
*dest++ = *src++;
sigprocmask(SIG_UNBLOCK, &hup, NULL);
Сложность обеспечения безопасности обработчика сигналов от состояния состязаний должно заставить вас писать обработчики, насколько возможно, простыми.
12.2.5. Нахождение набора ожидающих сигналов
Очень легко найти сигналы, находящиеся в состоянии ожидания (сигналы, которые должны быть доставлены, но в данный момент заблокированы).
#include <signal.h>
int sigpending(sigset_t *set);
Эта функция записывает по адресу, указанному set
, набор сигналов, которые в данный момент находятся в состоянии ожидания.
12.2.6. Ожидание сигналов
Когда программа построена преимущественно вокруг сигналов, часто необходимо, чтобы она ожидала появления какого-то сигнала, прежде чем продолжать работу. Системный вызов pause()
предоставляет простую возможность для этого.
#include <unistd.h>
int pause(void);
Функция pause()
не возвращает управления до тех пор, пока сигнал не будет доставлен процессу. Если зарегистрирован обработчик для этого сигнала, то он запускается до того, как pause()
вернет управление, pause()
всегда возвращает -1
и устанавливает errno
равным EINTR
.
Системный вызов sigsuspend()
предлагает альтернативный метод ожидания вызова сигнала.
#include <signal.h>
int sigsuspend(const sigset_t *mask);