]> git.cworth.org Git - tar/blob - lib/utime.c
Imported Upstream version 1.20
[tar] / lib / utime.c
1 /* Copyright (C) 1998, 2001, 2002, 2003, 2004, 2006 Free Software
2    Foundation, Inc.
3
4    This program is free software: you can redistribute it and/or modify it
5    under the terms of the GNU General Public License as published by the
6    Free Software Foundation; either version 3 of the License, or any
7    later version.
8
9    This program is distributed in the hope that it will be useful,
10    but WITHOUT ANY WARRANTY; without even the implied warranty of
11    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12    GNU General Public License for more details.
13
14    You should have received a copy of the GNU General Public License
15    along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
16
17 /* derived from a function in touch.c */
18
19 #include <config.h>
20 #undef utime
21
22 #include <sys/types.h>
23
24 #ifdef HAVE_UTIME_H
25 # include <utime.h>
26 #endif
27
28 #if !HAVE_UTIMES_NULL
29 # include <sys/stat.h>
30 # include <fcntl.h>
31 #endif
32
33 #include <unistd.h>
34 #include <errno.h>
35
36 #include "full-write.h"
37 #include "safe-read.h"
38
39 /* Some systems (even some that do have <utime.h>) don't declare this
40    structure anywhere.  */
41 #ifndef HAVE_STRUCT_UTIMBUF
42 struct utimbuf
43 {
44   long actime;
45   long modtime;
46 };
47 #endif
48
49 /* The results of open() in this file are not used with fchdir,
50    therefore save some unnecessary work in fchdir.c.  */
51 #undef open
52 #undef close
53
54 /* Emulate utime (file, NULL) for systems (like 4.3BSD) that do not
55    interpret it to set the access and modification times of FILE to
56    the current time.  Return 0 if successful, -1 if not. */
57
58 static int
59 utime_null (const char *file)
60 {
61 #if HAVE_UTIMES_NULL
62   return utimes (file, 0);
63 #else
64   int fd;
65   char c;
66   int status = 0;
67   struct stat st;
68   int saved_errno = 0;
69
70   fd = open (file, O_RDWR);
71   if (fd < 0
72       || fstat (fd, &st) < 0
73       || safe_read (fd, &c, sizeof c) == SAFE_READ_ERROR
74       || lseek (fd, (off_t) 0, SEEK_SET) < 0
75       || full_write (fd, &c, sizeof c) != sizeof c
76       /* Maybe do this -- it's necessary on SunOS 4.1.3 with some combination
77          of patches, but that system doesn't use this code: it has utimes.
78          || fsync (fd) < 0
79       */
80       || (st.st_size == 0 && ftruncate (fd, st.st_size) < 0))
81     {
82       saved_errno = errno;
83       status = -1;
84     }
85
86   if (0 <= fd)
87     {
88       if (close (fd) < 0)
89         status = -1;
90
91       /* If there was a prior failure, use the saved errno value.
92          But if the only failure was in the close, don't change errno.  */
93       if (saved_errno)
94         errno = saved_errno;
95     }
96
97   return status;
98 #endif
99 }
100
101 int
102 rpl_utime (const char *file, const struct utimbuf *times)
103 {
104   if (times)
105     return utime (file, times);
106
107   return utime_null (file);
108 }