xref: /aosp_15_r20/external/tcpdump/print-dvmrp.c (revision 05b00f6010a2396e3db2409989fc67270046269f)
1*05b00f60SXin Li /*
2*05b00f60SXin Li  * Copyright (c) 1995, 1996
3*05b00f60SXin Li  *	The Regents of the University of California.  All rights reserved.
4*05b00f60SXin Li  *
5*05b00f60SXin Li  * Redistribution and use in source and binary forms, with or without
6*05b00f60SXin Li  * modification, are permitted provided that: (1) source code distributions
7*05b00f60SXin Li  * retain the above copyright notice and this paragraph in its entirety, (2)
8*05b00f60SXin Li  * distributions including binary code include the above copyright notice and
9*05b00f60SXin Li  * this paragraph in its entirety in the documentation or other materials
10*05b00f60SXin Li  * provided with the distribution, and (3) all advertising materials mentioning
11*05b00f60SXin Li  * features or use of this software display the following acknowledgement:
12*05b00f60SXin Li  * ``This product includes software developed by the University of California,
13*05b00f60SXin Li  * Lawrence Berkeley Laboratory and its contributors.'' Neither the name of
14*05b00f60SXin Li  * the University nor the names of its contributors may be used to endorse
15*05b00f60SXin Li  * or promote products derived from this software without specific prior
16*05b00f60SXin Li  * written permission.
17*05b00f60SXin Li  * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED
18*05b00f60SXin Li  * WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF
19*05b00f60SXin Li  * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
20*05b00f60SXin Li  */
21*05b00f60SXin Li 
22*05b00f60SXin Li /* \summary: Distance Vector Multicast Routing Protocol printer */
23*05b00f60SXin Li 
24*05b00f60SXin Li #ifdef HAVE_CONFIG_H
25*05b00f60SXin Li #include <config.h>
26*05b00f60SXin Li #endif
27*05b00f60SXin Li 
28*05b00f60SXin Li #include "netdissect-stdinc.h"
29*05b00f60SXin Li 
30*05b00f60SXin Li #include "netdissect.h"
31*05b00f60SXin Li #include "extract.h"
32*05b00f60SXin Li #include "addrtoname.h"
33*05b00f60SXin Li 
34*05b00f60SXin Li /*
35*05b00f60SXin Li  * See: RFC 1075 and draft-ietf-idmr-dvmrp-v3
36*05b00f60SXin Li  *
37*05b00f60SXin Li  * DVMRP message types and flag values shamelessly stolen from
38*05b00f60SXin Li  * mrouted/dvmrp.h.
39*05b00f60SXin Li  */
40*05b00f60SXin Li #define DVMRP_PROBE		1	/* for finding neighbors */
41*05b00f60SXin Li #define DVMRP_REPORT		2	/* for reporting some or all routes */
42*05b00f60SXin Li #define DVMRP_ASK_NEIGHBORS	3	/* sent by mapper, asking for a list */
43*05b00f60SXin Li 					/* of this router's neighbors */
44*05b00f60SXin Li #define DVMRP_NEIGHBORS		4	/* response to such a request */
45*05b00f60SXin Li #define DVMRP_ASK_NEIGHBORS2	5	/* as above, want new format reply */
46*05b00f60SXin Li #define DVMRP_NEIGHBORS2	6
47*05b00f60SXin Li #define DVMRP_PRUNE		7	/* prune message */
48*05b00f60SXin Li #define DVMRP_GRAFT		8	/* graft message */
49*05b00f60SXin Li #define DVMRP_GRAFT_ACK		9	/* graft acknowledgement */
50*05b00f60SXin Li static const struct tok dvmrp_msgtype_str[] = {
51*05b00f60SXin Li 	{ DVMRP_PROBE,          "Probe"              },
52*05b00f60SXin Li 	{ DVMRP_REPORT,         "Report"             },
53*05b00f60SXin Li 	{ DVMRP_ASK_NEIGHBORS,  "Ask-neighbors(old)" },
54*05b00f60SXin Li 	{ DVMRP_NEIGHBORS,      "Neighbors(old)"     },
55*05b00f60SXin Li 	{ DVMRP_ASK_NEIGHBORS2, "Ask-neighbors2"     },
56*05b00f60SXin Li 	{ DVMRP_NEIGHBORS2,     "Neighbors2"         },
57*05b00f60SXin Li 	{ DVMRP_PRUNE,          "Prune"              },
58*05b00f60SXin Li 	{ DVMRP_GRAFT,          "Graft"              },
59*05b00f60SXin Li 	{ DVMRP_GRAFT_ACK,      "Graft-ACK"          },
60*05b00f60SXin Li 	{ 0, NULL }
61*05b00f60SXin Li };
62*05b00f60SXin Li 
63*05b00f60SXin Li /*
64*05b00f60SXin Li  * 'flags' byte values in DVMRP_NEIGHBORS2 reply.
65*05b00f60SXin Li  */
66*05b00f60SXin Li #define DVMRP_NF_TUNNEL		0x01	/* neighbors reached via tunnel */
67*05b00f60SXin Li #define DVMRP_NF_SRCRT		0x02	/* tunnel uses IP source routing */
68*05b00f60SXin Li #define DVMRP_NF_DOWN		0x10	/* kernel state of interface */
69*05b00f60SXin Li #define DVMRP_NF_DISABLED	0x20	/* administratively disabled */
70*05b00f60SXin Li #define DVMRP_NF_QUERIER	0x40	/* I am the subnet's querier */
71*05b00f60SXin Li 
72*05b00f60SXin Li static void print_probe(netdissect_options *, const u_char *, u_int);
73*05b00f60SXin Li static void print_report(netdissect_options *, const u_char *, u_int);
74*05b00f60SXin Li static void print_neighbors(netdissect_options *, const u_char *, u_int);
75*05b00f60SXin Li static void print_neighbors2(netdissect_options *, const u_char *, u_int, uint8_t, uint8_t);
76*05b00f60SXin Li 
77*05b00f60SXin Li void
dvmrp_print(netdissect_options * ndo,const u_char * bp,u_int len)78*05b00f60SXin Li dvmrp_print(netdissect_options *ndo,
79*05b00f60SXin Li             const u_char *bp, u_int len)
80*05b00f60SXin Li {
81*05b00f60SXin Li 	u_char type;
82*05b00f60SXin Li 	uint8_t major_version, minor_version;
83*05b00f60SXin Li 
84*05b00f60SXin Li 	ndo->ndo_protocol = "dvmrp";
85*05b00f60SXin Li 	if (len < 8) {
86*05b00f60SXin Li 		ND_PRINT(" [length %u < 8]", len);
87*05b00f60SXin Li 		goto invalid;
88*05b00f60SXin Li 	}
89*05b00f60SXin Li 
90*05b00f60SXin Li 	type = GET_U_1(bp + 1);
91*05b00f60SXin Li 
92*05b00f60SXin Li 	/* Skip IGMP header */
93*05b00f60SXin Li 	bp += 8;
94*05b00f60SXin Li 	len -= 8;
95*05b00f60SXin Li 
96*05b00f60SXin Li 	ND_PRINT(" %s", tok2str(dvmrp_msgtype_str, "[type %u]", type));
97*05b00f60SXin Li 	switch (type) {
98*05b00f60SXin Li 
99*05b00f60SXin Li 	case DVMRP_PROBE:
100*05b00f60SXin Li 		if (ndo->ndo_vflag) {
101*05b00f60SXin Li 			print_probe(ndo, bp, len);
102*05b00f60SXin Li 		}
103*05b00f60SXin Li 		break;
104*05b00f60SXin Li 
105*05b00f60SXin Li 	case DVMRP_REPORT:
106*05b00f60SXin Li 		if (ndo->ndo_vflag > 1) {
107*05b00f60SXin Li 			print_report(ndo, bp, len);
108*05b00f60SXin Li 		}
109*05b00f60SXin Li 		break;
110*05b00f60SXin Li 
111*05b00f60SXin Li 	case DVMRP_NEIGHBORS:
112*05b00f60SXin Li 		print_neighbors(ndo, bp, len);
113*05b00f60SXin Li 		break;
114*05b00f60SXin Li 
115*05b00f60SXin Li 	case DVMRP_NEIGHBORS2:
116*05b00f60SXin Li 		/*
117*05b00f60SXin Li 		 * extract version from IGMP group address field
118*05b00f60SXin Li 		 */
119*05b00f60SXin Li 		bp -= 4;
120*05b00f60SXin Li 		major_version = GET_U_1(bp + 3);
121*05b00f60SXin Li 		minor_version = GET_U_1(bp + 2);
122*05b00f60SXin Li 		bp += 4;
123*05b00f60SXin Li 		print_neighbors2(ndo, bp, len, major_version, minor_version);
124*05b00f60SXin Li 		break;
125*05b00f60SXin Li 
126*05b00f60SXin Li 	case DVMRP_PRUNE:
127*05b00f60SXin Li 		ND_PRINT(" src %s grp %s", GET_IPADDR_STRING(bp), GET_IPADDR_STRING(bp + 4));
128*05b00f60SXin Li 		ND_PRINT(" timer ");
129*05b00f60SXin Li 		unsigned_relts_print(ndo, GET_BE_U_4(bp + 8));
130*05b00f60SXin Li 		break;
131*05b00f60SXin Li 
132*05b00f60SXin Li 	case DVMRP_GRAFT:
133*05b00f60SXin Li 		ND_PRINT(" src %s grp %s", GET_IPADDR_STRING(bp), GET_IPADDR_STRING(bp + 4));
134*05b00f60SXin Li 		break;
135*05b00f60SXin Li 
136*05b00f60SXin Li 	case DVMRP_GRAFT_ACK:
137*05b00f60SXin Li 		ND_PRINT(" src %s grp %s", GET_IPADDR_STRING(bp), GET_IPADDR_STRING(bp + 4));
138*05b00f60SXin Li 		break;
139*05b00f60SXin Li 	}
140*05b00f60SXin Li 	return;
141*05b00f60SXin Li 
142*05b00f60SXin Li invalid:
143*05b00f60SXin Li 	nd_print_invalid(ndo);
144*05b00f60SXin Li }
145*05b00f60SXin Li 
146*05b00f60SXin Li static void
print_report(netdissect_options * ndo,const u_char * bp,u_int len)147*05b00f60SXin Li print_report(netdissect_options *ndo,
148*05b00f60SXin Li              const u_char *bp,
149*05b00f60SXin Li              u_int len)
150*05b00f60SXin Li {
151*05b00f60SXin Li 	uint32_t mask, origin;
152*05b00f60SXin Li 	u_int metric, done;
153*05b00f60SXin Li 	u_int i, width;
154*05b00f60SXin Li 
155*05b00f60SXin Li 	while (len > 0) {
156*05b00f60SXin Li 		if (len < 3) {
157*05b00f60SXin Li 			ND_PRINT(" [length %u < 3]", len);
158*05b00f60SXin Li 			goto invalid;
159*05b00f60SXin Li 		}
160*05b00f60SXin Li 		mask = (uint32_t)0xff << 24 | GET_U_1(bp) << 16 |
161*05b00f60SXin Li 			GET_U_1(bp + 1) << 8 | GET_U_1(bp + 2);
162*05b00f60SXin Li 		width = 1;
163*05b00f60SXin Li 		if (GET_U_1(bp))
164*05b00f60SXin Li 			width = 2;
165*05b00f60SXin Li 		if (GET_U_1(bp + 1))
166*05b00f60SXin Li 			width = 3;
167*05b00f60SXin Li 		if (GET_U_1(bp + 2))
168*05b00f60SXin Li 			width = 4;
169*05b00f60SXin Li 
170*05b00f60SXin Li 		ND_PRINT("\n\tMask %s", intoa(htonl(mask)));
171*05b00f60SXin Li 		bp += 3;
172*05b00f60SXin Li 		len -= 3;
173*05b00f60SXin Li 		do {
174*05b00f60SXin Li 			if (len < width + 1) {
175*05b00f60SXin Li 				ND_PRINT("\n\t  [Truncated Report]");
176*05b00f60SXin Li 				goto invalid;
177*05b00f60SXin Li 			}
178*05b00f60SXin Li 			origin = 0;
179*05b00f60SXin Li 			for (i = 0; i < width; ++i) {
180*05b00f60SXin Li 				origin = origin << 8 | GET_U_1(bp);
181*05b00f60SXin Li 				bp++;
182*05b00f60SXin Li 			}
183*05b00f60SXin Li 			for ( ; i < 4; ++i)
184*05b00f60SXin Li 				origin <<= 8;
185*05b00f60SXin Li 
186*05b00f60SXin Li 			metric = GET_U_1(bp);
187*05b00f60SXin Li 			bp++;
188*05b00f60SXin Li 			done = metric & 0x80;
189*05b00f60SXin Li 			metric &= 0x7f;
190*05b00f60SXin Li 			ND_PRINT("\n\t  %s metric %u", intoa(htonl(origin)),
191*05b00f60SXin Li 				metric);
192*05b00f60SXin Li 			len -= width + 1;
193*05b00f60SXin Li 		} while (!done);
194*05b00f60SXin Li 	}
195*05b00f60SXin Li 	return;
196*05b00f60SXin Li 
197*05b00f60SXin Li invalid:
198*05b00f60SXin Li 	nd_print_invalid(ndo);
199*05b00f60SXin Li }
200*05b00f60SXin Li 
201*05b00f60SXin Li static void
print_probe(netdissect_options * ndo,const u_char * bp,u_int len)202*05b00f60SXin Li print_probe(netdissect_options *ndo,
203*05b00f60SXin Li             const u_char *bp,
204*05b00f60SXin Li             u_int len)
205*05b00f60SXin Li {
206*05b00f60SXin Li 	if (len < 4) {
207*05b00f60SXin Li 		ND_PRINT(" [full length %u < 4]", len);
208*05b00f60SXin Li 		goto invalid;
209*05b00f60SXin Li 	}
210*05b00f60SXin Li 	ND_PRINT(ndo->ndo_vflag > 1 ? "\n\t" : " ");
211*05b00f60SXin Li 	ND_PRINT("genid %u", GET_BE_U_4(bp));
212*05b00f60SXin Li 	if (ndo->ndo_vflag < 2)
213*05b00f60SXin Li 		return;
214*05b00f60SXin Li 
215*05b00f60SXin Li 	bp += 4;
216*05b00f60SXin Li 	len -= 4;
217*05b00f60SXin Li 	while (len > 0) {
218*05b00f60SXin Li 		if (len < 4) {
219*05b00f60SXin Li 			ND_PRINT("[remaining length %u < 4]", len);
220*05b00f60SXin Li 			goto invalid;
221*05b00f60SXin Li 		}
222*05b00f60SXin Li 		ND_PRINT("\n\tneighbor %s", GET_IPADDR_STRING(bp));
223*05b00f60SXin Li 		bp += 4; len -= 4;
224*05b00f60SXin Li 	}
225*05b00f60SXin Li 	return;
226*05b00f60SXin Li 
227*05b00f60SXin Li invalid:
228*05b00f60SXin Li 	nd_print_invalid(ndo);
229*05b00f60SXin Li }
230*05b00f60SXin Li 
231*05b00f60SXin Li static void
print_neighbors(netdissect_options * ndo,const u_char * bp,u_int len)232*05b00f60SXin Li print_neighbors(netdissect_options *ndo,
233*05b00f60SXin Li                 const u_char *bp,
234*05b00f60SXin Li                 u_int len)
235*05b00f60SXin Li {
236*05b00f60SXin Li 	const u_char *laddr;
237*05b00f60SXin Li 	u_char metric;
238*05b00f60SXin Li 	u_char thresh;
239*05b00f60SXin Li 	int ncount;
240*05b00f60SXin Li 
241*05b00f60SXin Li 	while (len > 0) {
242*05b00f60SXin Li 		if (len < 7) {
243*05b00f60SXin Li 			ND_PRINT(" [length %u < 7]", len);
244*05b00f60SXin Li 			goto invalid;
245*05b00f60SXin Li 		}
246*05b00f60SXin Li 		laddr = bp;
247*05b00f60SXin Li 		bp += 4;
248*05b00f60SXin Li 		metric = GET_U_1(bp);
249*05b00f60SXin Li 		bp++;
250*05b00f60SXin Li 		thresh = GET_U_1(bp);
251*05b00f60SXin Li 		bp++;
252*05b00f60SXin Li 		ncount = GET_U_1(bp);
253*05b00f60SXin Li 		bp++;
254*05b00f60SXin Li 		len -= 7;
255*05b00f60SXin Li 		while (--ncount >= 0) {
256*05b00f60SXin Li 			if (len < 4) {
257*05b00f60SXin Li 				ND_PRINT(" [length %u < 4]", len);
258*05b00f60SXin Li 				goto invalid;
259*05b00f60SXin Li 			}
260*05b00f60SXin Li 			ND_PRINT(" [%s ->", GET_IPADDR_STRING(laddr));
261*05b00f60SXin Li 			ND_PRINT(" %s, (%u/%u)]",
262*05b00f60SXin Li 				   GET_IPADDR_STRING(bp), metric, thresh);
263*05b00f60SXin Li 			bp += 4;
264*05b00f60SXin Li 			len -= 4;
265*05b00f60SXin Li 		}
266*05b00f60SXin Li 	}
267*05b00f60SXin Li 	return;
268*05b00f60SXin Li 
269*05b00f60SXin Li invalid:
270*05b00f60SXin Li 	nd_print_invalid(ndo);
271*05b00f60SXin Li }
272*05b00f60SXin Li 
273*05b00f60SXin Li static void
print_neighbors2(netdissect_options * ndo,const u_char * bp,u_int len,uint8_t major_version,uint8_t minor_version)274*05b00f60SXin Li print_neighbors2(netdissect_options *ndo,
275*05b00f60SXin Li                  const u_char *bp,
276*05b00f60SXin Li                  u_int len, uint8_t major_version,
277*05b00f60SXin Li                  uint8_t minor_version)
278*05b00f60SXin Li {
279*05b00f60SXin Li 	const u_char *laddr;
280*05b00f60SXin Li 	u_char metric, thresh, flags;
281*05b00f60SXin Li 	int ncount;
282*05b00f60SXin Li 
283*05b00f60SXin Li 	ND_PRINT(" (v %u.%u):", major_version, minor_version);
284*05b00f60SXin Li 
285*05b00f60SXin Li 	while (len > 0) {
286*05b00f60SXin Li 		if (len < 8) {
287*05b00f60SXin Li 			ND_PRINT(" [length %u < 8]", len);
288*05b00f60SXin Li 			goto invalid;
289*05b00f60SXin Li 		}
290*05b00f60SXin Li 		laddr = bp;
291*05b00f60SXin Li 		bp += 4;
292*05b00f60SXin Li 		metric = GET_U_1(bp);
293*05b00f60SXin Li 		bp++;
294*05b00f60SXin Li 		thresh = GET_U_1(bp);
295*05b00f60SXin Li 		bp++;
296*05b00f60SXin Li 		flags = GET_U_1(bp);
297*05b00f60SXin Li 		bp++;
298*05b00f60SXin Li 		ncount = GET_U_1(bp);
299*05b00f60SXin Li 		bp++;
300*05b00f60SXin Li 		len -= 8;
301*05b00f60SXin Li 		while (--ncount >= 0 && len > 0) {
302*05b00f60SXin Li 			if (len < 4) {
303*05b00f60SXin Li 				ND_PRINT(" [length %u < 4]", len);
304*05b00f60SXin Li 				goto invalid;
305*05b00f60SXin Li 			}
306*05b00f60SXin Li 			ND_PRINT(" [%s -> ", GET_IPADDR_STRING(laddr));
307*05b00f60SXin Li 			ND_PRINT("%s (%u/%u", GET_IPADDR_STRING(bp),
308*05b00f60SXin Li 				     metric, thresh);
309*05b00f60SXin Li 			if (flags & DVMRP_NF_TUNNEL)
310*05b00f60SXin Li 				ND_PRINT("/tunnel");
311*05b00f60SXin Li 			if (flags & DVMRP_NF_SRCRT)
312*05b00f60SXin Li 				ND_PRINT("/srcrt");
313*05b00f60SXin Li 			if (flags & DVMRP_NF_QUERIER)
314*05b00f60SXin Li 				ND_PRINT("/querier");
315*05b00f60SXin Li 			if (flags & DVMRP_NF_DISABLED)
316*05b00f60SXin Li 				ND_PRINT("/disabled");
317*05b00f60SXin Li 			if (flags & DVMRP_NF_DOWN)
318*05b00f60SXin Li 				ND_PRINT("/down");
319*05b00f60SXin Li 			ND_PRINT(")]");
320*05b00f60SXin Li 			bp += 4;
321*05b00f60SXin Li 			len -= 4;
322*05b00f60SXin Li 		}
323*05b00f60SXin Li 		if (ncount != -1) {
324*05b00f60SXin Li 			ND_PRINT(" [invalid ncount]");
325*05b00f60SXin Li 			goto invalid;
326*05b00f60SXin Li 		}
327*05b00f60SXin Li 	}
328*05b00f60SXin Li 	return;
329*05b00f60SXin Li 
330*05b00f60SXin Li invalid:
331*05b00f60SXin Li 	nd_print_invalid(ndo);
332*05b00f60SXin Li }
333