Linux программирование в примерах
Шрифт:
Строки 319–322 сохраняют номер индекса и имя в
struct lbuf
. Если ep->lnum
возвращается из gstat
установленным в -1, это означает, что операция stat
с файлом завершилась неудачей. Наконец, строка 324 закрывает каталог. Следующая функция,
gstat
(строки 327–398), является центральной функцией для получения и сохранения сведений о файле.
327 struct lbuf * /* struct lbuf *gstat(char *file, int argfl) */
328 gstat(file, argfl)
329 char *file;
330 {
331 extern char *malloc;
332 struct stat statb;
333 register struct lbuf *rep;
334 static int nomocore;
335
336 if (nomocore) /*
Ранее была нехватка памяти */
337 return(NULL);
338 rep = (struct lbuf*)malloc(sizeof(struct lbuf));
339 if (rep==NULL) {
340 fprintf(stderr, "ls: out of memory\n");
341 nomocore = 1;
342 return(NULL);
343 }
344 if (lastp >= &flist[NFILES]) { /* Проверить, не дано ли слишком много файлов */
345 static int msg;
346 lastp--;
347 if (msg==0) {
348 fprintf(stderr, "ls: too many files\n");
349 msg++;
350 }
351 }
352 *lastp++ = rep; /* Заполнить сведения */
353 rep->lflags = 0;
354 rep->lnum = 0;
355 rep->ltype = '-'; /* Тип файла по умолчанию */
Статическая переменная
nomocore
[важно] указывает, что malloc
при предыдущем вызове завершилась неудачей. Поскольку она статическая, она автоматически инициализируется 0 (т.е. false
). Если на входе она равна true
, gstat
просто возвращает NULL
. В противном случае, если malloc
завершается неудачей, ls
выводит сообщение об ошибке, устанавливает в nomocore
true
и возвращает NULL
(строки 334–343). Строки 344–351 гарантируют, что в массиве
flist
все еще остается место. Если нет, ls
выдает сообщение (но лишь однажды; заметьте использование статической переменной msg
), а затем повторно использует последний слот flist
. Строка 352 заставляет слот
lastp
указывать на новую struct lbuf
(rep
). Это также обновляет lastp
, который используется для сортировки в main
(строки 142 и 152). Строки 353–355 устанавливают значения по умолчанию для полей флагов, номеров индексов и типов в struct lbuf
.
356 if (argfl || statreq) {
357 if (stat(file, &statb)<0) { /* stat завершилась неудачей */
358 printf("%s not found\n", file);
359 statb.st_ino = -1;
360 statb.st_size = 0;
361 statb.st_mode = 0;
362 if (argfl) {
363 lastp--;
364 return(0);
365 }
366 }
367 rep->lnum = statb.st_ino; /* stat OK,
копировать сведения */
368 rep->lsize = statb.st_size;
369 switch(statb.st_mode & S_IFMT) {
370
371 case S_IFDIR:
372 rep->ltype = 'd';
373 break;
374
375 case S_IFBLK:
376 rep->ltype = 'b';
377 rep->lsize = statb.st_rdev;
378 break;
379
380 case S_IFCHR:
381 rep->ltype = 'c';
382 rep->lsize = statb.st_rdfev;
383 break;
384 }
385 rep->lflags = statb.st_mode & ~S_IFMT;
386 rep->luid = statb.st_uid;
387 rep->lgid = statb.st_gid;
388 rep->lnl = statb.st_nlink;
389 if (uflg)
390 rep->lmtime = statb.st_atime;
391 else if (cflg)
392 rep->lmtime = statb.st_ctime;
393 else
394 rep->lmtime = statb.st_mtime;
395 tblocks += nblock(statb.st_size);
396 }
397 return(rep);
398 }
Строки 356–396 обрабатывают вызов
stat
. Если это аргумент командной строки или если statreq
установлен в true
благодаря опции, код заполняет struct lbuf
следующим образом: • Строки 357–366: вызывают
stat
, при ее неудаче выводится сообщение об ошибке с установкой соответствующих значений, затем возвращается NULL
(выраженный в виде 0). • Строки 367–368: устанавливают в struct stat поля номера индекса и размера, если вызов
stat
был успешным. • Строки 369–384: обрабатывают особые случаи каталогов, блочных и символьных устройств. Во всех случаях код обновляет поле
ltype
. Для устройств значение lsize
замещается значением st_rdev
. • Строки 385–388. заполняются поля
lflags
, luid
, lgid
и lnl
из соответствующих полей в struct stat
. Строка 385 удаляет биты типа файла, оставляя 12 битов прав доступа (на чтение/запись/исполнение для владельца/группы/остальных, а также setuid, setgid и save-text).
Поделиться с друзьями: