| /* vi: set sw=4 ts=4: */ |
| /* |
| * The Rdate command will ask a time server for the RFC 868 time |
| * and optionally set the system time. |
| * |
| * by Sterling Huxley <sterling@europa.com> |
| * |
| * This program is free software; you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License as published by |
| * the Free Software Foundation; either version 2 of the License, or |
| * (at your option) any later version. |
| * |
| * This program is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * General Public License for more details. |
| * |
| * You should have received a copy of the GNU General Public License |
| * along with this program; if not, write to the Free Software |
| * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
| * |
| */ |
| |
| #include <sys/time.h> |
| #include <sys/types.h> |
| #include <sys/socket.h> |
| #include <netinet/in.h> |
| #include <netdb.h> |
| #include <stdio.h> |
| #include <getopt.h> |
| #include <string.h> |
| #include <time.h> |
| #include <stdlib.h> |
| #include <unistd.h> |
| #include <signal.h> |
| |
| #include "busybox.h" |
| |
| |
| static const int RFC_868_BIAS = 2208988800UL; |
| |
| static void socket_timeout(int sig) |
| { |
| bb_error_msg_and_die("timeout connecting to time server"); |
| } |
| |
| static time_t askremotedate(const char *host) |
| { |
| unsigned long int nett, localt; |
| struct sockaddr_in s_in; |
| int fd; |
| |
| bb_lookup_host(&s_in, host); |
| s_in.sin_port = bb_lookup_port("time", "tcp", 37); |
| |
| /* Add a timeout for dead or non accessable servers */ |
| alarm(10); |
| signal(SIGALRM, socket_timeout); |
| |
| fd = xconnect(&s_in); |
| |
| if (safe_read(fd, (void *)&nett, 4) != 4) /* read time from server */ |
| bb_error_msg_and_die("%s did not send the complete time", host); |
| |
| close(fd); |
| |
| /* convert from network byte order to local byte order. |
| * RFC 868 time is the number of seconds |
| * since 00:00 (midnight) 1 January 1900 GMT |
| * the RFC 868 time 2,208,988,800 corresponds to 00:00 1 Jan 1970 GMT |
| * Subtract the RFC 868 time to get Linux epoch |
| */ |
| localt= ntohl(nett) - RFC_868_BIAS; |
| |
| return(localt); |
| } |
| |
| int rdate_main(int argc, char **argv) |
| { |
| time_t remote_time; |
| int opt; |
| int setdate = 1; |
| int printdate = 1; |
| |
| /* Interpret command line args */ |
| while ((opt = getopt(argc, argv, "sp")) > 0) { |
| switch (opt) { |
| case 's': |
| printdate = 0; |
| setdate = 1; |
| break; |
| case 'p': |
| printdate = 1; |
| setdate = 0; |
| break; |
| default: |
| bb_show_usage(); |
| } |
| } |
| |
| if (optind == argc) |
| bb_show_usage(); |
| |
| remote_time = askremotedate(argv[optind]); |
| |
| if (setdate) { |
| time_t current_time; |
| |
| time(¤t_time); |
| if (current_time == remote_time) |
| bb_error_msg("Current time matches remote time."); |
| else |
| if (stime(&remote_time) < 0) |
| bb_perror_msg_and_die("Could not set time of day"); |
| } |
| |
| if (printdate) |
| printf("%s", ctime(&remote_time)); |
| |
| return EXIT_SUCCESS; |
| } |