У каждого псевдотерминала есть запись в каталоге /dev/pts
:
% ls -l /dev/pts/4
crw--w---- 1 samuel tty 136, 4 Mar 8 02:56 /dev/pts/4
Обратите внимание на то, что псевдотерминал — это символьное устройство, а его владельцем является владелец процесса, для которого был создан псевдотерминал.
С псевдотерминалом можно обмениваться данными. При чтении перехватываются символы, вводимые с клавиатуры, а при записи данные отображаются в окне терминала.
Попробуйте открыть новое терминальное окно и определить номер псевдотерминала, выполнив команду ps -o pid,tty,cmd
. Теперь откройте другое окно и направьте какие-то данные на псевдотерминал. Например, если его номер 7, введите такую команду:
% echo "Hello, other window!" > /dev/pts/7
Заданная строка отобразится в первом окне. Когда терминальное окно будет закрыто, запись 7 исчезнет из каталога /dev/pts
.
Если ввести команду ps
в терминальном окне, работающем в текстовом режиме, окажется, что ему соответствует обычное терминальное устройство, а не псевдотерминал:
% ps -о pid,tty,cmd
PID TTY CMD
29325 tty1 -bash
29353 tty1 ps -o pid,tty,cmd
6.7. Функция ioctl()
Системный вызов ioctl()
— это универсальное средство управления аппаратными устройствами. Первым аргументом функции является дескриптор файла того устройства, которым требуется управлять. Второй аргумент — это код запроса, обозначающего выполняемую операцию. Разным устройствам соответствуют разные запросы. В зависимости от запроса функции ioctl()
могут потребоваться дополнительные аргументы.
Многие коды запросов перечислены на man
-странице ioctl_list
. При работе с функцией ioctl()
нужно хорошо понимать, как работает драйвер соответствующего устройства. В принципе, эти вопросы выходят за рамки нашей книги, но все же приведем небольшой пример.
#include
#include
#include
#include
#include
#include
int main(int argc, char* argv[]) {
/* Открытие файла устройства, указанного в командной строке. */
int fd = open(argv[1], O_RDONLY);
/* Извлечение компакт-диска из дисковода. */
ioctl(fd, CDROMEJECT);
/* Закрытие файла. */
close(fd);
return 0;
}
В листинге 6.2 представлена короткая программа, которая запрашивает извлечение компакт-диска из дисковода CD-ROM. Программа принимает единственный аргумент командной строки: имя дисковода CD-ROM. Программа открывает файл устройства и вызывает функцию ioctl()
с кодом запроса CDROMEJECT
. Этот код определен в файле
и служит устройству указанием извлечь компакт-диск из дисковода.
Например, если в системе имеется IDE-дисковод CD-ROM, подключенный в качестве главного устройства к дополнительному IDE-контроллеру, соответствующий файл устройства будет называться /dev/hdc
. Тогда компакт-диск извлекается из дисковода с помощью такой команды:
% ./cdrom-eject /dev/hdc
Глава 7
Файловая система /proc
Попробуйте запустить команду mount
без аргументов — она выдаст список файловых систем, смонтированных в настоящий момент. Среди прочих строк будет и такая:
none on /proc type proc (rw)
Она указывает на специальную файловую систему /proc
. Поле none
говорит о том, что эта система не связана с аппаратным устройством, например жестким диском. Она является своего рода "окном" в ядро Linux. Файлам в системе /proc
не соответствуют реальные файлы на физическом устройстве. Это особые объекты, которые ведут себя подобно файлам, открывал доступ к параметрам, служебным структурам и статистической информации ядра. "Содержимое" таких файлов генерируется ядром динамически в процессе чтения из файла. Осуществляя запись в некоторые файлы, можно менять конфигурацию работающего ядра системы. Рассмотрим пример:
% ls -l /proc/version
-r--r--r-- 1 root root 0 Jan 17 18:09 /proc/version
Обратите внимание на то, что размер файла равен нулю. Поскольку содержимое файла создается ядром "на лету", понятие размера файла здесь неприменимо. Соответственно время модификации файла равно времени запуска команды.