Описание
Неправильное употребление errno в обработчике сигнала происходит, когда вы вызываете одну из этих функций в обработчике сигнала:
signal
: Вы вызываете signal
функция в обработчике сигнала и затем считала значение errno
.
Например, функция-обработчик сигнала handler
вызовы signal
и затем вызовы perror
, который читает errno
.
void handler(int signum) {
pfv old_handler = signal(signum, SIG_DFL);
if (old_handler == SIG_ERR) {
perror("SIGINT handler");
}
}
errno
- установка функции POSIX®: Вы вызываете errno
- функция установки POSIX в обработчике сигнала, но не восстанавливает errno
при возврате из обработчика сигнала.
Например, функция-обработчик сигнала handler
вызовы waitpid
, который изменяет errno
, но не восстанавливает errno
перед возвратом.
void handler(int signum) {
int rc = waitpid(-1, NULL, WNOHANG);
if (ECHILD != errno) {
}
}
Риск
В каждом случае, который отмечает средство проверки, вы рискуете использовать неопределенное значение errno
.
signal
: Если вызов signal
в обработчике сигнала перестал работать, значение errno
неопределенно (см. Стандарт C11, Секунда. 7.14.1.1). Если вы используете определенное значение errno
, вы видите неожиданные результаты.
errno
- функция установки POSIX: errno
- установка функции устанавливает errno
onFailure. Если вы читаете errno
после того, как обработчик сигнала называется, и сам обработчик сигнала вызывает errno
- устанавливая функцию, вы видите неожиданные результаты.
Фиксация
Избегайте ситуаций, где вы рискуете использовать неопределенное значение errno
.
signal
: После вызова signal
функция в обработчике сигнала, не читайте errno
или используйте функцию, которая читает errno
.
errno
- функция установки POSIX: Прежде, чем вызвать errno
- установка функции в обработчике сигнала, сохраните errno
к временной переменной. Восстановите errno
от этой переменной прежде, чем возвратиться из обработчика сигнала.
Пример - чтение errno
После signal
Вызовите в обработчике сигнала
#include <signal.h>
#include <stdlib.h>
#include <stdio.h>
#define fatal_error() abort()
void handler(int signum) {
if (signal(signum, SIG_DFL) == SIG_ERR) {
perror("SIGINT handler");
}
}
int func(void) {
if (signal(SIGINT, handler) == SIG_ERR) {
/* Handle error */
fatal_error();
}
/* Program code */
if (raise(SIGINT) != 0) {
/* Handle error */
fatal_error();
}
return 0;
}
В этом примере, функциональном handler
называется, чтобы обработать SIGINT
сигнал. В теле handler
, signal
функция вызвана. После этого вызова, значения errno
неопределенно. Средство проверки повышает дефект когда perror
функция вызвана потому что perror
использует значение errno
.
Коррекция — старается не читать errno
После signal
Вызвать
Одна возможная коррекция не должна читать errno
после вызова signal
функция в обработчике сигнала. Откорректированный код здесь вызывает abort
функция через fatal_error
макрос вместо perror
функция.
#include <signal.h>
#include <stdlib.h>
#include <stdio.h>
#define fatal_error() abort()
void handler(int signum) {
if (signal(signum, SIG_DFL) == SIG_ERR) {
fatal_error();
}
}
int func(void) {
if (signal(SIGINT, handler) == SIG_ERR) {
/* Handle error */
fatal_error();
}
/* Program code */
if (raise(SIGINT) != 0) {
/* Handle error */
fatal_error();
}
return 0;
}