Выбрать главу

Пример программы

В листинге 4.3 показывается использование вызова lseek() в сочетании с read() и write(). Первый аргумент, передаваемый в командной строке для запуска этой программы, является именем открываемого файла. В остальных аргументах указываются операции ввода-вывода, выполняемые в отношении файла. Название каждой из этих операций состоит из буквы, за которой следует связанное с ней значение (без разделительного пробела):

• soffset — установка байтового смещения offset с начала файла;

• rlength — чтение length байтов из файла, начиная с текущего файлового смещения, и вывод их в текстовой форме;

• Rlength — чтение length байтов из файла, начиная с текущего файлового смещения и вывод их в виде шестнадцатеричных чисел;

• wstr — запись строки символов, указанной в str, начиная с позиции текущего файлового смещения.

Листинг 4.3. Демонстрация работы read(), write() и lseek()

fileio/seek_io.c

#include <sys/stat.h>

#include <fcntl.h>

#include <ctype.h>

#include "tlpi_hdr.h"

int

main(int argc, char *argv[])

{

size_t len;

off_t offset;

int fd, ap, j;

char *buf;

ssize_t numRead, numWritten;

if (argc < 3 || strcmp(argv[1], "-help") == 0)

usageErr("%s file {r<length>|R<length>|w<string>|s<offset>}…\n",

argv[0]);

fd = open(argv[1], O_RDWR | O_CREAT,

S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP |

S_IROTH | S_IWOTH); /* rw-rw-rw- */

if (fd == -1)

errExit("open");

for (ap = 2; ap < argc; ap++) {

switch (argv[ap][0]) {

case 'r': /* Вывод байтов с позиции текущего смещения в виде текста */

case 'R': /* Вывод байтов с позиции текущего смещения в виде hex-чисел */

len = getLong(&argv[ap][1], GN_ANY_BASE, argv[ap]);

buf = malloc(len);

if (buf == NULL)

errExit("malloc");

numRead = read(fd, buf, len);

if (numRead == -1)

errExit("read");

if (numRead == 0) {

printf("%s: end-of-file\n", argv[ap]);

} else {

printf("%s: ", argv[ap]);

for (j = 0; j < numRead; j++) {

if (argv[ap][0] == 'r')

printf("%c", isprint((unsigned char) buf[j])?

buf[j]: '?');

else

printf("%02x", (unsigned int) buf[j]);

}

printf("\n");

}

free(buf);

break;

case 'w': /* Запись строки, начиная с позиции текущего смещения */

numWritten = write(fd, &argv[ap][1], strlen(&argv[ap][1]));

if (numWritten == -1)

errExit("write");

printf("%s: wrote %ld bytes\n", argv[ap], (long) numWritten);

break;

case 's': /* Изменение файлового смещения */

offset = getLong(&argv[ap][1], GN_ANY_BASE, argv[ap]);

if (lseek(fd, offset, SEEK_SET) == -1)

errExit("lseek");

printf("%s: seek succeeded\n", argv[ap]);

break;

default:

cmdLineErr("Argument must start with [rRws]: %s\n", argv[ap]);

}

}

exit(EXIT_SUCCESS);

}

fileio/seek_io.c

Использование программы, приведенной в листинге 4.3, показано в следующих сессиях командной оболочки, с демонстрацией того, что произойдет при попытке чтения байтов из файловой дыры:

$ touch tfile Создание нового, пустого файла5

$ ./seek_io tfile s100000 wabc Установка смещения 100000, запись “abc”

s100000: seek succeeded

wabc: wrote 3 bytes

$ ls — l tfile Проверка размера файла

— rw-r-r- 1 mtk users 100003 Feb 10 10:35 tfile

$ ./seek_io tfile s10000 R5 Установка смещения 10000, чтение пяти байт из дыры

s10000: seek succeeded

R5: 00 00 00 00 00 В байтах дыры содержится 0

4.8. Операции, не вписывающиеся в модель универсального ввода-вывода: ioctl()

Системный вызов ioctl() — механизм общего назначения для выполнения операций в отношении файлов и устройств, выходящих за пределы универсальной модели ввода-вывода, рассмотренной ранее в данной главе.

#include <sys/ioctl.h>

int ioctl(int fd, int request… /* argp */);

Возвращаемое при успешном завершении значение зависит от request или при ошибке равно –1

Аргумент fd содержит дескриптор открываемого файла, представленного устройством или файлом, в отношении которого выполняется управляющая операция (указана в аргументе request). Как показывает стандартная для языка C запись в виде многоточия (…), третий аргумент для ioctl(), обозначенный как argp, может быть любого типа. Аргумент request позволяет ioctl() определить, какого типа значение следует ожидать в argp. Обычно argp представляет собой указатель либо на целое число, либо на структуру. В некоторых случаях этот аргумент не применяется.

Использование ioctl() будет показано в следующих главах (к примеру, в разделе 15.5).

Единственная спецификация, имеющаяся в SUSv3 для ioctl(), регламентирует операции по управлению STREAMS-устройствами. (Среда STREAMS относится к особенностям System V, не поддерживаемым основной ветвью ядра Linux, хотя было разработано несколько реализаций в виде дополнений.) Ни одна из других рассматриваемых в книге операций ioctl() в SUSv3 не регламентирована. Но вызов ioctl() был частью системы UNIX с самых ранних версий, вследствие чего несколько операций ioctl() предоставляются во многих других реализациях UNIX. По мере рассмотрения каждой операции ioctl() будут обсуждаться и вопросы портируемости.