]>
git.saurik.com Git - apple/network_cmds.git/blob - tcpdump.tproj/print-ospf.c
5012fb8f8299f69fa2f4fb92d2b7715928fe9146
   2  * Copyright (c) 1999 Apple Computer, Inc. All rights reserved. 
   4  * @APPLE_LICENSE_HEADER_START@ 
   6  * "Portions Copyright (c) 1999 Apple Computer, Inc.  All Rights 
   7  * Reserved.  This file contains Original Code and/or Modifications of 
   8  * Original Code as defined in and that are subject to the Apple Public 
   9  * Source License Version 1.0 (the 'License').  You may not use this file 
  10  * except in compliance with the License.  Please obtain a copy of the 
  11  * License at http://www.apple.com/publicsource and read it before using 
  14  * The Original Code and all software distributed under the License are 
  15  * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER 
  16  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES, 
  17  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY, 
  18  * FITNESS FOR A PARTICULAR PURPOSE OR NON-INFRINGEMENT.  Please see the 
  19  * License for the specific language governing rights and limitations 
  22  * @APPLE_LICENSE_HEADER_END@ 
  25  * Copyright (c) 1992, 1993, 1994, 1995, 1996 
  26  *      The Regents of the University of California.  All rights reserved. 
  28  * Redistribution and use in source and binary forms, with or without 
  29  * modification, are permitted provided that: (1) source code distributions 
  30  * retain the above copyright notice and this paragraph in its entirety, (2) 
  31  * distributions including binary code include the above copyright notice and 
  32  * this paragraph in its entirety in the documentation or other materials 
  33  * provided with the distribution, and (3) all advertising materials mentioning 
  34  * features or use of this software display the following acknowledgement: 
  35  * ``This product includes software developed by the University of California, 
  36  * Lawrence Berkeley Laboratory and its contributors.'' Neither the name of 
  37  * the University nor the names of its contributors may be used to endorse 
  38  * or promote products derived from this software without specific prior 
  40  * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED 
  41  * WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF 
  42  * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. 
  44  * OSPF support contributed by Jeffrey Honig (jch@mitchell.cit.cornell.edu) 
  48 static const char rcsid
[] = 
  49     "@(#) $Header: /cvs/Darwin/Commands/NeXT/network_cmds/tcpdump.tproj/print-ospf.c,v 1.1.1.1 1999/05/02 03:58:34 wsanchez Exp $ (LBL)"; 
  52 #include <sys/param.h> 
  54 #include <sys/socket.h> 
  56 #include <netinet/in.h> 
  57 #include <netinet/in_systm.h> 
  58 #include <netinet/ip.h> 
  59 #include <netinet/ip_var.h> 
  64 #include "interface.h" 
  65 #include "addrtoname.h" 
  74 static const struct bits ospf_option_bits
[] = { 
  75         { OSPF_OPTION_T
,        "T" }, 
  76         { OSPF_OPTION_E
,        "E" }, 
  77         { OSPF_OPTION_MC
,       "MC" }, 
  81 static const struct bits ospf_rla_flag_bits
[] = { 
  84         { RLA_FLAG_W1
,          "W1" }, 
  85         { RLA_FLAG_W2
,          "W2" }, 
  89 static struct tok type2str
[] = { 
  90         { OSPF_TYPE_UMD
,        "umd" }, 
  91         { OSPF_TYPE_HELLO
,      "hello" }, 
  92         { OSPF_TYPE_DB
,         "dd" }, 
  93         { OSPF_TYPE_LSR
,        "ls_req" }, 
  94         { OSPF_TYPE_LSU
,        "ls_upd" }, 
  95         { OSPF_TYPE_LSA
,        "ls_ack" }, 
  99 static char tstr
[] = " [|ospf]"; 
 102 static inline void ospf_print_seqage(u_int32_t
, time_t); 
 103 static inline void ospf_print_bits(const struct bits 
*, u_char
); 
 104 static void ospf_print_ls_type(u_int
, const struct in_addr 
*, 
 105     const struct in_addr 
*, const char *); 
 106 static int ospf_print_lshdr(const struct lsa_hdr 
*); 
 107 static int ospf_print_lsa(const struct lsa 
*); 
 108 static int ospf_decode_v2(const struct ospfhdr 
*, const u_char 
*); 
 111 ospf_print_seqage(register u_int32_t seq
, register time_t us
) 
 113         register time_t sec 
= us 
% 60; 
 114         register time_t mins 
= (us 
/ 60) % 60; 
 115         register time_t hour 
= us 
/ 3600; 
 117         printf(" S %X age ", seq
); 
 119                 printf("%u:%02u:%02u", 
 120                     (u_int32_t
) hour
, (u_int32_t
) mins
, (u_int32_t
) sec
); 
 122                 printf("%u:%02u", (u_int32_t
) mins
, (u_int32_t
) sec
); 
 124                 printf("%u", (u_int32_t
) sec
); 
 129 ospf_print_bits(register const struct bits 
*bp
, register u_char options
) 
 131         register char sep 
= ' '; 
 134                 if (options 
& bp
->bit
) { 
 135                         printf("%c%s", sep
, bp
->str
); 
 138         } while ((++bp
)->bit
); 
 142 ospf_print_ls_type(register u_int ls_type
, 
 143     register const struct in_addr 
*ls_stateid
, 
 144     register const struct in_addr 
*ls_router
, register const char *fmt
) 
 150                 printf(" rtr %s ", ipaddr_string(ls_router
)); 
 153         case LS_TYPE_NETWORK
: 
 154                 printf(" net dr %s if %s", 
 155                     ipaddr_string(ls_router
), 
 156                     ipaddr_string(ls_stateid
)); 
 160                 printf(" sum %s abr %s", 
 161                     ipaddr_string(ls_stateid
), 
 162                     ipaddr_string(ls_router
)); 
 165         case LS_TYPE_SUM_ABR
: 
 166                 printf(" abr %s rtr %s", 
 167                     ipaddr_string(ls_router
), 
 168                     ipaddr_string(ls_stateid
)); 
 172                 printf(" ase %s asbr %s", 
 173                     ipaddr_string(ls_stateid
), 
 174                     ipaddr_string(ls_router
)); 
 178                 printf(" group %s rtr %s", 
 179                     ipaddr_string(ls_stateid
), 
 180                     ipaddr_string(ls_router
)); 
 185                 printf(fmt
, ls_type
); 
 191 ospf_print_lshdr(register const struct lsa_hdr 
*lshp
) 
 194         TCHECK(lshp
->ls_type
); 
 195         printf(" {");                                           /* } (ctags) */ 
 197         TCHECK(lshp
->ls_options
); 
 198         ospf_print_bits(ospf_option_bits
, lshp
->ls_options
); 
 199         TCHECK(lshp
->ls_seq
); 
 200         ospf_print_seqage(ntohl(lshp
->ls_seq
), ntohs(lshp
->ls_age
)); 
 201         ospf_print_ls_type(lshp
->ls_type
, &lshp
->ls_stateid
, &lshp
->ls_router
, 
 211  * Print a single link state advertisement.  If truncated return 1, else 0. 
 214 ospf_print_lsa(register const struct lsa 
*lsap
) 
 216         register const u_char 
*ls_end
; 
 217         register const struct rlalink 
*rlp
; 
 218         register const struct tos_metric 
*tosp
; 
 219         register const struct in_addr 
*ap
; 
 220         register const struct aslametric 
*almp
; 
 221         register const struct mcla 
*mcp
; 
 222         register const u_int32_t 
*lp
; 
 225         if (ospf_print_lshdr(&lsap
->ls_hdr
)) 
 227         TCHECK(lsap
->ls_hdr
.ls_length
); 
 228         ls_end 
= (u_char 
*)lsap 
+ ntohs(lsap
->ls_hdr
.ls_length
); 
 229         switch (lsap
->ls_hdr
.ls_type
) { 
 232                 TCHECK(lsap
->lsa_un
.un_rla
.rla_flags
); 
 233                 ospf_print_bits(ospf_rla_flag_bits
, 
 234                     lsap
->lsa_un
.un_rla
.rla_flags
); 
 236                 TCHECK(lsap
->lsa_un
.un_rla
.rla_count
); 
 237                 j 
= ntohs(lsap
->lsa_un
.un_rla
.rla_count
); 
 238                 TCHECK(lsap
->lsa_un
.un_rla
.rla_link
); 
 239                 rlp 
= lsap
->lsa_un
.un_rla
.rla_link
; 
 241                         register struct rlalink 
*rln 
= 
 242                             (struct rlalink 
*)((u_char 
*)(rlp 
+ 1) + 
 243                             ((rlp
->link_toscount
) * sizeof(*tosp
))); 
 246                         printf(" {");                           /* } (ctags) */ 
 247                         switch (rlp
->link_type
) { 
 249                         case RLA_TYPE_VIRTUAL
: 
 253                         case RLA_TYPE_ROUTER
: 
 254                                 printf(" nbrid %s if %s", 
 255                                     ipaddr_string(&rlp
->link_id
), 
 256                                     ipaddr_string(&rlp
->link_data
)); 
 259                         case RLA_TYPE_TRANSIT
: 
 260                                 printf(" dr %s if %s", 
 261                                     ipaddr_string(&rlp
->link_id
), 
 262                                     ipaddr_string(&rlp
->link_data
)); 
 266                                 printf(" net %s mask %s", 
 267                                     ipaddr_string(&rlp
->link_id
), 
 268                                     ipaddr_string(&rlp
->link_data
)); 
 273                                 printf(" ??RouterLinksType %d?? }", 
 277                         printf(" tos 0 metric %d", ntohs(rlp
->link_tos0metric
)); 
 278                         tosp 
= (struct tos_metric 
*) 
 279                             ((sizeof rlp
->link_tos0metric
) + (u_char 
*) rlp
); 
 280                         for (k 
= 0; k 
< (int) rlp
->link_toscount
; ++k
, ++tosp
) { 
 282                                 printf(" tos %d metric %d", 
 284                                     ntohs(tosp
->tos_metric
)); 
 292         case LS_TYPE_NETWORK
: 
 293                 TCHECK(lsap
->lsa_un
.un_nla
.nla_mask
); 
 294                 printf(" mask %s rtrs", 
 295                     ipaddr_string(&lsap
->lsa_un
.un_nla
.nla_mask
)); 
 296                 ap 
= lsap
->lsa_un
.un_nla
.nla_router
; 
 297                 while ((u_char 
*)ap 
< ls_end
) { 
 299                         printf(" %s", ipaddr_string(ap
)); 
 305                 TCHECK(lsap
->lsa_un
.un_nla
.nla_mask
); 
 307                     ipaddr_string(&lsap
->lsa_un
.un_sla
.sla_mask
)); 
 310         case LS_TYPE_SUM_ABR
: 
 311                 TCHECK(lsap
->lsa_un
.un_sla
.sla_tosmetric
); 
 312                 lp 
= lsap
->lsa_un
.un_sla
.sla_tosmetric
; 
 313                 while ((u_char 
*)lp 
< ls_end
) { 
 314                         register u_int32_t ul
; 
 318                         printf(" tos %d metric %d", 
 319                             (ul 
& SLA_MASK_TOS
) >> SLA_SHIFT_TOS
, 
 320                             ul 
& SLA_MASK_METRIC
); 
 326                 TCHECK(lsap
->lsa_un
.un_nla
.nla_mask
); 
 328                     ipaddr_string(&lsap
->lsa_un
.un_asla
.asla_mask
)); 
 330                 TCHECK(lsap
->lsa_un
.un_sla
.sla_tosmetric
); 
 331                 almp 
= lsap
->lsa_un
.un_asla
.asla_metric
; 
 332                 while ((u_char 
*)almp 
< ls_end
) { 
 333                         register u_int32_t ul
; 
 335                         TCHECK(almp
->asla_tosmetric
); 
 336                         ul 
= ntohl(almp
->asla_tosmetric
); 
 337                         printf(" type %d tos %d metric %d", 
 338                             (ul 
& ASLA_FLAG_EXTERNAL
) ? 2 : 1, 
 339                             (ul 
& ASLA_MASK_TOS
) >> ASLA_SHIFT_TOS
, 
 340                             (ul 
& ASLA_MASK_METRIC
)); 
 341                         TCHECK(almp
->asla_forward
); 
 342                         if (almp
->asla_forward
.s_addr
) { 
 343                                 printf(" forward %s", 
 344                                     ipaddr_string(&almp
->asla_forward
)); 
 346                         TCHECK(almp
->asla_tag
); 
 347                         if (almp
->asla_tag
.s_addr
) { 
 349                                     ipaddr_string(&almp
->asla_tag
)); 
 356                 /* Multicast extensions as of 23 July 1991 */ 
 357                 mcp 
= lsap
->lsa_un
.un_mcla
; 
 358                 while ((u_char 
*)mcp 
< ls_end
) { 
 359                         TCHECK(mcp
->mcla_vid
); 
 360                         switch (ntohl(mcp
->mcla_vtype
)) { 
 362                         case MCLA_VERTEX_ROUTER
: 
 363                                 printf(" rtr rtrid %s", 
 364                                     ipaddr_string(&mcp
->mcla_vid
)); 
 367                         case MCLA_VERTEX_NETWORK
: 
 369                                     ipaddr_string(&mcp
->mcla_vid
)); 
 373                                 printf(" ??VertexType %u??", 
 374                                     (u_int32_t
)ntohl(mcp
->mcla_vtype
)); 
 390 ospf_decode_v2(register const struct ospfhdr 
*op
, 
 391     register const u_char 
*dataend
) 
 393         register const struct in_addr 
*ap
; 
 394         register const struct lsr 
*lsrp
; 
 395         register const struct lsa_hdr 
*lshp
; 
 396         register const struct lsa 
*lsap
; 
 400         switch (op
->ospf_type
) { 
 404                  * Rob Coltun's special monitoring packets; 
 409         case OSPF_TYPE_HELLO
: 
 411                         TCHECK(op
->ospf_hello
.hello_deadint
); 
 412                         ospf_print_bits(ospf_option_bits
, 
 413                             op
->ospf_hello
.hello_options
); 
 414                         printf(" mask %s int %d pri %d dead %u", 
 415                             ipaddr_string(&op
->ospf_hello
.hello_mask
), 
 416                             ntohs(op
->ospf_hello
.hello_helloint
), 
 417                             op
->ospf_hello
.hello_priority
, 
 418                             (u_int32_t
)ntohl(op
->ospf_hello
.hello_deadint
)); 
 420                 TCHECK(op
->ospf_hello
.hello_dr
); 
 421                 if (op
->ospf_hello
.hello_dr
.s_addr 
!= 0) 
 423                             ipaddr_string(&op
->ospf_hello
.hello_dr
)); 
 424                 TCHECK(op
->ospf_hello
.hello_bdr
); 
 425                 if (op
->ospf_hello
.hello_bdr
.s_addr 
!= 0) 
 427                             ipaddr_string(&op
->ospf_hello
.hello_bdr
)); 
 430                         ap 
= op
->ospf_hello
.hello_neighbor
; 
 431                         while ((u_char 
*)ap 
< dataend
) { 
 433                                 printf(" %s", ipaddr_string(ap
)); 
 440                 TCHECK(op
->ospf_db
.db_options
); 
 441                 ospf_print_bits(ospf_option_bits
, op
->ospf_db
.db_options
); 
 443                 TCHECK(op
->ospf_db
.db_flags
); 
 444                 if (op
->ospf_db
.db_flags 
& OSPF_DB_INIT
) { 
 448                 if (op
->ospf_db
.db_flags 
& OSPF_DB_MORE
) { 
 452                 if (op
->ospf_db
.db_flags 
& OSPF_DB_MASTER
) { 
 456                 TCHECK(op
->ospf_db
.db_seq
); 
 457                 printf(" S %X", (u_int32_t
)ntohl(op
->ospf_db
.db_seq
)); 
 460                         /* Print all the LS adv's */ 
 461                         lshp 
= op
->ospf_db
.db_lshdr
; 
 463                         while (!ospf_print_lshdr(lshp
)) { 
 474                         while ((u_char 
*)lsrp 
< dataend
) { 
 476                                 printf(" {");           /* } (ctags) */ 
 477                                 ospf_print_ls_type(ntohl(lsrp
->ls_type
), 
 490                         lsap 
= op
->ospf_lsu
.lsu_lsa
; 
 491                         TCHECK(op
->ospf_lsu
.lsu_count
); 
 492                         i 
= ntohl(op
->ospf_lsu
.lsu_count
); 
 494                                 if (ospf_print_lsa(lsap
)) 
 496                                 lsap 
= (struct lsa 
*)((u_char 
*)lsap 
+ 
 497                                     ntohs(lsap
->ls_hdr
.ls_length
)); 
 505                         lshp 
= op
->ospf_lsa
.lsa_lshdr
; 
 507                         while (!ospf_print_lshdr(lshp
)) { 
 516                 printf("v2 type %d", op
->ospf_type
); 
 525 ospf_print(register const u_char 
*bp
, register u_int length
, 
 526     register const u_char 
*bp2
) 
 528         register const struct ospfhdr 
*op
; 
 529         register const struct ip 
*ip
; 
 530         register const u_char 
*dataend
; 
 531         register const char *cp
; 
 533         op 
= (struct ospfhdr 
*)bp
; 
 534         ip 
= (struct ip 
*)bp2
; 
 535         /* Print the source and destination address  */ 
 536         (void) printf("%s > %s:", 
 537             ipaddr_string(&ip
->ip_src
), 
 538             ipaddr_string(&ip
->ip_dst
)); 
 540         /* If the type is valid translate it, or just print the type */ 
 541         /* value.  If it's not valid, say so and return */ 
 542         TCHECK(op
->ospf_type
); 
 543         cp 
= tok2str(type2str
, "type%d", op
->ospf_type
); 
 544         printf(" OSPFv%d-%s %d:", op
->ospf_version
, cp
, length
); 
 548         TCHECK(op
->ospf_len
); 
 549         if (length 
!= ntohs(op
->ospf_len
)) { 
 550                 printf(" [len %d]", ntohs(op
->ospf_len
)); 
 553         dataend 
= bp 
+ length
; 
 555         /* Print the routerid if it is not the same as the source */ 
 556         TCHECK(op
->ospf_routerid
); 
 557         if (ip
->ip_src
.s_addr 
!= op
->ospf_routerid
.s_addr
) 
 558                 printf(" rtrid %s", ipaddr_string(&op
->ospf_routerid
)); 
 560         TCHECK(op
->ospf_areaid
); 
 561         if (op
->ospf_areaid
.s_addr 
!= 0) 
 562                 printf(" area %s", ipaddr_string(&op
->ospf_areaid
)); 
 567                 /* Print authentication data (should we really do this?) */ 
 568                 TCHECK2(op
->ospf_authdata
[0], sizeof(op
->ospf_authdata
)); 
 569                 switch (ntohs(op
->ospf_authtype
)) { 
 574                 case OSPF_AUTH_SIMPLE
: 
 576                         (void)fn_printn(op
->ospf_authdata
, 
 577                             sizeof(op
->ospf_authdata
), NULL
); 
 582                         printf(" ??authtype-%d??", ntohs(op
->ospf_authtype
)); 
 586         /* Do rest according to version.         */ 
 587         switch (op
->ospf_version
) { 
 591                 if (ospf_decode_v2(op
, dataend
)) 
 596                 printf(" ospf [version %d]", op
->ospf_version
); 
 598         }                       /* end switch on version */