xref: /illumos-gate/usr/src/cmd/date/date.c (revision a288e5a9793fdffe5e842d7e61ab45263e75eaca)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License, Version 1.0 only
6  * (the "License").  You may not use this file except in compliance
7  * with the License.
8  *
9  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
10  * or http://www.opensolaris.org/os/licensing.
11  * See the License for the specific language governing permissions
12  * and limitations under the License.
13  *
14  * When distributing Covered Code, include this CDDL HEADER in each
15  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
16  * If applicable, add the following below this CDDL HEADER, with the
17  * fields enclosed by brackets "[]" replaced with your own identifying
18  * information: Portions Copyright [yyyy] [name of copyright owner]
19  *
20  * CDDL HEADER END
21  */
22 /*
23  * Copyright 2004 Sun Microsystems, Inc.  All rights reserved.
24  * Use is subject to license terms.
25  */
26 /*
27  * Copyright 2012 Nexenta Systems, Inc. All rights reserved.
28  */
29 
30 /*	Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T	*/
31 /*	  All Rights Reserved  	*/
32 
33 /*
34  * University Copyright- Copyright (c) 1982, 1986, 1988
35  * The Regents of the University of California
36  * All Rights Reserved
37  *
38  * University Acknowledgment- Portions of this document are derived from
39  * software developed by the University of California, Berkeley, and its
40  * contributors.
41  */
42 
43 /*
44  *	date - with format capabilities and international flair
45  */
46 
47 #include	<locale.h>
48 #include	<fcntl.h>
49 #include	<langinfo.h>
50 #include	<stdio.h>
51 #include	<stdlib.h>
52 #include	<string.h>
53 #include	<time.h>
54 #include	<unistd.h>
55 #include	<sys/time.h>
56 #include	<sys/types.h>
57 #include	<ctype.h>
58 #include	<utmpx.h>
59 #include	<tzfile.h>
60 
61 #define	year_size(A)	((isleap(A)) ? 366 : 365)
62 static 	char	buf[BUFSIZ];
63 static	time_t	clock_val;
64 static  short	month_size[12] =
65 	{ 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 };
66 static  struct  utmpx wtmpx[2] = {
67 	{"", "", OTIME_MSG, 0, OLD_TIME, 0, 0, 0},
68 	{"", "", NTIME_MSG, 0, NEW_TIME, 0, 0, 0}
69 	};
70 static char *usage =
71 	"usage:\tdate [-u] mmddHHMM[[cc]yy][.SS]\n\tdate [-Ru] [+format]\n"
72 	"\tdate -a [-]sss[.fff]\n";
73 static int uflag = 0;
74 static int Rflag = 0;
75 
76 static int get_adj(char *, struct timeval *);
77 static int setdate(struct tm *, char *);
78 
79 int
80 main(int argc, char **argv)
81 {
82 	struct tm *tp, tm;
83 	struct timeval tv;
84 	char *fmt;
85 	int c, aflag = 0, illflag = 0;
86 
87 	(void) setlocale(LC_ALL, "");
88 
89 #if !defined(TEXT_DOMAIN)
90 #define	TEXT_DOMAIN "SYS_TEST"
91 #endif
92 	(void) textdomain(TEXT_DOMAIN);
93 
94 	while ((c = getopt(argc, argv, "a:uR")) != EOF)
95 		switch (c) {
96 		case 'a':
97 			aflag++;
98 			if (get_adj(optarg, &tv) < 0) {
99 				(void) fprintf(stderr,
100 				    gettext("date: invalid argument -- %s\n"),
101 				    optarg);
102 				illflag++;
103 			}
104 			break;
105 		case 'u':
106 			uflag++;
107 			break;
108 		case 'R':
109 			Rflag++;
110 			break;
111 		default:
112 			illflag++;
113 		}
114 
115 	argc -= optind;
116 	argv  = &argv[optind];
117 
118 	/* -a is mutually exclusive with -u and -R */
119 	if (uflag && aflag)
120 		illflag++;
121 	if (Rflag && aflag)
122 		illflag++;
123 
124 	if (illflag) {
125 		(void) fprintf(stderr, gettext(usage));
126 		exit(1);
127 	}
128 
129 	(void) time(&clock_val);
130 
131 	if (aflag) {
132 		if (adjtime(&tv, 0) < 0) {
133 			perror(gettext("date: Failed to adjust date"));
134 			exit(1);
135 		}
136 		exit(0);
137 	}
138 
139 	if (argc > 0) {
140 		if (*argv[0] == '+')
141 			fmt = &argv[0][1];
142 		else {
143 			if (setdate(localtime(&clock_val), argv[0])) {
144 				(void) fprintf(stderr, gettext(usage));
145 				exit(1);
146 			}
147 			fmt = nl_langinfo(_DATE_FMT);
148 		}
149 	} else if (Rflag) {
150 		fmt = "%a, %d %h %Y %H:%M:%S %z";
151 	} else
152 		fmt = nl_langinfo(_DATE_FMT);
153 
154 	if (uflag) {
155 		(void) putenv("TZ=GMT0");
156 		tzset();
157 		tp = gmtime(&clock_val);
158 	} else
159 		tp = localtime(&clock_val);
160 	(void) memcpy(&tm, tp, sizeof (struct tm));
161 	(void) strftime(buf, BUFSIZ, fmt, &tm);
162 
163 	(void) puts(buf);
164 
165 	return (0);
166 }
167 
168 int
169 setdate(struct tm *current_date, char *date)
170 {
171 	int	i;
172 	int	mm;
173 	int	hh;
174 	int	min;
175 	int	sec = 0;
176 	char	*secptr;
177 	int	yy;
178 	int	dd	= 0;
179 	int	minidx	= 6;
180 	int	len;
181 	int	dd_check;
182 
183 	/*  Parse date string  */
184 	if ((secptr = strchr(date, '.')) != NULL && strlen(&secptr[1]) == 2 &&
185 	    isdigit(secptr[1]) && isdigit(secptr[2]) &&
186 	    (sec = atoi(&secptr[1])) >= 0 && sec < 60)
187 		secptr[0] = '\0';	/* eat decimal point only on success */
188 
189 	len = strlen(date);
190 
191 	for (i = 0; i < len; i++) {
192 		if (!isdigit(date[i])) {
193 			(void) fprintf(stderr,
194 			gettext("date: bad conversion\n"));
195 			exit(1);
196 		}
197 	}
198 	switch (strlen(date)) {
199 	case 12:
200 		yy = atoi(&date[8]);
201 		date[8] = '\0';
202 		break;
203 	case 10:
204 		/*
205 		 * The YY format has the following representation:
206 		 * 00-68 = 2000 thru 2068
207 		 * 69-99 = 1969 thru 1999
208 		 */
209 		if (atoi(&date[8]) <= 68) {
210 			yy = 1900 + (atoi(&date[8]) + 100);
211 		} else {
212 			yy = 1900 + atoi(&date[8]);
213 		}
214 		date[8] = '\0';
215 		break;
216 	case 8:
217 		yy = 1900 + current_date->tm_year;
218 		break;
219 	case 4:
220 		yy = 1900 + current_date->tm_year;
221 		mm = current_date->tm_mon + 1; 	/* tm_mon goes from 1 to 11 */
222 		dd = current_date->tm_mday;
223 		minidx = 2;
224 		break;
225 	default:
226 		(void) fprintf(stderr, gettext("date: bad conversion\n"));
227 		return (1);
228 	}
229 
230 	min = atoi(&date[minidx]);
231 	date[minidx] = '\0';
232 	hh = atoi(&date[minidx-2]);
233 	date[minidx-2] = '\0';
234 
235 	if (!dd) {
236 		/*
237 		 * if dd is 0 (not between 1 and 31), then
238 		 * read the value supplied by the user.
239 		 */
240 		dd = atoi(&date[2]);
241 		date[2] = '\0';
242 		mm = atoi(&date[0]);
243 	}
244 
245 	if (hh == 24)
246 		hh = 0, dd++;
247 
248 	/*  Validate date elements  */
249 	dd_check = 0;
250 	if (mm >= 1 && mm <= 12) {
251 		dd_check = month_size[mm - 1];	/* get days in this month */
252 		if (mm == 2 && isleap(yy))	/* adjust for leap year */
253 			dd_check++;
254 	}
255 	if (!((mm >= 1 && mm <= 12) && (dd >= 1 && dd <= dd_check) &&
256 	    (hh >= 0 && hh <= 23) && (min >= 0 && min <= 59))) {
257 		(void) fprintf(stderr, gettext("date: bad conversion\n"));
258 		return (1);
259 	}
260 
261 	/*  Build date and time number  */
262 	for (clock_val = 0, i = 1970; i < yy; i++)
263 		clock_val += year_size(i);
264 	/*  Adjust for leap year  */
265 	if (isleap(yy) && mm >= 3)
266 		clock_val += 1;
267 	/*  Adjust for different month lengths  */
268 	while (--mm)
269 		clock_val += (time_t)month_size[mm - 1];
270 	/*  Load up the rest  */
271 	clock_val += (time_t)(dd - 1);
272 	clock_val *= 24;
273 	clock_val += (time_t)hh;
274 	clock_val *= 60;
275 	clock_val += (time_t)min;
276 	clock_val *= 60;
277 	clock_val += sec;
278 
279 	if (!uflag) {
280 		/* convert to GMT assuming standard time */
281 		/* correction is made in localtime(3C) */
282 
283 		/*
284 		 * call localtime to set up "timezone" variable applicable
285 		 * for clock_val time, to support Olson timezones which
286 		 * can allow timezone rules to change.
287 		 */
288 		(void) localtime(&clock_val);
289 
290 		clock_val += (time_t)timezone;
291 
292 		/* correct if daylight savings time in effect */
293 
294 		if (localtime(&clock_val)->tm_isdst)
295 			clock_val = clock_val - (time_t)(timezone - altzone);
296 	}
297 
298 	(void) time(&wtmpx[0].ut_xtime);
299 	if (stime(&clock_val) < 0) {
300 		perror("date");
301 		return (1);
302 	}
303 #if defined(i386)
304 	/* correct the kernel's "gmt_lag" and the PC's RTC */
305 	(void) system("/usr/sbin/rtc -c > /dev/null 2>&1");
306 #endif
307 	(void) time(&wtmpx[1].ut_xtime);
308 	(void) pututxline(&wtmpx[0]);
309 	(void) pututxline(&wtmpx[1]);
310 	(void) updwtmpx(WTMPX_FILE, &wtmpx[0]);
311 	(void) updwtmpx(WTMPX_FILE, &wtmpx[1]);
312 	return (0);
313 }
314 
315 int
316 get_adj(char *cp, struct timeval *tp)
317 {
318 	register int mult;
319 	int sign;
320 
321 	/* arg must be [-]sss[.fff] */
322 
323 	tp->tv_sec = tp->tv_usec = 0;
324 	if (*cp == '-') {
325 		sign = -1;
326 		cp++;
327 	} else {
328 		sign = 1;
329 	}
330 
331 	while (*cp >= '0' && *cp <= '9') {
332 		tp->tv_sec *= 10;
333 		tp->tv_sec += *cp++ - '0';
334 	}
335 	if (*cp == '.') {
336 		cp++;
337 		mult = 100000;
338 		while (*cp >= '0' && *cp <= '9') {
339 			tp->tv_usec += (*cp++ - '0') * mult;
340 			mult /= 10;
341 		}
342 	}
343 	/*
344 	 * if there's anything left in the string,
345 	 * the input was invalid.
346 	 */
347 	if (*cp) {
348 		return (-1);
349 	} else {
350 		tp->tv_sec *= sign;
351 		tp->tv_usec *= sign;
352 		return (0);
353 	}
354 }
355