#include <ctype.h>
#include <errno.h>
#include <fcntl.h>
#include <inttypes.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <time.h>
#include <unistd.h>
#include <linux/android_alarm.h>
#include <linux/rtc.h>
#include <sys/ioctl.h>
static int settime_alarm(struct timespec *ts) {
int fd, ret;
fd = open("/dev/alarm", O_RDWR);
if (fd < 0)
return fd;
ret = ioctl(fd, ANDROID_ALARM_SET_RTC, ts);
close(fd);
return ret;
}
static int settime_alarm_tm(struct tm *tm) {
time_t t;
struct timespec ts;
t = mktime(tm);
ts.tv_sec = t;
ts.tv_nsec = 0;
return settime_alarm(&ts);
}
static int settime_alarm_timeval(struct timeval *tv) {
struct timespec ts;
ts.tv_sec = tv->tv_sec;
ts.tv_nsec = tv->tv_usec * 1000;
return settime_alarm(&ts);
}
static int settime_rtc_tm(struct tm *tm) {
int fd, ret;
struct timeval tv;
struct rtc_time rtc;
fd = open("/dev/rtc0", O_RDWR);
if (fd < 0)
return fd;
tv.tv_sec = mktime(tm);
tv.tv_usec = 0;
ret = settimeofday(&tv, NULL);
if (ret < 0)
goto done;
memset(&rtc, 0, sizeof(rtc));
rtc.tm_sec = tm->tm_sec;
rtc.tm_min = tm->tm_min;
rtc.tm_hour = tm->tm_hour;
rtc.tm_mday = tm->tm_mday;
rtc.tm_mon = tm->tm_mon;
rtc.tm_year = tm->tm_year;
rtc.tm_wday = tm->tm_wday;
rtc.tm_yday = tm->tm_yday;
rtc.tm_isdst = tm->tm_isdst;
ret = ioctl(fd, RTC_SET_TIME, rtc);
done:
close(fd);
return ret;
}
static int settime_rtc_timeval(struct timeval *tv) {
struct tm tm, *err;
time_t t = tv->tv_sec;
err = gmtime_r(&t, &tm);
if (!err)
return -1;
return settime_rtc_tm(&tm);
}
static void settime(char *s) {
struct tm tm;
int day = atoi(s);
int hour;
while (*s && *s != '.')
s++;
if (*s)
s++;
hour = atoi(s);
tm.tm_year = day / 10000 - 1900;
tm.tm_mon = (day % 10000) / 100 - 1;
tm.tm_mday = (day % 100);
tm.tm_hour = hour / 10000;
tm.tm_min = (hour % 10000) / 100;
tm.tm_sec = (hour % 100);
tm.tm_isdst = -1;
if (settime_alarm_tm(&tm) < 0)
settime_rtc_tm(&tm);
}
static char *parse_time(const char *str, struct timeval *ts) {
char *s;
long fs = 0; /* fractional seconds */
ts->tv_sec = strtoumax(str, &s, 10);
if (*s == '.') {
s++;
int count = 0;
/* read up to 6 digits (microseconds) */
while (*s && isdigit(*s)) {
if (++count < 7) {
fs = fs*10 + (*s - '0');
}
s++;
}
for (; count < 6; count++) {
fs *= 10;
}
}
ts->tv_usec = fs;
return s;
}
int date_main(int argc, char *argv[])
{
int c;
int res;
struct tm tm;
time_t t;
struct timeval tv;
char strbuf[260];
int useutc = 0;
tzset();
do {
c = getopt(argc, argv, "us:");
if (c == EOF)
break;
switch (c) {
case 'u':
useutc = 1;
break;
case 's':
settime(optarg);
break;
case '?':
fprintf(stderr, "%s: invalid option -%c\n",
argv[0], optopt);
exit(1);
}
} while (1);
if(optind + 2 < argc) {
fprintf(stderr,"%s [-u] [date]\n", argv[0]);
return 1;
}
int hasfmt = argc == optind + 1 && argv[optind][0] == '+';
if(optind == argc || hasfmt) {
time(&t);
if (useutc) {
gmtime_r(&t, &tm);
strftime(strbuf, sizeof(strbuf),
(hasfmt ? argv[optind] + 1 : "%a %b %e %H:%M:%S GMT %Y"),
&tm);
} else {
localtime_r(&t, &tm);
strftime(strbuf, sizeof(strbuf),
(hasfmt ? argv[optind] + 1 : "%a %b %e %H:%M:%S %Z %Y"),
&tm);
}
printf("%s\n", strbuf);
}
else if(optind + 1 == argc) {
#if 0
struct tm *tmptr;
tmptr = getdate(argv[optind]);
if(tmptr == NULL) {
fprintf(stderr,"getdate_r failed\n");
return 1;
}
tm = *tmptr;
#if 0
if(getdate_r(argv[optind], &tm) < 0) {
fprintf(stderr,"getdate_r failed %s\n", strerror(errno));
return 1;
}
#endif
#endif
//strptime(argv[optind], NULL, &tm);
//tv.tv_sec = mktime(&tm);
//tv.tv_usec = 0;
parse_time(argv[optind], &tv);
printf("time %s -> %lu.%lu\n", argv[optind], tv.tv_sec, tv.tv_usec);
res = settime_alarm_timeval(&tv);
if (res < 0)
res = settime_rtc_timeval(&tv);
if(res < 0) {
fprintf(stderr,"settimeofday failed %s\n", strerror(errno));
return 1;
}
}
else {
fprintf(stderr,"%s [-s 20070325.123456] [-u] [date]\n", argv[0]);
return 1;
}
return 0;
}