В главе 5 рассматриваемые здесь вопросы будут расширены дополнительными сведениями, касающимися файлового ввода-вывода. Еще одна особенность файлового ввода-вывода — буферизация — настолько сложна, что заслуживает отдельной главы. Буферизация ввода-вывода в ядре и с помощью средств библиотеки stdio будет описана в главе 13.
Все системные вызовы для выполнения ввода-вывода совершаются в отношении открытых файлов с использованием
Обычно от большинства программ ожидается возможность использования трех стандартных дескрипторов файлов, перечисленных в табл. 4.1. Эти три дескриптора открыты оболочкой от имени программы еще до запуска самой программы. Точнее говоря, программа наследует у оболочки копии дескрипторов файлов, а оболочка обычно работает с этими всегда открытыми тремя дескрипторами файлов. (В интерактивной оболочке эти три дескриптора обычно ссылаются на терминал, на котором запущена оболочка.) Если в командной строке указано перенаправление ввода-вывода, то оболочка перед запуском программы обеспечивает соответствующее изменение дескрипторов файлов.
Таблица 4.1. Стандартные дескрипторы файлов
Дескриптор файла — Назначение — Имя согласно POSIX — Поток stdio
0 — Стандартный ввод — STDIN_FILENO — stdin
1 — Стандартный вывод — STDOUT_FILENO — stdout
2 — Стандартная ошибка — STDERR_FILENO — stderr
При ссылке на эти дескрипторы файлов в программе можно использовать либо номера (0, 1 или 2), либо, что предпочтительнее, стандартные имена POSIX, определенные в файле
Хотя переменные stdin, stdout и stderr изначально ссылаются на стандартные ввод, вывод и ошибку процесса, с помощью библиотечной функции freopen() их можно изменить для ссылки на любой файл. В качестве части своей работы freopen() способен изменить дескриптор файла на основе вновь открытого потока. Иными словами, после вызова freopen() в отношении, к примеру, stdout, нельзя с полной уверенностью предполагать, что относящийся к нему дескриптор файла по-прежнему имеет значение 1.
Рассмотрим следующие четыре системных вызова, которые являются ключевыми для выполнения файлового ввода-вывода (языки программирования и программные пакеты обычно используют их исключительно опосредованно, через библиотеки ввода-вывода).
• fd = open(pathname, flags, mode) — открытие файла, идентифицированного по путевому имени — pathname, с возвращением дескриптора файла, который используется для обращения к открытому файлу в последующих вызовах. Если файл не существует, вызов open() может его создать в зависимости от установки битовой маски аргумента флагов — flags. В аргументе флагов также указывается, с какой целью открывается файл: для чтения, для записи или для проведения обеих операций. Аргумент mode (режим), определяет права доступа, которые будут накладываться на файл, если он создается этим вызовом. Если вызов open() не будет использоваться для создания файла, этот аргумент игнорируется и может быть опущен.
• numread = read(fd, buffer, count) — считывание не более указанного в count количества байтов из открытого файла, ссылка на который дана в fd, и сохранение их в буфере buffer. При вызове read() возвращается количество фактически считанных байтов. Если данные не могут быть считаны (то есть встретился конец файла), read() возвращает 0.
• numwritten = write(fd, buffer, count) — запись из буфера байтов, количество которых указано в count, в открытый файл, ссылка на который дана в fd. При вызове write() возвращается количество фактически записанных байтов, которое может быть меньше значения, указанного в count.
• status = close(fd) — вызывается после завершения ввода-вывода с целью высвобождения дескриптора файла fd и связанных с ним ресурсов ядра.
Перед тем как подробно разбирать эти системные вызовы, посмотрим на небольшую демонстрацию их использования в листинге 4.1. Эта программа является простой версией команды cp(1). Она копирует содержимое существующего файла, чье имя указано в первом аргументе командной строки, в новый файл с именем, указанным во втором аргументе командной строки.
Программой, показанной в листинге 4.1, можно воспользоваться следующим образом:
$ ./copy oldfile newfile
Листинг 4.1. Использование системных вызовов ввода-вывода
fileio/copy.c
#include
#include
#include "tlpi_hdr.h"
#ifndef BUF_SIZE /* Позволяет "cc — D" перекрыть определение */
#define BUF_SIZE 1024
#endif
int
main(int argc, char *argv[])
{
int inputFd, outputFd, openFlags;
mode_t filePerms;
ssize_t numRead;
char buf[BUF_SIZE];