]> Dogcows Code - chaz/tar/blob - src/list.c
Remove S_IS* ifdefs.
[chaz/tar] / src / list.c
1 /* List a tar archive, with support routines for reading a tar archive.
2 Copyright 1988,92,93,94,96,97,98,1999 Free Software Foundation, Inc.
3 Written by John Gilmore, on 1985-08-26.
4
5 This program is free software; you can redistribute it and/or modify it
6 under the terms of the GNU General Public License as published by the
7 Free Software Foundation; either version 2, or (at your option) any later
8 version.
9
10 This program is distributed in the hope that it will be useful, but
11 WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
13 Public License for more details.
14
15 You should have received a copy of the GNU General Public License along
16 with this program; if not, write to the Free Software Foundation, Inc.,
17 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
18
19 /* Define to non-zero for forcing old ctime() instead of isotime(). */
20 #undef USE_OLD_CTIME
21
22 #include "system.h"
23 #include <quotearg.h>
24
25 #include <time.h>
26
27 #ifndef FNM_LEADING_DIR
28 # include <fnmatch.h>
29 #endif
30
31 #include "common.h"
32
33 union block *current_header; /* points to current archive header */
34 struct stat current_stat; /* stat struct corresponding */
35 enum archive_format current_format; /* recognized format */
36
37 static uintmax_t from_oct PARAMS ((const char *, size_t, const char *, uintmax_t));
38
39
40 /*-----------------------------------.
41 | Main loop for reading an archive. |
42 `-----------------------------------*/
43
44 void
45 read_and (void (*do_something) ())
46 {
47 enum read_header status = HEADER_STILL_UNREAD;
48 enum read_header prev_status;
49 char save_typeflag;
50
51 name_gather ();
52 open_archive (ACCESS_READ);
53
54 while (1)
55 {
56 prev_status = status;
57 status = read_header ();
58 switch (status)
59 {
60 case HEADER_STILL_UNREAD:
61 abort ();
62
63 case HEADER_SUCCESS:
64
65 /* Valid header. We should decode next field (mode) first.
66 Ensure incoming names are null terminated. */
67
68 /* FIXME: This is a quick kludge before 1.12 goes out. */
69 current_stat.st_mtime = TIME_FROM_OCT (current_header->header.mtime);
70
71 if (!name_match (current_file_name)
72 || current_stat.st_mtime < newer_mtime_option
73 || excluded_pathname (excluded, current_file_name))
74 {
75 int isextended = 0;
76
77 if (current_header->header.typeflag == GNUTYPE_VOLHDR
78 || current_header->header.typeflag == GNUTYPE_MULTIVOL
79 || current_header->header.typeflag == GNUTYPE_NAMES)
80 {
81 (*do_something) ();
82 continue;
83 }
84 if (show_omitted_dirs_option
85 && current_header->header.typeflag == DIRTYPE)
86 WARN ((0, 0, _("Omitting %s"), current_file_name));
87
88 /* Skip past it in the archive. */
89
90 if (current_header->oldgnu_header.isextended)
91 isextended = 1;
92 save_typeflag = current_header->header.typeflag;
93 set_next_block_after (current_header);
94 if (isextended)
95 {
96 #if 0
97 union block *exhdr;
98
99 while (1)
100 {
101 exhdr = find_next_block ();
102 if (!exhdr->sparse_header.isextended)
103 {
104 set_next_block_after (exhdr);
105 break;
106 }
107 }
108 set_next_block_after (exhdr);
109 #endif
110 skip_extended_headers ();
111 }
112
113 /* Skip to the next header on the archive. */
114
115 if (save_typeflag != DIRTYPE)
116 skip_file (current_stat.st_size);
117 continue;
118 }
119
120 (*do_something) ();
121 continue;
122
123 case HEADER_ZERO_BLOCK:
124 if (block_number_option)
125 {
126 char buf[UINTMAX_STRSIZE_BOUND];
127 fprintf (stdlis, _("block %s: ** Block of NULs **\n"),
128 STRINGIFY_BIGINT (current_block_ordinal (), buf));
129 }
130
131 set_next_block_after (current_header);
132 status = prev_status;
133 if (ignore_zeros_option)
134 continue;
135 break;
136
137 case HEADER_END_OF_FILE:
138 if (block_number_option)
139 {
140 char buf[UINTMAX_STRSIZE_BOUND];
141 fprintf (stdlis, _("block %s: ** End of File **\n"),
142 STRINGIFY_BIGINT (current_block_ordinal (), buf));
143 }
144 break;
145
146 case HEADER_FAILURE:
147 /* If the previous header was good, tell them that we are
148 skipping bad ones. */
149 set_next_block_after (current_header);
150 switch (prev_status)
151 {
152 case HEADER_STILL_UNREAD:
153 WARN ((0, 0, _("Hmm, this doesn't look like a tar archive")));
154 /* Fall through. */
155
156 case HEADER_ZERO_BLOCK:
157 case HEADER_SUCCESS:
158 WARN ((0, 0, _("Skipping to next file header")));
159 break;
160
161 case HEADER_END_OF_FILE:
162 case HEADER_FAILURE:
163 /* We are in the middle of a cascade of errors. */
164 break;
165 }
166 continue;
167 }
168 break;
169 }
170
171 apply_delayed_set_stat ();
172 close_archive ();
173 names_notfound (); /* print names not found */
174 }
175
176 /*---------------------------------------------.
177 | Print a header block, based on tar options. |
178 `---------------------------------------------*/
179
180 void
181 list_archive (void)
182 {
183 int isextended = 0; /* to remember if current_header is extended */
184
185 /* Print the header block. */
186
187 if (verbose_option)
188 {
189 if (verbose_option > 1)
190 decode_header (current_header, &current_stat, &current_format, 0);
191 print_header ();
192 }
193
194 if (incremental_option && current_header->header.typeflag == GNUTYPE_DUMPDIR)
195 {
196 off_t size;
197 size_t written, check;
198 union block *data_block;
199
200 set_next_block_after (current_header);
201 if (multi_volume_option)
202 {
203 assign_string (&save_name, current_file_name);
204 save_totsize = current_stat.st_size;
205 }
206 for (size = current_stat.st_size; size > 0; size -= written)
207 {
208 if (multi_volume_option)
209 save_sizeleft = size;
210 data_block = find_next_block ();
211 if (!data_block)
212 {
213 ERROR ((0, 0, _("EOF in archive file")));
214 break; /* FIXME: What happens, then? */
215 }
216 written = available_space_after (data_block);
217 if (written > size)
218 written = size;
219 errno = 0; /* FIXME: errno should be read-only */
220 check = fwrite (data_block->buffer, sizeof (char), written, stdlis);
221 set_next_block_after ((union block *)
222 (data_block->buffer + written - 1));
223 if (check != written)
224 {
225 ERROR ((0, errno, _("Only wrote %lu of %lu bytes to file %s"),
226 (unsigned long) check,
227 (unsigned long) written, current_file_name));
228 skip_file (size - written);
229 break;
230 }
231 }
232 if (multi_volume_option)
233 assign_string (&save_name, NULL);
234 fputc ('\n', stdlis);
235 fflush (stdlis);
236 return;
237
238 }
239
240 /* Check to see if we have an extended header to skip over also. */
241
242 if (current_header->oldgnu_header.isextended)
243 isextended = 1;
244
245 /* Skip past the header in the archive. */
246
247 set_next_block_after (current_header);
248
249 /* If we needed to skip any extended headers, do so now, by reading
250 extended headers and skipping past them in the archive. */
251
252 if (isextended)
253 {
254 #if 0
255 union block *exhdr;
256
257 while (1)
258 {
259 exhdr = find_next_block ();
260
261 if (!exhdr->sparse_header.isextended)
262 {
263 set_next_block_after (exhdr);
264 break;
265 }
266 set_next_block_after (exhdr);
267 }
268 #endif
269 skip_extended_headers ();
270 }
271
272 if (multi_volume_option)
273 assign_string (&save_name, current_file_name);
274
275 /* Skip to the next header on the archive. */
276
277 skip_file (current_stat.st_size);
278
279 if (multi_volume_option)
280 assign_string (&save_name, NULL);
281 }
282
283 /*-----------------------------------------------------------------------.
284 | Read a block that's supposed to be a header block. Return its address |
285 | in "current_header", and if it is good, the file's size in |
286 | current_stat.st_size. |
287 | |
288 | Return 1 for success, 0 if the checksum is bad, EOF on eof, 2 for a |
289 | block full of zeros (EOF marker). |
290 | |
291 | You must always set_next_block_after(current_header) to skip past the |
292 | header which this routine reads. |
293 `-----------------------------------------------------------------------*/
294
295 /* The standard BSD tar sources create the checksum by adding up the
296 bytes in the header as type char. I think the type char was unsigned
297 on the PDP-11, but it's signed on the Next and Sun. It looks like the
298 sources to BSD tar were never changed to compute the checksum
299 currectly, so both the Sun and Next add the bytes of the header as
300 signed chars. This doesn't cause a problem until you get a file with
301 a name containing characters with the high bit set. So read_header
302 computes two checksums -- signed and unsigned. */
303
304 /* FIXME: The signed checksum computation is broken on machines where char's
305 are unsigned. It's uneasy to handle all cases correctly... */
306
307 enum read_header
308 read_header (void)
309 {
310 size_t i;
311 long unsigned_sum; /* the POSIX one :-) */
312 long signed_sum; /* the Sun one :-( */
313 long recorded_sum;
314 uintmax_t parsed_sum;
315 char *p;
316 union block *header;
317 char **longp;
318 char *bp;
319 union block *data_block;
320 size_t size, written;
321 static char *next_long_name, *next_long_link;
322
323 while (1)
324 {
325 header = find_next_block ();
326 current_header = header;
327 if (!header)
328 return HEADER_END_OF_FILE;
329
330 parsed_sum = from_oct (header->header.chksum,
331 sizeof header->header.chksum,
332 (char *) 0, TYPE_MAXIMUM (long));
333 if (parsed_sum == (uintmax_t) -1)
334 return HEADER_FAILURE;
335
336 recorded_sum = parsed_sum;
337 unsigned_sum = 0;
338 signed_sum = 0;
339 p = header->buffer;
340 for (i = sizeof (*header); i-- != 0;)
341 {
342 /* We can't use unsigned char here because of old compilers,
343 e.g. V7. */
344
345 unsigned_sum += 0xFF & *p;
346 signed_sum += *p++;
347 }
348
349 /* Adjust checksum to count the "chksum" field as blanks. */
350
351 for (i = sizeof (header->header.chksum); i-- != 0;)
352 {
353 unsigned_sum -= 0xFF & header->header.chksum[i];
354 signed_sum -= header->header.chksum[i];
355 }
356 unsigned_sum += ' ' * sizeof header->header.chksum;
357 signed_sum += ' ' * sizeof header->header.chksum;
358
359 if (unsigned_sum == sizeof header->header.chksum * ' ')
360 {
361 /* This is a zeroed block...whole block is 0's except for the
362 blanks we faked for the checksum field. */
363
364 return HEADER_ZERO_BLOCK;
365 }
366
367 if (unsigned_sum != recorded_sum && signed_sum != recorded_sum)
368 return HEADER_FAILURE;
369
370 /* Good block. Decode file size and return. */
371
372 if (header->header.typeflag == LNKTYPE)
373 current_stat.st_size = 0; /* links 0 size on tape */
374 else
375 current_stat.st_size = OFF_FROM_OCT (header->header.size);
376
377 header->header.name[NAME_FIELD_SIZE - 1] = '\0';
378 if (header->header.typeflag == GNUTYPE_LONGNAME
379 || header->header.typeflag == GNUTYPE_LONGLINK)
380 {
381 longp = ((header->header.typeflag == GNUTYPE_LONGNAME)
382 ? &next_long_name
383 : &next_long_link);
384
385 set_next_block_after (header);
386 if (*longp)
387 free (*longp);
388 size = current_stat.st_size;
389 if (size != current_stat.st_size)
390 FATAL_ERROR ((0, 0, _("Memory exhausted")));
391 bp = *longp = (char *) xmalloc (size);
392
393 for (; size > 0; size -= written)
394 {
395 data_block = find_next_block ();
396 if (data_block == NULL)
397 {
398 ERROR ((0, 0, _("Unexpected EOF on archive file")));
399 break;
400 }
401 written = available_space_after (data_block);
402 if (written > size)
403 written = size;
404
405 memcpy (bp, data_block->buffer, written);
406 bp += written;
407 set_next_block_after ((union block *)
408 (data_block->buffer + written - 1));
409 }
410
411 /* Loop! */
412
413 }
414 else
415 {
416 char *name = next_long_name;
417 struct posix_header *h = &current_header->header;
418 char namebuf[sizeof h->prefix + 1 + sizeof h->name + 1];
419
420 if (! name)
421 {
422 /* Accept file names as specified by POSIX.1-1996
423 section 10.1.1. */
424 char *np = namebuf;
425 if (h->prefix[0])
426 {
427 memcpy (np, h->prefix, sizeof h->prefix);
428 np[sizeof h->prefix] = '\0';
429 np += strlen (np);
430 *np++ = '/';
431 }
432 memcpy (np, h->name, sizeof h->name);
433 np[sizeof h->name] = '\0';
434 name = namebuf;
435 }
436
437 assign_string (&current_file_name, name);
438 assign_string (&current_link_name,
439 (next_long_link ? next_long_link
440 : current_header->header.linkname));
441 next_long_link = next_long_name = 0;
442 return HEADER_SUCCESS;
443 }
444 }
445 }
446
447 /*-------------------------------------------------------------------------.
448 | Decode things from a file HEADER block into STAT_INFO, also setting |
449 | *FORMAT_POINTER depending on the header block format. If DO_USER_GROUP, |
450 | decode the user/group information (this is useful for extraction, but |
451 | waste time when merely listing). |
452 | |
453 | read_header() has already decoded the checksum and length, so we don't. |
454 | |
455 | This routine should *not* be called twice for the same block, since the |
456 | two calls might use different DO_USER_GROUP values and thus might end up |
457 | with different uid/gid for the two calls. If anybody wants the uid/gid |
458 | they should decode it first, and other callers should decode it without |
459 | uid/gid before calling a routine, e.g. print_header, that assumes |
460 | decoded data. |
461 `-------------------------------------------------------------------------*/
462
463 void
464 decode_header (union block *header, struct stat *stat_info,
465 enum archive_format *format_pointer, int do_user_group)
466 {
467 enum archive_format format;
468
469 if (strcmp (header->header.magic, TMAGIC) == 0)
470 format = POSIX_FORMAT;
471 else if (strcmp (header->header.magic, OLDGNU_MAGIC) == 0)
472 format = OLDGNU_FORMAT;
473 else
474 format = V7_FORMAT;
475 *format_pointer = format;
476
477 stat_info->st_mode = MODE_FROM_OCT (header->header.mode);
478 stat_info->st_mtime = TIME_FROM_OCT (header->header.mtime);
479
480 if (format == OLDGNU_FORMAT && incremental_option)
481 {
482 stat_info->st_atime = TIME_FROM_OCT (header->oldgnu_header.atime);
483 stat_info->st_ctime = TIME_FROM_OCT (header->oldgnu_header.ctime);
484 }
485
486 if (format == V7_FORMAT)
487 {
488 stat_info->st_uid = UID_FROM_OCT (header->header.uid);
489 stat_info->st_gid = GID_FROM_OCT (header->header.gid);
490 stat_info->st_rdev = 0;
491 }
492 else
493 {
494 if (do_user_group)
495 {
496 /* FIXME: Decide if this should somewhat depend on -p. */
497
498 if (numeric_owner_option
499 || !*header->header.uname
500 || !uname_to_uid (header->header.uname, &stat_info->st_uid))
501 stat_info->st_uid = UID_FROM_OCT (header->header.uid);
502
503 if (numeric_owner_option
504 || !*header->header.gname
505 || !gname_to_gid (header->header.gname, &stat_info->st_gid))
506 stat_info->st_gid = GID_FROM_OCT (header->header.gid);
507 }
508 switch (header->header.typeflag)
509 {
510 case BLKTYPE:
511 stat_info->st_rdev
512 = makedev (MAJOR_FROM_OCT (header->header.devmajor),
513 MINOR_FROM_OCT (header->header.devminor));
514 break;
515
516 case CHRTYPE:
517 stat_info->st_rdev
518 = makedev (MAJOR_FROM_OCT (header->header.devmajor),
519 MINOR_FROM_OCT (header->header.devminor));
520 break;
521
522 default:
523 stat_info->st_rdev = 0;
524 }
525 }
526 }
527
528 /*------------------------------------------------------------------------.
529 | Quick and dirty octal conversion. Result is -1 if the field is invalid |
530 | (all blank, or nonoctal). |
531 `------------------------------------------------------------------------*/
532
533 static uintmax_t
534 from_oct (const char *where0, size_t digs0, const char *type, uintmax_t maxval)
535 {
536 uintmax_t value;
537 const char *where = where0;
538 size_t digs = digs0;
539
540 for (;;)
541 {
542 if (digs == 0)
543 {
544 if (type)
545 ERROR ((0, 0, _("Blanks in header where octal %s value expected"),
546 type));
547 return -1;
548 }
549 if (!ISSPACE ((unsigned char) *where))
550 break;
551 where++;
552 digs--;
553 }
554
555 value = 0;
556 while (digs != 0 && ISODIGIT (*where))
557 {
558 /* Scan til nonoctal. */
559
560 if (value << 3 >> 3 != value)
561 goto out_of_range;
562 value = (value << 3) | (*where++ - '0');
563 --digs;
564 }
565
566 if (digs != 0 && *where && !ISSPACE ((unsigned char) *where))
567 {
568 if (type)
569 {
570 char buf[1000]; /* Big enough to represent any header. */
571 static struct quoting_options *o;
572
573 if (!o)
574 {
575 o = clone_quoting_options ((struct quoting_options *) 0);
576 set_quoting_style (o, c_quoting_style);
577 }
578
579 for (digs = digs0; digs && ! where0[digs - 1]; digs--)
580 continue;
581 quotearg_buffer (buf, sizeof buf, where0, digs, o);
582 ERROR ((0, 0,
583 _("Header contains %.*s where octal %s value expected"),
584 (int) sizeof buf, buf, type));
585 }
586
587 return -1;
588 }
589
590 if (value <= maxval)
591 return value;
592
593 out_of_range:
594 if (type)
595 ERROR ((0, 0, _("Octal value `%.*s' is out of range for %s"),
596 (int) digs0, where0, type));
597 return -1;
598 }
599 gid_t
600 gid_from_oct (const char *p, size_t s)
601 {
602 return from_oct (p, s, "gid_t", (uintmax_t) TYPE_MAXIMUM (gid_t));
603 }
604 major_t
605 major_from_oct (const char *p, size_t s)
606 {
607 return from_oct (p, s, "major_t", (uintmax_t) TYPE_MAXIMUM (major_t));
608 }
609 minor_t
610 minor_from_oct (const char *p, size_t s)
611 {
612 return from_oct (p, s, "minor_t", (uintmax_t) TYPE_MAXIMUM (minor_t));
613 }
614 mode_t
615 mode_from_oct (const char *p, size_t s)
616 {
617 /* Do not complain about unrecognized mode bits. */
618 unsigned u = from_oct (p, s, "mode_t", TYPE_MAXIMUM (uintmax_t));
619 return ((u & TSUID ? S_ISUID : 0)
620 | (u & TSGID ? S_ISGID : 0)
621 | (u & TSVTX ? S_ISVTX : 0)
622 | (u & TUREAD ? S_IRUSR : 0)
623 | (u & TUWRITE ? S_IWUSR : 0)
624 | (u & TUEXEC ? S_IXUSR : 0)
625 | (u & TGREAD ? S_IRGRP : 0)
626 | (u & TGWRITE ? S_IWGRP : 0)
627 | (u & TGEXEC ? S_IXGRP : 0)
628 | (u & TOREAD ? S_IROTH : 0)
629 | (u & TOWRITE ? S_IWOTH : 0)
630 | (u & TOEXEC ? S_IXOTH : 0));
631 }
632 off_t
633 off_from_oct (const char *p, size_t s)
634 {
635 return from_oct (p, s, "off_t", (uintmax_t) TYPE_MAXIMUM (off_t));
636 }
637 size_t
638 size_from_oct (const char *p, size_t s)
639 {
640 return from_oct (p, s, "size_t", (uintmax_t) TYPE_MAXIMUM (size_t));
641 }
642 time_t
643 time_from_oct (const char *p, size_t s)
644 {
645 return from_oct (p, s, "time_t", (uintmax_t) TYPE_MAXIMUM (time_t));
646 }
647 uid_t
648 uid_from_oct (const char *p, size_t s)
649 {
650 return from_oct (p, s, "uid_t", (uintmax_t) TYPE_MAXIMUM (uid_t));
651 }
652 uintmax_t
653 uintmax_from_oct (const char *p, size_t s)
654 {
655 return from_oct (p, s, "uintmax_t", TYPE_MAXIMUM (uintmax_t));
656 }
657
658
659
660 /*----------------------------------------------------------------------.
661 | Format O as a null-terminated decimal string into BUF _backwards_; |
662 | return pointer to start of result. |
663 `----------------------------------------------------------------------*/
664 char *
665 stringify_uintmax_t_backwards (uintmax_t o, char *buf)
666 {
667 *--buf = '\0';
668 do
669 *--buf = '0' + (int) (o % 10);
670 while ((o /= 10) != 0);
671 return buf;
672 }
673
674 #if !USE_OLD_CTIME
675
676 /*-------------------------------------------.
677 | Return the time formatted along ISO 8601. |
678 `-------------------------------------------*/
679
680 /* Also, see http://www.ft.uni-erlangen.de/~mskuhn/iso-time.html. */
681
682 static char *
683 isotime (const time_t *time)
684 {
685 static char buffer[21];
686 struct tm *tm;
687
688 tm = localtime (time);
689 sprintf (buffer, "%4d-%02d-%02d %02d:%02d:%02d\n",
690 tm->tm_year + 1900, tm->tm_mon + 1, tm->tm_mday,
691 tm->tm_hour, tm->tm_min, tm->tm_sec);
692 return buffer;
693 }
694
695 #endif /* not USE_OLD_CTIME */
696
697 /*-------------------------------------------------------------------------.
698 | Decode MODE from its binary form in a stat structure, and encode it into |
699 | a 9 characters string STRING, terminated with a NUL. |
700 `-------------------------------------------------------------------------*/
701
702 static void
703 decode_mode (mode_t mode, char *string)
704 {
705 *string++ = mode & S_IRUSR ? 'r' : '-';
706 *string++ = mode & S_IWUSR ? 'w' : '-';
707 *string++ = (mode & S_ISUID
708 ? (mode & S_IXUSR ? 's' : 'S')
709 : (mode & S_IXUSR ? 'x' : '-'));
710 *string++ = mode & S_IRGRP ? 'r' : '-';
711 *string++ = mode & S_IWGRP ? 'w' : '-';
712 *string++ = (mode & S_ISGID
713 ? (mode & S_IXGRP ? 's' : 'S')
714 : (mode & S_IXGRP ? 'x' : '-'));
715 *string++ = mode & S_IROTH ? 'r' : '-';
716 *string++ = mode & S_IWOTH ? 'w' : '-';
717 *string++ = (mode & S_ISVTX
718 ? (mode & S_IXOTH ? 't' : 'T')
719 : (mode & S_IXOTH ? 'x' : '-'));
720 *string = '\0';
721 }
722
723 /*-------------------------------------------------------------------------.
724 | Actually print it. |
725 | |
726 | Plain and fancy file header block logging. Non-verbose just prints the |
727 | name, e.g. for "tar t" or "tar x". This should just contain file names, |
728 | so it can be fed back into tar with xargs or the "-T" option. The |
729 | verbose option can give a bunch of info, one line per file. I doubt |
730 | anybody tries to parse its format, or if they do, they shouldn't. Unix |
731 | tar is pretty random here anyway. |
732 `-------------------------------------------------------------------------*/
733
734 /* FIXME: Note that print_header uses the globals HEAD, HSTAT, and
735 HEAD_STANDARD, which must be set up in advance. Not very clean... */
736
737 /* UGSWIDTH starts with 18, so with user and group names <= 8 chars, the
738 columns never shift during the listing. */
739 #define UGSWIDTH 18
740 static int ugswidth = UGSWIDTH; /* maximum width encountered so far */
741
742 /* DATEWIDTH is the number of columns taken by the date and time fields. */
743 #if USE_OLD_CDATE
744 # define DATEWIDTH 19
745 #else
746 # define DATEWIDTH 18
747 #endif
748
749 void
750 print_header (void)
751 {
752 char modes[11];
753 char *timestamp;
754 /* These hold formatted ints. */
755 char uform[UINTMAX_STRSIZE_BOUND], gform[UINTMAX_STRSIZE_BOUND];
756 char *user, *group;
757 char size[2 * UINTMAX_STRSIZE_BOUND];
758 /* holds formatted size or major,minor */
759 char uintbuf[UINTMAX_STRSIZE_BOUND];
760 time_t longie; /* to make ctime() call portable */
761 int pad;
762 char *name;
763
764 if (block_number_option)
765 {
766 char buf[UINTMAX_STRSIZE_BOUND];
767 fprintf (stdlis, _("block %s: "),
768 STRINGIFY_BIGINT (current_block_ordinal (), buf));
769 }
770
771 if (verbose_option <= 1)
772 {
773 /* Just the fax, mam. */
774
775 char *quoted_name = quote_copy_string (current_file_name);
776
777 if (quoted_name)
778 {
779 fprintf (stdlis, "%s\n", quoted_name);
780 free (quoted_name);
781 }
782 else
783 fprintf (stdlis, "%s\n", current_file_name);
784 }
785 else
786 {
787 /* File type and modes. */
788
789 modes[0] = '?';
790 switch (current_header->header.typeflag)
791 {
792 case GNUTYPE_VOLHDR:
793 modes[0] = 'V';
794 break;
795
796 case GNUTYPE_MULTIVOL:
797 modes[0] = 'M';
798 break;
799
800 case GNUTYPE_NAMES:
801 modes[0] = 'N';
802 break;
803
804 case GNUTYPE_LONGNAME:
805 case GNUTYPE_LONGLINK:
806 ERROR ((0, 0, _("Visible longname error")));
807 break;
808
809 case GNUTYPE_SPARSE:
810 case REGTYPE:
811 case AREGTYPE:
812 case LNKTYPE:
813 modes[0] = '-';
814 if (current_file_name[strlen (current_file_name) - 1] == '/')
815 modes[0] = 'd';
816 break;
817 case GNUTYPE_DUMPDIR:
818 modes[0] = 'd';
819 break;
820 case DIRTYPE:
821 modes[0] = 'd';
822 break;
823 case SYMTYPE:
824 modes[0] = 'l';
825 break;
826 case BLKTYPE:
827 modes[0] = 'b';
828 break;
829 case CHRTYPE:
830 modes[0] = 'c';
831 break;
832 case FIFOTYPE:
833 modes[0] = 'p';
834 break;
835 case CONTTYPE:
836 modes[0] = 'C';
837 break;
838 }
839
840 decode_mode (current_stat.st_mode, modes + 1);
841
842 /* Timestamp. */
843
844 longie = current_stat.st_mtime;
845 #if USE_OLD_CTIME
846 timestamp = ctime (&longie);
847 timestamp[16] = '\0';
848 timestamp[24] = '\0';
849 #else
850 timestamp = isotime (&longie);
851 timestamp[16] = '\0';
852 #endif
853
854 /* User and group names. */
855
856 if (*current_header->header.uname && current_format != V7_FORMAT)
857 user = current_header->header.uname;
858 else
859 user = STRINGIFY_BIGINT (UINTMAX_FROM_OCT (current_header->header.uid),
860 uform);
861
862 if (*current_header->header.gname && current_format != V7_FORMAT)
863 group = current_header->header.gname;
864 else
865 group = STRINGIFY_BIGINT (UINTMAX_FROM_OCT
866 (current_header->header.gid),
867 gform);
868
869 /* Format the file size or major/minor device numbers. */
870
871 switch (current_header->header.typeflag)
872 {
873 case CHRTYPE:
874 case BLKTYPE:
875 sprintf (size, "%lu,%lu",
876 (unsigned long) major (current_stat.st_rdev),
877 (unsigned long) minor (current_stat.st_rdev));
878 break;
879 case GNUTYPE_SPARSE:
880 strcpy (size,
881 STRINGIFY_BIGINT
882 (UINTMAX_FROM_OCT (current_header->oldgnu_header.realsize),
883 uintbuf));
884 break;
885 default:
886 strcpy (size, STRINGIFY_BIGINT (current_stat.st_size, uintbuf));
887 break;
888 }
889
890 /* Figure out padding and print the whole line. */
891
892 pad = strlen (user) + strlen (group) + strlen (size) + 1;
893 if (pad > ugswidth)
894 ugswidth = pad;
895
896 #if USE_OLD_CTIME
897 fprintf (stdlis, "%s %s/%s %*s%s %s %s",
898 modes, user, group, ugswidth - pad, "",
899 size, timestamp + 4, timestamp + 20);
900 #else
901 fprintf (stdlis, "%s %s/%s %*s%s %s",
902 modes, user, group, ugswidth - pad, "", size, timestamp);
903 #endif
904
905 name = quote_copy_string (current_file_name);
906 if (name)
907 {
908 fprintf (stdlis, " %s", name);
909 free (name);
910 }
911 else
912 fprintf (stdlis, " %s", current_file_name);
913
914 switch (current_header->header.typeflag)
915 {
916 case SYMTYPE:
917 name = quote_copy_string (current_link_name);
918 if (name)
919 {
920 fprintf (stdlis, " -> %s\n", name);
921 free (name);
922 }
923 else
924 fprintf (stdlis, " -> %s\n", current_link_name);
925 break;
926
927 case LNKTYPE:
928 name = quote_copy_string (current_link_name);
929 if (name)
930 {
931 fprintf (stdlis, _(" link to %s\n"), name);
932 free (name);
933 }
934 else
935 fprintf (stdlis, _(" link to %s\n"), current_link_name);
936 break;
937
938 default:
939 fprintf (stdlis, _(" unknown file type `%c'\n"),
940 current_header->header.typeflag);
941 break;
942
943 case AREGTYPE:
944 case REGTYPE:
945 case GNUTYPE_SPARSE:
946 case CHRTYPE:
947 case BLKTYPE:
948 case DIRTYPE:
949 case FIFOTYPE:
950 case CONTTYPE:
951 case GNUTYPE_DUMPDIR:
952 putc ('\n', stdlis);
953 break;
954
955 case GNUTYPE_VOLHDR:
956 fprintf (stdlis, _("--Volume Header--\n"));
957 break;
958
959 case GNUTYPE_MULTIVOL:
960 strcpy (size,
961 STRINGIFY_BIGINT
962 (UINTMAX_FROM_OCT (current_header->oldgnu_header.offset),
963 uintbuf));
964 fprintf (stdlis, _("--Continued at byte %s--\n"), size);
965 break;
966
967 case GNUTYPE_NAMES:
968 fprintf (stdlis, _("--Mangled file names--\n"));
969 break;
970 }
971 }
972 fflush (stdlis);
973 }
974
975 /*--------------------------------------------------------------.
976 | Print a similar line when we make a directory automatically. |
977 `--------------------------------------------------------------*/
978
979 void
980 print_for_mkdir (char *pathname, int length, mode_t mode)
981 {
982 char modes[11];
983 char *name;
984
985 if (verbose_option > 1)
986 {
987 /* File type and modes. */
988
989 modes[0] = 'd';
990 decode_mode (mode, modes + 1);
991
992 if (block_number_option)
993 {
994 char buf[UINTMAX_STRSIZE_BOUND];
995 fprintf (stdlis, _("block %s: "),
996 STRINGIFY_BIGINT (current_block_ordinal (), buf));
997 }
998 name = quote_copy_string (pathname);
999 if (name)
1000 {
1001 fprintf (stdlis, "%s %*s %.*s\n", modes, ugswidth + DATEWIDTH,
1002 _("Creating directory:"), length, name);
1003 free (name);
1004 }
1005 else
1006 fprintf (stdlis, "%s %*s %.*s\n", modes, ugswidth + DATEWIDTH,
1007 _("Creating directory:"), length, pathname);
1008 }
1009 }
1010
1011 /*--------------------------------------------------------.
1012 | Skip over SIZE bytes of data in blocks in the archive. |
1013 `--------------------------------------------------------*/
1014
1015 void
1016 skip_file (off_t size)
1017 {
1018 union block *x;
1019
1020 if (multi_volume_option)
1021 {
1022 save_totsize = size;
1023 save_sizeleft = size;
1024 }
1025
1026 while (size > 0)
1027 {
1028 x = find_next_block ();
1029 if (x == NULL)
1030 FATAL_ERROR ((0, 0, _("Unexpected EOF on archive file")));
1031
1032 set_next_block_after (x);
1033 size -= BLOCKSIZE;
1034 if (multi_volume_option)
1035 save_sizeleft -= BLOCKSIZE;
1036 }
1037 }
1038
1039 /*---.
1040 | ? |
1041 `---*/
1042
1043 void
1044 skip_extended_headers (void)
1045 {
1046 union block *exhdr;
1047
1048 while (1)
1049 {
1050 exhdr = find_next_block ();
1051 if (!exhdr->sparse_header.isextended)
1052 {
1053 set_next_block_after (exhdr);
1054 break;
1055 }
1056 set_next_block_after (exhdr);
1057 }
1058 }
This page took 0.079552 seconds and 5 git commands to generate.