blob: 03aac3aa2f8f9a01cc8ceadb29138cf3d46c889f [file] [log] [blame]
Erik Andersene49d5ec2000-02-08 19:58:47 +00001/* vi: set sw=4 ts=4: */
Eric Andersen596e5461999-10-07 08:30:23 +00002/*
3 * Mini touch implementation for busybox
4 *
Eric Andersenc7bda1c2004-03-15 08:29:22 +00005 * Copyright (C) 1999-2004 by Erik Andersen <andersen@codepoet.org>
Eric Andersen596e5461999-10-07 08:30:23 +00006 *
Denys Vlasenko0ef64bd2010-08-16 20:14:46 +02007 * Licensed under GPLv2 or later, see file LICENSE in this source tree.
Eric Andersen596e5461999-10-07 08:30:23 +00008 */
Eric Andersencc8ed391999-10-05 16:24:54 +00009
Manuel Novoa III cad53642003-03-19 09:13:01 +000010/* BB_AUDIT SUSv3 _NOT_ compliant -- options -a, -m, -r, -t not supported. */
11/* http://www.opengroup.org/onlinepubs/007904975/utilities/touch.html */
12
13/* Mar 16, 2003 Manuel Novoa III (mjn3@codepoet.org)
14 *
15 * Previous version called open() and then utime(). While this will be
16 * be necessary to implement -r and -t, it currently only makes things bigger.
17 * Also, exiting on a failure was a bug. All args should be processed.
18 */
19
Denis Vlasenkob6adbf12007-05-26 19:00:18 +000020#include "libbb.h"
Eric Andersen596e5461999-10-07 08:30:23 +000021
Denis Vlasenko3f3aa2a2007-04-09 21:35:07 +000022/* This is a NOFORK applet. Be very careful! */
23
Denis Vlasenkoed90bda2008-06-28 01:18:09 +000024/* coreutils implements:
25 * -a change only the access time
26 * -c, --no-create
27 * do not create any files
28 * -d, --date=STRING
29 * parse STRING and use it instead of current time
30 * -f (ignored, BSD compat)
31 * -m change only the modification time
32 * -r, --reference=FILE
33 * use this file's times instead of current time
34 * -t STAMP
35 * use [[CC]YY]MMDDhhmm[.ss] instead of current time
36 * --time=WORD
37 * change the specified time: WORD is access, atime, or use
38 */
39
Denis Vlasenko9b49a5e2007-10-11 10:05:36 +000040int touch_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
Denis Vlasenkoa60f84e2008-07-05 09:18:54 +000041int touch_main(int argc UNUSED_PARAM, char **argv)
Eric Andersencc8ed391999-10-05 16:24:54 +000042{
Denys Vlasenko389cca42009-11-15 02:28:56 +010043 int fd;
44 int status = EXIT_SUCCESS;
45 int opts;
Denis Vlasenkoed90bda2008-06-28 01:18:09 +000046#if ENABLE_DESKTOP
Denys Vlasenko7aca89a2009-07-18 03:41:29 +020047# if ENABLE_LONG_OPTS
48 static const char touch_longopts[] ALIGN1 =
Denis Vlasenko7241e6d2009-03-15 01:28:30 +000049 /* name, has_arg, val */
50 "no-create\0" No_argument "c"
51 "reference\0" Required_argument "r"
Denys Vlasenko7aca89a2009-07-18 03:41:29 +020052 "date\0" Required_argument "d"
Denis Vlasenko7241e6d2009-03-15 01:28:30 +000053 ;
Denys Vlasenko7aca89a2009-07-18 03:41:29 +020054# endif
Denis Vlasenkoed90bda2008-06-28 01:18:09 +000055 char *reference_file = NULL;
Denys Vlasenko7aca89a2009-07-18 03:41:29 +020056 char *date_str = NULL;
Denys Vlasenkodcbfaba2009-11-29 19:40:36 +010057 struct timeval timebuf[2];
58 timebuf[1].tv_usec = timebuf[0].tv_usec = 0;
Denis Vlasenkoed90bda2008-06-28 01:18:09 +000059#else
Denys Vlasenko7aca89a2009-07-18 03:41:29 +020060# define reference_file NULL
61# define date_str NULL
Denys Vlasenkod0f9d0e2009-11-30 11:36:14 +010062# define timebuf ((struct timeval*)NULL)
Denis Vlasenkoed90bda2008-06-28 01:18:09 +000063#endif
Denis Vlasenko7241e6d2009-03-15 01:28:30 +000064
Denys Vlasenko7aca89a2009-07-18 03:41:29 +020065#if ENABLE_DESKTOP && ENABLE_LONG_OPTS
66 applet_long_options = touch_longopts;
Denis Vlasenko7241e6d2009-03-15 01:28:30 +000067#endif
Denys Vlasenko38dd8aa2009-07-18 04:49:20 +020068 /* -d and -t both set time. In coreutils,
69 * accepted data format differs a bit between -d and -t.
70 * We accept the same formats for both */
71 opts = getopt32(argv, "c" IF_DESKTOP("r:d:t:")
Denis Vlasenkoed90bda2008-06-28 01:18:09 +000072 /*ignored:*/ "fma"
Denys Vlasenko7aca89a2009-07-18 03:41:29 +020073 IF_DESKTOP(, &reference_file)
74 IF_DESKTOP(, &date_str)
Denys Vlasenko38dd8aa2009-07-18 04:49:20 +020075 IF_DESKTOP(, &date_str)
Denys Vlasenko7aca89a2009-07-18 03:41:29 +020076 );
Manuel Novoa III cad53642003-03-19 09:13:01 +000077
Denis Vlasenko7241e6d2009-03-15 01:28:30 +000078 opts &= 1; /* only -c bit is left */
Manuel Novoa III cad53642003-03-19 09:13:01 +000079 argv += optind;
Manuel Novoa III cad53642003-03-19 09:13:01 +000080 if (!*argv) {
81 bb_show_usage();
Eric Andersen5a0a2aa2000-06-02 23:26:44 +000082 }
83
Denis Vlasenkoed90bda2008-06-28 01:18:09 +000084 if (reference_file) {
85 struct stat stbuf;
86 xstat(reference_file, &stbuf);
Denys Vlasenkodcbfaba2009-11-29 19:40:36 +010087 timebuf[1].tv_sec = timebuf[0].tv_sec = stbuf.st_mtime;
Denis Vlasenkoed90bda2008-06-28 01:18:09 +000088 }
89
Denys Vlasenko7aca89a2009-07-18 03:41:29 +020090 if (date_str) {
91 struct tm tm_time;
92 time_t t;
93
94 //time(&t);
95 //localtime_r(&t, &tm_time);
96 memset(&tm_time, 0, sizeof(tm_time));
97 parse_datestr(date_str, &tm_time);
98
99 /* Correct any day of week and day of year etc. fields */
100 tm_time.tm_isdst = -1; /* Be sure to recheck dst */
101 t = validate_tm_time(date_str, &tm_time);
102
Denys Vlasenkodcbfaba2009-11-29 19:40:36 +0100103 timebuf[1].tv_sec = timebuf[0].tv_sec = t;
Denys Vlasenko7aca89a2009-07-18 03:41:29 +0200104 }
105
Manuel Novoa III cad53642003-03-19 09:13:01 +0000106 do {
Mikhail Gusarov927e4bb2010-03-21 14:22:47 +0600107 if (utimes(*argv, (reference_file || date_str) ? timebuf : NULL) != 0) {
Denis Vlasenkoed90bda2008-06-28 01:18:09 +0000108 if (errno == ENOENT) { /* no such file */
Denis Vlasenko7241e6d2009-03-15 01:28:30 +0000109 if (opts) { /* creation is disabled, so ignore */
Manuel Novoa III cad53642003-03-19 09:13:01 +0000110 continue;
111 }
Denys Vlasenkod0f9d0e2009-11-30 11:36:14 +0100112 /* Try to create the file */
113 fd = open(*argv, O_RDWR | O_CREAT, 0666);
114 if (fd >= 0) {
115 xclose(fd);
Mikhail Gusarov927e4bb2010-03-21 14:22:47 +0600116 if (reference_file || date_str)
Denys Vlasenkodcbfaba2009-11-29 19:40:36 +0100117 utimes(*argv, timebuf);
Manuel Novoa III cad53642003-03-19 09:13:01 +0000118 continue;
119 }
120 }
121 status = EXIT_FAILURE;
Denis Vlasenko0c97c9d2007-10-01 11:58:38 +0000122 bb_simple_perror_msg(*argv);
Eric Andersen5a0a2aa2000-06-02 23:26:44 +0000123 }
Manuel Novoa III cad53642003-03-19 09:13:01 +0000124 } while (*++argv);
Erik Andersene49d5ec2000-02-08 19:58:47 +0000125
Manuel Novoa III cad53642003-03-19 09:13:01 +0000126 return status;
Eric Andersencc8ed391999-10-05 16:24:54 +0000127}