/*
 * date - write the date and time
 *
 * Gunnar Ritter, Freiburg i. Br., Germany, June 2002.
 */
/*
 * Copyright (c) 2003 Gunnar Ritter
 *
 * This software is provided 'as-is', without any express or implied
 * warranty. In no event will the authors be held liable for any damages
 * arising from the use of this software.
 *
 * Permission is granted to anyone to use this software for any purpose,
 * including commercial applications, and to alter it and redistribute
 * it freely, subject to the following restrictions:
 *
 * 1. The origin of this software must not be misrepresented; you must not
 *    claim that you wrote the original software. If you use this software
 *    in a product, an acknowledgment in the product documentation would be
 *    appreciated but is not required.
 *
 * 2. Altered source versions must be plainly marked as such, and must not be
 *    misrepresented as being the original software.
 *
 * 3. This notice may not be removed or altered from any source distribution.
 */

#if __GNUC__ >= 3 && __GNUC_MINOR__ >= 4 || __GNUC__ >= 4
#define	USED	__attribute__ ((used))
#elif defined __GNUC__
#define	USED	__attribute__ ((unused))
#else
#define	USED
#endif
#if defined (SUS)
static const char sccsid[] USED = "@(#)date_sus.sl	1.26 (gritter) 1/22/06";
#else
static const char sccsid[] USED = "@(#)date.sl	1.26 (gritter) 1/22/06";
#endif

#include	<unistd.h>
#include	<stdio.h>
#include	<string.h>
#include	<stdlib.h>
#include	<errno.h>
#include	<libgen.h>
#include	<time.h>
#include	<locale.h>
#include	<utmpx.h>
#include	<langinfo.h>

static unsigned	errcnt;			/* count of errors */
static int	bflag;			/* boot flag: don't write wtmp */
static int	uflag;			/* use UTC */
static char	*progname;		/* argv[0] to main() */
static time_t	now;			/* current time */

static void
usage(void)
{
	fprintf(stderr,
	"usage: %s [-u] [+format] [hhmm | mmddhhmm[[cc]yy]] [-a sss.fff]\n",
		progname);
	exit(1);
}

static void
badconv(void)
{
	fprintf(stderr, "%s: bad conversion\n", progname);
	exit(1);
}

static int
atot(char *s, int m)
{
	char *x;
	int i;

	i = (int)strtol(s, &x, 10);
	if (*x != '\0' || i > m || i < 0 || *s == '-' || *s == '+')
		return -1;
	return i;
}

static time_t
timeop(char *op)
{
	struct tm *tp;

	tp = uflag ? gmtime(&now) : localtime(&now);
	tp->tm_isdst = -1;
	tp->tm_sec = 0;
	switch(strlen(op)) {
	case 12:
		if ((tp->tm_year = atot(&op[8], 30000)) < 1970)
			goto badtime;
		tp->tm_year -= 1900;
		goto bypass;
	case 10:
		if ((tp->tm_year = atot(&op[8], 99)) < 0)
			goto badtime;
		if (tp->tm_year < 69)
			tp->tm_year += 100;
bypass:
		op[8] = '\0';
		/*FALLTHRU*/
	case 8:
		if ((tp->tm_min = atot(&op[6], 59)) < 0)
			goto badtime;
		op[6] = '\0';
		if ((tp->tm_hour = atot(&op[4], 23)) < 0)
			goto badtime;
		op[4] = '\0';
		if ((tp->tm_mday = atot(&op[2], 31)) <= 0)
			goto badtime;
		op[2] = '\0';
		if ((tp->tm_mon = atot(op, 12)) <= 0)
			goto badtime;
		tp->tm_mon--;
		break;
	case 4:
		if ((tp->tm_min = atot(&op[2], 59)) < 0)
			goto badtime;
		op[2] = '\0';
		if ((tp->tm_hour = atot(op, 23)) < 0)
			goto badtime;
		break;
	default:
		goto badtime;
	}
	return mktime(tp);
badtime:
	return (time_t)-1;
}

#if defined (__FreeBSD__) || defined (__NetBSD__) || defined (__OpenBSD__) || \
	defined (__DragonFly__) || defined (__APPLE__)
static int
stime(time_t *t)
{
	struct timeval	tv;

	tv.tv_sec = *t;
	tv.tv_usec = 0;

	return settimeofday(&tv, NULL);
}
#endif	/* __FreeBSD__, __NetBSD__, __OpenBSD__, __DragonFly__, __APPLE__ */

static void
settime(char *op)
{
	struct utmpx before, after;
	const char wtmpxfile[] = "/var/log/wtmp";
	time_t newtime;

	memset(&before, 0, sizeof before);
	memset(&after, 0, sizeof after);
	before.ut_type = OLD_TIME;
	after.ut_type = NEW_TIME;
	strcpy(before.ut_line, "old time");
	strcpy(after.ut_line, "new time");
	if ((newtime = timeop(op)) == (time_t)-1)
		badconv();
	gettimeofday(&before.ut_tv, NULL);
	if (stime(&newtime) < 0) {
		fprintf(stderr, "%s: no permission\n", progname);
		exit(1);
	}
	gettimeofday(&after.ut_tv, NULL);
#ifdef	__linux__
	system("/sbin/hwclock -w >/dev/null 2>&1");
#endif	/* __linux__ */
	if (bflag == 0) {
		pututxline(&before);
		pututxline(&after);
#if !defined (__hpux) && !defined (_AIX)
		updwtmpx(wtmpxfile, &before);
		updwtmpx(wtmpxfile, &after);
#endif	/* !__hpux, !__AIX */
	}
	exit(0);
}

static void
printtime(const char *cp)
{
	struct tm *tp;
	char fmt[256];
	register char *fp;
	char buf[256];
	int mod;
	char	*date_fmt;

	tp = uflag ? gmtime(&now) : localtime(&now);
	while (*cp != '\0') {
		if (*cp == '%') {
			const char	valid[] = "aAbBc"
#ifdef	SUS
					"C"
#endif	/* SUS */
					"dDeFgGhHIjmMnprRStTuUVwWxXyYZ%";
			fp = fmt;
			*fp++ = *cp++;
			if (*cp == 'E' || *cp == 'O')
				mod = *cp++;
			else
				mod = '\0';
			if (*cp == '\0') {
				putchar('%');
				goto end;
			}
			if (!strchr(valid, *cp)) {
				switch (*cp) {
				case 'N':
#ifndef	SUS
				case 'C':
#endif	/* !SUS */
#ifdef	_DATE_FMT
					if ((date_fmt = nl_langinfo(_DATE_FMT))
							== NULL ||
							*date_fmt == '\0')
#endif	/* _DATE_FMT */
						date_fmt =
						"%a %b %e %H:%M:%S %Z %Y";
					strcpy(fmt, date_fmt);
					break;
				case 'E':
				case 'O':
					goto next;
				case 'z': {
						  int hour, min;
#if defined (__FreeBSD__) || defined (__OpenBSD__) || defined (__DragonFly__) \
	|| defined (__APPLE__)
						  long	timezone;
						  timezone = -tp->tm_gmtoff;
#endif	/* __FreeBSD__, __OpenBSD__, __DragonFly__, __APPLE__ */

						  hour = -timezone / 3600;
						  min = (timezone / 60) % 60;
						  if (tp->tm_isdst > 0)
							  hour++;
						  fp = fmt;
						  if (hour <= 0) {
							  *fp++ = '-';
							  hour = -hour;
						  }
						  sprintf(fp, "%02d%02d",
								  hour, min);
					  }
					break;
				default:
					*fp++ = '%';
					if (mod)
						*fp++ = mod;
					*fp++ = *cp;
					*fp++ = '\0';
				}
			} else {
				if ((mod == 'E' && strchr("cCxXyY", *cp)) ||
						(mod == 'O' &&
						 strchr("deHImMSuUVwWy", *cp)))
					*fp++ = mod;
				*fp++ = *cp;
				*fp = '\0';
			}
			strftime(buf, sizeof buf, fmt, tp);
			fputs(buf, stdout);
		} else
			putchar(*cp);
next:
		cp++;
	}
end:
	putchar('\n');
}

static void
adjust(char *op)
{
	struct timeval tv;
	char *cp;

	if ((cp = strchr(op, '.')) != NULL) {
		if ((tv.tv_usec = atot(&cp[1], 999)) < 0)
			usage();
		tv.tv_usec *= 1000;
		*cp = '\0';
	}
	if ((tv.tv_sec = atot(*op == '-' ? &op[1] : op, 999)) < 0)
		usage();
	if (*op == '-') {
		tv.tv_sec = -tv.tv_sec;
		tv.tv_usec = -tv.tv_usec;
	}
	if (adjtime(&tv, 0) < 0) {
		fprintf(stderr, "%s: Failed to adjust date: %s\n", progname,
				strerror(errno));
		exit(1);
	}
	return;
}

int
main(int argc, char **argv)
{
	const char optstring[] = ":a:bu";
	char *adjustment = NULL;
	int i;

	time(&now);
	progname = basename(argv[0]);
	setlocale(LC_TIME, "");
#ifdef	__GLIBC__
	putenv("POSIXLY_CORRECT=1");
#endif
	while ((i = getopt(argc, argv, optstring)) != EOF) {
		switch (i) {
		case 'a':
			adjustment = optarg;
			break;
		case 'b':
			bflag = 1;
			break;
		case 'u':
			uflag = 1;
			putenv("TZ=GMT");
			tzset();
			break;
		case ':':
			usage();
			/*NOTREACHED*/
		default:
			/*fprintf(stderr, "%s: no TOY clock\n", progname);
			exit(2);*/
			badconv();
		}
	}
	if (adjustment) {
		if (uflag)
			badconv();
		adjust(adjustment);
	} else if (argv[optind] && argv[optind][0] == '+') {
		if (argv[optind][1] == '\0')
			goto dfl;
		printtime(&argv[optind][1]);
	} else if (argv[optind])
		settime(argv[optind]);
	else
	dfl:	printtime("%N");
	return errcnt;
}


syntax highlighted by Code2HTML, v. 0.9.1