Прежде чем добавлять новую структуру, функция pututxline() сначала делает вызов getutxid() для обнаружения следующей записи, которая может быть заменена. Если такая запись существует, то она заменяется; в противном случае в конец файла добавляется новая структура. Часто перед вызовом pututxline() приложения используют одну из функций вида getutx*(), чтобы сделать текущей подходящую запись — отвечающую критериям структуры utmpx, на которую указывает аргумент ut вызова getutxid(). Если функция pututxline() обнаруживает, что данный шаг уже выполнен, то getutxid() не вызывается.
Если функция pututxline() делает внутри себя вызов getutxid(), то это не затрагивает статический участок памяти, в который функции getutx*() возвращают структуру utmpx. Данный аспект является одним из требований стандарта SUSv3.
Чтобы обновить файл wtmp, нужно просто открыть его и добавить новую запись. Поскольку это стандартная операция, в библиотеке glibc она реализована в виде функции updwtmpx().
Функция updwtmpx() добавляет запись типа utmpx, на которую указывает ut, в файл, заданный в аргументе wtmpx_file.
#define _GNU_SOURCE
#include
void updwtmpx(char *
Эта функция не входит в стандарт SUSv3 и поддерживается всего несколькими другими реализациями UNIX. В других системах доступны похожие функции — ogin(3), logout(3) и logwtmp(3); они являются частью библиотеки glibc и описаны на соответствующих справочных страницах. Если же они отсутствуют, вам придется самостоятельно написать их аналоги (они имеют несложную реализацию).
Листинг 40.3 обновляет файлы utmp и wtmp, используя функции, описанные в данном разделе. Эта программа вносит необходимые изменения, чтобы впустить в систему пользователя, указанного в командной строке, и затем, после нескольких секунд ожидания, выполняет выход из системы. Обычно такие действия связаны с созданием и завершением сессии входа в систему. Эта программа задействует вызов ttyname() (описанный в разделе 58.10) для получения имени терминального устройства, связанного с файловым дескриптором.
Работа программы из листинга 40.3 продемонстрирована на примере сессии командной оболочки, приведенной ниже. Мы повышаем привилегии, чтобы иметь возможность обновлять учетные файлы, связанные со входом в систему, после чего используем нашу программу для создания записи для пользователя mtk:
$ su
Password:
# ./utmpx_login mtk
Creating login entries in utmp and wtmp
using pid 1471, line pts/7, id /7
[1]+ Stopped./utmpx_login mtk
Пока программа utmpx_login находилась в состоянии ожидания, мы нажали Ctrl+Z, чтобы приостановить ее выполнение и переключить ее в фоновый режим. Теперь воспользуемся программой из листинга 40.2 для просмотра содержимого файла utmp:
# ./dump_utmpx /var/run/utmp
user type PID line id host date/time
cecilia USER_PR 249 tty1 1 Fri Feb 1 21:39:07 2008
mtk USER_PR 1471 pts/7 /7 Fri Feb 1 22:08:06 2008
# who
cecilia tty1 Feb 1 21:39
mtk pts/7 Feb 1 22:08
Мы воспользовались командой who(1) для демонстрации того, что ее вывод берется из файла utmp. Теперь просмотрим с помощью нашей программы содержимое файла wtmp:
# ./dump_utmpx /var/log/wtmp
user type PID line id host date/time
cecilia USER_PR 249 tty1 1 Fri Feb 1 21:39:07 2008
mtk USER_PR 1471 pts/7 /7 Fri Feb 1 22:08:06 2008
# last mtk
mtk pts/7 Fri Feb 1 22:08 still logged in
Команда last(1) применена для иллюстрации того, что ее вывод основывается на файле wtmp (чтобы вывод программ dump_utmpx и last был более лаконичным, здесь не приводятся строки, которые не относятся к теме обсуждения).
Теперь воспользуемся командой fg для возврата программы utmpx_login в активный режим. Продолжив работу, utmpx_login добавляет в файлы utmp и wtmp записи о выходе из системы.
# fg
./utmpx_login mtk
Creating logout entries in utmp and wtmp
Теперь еще раз просмотрим содержимое файла utmp. Как видите, внесенная нами запись была заменена:
# ./dump_utmpx /var/run/utmp
user type PID line id host date/time
cecilia USER_PR 249 tty1 1 Fri Feb 1 21:39:07 2008
DEAD_PR 1471 pts/7 /7 Fri Feb 1 22:09:09 2008
# who
Cecilia tty1 Feb 1 21:39
Последняя строчка этого вывода свидетельствует о том, что команда who проигнорировала запись DEAD_PROCESS.
Если вывести файл wtmp, мы увидим, что и здесь произошла замена нашей записи:
# ./dump_utmpx /var/log/wtmp
user type PID line id host date/time
cecilia USER_PR 249 tty1 1 Fri Feb 1 21:39:07 2008
mtk USER_PR 1471 pts/7 /7 Fri Feb 1 22:08:06 2008
DEAD_PR 1471 pts/7 /7 Fri Feb 1 22:09:09 2008
# last mtk
mtk pts/7 Fri Feb 1 22:08–22:09 (00:01)
В последней строке вышеприведенного вывода наглядно показано, как команда last ищет в файле wtmp записи о входе в систему и выходе из нее, чтобы вернуть время начала и завершения сессии пользователя.