if (fcntl(STDIN_FILENO, F_SETFL, flags | O_ASYNC | O_NONBLOCK) == -1)
errExit("fcntl(F_SETFL)");
/* Переключаем терминал в режим cbreak */
if (ttySetCbreak(STDIN_FILENO, &origTermios) == -1)
errExit("ttySetCbreak");
for (done = FALSE, cnt = 0;!done; cnt++) {
for (j = 0; j < 100000000; j++)
continue; /* Немного замедляем главный цикл */
if (gotSigio) { /* Доступен ли ввод? */
gotSigio = 0;
/* Считываем весь доступный ввод, пока не случится ошибка (вероятно,
EAGAIN), не обнаружится конец файла (что невозможно в режиме cbreak)
или не будет прочитан символ # */
while (read(STDIN_FILENO, &ch, 1) > 0 &&!done) {
printf("cnt=%d; read %c\n", cnt, ch);
done = ch == '#';
}
}
}
/* Восстанавливаем исходные параметры терминала */
if (tcsetattr(STDIN_FILENO, TCSAFLUSH, &origTermios) == -1)
errExit("tcsetattr");
exit(EXIT_SUCCESS);
}
altio/demo_sigio.c
По умолчанию действием сигнала SIGIO является завершение процесса, так что его обработчик следует устанавливать до включения ввода/вывода на основе сигналов. Если поменять местами эти две операции, то образует своеобразный «зазор», на протяжении которого появление сигнала SIGIO будет завершать процесс.
В ряде реализаций UNIX сигнал SIGIO по умолчанию игнорируется.
59.3.1. Установка владельца файлового дескриптора
Владелец файлового дескриптора устанавливается с помощью операции fcntl() следующего вида:
fcntl(fd, F_SETOWN, pid);
Для получения сигнала о возможности ввода/вывода можно выбрать как один процесс, так и все процессы в группе. Если значение pid больше нуля, то оно интерпретируется как идентификатор процесса. Если же является отрицательным, то его модуль берется в качестве идентификатора группы процессов.
В старых реализациях UNIX флаги FIOSETOWN или SIOCSPGRP в вызове ioctl() имели тот же эффект, что и F_SETOWN. В Linux они поддерживаются с целью совместимости.
Обычно аргумент pid равен идентификатору вызывающего процесса (это позволяет передавать сигнал процессу, который удерживает открытым файловый дескриптор). Но можно указать и другой процесс или целую группу (например, группу вызывающего процесса); таким образом сигнал будет доставляться всем адресатам, проходя проверку прав доступа, описанную в разделе 20.5. При этом отправителем считается процесс, выполняющий операцию F_SETOWN.
Операция fcntl() F_GETOWN возвращает идентификатор процесса или группы процессов, которые получат сигнал о возможности ввода/вывода для заданного дескриптора:
id = fcntl(fd, F_GETOWN);
if (id == -1)
errExit("fcntl");
Идентификатор группы процессов возвращается в виде отрицательного числа.
В старых реализациях UNIX аналогами флага F_SETOWN для вызова ioctl() были флаги FIOGETOWN и SIOCGPGRP. Оба они поддерживаются в Linux.
Ограничения, традиционно применяемые в Linux к системным вызовам (в некоторых архитектурах, таких как x86), имеют следующие последствия: если идентификатор группы процессов, владеющей файловым дескриптором, меньше 4096, то вместо возвращения его отрицательного значения операция fcntl() F_GETOWN завершается ошибкой. Таким образом, функция-обертка fcntl() возвращает –1, а глобальная переменная errno содержит (положительный) идентификатор группы процессов. Это является следствием того факта, что интерфейс системных вызовов ядра сигнализирует об ошибках, присваивая errno отрицательные значения. Существует несколько ситуаций, когда такие результаты следует отличать от успешного вызова, возвращающего корректное отрицательное значение. Для этого библиотека glibc интерпретирует отрицательные итоговые результаты в диапазоне от –1 до –4095 в качестве ошибки, копирует их значение (по модулю) в переменную errno и делает так, что функция возвращает вызывающему процессу –1. Обычно данных действий достаточно для работы с теми немногими системными вызовами, которые способны возвращать отрицательный результат.
Операция fcntl() F_GETOWN является единственным реальным случаем, когда описанный подход не работает. Это значит следующее: приложение, выбирающее для получения сигналов о «готовности ввода/вывода» группу процессов (что довольно необычно), не может достоверно определить группу, владеющую дескриптором, задействуя флаг F_GETOWN.
Начиная с версии 2.11 функция-обертка fcntl() в библиотеке glibc больше не имеет проблемы с использованием флага F_GETOWN для групп процессов, чей идентификатор меньше 4096. Для этого данный вызов реализован в пользовательском пространстве на основе операции F_GETOWN_EX (см. подраздел 59.3.3), которая поддерживается в Linux 2.6.32 и выше.
59.3.2. Когда генерируется сигнал о возможности ввода/вывода?
В этом подразделе вы узнаете, когда именно генерируется сигнал о возможности ввода/вывода для разных типов файлов.
Терминалы и псевдотерминалы