adjtimed.c revision 290000
119914Swollman/*************************************************************************/
250476Speter/* (c) Copyright Tai Jin, 1988.  All Rights Reserved.                    */
31558Srgrimes/*     Hewlett-Packard Laboratories.                                     */
4156905Sru/*                                                                       */
5156905Sru/* Permission is hereby granted for unlimited modification, use, and     */
61558Srgrimes/* distribution.  This software is made available with no warranty of    */
725451Speter/* any kind, express or implied.  This copyright notice must remain      */
8138593Ssam/* intact in all versions of this software.                              */
925451Speter/*                                                                       */
10138593Ssam/* The author would appreciate it if any bug fixes and enhancements were */
11138593Ssam/* to be sent back to him for incorporation into future versions of this */
12138593Ssam/* software.  Please send changes to tai@iag.hp.com or ken@sdd.hp.com.   */
13138593Ssam/*************************************************************************/
14138593Ssam
15138593Ssam#ifndef lint
16138593Ssamstatic char RCSid[] = "adjtimed.c,v 3.1 1993/07/06 01:04:45 jbj Exp";
17138593Ssam#endif
18222527Sbz
19138593Ssam/*
20222527Sbz * Adjust time daemon.
21222527Sbz * This daemon adjusts the rate of the system clock a la BSD's adjtime().
22138593Ssam * The adjtime() routine uses SYSV messages to communicate with this daemon.
23222527Sbz *
24138593Ssam * Caveat: This emulation uses an undocumented kernel variable.  As such, it
25222527Sbz * cannot be guaranteed to work in future HP-UX releases.  Fortunately,
26197138Shrs * it will no longer be needed in HPUX 10.01 and later.
27222527Sbz */
2844764Swpaul
29138593Ssam#include <sys/param.h>
30138593Ssam#include <sys/types.h>
31138593Ssam#include <sys/ipc.h>
32223735Sbz#include <sys/msg.h>
33138593Ssam#include <sys/lock.h>
34181224Sthompsa#include <time.h>
35193664Shrs#include <signal.h>
3677217Sphk#include <nlist.h>
37178354Ssam#include <fcntl.h>
38231642Srmh#include <stdio.h>
39231642Srmh#include <unistd.h>
40178354Ssam
41228577Sglebius#include "ntp_syslog.h"
42159781Smlaier#include "ntp_stdlib.h"
43257349Sglebius
44142215Sglebius#include "adjtime.h"
45257349Sglebius
46142215Sglebiusdouble atof (const char *);
47146987Sthompsa
48168793Sthompsaint InitClockRate (void);
49146987Sthompsaint AdjustClockRate (register struct timeval *delta, register struct timeval *olddelta);
50222527Sbzlong GetClockRate (void);
51222527Sbzint SetClockRate (long);
52222527Sbzvoid ResetClockRate (void);
53222527Sbzvoid Cleanup (void);
54222527Sbzvoid Exit (int);
55222527Sbz
56156905Sru#define MILLION		1000000L
57138593Ssam
58178354Ssam/* emacs cc-mode goes nuts if we split the next line... */
59178354Ssam#define tvtod(tv)	((double)tv.tv_sec + ((double)tv.tv_usec / (double)MILLION))
6080057Sobrien
61231642Srmhchar const *progname = NULL;
62231642Srmhint verbose = 0;
63231642Srmhint sysdebug = 0;
64231642Srmhstatic int mqid;
65231642Srmhstatic double oldrate = 0.0;
6680057Sobrien
67138593Ssamint
68138593Ssammain(
69144130Sbrooks	int argc,
70202532Sed	char *argv[]
711558Srgrimes	)
721558Srgrimes{
73	struct timeval remains;
74	struct sigvec vec;
75	MsgBuf msg;
76	char ch;
77	int nofork = 0;
78	int fd;
79
80	progname = argv[0];
81
82#ifdef LOG_LOCAL6
83	openlog("adjtimed", LOG_PID, LOG_LOCAL6);
84#else
85	openlog("adjtimed", LOG_PID);
86#endif
87
88	while ((ch = ntp_getopt(argc, argv, "hkrvdfp:")) != EOF) {
89		switch (ch) {
90		    case 'k':
91		    case 'r':
92			if ((mqid = msgget(KEY, 0)) != -1) {
93				if (msgctl(mqid, IPC_RMID, (struct msqid_ds *)0) == -1) {
94					msyslog(LOG_ERR, "remove old message queue: %m");
95					perror("adjtimed: remove old message queue");
96					exit(1);
97				}
98			}
99
100			if (ch == 'k')
101			    exit(0);
102
103			break;
104
105		    case 'v':
106			++verbose, nofork = 1;
107			break;
108
109		    case 'd':
110			++sysdebug;
111			break;
112
113		    case 'f':
114			nofork = 1;
115			break;
116
117		    case 'p':
118			fputs("adjtimed: -p option ignored\n", stderr);
119			break;
120
121		    default:
122			puts("usage: adjtimed -hkrvdf");
123			puts("-h\thelp");
124			puts("-k\tkill existing adjtimed, if any");
125			puts("-r\trestart (kills existing adjtimed, if any)");
126			puts("-v\tdebug output (repeat for more output)");
127			puts("-d\tsyslog output (repeat for more output)");
128			puts("-f\tno fork");
129			msyslog(LOG_ERR, "usage error");
130			exit(1);
131		} /* switch */
132	} /* while */
133
134	if (!nofork) {
135		switch (fork()) {
136		    case 0:
137			close(fileno(stdin));
138			close(fileno(stdout));
139			close(fileno(stderr));
140
141#ifdef TIOCNOTTY
142			if ((fd = open("/dev/tty")) != -1) {
143				ioctl(fd, TIOCNOTTY, 0);
144				close(fd);
145			}
146#else
147			setpgrp();
148#endif
149			break;
150
151		    case -1:
152			msyslog(LOG_ERR, "fork: %m");
153			perror("adjtimed: fork");
154			exit(1);
155
156		    default:
157			exit(0);
158		} /* switch */
159	} /* if */
160
161	if (nofork) {
162		setvbuf(stdout, NULL, _IONBF, BUFSIZ);
163		setvbuf(stderr, NULL, _IONBF, BUFSIZ);
164	}
165
166	msyslog(LOG_INFO, "started");
167	if (verbose) printf("adjtimed: started\n");
168
169	if (InitClockRate() == -1)
170	    Exit(2);
171
172	(void)signal(SIGHUP, SIG_IGN);
173	(void)signal(SIGINT, SIG_IGN);
174	(void)signal(SIGQUIT, SIG_IGN);
175	(void)signal(SIGTERM, Cleanup);
176
177	vec.sv_handler = ResetClockRate;
178	vec.sv_flags = 0;
179	vec.sv_mask = ~0;
180	sigvector(SIGALRM, &vec, (struct sigvec *)0);
181
182	if (msgget(KEY, IPC_CREAT|IPC_EXCL) == -1) {
183		if (errno == EEXIST) {
184			msyslog(LOG_ERR, "message queue already exists, use -r to remove it");
185			fputs("adjtimed: message queue already exists, use -r to remove it\n",
186			      stderr);
187			Exit(1);
188		}
189
190		msyslog(LOG_ERR, "create message queue: %m");
191		perror("adjtimed: create message queue");
192		Exit(1);
193	}
194
195	if ((mqid = msgget(KEY, 0)) == -1) {
196		msyslog(LOG_ERR, "get message queue id: %m");
197		perror("adjtimed: get message queue id");
198		Exit(1);
199	}
200
201	/* Lock process in memory to improve response time */
202	if (plock(PROCLOCK)) {
203		msyslog(LOG_ERR, "plock: %m");
204		perror("adjtimed: plock");
205		Cleanup();
206	}
207
208	/* Also raise process priority.
209	 * If we do not get run when we want, this leads to bad timekeeping
210	 * and "Previous time adjustment didn't complete" gripes from xntpd.
211	 */
212	if (nice(-10) == -1) {
213		msyslog(LOG_ERR, "nice: %m");
214		perror("adjtimed: nice");
215		Cleanup();
216	}
217
218	for (;;) {
219		if (msgrcv(mqid, &msg.msgp, MSGSIZE, CLIENT, 0) == -1) {
220			if (errno == EINTR) continue;
221			msyslog(LOG_ERR, "read message: %m");
222			perror("adjtimed: read message");
223			Cleanup();
224		}
225
226		switch (msg.msgb.code) {
227		    case DELTA1:
228		    case DELTA2:
229			AdjustClockRate(&msg.msgb.tv, &remains);
230
231			if (msg.msgb.code == DELTA2) {
232				msg.msgb.tv = remains;
233				msg.msgb.mtype = SERVER;
234
235				while (msgsnd(mqid, &msg.msgp, MSGSIZE, 0) == -1) {
236					if (errno == EINTR) continue;
237					msyslog(LOG_ERR, "send message: %m");
238					perror("adjtimed: send message");
239					Cleanup();
240				}
241			}
242
243			if (remains.tv_sec + remains.tv_usec != 0L) {
244				if (verbose) {
245					printf("adjtimed: previous correction remaining %.6fs\n",
246					       tvtod(remains));
247				}
248				if (sysdebug) {
249					msyslog(LOG_INFO, "previous correction remaining %.6fs",
250						tvtod(remains));
251				}
252			}
253			break;
254
255		    default:
256			fprintf(stderr, "adjtimed: unknown message code %d\n", msg.msgb.code);
257			msyslog(LOG_ERR, "unknown message code %d", msg.msgb.code);
258		} /* switch */
259	} /* loop */
260} /* main */
261
262/*
263 * Default clock rate (old_tick).
264 */
265#define DEFAULT_RATE	(MILLION / HZ)
266#define UNKNOWN_RATE	0L
267#define TICK_ADJ	5	/* standard adjustment rate, microsec/tick */
268
269static long default_rate = DEFAULT_RATE;
270static long tick_rate = HZ;	/* ticks per sec */
271static long slew_rate = TICK_ADJ * HZ; /* in microsec/sec */
272
273int
274AdjustClockRate(
275	register struct timeval *delta,
276	register struct timeval *olddelta
277	)
278{
279	register long rate, dt, leftover;
280	struct itimerval period, remains;
281
282	dt = (delta->tv_sec * MILLION) + delta->tv_usec;
283
284	if (verbose)
285	    printf("adjtimed: new correction %.6fs\n", (double)dt / (double)MILLION);
286	if (sysdebug)
287	    msyslog(LOG_INFO, "new correction %.6fs", (double)dt / (double)MILLION);
288	if (verbose > 2) printf("adjtimed: leftover %ldus\n", leftover);
289	if (sysdebug > 2) msyslog(LOG_INFO, "leftover %ldus", leftover);
290	rate = dt;
291
292	/*
293	 * Apply a slew rate of slew_rate over a period of dt/slew_rate seconds.
294	 */
295	if (dt > 0) {
296		rate = slew_rate;
297	} else {
298		rate = -slew_rate;
299		dt = -dt;
300	}
301	period.it_value.tv_sec = dt / slew_rate;
302	period.it_value.tv_usec = (dt % slew_rate) * (MILLION / slew_rate);
303	/*
304	 * Note: we assume the kernel will convert the specified period into ticks
305	 * using the modified clock rate rather than an assumed nominal clock rate,
306	 * and therefore will generate the timer interrupt after the specified
307	 * number of true seconds, not skewed seconds.
308	 */
309
310	if (verbose > 1)
311	    printf("adjtimed: will be complete in %lds %ldus\n",
312		   period.it_value.tv_sec, period.it_value.tv_usec);
313	if (sysdebug > 1)
314	    msyslog(LOG_INFO, "will be complete in %lds %ldus",
315		    period.it_value.tv_sec, period.it_value.tv_usec);
316	/*
317	 * adjust the clock rate
318	 */
319	if (dt) {
320		if (SetClockRate((rate / tick_rate) + default_rate) == -1) {
321			msyslog(LOG_ERR, "set clock rate: %m");
322			perror("adjtimed: set clock rate");
323		}
324	}
325	/*
326	 * start the timer
327	 * (do this after changing the rate because the period has been rounded down)
328	 */
329	period.it_interval.tv_sec = period.it_interval.tv_usec = 0L;
330	setitimer(ITIMER_REAL, &period, &remains);
331	/*
332	 * return old delta
333	 */
334	if (olddelta) {
335		dt = ((remains.it_value.tv_sec * MILLION) + remains.it_value.tv_usec) *
336			oldrate;
337		olddelta->tv_sec = dt / MILLION;
338		olddelta->tv_usec = dt - (olddelta->tv_sec * MILLION);
339	}
340
341	oldrate = (double)rate / (double)MILLION;
342	return(0);
343} /* AdjustClockRate */
344
345static struct nlist nl[] = {
346#ifdef __hp9000s800
347#ifdef PRE7_0
348	{ "tick" },
349#else
350	{ "old_tick" },
351#endif
352#else
353	{ "_old_tick" },
354#endif
355	{ "" }
356};
357
358static int kmem;
359
360/*
361 * The return value is the clock rate in old_tick units or -1 if error.
362 */
363long
364GetClockRate(void)
365{
366	long rate, mask;
367
368	if (lseek(kmem, (off_t)nl[0].n_value, 0) == -1L)
369	    return (-1L);
370
371	mask = sigblock(sigmask(SIGALRM));
372
373	if (read(kmem, (caddr_t)&rate, sizeof(rate)) != sizeof(rate))
374	    rate = UNKNOWN_RATE;
375
376	sigsetmask(mask);
377	return (rate);
378} /* GetClockRate */
379
380/*
381 * The argument is the new rate in old_tick units.
382 */
383int
384SetClockRate(
385	long rate
386	)
387{
388	long mask;
389
390	if (lseek(kmem, (off_t)nl[0].n_value, 0) == -1L)
391	    return (-1);
392
393	mask = sigblock(sigmask(SIGALRM));
394
395	if (write(kmem, (caddr_t)&rate, sizeof(rate)) != sizeof(rate)) {
396		sigsetmask(mask);
397		return (-1);
398	}
399
400	sigsetmask(mask);
401
402	if (rate != default_rate) {
403		if (verbose > 3) {
404			printf("adjtimed: clock rate (%lu) %ldus/s\n", rate,
405			       (rate - default_rate) * tick_rate);
406		}
407		if (sysdebug > 3) {
408			msyslog(LOG_INFO, "clock rate (%lu) %ldus/s", rate,
409				(rate - default_rate) * tick_rate);
410		}
411	}
412
413	return (0);
414} /* SetClockRate */
415
416int
417InitClockRate(void)
418{
419	if ((kmem = open("/dev/kmem", O_RDWR)) == -1) {
420		msyslog(LOG_ERR, "open(/dev/kmem): %m");
421		perror("adjtimed: open(/dev/kmem)");
422		return (-1);
423	}
424
425	nlist("/hp-ux", nl);
426
427	if (nl[0].n_type == 0) {
428		fputs("adjtimed: /hp-ux has no symbol table\n", stderr);
429		msyslog(LOG_ERR, "/hp-ux has no symbol table");
430		return (-1);
431	}
432	/*
433	 * Set the default to the system's original value
434	 */
435	default_rate = GetClockRate();
436	if (default_rate == UNKNOWN_RATE) default_rate = DEFAULT_RATE;
437	tick_rate = (MILLION / default_rate);
438	slew_rate = TICK_ADJ * tick_rate;
439	fprintf(stderr,"default_rate=%ld, tick_rate=%ld, slew_rate=%ld\n",default_rate,tick_rate,slew_rate);
440
441	return (0);
442} /* InitClockRate */
443
444/*
445 * Reset the clock rate to the default value.
446 */
447void
448ResetClockRate(void)
449{
450	struct itimerval it;
451
452	it.it_value.tv_sec = it.it_value.tv_usec = 0L;
453	setitimer(ITIMER_REAL, &it, (struct itimerval *)0);
454
455	if (verbose > 2) puts("adjtimed: resetting the clock");
456	if (sysdebug > 2) msyslog(LOG_INFO, "resetting the clock");
457
458	if (GetClockRate() != default_rate) {
459		if (SetClockRate(default_rate) == -1) {
460			msyslog(LOG_ERR, "set clock rate: %m");
461			perror("adjtimed: set clock rate");
462		}
463	}
464
465	oldrate = 0.0;
466} /* ResetClockRate */
467
468void
469Cleanup(void)
470{
471	ResetClockRate();
472
473	if (msgctl(mqid, IPC_RMID, (struct msqid_ds *)0) == -1) {
474		if (errno != EINVAL) {
475			msyslog(LOG_ERR, "remove message queue: %m");
476			perror("adjtimed: remove message queue");
477		}
478	}
479
480	Exit(2);
481} /* Cleanup */
482
483void
484Exit(status)
485     int status;
486{
487	msyslog(LOG_ERR, "terminated");
488	closelog();
489	if (kmem != -1) close(kmem);
490	exit(status);
491} /* Exit */
492