xref: /freebsd/contrib/tcpdump/print-ospf.c (revision 1b6c76a2fe091c74f08427e6c870851025a9cf67)
1 /*
2  * Copyright (c) 1992, 1993, 1994, 1995, 1996, 1997
3  *	The Regents of the University of California.  All rights reserved.
4  *
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
16  * written permission.
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.
20  *
21  * OSPF support contributed by Jeffrey Honig (jch@mitchell.cit.cornell.edu)
22  */
23 
24 #ifndef lint
25 static const char rcsid[] =
26     "@(#) $Header: /tcpdump/master/tcpdump/print-ospf.c,v 1.29 2000/09/29 04:58:45 guy Exp $ (LBL)";
27 #endif
28 
29 #ifdef HAVE_CONFIG_H
30 #include "config.h"
31 #endif
32 
33 #include <sys/param.h>
34 #include <sys/time.h>
35 #include <sys/socket.h>
36 
37 #include <netinet/in.h>
38 
39 #include <ctype.h>
40 #include <stdio.h>
41 
42 #include "interface.h"
43 #include "addrtoname.h"
44 
45 #include "ospf.h"
46 
47 #include "ip.h"
48 
49 struct bits {
50 	u_int32_t bit;
51 	const char *str;
52 };
53 
54 static const struct bits ospf_option_bits[] = {
55 	{ OSPF_OPTION_T,	"T" },
56 	{ OSPF_OPTION_E,	"E" },
57 	{ OSPF_OPTION_MC,	"MC" },
58 	{ 0,			NULL }
59 };
60 
61 static const struct bits ospf_rla_flag_bits[] = {
62 	{ RLA_FLAG_B,		"B" },
63 	{ RLA_FLAG_E,		"E" },
64 	{ RLA_FLAG_W1,		"W1" },
65 	{ RLA_FLAG_W2,		"W2" },
66 	{ 0,			NULL }
67 };
68 
69 static struct tok type2str[] = {
70 	{ OSPF_TYPE_UMD,	"umd" },
71 	{ OSPF_TYPE_HELLO,	"hello" },
72 	{ OSPF_TYPE_DB,		"dd" },
73 	{ OSPF_TYPE_LSR,	"ls_req" },
74 	{ OSPF_TYPE_LSU,	"ls_upd" },
75 	{ OSPF_TYPE_LSA,	"ls_ack" },
76 	{ 0,			NULL }
77 };
78 
79 static char tstr[] = " [|ospf]";
80 
81 /* Forwards */
82 static inline void ospf_print_seqage(u_int32_t, time_t);
83 static inline void ospf_print_bits(const struct bits *, u_char);
84 static void ospf_print_ls_type(u_int, const struct in_addr *,
85     const struct in_addr *, const char *);
86 static int ospf_print_lshdr(const struct lsa_hdr *);
87 static int ospf_print_lsa(const struct lsa *);
88 static int ospf_decode_v2(const struct ospfhdr *, const u_char *);
89 
90 static inline void
91 ospf_print_seqage(register u_int32_t seq, register time_t us)
92 {
93 	register time_t sec = us % 60;
94 	register time_t mins = (us / 60) % 60;
95 	register time_t hour = us / 3600;
96 
97 	printf(" S %X age ", seq);
98 	if (hour)
99 		printf("%u:%02u:%02u",
100 		    (u_int32_t) hour, (u_int32_t) mins, (u_int32_t) sec);
101 	else if (mins)
102 		printf("%u:%02u", (u_int32_t) mins, (u_int32_t) sec);
103 	else
104 		printf("%u", (u_int32_t) sec);
105 }
106 
107 
108 static inline void
109 ospf_print_bits(register const struct bits *bp, register u_char options)
110 {
111 	register char sep = ' ';
112 
113 	do {
114 		if (options & bp->bit) {
115 			printf("%c%s", sep, bp->str);
116 			sep = '/';
117 		}
118 	} while ((++bp)->bit);
119 }
120 
121 static void
122 ospf_print_ls_type(register u_int ls_type,
123     register const struct in_addr *ls_stateid,
124     register const struct in_addr *ls_router, register const char *fmt)
125 {
126 
127 	switch (ls_type) {
128 
129 	case LS_TYPE_ROUTER:
130 		printf(" rtr %s ", ipaddr_string(ls_router));
131 		break;
132 
133 	case LS_TYPE_NETWORK:
134 		printf(" net dr %s if %s",
135 		    ipaddr_string(ls_router),
136 		    ipaddr_string(ls_stateid));
137 		break;
138 
139 	case LS_TYPE_SUM_IP:
140 		printf(" sum %s abr %s",
141 		    ipaddr_string(ls_stateid),
142 		    ipaddr_string(ls_router));
143 		break;
144 
145 	case LS_TYPE_SUM_ABR:
146 		printf(" abr %s rtr %s",
147 		    ipaddr_string(ls_router),
148 		    ipaddr_string(ls_stateid));
149 		break;
150 
151 	case LS_TYPE_ASE:
152 		printf(" ase %s asbr %s",
153 		    ipaddr_string(ls_stateid),
154 		    ipaddr_string(ls_router));
155 		break;
156 
157 	case LS_TYPE_GROUP:
158 		printf(" group %s rtr %s",
159 		    ipaddr_string(ls_stateid),
160 		    ipaddr_string(ls_router));
161 		break;
162 
163 	default:
164 		putchar(' ');
165 		printf(fmt, ls_type);
166 		break;
167 	}
168 }
169 
170 static int
171 ospf_print_lshdr(register const struct lsa_hdr *lshp)
172 {
173 
174 	TCHECK(lshp->ls_type);
175 	printf(" {");						/* } (ctags) */
176 
177 	TCHECK(lshp->ls_options);
178 	ospf_print_bits(ospf_option_bits, lshp->ls_options);
179 	TCHECK(lshp->ls_seq);
180 	ospf_print_seqage(ntohl(lshp->ls_seq), ntohs(lshp->ls_age));
181 	ospf_print_ls_type(lshp->ls_type, &lshp->ls_stateid, &lshp->ls_router,
182 	    "ls_type %d");
183 
184 	return (0);
185 trunc:
186 	return (1);
187 }
188 
189 
190 /*
191  * Print a single link state advertisement.  If truncated return 1, else 0.
192  */
193 static int
194 ospf_print_lsa(register const struct lsa *lsap)
195 {
196 	register const u_char *ls_end;
197 	register const struct rlalink *rlp;
198 	register const struct tos_metric *tosp;
199 	register const struct in_addr *ap;
200 	register const struct aslametric *almp;
201 	register const struct mcla *mcp;
202 	register const u_int32_t *lp;
203 	register int j, k;
204 
205 	if (ospf_print_lshdr(&lsap->ls_hdr))
206 		return (1);
207 	TCHECK(lsap->ls_hdr.ls_length);
208 	ls_end = (u_char *)lsap + ntohs(lsap->ls_hdr.ls_length);
209 	switch (lsap->ls_hdr.ls_type) {
210 
211 	case LS_TYPE_ROUTER:
212 		TCHECK(lsap->lsa_un.un_rla.rla_flags);
213 		ospf_print_bits(ospf_rla_flag_bits,
214 		    lsap->lsa_un.un_rla.rla_flags);
215 
216 		TCHECK(lsap->lsa_un.un_rla.rla_count);
217 		j = ntohs(lsap->lsa_un.un_rla.rla_count);
218 		TCHECK(lsap->lsa_un.un_rla.rla_link);
219 		rlp = lsap->lsa_un.un_rla.rla_link;
220 		while (j--) {
221 			TCHECK(*rlp);
222 			printf(" {");				/* } (ctags) */
223 			switch (rlp->link_type) {
224 
225 			case RLA_TYPE_VIRTUAL:
226 				printf(" virt");
227 				/* Fall through */
228 
229 			case RLA_TYPE_ROUTER:
230 				printf(" nbrid %s if %s",
231 				    ipaddr_string(&rlp->link_id),
232 				    ipaddr_string(&rlp->link_data));
233 				break;
234 
235 			case RLA_TYPE_TRANSIT:
236 				printf(" dr %s if %s",
237 				    ipaddr_string(&rlp->link_id),
238 				    ipaddr_string(&rlp->link_data));
239 				break;
240 
241 			case RLA_TYPE_STUB:
242 				printf(" net %s mask %s",
243 				    ipaddr_string(&rlp->link_id),
244 				    ipaddr_string(&rlp->link_data));
245 				break;
246 
247 			default:
248 								/* { (ctags) */
249 				printf(" ??RouterLinksType %d?? }",
250 				    rlp->link_type);
251 				return (0);
252 			}
253 			printf(" tos 0 metric %d", ntohs(rlp->link_tos0metric));
254 			tosp = (struct tos_metric *)
255 			    ((sizeof rlp->link_tos0metric) + (u_char *) rlp);
256 			for (k = 0; k < (int) rlp->link_toscount; ++k, ++tosp) {
257 				TCHECK(*tosp);
258 				printf(" tos %d metric %d",
259 				    tosp->tos_type,
260 				    ntohs(tosp->tos_metric));
261 			}
262 								/* { (ctags) */
263 			printf(" }");
264 			rlp = (struct rlalink *)((u_char *)(rlp + 1) +
265 			    ((rlp->link_toscount) * sizeof(*tosp)));
266 		}
267 		break;
268 
269 	case LS_TYPE_NETWORK:
270 		TCHECK(lsap->lsa_un.un_nla.nla_mask);
271 		printf(" mask %s rtrs",
272 		    ipaddr_string(&lsap->lsa_un.un_nla.nla_mask));
273 		ap = lsap->lsa_un.un_nla.nla_router;
274 		while ((u_char *)ap < ls_end) {
275 			TCHECK(*ap);
276 			printf(" %s", ipaddr_string(ap));
277 			++ap;
278 		}
279 		break;
280 
281 	case LS_TYPE_SUM_IP:
282 		TCHECK(lsap->lsa_un.un_nla.nla_mask);
283 		printf(" mask %s",
284 		    ipaddr_string(&lsap->lsa_un.un_sla.sla_mask));
285 		/* Fall through */
286 
287 	case LS_TYPE_SUM_ABR:
288 		TCHECK(lsap->lsa_un.un_sla.sla_tosmetric);
289 		lp = lsap->lsa_un.un_sla.sla_tosmetric;
290 		while ((u_char *)lp < ls_end) {
291 			register u_int32_t ul;
292 
293 			TCHECK(*lp);
294 			ul = ntohl(*lp);
295 			printf(" tos %d metric %d",
296 			    (ul & SLA_MASK_TOS) >> SLA_SHIFT_TOS,
297 			    ul & SLA_MASK_METRIC);
298 			++lp;
299 		}
300 		break;
301 
302 	case LS_TYPE_ASE:
303 		TCHECK(lsap->lsa_un.un_nla.nla_mask);
304 		printf(" mask %s",
305 		    ipaddr_string(&lsap->lsa_un.un_asla.asla_mask));
306 
307 		TCHECK(lsap->lsa_un.un_sla.sla_tosmetric);
308 		almp = lsap->lsa_un.un_asla.asla_metric;
309 		while ((u_char *)almp < ls_end) {
310 			register u_int32_t ul;
311 
312 			TCHECK(almp->asla_tosmetric);
313 			ul = ntohl(almp->asla_tosmetric);
314 			printf(" type %d tos %d metric %d",
315 			    (ul & ASLA_FLAG_EXTERNAL) ? 2 : 1,
316 			    (ul & ASLA_MASK_TOS) >> ASLA_SHIFT_TOS,
317 			    (ul & ASLA_MASK_METRIC));
318 			TCHECK(almp->asla_forward);
319 			if (almp->asla_forward.s_addr) {
320 				printf(" forward %s",
321 				    ipaddr_string(&almp->asla_forward));
322 			}
323 			TCHECK(almp->asla_tag);
324 			if (almp->asla_tag.s_addr) {
325 				printf(" tag %s",
326 				    ipaddr_string(&almp->asla_tag));
327 			}
328 			++almp;
329 		}
330 		break;
331 
332 	case LS_TYPE_GROUP:
333 		/* Multicast extensions as of 23 July 1991 */
334 		mcp = lsap->lsa_un.un_mcla;
335 		while ((u_char *)mcp < ls_end) {
336 			TCHECK(mcp->mcla_vid);
337 			switch (ntohl(mcp->mcla_vtype)) {
338 
339 			case MCLA_VERTEX_ROUTER:
340 				printf(" rtr rtrid %s",
341 				    ipaddr_string(&mcp->mcla_vid));
342 				break;
343 
344 			case MCLA_VERTEX_NETWORK:
345 				printf(" net dr %s",
346 				    ipaddr_string(&mcp->mcla_vid));
347 				break;
348 
349 			default:
350 				printf(" ??VertexType %u??",
351 				    (u_int32_t)ntohl(mcp->mcla_vtype));
352 				break;
353 			}
354 		++mcp;
355 		}
356 	}
357 
358 								/* { (ctags) */
359 	fputs(" }", stdout);
360 	return (0);
361 trunc:
362 	fputs(" }", stdout);
363 	return (1);
364 }
365 
366 static int
367 ospf_decode_v2(register const struct ospfhdr *op,
368     register const u_char *dataend)
369 {
370 	register const struct in_addr *ap;
371 	register const struct lsr *lsrp;
372 	register const struct lsa_hdr *lshp;
373 	register const struct lsa *lsap;
374 	register char sep;
375 	register int i;
376 
377 	switch (op->ospf_type) {
378 
379 	case OSPF_TYPE_UMD:
380 		/*
381 		 * Rob Coltun's special monitoring packets;
382 		 * do nothing
383 		 */
384 		break;
385 
386 	case OSPF_TYPE_HELLO:
387 		if (vflag) {
388 			TCHECK(op->ospf_hello.hello_deadint);
389 			ospf_print_bits(ospf_option_bits,
390 			    op->ospf_hello.hello_options);
391 			printf(" mask %s int %d pri %d dead %u",
392 			    ipaddr_string(&op->ospf_hello.hello_mask),
393 			    ntohs(op->ospf_hello.hello_helloint),
394 			    op->ospf_hello.hello_priority,
395 			    (u_int32_t)ntohl(op->ospf_hello.hello_deadint));
396 		}
397 		TCHECK(op->ospf_hello.hello_dr);
398 		if (op->ospf_hello.hello_dr.s_addr != 0)
399 			printf(" dr %s",
400 			    ipaddr_string(&op->ospf_hello.hello_dr));
401 		TCHECK(op->ospf_hello.hello_bdr);
402 		if (op->ospf_hello.hello_bdr.s_addr != 0)
403 			printf(" bdr %s",
404 			    ipaddr_string(&op->ospf_hello.hello_bdr));
405 		if (vflag) {
406 			printf(" nbrs");
407 			ap = op->ospf_hello.hello_neighbor;
408 			while ((u_char *)ap < dataend) {
409 				TCHECK(*ap);
410 				printf(" %s", ipaddr_string(ap));
411 				++ap;
412 			}
413 		}
414 		break;	/* HELLO */
415 
416 	case OSPF_TYPE_DB:
417 		TCHECK(op->ospf_db.db_options);
418 		ospf_print_bits(ospf_option_bits, op->ospf_db.db_options);
419 		sep = ' ';
420 		TCHECK(op->ospf_db.db_flags);
421 		if (op->ospf_db.db_flags & OSPF_DB_INIT) {
422 			printf("%cI", sep);
423 			sep = '/';
424 		}
425 		if (op->ospf_db.db_flags & OSPF_DB_MORE) {
426 			printf("%cM", sep);
427 			sep = '/';
428 		}
429 		if (op->ospf_db.db_flags & OSPF_DB_MASTER) {
430 			printf("%cMS", sep);
431 			sep = '/';
432 		}
433 		TCHECK(op->ospf_db.db_seq);
434 		printf(" S %X", (u_int32_t)ntohl(op->ospf_db.db_seq));
435 
436 		if (vflag) {
437 			/* Print all the LS adv's */
438 			lshp = op->ospf_db.db_lshdr;
439 
440 			while (!ospf_print_lshdr(lshp)) {
441 							/* { (ctags) */
442 				printf(" }");
443 				++lshp;
444 			}
445 		}
446 		break;
447 
448 	case OSPF_TYPE_LSR:
449 		if (vflag) {
450 			lsrp = op->ospf_lsr;
451 			while ((u_char *)lsrp < dataend) {
452 				TCHECK(*lsrp);
453 				printf(" {");		/* } (ctags) */
454 				ospf_print_ls_type(ntohl(lsrp->ls_type),
455 				    &lsrp->ls_stateid,
456 				    &lsrp->ls_router,
457 				    "LinkStateType %d");
458 							/* { (ctags) */
459 				printf(" }");
460 				++lsrp;
461 			}
462 		}
463 		break;
464 
465 	case OSPF_TYPE_LSU:
466 		if (vflag) {
467 			lsap = op->ospf_lsu.lsu_lsa;
468 			TCHECK(op->ospf_lsu.lsu_count);
469 			i = ntohl(op->ospf_lsu.lsu_count);
470 			while (i--) {
471 				if (ospf_print_lsa(lsap))
472 					goto trunc;
473 				lsap = (struct lsa *)((u_char *)lsap +
474 				    ntohs(lsap->ls_hdr.ls_length));
475 			}
476 		}
477 		break;
478 
479 
480 	case OSPF_TYPE_LSA:
481 		if (vflag) {
482 			lshp = op->ospf_lsa.lsa_lshdr;
483 
484 			while (!ospf_print_lshdr(lshp)) {
485 							/* { (ctags) */
486 				printf(" }");
487 				++lshp;
488 			}
489 		}
490 		break;
491 
492 	default:
493 		printf("v2 type %d", op->ospf_type);
494 		break;
495 	}
496 	return (0);
497 trunc:
498 	return (1);
499 }
500 
501 void
502 ospf_print(register const u_char *bp, register u_int length,
503     register const u_char *bp2)
504 {
505 	register const struct ospfhdr *op;
506 	register const struct ip *ip;
507 	register const u_char *dataend;
508 	register const char *cp;
509 
510 	op = (struct ospfhdr *)bp;
511 	ip = (struct ip *)bp2;
512 	/* Print the source and destination address  */
513 #if 0
514 	(void) printf("%s > %s:",
515 	    ipaddr_string(&ip->ip_src),
516 	    ipaddr_string(&ip->ip_dst));
517 #endif
518 
519         /* XXX Before we do anything else, strip off the MD5 trailer */
520         TCHECK(op->ospf_authtype);
521         if (ntohs(op->ospf_authtype) == OSPF_AUTH_MD5) {
522                 length -= OSPF_AUTH_MD5_LEN;
523                 snapend -= OSPF_AUTH_MD5_LEN;
524         }
525 
526 	/* If the type is valid translate it, or just print the type */
527 	/* value.  If it's not valid, say so and return */
528 	TCHECK(op->ospf_type);
529 	cp = tok2str(type2str, "type%d", op->ospf_type);
530 	printf(" OSPFv%d-%s %d:", op->ospf_version, cp, length);
531 	if (*cp == 't')
532 		return;
533 
534 	TCHECK(op->ospf_len);
535 	if (length != ntohs(op->ospf_len)) {
536 		printf(" [len %d]", ntohs(op->ospf_len));
537 		return;
538 	}
539 	dataend = bp + length;
540 
541 	/* Print the routerid if it is not the same as the source */
542 	TCHECK(op->ospf_routerid);
543 	if (ip->ip_src.s_addr != op->ospf_routerid.s_addr)
544 		printf(" rtrid %s", ipaddr_string(&op->ospf_routerid));
545 
546 	TCHECK(op->ospf_areaid);
547 	if (op->ospf_areaid.s_addr != 0)
548 		printf(" area %s", ipaddr_string(&op->ospf_areaid));
549 	else
550 		printf(" backbone");
551 
552 	if (vflag) {
553 		/* Print authentication data (should we really do this?) */
554 		TCHECK2(op->ospf_authdata[0], sizeof(op->ospf_authdata));
555 		switch (ntohs(op->ospf_authtype)) {
556 
557 		case OSPF_AUTH_NONE:
558 			break;
559 
560 		case OSPF_AUTH_SIMPLE:
561 			printf(" auth \"");
562 			(void)fn_printn(op->ospf_authdata,
563 			    sizeof(op->ospf_authdata), NULL);
564 			printf("\"");
565 			break;
566 
567 		case OSPF_AUTH_MD5:
568 			printf(" auth MD5");
569 			break;
570 
571 		default:
572 			printf(" ??authtype-%d??", ntohs(op->ospf_authtype));
573 			return;
574 		}
575 	}
576 	/* Do rest according to version.	 */
577 	switch (op->ospf_version) {
578 
579 	case 2:
580 		/* ospf version 2 */
581 		if (ospf_decode_v2(op, dataend))
582 			goto trunc;
583 		break;
584 
585 	default:
586 		printf(" ospf [version %d]", op->ospf_version);
587 		break;
588 	}			/* end switch on version */
589 
590 	return;
591 trunc:
592 	fputs(tstr, stdout);
593 }
594