Skip to content

Commit 06e4fe5

Browse files
Ciaran Anscombtridge
Ciaran Anscomb
authored andcommitted
sanitise results of stat()/lstat() on Solaris
Solaris 11 (possibly only 11.4) can return negative values for st_atim.tv_nsec, st_mtim.tv_nsec and st_ctim.tv_nsec. Gnulib already identifies this as an issue and works around it with stat_time_normalize(). I've adapted that function for this patch, omitting st_ctim (unused by rsync) and using only a very simple test for time_t overflow. The feature test macros used mean this code path will be used on _all_ Solaris builds, but this shouldn't be an issue; it would probably be safe on anything that provides tv_nsec.
1 parent 99673f9 commit 06e4fe5

File tree

2 files changed

+55
-4
lines changed

2 files changed

+55
-4
lines changed

syscall.c

+14-4
Original file line numberDiff line numberDiff line change
@@ -357,19 +357,29 @@ int do_mkstemp(char *template, mode_t perms)
357357
int do_stat(const char *path, STRUCT_STAT *st)
358358
{
359359
#ifdef USE_STAT64_FUNCS
360-
return stat64(path, st);
360+
int r = stat64(path, st);
361361
#else
362-
return stat(path, st);
362+
int r = stat(path, st);
363+
#endif
364+
#if defined __sun && defined HAVE_STRUCT_STAT_ST_MTIM_TV_NSEC
365+
return stat_time_normalize(r, st);
366+
#else
367+
return r;
363368
#endif
364369
}
365370

366371
int do_lstat(const char *path, STRUCT_STAT *st)
367372
{
368373
#ifdef SUPPORT_LINKS
369374
# ifdef USE_STAT64_FUNCS
370-
return lstat64(path, st);
375+
int r = lstat64(path, st);
376+
# else
377+
int r = lstat(path, st);
378+
# endif
379+
# if defined __sun && defined HAVE_STRUCT_STAT_ST_MTIM_TV_NSEC
380+
return stat_time_normalize(r, st);
371381
# else
372-
return lstat(path, st);
382+
return r;
373383
# endif
374384
#else
375385
return do_stat(path, st);

util2.c

+41
Original file line numberDiff line numberDiff line change
@@ -143,3 +143,44 @@ const char *src_file(const char *file)
143143
return file + prefix;
144144
return file;
145145
}
146+
147+
/* If a stat-like function returned RESULT, normalize the timestamps
148+
in *ST, in case this platform suffers from the Solaris 11 bug where
149+
tv_nsec might be negative. Return the adjusted RESULT, setting
150+
errno to EOVERFLOW if normalization overflowed. This function
151+
is intended to be private to this .h file.
152+
153+
This is adapted from Gnulib. */
154+
155+
int stat_time_normalize(int result, STRUCT_STAT *st)
156+
{
157+
#if defined __sun && defined HAVE_STRUCT_STAT_ST_MTIM_TV_NSEC
158+
if (result == 0) {
159+
long int timespec_hz = 1000000000;
160+
short int const ts_off[] = { offsetof (struct stat, st_atim),
161+
offsetof (struct stat, st_mtim) };
162+
unsigned i;
163+
for (i = 0; i < sizeof ts_off / sizeof *ts_off; i++) {
164+
struct timespec *ts = (struct timespec *)((char *) st + ts_off[i]);
165+
long int q = ts->tv_nsec / timespec_hz;
166+
long int r = ts->tv_nsec % timespec_hz;
167+
if (r < 0) {
168+
r += timespec_hz;
169+
q--;
170+
}
171+
ts->tv_nsec = r;
172+
/* Overflow is possible, as Solaris 11 stat can yield
173+
tv_sec == TYPE_MINIMUM (time_t) && tv_nsec == -1000000000. */
174+
time_t sec = ts->tv_sec + q;
175+
if ((q > 0 && sec < ts->tv_sec) || (q < 0 && sec > ts->tv_sec)) {
176+
errno = EOVERFLOW;
177+
return -1;
178+
}
179+
ts->tv_sec = sec;
180+
}
181+
}
182+
#else
183+
(void)st;
184+
#endif
185+
return result;
186+
}

0 commit comments

Comments
 (0)