xref: /freebsd/contrib/tcpdump/print-zep.c (revision ee67461e56828dd1f8de165947ba83f6d9148a87)
1*ee67461eSJoseph Mingrone /*
2*ee67461eSJoseph Mingrone  * Copyright (c) 1988, 1989, 1990, 1991, 1992, 1993, 1994, 1995, 1996, 1997
3*ee67461eSJoseph Mingrone  *	The Regents of the University of California.  All rights reserved.
4*ee67461eSJoseph Mingrone  *
5*ee67461eSJoseph Mingrone  * Redistribution and use in source and binary forms, with or without
6*ee67461eSJoseph Mingrone  * modification, are permitted provided that: (1) source code distributions
7*ee67461eSJoseph Mingrone  * retain the above copyright notice and this paragraph in its entirety, (2)
8*ee67461eSJoseph Mingrone  * distributions including binary code include the above copyright notice and
9*ee67461eSJoseph Mingrone  * this paragraph in its entirety in the documentation or other materials
10*ee67461eSJoseph Mingrone  * provided with the distribution, and (3) all advertising materials mentioning
11*ee67461eSJoseph Mingrone  * features or use of this software display the following acknowledgement:
12*ee67461eSJoseph Mingrone  * ``This product includes software developed by the University of California,
13*ee67461eSJoseph Mingrone  * Lawrence Berkeley Laboratory and its contributors.'' Neither the name of
14*ee67461eSJoseph Mingrone  * the University nor the names of its contributors may be used to endorse
15*ee67461eSJoseph Mingrone  * or promote products derived from this software without specific prior
16*ee67461eSJoseph Mingrone  * written permission.
17*ee67461eSJoseph Mingrone  * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED
18*ee67461eSJoseph Mingrone  * WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF
19*ee67461eSJoseph Mingrone  * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
20*ee67461eSJoseph Mingrone  */
21*ee67461eSJoseph Mingrone 
22*ee67461eSJoseph Mingrone /* \summary: ZigBee Encapsulation Protocol (ZEP) printer */
23*ee67461eSJoseph Mingrone 
24*ee67461eSJoseph Mingrone #ifdef HAVE_CONFIG_H
25*ee67461eSJoseph Mingrone #include <config.h>
26*ee67461eSJoseph Mingrone #endif
27*ee67461eSJoseph Mingrone 
28*ee67461eSJoseph Mingrone #include "netdissect-stdinc.h"
29*ee67461eSJoseph Mingrone 
30*ee67461eSJoseph Mingrone #define ND_LONGJMP_FROM_TCHECK
31*ee67461eSJoseph Mingrone #include "netdissect.h"
32*ee67461eSJoseph Mingrone 
33*ee67461eSJoseph Mingrone #include "extract.h"
34*ee67461eSJoseph Mingrone 
35*ee67461eSJoseph Mingrone /* From wireshark packet-zep.c:
36*ee67461eSJoseph Mingrone  *
37*ee67461eSJoseph Mingrone  ***********************************************************************
38*ee67461eSJoseph Mingrone  *
39*ee67461eSJoseph Mingrone  * ZEP Packets must be received in the following format:
40*ee67461eSJoseph Mingrone  *
41*ee67461eSJoseph Mingrone  * |UDP Header|  ZEP Header |IEEE 802.15.4 Packet|
42*ee67461eSJoseph Mingrone  * | 8 bytes  | 16/32 bytes |    <= 127 bytes    |
43*ee67461eSJoseph Mingrone  *
44*ee67461eSJoseph Mingrone  ***********************************************************************
45*ee67461eSJoseph Mingrone  *
46*ee67461eSJoseph Mingrone  * ZEP v1 Header will have the following format:
47*ee67461eSJoseph Mingrone  * |Preamble|Version|Channel ID|Device ID|CRC/LQI Mode|LQI Val|Reserved|Length|
48*ee67461eSJoseph Mingrone  * |2 bytes |1 byte |  1 byte  | 2 bytes |   1 byte   |1 byte |7 bytes |1 byte|
49*ee67461eSJoseph Mingrone  *
50*ee67461eSJoseph Mingrone  * ZEP v2 Header will have the following format (if type=1/Data):
51*ee67461eSJoseph Mingrone  * |Prmbl|Ver  |Type |ChnlID|DevID|C/L Mode|LQI|NTP TS|Seq#|Res |Len|
52*ee67461eSJoseph Mingrone  * | 2   | 1   | 1   | 1    | 2   | 1      | 1 | 8    | 4  | 10 | 1 |
53*ee67461eSJoseph Mingrone  *
54*ee67461eSJoseph Mingrone  * ZEP v2 Header will have the following format (if type=2/Ack):
55*ee67461eSJoseph Mingrone  * |Preamble|Version| Type |Sequence#|
56*ee67461eSJoseph Mingrone  * |2 bytes |1 byte |1 byte| 4 bytes |
57*ee67461eSJoseph Mingrone  *------------------------------------------------------------
58*ee67461eSJoseph Mingrone  */
59*ee67461eSJoseph Mingrone 
60*ee67461eSJoseph Mingrone #define     JAN_1970        2208988800U
61*ee67461eSJoseph Mingrone 
62*ee67461eSJoseph Mingrone /* Print timestamp */
63*ee67461eSJoseph Mingrone static void zep_print_ts(netdissect_options *ndo, const u_char *p)
64*ee67461eSJoseph Mingrone {
65*ee67461eSJoseph Mingrone 	int32_t i;
66*ee67461eSJoseph Mingrone 	uint32_t uf;
67*ee67461eSJoseph Mingrone 	uint32_t f;
68*ee67461eSJoseph Mingrone 	float ff;
69*ee67461eSJoseph Mingrone 
70*ee67461eSJoseph Mingrone 	i = GET_BE_U_4(p);
71*ee67461eSJoseph Mingrone 	uf = GET_BE_U_4(p + 4);
72*ee67461eSJoseph Mingrone 	ff = (float) uf;
73*ee67461eSJoseph Mingrone 	if (ff < 0.0)           /* some compilers are buggy */
74*ee67461eSJoseph Mingrone 		ff += FMAXINT;
75*ee67461eSJoseph Mingrone 	ff = (float) (ff / FMAXINT); /* shift radix point by 32 bits */
76*ee67461eSJoseph Mingrone 	f = (uint32_t) (ff * 1000000000.0);  /* treat fraction as parts per
77*ee67461eSJoseph Mingrone 						billion */
78*ee67461eSJoseph Mingrone 	ND_PRINT("%u.%09d", i, f);
79*ee67461eSJoseph Mingrone 
80*ee67461eSJoseph Mingrone 	/*
81*ee67461eSJoseph Mingrone 	 * print the time in human-readable format.
82*ee67461eSJoseph Mingrone 	 */
83*ee67461eSJoseph Mingrone 	if (i) {
84*ee67461eSJoseph Mingrone 		time_t seconds = i - JAN_1970;
85*ee67461eSJoseph Mingrone 		char time_buf[128];
86*ee67461eSJoseph Mingrone 
87*ee67461eSJoseph Mingrone 		ND_PRINT(" (%s)",
88*ee67461eSJoseph Mingrone 		    nd_format_time(time_buf, sizeof (time_buf), "%Y/%m/%d %H:%M:%S",
89*ee67461eSJoseph Mingrone 		      localtime(&seconds)));
90*ee67461eSJoseph Mingrone 	}
91*ee67461eSJoseph Mingrone }
92*ee67461eSJoseph Mingrone 
93*ee67461eSJoseph Mingrone /*
94*ee67461eSJoseph Mingrone  * Main function to print packets.
95*ee67461eSJoseph Mingrone  */
96*ee67461eSJoseph Mingrone 
97*ee67461eSJoseph Mingrone void
98*ee67461eSJoseph Mingrone zep_print(netdissect_options *ndo,
99*ee67461eSJoseph Mingrone 	  const u_char *bp, u_int len)
100*ee67461eSJoseph Mingrone {
101*ee67461eSJoseph Mingrone 	uint8_t version, inner_len;
102*ee67461eSJoseph Mingrone 	uint32_t seq_no;
103*ee67461eSJoseph Mingrone 
104*ee67461eSJoseph Mingrone 	ndo->ndo_protocol = "zep";
105*ee67461eSJoseph Mingrone 
106*ee67461eSJoseph Mingrone 	nd_print_protocol_caps(ndo);
107*ee67461eSJoseph Mingrone 
108*ee67461eSJoseph Mingrone 	/* Preamble Code (must be "EX") */
109*ee67461eSJoseph Mingrone 	if (GET_U_1(bp) != 'E' || GET_U_1(bp + 1) != 'X') {
110*ee67461eSJoseph Mingrone 		ND_PRINT(" [Preamble Code: ");
111*ee67461eSJoseph Mingrone 		fn_print_char(ndo, GET_U_1(bp));
112*ee67461eSJoseph Mingrone 		fn_print_char(ndo, GET_U_1(bp + 1));
113*ee67461eSJoseph Mingrone 		ND_PRINT("]");
114*ee67461eSJoseph Mingrone 		nd_print_invalid(ndo);
115*ee67461eSJoseph Mingrone 		return;
116*ee67461eSJoseph Mingrone 	}
117*ee67461eSJoseph Mingrone 
118*ee67461eSJoseph Mingrone 	version = GET_U_1(bp + 2);
119*ee67461eSJoseph Mingrone 	ND_PRINT("v%u ", version);
120*ee67461eSJoseph Mingrone 
121*ee67461eSJoseph Mingrone 	if (version == 1) {
122*ee67461eSJoseph Mingrone 		/* ZEP v1 packet. */
123*ee67461eSJoseph Mingrone 		ND_LCHECK_U(len, 16);
124*ee67461eSJoseph Mingrone 		ND_PRINT("Channel ID %u, Device ID 0x%04x, ",
125*ee67461eSJoseph Mingrone 			 GET_U_1(bp + 3), GET_BE_U_2(bp + 4));
126*ee67461eSJoseph Mingrone 		if (GET_U_1(bp + 6))
127*ee67461eSJoseph Mingrone 			ND_PRINT("CRC, ");
128*ee67461eSJoseph Mingrone 		else
129*ee67461eSJoseph Mingrone 			ND_PRINT("LQI %u, ", GET_U_1(bp + 7));
130*ee67461eSJoseph Mingrone 		inner_len = GET_U_1(bp + 15);
131*ee67461eSJoseph Mingrone 		ND_PRINT("inner len = %u", inner_len);
132*ee67461eSJoseph Mingrone 
133*ee67461eSJoseph Mingrone 		bp += 16;
134*ee67461eSJoseph Mingrone 		len -= 16;
135*ee67461eSJoseph Mingrone 	} else {
136*ee67461eSJoseph Mingrone 		/* ZEP v2 packet. */
137*ee67461eSJoseph Mingrone 		if (GET_U_1(bp + 3) == 2) {
138*ee67461eSJoseph Mingrone 			/* ZEP v2 ack. */
139*ee67461eSJoseph Mingrone 			ND_LCHECK_U(len, 8);
140*ee67461eSJoseph Mingrone 			seq_no = GET_BE_U_4(bp + 4);
141*ee67461eSJoseph Mingrone 			ND_PRINT("ACK, seq# = %u", seq_no);
142*ee67461eSJoseph Mingrone 			inner_len = 0;
143*ee67461eSJoseph Mingrone 			bp += 8;
144*ee67461eSJoseph Mingrone 			len -= 8;
145*ee67461eSJoseph Mingrone 		} else {
146*ee67461eSJoseph Mingrone 			/* ZEP v2 data, or some other. */
147*ee67461eSJoseph Mingrone 			ND_LCHECK_U(len, 32);
148*ee67461eSJoseph Mingrone 			ND_PRINT("Type %u, Channel ID %u, Device ID 0x%04x, ",
149*ee67461eSJoseph Mingrone 				 GET_U_1(bp + 3), GET_U_1(bp + 4),
150*ee67461eSJoseph Mingrone 				 GET_BE_U_2(bp + 5));
151*ee67461eSJoseph Mingrone 			if (GET_U_1(bp + 7))
152*ee67461eSJoseph Mingrone 				ND_PRINT("CRC, ");
153*ee67461eSJoseph Mingrone 			else
154*ee67461eSJoseph Mingrone 				ND_PRINT("LQI %u, ", GET_U_1(bp + 8));
155*ee67461eSJoseph Mingrone 
156*ee67461eSJoseph Mingrone 			zep_print_ts(ndo, bp + 9);
157*ee67461eSJoseph Mingrone 			seq_no = GET_BE_U_4(bp + 17);
158*ee67461eSJoseph Mingrone 			inner_len = GET_U_1(bp + 31);
159*ee67461eSJoseph Mingrone 			ND_PRINT(", seq# = %u, inner len = %u",
160*ee67461eSJoseph Mingrone 				 seq_no, inner_len);
161*ee67461eSJoseph Mingrone 			bp += 32;
162*ee67461eSJoseph Mingrone 			len -= 32;
163*ee67461eSJoseph Mingrone 		}
164*ee67461eSJoseph Mingrone 	}
165*ee67461eSJoseph Mingrone 
166*ee67461eSJoseph Mingrone 	if (inner_len != 0) {
167*ee67461eSJoseph Mingrone 		/* Call 802.15.4 dissector. */
168*ee67461eSJoseph Mingrone 		ND_PRINT("\n\t");
169*ee67461eSJoseph Mingrone 		if (ieee802_15_4_print(ndo, bp, inner_len)) {
170*ee67461eSJoseph Mingrone 			ND_TCHECK_LEN(bp, len);
171*ee67461eSJoseph Mingrone 			bp += len;
172*ee67461eSJoseph Mingrone 			len = 0;
173*ee67461eSJoseph Mingrone 		}
174*ee67461eSJoseph Mingrone 	}
175*ee67461eSJoseph Mingrone 
176*ee67461eSJoseph Mingrone 	if (!ndo->ndo_suppress_default_print)
177*ee67461eSJoseph Mingrone 		ND_DEFAULTPRINT(bp, len);
178*ee67461eSJoseph Mingrone 	return;
179*ee67461eSJoseph Mingrone invalid:
180*ee67461eSJoseph Mingrone 	nd_print_invalid(ndo);
181*ee67461eSJoseph Mingrone }
182