xref: /aosp_15_r20/external/toybox/toys/posix/touch.c (revision cf5a6c84e2b8763fc1a7db14496fd4742913b199)
1*cf5a6c84SAndroid Build Coastguard Worker /* touch.c : change timestamp of a file
2*cf5a6c84SAndroid Build Coastguard Worker  *
3*cf5a6c84SAndroid Build Coastguard Worker  * Copyright 2012 Choubey Ji <[email protected]>
4*cf5a6c84SAndroid Build Coastguard Worker  *
5*cf5a6c84SAndroid Build Coastguard Worker  * See http://pubs.opengroup.org/onlinepubs/9699919799/utilities/touch.html
6*cf5a6c84SAndroid Build Coastguard Worker  *
7*cf5a6c84SAndroid Build Coastguard Worker  * -f is ignored for BSD/macOS compatibility. busybox/coreutils also support
8*cf5a6c84SAndroid Build Coastguard Worker  * this, but only coreutils documents it in --help output.
9*cf5a6c84SAndroid Build Coastguard Worker 
10*cf5a6c84SAndroid Build Coastguard Worker USE_TOUCH(NEWTOY(touch, "<1acd:fmr:t:h[!dtr]", TOYFLAG_BIN))
11*cf5a6c84SAndroid Build Coastguard Worker 
12*cf5a6c84SAndroid Build Coastguard Worker config TOUCH
13*cf5a6c84SAndroid Build Coastguard Worker   bool "touch"
14*cf5a6c84SAndroid Build Coastguard Worker   default y
15*cf5a6c84SAndroid Build Coastguard Worker   help
16*cf5a6c84SAndroid Build Coastguard Worker     usage: touch [-amch] [-d DATE] [-t TIME] [-r FILE] FILE...
17*cf5a6c84SAndroid Build Coastguard Worker 
18*cf5a6c84SAndroid Build Coastguard Worker     Update the access and modification times of each FILE to the current time.
19*cf5a6c84SAndroid Build Coastguard Worker 
20*cf5a6c84SAndroid Build Coastguard Worker     -a	Change access time
21*cf5a6c84SAndroid Build Coastguard Worker     -m	Change modification time
22*cf5a6c84SAndroid Build Coastguard Worker     -c	Don't create file
23*cf5a6c84SAndroid Build Coastguard Worker     -h	Change symlink
24*cf5a6c84SAndroid Build Coastguard Worker     -d	Set time to DATE (in YYYY-MM-DDThh:mm:SS[.frac][tz] format)
25*cf5a6c84SAndroid Build Coastguard Worker     -t	Set time to TIME (in [[CC]YY]MMDDhhmm[.ss][frac] format)
26*cf5a6c84SAndroid Build Coastguard Worker     -r	Set time same as reference FILE
27*cf5a6c84SAndroid Build Coastguard Worker */
28*cf5a6c84SAndroid Build Coastguard Worker 
29*cf5a6c84SAndroid Build Coastguard Worker #define FOR_touch
30*cf5a6c84SAndroid Build Coastguard Worker #include "toys.h"
31*cf5a6c84SAndroid Build Coastguard Worker 
32*cf5a6c84SAndroid Build Coastguard Worker GLOBALS(
33*cf5a6c84SAndroid Build Coastguard Worker   char *t, *r, *d;
34*cf5a6c84SAndroid Build Coastguard Worker )
35*cf5a6c84SAndroid Build Coastguard Worker 
touch_main(void)36*cf5a6c84SAndroid Build Coastguard Worker void touch_main(void)
37*cf5a6c84SAndroid Build Coastguard Worker {
38*cf5a6c84SAndroid Build Coastguard Worker   struct timespec ts[2];
39*cf5a6c84SAndroid Build Coastguard Worker   char **ss;
40*cf5a6c84SAndroid Build Coastguard Worker   int fd;
41*cf5a6c84SAndroid Build Coastguard Worker 
42*cf5a6c84SAndroid Build Coastguard Worker   // use current time if no -t or -d
43*cf5a6c84SAndroid Build Coastguard Worker   ts[0].tv_nsec = UTIME_NOW;
44*cf5a6c84SAndroid Build Coastguard Worker 
45*cf5a6c84SAndroid Build Coastguard Worker   if (FLAG(t) || FLAG(d)) {
46*cf5a6c84SAndroid Build Coastguard Worker     time_t t = time(0);
47*cf5a6c84SAndroid Build Coastguard Worker     unsigned nano;
48*cf5a6c84SAndroid Build Coastguard Worker 
49*cf5a6c84SAndroid Build Coastguard Worker     xparsedate(TT.t ? TT.t : TT.d, &t, &nano, 0);
50*cf5a6c84SAndroid Build Coastguard Worker     ts->tv_sec = t;
51*cf5a6c84SAndroid Build Coastguard Worker     ts->tv_nsec = nano;
52*cf5a6c84SAndroid Build Coastguard Worker   }
53*cf5a6c84SAndroid Build Coastguard Worker   ts[1]=ts[0];
54*cf5a6c84SAndroid Build Coastguard Worker 
55*cf5a6c84SAndroid Build Coastguard Worker   if (TT.r) {
56*cf5a6c84SAndroid Build Coastguard Worker     struct stat st;
57*cf5a6c84SAndroid Build Coastguard Worker 
58*cf5a6c84SAndroid Build Coastguard Worker     xstat(TT.r, &st);
59*cf5a6c84SAndroid Build Coastguard Worker     ts[0] = st.st_atim;
60*cf5a6c84SAndroid Build Coastguard Worker     ts[1] = st.st_mtim;
61*cf5a6c84SAndroid Build Coastguard Worker   }
62*cf5a6c84SAndroid Build Coastguard Worker 
63*cf5a6c84SAndroid Build Coastguard Worker   // Which time(s) should we actually change?
64*cf5a6c84SAndroid Build Coastguard Worker   if (FLAG(a)^FLAG(m)) ts[!FLAG(m)].tv_nsec = UTIME_OMIT;
65*cf5a6c84SAndroid Build Coastguard Worker 
66*cf5a6c84SAndroid Build Coastguard Worker   // Loop through files on command line
67*cf5a6c84SAndroid Build Coastguard Worker   for (ss = toys.optargs; *ss;) {
68*cf5a6c84SAndroid Build Coastguard Worker     char *s = *ss++;
69*cf5a6c84SAndroid Build Coastguard Worker 
70*cf5a6c84SAndroid Build Coastguard Worker     if (!strcmp(s, "-")) {
71*cf5a6c84SAndroid Build Coastguard Worker       if (!futimens(1, ts)) continue;
72*cf5a6c84SAndroid Build Coastguard Worker     } else {
73*cf5a6c84SAndroid Build Coastguard Worker       if (!utimensat(AT_FDCWD, s, ts, FLAG(h)*AT_SYMLINK_NOFOLLOW)) continue;
74*cf5a6c84SAndroid Build Coastguard Worker       if (FLAG(c)) continue;
75*cf5a6c84SAndroid Build Coastguard Worker       if (access(s, F_OK) && (-1!=(fd = open(s, O_CREAT, 0666)))) {
76*cf5a6c84SAndroid Build Coastguard Worker         close(fd);
77*cf5a6c84SAndroid Build Coastguard Worker         if (toys.optflags) ss--;
78*cf5a6c84SAndroid Build Coastguard Worker         continue;
79*cf5a6c84SAndroid Build Coastguard Worker       }
80*cf5a6c84SAndroid Build Coastguard Worker     }
81*cf5a6c84SAndroid Build Coastguard Worker     perror_msg("'%s'", s);
82*cf5a6c84SAndroid Build Coastguard Worker   }
83*cf5a6c84SAndroid Build Coastguard Worker }
84