print-ospf6.c revision 1.1.1.10 1 1.1 christos /*
2 1.1 christos * Copyright (c) 1992, 1993, 1994, 1995, 1996, 1997
3 1.1 christos * The Regents of the University of California. All rights reserved.
4 1.1 christos *
5 1.1 christos * Redistribution and use in source and binary forms, with or without
6 1.1 christos * modification, are permitted provided that: (1) source code distributions
7 1.1 christos * retain the above copyright notice and this paragraph in its entirety, (2)
8 1.1 christos * distributions including binary code include the above copyright notice and
9 1.1 christos * this paragraph in its entirety in the documentation or other materials
10 1.1 christos * provided with the distribution, and (3) all advertising materials mentioning
11 1.1 christos * features or use of this software display the following acknowledgement:
12 1.1 christos * ``This product includes software developed by the University of California,
13 1.1 christos * Lawrence Berkeley Laboratory and its contributors.'' Neither the name of
14 1.1 christos * the University nor the names of its contributors may be used to endorse
15 1.1 christos * or promote products derived from this software without specific prior
16 1.1 christos * written permission.
17 1.1 christos * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED
18 1.1 christos * WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF
19 1.1 christos * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
20 1.1 christos *
21 1.1 christos * OSPF support contributed by Jeffrey Honig (jch (at) mitchell.cit.cornell.edu)
22 1.1 christos */
23 1.1 christos
24 1.1.1.6 spz /* \summary: IPv6 Open Shortest Path First (OSPFv3) printer */
25 1.1.1.6 spz
26 1.1.1.9 christos #include <config.h>
27 1.1 christos
28 1.1.1.9 christos #include "netdissect-stdinc.h"
29 1.1 christos
30 1.1 christos #include <string.h>
31 1.1 christos
32 1.1.1.5 christos #include "netdissect.h"
33 1.1 christos #include "addrtoname.h"
34 1.1 christos #include "extract.h"
35 1.1 christos
36 1.1 christos #include "ospf.h"
37 1.1.1.4 christos
38 1.1.1.4 christos #define OSPF_TYPE_HELLO 1 /* Hello */
39 1.1.1.4 christos #define OSPF_TYPE_DD 2 /* Database Description */
40 1.1.1.4 christos #define OSPF_TYPE_LS_REQ 3 /* Link State Request */
41 1.1.1.4 christos #define OSPF_TYPE_LS_UPDATE 4 /* Link State Update */
42 1.1.1.4 christos #define OSPF_TYPE_LS_ACK 5 /* Link State Ack */
43 1.1.1.4 christos
44 1.1.1.4 christos /* Options *_options */
45 1.1.1.4 christos #define OSPF6_OPTION_V6 0x01 /* V6 bit: A bit for peeping tom */
46 1.1.1.4 christos #define OSPF6_OPTION_E 0x02 /* E bit: External routes advertised */
47 1.1.1.4 christos #define OSPF6_OPTION_MC 0x04 /* MC bit: Multicast capable */
48 1.1.1.4 christos #define OSPF6_OPTION_N 0x08 /* N bit: For type-7 LSA */
49 1.1.1.4 christos #define OSPF6_OPTION_R 0x10 /* R bit: Router bit */
50 1.1.1.4 christos #define OSPF6_OPTION_DC 0x20 /* DC bit: Demand circuits */
51 1.1.1.4 christos /* The field is actually 24-bit (RFC5340 Section A.2). */
52 1.1.1.4 christos #define OSPF6_OPTION_AF 0x0100 /* AF bit: Multiple address families */
53 1.1.1.4 christos #define OSPF6_OPTION_L 0x0200 /* L bit: Link-local signaling (LLS) */
54 1.1.1.4 christos #define OSPF6_OPTION_AT 0x0400 /* AT bit: Authentication trailer */
55 1.1.1.4 christos
56 1.1.1.4 christos
57 1.1.1.4 christos /* db_flags */
58 1.1.1.4 christos #define OSPF6_DB_INIT 0x04 /* */
59 1.1.1.4 christos #define OSPF6_DB_MORE 0x02
60 1.1.1.4 christos #define OSPF6_DB_MASTER 0x01
61 1.1.1.4 christos #define OSPF6_DB_M6 0x10 /* IPv6 MTU */
62 1.1.1.4 christos
63 1.1.1.4 christos /* ls_type */
64 1.1.1.4 christos #define LS_TYPE_ROUTER 1 /* router link */
65 1.1.1.4 christos #define LS_TYPE_NETWORK 2 /* network link */
66 1.1.1.4 christos #define LS_TYPE_INTER_AP 3 /* Inter-Area-Prefix */
67 1.1.1.4 christos #define LS_TYPE_INTER_AR 4 /* Inter-Area-Router */
68 1.1.1.4 christos #define LS_TYPE_ASE 5 /* ASE */
69 1.1.1.4 christos #define LS_TYPE_GROUP 6 /* Group membership */
70 1.1.1.4 christos #define LS_TYPE_NSSA 7 /* NSSA */
71 1.1.1.4 christos #define LS_TYPE_LINK 8 /* Link LSA */
72 1.1.1.4 christos #define LS_TYPE_INTRA_AP 9 /* Intra-Area-Prefix */
73 1.1.1.4 christos #define LS_TYPE_INTRA_ATE 10 /* Intra-Area-TE */
74 1.1.1.4 christos #define LS_TYPE_GRACE 11 /* Grace LSA */
75 1.1.1.4 christos #define LS_TYPE_RI 12 /* Router information */
76 1.1.1.4 christos #define LS_TYPE_INTER_ASTE 13 /* Inter-AS-TE */
77 1.1.1.4 christos #define LS_TYPE_L1VPN 14 /* L1VPN */
78 1.1.1.4 christos #define LS_TYPE_MASK 0x1fff
79 1.1.1.4 christos
80 1.1.1.4 christos #define LS_SCOPE_LINKLOCAL 0x0000
81 1.1.1.4 christos #define LS_SCOPE_AREA 0x2000
82 1.1.1.4 christos #define LS_SCOPE_AS 0x4000
83 1.1.1.4 christos #define LS_SCOPE_MASK 0x6000
84 1.1.1.4 christos #define LS_SCOPE_U 0x8000
85 1.1.1.4 christos
86 1.1.1.4 christos /* rla_link.link_type */
87 1.1.1.4 christos #define RLA_TYPE_ROUTER 1 /* point-to-point to another router */
88 1.1.1.4 christos #define RLA_TYPE_TRANSIT 2 /* connection to transit network */
89 1.1.1.4 christos #define RLA_TYPE_VIRTUAL 4 /* virtual link */
90 1.1.1.4 christos
91 1.1.1.4 christos /* rla_flags */
92 1.1.1.4 christos #define RLA_FLAG_B 0x01
93 1.1.1.4 christos #define RLA_FLAG_E 0x02
94 1.1.1.4 christos #define RLA_FLAG_V 0x04
95 1.1.1.4 christos #define RLA_FLAG_W 0x08
96 1.1.1.9 christos #define RLA_FLAG_Nt 0x10
97 1.1.1.4 christos
98 1.1.1.4 christos /* lsa_prefix options */
99 1.1.1.4 christos #define LSA_PREFIX_OPT_NU 0x01
100 1.1.1.4 christos #define LSA_PREFIX_OPT_LA 0x02
101 1.1.1.4 christos #define LSA_PREFIX_OPT_MC 0x04
102 1.1.1.4 christos #define LSA_PREFIX_OPT_P 0x08
103 1.1.1.4 christos #define LSA_PREFIX_OPT_DN 0x10
104 1.1.1.9 christos #define LSA_PREFIX_OPT_N 0x20
105 1.1.1.4 christos
106 1.1.1.4 christos /* sla_tosmetric breakdown */
107 1.1.1.4 christos #define SLA_MASK_TOS 0x7f000000
108 1.1.1.4 christos #define SLA_MASK_METRIC 0x00ffffff
109 1.1.1.4 christos #define SLA_SHIFT_TOS 24
110 1.1.1.4 christos
111 1.1.1.4 christos /* asla_metric */
112 1.1.1.4 christos #define ASLA_FLAG_FWDADDR 0x02000000
113 1.1.1.4 christos #define ASLA_FLAG_ROUTETAG 0x01000000
114 1.1.1.4 christos #define ASLA_MASK_METRIC 0x00ffffff
115 1.1.1.4 christos
116 1.1.1.4 christos /* RFC6506 Section 4.1 */
117 1.1.1.4 christos #define OSPF6_AT_HDRLEN 16U
118 1.1.1.4 christos #define OSPF6_AUTH_TYPE_HMAC 0x0001
119 1.1.1.4 christos
120 1.1.1.9 christos typedef nd_uint32_t rtrid_t;
121 1.1.1.4 christos
122 1.1.1.4 christos /* link state advertisement header */
123 1.1.1.4 christos struct lsa6_hdr {
124 1.1.1.9 christos nd_uint16_t ls_age;
125 1.1.1.9 christos nd_uint16_t ls_type;
126 1.1.1.4 christos rtrid_t ls_stateid;
127 1.1.1.4 christos rtrid_t ls_router;
128 1.1.1.9 christos nd_uint32_t ls_seq;
129 1.1.1.9 christos nd_uint16_t ls_chksum;
130 1.1.1.9 christos nd_uint16_t ls_length;
131 1.1.1.4 christos };
132 1.1.1.4 christos
133 1.1.1.4 christos /* Length of an IPv6 address, in bytes. */
134 1.1.1.4 christos #define IPV6_ADDR_LEN_BYTES (128/8)
135 1.1.1.4 christos
136 1.1.1.4 christos struct lsa6_prefix {
137 1.1.1.9 christos nd_uint8_t lsa_p_len;
138 1.1.1.9 christos nd_uint8_t lsa_p_opt;
139 1.1.1.9 christos nd_uint16_t lsa_p_metric;
140 1.1.1.9 christos nd_byte lsa_p_prefix[IPV6_ADDR_LEN_BYTES]; /* maximum length */
141 1.1.1.4 christos };
142 1.1.1.4 christos
143 1.1.1.4 christos /* link state advertisement */
144 1.1.1.4 christos struct lsa6 {
145 1.1.1.4 christos struct lsa6_hdr ls_hdr;
146 1.1.1.4 christos
147 1.1.1.4 christos /* Link state types */
148 1.1.1.4 christos union {
149 1.1.1.4 christos /* Router links advertisements */
150 1.1.1.4 christos struct {
151 1.1.1.4 christos union {
152 1.1.1.9 christos nd_uint8_t flg;
153 1.1.1.9 christos nd_uint32_t opt;
154 1.1.1.4 christos } rla_flgandopt;
155 1.1.1.4 christos #define rla_flags rla_flgandopt.flg
156 1.1.1.4 christos #define rla_options rla_flgandopt.opt
157 1.1.1.4 christos struct rlalink6 {
158 1.1.1.9 christos nd_uint8_t link_type;
159 1.1.1.9 christos nd_byte link_zero;
160 1.1.1.9 christos nd_uint16_t link_metric;
161 1.1.1.9 christos nd_uint32_t link_ifid;
162 1.1.1.9 christos nd_uint32_t link_nifid;
163 1.1.1.4 christos rtrid_t link_nrtid;
164 1.1.1.4 christos } rla_link[1]; /* may repeat */
165 1.1.1.4 christos } un_rla;
166 1.1.1.4 christos
167 1.1.1.4 christos /* Network links advertisements */
168 1.1.1.4 christos struct {
169 1.1.1.9 christos nd_uint32_t nla_options;
170 1.1.1.4 christos rtrid_t nla_router[1]; /* may repeat */
171 1.1.1.4 christos } un_nla;
172 1.1.1.4 christos
173 1.1.1.4 christos /* Inter Area Prefix LSA */
174 1.1.1.4 christos struct {
175 1.1.1.9 christos nd_uint32_t inter_ap_metric;
176 1.1.1.4 christos struct lsa6_prefix inter_ap_prefix[1];
177 1.1.1.4 christos } un_inter_ap;
178 1.1.1.4 christos
179 1.1.1.4 christos /* AS external links advertisements */
180 1.1.1.4 christos struct {
181 1.1.1.9 christos nd_uint32_t asla_metric;
182 1.1.1.4 christos struct lsa6_prefix asla_prefix[1];
183 1.1.1.4 christos /* some optional fields follow */
184 1.1.1.4 christos } un_asla;
185 1.1.1.4 christos
186 1.1.1.4 christos #if 0
187 1.1.1.4 christos /* Summary links advertisements */
188 1.1.1.4 christos struct {
189 1.1.1.9 christos nd_ipv4 sla_mask;
190 1.1.1.9 christos nd_uint32_t sla_tosmetric[1]; /* may repeat */
191 1.1.1.4 christos } un_sla;
192 1.1.1.4 christos
193 1.1.1.4 christos /* Multicast group membership */
194 1.1.1.4 christos struct mcla {
195 1.1.1.9 christos nd_uint32_t mcla_vtype;
196 1.1.1.9 christos nd_ipv4 mcla_vid;
197 1.1.1.4 christos } un_mcla[1];
198 1.1.1.4 christos #endif
199 1.1.1.4 christos
200 1.1.1.4 christos /* Type 7 LSA */
201 1.1.1.4 christos
202 1.1.1.4 christos /* Link LSA */
203 1.1.1.4 christos struct llsa {
204 1.1.1.4 christos union {
205 1.1.1.9 christos nd_uint8_t pri;
206 1.1.1.9 christos nd_uint32_t opt;
207 1.1.1.4 christos } llsa_priandopt;
208 1.1.1.4 christos #define llsa_priority llsa_priandopt.pri
209 1.1.1.4 christos #define llsa_options llsa_priandopt.opt
210 1.1.1.9 christos nd_ipv6 llsa_lladdr;
211 1.1.1.9 christos nd_uint32_t llsa_nprefix;
212 1.1.1.4 christos struct lsa6_prefix llsa_prefix[1];
213 1.1.1.4 christos } un_llsa;
214 1.1.1.4 christos
215 1.1.1.4 christos /* Intra-Area-Prefix */
216 1.1.1.4 christos struct {
217 1.1.1.9 christos nd_uint16_t intra_ap_nprefix;
218 1.1.1.9 christos nd_uint16_t intra_ap_lstype;
219 1.1.1.4 christos rtrid_t intra_ap_lsid;
220 1.1.1.4 christos rtrid_t intra_ap_rtid;
221 1.1.1.4 christos struct lsa6_prefix intra_ap_prefix[1];
222 1.1.1.4 christos } un_intra_ap;
223 1.1.1.4 christos } lsa_un;
224 1.1.1.4 christos };
225 1.1.1.4 christos
226 1.1.1.4 christos /*
227 1.1.1.4 christos * the main header
228 1.1.1.4 christos */
229 1.1.1.4 christos struct ospf6hdr {
230 1.1.1.9 christos nd_uint8_t ospf6_version;
231 1.1.1.9 christos nd_uint8_t ospf6_type;
232 1.1.1.9 christos nd_uint16_t ospf6_len;
233 1.1.1.4 christos rtrid_t ospf6_routerid;
234 1.1.1.4 christos rtrid_t ospf6_areaid;
235 1.1.1.9 christos nd_uint16_t ospf6_chksum;
236 1.1.1.9 christos nd_uint8_t ospf6_instanceid;
237 1.1.1.9 christos nd_uint8_t ospf6_rsvd;
238 1.1.1.4 christos };
239 1.1.1.4 christos
240 1.1.1.4 christos /*
241 1.1.1.4 christos * The OSPF6 header length is 16 bytes, regardless of how your compiler
242 1.1.1.4 christos * might choose to pad the above structure.
243 1.1.1.4 christos */
244 1.1.1.4 christos #define OSPF6HDR_LEN 16
245 1.1.1.4 christos
246 1.1.1.4 christos /* Hello packet */
247 1.1.1.4 christos struct hello6 {
248 1.1.1.9 christos nd_uint32_t hello_ifid;
249 1.1.1.4 christos union {
250 1.1.1.9 christos nd_uint8_t pri;
251 1.1.1.9 christos nd_uint32_t opt;
252 1.1.1.4 christos } hello_priandopt;
253 1.1.1.4 christos #define hello_priority hello_priandopt.pri
254 1.1.1.4 christos #define hello_options hello_priandopt.opt
255 1.1.1.9 christos nd_uint16_t hello_helloint;
256 1.1.1.9 christos nd_uint16_t hello_deadint;
257 1.1.1.4 christos rtrid_t hello_dr;
258 1.1.1.4 christos rtrid_t hello_bdr;
259 1.1.1.4 christos rtrid_t hello_neighbor[1]; /* may repeat */
260 1.1.1.4 christos };
261 1.1.1.4 christos
262 1.1.1.4 christos /* Database Description packet */
263 1.1.1.4 christos struct dd6 {
264 1.1.1.9 christos nd_uint32_t db_options;
265 1.1.1.9 christos nd_uint16_t db_mtu;
266 1.1.1.9 christos nd_uint8_t db_mbz;
267 1.1.1.9 christos nd_uint8_t db_flags;
268 1.1.1.9 christos nd_uint32_t db_seq;
269 1.1.1.4 christos struct lsa6_hdr db_lshdr[1]; /* may repeat */
270 1.1.1.4 christos };
271 1.1.1.4 christos
272 1.1.1.4 christos /* Link State Request */
273 1.1.1.4 christos struct lsr6 {
274 1.1.1.9 christos nd_uint16_t ls_mbz;
275 1.1.1.9 christos nd_uint16_t ls_type;
276 1.1.1.4 christos rtrid_t ls_stateid;
277 1.1.1.4 christos rtrid_t ls_router;
278 1.1.1.4 christos };
279 1.1.1.4 christos
280 1.1.1.4 christos /* Link State Update */
281 1.1.1.4 christos struct lsu6 {
282 1.1.1.9 christos nd_uint32_t lsu_count;
283 1.1.1.4 christos struct lsa6 lsu_lsa[1]; /* may repeat */
284 1.1.1.4 christos };
285 1.1.1.4 christos
286 1.1 christos
287 1.1 christos static const struct tok ospf6_option_values[] = {
288 1.1 christos { OSPF6_OPTION_V6, "V6" },
289 1.1 christos { OSPF6_OPTION_E, "External" },
290 1.1.1.4 christos { OSPF6_OPTION_MC, "Deprecated" },
291 1.1 christos { OSPF6_OPTION_N, "NSSA" },
292 1.1 christos { OSPF6_OPTION_R, "Router" },
293 1.1 christos { OSPF6_OPTION_DC, "Demand Circuit" },
294 1.1.1.4 christos { OSPF6_OPTION_AF, "AFs Support" },
295 1.1.1.4 christos { OSPF6_OPTION_L, "LLS" },
296 1.1.1.4 christos { OSPF6_OPTION_AT, "Authentication Trailer" },
297 1.1 christos { 0, NULL }
298 1.1 christos };
299 1.1 christos
300 1.1 christos static const struct tok ospf6_rla_flag_values[] = {
301 1.1 christos { RLA_FLAG_B, "ABR" },
302 1.1 christos { RLA_FLAG_E, "External" },
303 1.1 christos { RLA_FLAG_V, "Virtual-Link Endpoint" },
304 1.1.1.9 christos { RLA_FLAG_W, "Deprecated" },
305 1.1.1.9 christos { RLA_FLAG_Nt, "NSSA Translator" },
306 1.1 christos { 0, NULL }
307 1.1 christos };
308 1.1 christos
309 1.1 christos static const struct tok ospf6_asla_flag_values[] = {
310 1.1 christos { ASLA_FLAG_EXTERNAL, "External Type 2" },
311 1.1.1.4 christos { ASLA_FLAG_FWDADDR, "Forwarding" },
312 1.1 christos { ASLA_FLAG_ROUTETAG, "Tag" },
313 1.1 christos { 0, NULL }
314 1.1 christos };
315 1.1 christos
316 1.1.1.3 christos static const struct tok ospf6_type_values[] = {
317 1.1 christos { OSPF_TYPE_HELLO, "Hello" },
318 1.1 christos { OSPF_TYPE_DD, "Database Description" },
319 1.1 christos { OSPF_TYPE_LS_REQ, "LS-Request" },
320 1.1 christos { OSPF_TYPE_LS_UPDATE, "LS-Update" },
321 1.1 christos { OSPF_TYPE_LS_ACK, "LS-Ack" },
322 1.1 christos { 0, NULL }
323 1.1 christos };
324 1.1 christos
325 1.1.1.3 christos static const struct tok ospf6_lsa_values[] = {
326 1.1 christos { LS_TYPE_ROUTER, "Router" },
327 1.1 christos { LS_TYPE_NETWORK, "Network" },
328 1.1 christos { LS_TYPE_INTER_AP, "Inter-Area Prefix" },
329 1.1 christos { LS_TYPE_INTER_AR, "Inter-Area Router" },
330 1.1 christos { LS_TYPE_ASE, "External" },
331 1.1.1.4 christos { LS_TYPE_GROUP, "Deprecated" },
332 1.1 christos { LS_TYPE_NSSA, "NSSA" },
333 1.1 christos { LS_TYPE_LINK, "Link" },
334 1.1 christos { LS_TYPE_INTRA_AP, "Intra-Area Prefix" },
335 1.1 christos { LS_TYPE_INTRA_ATE, "Intra-Area TE" },
336 1.1 christos { LS_TYPE_GRACE, "Grace" },
337 1.1.1.4 christos { LS_TYPE_RI, "Router Information" },
338 1.1.1.4 christos { LS_TYPE_INTER_ASTE, "Inter-AS-TE" },
339 1.1.1.4 christos { LS_TYPE_L1VPN, "Layer 1 VPN" },
340 1.1 christos { 0, NULL }
341 1.1 christos };
342 1.1 christos
343 1.1.1.3 christos static const struct tok ospf6_ls_scope_values[] = {
344 1.1 christos { LS_SCOPE_LINKLOCAL, "Link Local" },
345 1.1 christos { LS_SCOPE_AREA, "Area Local" },
346 1.1 christos { LS_SCOPE_AS, "Domain Wide" },
347 1.1 christos { 0, NULL }
348 1.1 christos };
349 1.1 christos
350 1.1.1.3 christos static const struct tok ospf6_dd_flag_values[] = {
351 1.1 christos { OSPF6_DB_INIT, "Init" },
352 1.1 christos { OSPF6_DB_MORE, "More" },
353 1.1 christos { OSPF6_DB_MASTER, "Master" },
354 1.1.1.4 christos { OSPF6_DB_M6, "IPv6 MTU" },
355 1.1 christos { 0, NULL }
356 1.1 christos };
357 1.1 christos
358 1.1.1.3 christos static const struct tok ospf6_lsa_prefix_option_values[] = {
359 1.1 christos { LSA_PREFIX_OPT_NU, "No Unicast" },
360 1.1 christos { LSA_PREFIX_OPT_LA, "Local address" },
361 1.1.1.4 christos { LSA_PREFIX_OPT_MC, "Deprecated" },
362 1.1 christos { LSA_PREFIX_OPT_P, "Propagate" },
363 1.1 christos { LSA_PREFIX_OPT_DN, "Down" },
364 1.1.1.9 christos { LSA_PREFIX_OPT_N, "N-bit" },
365 1.1 christos { 0, NULL }
366 1.1 christos };
367 1.1 christos
368 1.1.1.4 christos static const struct tok ospf6_auth_type_str[] = {
369 1.1.1.4 christos { OSPF6_AUTH_TYPE_HMAC, "HMAC" },
370 1.1.1.4 christos { 0, NULL }
371 1.1.1.4 christos };
372 1.1 christos
373 1.1 christos static void
374 1.1.1.4 christos ospf6_print_ls_type(netdissect_options *ndo,
375 1.1.1.9 christos u_int ls_type, const rtrid_t *ls_stateid)
376 1.1 christos {
377 1.1.1.9 christos ND_PRINT("\n\t %s LSA (%u), %s Scope%s, LSA-ID %s",
378 1.1 christos tok2str(ospf6_lsa_values, "Unknown", ls_type & LS_TYPE_MASK),
379 1.1 christos ls_type & LS_TYPE_MASK,
380 1.1 christos tok2str(ospf6_ls_scope_values, "Unknown", ls_type & LS_SCOPE_MASK),
381 1.1 christos ls_type &0x8000 ? ", transitive" : "", /* U-bit */
382 1.1.1.9 christos GET_IPADDR_STRING(ls_stateid));
383 1.1 christos }
384 1.1 christos
385 1.1 christos static int
386 1.1.1.4 christos ospf6_print_lshdr(netdissect_options *ndo,
387 1.1.1.9 christos const struct lsa6_hdr *lshp, const u_char *dataend)
388 1.1 christos {
389 1.1.1.10 christos u_int ls_length;
390 1.1.1.10 christos
391 1.1.1.5 christos if ((const u_char *)(lshp + 1) > dataend)
392 1.1.1.4 christos goto trunc;
393 1.1 christos
394 1.1.1.10 christos ls_length = GET_BE_U_2(lshp->ls_length);
395 1.1.1.10 christos if (ls_length < sizeof(struct lsa_hdr)) {
396 1.1.1.10 christos ND_PRINT("\n\t Bogus length %u < header (%zu)", ls_length,
397 1.1.1.10 christos sizeof(struct lsa_hdr));
398 1.1.1.10 christos goto trunc;
399 1.1.1.10 christos }
400 1.1.1.10 christos
401 1.1.1.9 christos ND_PRINT("\n\t Advertising Router %s, seq 0x%08x, age %us, length %zu",
402 1.1.1.9 christos GET_IPADDR_STRING(lshp->ls_router),
403 1.1.1.9 christos GET_BE_U_4(lshp->ls_seq),
404 1.1.1.9 christos GET_BE_U_2(lshp->ls_age),
405 1.1.1.10 christos ls_length-sizeof(struct lsa6_hdr));
406 1.1 christos
407 1.1.1.9 christos ospf6_print_ls_type(ndo, GET_BE_U_2(lshp->ls_type),
408 1.1.1.9 christos &lshp->ls_stateid);
409 1.1 christos
410 1.1 christos return (0);
411 1.1 christos trunc:
412 1.1 christos return (1);
413 1.1 christos }
414 1.1 christos
415 1.1 christos static int
416 1.1.1.4 christos ospf6_print_lsaprefix(netdissect_options *ndo,
417 1.1.1.4 christos const uint8_t *tptr, u_int lsa_length)
418 1.1 christos {
419 1.1.1.5 christos const struct lsa6_prefix *lsapp = (const struct lsa6_prefix *)tptr;
420 1.1 christos u_int wordlen;
421 1.1.1.9 christos nd_ipv6 prefix;
422 1.1 christos
423 1.1.1.4 christos if (lsa_length < sizeof (*lsapp) - IPV6_ADDR_LEN_BYTES)
424 1.1.1.2 christos goto trunc;
425 1.1.1.4 christos lsa_length -= sizeof (*lsapp) - IPV6_ADDR_LEN_BYTES;
426 1.1.1.9 christos ND_TCHECK_LEN(lsapp, sizeof(*lsapp) - IPV6_ADDR_LEN_BYTES);
427 1.1.1.9 christos wordlen = (GET_U_1(lsapp->lsa_p_len) + 31) / 32;
428 1.1.1.9 christos if (wordlen * 4 > sizeof(nd_ipv6)) {
429 1.1.1.9 christos ND_PRINT(" bogus prefixlen /%u", GET_U_1(lsapp->lsa_p_len));
430 1.1 christos goto trunc;
431 1.1 christos }
432 1.1.1.2 christos if (lsa_length < wordlen * 4)
433 1.1.1.2 christos goto trunc;
434 1.1.1.2 christos lsa_length -= wordlen * 4;
435 1.1.1.9 christos memset(prefix, 0, sizeof(prefix));
436 1.1.1.9 christos GET_CPY_BYTES(prefix, lsapp->lsa_p_prefix, wordlen * 4);
437 1.1.1.9 christos ND_PRINT("\n\t\t%s/%u", ip6addr_string(ndo, prefix), /* local buffer, not packet data; don't use GET_IP6ADDR_STRING() */
438 1.1.1.9 christos GET_U_1(lsapp->lsa_p_len));
439 1.1.1.9 christos if (GET_U_1(lsapp->lsa_p_opt)) {
440 1.1.1.9 christos ND_PRINT(", Options [%s]",
441 1.1 christos bittok2str(ospf6_lsa_prefix_option_values,
442 1.1.1.9 christos "none", GET_U_1(lsapp->lsa_p_opt)));
443 1.1 christos }
444 1.1.1.9 christos ND_PRINT(", metric %u", GET_BE_U_2(lsapp->lsa_p_metric));
445 1.1.1.4 christos return sizeof(*lsapp) - IPV6_ADDR_LEN_BYTES + wordlen * 4;
446 1.1 christos
447 1.1 christos trunc:
448 1.1 christos return -1;
449 1.1 christos }
450 1.1 christos
451 1.1 christos
452 1.1 christos /*
453 1.1 christos * Print a single link state advertisement. If truncated return 1, else 0.
454 1.1 christos */
455 1.1 christos static int
456 1.1.1.4 christos ospf6_print_lsa(netdissect_options *ndo,
457 1.1.1.9 christos const struct lsa6 *lsap, const u_char *dataend)
458 1.1 christos {
459 1.1.1.9 christos const struct rlalink6 *rlp;
460 1.1 christos #if 0
461 1.1.1.9 christos const struct tos_metric *tosp;
462 1.1 christos #endif
463 1.1.1.9 christos const rtrid_t *ap;
464 1.1 christos #if 0
465 1.1.1.9 christos const struct aslametric *almp;
466 1.1.1.9 christos const struct mcla *mcp;
467 1.1 christos #endif
468 1.1.1.9 christos const struct llsa *llsap;
469 1.1.1.9 christos const struct lsa6_prefix *lsapp;
470 1.1 christos #if 0
471 1.1.1.9 christos const uint32_t *lp;
472 1.1 christos #endif
473 1.1.1.9 christos u_int prefixes;
474 1.1.1.9 christos int bytelen;
475 1.1.1.9 christos u_int length, lsa_length;
476 1.1.1.4 christos uint32_t flags32;
477 1.1.1.4 christos const uint8_t *tptr;
478 1.1 christos
479 1.1.1.4 christos if (ospf6_print_lshdr(ndo, &lsap->ls_hdr, dataend))
480 1.1 christos return (1);
481 1.1.1.9 christos length = GET_BE_U_2(lsap->ls_hdr.ls_length);
482 1.1.1.2 christos
483 1.1.1.2 christos /*
484 1.1.1.2 christos * The LSA length includes the length of the header;
485 1.1.1.2 christos * it must have a value that's at least that length.
486 1.1.1.2 christos * If it does, find the length of what follows the
487 1.1.1.2 christos * header.
488 1.1.1.2 christos */
489 1.1.1.5 christos if (length < sizeof(struct lsa6_hdr) || (const u_char *)lsap + length > dataend)
490 1.1.1.9 christos return (1);
491 1.1 christos lsa_length = length - sizeof(struct lsa6_hdr);
492 1.1.1.5 christos tptr = (const uint8_t *)lsap+sizeof(struct lsa6_hdr);
493 1.1 christos
494 1.1.1.9 christos switch (GET_BE_U_2(lsap->ls_hdr.ls_type)) {
495 1.1 christos case LS_TYPE_ROUTER | LS_SCOPE_AREA:
496 1.1.1.2 christos if (lsa_length < sizeof (lsap->lsa_un.un_rla.rla_options))
497 1.1.1.2 christos return (1);
498 1.1.1.2 christos lsa_length -= sizeof (lsap->lsa_un.un_rla.rla_options);
499 1.1.1.9 christos ND_PRINT("\n\t Options [%s]",
500 1.1.1.4 christos bittok2str(ospf6_option_values, "none",
501 1.1.1.9 christos GET_BE_U_4(lsap->lsa_un.un_rla.rla_options)));
502 1.1.1.9 christos ND_PRINT(", RLA-Flags [%s]",
503 1.1.1.4 christos bittok2str(ospf6_rla_flag_values, "none",
504 1.1.1.9 christos GET_U_1(lsap->lsa_un.un_rla.rla_flags)));
505 1.1 christos
506 1.1 christos rlp = lsap->lsa_un.un_rla.rla_link;
507 1.1.1.2 christos while (lsa_length != 0) {
508 1.1.1.2 christos if (lsa_length < sizeof (*rlp))
509 1.1.1.2 christos return (1);
510 1.1.1.2 christos lsa_length -= sizeof (*rlp);
511 1.1.1.9 christos ND_TCHECK_SIZE(rlp);
512 1.1.1.9 christos switch (GET_U_1(rlp->link_type)) {
513 1.1 christos
514 1.1 christos case RLA_TYPE_VIRTUAL:
515 1.1.1.9 christos ND_PRINT("\n\t Virtual Link: Neighbor Router-ID %s"
516 1.1 christos "\n\t Neighbor Interface-ID %s, Interface %s",
517 1.1.1.9 christos GET_IPADDR_STRING(rlp->link_nrtid),
518 1.1.1.9 christos GET_IPADDR_STRING(rlp->link_nifid),
519 1.1.1.9 christos GET_IPADDR_STRING(rlp->link_ifid));
520 1.1 christos break;
521 1.1 christos
522 1.1 christos case RLA_TYPE_ROUTER:
523 1.1.1.9 christos ND_PRINT("\n\t Neighbor Router-ID %s"
524 1.1 christos "\n\t Neighbor Interface-ID %s, Interface %s",
525 1.1.1.9 christos GET_IPADDR_STRING(rlp->link_nrtid),
526 1.1.1.9 christos GET_IPADDR_STRING(rlp->link_nifid),
527 1.1.1.9 christos GET_IPADDR_STRING(rlp->link_ifid));
528 1.1 christos break;
529 1.1 christos
530 1.1 christos case RLA_TYPE_TRANSIT:
531 1.1.1.9 christos ND_PRINT("\n\t Neighbor Network-ID %s"
532 1.1 christos "\n\t Neighbor Interface-ID %s, Interface %s",
533 1.1.1.9 christos GET_IPADDR_STRING(rlp->link_nrtid),
534 1.1.1.9 christos GET_IPADDR_STRING(rlp->link_nifid),
535 1.1.1.9 christos GET_IPADDR_STRING(rlp->link_ifid));
536 1.1 christos break;
537 1.1 christos
538 1.1 christos default:
539 1.1.1.9 christos ND_PRINT("\n\t Unknown Router Links Type 0x%02x",
540 1.1.1.9 christos GET_U_1(rlp->link_type));
541 1.1 christos return (0);
542 1.1 christos }
543 1.1.1.9 christos ND_PRINT(", metric %u", GET_BE_U_2(rlp->link_metric));
544 1.1 christos rlp++;
545 1.1 christos }
546 1.1 christos break;
547 1.1 christos
548 1.1 christos case LS_TYPE_NETWORK | LS_SCOPE_AREA:
549 1.1.1.2 christos if (lsa_length < sizeof (lsap->lsa_un.un_nla.nla_options))
550 1.1.1.2 christos return (1);
551 1.1.1.2 christos lsa_length -= sizeof (lsap->lsa_un.un_nla.nla_options);
552 1.1.1.9 christos ND_PRINT("\n\t Options [%s]",
553 1.1.1.4 christos bittok2str(ospf6_option_values, "none",
554 1.1.1.9 christos GET_BE_U_4(lsap->lsa_un.un_nla.nla_options)));
555 1.1.1.2 christos
556 1.1.1.9 christos ND_PRINT("\n\t Connected Routers:");
557 1.1 christos ap = lsap->lsa_un.un_nla.nla_router;
558 1.1.1.2 christos while (lsa_length != 0) {
559 1.1.1.2 christos if (lsa_length < sizeof (*ap))
560 1.1.1.2 christos return (1);
561 1.1.1.2 christos lsa_length -= sizeof (*ap);
562 1.1.1.9 christos ND_PRINT("\n\t\t%s", GET_IPADDR_STRING(ap));
563 1.1 christos ++ap;
564 1.1 christos }
565 1.1 christos break;
566 1.1 christos
567 1.1 christos case LS_TYPE_INTER_AP | LS_SCOPE_AREA:
568 1.1.1.2 christos if (lsa_length < sizeof (lsap->lsa_un.un_inter_ap.inter_ap_metric))
569 1.1.1.2 christos return (1);
570 1.1.1.2 christos lsa_length -= sizeof (lsap->lsa_un.un_inter_ap.inter_ap_metric);
571 1.1.1.9 christos ND_PRINT(", metric %u",
572 1.1.1.9 christos GET_BE_U_4(lsap->lsa_un.un_inter_ap.inter_ap_metric) & SLA_MASK_METRIC);
573 1.1.1.2 christos
574 1.1.1.5 christos tptr = (const uint8_t *)lsap->lsa_un.un_inter_ap.inter_ap_prefix;
575 1.1.1.2 christos while (lsa_length != 0) {
576 1.1.1.4 christos bytelen = ospf6_print_lsaprefix(ndo, tptr, lsa_length);
577 1.1.1.2 christos if (bytelen < 0)
578 1.1 christos goto trunc;
579 1.1.1.9 christos /*
580 1.1.1.9 christos * ospf6_print_lsaprefix() will return -1 if
581 1.1.1.9 christos * the length is too high, so this will not
582 1.1.1.9 christos * underflow.
583 1.1.1.9 christos */
584 1.1.1.2 christos lsa_length -= bytelen;
585 1.1.1.2 christos tptr += bytelen;
586 1.1 christos }
587 1.1 christos break;
588 1.1.1.2 christos
589 1.1.1.2 christos case LS_TYPE_ASE | LS_SCOPE_AS:
590 1.1.1.2 christos if (lsa_length < sizeof (lsap->lsa_un.un_asla.asla_metric))
591 1.1.1.2 christos return (1);
592 1.1.1.2 christos lsa_length -= sizeof (lsap->lsa_un.un_asla.asla_metric);
593 1.1.1.9 christos flags32 = GET_BE_U_4(lsap->lsa_un.un_asla.asla_metric);
594 1.1.1.9 christos ND_PRINT("\n\t Flags [%s]",
595 1.1.1.9 christos bittok2str(ospf6_asla_flag_values, "none", flags32));
596 1.1.1.9 christos ND_PRINT(" metric %u",
597 1.1.1.9 christos GET_BE_U_4(lsap->lsa_un.un_asla.asla_metric) &
598 1.1.1.9 christos ASLA_MASK_METRIC);
599 1.1.1.2 christos
600 1.1.1.5 christos tptr = (const uint8_t *)lsap->lsa_un.un_asla.asla_prefix;
601 1.1.1.5 christos lsapp = (const struct lsa6_prefix *)tptr;
602 1.1.1.4 christos bytelen = ospf6_print_lsaprefix(ndo, tptr, lsa_length);
603 1.1 christos if (bytelen < 0)
604 1.1 christos goto trunc;
605 1.1.1.9 christos /*
606 1.1.1.9 christos * ospf6_print_lsaprefix() will return -1 if
607 1.1.1.9 christos * the length is too high, so this will not
608 1.1.1.9 christos * underflow.
609 1.1.1.9 christos */
610 1.1.1.2 christos lsa_length -= bytelen;
611 1.1.1.2 christos tptr += bytelen;
612 1.1 christos
613 1.1.1.2 christos if ((flags32 & ASLA_FLAG_FWDADDR) != 0) {
614 1.1.1.9 christos if (lsa_length < sizeof (nd_ipv6))
615 1.1.1.2 christos return (1);
616 1.1.1.9 christos lsa_length -= sizeof (nd_ipv6);
617 1.1.1.9 christos ND_PRINT(" forward %s",
618 1.1.1.9 christos GET_IP6ADDR_STRING(tptr));
619 1.1.1.9 christos tptr += sizeof(nd_ipv6);
620 1.1.1.2 christos }
621 1.1 christos
622 1.1.1.2 christos if ((flags32 & ASLA_FLAG_ROUTETAG) != 0) {
623 1.1.1.4 christos if (lsa_length < sizeof (uint32_t))
624 1.1.1.2 christos return (1);
625 1.1.1.4 christos lsa_length -= sizeof (uint32_t);
626 1.1.1.9 christos ND_PRINT(" tag %s",
627 1.1.1.9 christos GET_IPADDR_STRING(tptr));
628 1.1.1.4 christos tptr += sizeof(uint32_t);
629 1.1.1.2 christos }
630 1.1 christos
631 1.1.1.9 christos if (GET_U_1(lsapp->lsa_p_metric)) {
632 1.1.1.4 christos if (lsa_length < sizeof (uint32_t))
633 1.1.1.2 christos return (1);
634 1.1.1.4 christos lsa_length -= sizeof (uint32_t);
635 1.1.1.9 christos ND_PRINT(" RefLSID: %s",
636 1.1.1.9 christos GET_IPADDR_STRING(tptr));
637 1.1.1.4 christos tptr += sizeof(uint32_t);
638 1.1 christos }
639 1.1 christos break;
640 1.1 christos
641 1.1 christos case LS_TYPE_LINK:
642 1.1 christos /* Link LSA */
643 1.1 christos llsap = &lsap->lsa_un.un_llsa;
644 1.1.1.2 christos if (lsa_length < sizeof (llsap->llsa_priandopt))
645 1.1.1.2 christos return (1);
646 1.1.1.2 christos lsa_length -= sizeof (llsap->llsa_priandopt);
647 1.1.1.9 christos ND_TCHECK_SIZE(&llsap->llsa_priandopt);
648 1.1.1.9 christos ND_PRINT("\n\t Options [%s]",
649 1.1.1.4 christos bittok2str(ospf6_option_values, "none",
650 1.1.1.9 christos GET_BE_U_4(llsap->llsa_options)));
651 1.1.1.2 christos
652 1.1.1.2 christos if (lsa_length < sizeof (llsap->llsa_lladdr) + sizeof (llsap->llsa_nprefix))
653 1.1.1.2 christos return (1);
654 1.1.1.2 christos lsa_length -= sizeof (llsap->llsa_lladdr) + sizeof (llsap->llsa_nprefix);
655 1.1.1.9 christos prefixes = GET_BE_U_4(llsap->llsa_nprefix);
656 1.1.1.9 christos ND_PRINT("\n\t Priority %u, Link-local address %s, Prefixes %u:",
657 1.1.1.9 christos GET_U_1(llsap->llsa_priority),
658 1.1.1.9 christos GET_IP6ADDR_STRING(llsap->llsa_lladdr),
659 1.1.1.9 christos prefixes);
660 1.1 christos
661 1.1.1.5 christos tptr = (const uint8_t *)llsap->llsa_prefix;
662 1.1.1.2 christos while (prefixes > 0) {
663 1.1.1.4 christos bytelen = ospf6_print_lsaprefix(ndo, tptr, lsa_length);
664 1.1.1.2 christos if (bytelen < 0)
665 1.1.1.2 christos goto trunc;
666 1.1.1.2 christos prefixes--;
667 1.1.1.9 christos /*
668 1.1.1.9 christos * ospf6_print_lsaprefix() will return -1 if
669 1.1.1.9 christos * the length is too high, so this will not
670 1.1.1.9 christos * underflow.
671 1.1.1.9 christos */
672 1.1.1.2 christos lsa_length -= bytelen;
673 1.1.1.2 christos tptr += bytelen;
674 1.1.1.2 christos }
675 1.1 christos break;
676 1.1 christos
677 1.1 christos case LS_TYPE_INTRA_AP | LS_SCOPE_AREA:
678 1.1 christos /* Intra-Area-Prefix LSA */
679 1.1.1.2 christos if (lsa_length < sizeof (lsap->lsa_un.un_intra_ap.intra_ap_rtid))
680 1.1.1.2 christos return (1);
681 1.1.1.2 christos lsa_length -= sizeof (lsap->lsa_un.un_intra_ap.intra_ap_rtid);
682 1.1.1.9 christos ND_TCHECK_4(lsap->lsa_un.un_intra_ap.intra_ap_rtid);
683 1.1.1.4 christos ospf6_print_ls_type(ndo,
684 1.1.1.9 christos GET_BE_U_2(lsap->lsa_un.un_intra_ap.intra_ap_lstype),
685 1.1 christos &lsap->lsa_un.un_intra_ap.intra_ap_lsid);
686 1.1.1.2 christos
687 1.1.1.2 christos if (lsa_length < sizeof (lsap->lsa_un.un_intra_ap.intra_ap_nprefix))
688 1.1.1.2 christos return (1);
689 1.1.1.2 christos lsa_length -= sizeof (lsap->lsa_un.un_intra_ap.intra_ap_nprefix);
690 1.1.1.9 christos prefixes = GET_BE_U_2(lsap->lsa_un.un_intra_ap.intra_ap_nprefix);
691 1.1.1.9 christos ND_PRINT("\n\t Prefixes %u:", prefixes);
692 1.1 christos
693 1.1.1.5 christos tptr = (const uint8_t *)lsap->lsa_un.un_intra_ap.intra_ap_prefix;
694 1.1.1.2 christos while (prefixes > 0) {
695 1.1.1.4 christos bytelen = ospf6_print_lsaprefix(ndo, tptr, lsa_length);
696 1.1.1.2 christos if (bytelen < 0)
697 1.1.1.2 christos goto trunc;
698 1.1.1.2 christos prefixes--;
699 1.1.1.9 christos /*
700 1.1.1.9 christos * ospf6_print_lsaprefix() will return -1 if
701 1.1.1.9 christos * the length is too high, so this will not
702 1.1.1.9 christos * underflow.
703 1.1.1.9 christos */
704 1.1.1.2 christos lsa_length -= bytelen;
705 1.1.1.2 christos tptr += bytelen;
706 1.1.1.2 christos }
707 1.1 christos break;
708 1.1 christos
709 1.1 christos case LS_TYPE_GRACE | LS_SCOPE_LINKLOCAL:
710 1.1.1.9 christos if (ospf_grace_lsa_print(ndo, tptr, lsa_length) == -1) {
711 1.1 christos return 1;
712 1.1 christos }
713 1.1.1.2 christos break;
714 1.1 christos
715 1.1 christos case LS_TYPE_INTRA_ATE | LS_SCOPE_LINKLOCAL:
716 1.1.1.9 christos if (ospf_te_lsa_print(ndo, tptr, lsa_length) == -1) {
717 1.1.1.2 christos return 1;
718 1.1.1.2 christos }
719 1.1.1.2 christos break;
720 1.1 christos
721 1.1 christos default:
722 1.1.1.4 christos if(!print_unknown_data(ndo,tptr,
723 1.1.1.2 christos "\n\t ",
724 1.1.1.2 christos lsa_length)) {
725 1.1.1.2 christos return (1);
726 1.1.1.2 christos }
727 1.1.1.2 christos break;
728 1.1 christos }
729 1.1 christos
730 1.1 christos return (0);
731 1.1 christos trunc:
732 1.1 christos return (1);
733 1.1 christos }
734 1.1 christos
735 1.1 christos static int
736 1.1.1.4 christos ospf6_decode_v3(netdissect_options *ndo,
737 1.1.1.9 christos const struct ospf6hdr *op,
738 1.1.1.9 christos const u_char *dataend)
739 1.1 christos {
740 1.1.1.9 christos const rtrid_t *ap;
741 1.1.1.9 christos const struct lsr6 *lsrp;
742 1.1.1.9 christos const struct lsa6_hdr *lshp;
743 1.1.1.9 christos const struct lsa6 *lsap;
744 1.1.1.10 christos uint32_t i;
745 1.1 christos
746 1.1.1.9 christos switch (GET_U_1(op->ospf6_type)) {
747 1.1 christos
748 1.1.1.4 christos case OSPF_TYPE_HELLO: {
749 1.1.1.9 christos const struct hello6 *hellop = (const struct hello6 *)((const uint8_t *)op + OSPF6HDR_LEN);
750 1.1.1.4 christos
751 1.1.1.9 christos ND_PRINT("\n\tOptions [%s]",
752 1.1.1.4 christos bittok2str(ospf6_option_values, "none",
753 1.1.1.9 christos GET_BE_U_4(hellop->hello_options)));
754 1.1.1.4 christos
755 1.1.1.9 christos ND_PRINT("\n\t Hello Timer %us, Dead Timer %us, Interface-ID %s, Priority %u",
756 1.1.1.9 christos GET_BE_U_2(hellop->hello_helloint),
757 1.1.1.9 christos GET_BE_U_2(hellop->hello_deadint),
758 1.1.1.9 christos GET_IPADDR_STRING(hellop->hello_ifid),
759 1.1.1.9 christos GET_U_1(hellop->hello_priority));
760 1.1.1.9 christos
761 1.1.1.9 christos if (GET_BE_U_4(hellop->hello_dr) != 0)
762 1.1.1.9 christos ND_PRINT("\n\t Designated Router %s",
763 1.1.1.9 christos GET_IPADDR_STRING(hellop->hello_dr));
764 1.1.1.9 christos if (GET_BE_U_4(hellop->hello_bdr) != 0)
765 1.1.1.9 christos ND_PRINT(", Backup Designated Router %s",
766 1.1.1.9 christos GET_IPADDR_STRING(hellop->hello_bdr));
767 1.1.1.4 christos if (ndo->ndo_vflag > 1) {
768 1.1.1.9 christos ND_PRINT("\n\t Neighbor List:");
769 1.1.1.4 christos ap = hellop->hello_neighbor;
770 1.1.1.5 christos while ((const u_char *)ap < dataend) {
771 1.1.1.9 christos ND_PRINT("\n\t %s", GET_IPADDR_STRING(ap));
772 1.1 christos ++ap;
773 1.1 christos }
774 1.1 christos }
775 1.1 christos break; /* HELLO */
776 1.1.1.4 christos }
777 1.1 christos
778 1.1.1.4 christos case OSPF_TYPE_DD: {
779 1.1.1.9 christos const struct dd6 *ddp = (const struct dd6 *)((const uint8_t *)op + OSPF6HDR_LEN);
780 1.1.1.4 christos
781 1.1.1.9 christos ND_PRINT("\n\tOptions [%s]",
782 1.1.1.4 christos bittok2str(ospf6_option_values, "none",
783 1.1.1.9 christos GET_BE_U_4(ddp->db_options)));
784 1.1.1.9 christos ND_PRINT(", DD Flags [%s]",
785 1.1.1.9 christos bittok2str(ospf6_dd_flag_values,"none",GET_U_1(ddp->db_flags)));
786 1.1.1.9 christos
787 1.1.1.9 christos ND_PRINT(", MTU %u, DD-Sequence 0x%08x",
788 1.1.1.9 christos GET_BE_U_2(ddp->db_mtu),
789 1.1.1.9 christos GET_BE_U_4(ddp->db_seq));
790 1.1.1.4 christos if (ndo->ndo_vflag > 1) {
791 1.1.1.4 christos /* Print all the LS adv's */
792 1.1.1.4 christos lshp = ddp->db_lshdr;
793 1.1.1.5 christos while ((const u_char *)lshp < dataend) {
794 1.1.1.4 christos if (ospf6_print_lshdr(ndo, lshp++, dataend))
795 1.1.1.4 christos goto trunc;
796 1.1.1.4 christos }
797 1.1.1.4 christos }
798 1.1 christos break;
799 1.1.1.4 christos }
800 1.1 christos
801 1.1 christos case OSPF_TYPE_LS_REQ:
802 1.1.1.4 christos if (ndo->ndo_vflag > 1) {
803 1.1.1.5 christos lsrp = (const struct lsr6 *)((const uint8_t *)op + OSPF6HDR_LEN);
804 1.1.1.5 christos while ((const u_char *)lsrp < dataend) {
805 1.1.1.9 christos ND_TCHECK_SIZE(lsrp);
806 1.1.1.9 christos ND_PRINT("\n\t Advertising Router %s",
807 1.1.1.9 christos GET_IPADDR_STRING(lsrp->ls_router));
808 1.1.1.9 christos ospf6_print_ls_type(ndo,
809 1.1.1.9 christos GET_BE_U_2(lsrp->ls_type),
810 1.1 christos &lsrp->ls_stateid);
811 1.1 christos ++lsrp;
812 1.1 christos }
813 1.1 christos }
814 1.1 christos break;
815 1.1 christos
816 1.1 christos case OSPF_TYPE_LS_UPDATE:
817 1.1.1.4 christos if (ndo->ndo_vflag > 1) {
818 1.1.1.9 christos const struct lsu6 *lsup = (const struct lsu6 *)((const uint8_t *)op + OSPF6HDR_LEN);
819 1.1.1.4 christos
820 1.1.1.9 christos i = GET_BE_U_4(lsup->lsu_count);
821 1.1.1.4 christos lsap = lsup->lsu_lsa;
822 1.1.1.5 christos while ((const u_char *)lsap < dataend && i--) {
823 1.1.1.4 christos if (ospf6_print_lsa(ndo, lsap, dataend))
824 1.1 christos goto trunc;
825 1.1.1.5 christos lsap = (const struct lsa6 *)((const u_char *)lsap +
826 1.1.1.9 christos GET_BE_U_2(lsap->ls_hdr.ls_length));
827 1.1 christos }
828 1.1 christos }
829 1.1 christos break;
830 1.1 christos
831 1.1 christos case OSPF_TYPE_LS_ACK:
832 1.1.1.4 christos if (ndo->ndo_vflag > 1) {
833 1.1.1.5 christos lshp = (const struct lsa6_hdr *)((const uint8_t *)op + OSPF6HDR_LEN);
834 1.1.1.5 christos while ((const u_char *)lshp < dataend) {
835 1.1.1.4 christos if (ospf6_print_lshdr(ndo, lshp++, dataend))
836 1.1.1.4 christos goto trunc;
837 1.1 christos }
838 1.1 christos }
839 1.1 christos break;
840 1.1 christos
841 1.1 christos default:
842 1.1 christos break;
843 1.1 christos }
844 1.1 christos return (0);
845 1.1 christos trunc:
846 1.1 christos return (1);
847 1.1 christos }
848 1.1 christos
849 1.1.1.4 christos /* RFC5613 Section 2.2 (w/o the TLVs) */
850 1.1.1.4 christos static int
851 1.1.1.4 christos ospf6_print_lls(netdissect_options *ndo,
852 1.1.1.4 christos const u_char *cp, const u_int len)
853 1.1.1.4 christos {
854 1.1.1.4 christos uint16_t llsdatalen;
855 1.1.1.4 christos
856 1.1.1.4 christos if (len == 0)
857 1.1.1.4 christos return 0;
858 1.1.1.4 christos if (len < OSPF_LLS_HDRLEN)
859 1.1.1.4 christos goto trunc;
860 1.1.1.4 christos /* Checksum */
861 1.1.1.9 christos ND_PRINT("\n\tLLS Checksum 0x%04x", GET_BE_U_2(cp));
862 1.1.1.4 christos cp += 2;
863 1.1.1.4 christos /* LLS Data Length */
864 1.1.1.9 christos llsdatalen = GET_BE_U_2(cp);
865 1.1.1.9 christos ND_PRINT(", Data Length %u", llsdatalen);
866 1.1.1.4 christos if (llsdatalen < OSPF_LLS_HDRLEN || llsdatalen > len)
867 1.1.1.4 christos goto trunc;
868 1.1.1.4 christos cp += 2;
869 1.1.1.4 christos /* LLS TLVs */
870 1.1.1.9 christos ND_TCHECK_LEN(cp, llsdatalen - OSPF_LLS_HDRLEN);
871 1.1.1.4 christos /* FIXME: code in print-ospf.c can be reused to decode the TLVs */
872 1.1.1.4 christos
873 1.1.1.4 christos return llsdatalen;
874 1.1.1.4 christos trunc:
875 1.1.1.4 christos return -1;
876 1.1.1.4 christos }
877 1.1.1.4 christos
878 1.1.1.4 christos /* RFC6506 Section 4.1 */
879 1.1.1.4 christos static int
880 1.1.1.4 christos ospf6_decode_at(netdissect_options *ndo,
881 1.1.1.4 christos const u_char *cp, const u_int len)
882 1.1.1.4 christos {
883 1.1.1.4 christos uint16_t authdatalen;
884 1.1.1.4 christos
885 1.1.1.4 christos if (len == 0)
886 1.1.1.4 christos return 0;
887 1.1.1.4 christos if (len < OSPF6_AT_HDRLEN)
888 1.1.1.4 christos goto trunc;
889 1.1.1.4 christos /* Authentication Type */
890 1.1.1.9 christos ND_PRINT("\n\tAuthentication Type %s",
891 1.1.1.9 christos tok2str(ospf6_auth_type_str, "unknown (0x%04x)", GET_BE_U_2(cp)));
892 1.1.1.4 christos cp += 2;
893 1.1.1.4 christos /* Auth Data Len */
894 1.1.1.9 christos authdatalen = GET_BE_U_2(cp);
895 1.1.1.9 christos ND_PRINT(", Length %u", authdatalen);
896 1.1.1.4 christos if (authdatalen < OSPF6_AT_HDRLEN || authdatalen > len)
897 1.1.1.4 christos goto trunc;
898 1.1.1.4 christos cp += 2;
899 1.1.1.4 christos /* Reserved */
900 1.1.1.4 christos cp += 2;
901 1.1.1.4 christos /* Security Association ID */
902 1.1.1.9 christos ND_PRINT(", SAID %u", GET_BE_U_2(cp));
903 1.1.1.4 christos cp += 2;
904 1.1.1.4 christos /* Cryptographic Sequence Number (High-Order 32 Bits) */
905 1.1.1.9 christos ND_PRINT(", CSN 0x%08x", GET_BE_U_4(cp));
906 1.1.1.4 christos cp += 4;
907 1.1.1.4 christos /* Cryptographic Sequence Number (Low-Order 32 Bits) */
908 1.1.1.9 christos ND_PRINT(":%08x", GET_BE_U_4(cp));
909 1.1.1.4 christos cp += 4;
910 1.1.1.4 christos /* Authentication Data */
911 1.1.1.9 christos ND_TCHECK_LEN(cp, authdatalen - OSPF6_AT_HDRLEN);
912 1.1.1.4 christos if (ndo->ndo_vflag > 1)
913 1.1.1.4 christos print_unknown_data(ndo,cp, "\n\tAuthentication Data ", authdatalen - OSPF6_AT_HDRLEN);
914 1.1.1.4 christos return 0;
915 1.1.1.4 christos
916 1.1.1.4 christos trunc:
917 1.1.1.4 christos return 1;
918 1.1.1.4 christos }
919 1.1.1.4 christos
920 1.1.1.4 christos /* The trailing data may include LLS and/or AT data (in this specific order).
921 1.1.1.4 christos * LLS data may be present only in Hello and DBDesc packets with the L-bit set.
922 1.1.1.4 christos * AT data may be present in Hello and DBDesc packets with the AT-bit set or in
923 1.1.1.4 christos * any other packet type, thus decode the AT data regardless of the AT-bit.
924 1.1.1.4 christos */
925 1.1.1.4 christos static int
926 1.1.1.4 christos ospf6_decode_v3_trailer(netdissect_options *ndo,
927 1.1.1.4 christos const struct ospf6hdr *op, const u_char *cp, const unsigned len)
928 1.1.1.4 christos {
929 1.1.1.9 christos uint8_t type;
930 1.1.1.4 christos int llslen = 0;
931 1.1.1.4 christos int lls_hello = 0;
932 1.1.1.4 christos int lls_dd = 0;
933 1.1.1.4 christos
934 1.1.1.9 christos type = GET_U_1(op->ospf6_type);
935 1.1.1.9 christos if (type == OSPF_TYPE_HELLO) {
936 1.1.1.5 christos const struct hello6 *hellop = (const struct hello6 *)((const uint8_t *)op + OSPF6HDR_LEN);
937 1.1.1.9 christos if (GET_BE_U_4(hellop->hello_options) & OSPF6_OPTION_L)
938 1.1.1.4 christos lls_hello = 1;
939 1.1.1.9 christos } else if (type == OSPF_TYPE_DD) {
940 1.1.1.5 christos const struct dd6 *ddp = (const struct dd6 *)((const uint8_t *)op + OSPF6HDR_LEN);
941 1.1.1.9 christos if (GET_BE_U_4(ddp->db_options) & OSPF6_OPTION_L)
942 1.1.1.4 christos lls_dd = 1;
943 1.1.1.4 christos }
944 1.1.1.4 christos if ((lls_hello || lls_dd) && (llslen = ospf6_print_lls(ndo, cp, len)) < 0)
945 1.1.1.4 christos goto trunc;
946 1.1.1.4 christos return ospf6_decode_at(ndo, cp + llslen, len - llslen);
947 1.1.1.4 christos
948 1.1.1.4 christos trunc:
949 1.1.1.4 christos return 1;
950 1.1.1.4 christos }
951 1.1.1.4 christos
952 1.1 christos void
953 1.1.1.4 christos ospf6_print(netdissect_options *ndo,
954 1.1.1.9 christos const u_char *bp, u_int length)
955 1.1 christos {
956 1.1.1.9 christos const struct ospf6hdr *op;
957 1.1.1.9 christos const u_char *dataend;
958 1.1.1.9 christos const char *cp;
959 1.1.1.4 christos uint16_t datalen;
960 1.1 christos
961 1.1.1.9 christos ndo->ndo_protocol = "ospf3";
962 1.1.1.5 christos op = (const struct ospf6hdr *)bp;
963 1.1 christos
964 1.1 christos /* If the type is valid translate it, or just print the type */
965 1.1 christos /* value. If it's not valid, say so and return */
966 1.1.1.9 christos cp = tok2str(ospf6_type_values, "unknown packet type (%u)",
967 1.1.1.9 christos GET_U_1(op->ospf6_type));
968 1.1.1.9 christos ND_PRINT("OSPFv%u, %s, length %u", GET_U_1(op->ospf6_version), cp,
969 1.1.1.9 christos length);
970 1.1 christos if (*cp == 'u') {
971 1.1 christos return;
972 1.1.1.4 christos }
973 1.1 christos
974 1.1.1.4 christos if(!ndo->ndo_vflag) { /* non verbose - so lets bail out here */
975 1.1.1.4 christos return;
976 1.1.1.4 christos }
977 1.1 christos
978 1.1.1.4 christos /* OSPFv3 data always comes first and optional trailing data may follow. */
979 1.1.1.9 christos datalen = GET_BE_U_2(op->ospf6_len);
980 1.1.1.4 christos if (datalen > length) {
981 1.1.1.9 christos ND_PRINT(" [len %u]", datalen);
982 1.1 christos return;
983 1.1 christos }
984 1.1.1.4 christos dataend = bp + datalen;
985 1.1.1.4 christos
986 1.1.1.9 christos ND_PRINT("\n\tRouter-ID %s", GET_IPADDR_STRING(op->ospf6_routerid));
987 1.1 christos
988 1.1.1.9 christos if (GET_BE_U_4(op->ospf6_areaid) != 0)
989 1.1.1.9 christos ND_PRINT(", Area %s", GET_IPADDR_STRING(op->ospf6_areaid));
990 1.1 christos else
991 1.1.1.9 christos ND_PRINT(", Backbone Area");
992 1.1.1.9 christos if (GET_U_1(op->ospf6_instanceid))
993 1.1.1.9 christos ND_PRINT(", Instance %u", GET_U_1(op->ospf6_instanceid));
994 1.1 christos
995 1.1 christos /* Do rest according to version. */
996 1.1.1.9 christos switch (GET_U_1(op->ospf6_version)) {
997 1.1 christos
998 1.1 christos case 3:
999 1.1 christos /* ospf version 3 */
1000 1.1.1.4 christos if (ospf6_decode_v3(ndo, op, dataend) ||
1001 1.1.1.4 christos ospf6_decode_v3_trailer(ndo, op, dataend, length - datalen))
1002 1.1 christos goto trunc;
1003 1.1 christos break;
1004 1.1 christos } /* end switch on version */
1005 1.1 christos
1006 1.1 christos return;
1007 1.1 christos trunc:
1008 1.1.1.9 christos nd_print_trunc(ndo);
1009 1.1 christos }
1010