2 * Copyright (c) 1988, 1989, 1990, 1991, 1992, 1993, 1994, 1995, 1996, 1997
3 * The Regents of the University of California. All rights reserved.
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
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.
22 /* \summary: Marvell Extended Distributed Switch Architecture (MEDSA) printer */
28 #include <netdissect-stdinc.h>
30 #include "netdissect.h"
32 #include "ethertype.h"
35 static const char tstr
[] = "[|MEDSA]";
38 * Marvell Extended Distributed Switch Archiecture.
40 * A Marvell propriatary header used for passing packets to/from
41 * specific ports of a switch. There is no open specification of this
42 * header, but is documented in the Marvell Switch data sheets. For
45 * https://lwn.net/Articles/302333/
52 /* Bytes 0 and 1 are reserved and should contain 0 */
53 #define TAG(medsa) (medsa->bytes[2] >> 6)
55 #define TAG_FROM_CPU 1
57 #define SRC_TAG(medsa) ((medsa->bytes[2] >> 5) & 0x01)
58 #define SRC_DEV(medsa) (medsa->bytes[2] & 0x1f)
59 #define SRC_PORT(medsa) ((medsa->bytes[3] >> 3) & 0x01f)
60 #define TRUNK(medsa) ((medsa->bytes[3] >> 2) & 0x01)
61 #define CODE(medsa) ((medsa->bytes[3] & 0x06) | \
62 ((medsa->bytes[4] >> 4) & 0x01))
64 #define CODE_IGMP_MLD 2
65 #define CODE_ARP_MIRROR 4
66 #define CFI(medsa) (medsa->bytes[3] & 0x01)
67 #define PRI(medsa) (medsa->bytes[4] >> 5)
68 #define VID(medsa) (((u_short)(medsa->bytes[4] & 0xf) << 8 | \
71 static const struct tok tag_values
[] = {
72 { TAG_TO_CPU
, "To_CPU" },
73 { TAG_FROM_CPU
, "From_CPU" },
74 { TAG_FORWARD
, "Forward" },
78 static const struct tok code_values
[] = {
79 { CODE_BDPU
, "BDPU" },
80 { CODE_IGMP_MLD
, "IGMP/MLD" },
81 { CODE_ARP_MIRROR
, "APR_Mirror" },
86 medsa_print_full(netdissect_options
*ndo
,
87 const struct medsa_pkthdr
*medsa
,
90 u_char tag
= TAG(medsa
);
93 tok2str(tag_values
, "Unknown (%u)", tag
)));
97 ND_PRINT((ndo
, ", %stagged", SRC_TAG(medsa
) ? "" : "un"));
98 ND_PRINT((ndo
, ", dev.port:vlan %d.%d:%d",
99 SRC_DEV(medsa
), SRC_PORT(medsa
), VID(medsa
)));
101 ND_PRINT((ndo
, ", %s",
102 tok2str(code_values
, "Unknown (%u)", CODE(medsa
))));
104 ND_PRINT((ndo
, ", CFI"));
106 ND_PRINT((ndo
, ", pri %d: ", PRI(medsa
)));
109 ND_PRINT((ndo
, ", %stagged", SRC_TAG(medsa
) ? "" : "un"));
110 ND_PRINT((ndo
, ", dev.port:vlan %d.%d:%d",
111 SRC_DEV(medsa
), SRC_PORT(medsa
), VID(medsa
)));
114 ND_PRINT((ndo
, ", CFI"));
116 ND_PRINT((ndo
, ", pri %d: ", PRI(medsa
)));
119 ND_PRINT((ndo
, ", %stagged", SRC_TAG(medsa
) ? "" : "un"));
121 ND_PRINT((ndo
, ", dev.trunk:vlan %d.%d:%d",
122 SRC_DEV(medsa
), SRC_PORT(medsa
), VID(medsa
)));
124 ND_PRINT((ndo
, ", dev.port:vlan %d.%d:%d",
125 SRC_DEV(medsa
), SRC_PORT(medsa
), VID(medsa
)));
128 ND_PRINT((ndo
, ", CFI"));
130 ND_PRINT((ndo
, ", pri %d: ", PRI(medsa
)));
133 ND_DEFAULTPRINT((const u_char
*)medsa
, caplen
);
139 medsa_print(netdissect_options
*ndo
,
140 const u_char
*bp
, u_int length
, u_int caplen
)
142 register const struct ether_header
*ep
;
143 const struct medsa_pkthdr
*medsa
;
146 medsa
= (const struct medsa_pkthdr
*)bp
;
147 ep
= (const struct ether_header
*)(bp
- sizeof(*ep
));
151 ND_PRINT((ndo
, "MEDSA %d.%d:%d: ",
152 SRC_DEV(medsa
), SRC_PORT(medsa
), VID(medsa
)));
154 medsa_print_full(ndo
, medsa
, caplen
);
160 ether_type
= EXTRACT_16BITS(&medsa
->ether_type
);
161 if (ether_type
<= ETHERMTU
) {
162 /* Try to print the LLC-layer header & higher layers */
163 if (llc_print(ndo
, bp
, length
, caplen
, ESRC(ep
), EDST(ep
)) < 0) {
164 /* packet type not known, print raw packet */
165 if (!ndo
->ndo_suppress_default_print
)
166 ND_DEFAULTPRINT(bp
, caplen
);
170 ND_PRINT((ndo
, "ethertype %s (0x%04x) ",
171 tok2str(ethertype_values
, "Unknown",
175 if (ethertype_print(ndo
, ether_type
, bp
, length
, caplen
) == 0) {
176 /* ether_type not known, print raw packet */
178 ND_PRINT((ndo
, "ethertype %s (0x%04x) ",
179 tok2str(ethertype_values
, "Unknown",
183 if (!ndo
->ndo_suppress_default_print
)
184 ND_DEFAULTPRINT(bp
, caplen
);
189 ND_PRINT((ndo
, "%s", tstr
));