устанавливается обработчик сигнала. Текущий обработчик сигнала (если таковой имеется) сохраняется, и таким образом мы можем восстановить его в конце функции.
Установка таймера
9-10
Таймер для процесса устанавливается на время (число секунд), заданное вызывающим процессом. Возвращаемое значение функции
alarm
— это число секунд, остающихся в таймере для процесса (если он уже установлен для процесса) в настоящий момент или 0 (если таймер не был установлен прежде). В первом случае мы выводим сообщение с предупреждением, поскольку мы стираем предыдущую установку таймера (см. упражнение 14.2).
Вызов функции connect
11-15
Вызывается функция
connect
, и если функция прерывается (
EINTR
), мы присваиваем переменной errno значение
ETIMEDOUT
. Сокет закрывается, чтобы не допустить продолжения трехэтапного рукопожатия.
Выключение таймера и восстановление предыдущего обработчика сигнала
16-18
Таймер при обнулении выключается, и восстанавливается предыдущий обработчик сигналов (если таковой имеется).
Обработка сигнала SIGALRM
20-24
Обработчик сигнала просто возвращает управление. Предполагается, что это прервет ожидание функции
connect
, заставив ее возвратить ошибку
EINTR
. Вспомните нашу функцию
signal
(см. листинг 5.5), которая не устанавливает флага
SA_RESTART
, когда перехватываемый сигнал — это сигнал
SIGALRM
.
Одним из важных моментов в этом примере является то, что мы всегда можем сократить период ожидания для функции
connect
, используя эту технологию, но мы не можем увеличить период, заданный для ядра. В Беркли-ядре тайм-аут для функции
connect
обычно равен 75 с. Мы можем задать меньшее значение для нашей функции, допустим 10, но если мы задаем большее значение, скажем 80, тайм- аут самой функции
connect
все равно составит 75 с.
Другой важный момент в данном примере — то, что мы используем возможность прерывания системного вызова (
connect
) для того, чтобы возвратить управление, прежде чем истечет время ожидания ядра. Такой подход допустим, когда мы выполняем системный вызов и можем обработать возвращение ошибки
EINTR
. Но в разделе 29.7 мы встретимся с библиотечной функцией, выполняющей системный вызов, которая сама выполняет заново системный вызов при возвращении ошибки
EINTR
. Мы можем продолжать работать с сигналом
SIGALRM
и в этом случае, но в листинге 29.6 мы увидим, что нам придется воспользоваться функциями
sigsetjmp
и
siglongjmp
, поскольку библиотечная функция игнорирует ошибку
EINTR
.
Тайм-аут для функции recvfrom (сигнал SIGALRM)
В листинге 14.2 показана новая версия функции
dg_cli
, приведенной в листинге 8.4, в которую добавлен вызов функции
alarm
для прерывания функции recvfrom при отсутствии ответа в течение 5 с.
Листинг 14.2. Функция dg_cli, в которой при установке тайм-аута для функции recvfrom используется функция alarm
//advio/dgclitimeo3.c
1 #include "unp.h"
2 static void signalrm(int);
3 void
4 dg_cli(FILE *fp, int sockfd, const SA *pservaddr, socklen_t servlen)