231 lines
4.8 KiB
C
231 lines
4.8 KiB
C
#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;
|
|
struct timespec ts;
|
|
char strbuf[260];
|
|
int fd;
|
|
|
|
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) {
|
|
char buf[2000];
|
|
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;
|
|
}
|