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
brcm_tag_print(netdissect_options * ndo,const u_char * bp)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
brcm_tag_if_print(netdissect_options * ndo,const struct pcap_pkthdr * h,const u_char * p)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
brcm_tag_prepend_if_print(netdissect_options * ndo,const struct pcap_pkthdr * h,const u_char * p)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