1*ee67461eSJoseph Mingrone /* 2*ee67461eSJoseph Mingrone * Copyright (c) 1988, 1989, 1990, 1991, 1992, 1993, 1994, 1995, 1996, 1997 3*ee67461eSJoseph Mingrone * The Regents of the University of California. All rights reserved. 4*ee67461eSJoseph Mingrone * 5*ee67461eSJoseph Mingrone * Redistribution and use in source and binary forms, with or without 6*ee67461eSJoseph Mingrone * modification, are permitted provided that: (1) source code distributions 7*ee67461eSJoseph Mingrone * retain the above copyright notice and this paragraph in its entirety, (2) 8*ee67461eSJoseph Mingrone * distributions including binary code include the above copyright notice and 9*ee67461eSJoseph Mingrone * this paragraph in its entirety in the documentation or other materials 10*ee67461eSJoseph Mingrone * provided with the distribution, and (3) all advertising materials mentioning 11*ee67461eSJoseph Mingrone * features or use of this software display the following acknowledgement: 12*ee67461eSJoseph Mingrone * ``This product includes software developed by the University of California, 13*ee67461eSJoseph Mingrone * Lawrence Berkeley Laboratory and its contributors.'' Neither the name of 14*ee67461eSJoseph Mingrone * the University nor the names of its contributors may be used to endorse 15*ee67461eSJoseph Mingrone * or promote products derived from this software without specific prior 16*ee67461eSJoseph Mingrone * written permission. 17*ee67461eSJoseph Mingrone * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED 18*ee67461eSJoseph Mingrone * WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF 19*ee67461eSJoseph Mingrone * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. 20*ee67461eSJoseph Mingrone */ 21*ee67461eSJoseph Mingrone 22*ee67461eSJoseph Mingrone /* \summary: Broadcom Ethernet switches tag (4 bytes) printer */ 23*ee67461eSJoseph Mingrone 24*ee67461eSJoseph Mingrone #include <config.h> 25*ee67461eSJoseph Mingrone 26*ee67461eSJoseph Mingrone #include "netdissect-stdinc.h" 27*ee67461eSJoseph Mingrone 28*ee67461eSJoseph Mingrone #define ND_LONGJMP_FROM_TCHECK 29*ee67461eSJoseph Mingrone #include "netdissect.h" 30*ee67461eSJoseph Mingrone #include "addrtoname.h" 31*ee67461eSJoseph Mingrone #include "extract.h" 32*ee67461eSJoseph Mingrone 33*ee67461eSJoseph Mingrone #define ETHER_TYPE_LEN 2 34*ee67461eSJoseph Mingrone 35*ee67461eSJoseph Mingrone #define BRCM_TAG_LEN 4 36*ee67461eSJoseph Mingrone #define BRCM_OPCODE_SHIFT 5 37*ee67461eSJoseph Mingrone #define BRCM_OPCODE_MASK 0x7 38*ee67461eSJoseph Mingrone 39*ee67461eSJoseph Mingrone /* Ingress fields */ 40*ee67461eSJoseph Mingrone #define BRCM_IG_TC_SHIFT 2 41*ee67461eSJoseph Mingrone #define BRCM_IG_TC_MASK 0x7 42*ee67461eSJoseph Mingrone #define BRCM_IG_TE_MASK 0x3 43*ee67461eSJoseph Mingrone #define BRCM_IG_TS_SHIFT 7 44*ee67461eSJoseph Mingrone #define BRCM_IG_DSTMAP_MASK 0x1ff 45*ee67461eSJoseph Mingrone 46*ee67461eSJoseph Mingrone /* Egress fields */ 47*ee67461eSJoseph Mingrone #define BRCM_EG_CID_MASK 0xff 48*ee67461eSJoseph Mingrone #define BRCM_EG_RC_MASK 0xff 49*ee67461eSJoseph Mingrone #define BRCM_EG_RC_RSVD (3 << 6) 50*ee67461eSJoseph Mingrone #define BRCM_EG_RC_EXCEPTION (1 << 5) 51*ee67461eSJoseph Mingrone #define BRCM_EG_RC_PROT_SNOOP (1 << 4) 52*ee67461eSJoseph Mingrone #define BRCM_EG_RC_PROT_TERM (1 << 3) 53*ee67461eSJoseph Mingrone #define BRCM_EG_RC_SWITCH (1 << 2) 54*ee67461eSJoseph Mingrone #define BRCM_EG_RC_MAC_LEARN (1 << 1) 55*ee67461eSJoseph Mingrone #define BRCM_EG_RC_MIRROR (1 << 0) 56*ee67461eSJoseph Mingrone #define BRCM_EG_TC_SHIFT 5 57*ee67461eSJoseph Mingrone #define BRCM_EG_TC_MASK 0x7 58*ee67461eSJoseph Mingrone #define BRCM_EG_PID_MASK 0x1f 59*ee67461eSJoseph Mingrone 60*ee67461eSJoseph Mingrone static const struct tok brcm_tag_te_values[] = { 61*ee67461eSJoseph Mingrone { 0, "None" }, 62*ee67461eSJoseph Mingrone { 1, "Untag" }, 63*ee67461eSJoseph Mingrone { 2, "Header"}, 64*ee67461eSJoseph Mingrone { 3, "Reserved" }, 65*ee67461eSJoseph Mingrone { 0, NULL } 66*ee67461eSJoseph Mingrone }; 67*ee67461eSJoseph Mingrone 68*ee67461eSJoseph Mingrone static const struct tok brcm_tag_rc_values[] = { 69*ee67461eSJoseph Mingrone { 1, "mirror" }, 70*ee67461eSJoseph Mingrone { 2, "MAC learning" }, 71*ee67461eSJoseph Mingrone { 4, "switching" }, 72*ee67461eSJoseph Mingrone { 8, "prot term" }, 73*ee67461eSJoseph Mingrone { 16, "prot snoop" }, 74*ee67461eSJoseph Mingrone { 32, "exception" }, 75*ee67461eSJoseph Mingrone { 0, NULL } 76*ee67461eSJoseph Mingrone }; 77*ee67461eSJoseph Mingrone 78*ee67461eSJoseph Mingrone static void 79*ee67461eSJoseph Mingrone brcm_tag_print(netdissect_options *ndo, const u_char *bp) 80*ee67461eSJoseph Mingrone { 81*ee67461eSJoseph Mingrone uint8_t tag[BRCM_TAG_LEN]; 82*ee67461eSJoseph Mingrone uint16_t dst_map; 83*ee67461eSJoseph Mingrone unsigned int i; 84*ee67461eSJoseph Mingrone 85*ee67461eSJoseph Mingrone for (i = 0; i < BRCM_TAG_LEN; i++) 86*ee67461eSJoseph Mingrone tag[i] = GET_U_1(bp + i); 87*ee67461eSJoseph Mingrone 88*ee67461eSJoseph Mingrone ND_PRINT("BRCM tag OP: %s", tag[0] ? "IG" : "EG"); 89*ee67461eSJoseph Mingrone if (tag[0] & (1 << BRCM_OPCODE_SHIFT)) { 90*ee67461eSJoseph Mingrone /* Ingress Broadcom tag */ 91*ee67461eSJoseph Mingrone ND_PRINT(", TC: %d", (tag[1] >> BRCM_IG_TC_SHIFT) & 92*ee67461eSJoseph Mingrone BRCM_IG_TC_MASK); 93*ee67461eSJoseph Mingrone ND_PRINT(", TE: %s", 94*ee67461eSJoseph Mingrone tok2str(brcm_tag_te_values, "unknown", 95*ee67461eSJoseph Mingrone (tag[1] & BRCM_IG_TE_MASK))); 96*ee67461eSJoseph Mingrone ND_PRINT(", TS: %d", tag[1] >> BRCM_IG_TS_SHIFT); 97*ee67461eSJoseph Mingrone dst_map = (uint16_t)tag[2] << 8 | tag[3]; 98*ee67461eSJoseph Mingrone ND_PRINT(", DST map: 0x%04x", dst_map & BRCM_IG_DSTMAP_MASK); 99*ee67461eSJoseph Mingrone } else { 100*ee67461eSJoseph Mingrone /* Egress Broadcom tag */ 101*ee67461eSJoseph Mingrone ND_PRINT(", CID: %d", tag[1]); 102*ee67461eSJoseph Mingrone ND_PRINT(", RC: %s", tok2str(brcm_tag_rc_values, 103*ee67461eSJoseph Mingrone "reserved", tag[2])); 104*ee67461eSJoseph Mingrone ND_PRINT(", TC: %d", (tag[3] >> BRCM_EG_TC_SHIFT) & 105*ee67461eSJoseph Mingrone BRCM_EG_TC_MASK); 106*ee67461eSJoseph Mingrone ND_PRINT(", port: %d", tag[3] & BRCM_EG_PID_MASK); 107*ee67461eSJoseph Mingrone } 108*ee67461eSJoseph Mingrone ND_PRINT(", "); 109*ee67461eSJoseph Mingrone } 110*ee67461eSJoseph Mingrone 111*ee67461eSJoseph Mingrone void 112*ee67461eSJoseph Mingrone brcm_tag_if_print(netdissect_options *ndo, const struct pcap_pkthdr *h, 113*ee67461eSJoseph Mingrone const u_char *p) 114*ee67461eSJoseph Mingrone { 115*ee67461eSJoseph Mingrone u_int caplen = h->caplen; 116*ee67461eSJoseph Mingrone u_int length = h->len; 117*ee67461eSJoseph Mingrone 118*ee67461eSJoseph Mingrone ndo->ndo_protocol = "brcm-tag"; 119*ee67461eSJoseph Mingrone ndo->ndo_ll_hdr_len += 120*ee67461eSJoseph Mingrone ether_switch_tag_print(ndo, p, length, caplen, 121*ee67461eSJoseph Mingrone brcm_tag_print, BRCM_TAG_LEN); 122*ee67461eSJoseph Mingrone } 123*ee67461eSJoseph Mingrone 124*ee67461eSJoseph Mingrone void 125*ee67461eSJoseph Mingrone brcm_tag_prepend_if_print(netdissect_options *ndo, const struct pcap_pkthdr *h, 126*ee67461eSJoseph Mingrone const u_char *p) 127*ee67461eSJoseph Mingrone { 128*ee67461eSJoseph Mingrone u_int caplen = h->caplen; 129*ee67461eSJoseph Mingrone u_int length = h->len; 130*ee67461eSJoseph Mingrone 131*ee67461eSJoseph Mingrone ndo->ndo_protocol = "brcm-tag-prepend"; 132*ee67461eSJoseph Mingrone ND_TCHECK_LEN(p, BRCM_TAG_LEN); 133*ee67461eSJoseph Mingrone ndo->ndo_ll_hdr_len += BRCM_TAG_LEN; 134*ee67461eSJoseph Mingrone 135*ee67461eSJoseph Mingrone if (ndo->ndo_eflag) { 136*ee67461eSJoseph Mingrone /* Print the prepended Broadcom tag. */ 137*ee67461eSJoseph Mingrone brcm_tag_print(ndo, p); 138*ee67461eSJoseph Mingrone } 139*ee67461eSJoseph Mingrone p += BRCM_TAG_LEN; 140*ee67461eSJoseph Mingrone length -= BRCM_TAG_LEN; 141*ee67461eSJoseph Mingrone caplen -= BRCM_TAG_LEN; 142*ee67461eSJoseph Mingrone 143*ee67461eSJoseph Mingrone /* 144*ee67461eSJoseph Mingrone * Now print the Ethernet frame following it. 145*ee67461eSJoseph Mingrone */ 146*ee67461eSJoseph Mingrone ndo->ndo_ll_hdr_len += 147*ee67461eSJoseph Mingrone ether_print(ndo, p, length, caplen, NULL, NULL); 148*ee67461eSJoseph Mingrone } 149