summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJim Meyering <jim@meyering.net>2003-11-17 16:13:04 +0000
committerJim Meyering <jim@meyering.net>2003-11-17 16:13:04 +0000
commit476ff1e6232f7d636b5665496f5346d161eeb47b (patch)
tree165c6bd3b7ae747b0099c727348f7416e6508d7b
parente81926abd0071068caad93eff8d9751a3e0baffa (diff)
downloadcoreutils-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.
-rw-r--r--lib/utime.c27
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
}