diff options
author | Jim Meyering <jim@meyering.net> | 2004-03-17 10:06:57 +0000 |
---|---|---|
committer | Jim Meyering <jim@meyering.net> | 2004-03-17 10:06:57 +0000 |
commit | 553e0746b0b5c8463fab72fdb924bccb85fefb4c (patch) | |
tree | 84a363aaccdfb5d6f3ae406d642db3e665827adb | |
parent | 1188f599d9e0140cfe2109deb5d4fa573c4ff551 (diff) | |
download | coreutils-553e0746b0b5c8463fab72fdb924bccb85fefb4c.tar.xz |
(newtime): Now an array of two timespecs, one
for access and one for modification.
(ref_stats): Remove.
(get_reldate): Use get_date's parameter profile.
(touch, main): Adjust to above changes.
(main): Work even if tm_year == INT_MAX (so long as long int is wider).
Use gettime instead of gettimeofday, for new get_date signature.
-rw-r--r-- | src/touch.c | 78 |
1 files changed, 29 insertions, 49 deletions
diff --git a/src/touch.c b/src/touch.c index 728f62c3f..8dd204c85 100644 --- a/src/touch.c +++ b/src/touch.c @@ -69,15 +69,12 @@ static int posix_date; we have neither read nor write access to it. */ static int amtime_now; -/* New time to use when setting time. */ -static struct timespec newtime; +/* New access and modification times to use when setting time. */ +static struct timespec newtime[2]; /* File to use for -r. */ static char *ref_file; -/* Info about the reference file. */ -static struct stat ref_stats; - /* For long options that have no equivalent short option, use a non-character as a pseudo short option, starting with CHAR_MAX + 1. */ enum @@ -109,17 +106,15 @@ static int const time_masks[] = CH_ATIME, CH_ATIME, CH_ATIME, CH_MTIME, CH_MTIME }; -/* Interpret FLEX_DATE as a date, relative to NOW, and return the - respresented time. If NOW is null, use the current time. - FIXME: add support for subsecond resolution. */ +/* Store into *RESULT the result of interpreting FLEX_DATE as a date, + relative to NOW. If NOW is null, use the current time. */ -static time_t -get_reldate (char const *flex_date, time_t const *now) +static void +get_reldate (struct timespec *result, + char const *flex_date, struct timespec const *now) { - time_t r = get_date (flex_date, now); - if (r == (time_t) -1) + if (! get_date (result, flex_date, now)) error (EXIT_FAILURE, 0, _("invalid date format %s"), quote (flex_date)); - return r; } /* Update the time of file FILE according to the options given. @@ -185,20 +180,7 @@ touch (const char *file) else { struct timespec timespec[2]; - - /* There's currently no interface to set file timestamps with - better than 1-second resolution, so discard any fractional - part of the source timestamp. */ - - if (use_ref) - { - timespec[0].tv_sec = ref_stats.st_atime; - timespec[0].tv_nsec = TIMESPEC_NS (ref_stats.st_atim); - timespec[1].tv_sec = ref_stats.st_mtime; - timespec[1].tv_nsec = TIMESPEC_NS (ref_stats.st_mtim); - } - else - timespec[0] = timespec[1] = newtime; + memcpy (timespec, newtime, sizeof timespec); if (!(change_times & CH_ATIME)) { @@ -329,10 +311,11 @@ main (int argc, char **argv) case 't': posix_date++; - if (! posixtime (&newtime.tv_sec, optarg, + if (! posixtime (&newtime[0].tv_sec, optarg, PDS_LEADING_YEAR | PDS_CENTURY | PDS_SECONDS)) error (EXIT_FAILURE, 0, _("invalid date format %s"), quote (optarg)); - newtime.tv_nsec = 0; + newtime[0].tv_nsec = 0; + newtime[1] = newtime[0]; date_set++; break; @@ -361,15 +344,20 @@ main (int argc, char **argv) if (use_ref) { + struct stat ref_stats; if (stat (ref_file, &ref_stats)) error (EXIT_FAILURE, errno, _("failed to get attributes of %s"), quote (ref_file)); + newtime[0].tv_sec = ref_stats.st_atime; + newtime[0].tv_nsec = TIMESPEC_NS (ref_stats.st_atim); + newtime[1].tv_sec = ref_stats.st_mtime; + newtime[1].tv_nsec = TIMESPEC_NS (ref_stats.st_mtim); if (flex_date) { if (change_times & CH_ATIME) - ref_stats.st_atime = get_reldate (flex_date, &ref_stats.st_atime); + get_reldate (&newtime[0], flex_date, &newtime[0]); if (change_times & CH_MTIME) - ref_stats.st_mtime = get_reldate (flex_date, &ref_stats.st_mtime); + get_reldate (&newtime[1], flex_date, &newtime[1]); } date_set++; } @@ -377,8 +365,8 @@ main (int argc, char **argv) { if (flex_date) { - newtime.tv_sec = get_reldate (flex_date, NULL); - newtime.tv_nsec = 0; + get_reldate (&newtime[0], flex_date, NULL); + newtime[1] = newtime[0]; } } @@ -387,17 +375,18 @@ main (int argc, char **argv) if (!date_set && 2 <= argc - optind && !STREQ (argv[optind - 1], "--") && posix2_version () < 200112) { - if (posixtime (&newtime.tv_sec, argv[optind], PDS_TRAILING_YEAR)) + if (posixtime (&newtime[0].tv_sec, argv[optind], PDS_TRAILING_YEAR)) { - newtime.tv_nsec = 0; + newtime[0].tv_nsec = 0; + newtime[1] = newtime[0]; if (! getenv ("POSIXLY_CORRECT")) { - struct tm const *tm = localtime (&newtime.tv_sec); + struct tm const *tm = localtime (&newtime[0].tv_sec); error (0, 0, _("warning: `touch %s' is obsolete; use\ - `touch -t %04d%02d%02d%02d%02d.%02d'"), + `touch -t %04ld%02d%02d%02d%02d.%02d'"), argv[optind], - tm->tm_year + 1900, tm->tm_mon + 1, tm->tm_mday, + tm->tm_year + 1900L, tm->tm_mon + 1, tm->tm_mday, tm->tm_hour, tm->tm_min, tm->tm_sec); } @@ -411,18 +400,9 @@ main (int argc, char **argv) amtime_now = 1; else { - /* Get time of day, but only to microsecond resolution, - since 'utimes' currently supports only microsecond - resolution at best. It would be cleaner here to invoke - gettime, but then we would have to link in more shared - libraries on platforms like Solaris, and we'd rather not - have 'touch' depend on libraries that it doesn't - need. */ - struct timeval timeval; - if (gettimeofday (&timeval, NULL) != 0) + if (gettime (&newtime[0]) != 0) error (EXIT_FAILURE, errno, _("cannot get time of day")); - newtime.tv_sec = timeval.tv_sec; - newtime.tv_nsec = timeval.tv_usec * 1000; + newtime[1] = newtime[0]; } } |