aboutsummaryrefslogtreecommitdiff
path: root/toys/posix/cpio.c
diff options
context:
space:
mode:
authorRob Landley <rob@landley.net>2015-12-21 11:28:30 -0600
committerRob Landley <rob@landley.net>2015-12-21 11:28:30 -0600
commit6be15f4143a40cf7b2985be88ad16bc996310b59 (patch)
treee4277d2517b48377ae17d5f2552cab2cf59af0e6 /toys/posix/cpio.c
parentfde3a9300868b7505b76a652be27ac42ee534df3 (diff)
downloadtoybox-6be15f4143a40cf7b2985be88ad16bc996310b59.tar.gz
Rich Felker's reading of posix is that fchown() and fchmod() must be allowed
on files opened read-only, so revert last commit (force writeable logic).
Diffstat (limited to 'toys/posix/cpio.c')
-rw-r--r--toys/posix/cpio.c37
1 files changed, 14 insertions, 23 deletions
diff --git a/toys/posix/cpio.c b/toys/posix/cpio.c
index 712980cc..0316cc9d 100644
--- a/toys/posix/cpio.c
+++ b/toys/posix/cpio.c
@@ -109,7 +109,7 @@ void cpio_main(void)
if (toys.optflags & (FLAG_i|FLAG_t)) for (;;) {
char *name, *tofree, *data;
- unsigned size, mode, uid, gid, timestamp, ala = 0;
+ unsigned size, mode, uid, gid, timestamp;
int test = toys.optflags & FLAG_t, err = 0;
// Read header and name.
@@ -132,10 +132,6 @@ void cpio_main(void)
if (toys.optflags & (FLAG_t|FLAG_v)) puts(name);
- // If we need to reopen dir or mknod for write later, force u+w now
- if (!(toys.optflags & FLAG_no_preserve_owner)
- && !S_ISREG(mode) && !S_ISLNK(mode) && !geteuid()) ala = 0200;
-
if (!test && strrchr(name, '/') && mkpathat(AT_FDCWD, name, 0, 2)) {
perror_msg("mkpath '%s'", name);
test++;
@@ -145,7 +141,7 @@ void cpio_main(void)
// properly aligned with next file.
if (S_ISDIR(mode)) {
- if (!test) err = mkdir(name, ala|mode);
+ if (!test) err = mkdir(name, mode);
} else if (S_ISLNK(mode)) {
data = strpad(afd, size, 0);
if (!test) err = symlink(data, name);
@@ -184,29 +180,24 @@ void cpio_main(void)
close(fd);
}
} else if (!test)
- err = mknod(name, ala|mode, makedev(x8u(toybuf+78), x8u(toybuf+86)));
+ err = mknod(name, mode, makedev(x8u(toybuf+78), x8u(toybuf+86)));
// Set ownership and timestamp.
if (!test && !err) {
- // Creating dir/dev doesn't give us a filehandle, we have to refer to it
- // by name to chown/utime, but how do we know it's the same item? Use
- // filehandle to check that we at least have the right type of entity
- // open. If we forced it writeable, chmod it back, but do _not_ restore
- // dropped suid/sgid bit.
- if (ala) {
- int fd = open(name, O_WRONLY|O_NOFOLLOW);
+ // Creading dir/dev doesn't give us a filehandle, we have to refer to it
+ // by name to chown/utime, but how do we know it's the same item?
+ // Check that we at least have the right type of entity open, and do
+ // NOT restore dropped suid bit in this case.
+ if (!S_ISREG(mode) && !S_ISLNK(mode) && !geteuid()
+ && !(toys.optflags & FLAG_no_preserve_owner))
+ {
+ int fd = open(name, O_RDONLY|O_NOFOLLOW);
struct stat st;
- // If we forced it writeable, change it back, but do _not_ restore
- // dropped suid/sgid bit.
-
- if (fd != -1 && !fstat(fd, &st) && (st.st_mode&S_IFMT)==(mode&S_IFMT)) {
- if (!(err = fchown(fd, uid, gid)))
- if ((ala|mode)!=mode) err = fchmod(fd, mode&~(S_ISUID|S_ISGID));
- } else err = 1;
+ if (fd != -1 && !fstat(fd, &st) && (st.st_mode&S_IFMT) == (mode&S_IFMT))
+ err = fchown(fd, uid, gid);
+ else err = 1;
- // If we forced it writeable, change it back, but do _not_ restore
- // dropped suid/sgid bit.
close(fd);
}