/* touch.c : change timestamp of a file * * Copyright 2012 Choubey Ji <warior.linux@gmail.com> * * See http://pubs.opengroup.org/onlinepubs/9699919799/utilities/touch.html * * -f is ignored for BSD/macOS compatibility. busybox/coreutils also support * this, but only coreutils documents it in --help output. USE_TOUCH(NEWTOY(touch, "<1acd:fmr:t:h[!dtr]", TOYFLAG_BIN)) config TOUCH bool "touch" default y help usage: touch [-amch] [-d DATE] [-t TIME] [-r FILE] FILE... Update the access and modification times of each FILE to the current time. -a Change access time -m Change modification time -c Don't create file -h Change symlink -d Set time to DATE (in YYYY-MM-DDThh:mm:SS[.frac][tz] format) -t Set time to TIME (in [[CC]YY]MMDDhhmm[.ss][frac] format) -r Set time same as reference FILE */ #define FOR_touch #include "toys.h" GLOBALS( char *t, *r, *d; ) void touch_main(void) { struct timespec ts[2]; char **ss; int fd, i; // use current time if no -t or -d ts[0].tv_nsec = UTIME_NOW; if (FLAG(t) || FLAG(d)) { char *s, *date, **format; struct tm tm; int len = 0; // Initialize default values for time fields ts->tv_sec = time(0); ts->tv_nsec = 0; // List of search types if (FLAG(d)) { format = (char *[]){"%Y-%m-%dT%T", "%Y-%m-%d %T", 0}; date = TT.d; } else { format = (char *[]){"%m%d%H%M", "%y%m%d%H%M", "%C%y%m%d%H%M", 0}; date = TT.t; } // Trailing Z means UTC timezone, don't expect libc to know this. i = strlen(s = date); if (i && toupper(date[i-1])=='Z') { date[i-1] = 0; setenv("TZ", "UTC0", 1); } while (*format) { if (FLAG(t)) { s = strchr(date, '.'); if ((s ? s-date : strlen(date)) != strlen(*format)) { format++; continue; } } localtime_r(&(ts->tv_sec), &tm); // Adjusting for daylight savings time gives the wrong answer. tm.tm_isdst = 0; tm.tm_sec = 0; s = strptime(date, *format++, &tm); // parse nanoseconds if (s && *s=='.' && isdigit(s[1])) { s++; if (FLAG(t)) if (1 == sscanf(s, "%2u%n", &(tm.tm_sec), &len)) s += len; if (1 == sscanf(s, "%lu%n", &ts->tv_nsec, &len)) { s += len; if (ts->tv_nsec > 999999999) s = 0; else while (len++ < 9) ts->tv_nsec *= 10; } } if (s && !*s) break; } errno = 0; ts->tv_sec = mktime(&tm); if (!s || *s || ts->tv_sec == -1) perror_exit("bad '%s'", date); } ts[1]=ts[0]; // Set time from -r? if (TT.r) { struct stat st; xstat(TT.r, &st); ts[0] = st.st_atim; ts[1] = st.st_mtim; } // Which time(s) should we actually change? i = toys.optflags & (FLAG_a|FLAG_m); if (i && i!=(FLAG_a|FLAG_m)) ts[i!=FLAG_m].tv_nsec = UTIME_OMIT; // Loop through files on command line for (ss = toys.optargs; *ss;) { char *s = *ss++; if (!strcmp(s, "-")) { if (!futimens(1, ts)) continue; } else { // cheat: FLAG_h is rightmost flag, so its value is 1 if (!utimensat(AT_FDCWD, s, ts, FLAG(h)*AT_SYMLINK_NOFOLLOW)) continue; if (FLAG(c)) continue; if (access(s, F_OK) && (-1!=(fd = open(s, O_CREAT, 0666)))) { close(fd); if (toys.optflags) ss--; continue; } } perror_msg("'%s'", s); } }