Lines 24-34
Link Here
|
24 |
|
24 |
|
25 |
#include "utimens.h" |
25 |
#include "utimens.h" |
26 |
|
26 |
|
27 |
#include <errno.h> |
27 |
#include <errno.h> |
28 |
#include <fcntl.h> |
28 |
#include <fcntl.h> |
29 |
#include <sys/stat.h> |
|
|
30 |
#include <sys/time.h> |
29 |
#include <sys/time.h> |
31 |
#include <unistd.h> |
30 |
#include <unistd.h> |
32 |
|
31 |
|
33 |
#if HAVE_UTIME_H |
32 |
#if HAVE_UTIME_H |
34 |
# include <utime.h> |
33 |
# include <utime.h> |
Lines 93-197
Link Here
|
93 |
sync (); |
92 |
sync (); |
94 |
else |
93 |
else |
95 |
fsync (fd); |
94 |
fsync (fd); |
96 |
#endif |
95 |
#endif |
97 |
|
96 |
|
98 |
/* POSIX 200x added two interfaces to set file timestamps with |
97 |
/* There's currently no interface to set file timestamps with |
99 |
nanosecond resolution. */ |
|
|
100 |
#if HAVE_UTIMENSAT |
101 |
if (fd < 0) |
102 |
return utimensat (AT_FDCWD, file, timespec, 0); |
103 |
#endif |
104 |
#if HAVE_FUTIMENS |
105 |
return futimens (fd, timespec); |
106 |
#else |
107 |
|
108 |
/* The platform lacks an interface to set file timestamps with |
109 |
nanosecond resolution, so do the best we can, discarding any |
98 |
nanosecond resolution, so do the best we can, discarding any |
110 |
fractional part of the timestamp. */ |
99 |
fractional part of the timestamp. */ |
111 |
{ |
100 |
#if HAVE_FUTIMESAT || HAVE_WORKING_UTIMES |
112 |
# if HAVE_FUTIMESAT || HAVE_WORKING_UTIMES |
101 |
struct timeval timeval[2]; |
113 |
struct timeval timeval[2]; |
102 |
struct timeval const *t; |
114 |
struct timeval const *t; |
103 |
if (timespec) |
115 |
if (timespec) |
104 |
{ |
116 |
{ |
105 |
timeval[0].tv_sec = timespec[0].tv_sec; |
117 |
timeval[0].tv_sec = timespec[0].tv_sec; |
106 |
timeval[0].tv_usec = timespec[0].tv_nsec / 1000; |
118 |
timeval[0].tv_usec = timespec[0].tv_nsec / 1000; |
107 |
timeval[1].tv_sec = timespec[1].tv_sec; |
119 |
timeval[1].tv_sec = timespec[1].tv_sec; |
108 |
timeval[1].tv_usec = timespec[1].tv_nsec / 1000; |
120 |
timeval[1].tv_usec = timespec[1].tv_nsec / 1000; |
109 |
t = timeval; |
121 |
t = timeval; |
110 |
} |
122 |
} |
111 |
else |
123 |
else |
112 |
t = NULL; |
124 |
t = NULL; |
|
|
125 |
|
113 |
|
126 |
if (fd < 0) |
|
|
127 |
{ |
128 |
# if HAVE_FUTIMESAT |
129 |
return futimesat (AT_FDCWD, file, t); |
130 |
# endif |
131 |
} |
132 |
else |
133 |
{ |
134 |
/* If futimesat or futimes fails here, don't try to speed things |
135 |
up by returning right away. glibc can incorrectly fail with |
136 |
errno == ENOENT if /proc isn't mounted. Also, Mandrake 10.0 |
137 |
in high security mode doesn't allow ordinary users to read |
138 |
/proc/self, so glibc incorrectly fails with errno == EACCES. |
139 |
If errno == EIO, EPERM, or EROFS, it's probably safe to fail |
140 |
right away, but these cases are rare enough that they're not |
141 |
worth optimizing, and who knows what other messed-up systems |
142 |
are out there? So play it safe and fall back on the code |
143 |
below. */ |
144 |
# if HAVE_FUTIMESAT |
145 |
if (futimesat (fd, NULL, t) == 0) |
146 |
return 0; |
147 |
# elif HAVE_FUTIMES |
148 |
if (futimes (fd, t) == 0) |
149 |
return 0; |
150 |
# endif |
151 |
} |
152 |
# endif /* HAVE_FUTIMESAT || HAVE_WORKING_UTIMES */ |
153 |
|
114 |
|
154 |
if (!file) |
115 |
if (fd < 0) |
155 |
{ |
116 |
{ |
156 |
# if ! (HAVE_FUTIMESAT || (HAVE_WORKING_UTIMES && HAVE_FUTIMES)) |
117 |
# if HAVE_FUTIMESAT |
157 |
errno = ENOSYS; |
118 |
return futimesat (AT_FDCWD, file, t); |
158 |
# endif |
119 |
# endif |
|
|
120 |
} |
121 |
else |
122 |
{ |
123 |
/* If futimesat or futimes fails here, don't try to speed things |
124 |
up by returning right away. glibc can incorrectly fail with |
125 |
errno == ENOENT if /proc isn't mounted. Also, Mandrake 10.0 |
126 |
in high security mode doesn't allow ordinary users to read |
127 |
/proc/self, so glibc incorrectly fails with errno == EACCES. |
128 |
If errno == EIO, EPERM, or EROFS, it's probably safe to fail |
129 |
right away, but these cases are rare enough that they're not |
130 |
worth optimizing, and who knows what other messed-up systems |
131 |
are out there? So play it safe and fall back on the code |
132 |
below. */ |
133 |
# if HAVE_FUTIMESAT |
134 |
if (futimesat (fd, NULL, t) == 0) |
135 |
return 0; |
136 |
# elif HAVE_FUTIMES |
137 |
if (futimes (fd, t) == 0) |
138 |
return 0; |
139 |
# endif |
140 |
} |
141 |
#endif |
159 |
|
142 |
|
160 |
/* Prefer EBADF to ENOSYS if both error numbers apply. */ |
143 |
if (!file) |
161 |
if (errno == ENOSYS) |
|
|
162 |
{ |
163 |
int fd2 = dup (fd); |
164 |
int dup_errno = errno; |
165 |
if (0 <= fd2) |
166 |
close (fd2); |
167 |
errno = (fd2 < 0 && dup_errno == EBADF ? EBADF : ENOSYS); |
168 |
} |
169 |
|
170 |
return -1; |
171 |
} |
172 |
|
173 |
# if HAVE_WORKING_UTIMES |
174 |
return utimes (file, t); |
175 |
# else |
176 |
{ |
144 |
{ |
177 |
struct utimbuf utimbuf; |
145 |
#if ! (HAVE_FUTIMESAT || (HAVE_WORKING_UTIMES && HAVE_FUTIMES)) |
178 |
struct utimbuf const *ut; |
146 |
errno = ENOSYS; |
179 |
if (timespec) |
147 |
#endif |
|
|
148 |
|
149 |
/* Prefer EBADF to ENOSYS if both error numbers apply. */ |
150 |
if (errno == ENOSYS) |
180 |
{ |
151 |
{ |
181 |
utimbuf.actime = timespec[0].tv_sec; |
152 |
int fd2 = dup (fd); |
182 |
utimbuf.modtime = timespec[1].tv_sec; |
153 |
int dup_errno = errno; |
183 |
ut = &utimbuf; |
154 |
if (0 <= fd2) |
|
|
155 |
close (fd2); |
156 |
errno = (fd2 < 0 && dup_errno == EBADF ? EBADF : ENOSYS); |
184 |
} |
157 |
} |
185 |
else |
|
|
186 |
ut = NULL; |
187 |
|
158 |
|
188 |
return utime (file, ut); |
159 |
return -1; |
189 |
} |
160 |
} |
190 |
# endif /* !HAVE_WORKING_UTIMES */ |
161 |
|
|
|
162 |
#if HAVE_WORKING_UTIMES |
163 |
return utimes (file, t); |
164 |
#else |
165 |
{ |
166 |
struct utimbuf utimbuf; |
167 |
struct utimbuf const *ut; |
168 |
if (timespec) |
169 |
{ |
170 |
utimbuf.actime = timespec[0].tv_sec; |
171 |
utimbuf.modtime = timespec[1].tv_sec; |
172 |
ut = &utimbuf; |
173 |
} |
174 |
else |
175 |
ut = NULL; |
176 |
|
177 |
return utime (file, ut); |
191 |
} |
178 |
} |
192 |
#endif /* !HAVE_FUTIMENS */ |
179 |
#endif |
193 |
} |
180 |
} |
194 |
|
181 |
|
195 |
/* Set the access and modification time stamps of FILE to be |
182 |
/* Set the access and modification time stamps of FILE to be |
196 |
TIMESPEC[0] and TIMESPEC[1], respectively. */ |
183 |
TIMESPEC[0] and TIMESPEC[1], respectively. */ |
197 |
int |
184 |
int |