/* NO WARRANTY THERE IS NO WARRANTY FOR THIS PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR REDISTRIBUTE THE PROGRAM, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. */ /* fixes for netdate 1.16 1) utmp handling 2) trying to read 64 bit on 64bit machines 3) getdate is a UNIX98 function renamed it to mygetdate */ /* fixes 1.2 1) did not write old AND newtime to utmp 2) init wtmp now correcly with OLD/NEW_TIME or atleast mit O/N in line 3) cleanup 4) removed references to bcopy bzero */ #ifndef lint char rcsid[]="$ID$ "; #endif #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include /* ok, we stick to the old stuff .... */ /* but only if we can find anything better */ #ifndef WTMP_FILE #define WTMP "/etc/wtmp" #else #define WTMP WTMP_FILE #endif char *service = "time"; char *defaultproto = "udp"; /* difference between 1900 (RFC868) and 1970 (UNIX) base times */ #define NETBASE 2208988800 long limit = 5; #define MAXHOSTS 20 #define LOCALHOST "localhost" char *whoami; char hostname[65]; struct timeval now; struct timehost { char *hostname; short local; short bad; char *protoname; long protonumber; int socktype; struct timeval asked; struct timeval then; struct timeval acked; long difference; long count; } timehosts[MAXHOSTS]; struct timehost *tophost = &timehosts[MAXHOSTS]; void usage (void); int setproto (char *, struct timehost *); int main (int, char **); int getdiff (struct timehost *); int mygetdate (struct timehost *); void printit (struct timehost *); void tvsub (struct timeval *, struct timeval *, struct timeval *); void printdiff (char *, struct timeval *); void timeout (); static int internettime (struct timehost *thishost); long getport (char *protoname); static int internettime (struct timehost *thishost); struct timehost *mungediffs(struct timehost *); void usage (void) { fprintf (stderr, "usage: %s [ -l limit ] host ...\n" "%s tries to find a group of at least two hosts whose times agree\n" "within %ld seconds, and sets the time to that of the first host in the group.\n", whoami, whoami, limit); fprintf (stderr, "The limit may be set with the -l option. Setting it to zero (or supplying\n" "only one host name argument) will set the time to that of the first host to\n" "respond. The caller must be super-user for the system time to be set.\n"); exit (1); } int rdate = 0; int verbose = 0; int debug = 0; int main (int argc, char **argv) { extern char *rindex(); struct timehost *mungediffs(); register struct timehost *thishost; int hostargs = 0; if ((whoami = rindex(*argv, '/')) != NULL) whoami++; else whoami = *argv; if (strcmp (whoami, "rdate") == 0) { /* emulate SMI rdate command */ rdate = 1; defaultproto = "tcp"; limit = 0; } if (gethostname(hostname, (int)sizeof (hostname)) == -1) { perror ("gethostname"); exit (1); } while (*++argv != NULL && **argv == '-') { switch (argv[0][1]) { case 'd': debug++; break; case 'v': verbose++; break; case 'l': if (*++argv == NULL) usage(); limit = atoi(*argv); break; default: fprintf (stderr, "Unknown option: %s\n", *argv); usage(); break; } } if (*argv == NULL) usage(); if (debug) fprintf (stderr, "%s: rdate %d; verbose %d; limit %ld.\n", whoami, rdate, verbose, limit); for (thishost = &timehosts[0]; *argv != NULL; argv++) { if (thishost >= tophost) { fprintf(stderr, "Too many hosts: ignoring"); do { fprintf (stderr, " %s", *argv); } while (*++argv != NULL); fprintf (stderr, "\n"); break; } if (setproto(*argv, thishost)) continue; thishost -> hostname = *argv; thishost -> bad = 0; if (strcmp (thishost -> hostname, LOCALHOST) == 0) thishost -> local = 1; if (++hostargs == 1 && argv[1] == NULL) /* Only one host arg, */ limit = 0; /* so just set to it. */ if (limit == 0) { if (!mygetdate(thishost)) continue; exit(0); } if (!getdiff (thishost)) continue; thishost++; } if (limit == 0) exit(1); if (thishost == &timehosts[0]) exit(1); if ((thishost = mungediffs(thishost)) == NULL) { fprintf (stderr, "No two hosts agree on the time within %ld seconds\n", limit); exit(1); } if (!mygetdate (thishost)) exit (1); exit(0); } int setproto(char *what, struct timehost *thishost) { static char *protoname; static long protonumber; static int socktype; register struct protoent *pp; setprotoent(1); if ((pp = getprotobyname (what)) == NULL) { if (protoname == NULL) if (!setproto(defaultproto, thishost)) { fprintf(stderr, "Default protocol %s was not found in /etc/protocols.\n", defaultproto); exit(1); } thishost -> protoname = protoname; thishost -> protonumber = protonumber; thishost -> socktype = socktype; return(0); } protoname = what; /*pp -> p_name; this is static: don't use it.*/ protonumber = pp -> p_proto; switch (protonumber) { case IPPROTO_TCP: socktype = SOCK_STREAM; if (debug) fprintf(stderr, "%s SOCK_STREAM\n", protoname); break; case IPPROTO_UDP: socktype = SOCK_DGRAM; if (debug) fprintf(stderr, "%s SOCK_DGRAM\n", protoname); break; default: fprintf(stderr, "Unknown protocol: %s\n", protoname); exit(1); break; } return(1); } int getdiff(struct timehost *thishost) { if (!internettime (thishost)) return(0); thishost -> difference = thishost -> then.tv_sec - now.tv_sec; if (!rdate) printit(thishost); return(1); } /* Find the largest group of hosts which agree within the limit and return the first of that group. If no two hosts agree, give up. */ struct timehost * mungediffs(struct timehost *tophost) { register struct timehost *thishost, *ahost, *goodhost; long diff; tophost--; /* simplifies the comparisons */ goodhost = &timehosts[0]; for (thishost = &timehosts[0]; thishost < tophost; thishost++) { if (thishost -> bad) continue; thishost -> count = 1; if (verbose) printf ("%s", thishost -> hostname); for (ahost = thishost + 1; ahost <= tophost; ahost++) { if (thishost -> bad) continue; diff = ahost -> difference - thishost -> difference; if (abs(diff) < limit) { thishost -> count++; if (verbose) printf (" %s", ahost -> hostname); } } if (verbose) { printf (" %ld\n", thishost -> count); (void)fflush(stdout); } if (thishost -> count > goodhost -> count) goodhost = thishost; } if (goodhost -> count > 1) return(goodhost); return(NULL); } int mygetdate (struct timehost *thishost) { int set = 0; /* we only use wtmp here so why use global vars at all ? */ /* additionaly we now init it with zero if somebody cares he */ /* can fill the structures with something meaningfull */ struct utmp *wtmp[2]; wtmp[0]=(struct utmp *)malloc(sizeof(struct utmp)); wtmp[1]=(struct utmp *)malloc(sizeof(struct utmp)); memset(wtmp[0],0,sizeof(struct utmp)); /* zero the mem */ memset(wtmp[1],0,sizeof(struct utmp)); /* zero the mem */ if (!internettime (thishost)) return (0); if (thishost -> local) { printf ("Local host %s has best time, so not setting date\n", hostname); printit(thishost); exit(0); } if (limit != 0 && abs((thishost -> then.tv_sec - now.tv_sec) - thishost -> difference) > limit) { fprintf (stderr, "Time from %s has varied more than the limit of %ld seconds\n", thishost -> hostname, limit); printit(thishost); exit(1); } if (settimeofday (&thishost -> then, (struct timezone *)0) == -1) perror ("netdate: settimeofday"); else { int wf; if ((wf = open(WTMP, 1)) >= 0) { /* this ist at least UGLY sometimes its timeval ut_tv */ /* on others there is ut_time */ #ifdef ut_type wtmp[1]->ut_type=OLD_TIME; wtmp[0]->ut_type=NEW_TIME; #else wtmp[0]->ut_line[0] = 'O'; wtmp[1]->ut_line[0] = 'N'; #endif #ifndef _HAVE_UT_TV wtmp[0]->ut_time = now.tv_sec; wtmp[1]->ut_time = thishost -> then.tv_sec; #else wtmp[0]->ut_tv.tv_sec = now.tv_sec; wtmp[1]->ut_tv.tv_sec = thishost -> then.tv_sec; #endif (void)lseek(wf, 0L, 2); (void)write(wf, (char *)wtmp[0], sizeof(struct utmp)); (void)write(wf, (char *)wtmp[1], sizeof(struct utmp)); (void)close(wf); } set = 1; } printit(thishost); return(set); } void printit(struct timehost *thishost) { extern char *ctime(); struct tm *tp, *localtime(); struct timeval diff; char newstring[128]; if (rdate) printf ("%s", ctime((unsigned long *)&thishost -> then.tv_sec)); else { (void)sprintf(newstring, "%s ", thishost -> hostname); tvsub(&diff, &thishost -> then, &now); printdiff(&newstring[strlen(newstring)], &diff); printf ("%-24s %.19s.%03ld", newstring, ctime((unsigned long *)&thishost -> then.tv_sec), thishost -> then.tv_usec / 1000); if (verbose) { tp = localtime((unsigned long *)&thishost -> acked); printf(" at %02d:%02d:%02d.%03ld", tp -> tm_hour, tp -> tm_min, tp -> tm_sec, thishost -> acked.tv_usec / 1000); tvsub(&diff, &thishost -> acked, &thishost -> asked); printdiff(newstring, &diff); printf(" delay %s", newstring); } printf("\n"); } (void)fflush (stdout); } void tvsub(tdiff, t1, t0) struct timeval *tdiff, *t1, *t0; { tdiff->tv_sec = t1->tv_sec - t0->tv_sec; tdiff->tv_usec = t1->tv_usec - t0->tv_usec; if (tdiff->tv_sec < 0 && tdiff->tv_usec > 0) tdiff->tv_sec++, tdiff->tv_usec -= 1000000; if (tdiff->tv_sec > 0 && tdiff->tv_usec < 0) tdiff->tv_sec--, tdiff->tv_usec += 1000000; } void printdiff(char *where, struct timeval *diff) { (void) sprintf (where, "%c%d.%.03d", (diff->tv_sec < 0 || diff->tv_usec < 0) ? '-' : '+', abs(diff->tv_sec), abs(diff->tv_usec) / 1000); } static jmp_buf jb; void timeout() { longjmp(jb, 1); } static int internettime (struct timehost *thishost) { register struct hostent *hp; struct sockaddr_in sin; long port; int nread; int buf; /* must be 32bit !! */ static int s = -1; if (thishost -> local) { if (gettimeofday (&now, (struct timezone *)0) == -1) { perror ("netdate: gettimeofday"); exit (1); } thishost -> asked = now; thishost -> then = now; thishost -> acked = now; return(1); } timerclear(&thishost -> then); if (setjmp(jb)) goto bad; (void)signal(SIGALRM, timeout); if (s != -1) (void) close (s), s = -1; port = getport(thishost -> protoname); // bzero((char *)&sin, sizeof (sin)); (void)memset((char *)&sin,0,sizeof(sin)); sethostent(1); if ((hp = gethostbyname(thishost -> hostname)) == NULL) { fprintf(stderr, "%s: %s: unknown host\n", whoami, thishost -> hostname); goto out; } sin.sin_family = hp->h_addrtype; (void)alarm(20); s = socket(hp->h_addrtype, thishost -> socktype, 0 /*protonumber*/); if (s < 0) { perror("netdate: socket"); (void)alarm(0); goto out; } if (thishost -> socktype == SOCK_STREAM) { if (bind(s, (struct sockaddr *)&sin, sizeof (sin)) < 0) { perror("netdate: bind"); goto bad; } } memcpy((char *)&sin.sin_addr,hp->h_addr, hp->h_length); // bcopy(hp->h_addr, (char *)&sin.sin_addr, hp->h_length); sin.sin_port = port; (void)gettimeofday (&thishost -> asked, (struct timezone *)0); if (connect(s, (struct sockaddr *)&sin, sizeof (sin)) < 0) { perror("netdate: connect"); goto bad; } if (thishost -> socktype == SOCK_DGRAM) { if (write (s, "\n", 1) < 0) { perror ("netdate: send"); goto bad; } } /* see note below: */ /* nread = read (s, (char *)&thishost -> then, sizeof (thishost -> then)); */ thishost -> then.tv_sec=0L; nread = read (s, (char *)&buf,4); (void)gettimeofday (&thishost -> acked, (struct timezone *)0); (void)alarm(0); now = thishost -> acked; /* this breaks netdate when you have a 64bit host and a 32bit server */ /* RFC 868 says: Send the time as a 32 bit binary number */ /* so we will set this to 4 byte fixed */ /* if (nread < sizeof(thishost -> then.tv_sec) ) */ if (nread != 4) { perror ("netdate: read"); goto bad; } thishost -> then.tv_sec=buf; /* everything is fine we set the time */ /* RFC 868 only allows seconds, but what the hell */ if (nread == sizeof(thishost -> then)) thishost -> then.tv_usec = ntohl(thishost -> then.tv_usec); else thishost -> then.tv_usec = 0L; thishost -> then.tv_sec = ntohl (thishost -> then.tv_sec) - NETBASE; return (1); /* don't close before returning to avoid delays */ bad: (void)alarm(0); (void) close (s), s = -1; out: if (gettimeofday (&now, (struct timezone *)0) == -1) { perror ("netdate: gettimeofday"); exit (1); } thishost -> asked = now; thishost -> then = now; thishost -> acked = now; thishost -> bad = 1; fprintf (stderr, "Connection with %s to %s failed.\n", thishost -> protoname, thishost -> hostname); return(0); } long getport(char *protoname) { register struct servent *sp; static long port; if (port != 0) return(port); if ((sp = getservbyname(service, protoname)) == 0) { fprintf(stderr, "%s: %s/%s: unknown service\n", whoami, service, protoname); exit(1); } return (port = sp->s_port); }