Когда лидер сессии открывает контролирующий терминал, он сразу же становится его контролирующим процессом. Если впоследствии произойдет отключение терминала, ядро уведомит об этом контролирующий процесс, отправив ему сигнал SIGHUP. Более подробно об этом моменте мы поговорим в подразделе 34.6.2.

Если процесс имеет контролирующий терминал, он может получить его файловый дескриптор, открыв специальный файл /dev/tty. Это может пригодиться в ситуации, когда стандартные ввод и вывод перенаправлены, а программа хочет удостовериться, что она соединена с контролирующим терминалом. Данный подход, к примеру, используется в функции getpass(), описанной в разделе 8.5. Если у процесса нет контролирующего терминала, открытие файла /dev/tty завершается ошибкой ENXIO.

Отключение процесса от контролирующего терминала

С помощью операции ioctl(fd, TIOCNOTTY) можно отключить процесс от его контролирующего терминала, указанного в виде файлового дескриптора fd. После этого вызова все попытки открыть файл /dev/tty будут завершаться неудачей (операция TIOCNOTTY не входит в стандарт SUSv3, однако поддерживается в большинстве UNIX-систем).

Если вызывающий процесс является контролирующим для терминала, во время его завершения (см. подраздел 34.6.2) происходит следующее.

1. Все процессы в сессии теряют соединение с контролирующим терминалом.

2. Контролирующий терминал теряет соединение с сессией и впоследствии может стать контролирующим процессом для другой сессии.

3. Ядро шлет всем участникам активной группы процессов сигнал SIGHUP (и SIGCONT), чтобы проинформировать их о потере контролирующего терминала.

Получение пути к контролирующему терминалу: ctermid()

Функция ctermid() возвращает путь к контролирующему терминалу.

#include  /* Объявляет константу L_ctermid */

char *ctermid(char *ttyname);

Возвращает указатель на строку, содержащую путь к контролирующему терминалу, или NULL, если путь не удается определить

Функция ctermid() возвращает путь к контролирующему терминалу двумя разными способами: через результат своего выполнения и посредством буфера, на который указывает ttyname.

Если аргумент ttyname не равен NULL, то он должен быть буфером размера как минимум L_ctermid и содержать копию пути. В этом случае значение, возвращаемое функцией, тоже является указателем на этот буфер. Если аргумент ttyname равен NULL, то ctermid() возвращает указатель на статически выделенный массив, содержащий путь; при этом функция ctermid() не является реентерабельной.

В Linux и других реализациях UNIX ctermid() обычно возвращает строку /dev/tty. Назначение этой функции заключается в упрощении переносимости программы на системы, отличные от UNIX.

34.5. Активные и фоновые группы процессов

Контролирующий терминал несет в себе понятие активной группы процессов. В любой момент времени только один процесс в рамках сессии может быть активным; остальные группы процессов в этой сессии являются фоновыми. Только активная группа процессов может свободно выполнять чтение и запись данных в контролирующем терминале. Сигнал, который генерируется в результате нажатия подходящей комбинации клавиш, передается драйвером терминала участникам активной группы процессов. Мы остановимся на этом подробнее в разделе 34.7.

Теоретически может возникнуть ситуация, в которой сессия не имеет активной группы процессов. Это может случиться, к примеру, если все процессы в активной группе завершатся и ни один другой процесс этого не заметит и не станет активным. Однако на практике такие ситуации крайне маловероятны. Обычно отслеживанием состояния активной группы занимается командная оболочка; заметив, что такая группа завершила свою работу (через вызов wait()), она сама становится активной.

Функции tcgetpgrp() и tcsetpgrp() используются соответственно для получения и изменения группы процессов терминала. Они в основном применяются командными оболочками для управления заданиями.

#include

pid_t tcgetpgrp(int fd);

Возвращает идентификатор активной группы процессов для терминала или –1, если случилась ошибка

int tcsetpgrp(int fd, pid_t pgid);

Возвращает 0 при успешном завершении или –1, если произошла ошибка

Функция tcgetpgrp() возвращает идентификатор активной группы процессов для терминала, заданного с помощью файлового дескриптора fd, который должен быть управляющим по отношению к вызывающему процессу.

Если у текущего терминала нет активной группы процессов, tcgetpgrp() возвращает значение больше 1, которое не совпадает с идентификаторами любой существующей группы (такое поведение описано в стандарте SUSv3).

Перейти на страницу:

Похожие книги