$ ./demo_SIGWINCH
Caught SIGWINCH, new window size: 35 rows * 80 columns
Caught SIGWINCH, new window size: 35 rows * 73 columns
Caught SIGWINCH, new window size: 22 rows * 73 columns
Листинг 58.5. Мониторинг изменений размера окна терминала
tty/demo_SIGWINCH.c
#include
#include
#include
#include "tlpi_hdr.h"
static void
sigwinchHandler(int sig)
{
}
int
main(int argc, char *argv[])
{
struct winsize ws;
struct sigaction sa;
sigemptyset(&sa.sa_mask);
sa.sa_flags = 0;
sa.sa_handler = sigwinchHandler;
if (sigaction(SIGWINCH, &sa, NULL) == -1)
errExit("sigaction");
for (;;) {
pause(); /* Ждем сигнала SIGWINCH */
if (ioctl(STDIN_FILENO, TIOCGWINSZ, &ws) == -1)
errExit("ioctl");
printf("Caught SIGWINCH, new window size: "
"%d rows * %d columns\n", ws.ws_row, ws.ws_col);
}
}
tty/demo_SIGWINCH.c
Можно также изменить представление драйвера терминала о размере окна, передав вызову ioctl() с флагом TIOCSWINSZ инициализированную структуру winsize:
ws.ws_row = 40;
ws.ws_col = 100;
if (ioctl(fd, TIOCSWINSZ, &ws) == -1)
errExit("ioctl");
Если новые значения в структуре winsize отличаются от текущего представления драйвера терминала о размере окна, происходят две вещи:
• структуры данных драйвера терминала обновляются с помощью новых значений, указанных в аргументе ws;
• активной группе процессов терминала передается сигнал SIGWINCH.
Стоит отметить, что эти события сами по себе не могут изменить реальный размер отображаемого окна, который контролируется кодом за пределами ядра (например, оконным менеджером или эмулятором терминала).
Большинство UNIX-систем предоставляют доступ к размеру окна терминала, используя операции ioctl(), описанные в данном разделе, хотя это не предусмотрено стандартом SUSv3.
В разделе 34.4 мы познакомились с функцией ctermid(), возвращающей имя управляющего терминала, в котором запущен процесс (в системах UNIX это обычно /dev/tty). Функции, описанные в данном разделе, тоже могут пригодиться для идентификации терминала.
Функция isatty() позволяет определить, связан ли файловый дескриптор fd с каким-либо терминалом (а не с файлом другого типа).
#include
int isatty(int
Возвращает true (1), если дескриптор fd связан с терминалом, или false (0), если нет
Функция isatty() может пригодиться в редакторах и других программах, работающих с экраном, которым нужно знать, связаны ли их ввод и вывод с терминалом.
На основе файлового дескриптора функция ttyname() возвращает имя соответствующего терминального устройства.
#include
char *ttyname(int
Возвращает указатель на (статически выделенную) строку с именем терминала или NULL при ошибке
Для нахождения имени терминала ttyname() используются функции opendir() и readdir(), описанные в разделе 18.8. Они перебирают каталоги в поисках идентификатора (поля st_rdev в структуре stat), принадлежащего устройству, на которое ссылается файловый дескриптор fd. Записи о терминальных устройствах обычно находятся в двух каталогах: /dev и /dev/pts. В первом перечислены виртуальные консоли (например, /dev/tty1) и псевдотерминалы систем BSD, а во втором — псевдотерминалы вторичных устройств (в стиле System V). Мы вернемся к псевдотерминалам в главе 60.
У функции ttyname() есть реентерабельный аналог под названием ttyname_r().
Команда tty(1), отображающая имя терминала, с которым связан ее стандартный ввод, является консольной альтернативой функции ttyname().
Когда-то терминалы в UNIX-системах представляли собой настоящие аппаратные устройства, подключаемые к компьютерам через последовательный порт. Ранние разновидности терминалов не были стандартизованы — то есть управляющие последовательности, требуемые для работы с терминалом, зависели от его производителя. В современных рабочих станциях вместо терминалов используются графические мониторы под управлением X Window System. Однако возможность управления терминалами все еще требуется при работе с виртуальными устройствами, такими как виртуальные консоли и эмуляторы терминалов (основанных на псевдотерминалах), а также с реальными устройствами, подключенными через последовательный порт.
Параметры терминала (за исключением размеров его окна) хранятся в структуре типа termios, состоящей из четырех битовых масок с различными атрибутами терминала и массива, который определяет разные управляющие последовательности, интерпретируемые драйвером терминала. Для получения и изменения этих параметров предусмотрены функции tcgetattr() и tcsetattr().