1*05b00f60SXin Li /*
2*05b00f60SXin Li * Copyright (c) 2003 Bruce M. Simpson <[email protected]>
3*05b00f60SXin Li * 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 the following conditions
7*05b00f60SXin Li * are met:
8*05b00f60SXin Li * 1. Redistributions of source code must retain the above copyright
9*05b00f60SXin Li * notice, this list of conditions and the following disclaimer.
10*05b00f60SXin Li * 2. Redistributions in binary form must reproduce the above copyright
11*05b00f60SXin Li * notice, this list of conditions and the following disclaimer in the
12*05b00f60SXin Li * documentation and/or other materials provided with the distribution.
13*05b00f60SXin Li * 3. All advertising materials mentioning features or use of this software
14*05b00f60SXin Li * must display the following acknowledgement:
15*05b00f60SXin Li * This product includes software developed by Bruce M. Simpson.
16*05b00f60SXin Li * 4. Neither the name of Bruce M. Simpson nor the names of co-
17*05b00f60SXin Li * contributors may be used to endorse or promote products derived
18*05b00f60SXin Li * from this software without specific prior written permission.
19*05b00f60SXin Li *
20*05b00f60SXin Li * THIS SOFTWARE IS PROVIDED BY Bruce M. Simpson AND CONTRIBUTORS
21*05b00f60SXin Li * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
22*05b00f60SXin Li * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
23*05b00f60SXin Li * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL Bruce M. Simpson OR CONTRIBUTORS
24*05b00f60SXin Li * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
25*05b00f60SXin Li * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
26*05b00f60SXin Li * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
27*05b00f60SXin Li * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
28*05b00f60SXin Li * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
29*05b00f60SXin Li * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
30*05b00f60SXin Li * POSSIBILITY OF SUCH DAMAGE.
31*05b00f60SXin Li */
32*05b00f60SXin Li
33*05b00f60SXin Li /* \summary: Ad hoc On-Demand Distance Vector (AODV) Routing printer */
34*05b00f60SXin Li
35*05b00f60SXin Li #ifdef HAVE_CONFIG_H
36*05b00f60SXin Li #include <config.h>
37*05b00f60SXin Li #endif
38*05b00f60SXin Li
39*05b00f60SXin Li #include "netdissect-stdinc.h"
40*05b00f60SXin Li
41*05b00f60SXin Li #include "netdissect.h"
42*05b00f60SXin Li #include "addrtoname.h"
43*05b00f60SXin Li #include "extract.h"
44*05b00f60SXin Li
45*05b00f60SXin Li /*
46*05b00f60SXin Li * RFC 3561
47*05b00f60SXin Li */
48*05b00f60SXin Li struct aodv_rreq {
49*05b00f60SXin Li nd_uint8_t rreq_type; /* AODV message type (1) */
50*05b00f60SXin Li nd_uint8_t rreq_flags; /* various flags */
51*05b00f60SXin Li nd_uint8_t rreq_zero0; /* reserved, set to zero */
52*05b00f60SXin Li nd_uint8_t rreq_hops; /* number of hops from originator */
53*05b00f60SXin Li nd_uint32_t rreq_id; /* request ID */
54*05b00f60SXin Li nd_ipv4 rreq_da; /* destination IPv4 address */
55*05b00f60SXin Li nd_uint32_t rreq_ds; /* destination sequence number */
56*05b00f60SXin Li nd_ipv4 rreq_oa; /* originator IPv4 address */
57*05b00f60SXin Li nd_uint32_t rreq_os; /* originator sequence number */
58*05b00f60SXin Li };
59*05b00f60SXin Li struct aodv_rreq6 {
60*05b00f60SXin Li nd_uint8_t rreq_type; /* AODV message type (1) */
61*05b00f60SXin Li nd_uint8_t rreq_flags; /* various flags */
62*05b00f60SXin Li nd_uint8_t rreq_zero0; /* reserved, set to zero */
63*05b00f60SXin Li nd_uint8_t rreq_hops; /* number of hops from originator */
64*05b00f60SXin Li nd_uint32_t rreq_id; /* request ID */
65*05b00f60SXin Li nd_ipv6 rreq_da; /* destination IPv6 address */
66*05b00f60SXin Li nd_uint32_t rreq_ds; /* destination sequence number */
67*05b00f60SXin Li nd_ipv6 rreq_oa; /* originator IPv6 address */
68*05b00f60SXin Li nd_uint32_t rreq_os; /* originator sequence number */
69*05b00f60SXin Li };
70*05b00f60SXin Li struct aodv_rreq6_draft_01 {
71*05b00f60SXin Li nd_uint8_t rreq_type; /* AODV message type (16) */
72*05b00f60SXin Li nd_uint8_t rreq_flags; /* various flags */
73*05b00f60SXin Li nd_uint8_t rreq_zero0; /* reserved, set to zero */
74*05b00f60SXin Li nd_uint8_t rreq_hops; /* number of hops from originator */
75*05b00f60SXin Li nd_uint32_t rreq_id; /* request ID */
76*05b00f60SXin Li nd_uint32_t rreq_ds; /* destination sequence number */
77*05b00f60SXin Li nd_uint32_t rreq_os; /* originator sequence number */
78*05b00f60SXin Li nd_ipv6 rreq_da; /* destination IPv6 address */
79*05b00f60SXin Li nd_ipv6 rreq_oa; /* originator IPv6 address */
80*05b00f60SXin Li };
81*05b00f60SXin Li
82*05b00f60SXin Li #define RREQ_JOIN 0x80 /* join (reserved for multicast */
83*05b00f60SXin Li #define RREQ_REPAIR 0x40 /* repair (reserved for multicast */
84*05b00f60SXin Li #define RREQ_GRAT 0x20 /* gratuitous RREP */
85*05b00f60SXin Li #define RREQ_DEST 0x10 /* destination only */
86*05b00f60SXin Li #define RREQ_UNKNOWN 0x08 /* unknown destination sequence num */
87*05b00f60SXin Li #define RREQ_FLAGS_MASK 0xF8 /* mask for rreq_flags */
88*05b00f60SXin Li
89*05b00f60SXin Li struct aodv_rrep {
90*05b00f60SXin Li nd_uint8_t rrep_type; /* AODV message type (2) */
91*05b00f60SXin Li nd_uint8_t rrep_flags; /* various flags */
92*05b00f60SXin Li nd_uint8_t rrep_ps; /* prefix size */
93*05b00f60SXin Li nd_uint8_t rrep_hops; /* number of hops from o to d */
94*05b00f60SXin Li nd_ipv4 rrep_da; /* destination IPv4 address */
95*05b00f60SXin Li nd_uint32_t rrep_ds; /* destination sequence number */
96*05b00f60SXin Li nd_ipv4 rrep_oa; /* originator IPv4 address */
97*05b00f60SXin Li nd_uint32_t rrep_life; /* lifetime of this route */
98*05b00f60SXin Li };
99*05b00f60SXin Li struct aodv_rrep6 {
100*05b00f60SXin Li nd_uint8_t rrep_type; /* AODV message type (2) */
101*05b00f60SXin Li nd_uint8_t rrep_flags; /* various flags */
102*05b00f60SXin Li nd_uint8_t rrep_ps; /* prefix size */
103*05b00f60SXin Li nd_uint8_t rrep_hops; /* number of hops from o to d */
104*05b00f60SXin Li nd_ipv6 rrep_da; /* destination IPv6 address */
105*05b00f60SXin Li nd_uint32_t rrep_ds; /* destination sequence number */
106*05b00f60SXin Li nd_ipv6 rrep_oa; /* originator IPv6 address */
107*05b00f60SXin Li nd_uint32_t rrep_life; /* lifetime of this route */
108*05b00f60SXin Li };
109*05b00f60SXin Li struct aodv_rrep6_draft_01 {
110*05b00f60SXin Li nd_uint8_t rrep_type; /* AODV message type (17) */
111*05b00f60SXin Li nd_uint8_t rrep_flags; /* various flags */
112*05b00f60SXin Li nd_uint8_t rrep_ps; /* prefix size */
113*05b00f60SXin Li nd_uint8_t rrep_hops; /* number of hops from o to d */
114*05b00f60SXin Li nd_uint32_t rrep_ds; /* destination sequence number */
115*05b00f60SXin Li nd_ipv6 rrep_da; /* destination IPv6 address */
116*05b00f60SXin Li nd_ipv6 rrep_oa; /* originator IPv6 address */
117*05b00f60SXin Li nd_uint32_t rrep_life; /* lifetime of this route */
118*05b00f60SXin Li };
119*05b00f60SXin Li
120*05b00f60SXin Li #define RREP_REPAIR 0x80 /* repair (reserved for multicast */
121*05b00f60SXin Li #define RREP_ACK 0x40 /* acknowledgement required */
122*05b00f60SXin Li #define RREP_FLAGS_MASK 0xC0 /* mask for rrep_flags */
123*05b00f60SXin Li #define RREP_PREFIX_MASK 0x1F /* mask for prefix size */
124*05b00f60SXin Li
125*05b00f60SXin Li struct rerr_unreach {
126*05b00f60SXin Li nd_ipv4 u_da; /* IPv4 address */
127*05b00f60SXin Li nd_uint32_t u_ds; /* sequence number */
128*05b00f60SXin Li };
129*05b00f60SXin Li struct rerr_unreach6 {
130*05b00f60SXin Li nd_ipv6 u_da; /* IPv6 address */
131*05b00f60SXin Li nd_uint32_t u_ds; /* sequence number */
132*05b00f60SXin Li };
133*05b00f60SXin Li struct rerr_unreach6_draft_01 {
134*05b00f60SXin Li nd_ipv6 u_da; /* IPv6 address */
135*05b00f60SXin Li nd_uint32_t u_ds; /* sequence number */
136*05b00f60SXin Li };
137*05b00f60SXin Li
138*05b00f60SXin Li struct aodv_rerr {
139*05b00f60SXin Li nd_uint8_t rerr_type; /* AODV message type (3 or 18) */
140*05b00f60SXin Li nd_uint8_t rerr_flags; /* various flags */
141*05b00f60SXin Li nd_uint8_t rerr_zero0; /* reserved, set to zero */
142*05b00f60SXin Li nd_uint8_t rerr_dc; /* destination count */
143*05b00f60SXin Li };
144*05b00f60SXin Li
145*05b00f60SXin Li #define RERR_NODELETE 0x80 /* don't delete the link */
146*05b00f60SXin Li #define RERR_FLAGS_MASK 0x80 /* mask for rerr_flags */
147*05b00f60SXin Li
148*05b00f60SXin Li struct aodv_rrep_ack {
149*05b00f60SXin Li nd_uint8_t ra_type;
150*05b00f60SXin Li nd_uint8_t ra_zero0;
151*05b00f60SXin Li };
152*05b00f60SXin Li
153*05b00f60SXin Li #define AODV_RREQ 1 /* route request */
154*05b00f60SXin Li #define AODV_RREP 2 /* route response */
155*05b00f60SXin Li #define AODV_RERR 3 /* error report */
156*05b00f60SXin Li #define AODV_RREP_ACK 4 /* route response acknowledgement */
157*05b00f60SXin Li
158*05b00f60SXin Li #define AODV_V6_DRAFT_01_RREQ 16 /* IPv6 route request */
159*05b00f60SXin Li #define AODV_V6_DRAFT_01_RREP 17 /* IPv6 route response */
160*05b00f60SXin Li #define AODV_V6_DRAFT_01_RERR 18 /* IPv6 error report */
161*05b00f60SXin Li #define AODV_V6_DRAFT_01_RREP_ACK 19 /* IPV6 route response acknowledgment */
162*05b00f60SXin Li
163*05b00f60SXin Li struct aodv_ext {
164*05b00f60SXin Li nd_uint8_t type; /* extension type */
165*05b00f60SXin Li nd_uint8_t length; /* extension length */
166*05b00f60SXin Li };
167*05b00f60SXin Li
168*05b00f60SXin Li struct aodv_hello {
169*05b00f60SXin Li struct aodv_ext eh; /* extension header */
170*05b00f60SXin Li nd_uint32_t interval; /* expect my next hello in
171*05b00f60SXin Li * (n) ms
172*05b00f60SXin Li * NOTE: this is not aligned */
173*05b00f60SXin Li };
174*05b00f60SXin Li
175*05b00f60SXin Li #define AODV_EXT_HELLO 1
176*05b00f60SXin Li
177*05b00f60SXin Li static void
aodv_extension(netdissect_options * ndo,const struct aodv_ext * ep,u_int length)178*05b00f60SXin Li aodv_extension(netdissect_options *ndo,
179*05b00f60SXin Li const struct aodv_ext *ep, u_int length)
180*05b00f60SXin Li {
181*05b00f60SXin Li const struct aodv_hello *ah;
182*05b00f60SXin Li
183*05b00f60SXin Li ND_TCHECK_SIZE(ep);
184*05b00f60SXin Li switch (GET_U_1(ep->type)) {
185*05b00f60SXin Li case AODV_EXT_HELLO:
186*05b00f60SXin Li ah = (const struct aodv_hello *)(const void *)ep;
187*05b00f60SXin Li ND_TCHECK_SIZE(ah);
188*05b00f60SXin Li if (length < sizeof(struct aodv_hello))
189*05b00f60SXin Li goto trunc;
190*05b00f60SXin Li if (GET_U_1(ep->length) < 4) {
191*05b00f60SXin Li ND_PRINT("\n\text HELLO - bad length %u",
192*05b00f60SXin Li GET_U_1(ep->length));
193*05b00f60SXin Li break;
194*05b00f60SXin Li }
195*05b00f60SXin Li ND_PRINT("\n\text HELLO %u ms",
196*05b00f60SXin Li GET_BE_U_4(ah->interval));
197*05b00f60SXin Li break;
198*05b00f60SXin Li
199*05b00f60SXin Li default:
200*05b00f60SXin Li ND_PRINT("\n\text %u %u", GET_U_1(ep->type),
201*05b00f60SXin Li GET_U_1(ep->length));
202*05b00f60SXin Li break;
203*05b00f60SXin Li }
204*05b00f60SXin Li return;
205*05b00f60SXin Li
206*05b00f60SXin Li trunc:
207*05b00f60SXin Li nd_print_trunc(ndo);
208*05b00f60SXin Li }
209*05b00f60SXin Li
210*05b00f60SXin Li static void
aodv_rreq(netdissect_options * ndo,const u_char * dat,u_int length)211*05b00f60SXin Li aodv_rreq(netdissect_options *ndo, const u_char *dat, u_int length)
212*05b00f60SXin Li {
213*05b00f60SXin Li u_int i;
214*05b00f60SXin Li const struct aodv_rreq *ap = (const struct aodv_rreq *)dat;
215*05b00f60SXin Li
216*05b00f60SXin Li ND_TCHECK_SIZE(ap);
217*05b00f60SXin Li if (length < sizeof(*ap))
218*05b00f60SXin Li goto trunc;
219*05b00f60SXin Li ND_PRINT(" rreq %u %s%s%s%s%shops %u id 0x%08x\n"
220*05b00f60SXin Li "\tdst %s seq %u src %s seq %u", length,
221*05b00f60SXin Li GET_U_1(ap->rreq_type) & RREQ_JOIN ? "[J]" : "",
222*05b00f60SXin Li GET_U_1(ap->rreq_type) & RREQ_REPAIR ? "[R]" : "",
223*05b00f60SXin Li GET_U_1(ap->rreq_type) & RREQ_GRAT ? "[G]" : "",
224*05b00f60SXin Li GET_U_1(ap->rreq_type) & RREQ_DEST ? "[D]" : "",
225*05b00f60SXin Li GET_U_1(ap->rreq_type) & RREQ_UNKNOWN ? "[U] " : " ",
226*05b00f60SXin Li GET_U_1(ap->rreq_hops),
227*05b00f60SXin Li GET_BE_U_4(ap->rreq_id),
228*05b00f60SXin Li GET_IPADDR_STRING(ap->rreq_da),
229*05b00f60SXin Li GET_BE_U_4(ap->rreq_ds),
230*05b00f60SXin Li GET_IPADDR_STRING(ap->rreq_oa),
231*05b00f60SXin Li GET_BE_U_4(ap->rreq_os));
232*05b00f60SXin Li i = length - sizeof(*ap);
233*05b00f60SXin Li if (i >= sizeof(struct aodv_ext))
234*05b00f60SXin Li aodv_extension(ndo, (const struct aodv_ext *)(dat + sizeof(*ap)), i);
235*05b00f60SXin Li return;
236*05b00f60SXin Li
237*05b00f60SXin Li trunc:
238*05b00f60SXin Li nd_print_trunc(ndo);
239*05b00f60SXin Li }
240*05b00f60SXin Li
241*05b00f60SXin Li static void
aodv_rrep(netdissect_options * ndo,const u_char * dat,u_int length)242*05b00f60SXin Li aodv_rrep(netdissect_options *ndo, const u_char *dat, u_int length)
243*05b00f60SXin Li {
244*05b00f60SXin Li u_int i;
245*05b00f60SXin Li const struct aodv_rrep *ap = (const struct aodv_rrep *)dat;
246*05b00f60SXin Li
247*05b00f60SXin Li ND_TCHECK_SIZE(ap);
248*05b00f60SXin Li if (length < sizeof(*ap))
249*05b00f60SXin Li goto trunc;
250*05b00f60SXin Li ND_PRINT(" rrep %u %s%sprefix %u hops %u\n"
251*05b00f60SXin Li "\tdst %s dseq %u src %s %u ms", length,
252*05b00f60SXin Li GET_U_1(ap->rrep_type) & RREP_REPAIR ? "[R]" : "",
253*05b00f60SXin Li GET_U_1(ap->rrep_type) & RREP_ACK ? "[A] " : " ",
254*05b00f60SXin Li GET_U_1(ap->rrep_ps) & RREP_PREFIX_MASK,
255*05b00f60SXin Li GET_U_1(ap->rrep_hops),
256*05b00f60SXin Li GET_IPADDR_STRING(ap->rrep_da),
257*05b00f60SXin Li GET_BE_U_4(ap->rrep_ds),
258*05b00f60SXin Li GET_IPADDR_STRING(ap->rrep_oa),
259*05b00f60SXin Li GET_BE_U_4(ap->rrep_life));
260*05b00f60SXin Li i = length - sizeof(*ap);
261*05b00f60SXin Li if (i >= sizeof(struct aodv_ext))
262*05b00f60SXin Li aodv_extension(ndo, (const struct aodv_ext *)(dat + sizeof(*ap)), i);
263*05b00f60SXin Li return;
264*05b00f60SXin Li
265*05b00f60SXin Li trunc:
266*05b00f60SXin Li nd_print_trunc(ndo);
267*05b00f60SXin Li }
268*05b00f60SXin Li
269*05b00f60SXin Li static void
aodv_rerr(netdissect_options * ndo,const u_char * dat,u_int length)270*05b00f60SXin Li aodv_rerr(netdissect_options *ndo, const u_char *dat, u_int length)
271*05b00f60SXin Li {
272*05b00f60SXin Li u_int i, dc;
273*05b00f60SXin Li const struct aodv_rerr *ap = (const struct aodv_rerr *)dat;
274*05b00f60SXin Li const struct rerr_unreach *dp;
275*05b00f60SXin Li
276*05b00f60SXin Li ND_TCHECK_SIZE(ap);
277*05b00f60SXin Li if (length < sizeof(*ap))
278*05b00f60SXin Li goto trunc;
279*05b00f60SXin Li ND_PRINT(" rerr %s [items %u] [%u]:",
280*05b00f60SXin Li GET_U_1(ap->rerr_flags) & RERR_NODELETE ? "[D]" : "",
281*05b00f60SXin Li GET_U_1(ap->rerr_dc), length);
282*05b00f60SXin Li dp = (const struct rerr_unreach *)(dat + sizeof(*ap));
283*05b00f60SXin Li i = length - sizeof(*ap);
284*05b00f60SXin Li for (dc = GET_U_1(ap->rerr_dc); dc != 0; dc--) {
285*05b00f60SXin Li ND_TCHECK_SIZE(dp);
286*05b00f60SXin Li if (i < sizeof(*dp))
287*05b00f60SXin Li goto trunc;
288*05b00f60SXin Li ND_PRINT(" {%s}(%u)", GET_IPADDR_STRING(dp->u_da),
289*05b00f60SXin Li GET_BE_U_4(dp->u_ds));
290*05b00f60SXin Li dp++;
291*05b00f60SXin Li i -= sizeof(*dp);
292*05b00f60SXin Li }
293*05b00f60SXin Li return;
294*05b00f60SXin Li
295*05b00f60SXin Li trunc:
296*05b00f60SXin Li nd_print_trunc(ndo);
297*05b00f60SXin Li }
298*05b00f60SXin Li
299*05b00f60SXin Li static void
aodv_v6_rreq(netdissect_options * ndo,const u_char * dat,u_int length)300*05b00f60SXin Li aodv_v6_rreq(netdissect_options *ndo, const u_char *dat, u_int length)
301*05b00f60SXin Li {
302*05b00f60SXin Li u_int i;
303*05b00f60SXin Li const struct aodv_rreq6 *ap = (const struct aodv_rreq6 *)dat;
304*05b00f60SXin Li
305*05b00f60SXin Li ND_TCHECK_SIZE(ap);
306*05b00f60SXin Li if (length < sizeof(*ap))
307*05b00f60SXin Li goto trunc;
308*05b00f60SXin Li ND_PRINT(" v6 rreq %u %s%s%s%s%shops %u id 0x%08x\n"
309*05b00f60SXin Li "\tdst %s seq %u src %s seq %u", length,
310*05b00f60SXin Li GET_U_1(ap->rreq_type) & RREQ_JOIN ? "[J]" : "",
311*05b00f60SXin Li GET_U_1(ap->rreq_type) & RREQ_REPAIR ? "[R]" : "",
312*05b00f60SXin Li GET_U_1(ap->rreq_type) & RREQ_GRAT ? "[G]" : "",
313*05b00f60SXin Li GET_U_1(ap->rreq_type) & RREQ_DEST ? "[D]" : "",
314*05b00f60SXin Li GET_U_1(ap->rreq_type) & RREQ_UNKNOWN ? "[U] " : " ",
315*05b00f60SXin Li GET_U_1(ap->rreq_hops),
316*05b00f60SXin Li GET_BE_U_4(ap->rreq_id),
317*05b00f60SXin Li GET_IP6ADDR_STRING(ap->rreq_da),
318*05b00f60SXin Li GET_BE_U_4(ap->rreq_ds),
319*05b00f60SXin Li GET_IP6ADDR_STRING(ap->rreq_oa),
320*05b00f60SXin Li GET_BE_U_4(ap->rreq_os));
321*05b00f60SXin Li i = length - sizeof(*ap);
322*05b00f60SXin Li if (i >= sizeof(struct aodv_ext))
323*05b00f60SXin Li aodv_extension(ndo, (const struct aodv_ext *)(dat + sizeof(*ap)), i);
324*05b00f60SXin Li return;
325*05b00f60SXin Li
326*05b00f60SXin Li trunc:
327*05b00f60SXin Li nd_print_trunc(ndo);
328*05b00f60SXin Li }
329*05b00f60SXin Li
330*05b00f60SXin Li static void
aodv_v6_rrep(netdissect_options * ndo,const u_char * dat,u_int length)331*05b00f60SXin Li aodv_v6_rrep(netdissect_options *ndo, const u_char *dat, u_int length)
332*05b00f60SXin Li {
333*05b00f60SXin Li u_int i;
334*05b00f60SXin Li const struct aodv_rrep6 *ap = (const struct aodv_rrep6 *)dat;
335*05b00f60SXin Li
336*05b00f60SXin Li ND_TCHECK_SIZE(ap);
337*05b00f60SXin Li if (length < sizeof(*ap))
338*05b00f60SXin Li goto trunc;
339*05b00f60SXin Li ND_PRINT(" rrep %u %s%sprefix %u hops %u\n"
340*05b00f60SXin Li "\tdst %s dseq %u src %s %u ms", length,
341*05b00f60SXin Li GET_U_1(ap->rrep_type) & RREP_REPAIR ? "[R]" : "",
342*05b00f60SXin Li GET_U_1(ap->rrep_type) & RREP_ACK ? "[A] " : " ",
343*05b00f60SXin Li GET_U_1(ap->rrep_ps) & RREP_PREFIX_MASK,
344*05b00f60SXin Li GET_U_1(ap->rrep_hops),
345*05b00f60SXin Li GET_IP6ADDR_STRING(ap->rrep_da),
346*05b00f60SXin Li GET_BE_U_4(ap->rrep_ds),
347*05b00f60SXin Li GET_IP6ADDR_STRING(ap->rrep_oa),
348*05b00f60SXin Li GET_BE_U_4(ap->rrep_life));
349*05b00f60SXin Li i = length - sizeof(*ap);
350*05b00f60SXin Li if (i >= sizeof(struct aodv_ext))
351*05b00f60SXin Li aodv_extension(ndo, (const struct aodv_ext *)(dat + sizeof(*ap)), i);
352*05b00f60SXin Li return;
353*05b00f60SXin Li
354*05b00f60SXin Li trunc:
355*05b00f60SXin Li nd_print_trunc(ndo);
356*05b00f60SXin Li }
357*05b00f60SXin Li
358*05b00f60SXin Li static void
aodv_v6_rerr(netdissect_options * ndo,const u_char * dat,u_int length)359*05b00f60SXin Li aodv_v6_rerr(netdissect_options *ndo, const u_char *dat, u_int length)
360*05b00f60SXin Li {
361*05b00f60SXin Li u_int i, dc;
362*05b00f60SXin Li const struct aodv_rerr *ap = (const struct aodv_rerr *)dat;
363*05b00f60SXin Li const struct rerr_unreach6 *dp6;
364*05b00f60SXin Li
365*05b00f60SXin Li ND_TCHECK_SIZE(ap);
366*05b00f60SXin Li if (length < sizeof(*ap))
367*05b00f60SXin Li goto trunc;
368*05b00f60SXin Li ND_PRINT(" rerr %s [items %u] [%u]:",
369*05b00f60SXin Li GET_U_1(ap->rerr_flags) & RERR_NODELETE ? "[D]" : "",
370*05b00f60SXin Li GET_U_1(ap->rerr_dc), length);
371*05b00f60SXin Li dp6 = (const struct rerr_unreach6 *)(const void *)(ap + 1);
372*05b00f60SXin Li i = length - sizeof(*ap);
373*05b00f60SXin Li for (dc = GET_U_1(ap->rerr_dc); dc != 0; dc--) {
374*05b00f60SXin Li ND_TCHECK_SIZE(dp6);
375*05b00f60SXin Li if (i < sizeof(*dp6))
376*05b00f60SXin Li goto trunc;
377*05b00f60SXin Li ND_PRINT(" {%s}(%u)", GET_IP6ADDR_STRING(dp6->u_da),
378*05b00f60SXin Li GET_BE_U_4(dp6->u_ds));
379*05b00f60SXin Li dp6++;
380*05b00f60SXin Li i -= sizeof(*dp6);
381*05b00f60SXin Li }
382*05b00f60SXin Li return;
383*05b00f60SXin Li
384*05b00f60SXin Li trunc:
385*05b00f60SXin Li nd_print_trunc(ndo);
386*05b00f60SXin Li }
387*05b00f60SXin Li
388*05b00f60SXin Li static void
aodv_v6_draft_01_rreq(netdissect_options * ndo,const u_char * dat,u_int length)389*05b00f60SXin Li aodv_v6_draft_01_rreq(netdissect_options *ndo, const u_char *dat, u_int length)
390*05b00f60SXin Li {
391*05b00f60SXin Li u_int i;
392*05b00f60SXin Li const struct aodv_rreq6_draft_01 *ap = (const struct aodv_rreq6_draft_01 *)dat;
393*05b00f60SXin Li
394*05b00f60SXin Li ND_TCHECK_SIZE(ap);
395*05b00f60SXin Li if (length < sizeof(*ap))
396*05b00f60SXin Li goto trunc;
397*05b00f60SXin Li ND_PRINT(" rreq %u %s%s%s%s%shops %u id 0x%08x\n"
398*05b00f60SXin Li "\tdst %s seq %u src %s seq %u", length,
399*05b00f60SXin Li GET_U_1(ap->rreq_type) & RREQ_JOIN ? "[J]" : "",
400*05b00f60SXin Li GET_U_1(ap->rreq_type) & RREQ_REPAIR ? "[R]" : "",
401*05b00f60SXin Li GET_U_1(ap->rreq_type) & RREQ_GRAT ? "[G]" : "",
402*05b00f60SXin Li GET_U_1(ap->rreq_type) & RREQ_DEST ? "[D]" : "",
403*05b00f60SXin Li GET_U_1(ap->rreq_type) & RREQ_UNKNOWN ? "[U] " : " ",
404*05b00f60SXin Li GET_U_1(ap->rreq_hops),
405*05b00f60SXin Li GET_BE_U_4(ap->rreq_id),
406*05b00f60SXin Li GET_IP6ADDR_STRING(ap->rreq_da),
407*05b00f60SXin Li GET_BE_U_4(ap->rreq_ds),
408*05b00f60SXin Li GET_IP6ADDR_STRING(ap->rreq_oa),
409*05b00f60SXin Li GET_BE_U_4(ap->rreq_os));
410*05b00f60SXin Li i = length - sizeof(*ap);
411*05b00f60SXin Li if (i >= sizeof(struct aodv_ext))
412*05b00f60SXin Li aodv_extension(ndo, (const struct aodv_ext *)(dat + sizeof(*ap)), i);
413*05b00f60SXin Li return;
414*05b00f60SXin Li
415*05b00f60SXin Li trunc:
416*05b00f60SXin Li nd_print_trunc(ndo);
417*05b00f60SXin Li }
418*05b00f60SXin Li
419*05b00f60SXin Li static void
aodv_v6_draft_01_rrep(netdissect_options * ndo,const u_char * dat,u_int length)420*05b00f60SXin Li aodv_v6_draft_01_rrep(netdissect_options *ndo, const u_char *dat, u_int length)
421*05b00f60SXin Li {
422*05b00f60SXin Li u_int i;
423*05b00f60SXin Li const struct aodv_rrep6_draft_01 *ap = (const struct aodv_rrep6_draft_01 *)dat;
424*05b00f60SXin Li
425*05b00f60SXin Li ND_TCHECK_SIZE(ap);
426*05b00f60SXin Li if (length < sizeof(*ap))
427*05b00f60SXin Li goto trunc;
428*05b00f60SXin Li ND_PRINT(" rrep %u %s%sprefix %u hops %u\n"
429*05b00f60SXin Li "\tdst %s dseq %u src %s %u ms", length,
430*05b00f60SXin Li GET_U_1(ap->rrep_type) & RREP_REPAIR ? "[R]" : "",
431*05b00f60SXin Li GET_U_1(ap->rrep_type) & RREP_ACK ? "[A] " : " ",
432*05b00f60SXin Li GET_U_1(ap->rrep_ps) & RREP_PREFIX_MASK,
433*05b00f60SXin Li GET_U_1(ap->rrep_hops),
434*05b00f60SXin Li GET_IP6ADDR_STRING(ap->rrep_da),
435*05b00f60SXin Li GET_BE_U_4(ap->rrep_ds),
436*05b00f60SXin Li GET_IP6ADDR_STRING(ap->rrep_oa),
437*05b00f60SXin Li GET_BE_U_4(ap->rrep_life));
438*05b00f60SXin Li i = length - sizeof(*ap);
439*05b00f60SXin Li if (i >= sizeof(struct aodv_ext))
440*05b00f60SXin Li aodv_extension(ndo, (const struct aodv_ext *)(dat + sizeof(*ap)), i);
441*05b00f60SXin Li return;
442*05b00f60SXin Li
443*05b00f60SXin Li trunc:
444*05b00f60SXin Li nd_print_trunc(ndo);
445*05b00f60SXin Li }
446*05b00f60SXin Li
447*05b00f60SXin Li static void
aodv_v6_draft_01_rerr(netdissect_options * ndo,const u_char * dat,u_int length)448*05b00f60SXin Li aodv_v6_draft_01_rerr(netdissect_options *ndo, const u_char *dat, u_int length)
449*05b00f60SXin Li {
450*05b00f60SXin Li u_int i, dc;
451*05b00f60SXin Li const struct aodv_rerr *ap = (const struct aodv_rerr *)dat;
452*05b00f60SXin Li const struct rerr_unreach6_draft_01 *dp6;
453*05b00f60SXin Li
454*05b00f60SXin Li ND_TCHECK_SIZE(ap);
455*05b00f60SXin Li if (length < sizeof(*ap))
456*05b00f60SXin Li goto trunc;
457*05b00f60SXin Li ND_PRINT(" rerr %s [items %u] [%u]:",
458*05b00f60SXin Li GET_U_1(ap->rerr_flags) & RERR_NODELETE ? "[D]" : "",
459*05b00f60SXin Li GET_U_1(ap->rerr_dc), length);
460*05b00f60SXin Li dp6 = (const struct rerr_unreach6_draft_01 *)(const void *)(ap + 1);
461*05b00f60SXin Li i = length - sizeof(*ap);
462*05b00f60SXin Li for (dc = GET_U_1(ap->rerr_dc); dc != 0; dc--) {
463*05b00f60SXin Li ND_TCHECK_SIZE(dp6);
464*05b00f60SXin Li if (i < sizeof(*dp6))
465*05b00f60SXin Li goto trunc;
466*05b00f60SXin Li ND_PRINT(" {%s}(%u)", GET_IP6ADDR_STRING(dp6->u_da),
467*05b00f60SXin Li GET_BE_U_4(dp6->u_ds));
468*05b00f60SXin Li dp6++;
469*05b00f60SXin Li i -= sizeof(*dp6);
470*05b00f60SXin Li }
471*05b00f60SXin Li return;
472*05b00f60SXin Li
473*05b00f60SXin Li trunc:
474*05b00f60SXin Li nd_print_trunc(ndo);
475*05b00f60SXin Li }
476*05b00f60SXin Li
477*05b00f60SXin Li void
aodv_print(netdissect_options * ndo,const u_char * dat,u_int length,int is_ip6)478*05b00f60SXin Li aodv_print(netdissect_options *ndo,
479*05b00f60SXin Li const u_char *dat, u_int length, int is_ip6)
480*05b00f60SXin Li {
481*05b00f60SXin Li uint8_t msg_type;
482*05b00f60SXin Li
483*05b00f60SXin Li ndo->ndo_protocol = "aodv";
484*05b00f60SXin Li /*
485*05b00f60SXin Li * The message type is the first byte; make sure we have it
486*05b00f60SXin Li * and then fetch it.
487*05b00f60SXin Li */
488*05b00f60SXin Li msg_type = GET_U_1(dat);
489*05b00f60SXin Li ND_PRINT(" aodv");
490*05b00f60SXin Li
491*05b00f60SXin Li switch (msg_type) {
492*05b00f60SXin Li
493*05b00f60SXin Li case AODV_RREQ:
494*05b00f60SXin Li if (is_ip6)
495*05b00f60SXin Li aodv_v6_rreq(ndo, dat, length);
496*05b00f60SXin Li else
497*05b00f60SXin Li aodv_rreq(ndo, dat, length);
498*05b00f60SXin Li break;
499*05b00f60SXin Li
500*05b00f60SXin Li case AODV_RREP:
501*05b00f60SXin Li if (is_ip6)
502*05b00f60SXin Li aodv_v6_rrep(ndo, dat, length);
503*05b00f60SXin Li else
504*05b00f60SXin Li aodv_rrep(ndo, dat, length);
505*05b00f60SXin Li break;
506*05b00f60SXin Li
507*05b00f60SXin Li case AODV_RERR:
508*05b00f60SXin Li if (is_ip6)
509*05b00f60SXin Li aodv_v6_rerr(ndo, dat, length);
510*05b00f60SXin Li else
511*05b00f60SXin Li aodv_rerr(ndo, dat, length);
512*05b00f60SXin Li break;
513*05b00f60SXin Li
514*05b00f60SXin Li case AODV_RREP_ACK:
515*05b00f60SXin Li ND_PRINT(" rrep-ack %u", length);
516*05b00f60SXin Li break;
517*05b00f60SXin Li
518*05b00f60SXin Li case AODV_V6_DRAFT_01_RREQ:
519*05b00f60SXin Li aodv_v6_draft_01_rreq(ndo, dat, length);
520*05b00f60SXin Li break;
521*05b00f60SXin Li
522*05b00f60SXin Li case AODV_V6_DRAFT_01_RREP:
523*05b00f60SXin Li aodv_v6_draft_01_rrep(ndo, dat, length);
524*05b00f60SXin Li break;
525*05b00f60SXin Li
526*05b00f60SXin Li case AODV_V6_DRAFT_01_RERR:
527*05b00f60SXin Li aodv_v6_draft_01_rerr(ndo, dat, length);
528*05b00f60SXin Li break;
529*05b00f60SXin Li
530*05b00f60SXin Li case AODV_V6_DRAFT_01_RREP_ACK:
531*05b00f60SXin Li ND_PRINT(" rrep-ack %u", length);
532*05b00f60SXin Li break;
533*05b00f60SXin Li
534*05b00f60SXin Li default:
535*05b00f60SXin Li ND_PRINT(" type %u %u", msg_type, length);
536*05b00f60SXin Li }
537*05b00f60SXin Li }
538