⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 adjtimed.c

📁 网络时间协议NTP 源码 版本v4.2.0b 该源码用于linux平台下
💻 C
字号:
/*************************************************************************//* (c) Copyright Tai Jin, 1988.  All Rights Reserved.                    *//*     Hewlett-Packard Laboratories.                                     *//*                                                                       *//* Permission is hereby granted for unlimited modification, use, and     *//* distribution.  This software is made available with no warranty of    *//* any kind, express or implied.  This copyright notice must remain      *//* intact in all versions of this software.                              *//*                                                                       *//* The author would appreciate it if any bug fixes and enhancements were *//* to be sent back to him for incorporation into future versions of this *//* software.  Please send changes to tai@iag.hp.com or ken@sdd.hp.com.   *//*************************************************************************/#ifndef lintstatic char RCSid[] = "adjtimed.c,v 3.1 1993/07/06 01:04:45 jbj Exp";#endif/* * Adjust time daemon. * This daemon adjusts the rate of the system clock a la BSD's adjtime(). * The adjtime() routine uses SYSV messages to communicate with this daemon. * * Caveat: This emulation uses an undocumented kernel variable.  As such, it * cannot be guaranteed to work in future HP-UX releases.  Fortunately, * it will no longer be needed in HPUX 10.01 and later. */#include <sys/param.h>#include <sys/types.h>#include <sys/ipc.h>#include <sys/msg.h>#include <sys/lock.h>#include <time.h>#include <signal.h>#include <nlist.h>#include <fcntl.h>#include <stdio.h>#include <unistd.h>#include "ntp_syslog.h"#include "ntp_stdlib.h"#include "adjtime.h"double atof (const char *);int InitClockRate (void);int AdjustClockRate (register struct timeval *delta, register struct timeval *olddelta);long GetClockRate (void);int SetClockRate (long);void ResetClockRate (void);void Cleanup (void);void Exit (int);#define MILLION		1000000L/* emacs cc-mode goes nuts if we split the next line... */#define tvtod(tv)	((double)tv.tv_sec + ((double)tv.tv_usec / (double)MILLION))char *progname = NULL;int verbose = 0;int sysdebug = 0;static int mqid;static double oldrate = 0.0;intmain(	int argc,	char *argv[]	){	struct timeval remains;	struct sigvec vec;	MsgBuf msg;	char ch;	int nofork = 0;	int fd;	progname = argv[0];#ifdef LOG_LOCAL6	openlog("adjtimed", LOG_PID, LOG_LOCAL6);#else	openlog("adjtimed", LOG_PID);#endif	while ((ch = ntp_getopt(argc, argv, "hkrvdfp:")) != EOF) {		switch (ch) {		    case 'k':		    case 'r':			if ((mqid = msgget(KEY, 0)) != -1) {				if (msgctl(mqid, IPC_RMID, (struct msqid_ds *)0) == -1) {					msyslog(LOG_ERR, "remove old message queue: %m");					perror("adjtimed: remove old message queue");					exit(1);				}			}			if (ch == 'k')			    exit(0);			break;		    case 'v':			++verbose, nofork = 1;			break;		    case 'd':			++sysdebug;			break;		    case 'f':			nofork = 1;			break;		    case 'p':			fputs("adjtimed: -p option ignored\n", stderr);			break;		    default:			puts("usage: adjtimed -hkrvdf");			puts("-h\thelp");			puts("-k\tkill existing adjtimed, if any");			puts("-r\trestart (kills existing adjtimed, if any)");			puts("-v\tdebug output (repeat for more output)");			puts("-d\tsyslog output (repeat for more output)");			puts("-f\tno fork");			msyslog(LOG_ERR, "usage error");			exit(1);		} /* switch */	} /* while */	if (!nofork) {		switch (fork()) {		    case 0:			close(fileno(stdin));			close(fileno(stdout));			close(fileno(stderr));#ifdef TIOCNOTTY			if ((fd = open("/dev/tty")) != -1) {				ioctl(fd, TIOCNOTTY, 0);				close(fd);			}#else			setpgrp();#endif			break;		    case -1:			msyslog(LOG_ERR, "fork: %m");			perror("adjtimed: fork");			exit(1);		    default:			exit(0);		} /* switch */	} /* if */	if (nofork) {		setvbuf(stdout, NULL, _IONBF, BUFSIZ);		setvbuf(stderr, NULL, _IONBF, BUFSIZ);	}	msyslog(LOG_INFO, "started");	if (verbose) printf("adjtimed: started\n");	if (InitClockRate() == -1)	    Exit(2);	(void)signal(SIGHUP, SIG_IGN);	(void)signal(SIGINT, SIG_IGN);	(void)signal(SIGQUIT, SIG_IGN);	(void)signal(SIGTERM, Cleanup);	vec.sv_handler = ResetClockRate;	vec.sv_flags = 0;	vec.sv_mask = ~0;	sigvector(SIGALRM, &vec, (struct sigvec *)0);	if (msgget(KEY, IPC_CREAT|IPC_EXCL) == -1) {		if (errno == EEXIST) {			msyslog(LOG_ERR, "message queue already exists, use -r to remove it");			fputs("adjtimed: message queue already exists, use -r to remove it\n",			      stderr);			Exit(1);		}		msyslog(LOG_ERR, "create message queue: %m");		perror("adjtimed: create message queue");		Exit(1);	}	if ((mqid = msgget(KEY, 0)) == -1) {		msyslog(LOG_ERR, "get message queue id: %m");		perror("adjtimed: get message queue id");		Exit(1);	}  	/* Lock process in memory to improve response time */	if (plock(PROCLOCK)) {		msyslog(LOG_ERR, "plock: %m");		perror("adjtimed: plock");		Cleanup();	}	/* Also raise process priority.	 * If we do not get run when we want, this leads to bad timekeeping	 * and "Previous time adjustment didn't complete" gripes from xntpd.	 */	if (nice(-10) == -1) {		msyslog(LOG_ERR, "nice: %m");		perror("adjtimed: nice");		Cleanup();	}	for (;;) {		if (msgrcv(mqid, &msg.msgp, MSGSIZE, CLIENT, 0) == -1) {			if (errno == EINTR) continue;			msyslog(LOG_ERR, "read message: %m");			perror("adjtimed: read message");			Cleanup();		}		switch (msg.msgb.code) {		    case DELTA1:		    case DELTA2:			AdjustClockRate(&msg.msgb.tv, &remains);			if (msg.msgb.code == DELTA2) {				msg.msgb.tv = remains;				msg.msgb.mtype = SERVER;				while (msgsnd(mqid, &msg.msgp, MSGSIZE, 0) == -1) {					if (errno == EINTR) continue;					msyslog(LOG_ERR, "send message: %m");					perror("adjtimed: send message");					Cleanup();				}			}			if (remains.tv_sec + remains.tv_usec != 0L) {				if (verbose) {					printf("adjtimed: previous correction remaining %.6fs\n",					       tvtod(remains));				}				if (sysdebug) {					msyslog(LOG_INFO, "previous correction remaining %.6fs",						tvtod(remains));				}			}			break;		    default:			fprintf(stderr, "adjtimed: unknown message code %d\n", msg.msgb.code);			msyslog(LOG_ERR, "unknown message code %d", msg.msgb.code);		} /* switch */	} /* loop */} /* main *//* * Default clock rate (old_tick). */#define DEFAULT_RATE	(MILLION / HZ)#define UNKNOWN_RATE	0L#define TICK_ADJ	5	/* standard adjustment rate, microsec/tick */static long default_rate = DEFAULT_RATE;static long tick_rate = HZ;	/* ticks per sec */static long slew_rate = TICK_ADJ * HZ; /* in microsec/sec */intAdjustClockRate(	register struct timeval *delta,	register struct timeval *olddelta	){	register long rate, dt, leftover;	struct itimerval period, remains; 	dt = (delta->tv_sec * MILLION) + delta->tv_usec;	if (verbose)	    printf("adjtimed: new correction %.6fs\n", (double)dt / (double)MILLION);	if (sysdebug)	    msyslog(LOG_INFO, "new correction %.6fs", (double)dt / (double)MILLION);	if (verbose > 2) printf("adjtimed: leftover %ldus\n", leftover);	if (sysdebug > 2) msyslog(LOG_INFO, "leftover %ldus", leftover);	rate = dt;	/*	 * Apply a slew rate of slew_rate over a period of dt/slew_rate seconds.	 */	if (dt > 0) {		rate = slew_rate;	} else {		rate = -slew_rate;		dt = -dt;	}	period.it_value.tv_sec = dt / slew_rate;	period.it_value.tv_usec = (dt % slew_rate) * (MILLION / slew_rate);	/*	 * Note: we assume the kernel will convert the specified period into ticks	 * using the modified clock rate rather than an assumed nominal clock rate,	 * and therefore will generate the timer interrupt after the specified	 * number of true seconds, not skewed seconds.	 */	if (verbose > 1)	    printf("adjtimed: will be complete in %lds %ldus\n",		   period.it_value.tv_sec, period.it_value.tv_usec);	if (sysdebug > 1)	    msyslog(LOG_INFO, "will be complete in %lds %ldus",		    period.it_value.tv_sec, period.it_value.tv_usec);	/*	 * adjust the clock rate	 */	if (dt) {		if (SetClockRate((rate / tick_rate) + default_rate) == -1) {			msyslog(LOG_ERR, "set clock rate: %m");			perror("adjtimed: set clock rate");		}	}	/*	 * start the timer	 * (do this after changing the rate because the period has been rounded down)	 */	period.it_interval.tv_sec = period.it_interval.tv_usec = 0L;	setitimer(ITIMER_REAL, &period, &remains);	/*	 * return old delta	 */	if (olddelta) {		dt = ((remains.it_value.tv_sec * MILLION) + remains.it_value.tv_usec) *			oldrate;		olddelta->tv_sec = dt / MILLION;		olddelta->tv_usec = dt - (olddelta->tv_sec * MILLION); 	}	oldrate = (double)rate / (double)MILLION;	return(0);} /* AdjustClockRate */static struct nlist nl[] = {#ifdef __hp9000s800#ifdef PRE7_0	{ "tick" },#else	{ "old_tick" },#endif#else	{ "_old_tick" },#endif	{ "" }};static int kmem;/* * The return value is the clock rate in old_tick units or -1 if error. */longGetClockRate(void){	long rate, mask;	if (lseek(kmem, (off_t)nl[0].n_value, 0) == -1L)	    return (-1L);	mask = sigblock(sigmask(SIGALRM));	if (read(kmem, (caddr_t)&rate, sizeof(rate)) != sizeof(rate))	    rate = UNKNOWN_RATE;	sigsetmask(mask);	return (rate);} /* GetClockRate *//* * The argument is the new rate in old_tick units. */intSetClockRate(	long rate	){	long mask;	if (lseek(kmem, (off_t)nl[0].n_value, 0) == -1L)	    return (-1);	mask = sigblock(sigmask(SIGALRM));	if (write(kmem, (caddr_t)&rate, sizeof(rate)) != sizeof(rate)) {		sigsetmask(mask);		return (-1);	}	sigsetmask(mask);	if (rate != default_rate) {		if (verbose > 3) {			printf("adjtimed: clock rate (%lu) %ldus/s\n", rate,			       (rate - default_rate) * tick_rate);		}		if (sysdebug > 3) {			msyslog(LOG_INFO, "clock rate (%lu) %ldus/s", rate,				(rate - default_rate) * tick_rate);		}	}	return (0);} /* SetClockRate */intInitClockRate(void){	if ((kmem = open("/dev/kmem", O_RDWR)) == -1) {		msyslog(LOG_ERR, "open(/dev/kmem): %m");		perror("adjtimed: open(/dev/kmem)");		return (-1);	}	nlist("/hp-ux", nl);	if (nl[0].n_type == 0) {		fputs("adjtimed: /hp-ux has no symbol table\n", stderr);		msyslog(LOG_ERR, "/hp-ux has no symbol table");		return (-1);	}	/*	 * Set the default to the system's original value	 */	default_rate = GetClockRate();	if (default_rate == UNKNOWN_RATE) default_rate = DEFAULT_RATE;	tick_rate = (MILLION / default_rate);	slew_rate = TICK_ADJ * tick_rate;	fprintf(stderr,"default_rate=%ld, tick_rate=%ld, slew_rate=%ld\n",default_rate,tick_rate,slew_rate);	return (0);} /* InitClockRate *//* * Reset the clock rate to the default value. */voidResetClockRate(void){	struct itimerval it;	it.it_value.tv_sec = it.it_value.tv_usec = 0L;	setitimer(ITIMER_REAL, &it, (struct itimerval *)0);	if (verbose > 2) puts("adjtimed: resetting the clock");	if (sysdebug > 2) msyslog(LOG_INFO, "resetting the clock");	if (GetClockRate() != default_rate) {		if (SetClockRate(default_rate) == -1) {			msyslog(LOG_ERR, "set clock rate: %m");			perror("adjtimed: set clock rate");		}	}	oldrate = 0.0;} /* ResetClockRate */voidCleanup(void){	ResetClockRate();	if (msgctl(mqid, IPC_RMID, (struct msqid_ds *)0) == -1) {		if (errno != EINVAL) {			msyslog(LOG_ERR, "remove message queue: %m");			perror("adjtimed: remove message queue");		}	}	Exit(2);} /* Cleanup */voidExit(status)     int status;{	msyslog(LOG_ERR, "terminated");	closelog();	if (kmem != -1) close(kmem);	exit(status);} /* Exit */

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -