]>
The Tcpdump Group git mirrors - tcpdump/blob - print-ospf.c
4d24755691e7c3bcf584a334942d87e9be2726bf
2 * Copyright (c) 1992, 1993, 1994, 1995, 1996, 1997
3 * The Regents of the University of California. All rights reserved.
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that: (1) source code distributions
7 * retain the above copyright notice and this paragraph in its entirety, (2)
8 * distributions including binary code include the above copyright notice and
9 * this paragraph in its entirety in the documentation or other materials
10 * provided with the distribution, and (3) all advertising materials mentioning
11 * features or use of this software display the following acknowledgement:
12 * ``This product includes software developed by the University of California,
13 * Lawrence Berkeley Laboratory and its contributors.'' Neither the name of
14 * the University nor the names of its contributors may be used to endorse
15 * or promote products derived from this software without specific prior
17 * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED
18 * WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF
19 * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
21 * OSPF support contributed by Jeffrey Honig (jch@mitchell.cit.cornell.edu)
25 static const char rcsid
[] =
26 "@(#) $Header: /tcpdump/master/tcpdump/print-ospf.c,v 1.26 1999-10-30 05:11:19 itojun Exp $ (LBL)";
29 #include <sys/param.h>
31 #include <sys/socket.h>
33 #include <netinet/in.h>
34 #include <netinet/in_systm.h>
35 #include <netinet/ip.h>
36 #include <netinet/ip_var.h>
41 #include "interface.h"
42 #include "addrtoname.h"
51 static const struct bits ospf_option_bits
[] = {
52 { OSPF_OPTION_T
, "T" },
53 { OSPF_OPTION_E
, "E" },
54 { OSPF_OPTION_MC
, "MC" },
58 static const struct bits ospf_rla_flag_bits
[] = {
61 { RLA_FLAG_W1
, "W1" },
62 { RLA_FLAG_W2
, "W2" },
66 static struct tok type2str
[] = {
67 { OSPF_TYPE_UMD
, "umd" },
68 { OSPF_TYPE_HELLO
, "hello" },
69 { OSPF_TYPE_DB
, "dd" },
70 { OSPF_TYPE_LSR
, "ls_req" },
71 { OSPF_TYPE_LSU
, "ls_upd" },
72 { OSPF_TYPE_LSA
, "ls_ack" },
76 static char tstr
[] = " [|ospf]";
79 static inline void ospf_print_seqage(u_int32_t
, time_t);
80 static inline void ospf_print_bits(const struct bits
*, u_char
);
81 static void ospf_print_ls_type(u_int
, const struct in_addr
*,
82 const struct in_addr
*, const char *);
83 static int ospf_print_lshdr(const struct lsa_hdr
*);
84 static int ospf_print_lsa(const struct lsa
*);
85 static int ospf_decode_v2(const struct ospfhdr
*, const u_char
*);
88 ospf_print_seqage(register u_int32_t seq
, register time_t us
)
90 register time_t sec
= us
% 60;
91 register time_t mins
= (us
/ 60) % 60;
92 register time_t hour
= us
/ 3600;
94 printf(" S %X age ", seq
);
96 printf("%u:%02u:%02u",
97 (u_int32_t
) hour
, (u_int32_t
) mins
, (u_int32_t
) sec
);
99 printf("%u:%02u", (u_int32_t
) mins
, (u_int32_t
) sec
);
101 printf("%u", (u_int32_t
) sec
);
106 ospf_print_bits(register const struct bits
*bp
, register u_char options
)
108 register char sep
= ' ';
111 if (options
& bp
->bit
) {
112 printf("%c%s", sep
, bp
->str
);
115 } while ((++bp
)->bit
);
119 ospf_print_ls_type(register u_int ls_type
,
120 register const struct in_addr
*ls_stateid
,
121 register const struct in_addr
*ls_router
, register const char *fmt
)
127 printf(" rtr %s ", ipaddr_string(ls_router
));
130 case LS_TYPE_NETWORK
:
131 printf(" net dr %s if %s",
132 ipaddr_string(ls_router
),
133 ipaddr_string(ls_stateid
));
137 printf(" sum %s abr %s",
138 ipaddr_string(ls_stateid
),
139 ipaddr_string(ls_router
));
142 case LS_TYPE_SUM_ABR
:
143 printf(" abr %s rtr %s",
144 ipaddr_string(ls_router
),
145 ipaddr_string(ls_stateid
));
149 printf(" ase %s asbr %s",
150 ipaddr_string(ls_stateid
),
151 ipaddr_string(ls_router
));
155 printf(" group %s rtr %s",
156 ipaddr_string(ls_stateid
),
157 ipaddr_string(ls_router
));
162 printf(fmt
, ls_type
);
168 ospf_print_lshdr(register const struct lsa_hdr
*lshp
)
171 TCHECK(lshp
->ls_type
);
172 printf(" {"); /* } (ctags) */
174 TCHECK(lshp
->ls_options
);
175 ospf_print_bits(ospf_option_bits
, lshp
->ls_options
);
176 TCHECK(lshp
->ls_seq
);
177 ospf_print_seqage(ntohl(lshp
->ls_seq
), ntohs(lshp
->ls_age
));
178 ospf_print_ls_type(lshp
->ls_type
, &lshp
->ls_stateid
, &lshp
->ls_router
,
188 * Print a single link state advertisement. If truncated return 1, else 0.
191 ospf_print_lsa(register const struct lsa
*lsap
)
193 register const u_char
*ls_end
;
194 register const struct rlalink
*rlp
;
195 register const struct tos_metric
*tosp
;
196 register const struct in_addr
*ap
;
197 register const struct aslametric
*almp
;
198 register const struct mcla
*mcp
;
199 register const u_int32_t
*lp
;
202 if (ospf_print_lshdr(&lsap
->ls_hdr
))
204 TCHECK(lsap
->ls_hdr
.ls_length
);
205 ls_end
= (u_char
*)lsap
+ ntohs(lsap
->ls_hdr
.ls_length
);
206 switch (lsap
->ls_hdr
.ls_type
) {
209 TCHECK(lsap
->lsa_un
.un_rla
.rla_flags
);
210 ospf_print_bits(ospf_rla_flag_bits
,
211 lsap
->lsa_un
.un_rla
.rla_flags
);
213 TCHECK(lsap
->lsa_un
.un_rla
.rla_count
);
214 j
= ntohs(lsap
->lsa_un
.un_rla
.rla_count
);
215 TCHECK(lsap
->lsa_un
.un_rla
.rla_link
);
216 rlp
= lsap
->lsa_un
.un_rla
.rla_link
;
219 printf(" {"); /* } (ctags) */
220 switch (rlp
->link_type
) {
222 case RLA_TYPE_VIRTUAL
:
226 case RLA_TYPE_ROUTER
:
227 printf(" nbrid %s if %s",
228 ipaddr_string(&rlp
->link_id
),
229 ipaddr_string(&rlp
->link_data
));
232 case RLA_TYPE_TRANSIT
:
233 printf(" dr %s if %s",
234 ipaddr_string(&rlp
->link_id
),
235 ipaddr_string(&rlp
->link_data
));
239 printf(" net %s mask %s",
240 ipaddr_string(&rlp
->link_id
),
241 ipaddr_string(&rlp
->link_data
));
246 printf(" ??RouterLinksType %d?? }",
250 printf(" tos 0 metric %d", ntohs(rlp
->link_tos0metric
));
251 tosp
= (struct tos_metric
*)
252 ((sizeof rlp
->link_tos0metric
) + (u_char
*) rlp
);
253 for (k
= 0; k
< (int) rlp
->link_toscount
; ++k
, ++tosp
) {
255 printf(" tos %d metric %d",
257 ntohs(tosp
->tos_metric
));
261 rlp
= (struct rlalink
*)((u_char
*)(rlp
+ 1) +
262 ((rlp
->link_toscount
) * sizeof(*tosp
)));
266 case LS_TYPE_NETWORK
:
267 TCHECK(lsap
->lsa_un
.un_nla
.nla_mask
);
268 printf(" mask %s rtrs",
269 ipaddr_string(&lsap
->lsa_un
.un_nla
.nla_mask
));
270 ap
= lsap
->lsa_un
.un_nla
.nla_router
;
271 while ((u_char
*)ap
< ls_end
) {
273 printf(" %s", ipaddr_string(ap
));
279 TCHECK(lsap
->lsa_un
.un_nla
.nla_mask
);
281 ipaddr_string(&lsap
->lsa_un
.un_sla
.sla_mask
));
284 case LS_TYPE_SUM_ABR
:
285 TCHECK(lsap
->lsa_un
.un_sla
.sla_tosmetric
);
286 lp
= lsap
->lsa_un
.un_sla
.sla_tosmetric
;
287 while ((u_char
*)lp
< ls_end
) {
288 register u_int32_t ul
;
292 printf(" tos %d metric %d",
293 (ul
& SLA_MASK_TOS
) >> SLA_SHIFT_TOS
,
294 ul
& SLA_MASK_METRIC
);
300 TCHECK(lsap
->lsa_un
.un_nla
.nla_mask
);
302 ipaddr_string(&lsap
->lsa_un
.un_asla
.asla_mask
));
304 TCHECK(lsap
->lsa_un
.un_sla
.sla_tosmetric
);
305 almp
= lsap
->lsa_un
.un_asla
.asla_metric
;
306 while ((u_char
*)almp
< ls_end
) {
307 register u_int32_t ul
;
309 TCHECK(almp
->asla_tosmetric
);
310 ul
= ntohl(almp
->asla_tosmetric
);
311 printf(" type %d tos %d metric %d",
312 (ul
& ASLA_FLAG_EXTERNAL
) ? 2 : 1,
313 (ul
& ASLA_MASK_TOS
) >> ASLA_SHIFT_TOS
,
314 (ul
& ASLA_MASK_METRIC
));
315 TCHECK(almp
->asla_forward
);
316 if (almp
->asla_forward
.s_addr
) {
317 printf(" forward %s",
318 ipaddr_string(&almp
->asla_forward
));
320 TCHECK(almp
->asla_tag
);
321 if (almp
->asla_tag
.s_addr
) {
323 ipaddr_string(&almp
->asla_tag
));
330 /* Multicast extensions as of 23 July 1991 */
331 mcp
= lsap
->lsa_un
.un_mcla
;
332 while ((u_char
*)mcp
< ls_end
) {
333 TCHECK(mcp
->mcla_vid
);
334 switch (ntohl(mcp
->mcla_vtype
)) {
336 case MCLA_VERTEX_ROUTER
:
337 printf(" rtr rtrid %s",
338 ipaddr_string(&mcp
->mcla_vid
));
341 case MCLA_VERTEX_NETWORK
:
343 ipaddr_string(&mcp
->mcla_vid
));
347 printf(" ??VertexType %u??",
348 (u_int32_t
)ntohl(mcp
->mcla_vtype
));
364 ospf_decode_v2(register const struct ospfhdr
*op
,
365 register const u_char
*dataend
)
367 register const struct in_addr
*ap
;
368 register const struct lsr
*lsrp
;
369 register const struct lsa_hdr
*lshp
;
370 register const struct lsa
*lsap
;
374 switch (op
->ospf_type
) {
378 * Rob Coltun's special monitoring packets;
383 case OSPF_TYPE_HELLO
:
385 TCHECK(op
->ospf_hello
.hello_deadint
);
386 ospf_print_bits(ospf_option_bits
,
387 op
->ospf_hello
.hello_options
);
388 printf(" mask %s int %d pri %d dead %u",
389 ipaddr_string(&op
->ospf_hello
.hello_mask
),
390 ntohs(op
->ospf_hello
.hello_helloint
),
391 op
->ospf_hello
.hello_priority
,
392 (u_int32_t
)ntohl(op
->ospf_hello
.hello_deadint
));
394 TCHECK(op
->ospf_hello
.hello_dr
);
395 if (op
->ospf_hello
.hello_dr
.s_addr
!= 0)
397 ipaddr_string(&op
->ospf_hello
.hello_dr
));
398 TCHECK(op
->ospf_hello
.hello_bdr
);
399 if (op
->ospf_hello
.hello_bdr
.s_addr
!= 0)
401 ipaddr_string(&op
->ospf_hello
.hello_bdr
));
404 ap
= op
->ospf_hello
.hello_neighbor
;
405 while ((u_char
*)ap
< dataend
) {
407 printf(" %s", ipaddr_string(ap
));
414 TCHECK(op
->ospf_db
.db_options
);
415 ospf_print_bits(ospf_option_bits
, op
->ospf_db
.db_options
);
417 TCHECK(op
->ospf_db
.db_flags
);
418 if (op
->ospf_db
.db_flags
& OSPF_DB_INIT
) {
422 if (op
->ospf_db
.db_flags
& OSPF_DB_MORE
) {
426 if (op
->ospf_db
.db_flags
& OSPF_DB_MASTER
) {
430 TCHECK(op
->ospf_db
.db_seq
);
431 printf(" S %X", (u_int32_t
)ntohl(op
->ospf_db
.db_seq
));
434 /* Print all the LS adv's */
435 lshp
= op
->ospf_db
.db_lshdr
;
437 while (!ospf_print_lshdr(lshp
)) {
448 while ((u_char
*)lsrp
< dataend
) {
450 printf(" {"); /* } (ctags) */
451 ospf_print_ls_type(ntohl(lsrp
->ls_type
),
464 lsap
= op
->ospf_lsu
.lsu_lsa
;
465 TCHECK(op
->ospf_lsu
.lsu_count
);
466 i
= ntohl(op
->ospf_lsu
.lsu_count
);
468 if (ospf_print_lsa(lsap
))
470 lsap
= (struct lsa
*)((u_char
*)lsap
+
471 ntohs(lsap
->ls_hdr
.ls_length
));
479 lshp
= op
->ospf_lsa
.lsa_lshdr
;
481 while (!ospf_print_lshdr(lshp
)) {
490 printf("v2 type %d", op
->ospf_type
);
499 ospf_print(register const u_char
*bp
, register u_int length
,
500 register const u_char
*bp2
)
502 register const struct ospfhdr
*op
;
503 register const struct ip
*ip
;
504 register const u_char
*dataend
;
505 register const char *cp
;
507 op
= (struct ospfhdr
*)bp
;
508 ip
= (struct ip
*)bp2
;
509 /* Print the source and destination address */
511 (void) printf("%s > %s:",
512 ipaddr_string(&ip
->ip_src
),
513 ipaddr_string(&ip
->ip_dst
));
516 /* XXX Before we do anything else, strip off the MD5 trailer */
517 TCHECK(op
->ospf_authtype
);
518 if (ntohs(op
->ospf_authtype
) == OSPF_AUTH_MD5
) {
519 length
-= OSPF_AUTH_MD5_LEN
;
520 snapend
-= OSPF_AUTH_MD5_LEN
;
523 /* If the type is valid translate it, or just print the type */
524 /* value. If it's not valid, say so and return */
525 TCHECK(op
->ospf_type
);
526 cp
= tok2str(type2str
, "type%d", op
->ospf_type
);
527 printf(" OSPFv%d-%s %d:", op
->ospf_version
, cp
, length
);
531 TCHECK(op
->ospf_len
);
532 if (length
!= ntohs(op
->ospf_len
)) {
533 printf(" [len %d]", ntohs(op
->ospf_len
));
536 dataend
= bp
+ length
;
538 /* Print the routerid if it is not the same as the source */
539 TCHECK(op
->ospf_routerid
);
540 if (ip
->ip_src
.s_addr
!= op
->ospf_routerid
.s_addr
)
541 printf(" rtrid %s", ipaddr_string(&op
->ospf_routerid
));
543 TCHECK(op
->ospf_areaid
);
544 if (op
->ospf_areaid
.s_addr
!= 0)
545 printf(" area %s", ipaddr_string(&op
->ospf_areaid
));
550 /* Print authentication data (should we really do this?) */
551 TCHECK2(op
->ospf_authdata
[0], sizeof(op
->ospf_authdata
));
552 switch (ntohs(op
->ospf_authtype
)) {
557 case OSPF_AUTH_SIMPLE
:
559 (void)fn_printn(op
->ospf_authdata
,
560 sizeof(op
->ospf_authdata
), NULL
);
569 printf(" ??authtype-%d??", ntohs(op
->ospf_authtype
));
573 /* Do rest according to version. */
574 switch (op
->ospf_version
) {
578 if (ospf_decode_v2(op
, dataend
))
583 printf(" ospf [version %d]", op
->ospf_version
);
585 } /* end switch on version */