diff options
author | Jim Meyering <jim@meyering.net> | 2003-11-17 16:13:04 +0000 |
---|---|---|
committer | Jim Meyering <jim@meyering.net> | 2003-11-17 16:13:04 +0000 |
commit | 476ff1e6232f7d636b5665496f5346d161eeb47b (patch) | |
tree | 165c6bd3b7ae747b0099c727348f7416e6508d7b /lib | |
parent | e81926abd0071068caad93eff8d9751a3e0baffa (diff) | |
download | coreutils-476ff1e6232f7d636b5665496f5346d161eeb47b.tar.xz |
On systems without utime and without a utimes function capable of
dealing with a NULL struct utimbuf* argument, this utime replacement
could -- in unusual circumstances -- leak a file descriptor.
Include <unistd.h> and <errno.h>.
(utime_null): Be sure to close `fd' and to preserve errno.
Diffstat (limited to 'lib')
-rw-r--r-- | lib/utime.c | 27 |
1 files changed, 24 insertions, 3 deletions
diff --git a/lib/utime.c b/lib/utime.c index 25657eea9..7b87a969d 100644 --- a/lib/utime.c +++ b/lib/utime.c @@ -32,6 +32,12 @@ # include <fcntl.h> #endif +#include <unistd.h> +#include <errno.h> +#ifndef errno +extern int errno; +#endif + #include "full-write.h" #include "safe-read.h" @@ -59,6 +65,7 @@ utime_null (const char *file) char c; int status = 0; struct stat st; + int saved_errno = 0; fd = open (file, O_RDWR); if (fd < 0 @@ -70,9 +77,23 @@ utime_null (const char *file) of patches, but that system doesn't use this code: it has utimes. || fsync (fd) < 0 */ - || (st.st_size == 0 && ftruncate (fd, st.st_size) < 0) - || close (fd) < 0) - status = -1; + || (st.st_size == 0 && ftruncate (fd, st.st_size) < 0)) + { + saved_errno = errno; + status = -1; + } + + if (0 <= fd) + { + if (close (fd) < 0) + status = -1; + + /* If there was a prior failure, use the saved errno value. + But if the only failure was in the close, don't change errno. */ + if (saved_errno) + errno = saved_errno; + } + return status; #endif } |