Чтение онлайн

ЖАНРЫ

Linux программирование в примерах

Роббинс Арнольд

Шрифт:

8.5. Обход дерева файлов: GNU

du

GNU версия

du
в GNU Coreutils использует
nftw
для обхода одной или более иерархии файлов, собирая и выводя сведения, касающиеся количества используемого дискового пространства. У нее большое число опций, которые управляют ее поведением но отношению к символическим ссылкам, форматом вывода чисел и т.д. Это делает разбор кода труднее, чем могло бы быть при более простой версии. (Однако, мы не собираемся позволить этому остановить нас.) Вот сводка опций
du
, которые
вскоре будут полезны, когда мы рассмотрим код.

$ du --help

Usage: du [OPTION]... [FILE]...

Дает сводку использования диска для каждого FILE,

рекурсивно для каталогов.

Обязательные для длинных опций аргументы являются обязательными

также и для коротких опций.

– a, --all записать число всех файлов, а не только

каталогов

– -apparent-size вывести видимые размеры, а не использование

диска; хотя видимый размер обычно меньше, он

может быть и больше из-за дыр в файлах,

внутренней фрагментации, косвенных блоков и т.п.

– В, --block-size=SIZE использовать блоки размером SIZE байтов

– b, --bytes эквивалентно '--apparent-size --block-size=1'

– с, --total выводит итоговую сумму

– D, --dereference-args разыменовывать FILE, которые являются

символическими ссылками

– h, --human-readable вывести размеры в удобном для восприятия

формате (например, 1K 234М 2G)

– Н, --si так же, но использовать степени 1000, не 1024

– k подобно --block-size=1K

– l, --count-links считать размеры несколько раз при прямых

ссылках

– L, --dereference разыменовывать все символические ссылки

– S, --separate-dirs не включать размер подкаталогов

– s, --summarize отобразить для каждого аргумента лишь итоги

– х, --one-file-system пропускать каталоги на различных файловых

системах

– X --exclude- исключить файлы, подходящие под любой

FILE from=FILE образец в FILE

– -exclude=PATTERN исключить файлы, соответствующие PATTERN

– -max-depth=N вывести итог для каталога (или файла, с --all)

только если он находится на N или менее уровней

глубже аргумента командной строки;

– -max-depth=0 то же самое, что и --summarize

– -help отобразить экран справки и выйти

– -version вывести сведения о версии и выйти

SIZE может быть (или
может быть целым, за которым

может следовать это) одним из

следующих: kB 1000, K 1024, MB 1 000 000, M 1 048 576 и т.д.

для G, T, Р, E, Z, Y.

Чтобы еще больше усложнить дело,

du
использует частную версию
nftw
, у которой есть несколько расширений. Сначала имеются дополнительные значения флагов для функции обратного вызова:

FTW_DCHP

Это значение означает, что

nftw
не может выполнять '
chdir("..")
'.

FTW_DCH

Это значение означает, что

nftw
не может использовать
chdir
для перехода в сам каталог.

FTW_DPRE

Частная

nftw
вызывает для каталогов функцию обратного вызова дважды. Это значение используется при первой встрече с каталогом. После обработки всех нижележащих объектов каталога используется стандартное значение
FTW_DP
.

Частная

nftw
добавляет также в
struct FTW
новый член,
int skip
. Если текущий объект является каталогом и функция обратного вызова устанавливает в поле
skip
ненулевое значение,
nftw
не будет больше обрабатывать этот каталог. (Функция обратного вызова должна установить
skip
таким образом, когда
flag
равен
FTW_DPRE
; делать это для
FTW_DP
слишком поздно.)

С этим объяснением за поясом, вот функция

process_file
из
du.c
. Номера строк приведены относительно начала функции:

1 /* Эта функция вызывается один раз для каждого объекта файловой

2 системы, с которой сталкивается nftw. nftw осуществляет сначала

3 поиск вглубь. Эта функция знает это и собирает итоги для каталогов

4 на основе изменений в глубине текущего элемента. */

5

6 static int

7 process_file(const char *file, const struct stat *sb,

8 int file_type, struct FTW *info)

9 {

10 uintmax_t size;

11 uintmax_t size_to_print;

12 static int first_call = 1;

13 static size_t prev_level;

14 static size_t n_alloc;

15 static uintmax_t *sum_ent;

16 static uintmax_t *sum_subdir;

17 int print = 1;

18

19 /* Всегда определяйте info->skip перед возвратом. */

Поделиться с друзьями: