#define XOPEN_SOURCE 500
#include
size_tpread(int fd, void * buf, size_t count, off_t offset);
size_t pwrite(int fd, void * buf, size_t count, off_t offset);
#endif
Это выглядит подобно прототипам read()
и write()
с четвертым параметром, offset
. offset определяет, с какой точки файла следует читать, а в какую — записывать. Как и их "тезки", эти функции возвращают количество переданных байтов. Ниже приведена версия pread()
, реализованная с помощью read()
и lseek()
, что облегчает понимание ее функции [99] Эта эмулированная версия в большинстве случаев ведет себя корректно, но действует не так, как фактический системный вызов, если сигналы принимаются во время его выполнения.
.
int pread (int fd, void * data, int size, int offset) {
int oldOffset;
int rc;
int oldErrno;
/* переместить указатель файла в новое расположение */
oldOffset = lseek(fd, SEEK_SET, offset);
if (oldOffset < 0) return -1;
rc = read(fd, data, size);
/* восстановить указатель файла, предварительно сохранив errno */
oldErrno = errno;
lseek(fd, SEEK_SET, oldOffset);
errno = oldErrno;
return rc;
}
Глава 14
Операции с каталогами
Как и во многих других операционных системах, для организации файлов в Linux используются каталоги. Каталоги (представляющие собой особые типы файлов, которые содержат списки имен файлов) состоят из файлов, а также других каталогов, образуя иерархию файлов. Все системы Linux содержат корневой каталог, известный как /
, через который (прямо или непрямо) можно получить доступ ко всем файлам системы.
14.1. Текущий рабочий каталог
14.1.1. Поиск текущего рабочего каталога
Функция getcwd()
позволяет процессу найти имя своего текущего каталога относительно корневого каталога системы.
#include
char * getcwd(char * buf, size_t size);
Первый параметр, buf
, указывает на буфер, хранящий путь к текущему каталогу. Если длина текущего пути превышает size - 1
байт (-1 позволяет пути завершаться символом '\0'
), функция возвращает ошибку ЕRANGE
. Если вызов удается, возвращается buf
; в случае ошибки возвращается NULL
. Несмотря на то что в большинстве современных оболочек поддерживается переменная окружения PWD
, хранящая путь в текущий каталог, ее значение необязательно равняется значению, возвращаемому getcwd()
. PWD
часто содержит элементы путей, являющиеся символическими ссылками на другие каталоги, но getcwd()
всегда возвращает путь, свободный от символических ссылок.
Если текущий путь неизвестен (например, во время запуска программы), буфер, содержащий текущий каталог, должен быть динамически распределен, поскольку размер текущего пути может быть произвольным. Код, должным образом читающий текущий путь, выглядит так, как показано ниже.
char * buf;
int len = 50;
buf = malloc(len);
while (!getcwd(buf, len) && errno == ERANGE) {
len += 50;
buf = realloc(buf, len);
}
Как и многие другие Unix-подобные системы, Linux предоставляет полезное расширение POSIX-спецификации getcwd()
. Если buf
является NULL
, функция распределяет буфер, размер которого достаточен для содержания текущего пути, с помощью нормального механизма malloc()
. Несмотря на то что вызывающий код должен позаботиться о надлежащем освобождении памяти, используемой результатом, это расширение обеспечивает лучшую очистку, нежели цикл, как показано в предыдущем примере.
Функция BSD по имени getwd()
является наиболее распространенной альтернативой getcwd()
, но ее определенные дефекты привели к разработке getcwd()
.
#include
char * getwd(char * buf);
Как и getcwd()
, getwd()
заполняет buf
текущим путем, хотя функция не имеет представления о размере buf
. getwd()
никогда не записывает в буфер больше, чем PATH_MAX
(определенная в ), что позволяет программам избегать переполнения буферов, но не предоставляет программе механизма поиска правильного пути, если он превышает PATH_MAX
байт [100] Это верно; PATH_MAX не является фактическим пределом. POSIX считает его неопределенным , что обычно является эквивалентом "не используйте его".
. Эта функция поддерживается Linux только для унаследованных приложений и не может использоваться новыми приложениями. Вместо этого применяйте правильную и более переносимую функцию getcwd()
.
Если для пользователей необходимо отображать текущий путь каталога, хорошим решением будет проверка переменной окружения PWD
. Если она установлена, она содержит путь, который применяется пользователем и который может содержать символические ссылки на некоторые элементы пути. Этот путь обычно и должен отображаться приложением по желанию пользователя. Для облегчения задачи библиотека С в Linux предоставляет функцию get_current_dir_name()
, реализуемую следующим образом.
Читать дальше