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