1b0453382SBill Fenner /* 2b0453382SBill Fenner * Copyright (c) 1990, 1991, 1993, 1994, 1995, 1996, 1997 3b0453382SBill Fenner * The Regents of the University of California. All rights reserved. 4b0453382SBill Fenner * 5b0453382SBill Fenner * Redistribution and use in source and binary forms, with or without 6b0453382SBill Fenner * modification, are permitted provided that: (1) source code distributions 7b0453382SBill Fenner * retain the above copyright notice and this paragraph in its entirety, (2) 8b0453382SBill Fenner * distributions including binary code include the above copyright notice and 9b0453382SBill Fenner * this paragraph in its entirety in the documentation or other materials 10b0453382SBill Fenner * provided with the distribution, and (3) all advertising materials mentioning 11b0453382SBill Fenner * features or use of this software display the following acknowledgement: 12b0453382SBill Fenner * ``This product includes software developed by the University of California, 13b0453382SBill Fenner * Lawrence Berkeley Laboratory and its contributors.'' Neither the name of 14b0453382SBill Fenner * the University nor the names of its contributors may be used to endorse 15b0453382SBill Fenner * or promote products derived from this software without specific prior 16b0453382SBill Fenner * written permission. 17b0453382SBill Fenner * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED 18b0453382SBill Fenner * WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF 19b0453382SBill Fenner * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. 20b0453382SBill Fenner */ 21b0453382SBill Fenner 22b0453382SBill Fenner #ifndef lint 235b0fe478SBruce M Simpson static const char rcsid[] _U_ = 24b5bfcb5dSMax Laier "@(#) $Header: /tcpdump/master/tcpdump/print-chdlc.c,v 1.32.2.11 2005/11/29 08:57:10 hannes Exp $ (LBL)"; 25b0453382SBill Fenner #endif 26b0453382SBill Fenner 27b0453382SBill Fenner #ifdef HAVE_CONFIG_H 28b0453382SBill Fenner #include "config.h" 29b0453382SBill Fenner #endif 30b0453382SBill Fenner 315b0fe478SBruce M Simpson #include <tcpdump-stdinc.h> 32b0453382SBill Fenner 33b0453382SBill Fenner #include <pcap.h> 34b0453382SBill Fenner #include <stdio.h> 35b0453382SBill Fenner 36b0453382SBill Fenner #include "interface.h" 37b0453382SBill Fenner #include "addrtoname.h" 38685295f4SBill Fenner #include "ethertype.h" 39a90e161bSBill Fenner #include "extract.h" 40b0453382SBill Fenner #include "ppp.h" 41685295f4SBill Fenner #include "chdlc.h" 42b0453382SBill Fenner 43b0453382SBill Fenner static void chdlc_slarp_print(const u_char *, u_int); 44b0453382SBill Fenner 45b5bfcb5dSMax Laier const struct tok chdlc_cast_values[] = { 46b5bfcb5dSMax Laier { CHDLC_UNICAST, "unicast" }, 47b5bfcb5dSMax Laier { CHDLC_BCAST, "bcast" }, 48b5bfcb5dSMax Laier { 0, NULL} 49b5bfcb5dSMax Laier }; 50b5bfcb5dSMax Laier 51b5bfcb5dSMax Laier 52b0453382SBill Fenner /* Standard CHDLC printer */ 535b0fe478SBruce M Simpson u_int 545b0fe478SBruce M Simpson chdlc_if_print(const struct pcap_pkthdr *h, register const u_char *p) 55b0453382SBill Fenner { 56b0453382SBill Fenner register u_int length = h->len; 57b0453382SBill Fenner register u_int caplen = h->caplen; 58a90e161bSBill Fenner 59a90e161bSBill Fenner if (caplen < CHDLC_HDRLEN) { 60a90e161bSBill Fenner printf("[|chdlc]"); 615b0fe478SBruce M Simpson return (caplen); 62a90e161bSBill Fenner } 632ebc47dbSSam Leffler return (chdlc_print(p,length)); 642ebc47dbSSam Leffler } 652ebc47dbSSam Leffler 662ebc47dbSSam Leffler u_int 672ebc47dbSSam Leffler chdlc_print(register const u_char *p, u_int length) { 682ebc47dbSSam Leffler u_int proto; 69a90e161bSBill Fenner 70a90e161bSBill Fenner proto = EXTRACT_16BITS(&p[2]); 71b0453382SBill Fenner if (eflag) { 72b5bfcb5dSMax Laier printf("%s, ethertype %s (0x%04x), length %u: ", 73b5bfcb5dSMax Laier tok2str(chdlc_cast_values, "0x%02x", p[0]), 74b5bfcb5dSMax Laier tok2str(ethertype_values, "Unknown", proto), 75b5bfcb5dSMax Laier proto, 76b5bfcb5dSMax Laier length); 77b0453382SBill Fenner } 78b0453382SBill Fenner 79b0453382SBill Fenner length -= CHDLC_HDRLEN; 80b5bfcb5dSMax Laier p += CHDLC_HDRLEN; 81b5bfcb5dSMax Laier 82b0453382SBill Fenner switch (proto) { 83b0453382SBill Fenner case ETHERTYPE_IP: 84b5bfcb5dSMax Laier ip_print(gndo, p, length); 85b0453382SBill Fenner break; 86b0453382SBill Fenner #ifdef INET6 87b0453382SBill Fenner case ETHERTYPE_IPV6: 88b5bfcb5dSMax Laier ip6_print(p, length); 89b0453382SBill Fenner break; 90b0453382SBill Fenner #endif 91b0453382SBill Fenner case CHDLC_TYPE_SLARP: 92b5bfcb5dSMax Laier chdlc_slarp_print(p, length); 93b0453382SBill Fenner break; 94b0453382SBill Fenner #if 0 95b0453382SBill Fenner case CHDLC_TYPE_CDP: 96b5bfcb5dSMax Laier chdlc_cdp_print(p, length); 97b0453382SBill Fenner break; 98b0453382SBill Fenner #endif 995b0fe478SBruce M Simpson case ETHERTYPE_MPLS: 1005b0fe478SBruce M Simpson case ETHERTYPE_MPLS_MULTI: 101b5bfcb5dSMax Laier mpls_print(p, length); 1025b0fe478SBruce M Simpson break; 1035b0fe478SBruce M Simpson case ETHERTYPE_ISO: 1045b0fe478SBruce M Simpson /* is the fudge byte set ? lets verify by spotting ISO headers */ 105b5bfcb5dSMax Laier if (*(p+1) == 0x81 || 106b5bfcb5dSMax Laier *(p+1) == 0x82 || 107b5bfcb5dSMax Laier *(p+1) == 0x83) 108b5bfcb5dSMax Laier isoclns_print(p+1, length-1, length-1); 1095b0fe478SBruce M Simpson else 110b5bfcb5dSMax Laier isoclns_print(p, length, length); 1115b0fe478SBruce M Simpson break; 1125b0fe478SBruce M Simpson default: 113b5bfcb5dSMax Laier if (!eflag) 1145b0fe478SBruce M Simpson printf("unknown CHDLC protocol (0x%04x)", proto); 1155b0fe478SBruce M Simpson break; 116b0453382SBill Fenner } 1175b0fe478SBruce M Simpson 1185b0fe478SBruce M Simpson return (CHDLC_HDRLEN); 119b0453382SBill Fenner } 120b0453382SBill Fenner 121f4d0c64aSSam Leffler /* 122f4d0c64aSSam Leffler * The fixed-length portion of a SLARP packet. 123f4d0c64aSSam Leffler */ 124b0453382SBill Fenner struct cisco_slarp { 125f4d0c64aSSam Leffler u_int8_t code[4]; 126b0453382SBill Fenner #define SLARP_REQUEST 0 127b0453382SBill Fenner #define SLARP_REPLY 1 128b0453382SBill Fenner #define SLARP_KEEPALIVE 2 129b0453382SBill Fenner union { 130b0453382SBill Fenner struct { 131f4d0c64aSSam Leffler u_int8_t addr[4]; 132f4d0c64aSSam Leffler u_int8_t mask[4]; 133b0453382SBill Fenner } addr; 134b0453382SBill Fenner struct { 135f4d0c64aSSam Leffler u_int8_t myseq[4]; 136f4d0c64aSSam Leffler u_int8_t yourseq[4]; 137f4d0c64aSSam Leffler u_int8_t rel[2]; 138b0453382SBill Fenner } keep; 139b0453382SBill Fenner } un; 140b0453382SBill Fenner }; 141b0453382SBill Fenner 142f4d0c64aSSam Leffler #define SLARP_MIN_LEN 14 143f4d0c64aSSam Leffler #define SLARP_MAX_LEN 18 144b0453382SBill Fenner 145b0453382SBill Fenner static void 146b0453382SBill Fenner chdlc_slarp_print(const u_char *cp, u_int length) 147b0453382SBill Fenner { 148a90e161bSBill Fenner const struct cisco_slarp *slarp; 149f4d0c64aSSam Leffler u_int sec,min,hrs,days; 150b0453382SBill Fenner 151f4d0c64aSSam Leffler printf("SLARP (length: %u), ",length); 152f4d0c64aSSam Leffler if (length < SLARP_MIN_LEN) 1535b0fe478SBruce M Simpson goto trunc; 154b0453382SBill Fenner 155a90e161bSBill Fenner slarp = (const struct cisco_slarp *)cp; 156f4d0c64aSSam Leffler TCHECK2(*slarp, SLARP_MIN_LEN); 1575b0fe478SBruce M Simpson switch (EXTRACT_32BITS(&slarp->code)) { 158b0453382SBill Fenner case SLARP_REQUEST: 1595b0fe478SBruce M Simpson printf("request"); 160f4d0c64aSSam Leffler /* 161f4d0c64aSSam Leffler * At least according to William "Chops" Westfield's 162f4d0c64aSSam Leffler * message in 163f4d0c64aSSam Leffler * 164f4d0c64aSSam Leffler * http://www.nethelp.no/net/cisco-hdlc.txt 165f4d0c64aSSam Leffler * 166f4d0c64aSSam Leffler * the address and mask aren't used in requests - 167f4d0c64aSSam Leffler * they're just zero. 168f4d0c64aSSam Leffler */ 169b0453382SBill Fenner break; 170b0453382SBill Fenner case SLARP_REPLY: 1715b0fe478SBruce M Simpson printf("reply %s/%s", 172b0453382SBill Fenner ipaddr_string(&slarp->un.addr.addr), 173b0453382SBill Fenner ipaddr_string(&slarp->un.addr.mask)); 174b0453382SBill Fenner break; 175b0453382SBill Fenner case SLARP_KEEPALIVE: 176f4d0c64aSSam Leffler printf("keepalive: mineseen=0x%08x, yourseen=0x%08x, reliability=0x%04x", 1775b0fe478SBruce M Simpson EXTRACT_32BITS(&slarp->un.keep.myseq), 178f4d0c64aSSam Leffler EXTRACT_32BITS(&slarp->un.keep.yourseq), 179f4d0c64aSSam Leffler EXTRACT_16BITS(&slarp->un.keep.rel)); 180f4d0c64aSSam Leffler 181f4d0c64aSSam Leffler if (length >= SLARP_MAX_LEN) { /* uptime-stamp is optional */ 182f4d0c64aSSam Leffler cp += SLARP_MIN_LEN; 183f4d0c64aSSam Leffler if (!TTEST2(*cp, 4)) 184f4d0c64aSSam Leffler goto trunc; 185f4d0c64aSSam Leffler sec = EXTRACT_32BITS(cp) / 1000; 186f4d0c64aSSam Leffler min = sec / 60; sec -= min * 60; 187f4d0c64aSSam Leffler hrs = min / 60; min -= hrs * 60; 188f4d0c64aSSam Leffler days = hrs / 24; hrs -= days * 24; 189f4d0c64aSSam Leffler printf(", link uptime=%ud%uh%um%us",days,hrs,min,sec); 190f4d0c64aSSam Leffler } 191b0453382SBill Fenner break; 192b0453382SBill Fenner default: 1935b0fe478SBruce M Simpson printf("0x%02x unknown", EXTRACT_32BITS(&slarp->code)); 1945b0fe478SBruce M Simpson if (vflag <= 1) 1955b0fe478SBruce M Simpson print_unknown_data(cp+4,"\n\t",length-4); 196b0453382SBill Fenner break; 197b0453382SBill Fenner } 198b0453382SBill Fenner 199f4d0c64aSSam Leffler if (SLARP_MAX_LEN < length && vflag) 200f4d0c64aSSam Leffler printf(", (trailing junk: %d bytes)", length - SLARP_MAX_LEN); 2015b0fe478SBruce M Simpson if (vflag > 1) 2025b0fe478SBruce M Simpson print_unknown_data(cp+4,"\n\t",length-4); 2035b0fe478SBruce M Simpson return; 2045b0fe478SBruce M Simpson 2055b0fe478SBruce M Simpson trunc: 2065b0fe478SBruce M Simpson printf("[|slarp]"); 207b0453382SBill Fenner } 2081de50e9fSSam Leffler 2091de50e9fSSam Leffler 2101de50e9fSSam Leffler /* 2111de50e9fSSam Leffler * Local Variables: 2121de50e9fSSam Leffler * c-style: whitesmith 2131de50e9fSSam Leffler * c-basic-offset: 8 2141de50e9fSSam Leffler * End: 2151de50e9fSSam Leffler */ 216