1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
|
/* vi: set sw=4 ts=4: */
/*
* Mini hwclock implementation for busybox
*
* Copyright (C) 2002 Robert Griebl <griebl@gmx.de>
*
* Licensed under GPLv2 or later, see file LICENSE in this source tree.
*/
//config:config HWCLOCK
//config: bool "hwclock (5.9 kb)"
//config: default y
//config: select LONG_OPTS
//config: help
//config: The hwclock utility is used to read and set the hardware clock
//config: on a system. This is primarily used to set the current time on
//config: shutdown in the hardware clock, so the hardware will keep the
//config: correct time when Linux is _not_ running.
//config:
//config:config FEATURE_HWCLOCK_ADJTIME_FHS
//config: bool "Use FHS /var/lib/hwclock/adjtime"
//config: default n # util-linux-ng in Fedora 13 still uses /etc/adjtime
//config: depends on HWCLOCK
//config: help
//config: Starting with FHS 2.3, the adjtime state file is supposed to exist
//config: at /var/lib/hwclock/adjtime instead of /etc/adjtime. If you wish
//config: to use the FHS behavior, answer Y here, otherwise answer N for the
//config: classic /etc/adjtime path.
//config:
//config: pathname.com/fhs/pub/fhs-2.3.html#VARLIBHWCLOCKSTATEDIRECTORYFORHWCLO
//applet:IF_HWCLOCK(APPLET(hwclock, BB_DIR_SBIN, BB_SUID_DROP))
//kbuild:lib-$(CONFIG_HWCLOCK) += hwclock.o
#include "libbb.h"
/* After libbb.h, since it needs sys/types.h on some systems */
#include <sys/utsname.h>
#include "rtc_.h"
//musl has no __MUSL__ or similar define to check for,
//but its <sys/types.h> has these lines:
// #define __NEED_fsblkcnt_t
// #define __NEED_fsfilcnt_t
#if defined(__linux__) && defined(__NEED_fsblkcnt_t) && defined(__NEED_fsfilcnt_t)
# define LIBC_IS_MUSL 1
# include <sys/syscall.h>
#else
# define LIBC_IS_MUSL 0
#endif
/* diff code is disabled: it's not sys/hw clock diff, it's some useless
* "time between hwclock was started and we saw CMOS tick" quantity.
* It's useless since hwclock is started at a random moment,
* thus the quantity is also random, useless. Showing 0.000000 does not
* deprive us from any useful info.
*
* SHOW_HWCLOCK_DIFF code in this file shows the difference between system
* and hw clock. It is useful, but not compatible with standard hwclock.
* Thus disabled.
*/
#define SHOW_HWCLOCK_DIFF 0
#if !SHOW_HWCLOCK_DIFF
# define read_rtc(pp_rtcname, sys_tv, utc) read_rtc(pp_rtcname, utc)
#endif
static time_t read_rtc(const char **pp_rtcname, struct timeval *sys_tv, int utc)
{
struct tm tm_time;
int fd;
fd = rtc_xopen(pp_rtcname, O_RDONLY);
rtc_read_tm(&tm_time, fd);
#if SHOW_HWCLOCK_DIFF
{
int before = tm_time.tm_sec;
while (1) {
rtc_read_tm(&tm_time, fd);
xgettimeofday(sys_tv);
if (before != (int)tm_time.tm_sec)
break;
}
}
#endif
if (ENABLE_FEATURE_CLEAN_UP)
close(fd);
return rtc_tm2time(&tm_time, utc);
}
static void show_clock(const char **pp_rtcname, int utc)
{
#if SHOW_HWCLOCK_DIFF
struct timeval sys_tv;
#endif
time_t t = read_rtc(pp_rtcname, &sys_tv, utc);
#if ENABLE_LOCALE_SUPPORT
/* Standard hwclock uses locale-specific output format */
char cp[64];
struct tm *ptm = localtime(&t);
strftime(cp, sizeof(cp), "%c", ptm);
#else
char *cp = ctime(&t);
chomp(cp);
#endif
#if !SHOW_HWCLOCK_DIFF
printf("%s 0.000000 seconds\n", cp);
#else
{
long diff = sys_tv.tv_sec - t;
if (diff < 0 /*&& tv.tv_usec != 0*/) {
/* Why we need diff++? */
/* diff >= 0 is ok: | diff < 0, can't just use tv.tv_usec: */
/* 45.520820 | 43.520820 */
/* - 44.000000 | - 45.000000 */
/* = 1.520820 | = -1.479180, not -2.520820! */
diff++;
/* Should be 1000000 - tv.tv_usec, but then we must check tv.tv_usec != 0 */
sys_tv.tv_usec = 999999 - sys_tv.tv_usec;
}
printf("%s %ld.%06lu seconds\n", cp, diff, (unsigned long)sys_tv.tv_usec);
}
#endif
}
static void set_kernel_tz(const struct timezone *tz)
{
#if LIBC_IS_MUSL
/* musl libc does not pass tz argument to syscall
* because "it's deprecated by POSIX, therefore it's fine
* if we gratuitously break stuff" :(
*/
#if !defined(SYS_settimeofday) && defined(SYS_settimeofday_time32)
# define SYS_settimeofday SYS_settimeofday_time32
#endif
int ret = syscall(SYS_settimeofday, NULL, tz);
#else
int ret = settimeofday(NULL, tz);
#endif
if (ret)
bb_simple_perror_msg_and_die("settimeofday");
}
/*
* At system boot, kernel may set system time from RTC,
* but it knows nothing about timezones. If RTC is in local time,
* then system time is wrong - it is offset by timezone.
* --systz option corrects system time if RTC is in local time,
* and (always) sets in-kernel timezone.
* (Unlike --hctosys, it does not read the RTC).
*
* util-linux's code has this comment:
* RTC | settimeofday calls
* ------|-------------------------------------------------
* Local | 1st) warps system time*, sets PCIL* and kernel tz
* UTC | 1st) locks warp_clock 2nd) sets kernel tz
* * only on first call after boot
* (PCIL is "persistent_clock_is_local" kernel internal flag,
* it makes kernel save RTC in local time, not UTC.)
*/
static void set_kernel_timezone_and_clock(int utc, const struct timeval *hctosys)
{
time_t cur;
struct tm *broken;
struct timezone tz = { 0 };
/* if --utc, prevent kernel's warp_clock() with a dummy call */
if (utc)
set_kernel_tz(&tz);
/* Set kernel's timezone offset based on userspace one */
//It's tempting to call tzset() and use libc global "timezone" variable
//...but it does NOT include DST shift (IOW: it's WRONG, usually by one hour,
//if DST is in effect!) Thus this ridiculous dance:
cur = time(NULL);
broken = localtime(&cur);
tz.tz_minuteswest = -broken->tm_gmtoff / 60;
/*tz.tz_dsttime = 0; already is */
set_kernel_tz(&tz); /* MIGHT warp_clock() if 1st call since boot */
if (hctosys) /* it's --hctosys: set time too */
xsettimeofday(hctosys);
}
static void to_sys_clock(const char **pp_rtcname, int utc)
{
struct timeval tv;
tv.tv_sec = read_rtc(pp_rtcname, NULL, utc);
tv.tv_usec = 0;
return set_kernel_timezone_and_clock(utc, &tv);
}
static void from_sys_clock(const char **pp_rtcname, int utc)
{
#if 1
struct timeval tv;
struct tm tm_time;
int rtc;
rtc = rtc_xopen(pp_rtcname, O_WRONLY);
xgettimeofday(&tv);
/* Prepare tm_time */
if (sizeof(time_t) == sizeof(tv.tv_sec)) {
if (utc)
gmtime_r((time_t*)&tv.tv_sec, &tm_time);
else
localtime_r((time_t*)&tv.tv_sec, &tm_time);
} else {
time_t t = tv.tv_sec;
if (utc)
gmtime_r(&t, &tm_time);
else
localtime_r(&t, &tm_time);
}
#else
/* Bloated code which tries to set hw clock with better precision.
* On x86, even though code does set hw clock within <1ms of exact
* whole seconds, apparently hw clock (at least on some machines)
* doesn't reset internal fractional seconds to 0,
* making all this a pointless exercise.
*/
/* If we see that we are N usec away from whole second,
* we'll sleep for N-ADJ usecs. ADJ corrects for the fact
* that CPU is not infinitely fast.
* On infinitely fast CPU, next wakeup would be
* on (exactly_next_whole_second - ADJ). On real CPUs,
* this difference between current time and whole second
* is less than ADJ (assuming system isn't heavily loaded).
*/
/* Small value of 256us gives very precise sync for 2+ GHz CPUs.
* Slower CPUs will fail to sync and will go to bigger
* ADJ values. qemu-emulated armv4tl with ~100 MHz
* performance ends up using ADJ ~= 4*1024 and it takes
* 2+ secs (2 tries with successively larger ADJ)
* to sync. Even straced one on the same qemu (very slow)
* takes only 4 tries.
*/
#define TWEAK_USEC 256
unsigned adj = TWEAK_USEC;
struct tm tm_time;
struct timeval tv;
int rtc = rtc_xopen(pp_rtcname, O_WRONLY);
/* Try to catch the moment when whole second is close */
while (1) {
unsigned rem_usec;
time_t t;
xgettimeofday(&tv);
t = tv.tv_sec;
rem_usec = 1000000 - tv.tv_usec;
if (rem_usec < adj) {
/* Close enough */
small_rem:
t++;
}
/* Prepare tm_time from t */
if (utc)
gmtime_r(&t, &tm_time); /* may read /etc/xxx (it takes time) */
else
localtime_r(&t, &tm_time); /* same */
if (adj >= 32*1024) {
break; /* 32 ms diff and still no luck?? give up trying to sync */
}
/* gmtime/localtime took some time, re-get cur time */
xgettimeofday(&tv);
if (tv.tv_sec < t /* we are still in old second */
|| (tv.tv_sec == t && tv.tv_usec < adj) /* not too far into next second */
) {
break; /* good, we are in sync! */
}
rem_usec = 1000000 - tv.tv_usec;
if (rem_usec < adj) {
t = tv.tv_sec;
goto small_rem; /* already close to next sec, don't sleep */
}
/* Try to sync up by sleeping */
usleep(rem_usec - adj);
/* Jump to 1ms diff, then increase fast (x2): EVERY loop
* takes ~1 sec, people won't like slowly converging code here!
*/
//bb_error_msg("adj:%d tv.tv_usec:%d", adj, (int)tv.tv_usec);
if (adj < 512)
adj = 512;
/* ... and if last "overshoot" does not look insanely big,
* just use it as adj increment. This makes convergence faster.
*/
if (tv.tv_usec < adj * 8) {
adj += tv.tv_usec;
continue;
}
adj *= 2;
}
/* Debug aid to find "optimal" TWEAK_USEC with nearly exact sync.
* Look for a value which makes tv_usec close to 999999 or 0.
* For 2.20GHz Intel Core 2: optimal TWEAK_USEC ~= 200
*/
//bb_error_msg("tv.tv_usec:%d", (int)tv.tv_usec);
#endif
tm_time.tm_isdst = 0;
xioctl(rtc, RTC_SET_TIME, &tm_time);
if (ENABLE_FEATURE_CLEAN_UP)
close(rtc);
}
static uint64_t resolve_rtc_param_alias(const char *alias)
{
int n;
BUILD_BUG_ON(RTC_PARAM_FEATURES != 0
|| RTC_PARAM_CORRECTION != 1
|| RTC_PARAM_BACKUP_SWITCH_MODE != 2
);
n = index_in_strings(
"features" "\0"
"correction" "\0"
"bsm" "\0"
, alias);
if (n >= 0)
return n;
return xstrtoull(alias, 0);
}
static void get_rtc_param(const char **pp_rtcname, const char *rtc_param)
{
int rtc;
struct rtc_param param;
param.param = resolve_rtc_param_alias(rtc_param);
rtc = rtc_xopen(pp_rtcname, O_RDONLY);
xioctl(rtc, RTC_PARAM_GET, ¶m);
printf("The RTC parameter 0x%llx is set to 0x%llx.\n",
(unsigned long long) param.param, (unsigned long long) param.uvalue);
if (ENABLE_FEATURE_CLEAN_UP)
close(rtc);
}
static void set_rtc_param(const char **pp_rtcname, char *rtc_param)
{
int rtc;
struct rtc_param param;
char *eq;
/* handle param name */
eq = strchr(rtc_param, '=');
if (!eq)
bb_error_msg_and_die("expected <param>=<value>");
*eq = '\0';
param.param = resolve_rtc_param_alias(rtc_param);
*eq = '=';
/* handle param value */
param.uvalue = xstrtoull(eq + 1, 0);
rtc = rtc_xopen(pp_rtcname, O_WRONLY);
printf("The RTC parameter 0x%llx will be set to 0x%llx.\n",
(unsigned long long) param.param, (unsigned long long) param.uvalue);
xioctl(rtc, RTC_PARAM_SET, ¶m);
if (ENABLE_FEATURE_CLEAN_UP)
close(rtc);
}
// hwclock from util-linux 2.36.1
// hwclock [function] [option...]
//Functions:
// -r, --show display the RTC time
// --get display drift corrected RTC time
// --set set the RTC according to --date
// -s, --hctosys set the system time from the RTC
// -w, --systohc set the RTC from the system time
// --systz send timescale configurations to the kernel
// -a, --adjust adjust the RTC to account for systematic drift
// --predict predict the drifted RTC time according to --date
//Options:
// -u, --utc the RTC timescale is UTC
// -l, --localtime the RTC timescale is Local
// -f, --rtc <file> use an alternate file to /dev/rtc0
// --directisa use the ISA bus instead of /dev/rtc0 access
// --date <time> date/time input for --set and --predict
// --delay <sec> delay used when set new RTC time
// --update-drift update the RTC drift factor
// --noadjfile do not use /etc/adjtime
// --adjfile <file> use an alternate file to /etc/adjtime
// --test dry run; implies --verbose
// -v, --verbose display more details
//usage:#define hwclock_trivial_usage
//usage: "[-ul] [-f DEV] [-s|-w|--systz|--param-get PARAM|--param-set PARAM=VAL]"
//usage:#define hwclock_full_usage "\n\n"
//usage: "Show or set hardware clock (RTC)\n"
//usage: "\n -f DEV Use this device (e.g. /dev/rtc2)"
//usage: "\n -u Assume RTC is kept in UTC"
//usage: "\n -l Assume RTC is kept in local time"
//usage: "\n (if neither is given, read from "ADJTIME_PATH")"
///////: "\n -r Show RTC time"
///////-r is default, don't bother showing it in help
//usage: "\n -s Set system time from RTC"
//usage: "\n -w Set RTC from system time"
//usage: "\n --systz Set in-kernel timezone, correct system time"
//usage: "\n if RTC is kept in local time"
//usage: "\n --param-get PARAM Get RTC parameter"
//usage: "\n --param-set PARAM=VAL Set RTC parameter"
int hwclock_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
int hwclock_main(int argc UNUSED_PARAM, char **argv)
{
const char *rtcname = NULL;
char *param;
unsigned opt, exclusive;
int utc;
#define OPT_LOCALTIME (1 << 0)
#define OPT_UTC (1 << 1)
#define OPT_RTCFILE (1 << 2)
#define OPT_SHOW (1 << 3)
#define OPT_HCTOSYS (1 << 4)
#define OPT_SYSTOHC (1 << 5)
#define OPT_PARAM_GET (1 << 6)
#define OPT_PARAM_SET (1 << 7)
//#define OPT_VERBOSE (1 << 8) UNUSED
#define OPT_SYSTZ (1 << 9)
static const char hwclock_longopts[] ALIGN1 =
"localtime\0" No_argument "l"
"utc\0" No_argument "u"
"rtc\0" Required_argument "f"
"show\0" No_argument "r"
"hctosys\0" No_argument "s"
"systohc\0" No_argument "w"
"param-get\0" Required_argument "\xfd" /* no short equivalent */
"param-set\0" Required_argument "\xfe" /* no short equivalent */
"systz\0" No_argument "\xff" /* no short equivalent */
;
opt = getopt32long(argv,
"^""luf:rsw\xfd:\xfe:v" /* -v is accepted and ignored */
"\0"
"l--u:u--l",
hwclock_longopts,
&rtcname,
¶m,
¶m
);
#if 0 //DEBUG
bb_error_msg("opt:0x%x", opt);
if (opt & OPT_PARAM_GET) bb_error_msg("OPT_PARAM_GET %s", param);
if (opt & OPT_PARAM_SET) bb_error_msg("OPT_PARAM_SET %s", param);
if (opt & OPT_SYSTZ ) bb_error_msg("OPT_SYSTZ");
return 0;
#endif
/* All options apart from -luf are exclusive, enforce */
exclusive = opt >> 3;
if ((exclusive - 1) & exclusive) /* more than one bit set? */
bb_show_usage();
/* If -u or -l wasn't given, check if we are using utc */
if (opt & (OPT_UTC | OPT_LOCALTIME))
utc = (opt & OPT_UTC);
else
utc = rtc_adjtime_is_utc();
if (opt & OPT_HCTOSYS)
to_sys_clock(&rtcname, utc);
else if (opt & OPT_SYSTOHC)
from_sys_clock(&rtcname, utc);
else if (opt & OPT_SYSTZ)
set_kernel_timezone_and_clock(utc, NULL);
else if (opt & OPT_PARAM_GET)
get_rtc_param(&rtcname, param);
else if (opt & OPT_PARAM_SET)
set_rtc_param(&rtcname, param);
else
/* default OPT_SHOW */
show_clock(&rtcname, utc);
return 0;
}
|