xref: /illumos-gate/usr/src/cmd/cmd-inet/usr.sbin/snoop/snoop_ntp.c (revision 7a6d80f1660abd4755c68cbd094d4a914681d26e)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License, Version 1.0 only
6  * (the "License").  You may not use this file except in compliance
7  * with the License.
8  *
9  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
10  * or http://www.opensolaris.org/os/licensing.
11  * See the License for the specific language governing permissions
12  * and limitations under the License.
13  *
14  * When distributing Covered Code, include this CDDL HEADER in each
15  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
16  * If applicable, add the following below this CDDL HEADER, with the
17  * fields enclosed by brackets "[]" replaced with your own identifying
18  * information: Portions Copyright [yyyy] [name of copyright owner]
19  *
20  * CDDL HEADER END
21  */
22 /*
23  * Copyright 2005 Sun Microsystems, Inc.  All rights reserved.
24  * Use is subject to license terms.
25  */
26 
27 #include <fcntl.h>
28 #include <sys/socket.h>
29 #include <sys/sysmacros.h>
30 #include <netinet/in.h>
31 #include <netdb.h>
32 #include <stdio.h>
33 #include <string.h>
34 #include <tzfile.h>
35 #include "snoop.h"
36 #include "ntp.h"
37 
38 /*
39  * In verbose mode, how many octets of the control-mode data payload
40  * are displayed per line of output.  The value 64 fits well on an
41  * 80-column screen and, as a power of 2, is easily correlated to
42  * hexadecimal output.
43  */
44 #define	OCTETS_PER_LINE	64
45 
46 extern char *dlc_header;
47 
48 static	char	*show_leap(int);
49 static	char	*show_mode(int);
50 static	char	*show_ref(int, ulong_t);
51 static	char	*show_time(struct l_fixedpt);
52 static	double	s_fixed_to_double(struct s_fixedpt *);
53 static	char	*iso_date_time(time_t);
54 static	char	*show_operation(int);
55 
56 int
57 interpret_ntp(int flags, struct ntpdata *ntp_pkt, int fraglen)
58 {
59 	unsigned int	i, j, macbytes;
60 	unsigned int	proto_version;
61 	unsigned int	datalen;
62 	unsigned int	linelen = OCTETS_PER_LINE;
63 	unsigned int	sofar = 0;
64 
65 	char	*datap;
66 	char	hbuf[2 * MAC_OCTETS_MAX + 1];
67 	static	char *hexstr = "0123456789ABCDEF";
68 
69 	union	ntp_pkt_buf {
70 		struct	ntpdata ntp_msg;
71 		union ntpc_buf {
72 			struct	ntp_control chdr;
73 			uchar_t	data2[NTPC_DATA_MAXLEN - 1];
74 		} ntpc_msg;
75 		union ntpp_buf {
76 			struct	ntp_private phdr;
77 			uchar_t	data2[1];
78 		} ntpp_msg;
79 	} fragbuf;
80 
81 	struct	ntpdata		*ntp = &fragbuf.ntp_msg;
82 	struct	ntp_control	*ntpc = (struct ntp_control *)&fragbuf.ntpc_msg;
83 	struct	ntp_private	*ntpp = (struct ntp_private *)&fragbuf.ntpp_msg;
84 
85 	/*
86 	 * Copying packet contents into a local buffer avoids
87 	 * problems of interpretation if the packet is truncated.
88 	 */
89 	(void) memcpy(&fragbuf, ntp_pkt, MIN(sizeof (fragbuf), fraglen));
90 
91 	if (flags & F_SUM) {
92 		switch (ntp->li_vn_mode & NTPMODEMASK) {
93 		case MODE_SYM_ACT:
94 		case MODE_SYM_PAS:
95 		case MODE_CLIENT:
96 		case MODE_SERVER:
97 		case MODE_BROADCAST:
98 		    (void) sprintf(get_sum_line(),
99 			"NTP  %s [st=%hd] (%s)",
100 			show_mode(ntp->li_vn_mode & NTPMODEMASK),
101 			ntp->stratum,
102 			show_time(ntp->xmt));
103 		    break;
104 		case MODE_CONTROL:
105 		    (void) sprintf(get_sum_line(),
106 			"NTP  %s "
107 			"(Flags/op=0x%02x Seq=%hu Status=0x%04hx Assoc=%hu)",
108 			show_mode(ntpc->li_vn_mode & NTPMODEMASK),
109 			ntpc->r_m_e_op,
110 			ntohs(ntpc->sequence),
111 			ntohs(ntpc->status),
112 			ntohs(ntpc->associd));
113 		    break;
114 		default:
115 		    (void) sprintf(get_sum_line(),
116 			"NTP  %s",
117 			show_mode(ntpp->rm_vn_mode & NTPMODEMASK));
118 		    break;
119 		}
120 	}
121 
122 	proto_version = (ntp->li_vn_mode & VERSIONMASK) >> 3;
123 
124 	if (flags & F_DTAIL) {
125 		show_header("NTP:  ", "Network Time Protocol", fraglen);
126 		show_space();
127 		switch (ntp->li_vn_mode & NTPMODEMASK) {
128 		case MODE_SYM_ACT:
129 		case MODE_SYM_PAS:
130 		case MODE_CLIENT:
131 		case MODE_SERVER:
132 		case MODE_BROADCAST:
133 		    (void) sprintf(get_line((char *)(uintptr_t)ntp->li_vn_mode -
134 			dlc_header, 1),
135 			"Leap    = 0x%x (%s)",
136 			(int)(ntp->li_vn_mode & LEAPMASK) >> 6,
137 			show_leap(ntp->li_vn_mode & LEAPMASK));
138 		    (void) sprintf(get_line((char *)(uintptr_t)ntp->li_vn_mode -
139 			dlc_header, 1),
140 			"Version = %lu", proto_version);
141 		    (void) sprintf(get_line((char *)(uintptr_t)ntp->li_vn_mode -
142 			dlc_header, 1),
143 			"Mode    = %hu (%s)",
144 			ntp->li_vn_mode & NTPMODEMASK,
145 			show_mode(ntp->li_vn_mode & NTPMODEMASK));
146 		    (void) sprintf(get_line((char *)(uintptr_t)ntp->stratum -
147 			dlc_header, 1),
148 			"Stratum = %d (%s)",
149 			ntp->stratum,
150 			ntp->stratum == 0 ? "unspecified" :
151 			ntp->stratum == 1 ? "primary reference" :
152 			"secondary reference");
153 		    (void) sprintf(get_line((char *)(uintptr_t)ntp->ppoll -
154 			dlc_header, 1),	"Poll    = %hu", ntp->ppoll);
155 		    (void) sprintf(get_line((char *)(uintptr_t)ntp->precision -
156 			dlc_header, 1),
157 			"Precision = %d seconds",
158 			ntp->precision);
159 		    (void) sprintf(get_line(
160 			(char *)(uintptr_t)ntp->distance.int_part -
161 			dlc_header, 1),
162 			"Synchronizing distance   = 0x%04x.%04x  (%f)",
163 			ntohs(ntp->distance.int_part),
164 			ntohs(ntp->distance.fraction),
165 			s_fixed_to_double(&ntp->distance));
166 		    (void) sprintf(get_line(
167 			(char *)(uintptr_t)ntp->dispersion.int_part -
168 			dlc_header, 1),
169 			"Synchronizing dispersion = 0x%04x.%04x  (%f)",
170 			ntohs(ntp->dispersion.int_part),
171 			ntohs(ntp->dispersion.fraction),
172 			s_fixed_to_double(&ntp->dispersion));
173 		    (void) sprintf(get_line((char *)(uintptr_t)ntp->refid -
174 			dlc_header, 1), "Reference clock = %s",
175 			show_ref(ntp->stratum, ntp->refid));
176 
177 		    (void) sprintf(get_line(
178 			(char *)(uintptr_t)ntp->reftime.int_part - dlc_header,
179 			1), "Reference time = 0x%08lx.%08lx (%s)",
180 			ntohl(ntp->reftime.int_part),
181 			ntohl(ntp->reftime.fraction),
182 			show_time(ntp->reftime));
183 
184 		    (void) sprintf(get_line(
185 			(char *)(uintptr_t)ntp->org.int_part - dlc_header, 1),
186 			"Originate time = 0x%08lx.%08lx (%s)",
187 			ntohl(ntp->org.int_part),
188 			ntohl(ntp->org.fraction),
189 			show_time(ntp->org));
190 
191 		    (void) sprintf(get_line(
192 			(char *)(uintptr_t)ntp->rec.int_part - dlc_header, 1),
193 			"Receive   time = 0x%08lx.%08lx (%s)",
194 			ntohl(ntp->rec.int_part),
195 			ntohl(ntp->rec.fraction),
196 			show_time(ntp->rec));
197 
198 		    (void) sprintf(get_line(
199 			(char *)(uintptr_t)ntp->xmt.int_part - dlc_header, 1),
200 			"Transmit  time = 0x%08lx.%08lx (%s)",
201 			ntohl(ntp->xmt.int_part),
202 			ntohl(ntp->xmt.fraction),
203 			show_time(ntp->xmt));
204 
205 		    if (proto_version > 3 ||
206 			fraglen < (LEN_PKT_NOMAC + MAC_OCTETS_MIN)) {
207 				/*
208 				 * A newer protocol version we can't parse,
209 				 * or v3 packet with no valid authentication.
210 				 */
211 				break;
212 		    }
213 		    (void) sprintf(get_line((char *)ntp->keyid -
214 			dlc_header, 1),
215 			"Key ID  = %8lu", ntohl(ntp->keyid));
216 
217 		    macbytes = fraglen - (LEN_PKT_NOMAC + sizeof (uint32_t));
218 
219 		    for (i = 0, j = 0; i < macbytes; i++) {
220 			    hbuf[j++] = hexstr[ntp->mac[i] >> 4 & 0x0f];
221 			    hbuf[j++] = hexstr[ntp->mac[i] & 0x0f];
222 		    }
223 		    hbuf[j] = '\0';
224 		    (void) sprintf(get_line((char *)ntp->mac -
225 			dlc_header, 1),
226 			"Authentication code = %s", hbuf);
227 		    break;
228 
229 		case MODE_CONTROL:
230 		    /* NTP Control Message, mode 6 */
231 
232 		    (void) sprintf(get_line((char *)(uintptr_t)ntp->li_vn_mode -
233 			dlc_header, 1),
234 			"Leap    = 0x%x (%s)",
235 			(int)(ntp->li_vn_mode & LEAPMASK) >> 6,
236 			show_leap(ntp->li_vn_mode & LEAPMASK));
237 		    (void) sprintf(get_line((char *)(uintptr_t)ntp->li_vn_mode -
238 			dlc_header, 1),
239 			"Version = %lu", proto_version);
240 		    (void) sprintf(get_line((char *)(uintptr_t)ntp->li_vn_mode -
241 			dlc_header, 1),
242 			"Mode    = %hu (%s)",
243 			ntp->li_vn_mode & NTPMODEMASK,
244 			show_mode(ntp->li_vn_mode & NTPMODEMASK));
245 		    (void) sprintf(get_line((char *)(uintptr_t)ntpc->r_m_e_op -
246 			dlc_header, 1),
247 			"Flags and operation code = 0x%02x",
248 			ntpc->r_m_e_op);
249 		    (void) sprintf(get_line((char *)(uintptr_t)ntpc->r_m_e_op -
250 			dlc_header, 1),
251 			"      %s",
252 			getflag(ntpc->r_m_e_op, CTL_RESPONSE, "response",
253 			"request"));
254 		    (void) sprintf(get_line((char *)(uintptr_t)ntpc->r_m_e_op -
255 			dlc_header, 1),
256 			"      %s",
257 			getflag(ntpc->r_m_e_op, CTL_ERROR, "error",
258 			"success"));
259 		    (void) sprintf(get_line((char *)(uintptr_t)ntpc->r_m_e_op -
260 			dlc_header, 1),
261 			"      %s",
262 			getflag(ntpc->r_m_e_op, CTL_MORE, "more",
263 			"no more"));
264 		    (void) sprintf(get_line((char *)(uintptr_t)ntpc->r_m_e_op -
265 			dlc_header, 1),
266 			"      ...x xxxx = %hd (%s)",
267 			ntpc->r_m_e_op & CTL_OP_MASK,
268 			show_operation(ntpc->r_m_e_op & CTL_OP_MASK));
269 		    (void) sprintf(get_line((char *)(uintptr_t)ntpc->sequence -
270 			dlc_header, 1),
271 			"Sequence = %hu",
272 			ntohs(ntpc->sequence));
273 		    (void) sprintf(get_line((char *)(uintptr_t)ntpc->status -
274 			dlc_header, 1),
275 			"Status = 0x%04hx",
276 			ntohs(ntpc->status));
277 		    (void) sprintf(get_line((char *)(uintptr_t)ntpc->associd -
278 			dlc_header, 1),
279 			"Assoc ID = %hu",
280 			ntohs(ntpc->associd));
281 		    (void) sprintf(get_line((char *)(uintptr_t)ntpc->offset -
282 			dlc_header, 1),
283 			"Data offset = %hu",
284 			ntohs(ntpc->offset));
285 		    (void) sprintf(get_line((char *)(uintptr_t)ntpc->count -
286 			dlc_header, 1),
287 			"Data bytes = %hu",
288 			ntohs(ntpc->count));
289 		    datalen = ntohs(ntpc->count);
290 		    if (datalen == 0) {
291 			    break;
292 		    } else if (datalen > NTPC_DATA_MAXLEN) {
293 			    datalen = NTPC_DATA_MAXLEN;
294 		    }
295 		    show_space();
296 		    datap = (char *)ntpc->data;
297 		    do {
298 			    (void) sprintf(get_line(datap -
299 				dlc_header, 1),
300 				"\"%s\"",
301 				show_string(datap, linelen, datalen));
302 			    sofar += linelen;
303 			    datap += linelen;
304 			    if ((sofar + linelen) > datalen) {
305 				    linelen = datalen - sofar;
306 			    }
307 		    } while (sofar < datalen);
308 		    show_trailer();
309 		    break;
310 
311 		case MODE_PRIVATE:
312 		    /* NTP Private Message, mode 7 */
313 
314 		    (void) sprintf(get_line(
315 			(char *)(uintptr_t)ntpp->rm_vn_mode - dlc_header, 1),
316 			"Version = %hu", INFO_VERSION(ntpp->rm_vn_mode));
317 		    (void) sprintf(get_line(
318 			(char *)(uintptr_t)ntpp->rm_vn_mode - dlc_header, 1),
319 			"Mode    = %hu (%s)", INFO_MODE(ntpp->rm_vn_mode),
320 			show_mode(INFO_MODE(ntpp->rm_vn_mode)));
321 		    (void) sprintf(get_line(
322 			(char *)(uintptr_t)ntpp->rm_vn_mode - dlc_header, 1),
323 			"Flags = 0x%02hx", ntpp->rm_vn_mode);
324 		    (void) sprintf(get_line(
325 			(char *)(uintptr_t)ntpp->rm_vn_mode - dlc_header, 1),
326 			"      %s",
327 			getflag(ntpp->rm_vn_mode, RESP_BIT, "response",
328 			"request"));
329 		    (void) sprintf(get_line(
330 			(char *)(uintptr_t)ntpp->rm_vn_mode - dlc_header, 1),
331 			"      %s",
332 			getflag(ntpp->rm_vn_mode, MORE_BIT, "more", "no more"));
333 		    (void) sprintf(get_line((char *)(uintptr_t)ntpp->auth_seq -
334 			dlc_header, 1),
335 			"Authentication and sequence = 0x%02x", ntpp->auth_seq);
336 		    (void) sprintf(get_line((char *)(uintptr_t)ntpp->auth_seq -
337 			dlc_header, 1),
338 			"      %s",
339 			getflag(ntpp->auth_seq, AUTH_BIT, "authenticated",
340 			"unauthenticated"));
341 		    (void) sprintf(get_line((char *)(uintptr_t)ntpp->auth_seq -
342 			dlc_header, 1),
343 			"      .xxx xxxx = %hu (sequence number)",
344 			INFO_SEQ(ntpp->auth_seq));
345 		    (void) sprintf(get_line(
346 			(char *)(uintptr_t)ntpp->implementation - dlc_header,
347 			1), "Implementation = %hu", ntpp->implementation);
348 		    (void) sprintf(get_line((char *)(uintptr_t)ntpp->request -
349 			dlc_header, 1), "Request = %hu", ntpp->request);
350 		    (void) sprintf(get_line(
351 			(char *)(uintptr_t)ntpp->err_nitems - dlc_header, 1),
352 			"Error = %hu", INFO_ERR(ntpp->err_nitems));
353 		    (void) sprintf(get_line(
354 			(char *)(uintptr_t)ntpp->err_nitems - dlc_header, 1),
355 			"Items = %hu", INFO_NITEMS(ntpp->err_nitems));
356 		    (void) sprintf(get_line(
357 			(char *)(uintptr_t)ntpp->mbz_itemsize - dlc_header, 1),
358 			"Item size = %hu", INFO_ITEMSIZE(ntpp->mbz_itemsize));
359 		    break;
360 
361 		default:
362 		    /* Unknown mode */
363 		    (void) sprintf(get_line((char *)(uintptr_t)ntp->li_vn_mode -
364 			dlc_header, 1),	"Mode    = %hu (%s)",
365 			ntp->li_vn_mode & NTPMODEMASK,
366 			show_mode(ntp->li_vn_mode & NTPMODEMASK));
367 		    break;
368 		}
369 	}
370 
371 	return (fraglen);
372 }
373 
374 char *
375 show_leap(int leap)
376 {
377 	switch (leap) {
378 	case NO_WARNING: return ("OK");
379 	case PLUS_SEC:	return ("add a second (61 seconds)");
380 	case MINUS_SEC: return ("minus a second (59 seconds)");
381 	case ALARM:	return ("alarm condition (clock unsynchronized)");
382 	default:	return ("unknown");
383 	}
384 }
385 
386 char *
387 show_mode(int mode)
388 {
389 	switch (mode) {
390 	case MODE_UNSPEC:	return ("unspecified");
391 	case MODE_SYM_ACT:	return ("symmetric active");
392 	case MODE_SYM_PAS:	return ("symmetric passive");
393 	case MODE_CLIENT:	return ("client");
394 	case MODE_SERVER:	return ("server");
395 	case MODE_BROADCAST:	return ("broadcast");
396 	case MODE_CONTROL:	return ("control");
397 	case MODE_PRIVATE:	return ("private");
398 	default:		return ("unknown");
399 	}
400 }
401 
402 char *
403 show_ref(int mode, ulong_t refid)
404 {
405 	static char buff[MAXHOSTNAMELEN + 32];
406 	struct in_addr host;
407 	extern char *inet_ntoa();
408 
409 	switch (mode) {
410 	case 0:
411 	case 1:
412 		(void) strncpy(buff, (char *)&refid, 4);
413 		buff[4] = '\0';
414 		break;
415 
416 	default:
417 		host.s_addr = refid;
418 		(void) sprintf(buff, "%s (%s)",
419 		    inet_ntoa(host),
420 		    addrtoname(AF_INET, &host));
421 		break;
422 	}
423 
424 	return (buff);
425 }
426 
427 /*
428  *  Here we have to worry about the high order bit being signed
429  */
430 double
431 s_fixed_to_double(struct s_fixedpt *t)
432 {
433 	double a;
434 
435 	if (ntohs(t->int_part) & 0x8000) {
436 		a = ntohs((int)(~t->fraction) & 0xFFFF);
437 		a = a / 65536.0;	/* shift dec point over by 16 bits */
438 		a +=  ntohs((int)(~t->int_part) & 0xFFFF);
439 		a = -a;
440 	} else {
441 		a = ntohs(t->fraction);
442 		a = a / 65536.0;	/* shift dec point over by 16 bits */
443 		a += ntohs(t->int_part);
444 	}
445 	return (a);
446 }
447 
448 /*
449  * Consistent with RFC-3339, ISO 8601.
450  */
451 char *
452 iso_date_time(time_t input_time)
453 {
454 	struct tm	*time_parts;
455 	static char	tbuf[sizeof ("yyyy-mm-dd hh:mm:ss")];
456 
457 	time_parts = localtime(&input_time);
458 	(void) strftime(tbuf, sizeof (tbuf), "%Y-%m-%d %H:%M:%S", time_parts);
459 	return (tbuf);
460 }
461 
462 /*
463  * The base of NTP timestamps is 1900-01-01 00:00:00.00000
464  */
465 char *
466 show_time(struct l_fixedpt pkt_time)
467 {
468 	struct l_fixedpt net_time;
469 	unsigned long	fracsec;
470 	static char	buff[32];
471 
472 	if (pkt_time.int_part == 0) {
473 		buff[0] = '\0';
474 		return (buff);
475 	}
476 
477 	net_time.int_part = ntohl(pkt_time.int_part) - JAN_1970;
478 	net_time.fraction = ntohl(pkt_time.fraction);
479 
480 	fracsec = net_time.fraction / 42949;	/* fract / (2**32/10**6) */
481 
482 	(void) strlcpy(buff, iso_date_time(net_time.int_part), sizeof (buff));
483 	(void) snprintf(buff, sizeof (buff), "%s.%05lu", buff, fracsec);
484 
485 	return (buff);
486 }
487 
488 char *
489 show_operation(int op)
490 {
491 	switch (op) {
492 	case CTL_OP_UNSPEC:	return ("unspecified");
493 	case CTL_OP_READSTAT:	return ("read stats");
494 	case CTL_OP_READVAR:	return ("read var");
495 	case CTL_OP_WRITEVAR:	return ("write var");
496 	case CTL_OP_READCLOCK:	return ("read clock");
497 	case CTL_OP_WRITECLOCK: return ("write clock");
498 	case CTL_OP_SETTRAP:	return ("set trap");
499 	case CTL_OP_ASYNCMSG:	return ("async msg");
500 	case CTL_OP_UNSETTRAP:	return ("unset trap");
501 	default:		return ("unknown");
502 	}
503 }
504