По аналогии с вызовом open() функция posix_openpt() использует для открытия первичного конца псевдотерминала файловый дескриптор с наименьшим доступным номером.
Вызов posix_openpt() также приводит к созданию в каталоге /dev/pts файла, связанного с первичным устройством псевдотерминала. Мы подробно остановимся на этом моменте, когда будем рассматривать функцию ptsname().
Функция posix_openpt() появилась в стандарте SUSv3 относительно недавно по инициативе комитета стандартизации POSIX. В исходной реализации псевдотерминала в System V получение доступного первичного устройства достигалось за счет открытия его
int
posix_openpt(int flags)
{
return open("/dev/ptmx", flags);
}
Каждый псевдотерминал использует определенное, пусть и небольшое, количество памяти, которую нельзя сбросить на диск, поэтому ядро вводит ограничение на количество псевдотерминалов стандарта UNIX 98 в системе. В версиях Linux вплоть до 2.6.3 данное ограничение определялось параметром конфигурации ядра (CONFIG_UNIX98_PTYS), по умолчанию равным 256. Но мы могли указать любое значение от 0 до 2048.
Начиная с версии 2.6.4 параметр конфигурации ядра CONFIG_UNIX98_PTYS был заменен более гибким подходом. Теперь максимальное количество псевдотерминалов ограничивается значением, указанным в файле /proc/sys/kernel/pty/max (который доступен лишь в Linux). По умолчанию это значение равно 4096 и не должно превышать 1 048 576. Существует также вспомогательный файл, доступный только для чтения, показывающий, сколько всего псевдотерминалов стандарта UNIX 98 используется на данный момент.
60.2.2. Изменение владельца и прав доступа ко вторичному устройству: вызов grantpt()
В стандарте SUSv3 предусмотрена функция grantpt(), позволяющая изменить владельца и права доступа ко вторичному устройству, связанному с первичным концом псевдотерминала, на который ссылается файловый дескриптор mfd. В Linux вызов grantpt() не является обязательным. Однако он требуется для корректной работы в ряде других систем, поэтому портируемые приложения должны выполнять его после вызова posix_openpt().
#define _XOPEN_SOURCE 500
#include
int grantpt(int
Возвращает 0 при успешном завершении или -1 при ошибке
Если вызов grantpt() является обязательным, то создает дочерний процесс, который выполняет программу, устанавливающую идентификатор администратора. Эта программа обычно называется pt_chown и совершает следующие операции со вторичным устройством псевдотерминала:
• изменяет владельца вторичного устройства в соответствии с действующим пользовательским идентификатором вызывающего процесса;
• переносит вторичное устройство в группу tty;
• изменяет права доступа ко вторичному устройству таким образом, чтобы владелец мог выполнять запись и чтение, а группа имела только право на запись.
Причина изменения группы терминала на tty и предоставления ей права на запись заключается в том, что программы wall(1) и write(1) устанавливают идентификаторы группы и сами входят в tty.
В Linux вторичное устройство псевдотерминала автоматически конфигурируется вышеописанным образом, поэтому вызова grantpt() не требуется (но его все равно нужно сделать).
Вызов grantpt() может создать дочерний процесс, вследствие чего в стандарте SUSv3 сказано, что его поведение является неопределенным, если в вызывающей программе установлен обработчик для сигнала SIGCHLD.
60.2.3. Разблокировка вторичного устройства: вызов unlockpt()
Функция unlockpt() убирает внутреннюю блокировку вторичного устройства, связанного с первичным концом псевдотерминала, на который ссылается файловый дескриптор mfd. Такой механизм блокировки позволяет вызывающему процессу выполнять любые действия, необходимые для инициализации вторичного устройства (например, вызов grantpt()) до того, как его сможет открыть какой-то другой процесс.
#define _XOPEN_SOURCE 500
#include
int unlockpt(int
Возвращает 0 при успешном завершении или -1 при ошибке
Попытка открыть вторичное устройство псевдотерминала до его разблокировки с помощью вызова unlockpt() завершается ошибкой EIO.
60.2.4. Получение имени вторичного устройства: вызов ptsname()
Функция ptsname() возвращает имя вторичного устройства псевдотерминала, связанного с первичным концом псевдотерминала, на который ссылается файловый дескриптор mfd.
В Linux (как и в большинстве других реализаций) вызов ptsname() возвращает имя в виде /dev/pts/nn, где nn — уникальный номер вторичного устройства псевдотерминала.