Initial public busybox upstream commit
[busybox4maemo] / util-linux / hwclock.c
1 /* vi: set sw=4 ts=4: */
2 /*
3  * Mini hwclock implementation for busybox
4  *
5  * Copyright (C) 2002 Robert Griebl <griebl@gmx.de>
6  *
7  * Licensed under GPLv2 or later, see file LICENSE in this tarball for details.
8 */
9
10 #include <sys/utsname.h>
11 #include <getopt.h>
12 #include "libbb.h"
13 #include "rtc_.h"
14
15 #if ENABLE_FEATURE_HWCLOCK_LONG_OPTIONS
16 # ifndef _GNU_SOURCE
17 #  define _GNU_SOURCE
18 # endif
19 #endif
20
21 static const char *rtcname;
22
23 static time_t read_rtc(int utc)
24 {
25         time_t ret;
26         int fd;
27
28         fd = rtc_xopen(&rtcname, O_RDONLY);
29         ret = rtc_read_time(fd, utc);
30         close(fd);
31
32         return ret;
33 }
34
35 static void write_rtc(time_t t, int utc)
36 {
37         struct tm tm;
38         int rtc = rtc_xopen(&rtcname, O_WRONLY);
39
40         tm = *(utc ? gmtime(&t) : localtime(&t));
41         tm.tm_isdst = 0;
42
43         xioctl(rtc, RTC_SET_TIME, &tm);
44
45         close(rtc);
46 }
47
48 static void show_clock(int utc)
49 {
50         //struct tm *ptm;
51         time_t t;
52         char *cp;
53
54         t = read_rtc(utc);
55         //ptm = localtime(&t);  /* Sets 'tzname[]' */
56
57         cp = ctime(&t);
58         if (cp[0])
59                 cp[strlen(cp) - 1] = '\0';
60
61         //printf("%s  %.6f seconds %s\n", cp, 0.0, utc ? "" : (ptm->tm_isdst ? tzname[1] : tzname[0]));
62         printf("%s  0.000000 seconds\n", cp);
63 }
64
65 static void to_sys_clock(int utc)
66 {
67         struct timeval tv;
68         const struct timezone tz = { timezone/60 - 60*daylight, 0 };
69
70         tv.tv_sec = read_rtc(utc);
71         tv.tv_usec = 0;
72         if (settimeofday(&tv, &tz))
73                 bb_perror_msg_and_die("settimeofday() failed");
74 }
75
76 static void from_sys_clock(int utc)
77 {
78         struct timeval tv;
79
80         gettimeofday(&tv, NULL);
81         //if (gettimeofday(&tv, NULL))
82         //      bb_perror_msg_and_die("gettimeofday() failed");
83         write_rtc(tv.tv_sec, utc);
84 }
85
86 #define HWCLOCK_OPT_LOCALTIME   0x01
87 #define HWCLOCK_OPT_UTC         0x02
88 #define HWCLOCK_OPT_SHOW        0x04
89 #define HWCLOCK_OPT_HCTOSYS     0x08
90 #define HWCLOCK_OPT_SYSTOHC     0x10
91 #define HWCLOCK_OPT_RTCFILE     0x20
92
93 int hwclock_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
94 int hwclock_main(int argc ATTRIBUTE_UNUSED, char **argv)
95 {
96         unsigned opt;
97         int utc;
98
99 #if ENABLE_FEATURE_HWCLOCK_LONG_OPTIONS
100         static const char hwclock_longopts[] ALIGN1 =
101                 "localtime\0" No_argument "l"
102                 "utc\0"       No_argument "u"
103                 "show\0"      No_argument "r"
104                 "hctosys\0"   No_argument "s"
105                 "systohc\0"   No_argument "w"
106                 "file\0"      Required_argument "f"
107                 ;
108         applet_long_options = hwclock_longopts;
109 #endif
110         opt_complementary = "r--ws:w--rs:s--wr:l--u:u--l";
111         opt = getopt32(argv, "lurswf:", &rtcname);
112
113         /* If -u or -l wasn't given check if we are using utc */
114         if (opt & (HWCLOCK_OPT_UTC | HWCLOCK_OPT_LOCALTIME))
115                 utc = (opt & HWCLOCK_OPT_UTC);
116         else
117                 utc = rtc_adjtime_is_utc();
118
119         if (opt & HWCLOCK_OPT_HCTOSYS)
120                 to_sys_clock(utc);
121         else if (opt & HWCLOCK_OPT_SYSTOHC)
122                 from_sys_clock(utc);
123         else
124                 /* default HWCLOCK_OPT_SHOW */
125                 show_clock(utc);
126
127         return 0;
128 }