Eric Andersen | 6ab2202 | 2000-08-21 23:04:00 +0000 | [diff] [blame] | 1 | /* vi: set sw=4 ts=4: */ |
| 2 | /* |
| 3 | * The Rdate command will ask a time server for the RFC 868 time |
Denys Vlasenko | 93c32f2 | 2012-06-11 02:06:11 +0200 | [diff] [blame] | 4 | * and optionally set the system time. |
Eric Andersen | 6ab2202 | 2000-08-21 23:04:00 +0000 | [diff] [blame] | 5 | * |
| 6 | * by Sterling Huxley <sterling@europa.com> |
| 7 | * |
Denys Vlasenko | 0ef64bd | 2010-08-16 20:14:46 +0200 | [diff] [blame] | 8 | * Licensed under GPLv2 or later, see file LICENSE in this source tree. |
Denys Vlasenko | dd898c9 | 2016-11-23 11:46:32 +0100 | [diff] [blame] | 9 | */ |
| 10 | //config:config RDATE |
| 11 | //config: bool "rdate" |
| 12 | //config: default y |
| 13 | //config: help |
| 14 | //config: The rdate utility allows you to synchronize the date and time of your |
| 15 | //config: system clock with the date and time of a remote networked system using |
| 16 | //config: the RFC868 protocol, which is built into the inetd daemon on most |
| 17 | //config: systems. |
| 18 | |
| 19 | //applet:IF_RDATE(APPLET(rdate, BB_DIR_USR_SBIN, BB_SUID_DROP)) |
| 20 | |
| 21 | //kbuild:lib-$(CONFIG_RDATE) += rdate.o |
Eric Andersen | 6ab2202 | 2000-08-21 23:04:00 +0000 | [diff] [blame] | 22 | |
Pere Orga | 5bc8c00 | 2011-04-11 03:29:49 +0200 | [diff] [blame] | 23 | //usage:#define rdate_trivial_usage |
Denys Vlasenko | 179e88b | 2017-01-20 16:03:48 +0100 | [diff] [blame] | 24 | //usage: "[-s/-p] HOST" |
Pere Orga | 5bc8c00 | 2011-04-11 03:29:49 +0200 | [diff] [blame] | 25 | //usage:#define rdate_full_usage "\n\n" |
Denys Vlasenko | 179e88b | 2017-01-20 16:03:48 +0100 | [diff] [blame] | 26 | //usage: "Set and print time from HOST using RFC 868\n" |
| 27 | //usage: "\n -s Only set system time" |
| 28 | //usage: "\n -p Only print time" |
Pere Orga | 5bc8c00 | 2011-04-11 03:29:49 +0200 | [diff] [blame] | 29 | |
Denis Vlasenko | b6adbf1 | 2007-05-26 19:00:18 +0000 | [diff] [blame] | 30 | #include "libbb.h" |
Eric Andersen | 6ab2202 | 2000-08-21 23:04:00 +0000 | [diff] [blame] | 31 | |
Denis Vlasenko | 8e9ccba | 2007-01-11 16:50:23 +0000 | [diff] [blame] | 32 | enum { RFC_868_BIAS = 2208988800UL }; |
Eric Andersen | 6ab2202 | 2000-08-21 23:04:00 +0000 | [diff] [blame] | 33 | |
Denis Vlasenko | a60f84e | 2008-07-05 09:18:54 +0000 | [diff] [blame] | 34 | static void socket_timeout(int sig UNUSED_PARAM) |
Eric Andersen | e15138a | 2003-09-12 05:50:51 +0000 | [diff] [blame] | 35 | { |
Eric Andersen | 7f2935b | 2003-09-12 08:32:24 +0000 | [diff] [blame] | 36 | bb_error_msg_and_die("timeout connecting to time server"); |
Eric Andersen | e15138a | 2003-09-12 05:50:51 +0000 | [diff] [blame] | 37 | } |
| 38 | |
Eric Andersen | 40eaa9f | 2001-03-19 19:41:54 +0000 | [diff] [blame] | 39 | static time_t askremotedate(const char *host) |
Eric Andersen | 6ab2202 | 2000-08-21 23:04:00 +0000 | [diff] [blame] | 40 | { |
Denis Vlasenko | 8e9ccba | 2007-01-11 16:50:23 +0000 | [diff] [blame] | 41 | uint32_t nett; |
Eric Andersen | 6ab2202 | 2000-08-21 23:04:00 +0000 | [diff] [blame] | 42 | int fd; |
| 43 | |
Rob Landley | 1b751c8 | 2005-10-28 09:24:33 +0000 | [diff] [blame] | 44 | /* Add a timeout for dead or inaccessible servers */ |
Eric Andersen | e15138a | 2003-09-12 05:50:51 +0000 | [diff] [blame] | 45 | alarm(10); |
| 46 | signal(SIGALRM, socket_timeout); |
| 47 | |
Denis Vlasenko | 8e9ccba | 2007-01-11 16:50:23 +0000 | [diff] [blame] | 48 | fd = create_and_connect_stream_or_die(host, bb_lookup_port("time", "tcp", 37)); |
Eric Andersen | 40eaa9f | 2001-03-19 19:41:54 +0000 | [diff] [blame] | 49 | |
Denys Vlasenko | 93c32f2 | 2012-06-11 02:06:11 +0200 | [diff] [blame] | 50 | if (safe_read(fd, &nett, 4) != 4) /* read time from server */ |
Denys Vlasenko | 54e9585 | 2015-02-18 13:47:27 +0100 | [diff] [blame] | 51 | bb_error_msg_and_die("%s: %s", host, "short read"); |
Denys Vlasenko | 93c32f2 | 2012-06-11 02:06:11 +0200 | [diff] [blame] | 52 | if (ENABLE_FEATURE_CLEAN_UP) |
| 53 | close(fd); |
Eric Andersen | 6ab2202 | 2000-08-21 23:04:00 +0000 | [diff] [blame] | 54 | |
Denys Vlasenko | 93c32f2 | 2012-06-11 02:06:11 +0200 | [diff] [blame] | 55 | /* Convert from network byte order to local byte order. |
Eric Andersen | 6ab2202 | 2000-08-21 23:04:00 +0000 | [diff] [blame] | 56 | * RFC 868 time is the number of seconds |
Denis Vlasenko | 8e9ccba | 2007-01-11 16:50:23 +0000 | [diff] [blame] | 57 | * since 00:00 (midnight) 1 January 1900 GMT |
| 58 | * the RFC 868 time 2,208,988,800 corresponds to 00:00 1 Jan 1970 GMT |
Denys Vlasenko | 93c32f2 | 2012-06-11 02:06:11 +0200 | [diff] [blame] | 59 | * Subtract the RFC 868 time to get Linux epoch. |
Eric Andersen | 6ab2202 | 2000-08-21 23:04:00 +0000 | [diff] [blame] | 60 | */ |
Denys Vlasenko | 179e88b | 2017-01-20 16:03:48 +0100 | [diff] [blame] | 61 | nett = ntohl(nett) - RFC_868_BIAS; |
Tim Riker | c1ef7bd | 2006-01-25 00:08:53 +0000 | [diff] [blame] | 62 | |
Denys Vlasenko | 179e88b | 2017-01-20 16:03:48 +0100 | [diff] [blame] | 63 | if (sizeof(time_t) > 4) { |
| 64 | /* Now we have 32-bit lsb of a wider time_t |
| 65 | * Imagine that nett = 0x00000001, |
| 66 | * current time cur = 0x123ffffffff. |
| 67 | * Assuming our time is not some 40 years off, |
| 68 | * remote time must be 0x12400000001. |
| 69 | * Need to adjust out time by (int32_t)(nett - cur). |
| 70 | */ |
| 71 | time_t cur = time(NULL); |
| 72 | int32_t adjust = (int32_t)(nett - (uint32_t)cur); |
| 73 | return cur + adjust; |
| 74 | } |
| 75 | /* This is not going to work, but what can we do */ |
| 76 | return (time_t)nett; |
Eric Andersen | 6ab2202 | 2000-08-21 23:04:00 +0000 | [diff] [blame] | 77 | } |
| 78 | |
Denis Vlasenko | 9b49a5e | 2007-10-11 10:05:36 +0000 | [diff] [blame] | 79 | int rdate_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE; |
Denis Vlasenko | a60f84e | 2008-07-05 09:18:54 +0000 | [diff] [blame] | 80 | int rdate_main(int argc UNUSED_PARAM, char **argv) |
Eric Andersen | 6ab2202 | 2000-08-21 23:04:00 +0000 | [diff] [blame] | 81 | { |
Eric Andersen | 1ca20a7 | 2001-03-21 07:34:27 +0000 | [diff] [blame] | 82 | time_t remote_time; |
Denys Vlasenko | 93c32f2 | 2012-06-11 02:06:11 +0200 | [diff] [blame] | 83 | unsigned flags; |
Tim Riker | c1ef7bd | 2006-01-25 00:08:53 +0000 | [diff] [blame] | 84 | |
Denis Vlasenko | 67b23e6 | 2006-10-03 21:00:06 +0000 | [diff] [blame] | 85 | opt_complementary = "-1"; |
Denis Vlasenko | fe7cd64 | 2007-08-18 15:32:12 +0000 | [diff] [blame] | 86 | flags = getopt32(argv, "sp"); |
Tim Riker | c1ef7bd | 2006-01-25 00:08:53 +0000 | [diff] [blame] | 87 | |
Eric Andersen | 1ca20a7 | 2001-03-21 07:34:27 +0000 | [diff] [blame] | 88 | remote_time = askremotedate(argv[optind]); |
Eric Andersen | 40eaa9f | 2001-03-19 19:41:54 +0000 | [diff] [blame] | 89 | |
Denys Vlasenko | 179e88b | 2017-01-20 16:03:48 +0100 | [diff] [blame] | 90 | /* Manpages of various Unixes are confusing. What happens is: |
| 91 | * (no opts) set and print time |
| 92 | * -s: set time ("do not print the time") |
| 93 | * -p: print time ("do not set, just print the remote time") |
| 94 | * -sp: print time (that's what we do, not sure this is right) |
| 95 | */ |
| 96 | |
Denys Vlasenko | 39f82d4 | 2012-06-11 14:57:29 +0200 | [diff] [blame] | 97 | if (!(flags & 2)) { /* no -p (-s may be present) */ |
Denys Vlasenko | 6e51139 | 2017-01-20 16:07:14 +0100 | [diff] [blame] | 98 | if (time(NULL) == remote_time) |
Denis Vlasenko | 000b9ba | 2006-10-05 23:12:49 +0000 | [diff] [blame] | 99 | bb_error_msg("current time matches remote time"); |
Eric Andersen | 89f10bc | 2003-12-19 11:29:29 +0000 | [diff] [blame] | 100 | else |
| 101 | if (stime(&remote_time) < 0) |
Denys Vlasenko | 6331cf0 | 2009-11-13 09:08:27 +0100 | [diff] [blame] | 102 | bb_perror_msg_and_die("can't set time of day"); |
Bernhard Reutner-Fischer | 5c0ae06 | 2006-06-03 10:24:20 +0000 | [diff] [blame] | 103 | } |
Tim Riker | c1ef7bd | 2006-01-25 00:08:53 +0000 | [diff] [blame] | 104 | |
Denys Vlasenko | 39f82d4 | 2012-06-11 14:57:29 +0200 | [diff] [blame] | 105 | if (flags != 1) /* not lone -s */ |
Bernhard Reutner-Fischer | 5c0ae06 | 2006-06-03 10:24:20 +0000 | [diff] [blame] | 106 | printf("%s", ctime(&remote_time)); |
Eric Andersen | 6ab2202 | 2000-08-21 23:04:00 +0000 | [diff] [blame] | 107 | |
Matt Kraai | 3e856ce | 2000-12-01 02:55:13 +0000 | [diff] [blame] | 108 | return EXIT_SUCCESS; |
Eric Andersen | 6ab2202 | 2000-08-21 23:04:00 +0000 | [diff] [blame] | 109 | } |