summaryrefslogtreecommitdiffstats
path: root/gnu/usr.bin/tar
diff options
context:
space:
mode:
authorbde <bde@FreeBSD.org>1995-06-26 06:24:48 +0000
committerbde <bde@FreeBSD.org>1995-06-26 06:24:48 +0000
commitf0ca90235ffbc38533e8d664d8c006ab3432d41a (patch)
tree40316348fc3ad4b1c9c40d02e1e3d943f37eb6ac /gnu/usr.bin/tar
parent1aca6d86947608971056aac349a058ddb79b5ca6 (diff)
downloadFreeBSD-src-f0ca90235ffbc38533e8d664d8c006ab3432d41a.zip
FreeBSD-src-f0ca90235ffbc38533e8d664d8c006ab3432d41a.tar.gz
Improve the handling of large minor numbers:
cpio/copyout.c: Don't output a file if the major, minor or totality of its rdev would be truncated. Print a message about the skipped files to stderr but don't report the error in the exit status. cpio's abysmal error handling doesn't allow continuing after an error, and the rdev checks had to be misplaced to avoid the problem of returning an error code from routines that return void. pax/pax.h: Use the system macros for major(), minor() and makedev(). pax already checks _all_ output conversions for overflow. This has the undesirable effect that failure to convert relatively useless fields such as st_dev for regular files causes files not to be output. pax doesn't report exactly which fields couldn't be converted. tar/create.c: Don't output a file if the major or minor its rdev would be truncated. Print a message about the skipped files to stderr and report the error in the exit status. tar/tar.c: For not immediately fatal errors, exit with status 1, not the error count (mod 256). All: Minor numbers are limited to 21 bits in pax's ustar format and to 18 bits in archives created by gnu tar (gnu tar wastes 3 bits for padding). pax's and cpio's ustar format is incompatible with gnu tar's ustar format for other reasons (see cpio/README).
Diffstat (limited to 'gnu/usr.bin/tar')
-rw-r--r--gnu/usr.bin/tar/create.c34
-rw-r--r--gnu/usr.bin/tar/tar.c2
2 files changed, 31 insertions, 5 deletions
diff --git a/gnu/usr.bin/tar/create.c b/gnu/usr.bin/tar/create.c
index 5d1b71c..c3be00c 100644
--- a/gnu/usr.bin/tar/create.c
+++ b/gnu/usr.bin/tar/create.c
@@ -875,10 +875,20 @@ dump_file (p, curdev, toplevel)
#if defined(S_IFBLK) || defined(S_IFCHR)
if (type != LF_FIFO)
{
- to_oct ((long) major (hstat.st_rdev), 8,
- header->header.devmajor);
- to_oct ((long) minor (hstat.st_rdev), 8,
- header->header.devminor);
+ if (checked_to_oct ((long) major (hstat.st_rdev), 8,
+ header->header.devmajor))
+ {
+ msg ("%s: major number too large; not dumped", p);
+ critical_error = 1;
+ goto badfile;
+ }
+ if (checked_to_oct ((long) minor (hstat.st_rdev), 8,
+ header->header.devminor))
+ {
+ msg ("%s: minor number too large; not dumped", p);
+ critical_error = 1;
+ goto badfile;
+ }
}
#endif
@@ -1397,6 +1407,22 @@ to_oct (value, digs, where)
/*
+ * Call to_oct (), then return nonzero iff the conversion failed.
+ */
+int
+checked_to_oct (value, digs, where)
+ register long value;
+ register int digs;
+ register char *where;
+{
+ long from_oct ();
+
+ to_oct (value, digs, where);
+ return from_oct (digs, where) != value;
+}
+
+
+/*
* Write the EOT record(s).
* We actually zero at least one record, through the end of the block.
* Old tar writes garbage after two zeroed records -- and PDtar used to.
diff --git a/gnu/usr.bin/tar/tar.c b/gnu/usr.bin/tar/tar.c
index 6385986..84e2b94 100644
--- a/gnu/usr.bin/tar/tar.c
+++ b/gnu/usr.bin/tar/tar.c
@@ -281,7 +281,7 @@ main (argc, argv)
}
if (f_volno_file)
closeout_volume_number ();
- exit (errors);
+ exit (errors ? EX_ARGSBAD : 0); /* FIXME (should be EX_NONDESCRIPT) */
/* NOTREACHED */
}
OpenPOWER on IntegriCloud