]>
The Tcpdump Group git mirrors - tcpdump/blob - print-arcnet.c
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.
21 * From: NetBSD: print-arcnet.c,v 1.2 2000/04/24 13:02:28 itojun Exp
24 /* \summary: Attached Resource Computer NETwork (ARCNET) printer */
28 #include "netdissect-stdinc.h"
30 #include "netdissect.h"
34 * from: NetBSD: if_arc.h,v 1.13 1999/11/19 20:41:19 thorpej Exp
38 * Structure of a 2.5MB/s Arcnet header on the BSDs,
39 * as given to interface code.
46 * only present for newstyle encoding with LL fragmentation.
47 * Don't use sizeof(anything), use ARC_HDR{,NEW}LEN instead.
50 nd_uint16_t arc_seqid
;
53 * only present in exception packets (arc_flag == 0xff)
55 nd_uint8_t arc_type2
; /* same as arc_type */
56 nd_uint8_t arc_flag2
; /* real flag value */
57 nd_uint16_t arc_seqid2
; /* real seqid value */
61 #define ARC_HDRNEWLEN 6
62 #define ARC_HDRNEWLEN_EXC 10
65 #define ARCTYPE_IP_OLD 240 /* IP protocol */
66 #define ARCTYPE_ARP_OLD 241 /* address resolution protocol */
69 #define ARCTYPE_IP 212 /* IP protocol */
70 #define ARCTYPE_ARP 213 /* address resolution protocol */
71 #define ARCTYPE_REVARP 214 /* reverse addr resolution protocol */
73 #define ARCTYPE_ATALK 221 /* Appletalk */
74 #define ARCTYPE_BANIAN 247 /* Banyan Vines */
75 #define ARCTYPE_IPX 250 /* Novell IPX */
77 #define ARCTYPE_INET6 0xc4 /* IPng */
78 #define ARCTYPE_DIAGNOSE 0x80 /* as per ANSI/ATA 878.1 */
81 * Structure of a 2.5MB/s Arcnet header on Linux. Linux has
82 * an extra "offset" field when given to interface code, and
83 * never presents packets that look like exception frames.
85 struct arc_linux_header
{
88 nd_uint16_t arc_offset
;
91 * only present for newstyle encoding with LL fragmentation.
92 * Don't use sizeof(anything), use ARC_LINUX_HDR{,NEW}LEN
96 nd_uint16_t arc_seqid
;
99 #define ARC_LINUX_HDRLEN 5
100 #define ARC_LINUX_HDRNEWLEN 8
102 static int arcnet_encap_print(netdissect_options
*, u_char arctype
, const u_char
*p
,
103 u_int length
, u_int caplen
);
105 static const struct tok arctypemap
[] = {
106 { ARCTYPE_IP_OLD
, "oldip" },
107 { ARCTYPE_ARP_OLD
, "oldarp" },
108 { ARCTYPE_IP
, "ip" },
109 { ARCTYPE_ARP
, "arp" },
110 { ARCTYPE_REVARP
, "rarp" },
111 { ARCTYPE_ATALK
, "atalk" },
112 { ARCTYPE_BANIAN
, "banyan" },
113 { ARCTYPE_IPX
, "ipx" },
114 { ARCTYPE_INET6
, "ipv6" },
115 { ARCTYPE_DIAGNOSE
, "diag" },
120 arcnet_print(netdissect_options
*ndo
, const u_char
*bp
, u_int length
, int phds
,
121 u_int flag
, u_int seqid
)
123 const struct arc_header
*ap
;
124 const char *arctypename
;
126 ndo
->ndo_protocol
= "arcnet";
127 ap
= (const struct arc_header
*)bp
;
129 if (ndo
->ndo_qflag
) {
130 ND_PRINT("%02x %02x %u: ",
131 GET_U_1(ap
->arc_shost
),
132 GET_U_1(ap
->arc_dhost
),
137 arctypename
= tok2str(arctypemap
, "%02x", GET_U_1(ap
->arc_type
));
140 ND_PRINT("%02x %02x %s %u: ",
141 GET_U_1(ap
->arc_shost
),
142 GET_U_1(ap
->arc_dhost
),
149 ND_PRINT("%02x %02x %s seqid %04x %u: ",
150 GET_U_1(ap
->arc_shost
),
151 GET_U_1(ap
->arc_dhost
),
158 ND_PRINT("%02x %02x %s seqid %04x "
159 "(first of %u fragments) %u: ",
160 GET_U_1(ap
->arc_shost
),
161 GET_U_1(ap
->arc_dhost
),
163 (flag
+ 3) / 2, length
);
165 ND_PRINT("%02x %02x %s seqid %04x "
166 "(fragment %u) %u: ",
167 GET_U_1(ap
->arc_shost
),
168 GET_U_1(ap
->arc_dhost
),
174 * This is the top level routine of the printer. 'p' points
175 * to the ARCNET header of the packet, 'h->ts' is the timestamp,
176 * 'h->len' is the length of the packet off the wire, and 'h->caplen'
177 * is the number of bytes actually captured.
180 arcnet_if_print(netdissect_options
*ndo
, const struct pcap_pkthdr
*h
, const u_char
*p
)
182 u_int caplen
= h
->caplen
;
183 u_int length
= h
->len
;
184 const struct arc_header
*ap
;
187 u_int flag
= 0, archdrlen
= 0;
191 ndo
->ndo_protocol
= "arcnet";
192 if (caplen
< ARC_HDRLEN
) {
193 ndo
->ndo_ll_hdr_len
+= caplen
;
194 nd_trunc_longjmp(ndo
);
197 ap
= (const struct arc_header
*)p
;
198 arc_type
= GET_U_1(ap
->arc_type
);
205 case ARCTYPE_ARP_OLD
:
206 case ARCTYPE_DIAGNOSE
:
208 archdrlen
= ARC_HDRLEN
;
213 if (caplen
< ARC_HDRNEWLEN
) {
214 arcnet_print(ndo
, p
, length
, 0, 0, 0);
216 ndo
->ndo_ll_hdr_len
+= caplen
;
217 nd_trunc_longjmp(ndo
);
220 flag
= GET_U_1(ap
->arc_flag
);
222 if (caplen
< ARC_HDRNEWLEN_EXC
) {
223 arcnet_print(ndo
, p
, length
, 0, 0, 0);
224 ND_PRINT(" phds extended");
225 ndo
->ndo_ll_hdr_len
+= caplen
;
226 nd_trunc_longjmp(ndo
);
228 flag
= GET_U_1(ap
->arc_flag2
);
229 seqid
= GET_BE_U_2(ap
->arc_seqid2
);
230 archdrlen
= ARC_HDRNEWLEN_EXC
;
232 seqid
= GET_BE_U_2(ap
->arc_seqid
);
233 archdrlen
= ARC_HDRNEWLEN
;
239 arcnet_print(ndo
, p
, length
, phds
, flag
, seqid
);
242 * Go past the ARCNET header.
248 if (phds
&& flag
&& (flag
& 1) == 0) {
250 * This is a middle fragment.
252 ndo
->ndo_ll_hdr_len
+= archdrlen
;
256 if (!arcnet_encap_print(ndo
, arc_type
, p
, length
, caplen
))
257 ND_DEFAULTPRINT(p
, caplen
);
259 ndo
->ndo_ll_hdr_len
+= archdrlen
;
263 * This is the top level routine of the printer. 'p' points
264 * to the ARCNET header of the packet, 'h->ts' is the timestamp,
265 * 'h->len' is the length of the packet off the wire, and 'h->caplen'
266 * is the number of bytes actually captured. It is quite similar
267 * to the non-Linux style printer except that Linux doesn't ever
268 * supply packets that look like exception frames, it always supplies
269 * reassembled packets rather than raw frames, and headers have an
270 * extra "offset" field between the src/dest and packet type.
273 arcnet_linux_if_print(netdissect_options
*ndo
, const struct pcap_pkthdr
*h
, const u_char
*p
)
275 u_int caplen
= h
->caplen
;
276 u_int length
= h
->len
;
277 const struct arc_linux_header
*ap
;
282 ndo
->ndo_protocol
= "arcnet_linux";
283 if (caplen
< ARC_LINUX_HDRLEN
) {
284 ndo
->ndo_ll_hdr_len
+= caplen
;
285 nd_trunc_longjmp(ndo
);
288 ap
= (const struct arc_linux_header
*)p
;
289 arc_type
= GET_U_1(ap
->arc_type
);
293 archdrlen
= ARC_LINUX_HDRNEWLEN
;
294 if (caplen
< ARC_LINUX_HDRNEWLEN
) {
295 ndo
->ndo_ll_hdr_len
+= caplen
;
296 nd_trunc_longjmp(ndo
);
300 case ARCTYPE_ARP_OLD
:
301 case ARCTYPE_DIAGNOSE
:
302 archdrlen
= ARC_LINUX_HDRLEN
;
307 arcnet_print(ndo
, p
, length
, 0, 0, 0);
310 * Go past the ARCNET header.
316 if (!arcnet_encap_print(ndo
, arc_type
, p
, length
, caplen
))
317 ND_DEFAULTPRINT(p
, caplen
);
319 ndo
->ndo_ll_hdr_len
+= archdrlen
;
323 * Prints the packet encapsulated in an ARCnet data field,
324 * given the ARCnet system code.
326 * Returns non-zero if it can do so, zero if the system code is unknown.
331 arcnet_encap_print(netdissect_options
*ndo
, u_char arctype
, const u_char
*p
,
332 u_int length
, u_int caplen
)
338 ip_print(ndo
, p
, length
);
342 ip6_print(ndo
, p
, length
);
345 case ARCTYPE_ARP_OLD
:
348 arp_print(ndo
, p
, length
, caplen
);
351 case ARCTYPE_ATALK
: /* XXX was this ever used? */
354 atalk_print(ndo
, p
, length
);
358 ipx_print(ndo
, p
, length
);