1*05b00f60SXin Li /*
2*05b00f60SXin Li * Copyright (c) 1988, 1989, 1990, 1991, 1992, 1993, 1994, 1995, 1996, 1997
3*05b00f60SXin Li * The Regents of the University of California. All rights reserved.
4*05b00f60SXin Li *
5*05b00f60SXin Li * Redistribution and use in source and binary forms, with or without
6*05b00f60SXin Li * modification, are permitted provided that: (1) source code distributions
7*05b00f60SXin Li * retain the above copyright notice and this paragraph in its entirety, (2)
8*05b00f60SXin Li * distributions including binary code include the above copyright notice and
9*05b00f60SXin Li * this paragraph in its entirety in the documentation or other materials
10*05b00f60SXin Li * provided with the distribution, and (3) all advertising materials mentioning
11*05b00f60SXin Li * features or use of this software display the following acknowledgement:
12*05b00f60SXin Li * ``This product includes software developed by the University of California,
13*05b00f60SXin Li * Lawrence Berkeley Laboratory and its contributors.'' Neither the name of
14*05b00f60SXin Li * the University nor the names of its contributors may be used to endorse
15*05b00f60SXin Li * or promote products derived from this software without specific prior
16*05b00f60SXin Li * written permission.
17*05b00f60SXin Li * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED
18*05b00f60SXin Li * WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF
19*05b00f60SXin Li * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
20*05b00f60SXin Li *
21*05b00f60SXin Li * From: NetBSD: print-arcnet.c,v 1.2 2000/04/24 13:02:28 itojun Exp
22*05b00f60SXin Li */
23*05b00f60SXin Li
24*05b00f60SXin Li /* \summary: Attached Resource Computer NETwork (ARCNET) printer */
25*05b00f60SXin Li
26*05b00f60SXin Li #ifdef HAVE_CONFIG_H
27*05b00f60SXin Li #include <config.h>
28*05b00f60SXin Li #endif
29*05b00f60SXin Li
30*05b00f60SXin Li #include "netdissect-stdinc.h"
31*05b00f60SXin Li
32*05b00f60SXin Li #include "netdissect.h"
33*05b00f60SXin Li #include "extract.h"
34*05b00f60SXin Li
35*05b00f60SXin Li /*
36*05b00f60SXin Li * from: NetBSD: if_arc.h,v 1.13 1999/11/19 20:41:19 thorpej Exp
37*05b00f60SXin Li */
38*05b00f60SXin Li
39*05b00f60SXin Li /*
40*05b00f60SXin Li * Structure of a 2.5MB/s Arcnet header on the BSDs,
41*05b00f60SXin Li * as given to interface code.
42*05b00f60SXin Li */
43*05b00f60SXin Li struct arc_header {
44*05b00f60SXin Li nd_uint8_t arc_shost;
45*05b00f60SXin Li nd_uint8_t arc_dhost;
46*05b00f60SXin Li nd_uint8_t arc_type;
47*05b00f60SXin Li /*
48*05b00f60SXin Li * only present for newstyle encoding with LL fragmentation.
49*05b00f60SXin Li * Don't use sizeof(anything), use ARC_HDR{,NEW}LEN instead.
50*05b00f60SXin Li */
51*05b00f60SXin Li nd_uint8_t arc_flag;
52*05b00f60SXin Li nd_uint16_t arc_seqid;
53*05b00f60SXin Li
54*05b00f60SXin Li /*
55*05b00f60SXin Li * only present in exception packets (arc_flag == 0xff)
56*05b00f60SXin Li */
57*05b00f60SXin Li nd_uint8_t arc_type2; /* same as arc_type */
58*05b00f60SXin Li nd_uint8_t arc_flag2; /* real flag value */
59*05b00f60SXin Li nd_uint16_t arc_seqid2; /* real seqid value */
60*05b00f60SXin Li };
61*05b00f60SXin Li
62*05b00f60SXin Li #define ARC_HDRLEN 3
63*05b00f60SXin Li #define ARC_HDRNEWLEN 6
64*05b00f60SXin Li #define ARC_HDRNEWLEN_EXC 10
65*05b00f60SXin Li
66*05b00f60SXin Li /* RFC 1051 */
67*05b00f60SXin Li #define ARCTYPE_IP_OLD 240 /* IP protocol */
68*05b00f60SXin Li #define ARCTYPE_ARP_OLD 241 /* address resolution protocol */
69*05b00f60SXin Li
70*05b00f60SXin Li /* RFC 1201 */
71*05b00f60SXin Li #define ARCTYPE_IP 212 /* IP protocol */
72*05b00f60SXin Li #define ARCTYPE_ARP 213 /* address resolution protocol */
73*05b00f60SXin Li #define ARCTYPE_REVARP 214 /* reverse addr resolution protocol */
74*05b00f60SXin Li
75*05b00f60SXin Li #define ARCTYPE_ATALK 221 /* Appletalk */
76*05b00f60SXin Li #define ARCTYPE_BANIAN 247 /* Banyan Vines */
77*05b00f60SXin Li #define ARCTYPE_IPX 250 /* Novell IPX */
78*05b00f60SXin Li
79*05b00f60SXin Li #define ARCTYPE_INET6 0xc4 /* IPng */
80*05b00f60SXin Li #define ARCTYPE_DIAGNOSE 0x80 /* as per ANSI/ATA 878.1 */
81*05b00f60SXin Li
82*05b00f60SXin Li /*
83*05b00f60SXin Li * Structure of a 2.5MB/s Arcnet header on Linux. Linux has
84*05b00f60SXin Li * an extra "offset" field when given to interface code, and
85*05b00f60SXin Li * never presents packets that look like exception frames.
86*05b00f60SXin Li */
87*05b00f60SXin Li struct arc_linux_header {
88*05b00f60SXin Li nd_uint8_t arc_shost;
89*05b00f60SXin Li nd_uint8_t arc_dhost;
90*05b00f60SXin Li nd_uint16_t arc_offset;
91*05b00f60SXin Li nd_uint8_t arc_type;
92*05b00f60SXin Li /*
93*05b00f60SXin Li * only present for newstyle encoding with LL fragmentation.
94*05b00f60SXin Li * Don't use sizeof(anything), use ARC_LINUX_HDR{,NEW}LEN
95*05b00f60SXin Li * instead.
96*05b00f60SXin Li */
97*05b00f60SXin Li nd_uint8_t arc_flag;
98*05b00f60SXin Li nd_uint16_t arc_seqid;
99*05b00f60SXin Li };
100*05b00f60SXin Li
101*05b00f60SXin Li #define ARC_LINUX_HDRLEN 5
102*05b00f60SXin Li #define ARC_LINUX_HDRNEWLEN 8
103*05b00f60SXin Li
104*05b00f60SXin Li static int arcnet_encap_print(netdissect_options *, u_char arctype, const u_char *p,
105*05b00f60SXin Li u_int length, u_int caplen);
106*05b00f60SXin Li
107*05b00f60SXin Li static const struct tok arctypemap[] = {
108*05b00f60SXin Li { ARCTYPE_IP_OLD, "oldip" },
109*05b00f60SXin Li { ARCTYPE_ARP_OLD, "oldarp" },
110*05b00f60SXin Li { ARCTYPE_IP, "ip" },
111*05b00f60SXin Li { ARCTYPE_ARP, "arp" },
112*05b00f60SXin Li { ARCTYPE_REVARP, "rarp" },
113*05b00f60SXin Li { ARCTYPE_ATALK, "atalk" },
114*05b00f60SXin Li { ARCTYPE_BANIAN, "banyan" },
115*05b00f60SXin Li { ARCTYPE_IPX, "ipx" },
116*05b00f60SXin Li { ARCTYPE_INET6, "ipv6" },
117*05b00f60SXin Li { ARCTYPE_DIAGNOSE, "diag" },
118*05b00f60SXin Li { 0, NULL }
119*05b00f60SXin Li };
120*05b00f60SXin Li
121*05b00f60SXin Li static void
arcnet_print(netdissect_options * ndo,const u_char * bp,u_int length,int phds,u_int flag,u_int seqid)122*05b00f60SXin Li arcnet_print(netdissect_options *ndo, const u_char *bp, u_int length, int phds,
123*05b00f60SXin Li u_int flag, u_int seqid)
124*05b00f60SXin Li {
125*05b00f60SXin Li const struct arc_header *ap;
126*05b00f60SXin Li const char *arctypename;
127*05b00f60SXin Li
128*05b00f60SXin Li ndo->ndo_protocol = "arcnet";
129*05b00f60SXin Li ap = (const struct arc_header *)bp;
130*05b00f60SXin Li
131*05b00f60SXin Li if (ndo->ndo_qflag) {
132*05b00f60SXin Li ND_PRINT("%02x %02x %u: ",
133*05b00f60SXin Li GET_U_1(ap->arc_shost),
134*05b00f60SXin Li GET_U_1(ap->arc_dhost),
135*05b00f60SXin Li length);
136*05b00f60SXin Li return;
137*05b00f60SXin Li }
138*05b00f60SXin Li
139*05b00f60SXin Li arctypename = tok2str(arctypemap, "%02x", GET_U_1(ap->arc_type));
140*05b00f60SXin Li
141*05b00f60SXin Li if (!phds) {
142*05b00f60SXin Li ND_PRINT("%02x %02x %s %u: ",
143*05b00f60SXin Li GET_U_1(ap->arc_shost),
144*05b00f60SXin Li GET_U_1(ap->arc_dhost),
145*05b00f60SXin Li arctypename,
146*05b00f60SXin Li length);
147*05b00f60SXin Li return;
148*05b00f60SXin Li }
149*05b00f60SXin Li
150*05b00f60SXin Li if (flag == 0) {
151*05b00f60SXin Li ND_PRINT("%02x %02x %s seqid %04x %u: ",
152*05b00f60SXin Li GET_U_1(ap->arc_shost),
153*05b00f60SXin Li GET_U_1(ap->arc_dhost),
154*05b00f60SXin Li arctypename, seqid,
155*05b00f60SXin Li length);
156*05b00f60SXin Li return;
157*05b00f60SXin Li }
158*05b00f60SXin Li
159*05b00f60SXin Li if (flag & 1)
160*05b00f60SXin Li ND_PRINT("%02x %02x %s seqid %04x "
161*05b00f60SXin Li "(first of %u fragments) %u: ",
162*05b00f60SXin Li GET_U_1(ap->arc_shost),
163*05b00f60SXin Li GET_U_1(ap->arc_dhost),
164*05b00f60SXin Li arctypename, seqid,
165*05b00f60SXin Li (flag + 3) / 2, length);
166*05b00f60SXin Li else
167*05b00f60SXin Li ND_PRINT("%02x %02x %s seqid %04x "
168*05b00f60SXin Li "(fragment %u) %u: ",
169*05b00f60SXin Li GET_U_1(ap->arc_shost),
170*05b00f60SXin Li GET_U_1(ap->arc_dhost),
171*05b00f60SXin Li arctypename, seqid,
172*05b00f60SXin Li flag/2 + 1, length);
173*05b00f60SXin Li }
174*05b00f60SXin Li
175*05b00f60SXin Li /*
176*05b00f60SXin Li * This is the top level routine of the printer. 'p' points
177*05b00f60SXin Li * to the ARCNET header of the packet, 'h->ts' is the timestamp,
178*05b00f60SXin Li * 'h->len' is the length of the packet off the wire, and 'h->caplen'
179*05b00f60SXin Li * is the number of bytes actually captured.
180*05b00f60SXin Li */
181*05b00f60SXin Li void
arcnet_if_print(netdissect_options * ndo,const struct pcap_pkthdr * h,const u_char * p)182*05b00f60SXin Li arcnet_if_print(netdissect_options *ndo, const struct pcap_pkthdr *h, const u_char *p)
183*05b00f60SXin Li {
184*05b00f60SXin Li u_int caplen = h->caplen;
185*05b00f60SXin Li u_int length = h->len;
186*05b00f60SXin Li const struct arc_header *ap;
187*05b00f60SXin Li
188*05b00f60SXin Li int phds;
189*05b00f60SXin Li u_int flag = 0, archdrlen = 0;
190*05b00f60SXin Li u_int seqid = 0;
191*05b00f60SXin Li u_char arc_type;
192*05b00f60SXin Li
193*05b00f60SXin Li ndo->ndo_protocol = "arcnet";
194*05b00f60SXin Li if (caplen < ARC_HDRLEN) {
195*05b00f60SXin Li ndo->ndo_ll_hdr_len += caplen;
196*05b00f60SXin Li nd_trunc_longjmp(ndo);
197*05b00f60SXin Li }
198*05b00f60SXin Li
199*05b00f60SXin Li ap = (const struct arc_header *)p;
200*05b00f60SXin Li arc_type = GET_U_1(ap->arc_type);
201*05b00f60SXin Li
202*05b00f60SXin Li switch (arc_type) {
203*05b00f60SXin Li default:
204*05b00f60SXin Li phds = 1;
205*05b00f60SXin Li break;
206*05b00f60SXin Li case ARCTYPE_IP_OLD:
207*05b00f60SXin Li case ARCTYPE_ARP_OLD:
208*05b00f60SXin Li case ARCTYPE_DIAGNOSE:
209*05b00f60SXin Li phds = 0;
210*05b00f60SXin Li archdrlen = ARC_HDRLEN;
211*05b00f60SXin Li break;
212*05b00f60SXin Li }
213*05b00f60SXin Li
214*05b00f60SXin Li if (phds) {
215*05b00f60SXin Li if (caplen < ARC_HDRNEWLEN) {
216*05b00f60SXin Li arcnet_print(ndo, p, length, 0, 0, 0);
217*05b00f60SXin Li ND_PRINT(" phds");
218*05b00f60SXin Li ndo->ndo_ll_hdr_len += caplen;
219*05b00f60SXin Li nd_trunc_longjmp(ndo);
220*05b00f60SXin Li }
221*05b00f60SXin Li
222*05b00f60SXin Li flag = GET_U_1(ap->arc_flag);
223*05b00f60SXin Li if (flag == 0xff) {
224*05b00f60SXin Li if (caplen < ARC_HDRNEWLEN_EXC) {
225*05b00f60SXin Li arcnet_print(ndo, p, length, 0, 0, 0);
226*05b00f60SXin Li ND_PRINT(" phds extended");
227*05b00f60SXin Li ndo->ndo_ll_hdr_len += caplen;
228*05b00f60SXin Li nd_trunc_longjmp(ndo);
229*05b00f60SXin Li }
230*05b00f60SXin Li flag = GET_U_1(ap->arc_flag2);
231*05b00f60SXin Li seqid = GET_BE_U_2(ap->arc_seqid2);
232*05b00f60SXin Li archdrlen = ARC_HDRNEWLEN_EXC;
233*05b00f60SXin Li } else {
234*05b00f60SXin Li seqid = GET_BE_U_2(ap->arc_seqid);
235*05b00f60SXin Li archdrlen = ARC_HDRNEWLEN;
236*05b00f60SXin Li }
237*05b00f60SXin Li }
238*05b00f60SXin Li
239*05b00f60SXin Li
240*05b00f60SXin Li if (ndo->ndo_eflag)
241*05b00f60SXin Li arcnet_print(ndo, p, length, phds, flag, seqid);
242*05b00f60SXin Li
243*05b00f60SXin Li /*
244*05b00f60SXin Li * Go past the ARCNET header.
245*05b00f60SXin Li */
246*05b00f60SXin Li length -= archdrlen;
247*05b00f60SXin Li caplen -= archdrlen;
248*05b00f60SXin Li p += archdrlen;
249*05b00f60SXin Li
250*05b00f60SXin Li if (phds && flag && (flag & 1) == 0) {
251*05b00f60SXin Li /*
252*05b00f60SXin Li * This is a middle fragment.
253*05b00f60SXin Li */
254*05b00f60SXin Li ndo->ndo_ll_hdr_len += archdrlen;
255*05b00f60SXin Li return;
256*05b00f60SXin Li }
257*05b00f60SXin Li
258*05b00f60SXin Li if (!arcnet_encap_print(ndo, arc_type, p, length, caplen))
259*05b00f60SXin Li ND_DEFAULTPRINT(p, caplen);
260*05b00f60SXin Li
261*05b00f60SXin Li ndo->ndo_ll_hdr_len += archdrlen;
262*05b00f60SXin Li }
263*05b00f60SXin Li
264*05b00f60SXin Li /*
265*05b00f60SXin Li * This is the top level routine of the printer. 'p' points
266*05b00f60SXin Li * to the ARCNET header of the packet, 'h->ts' is the timestamp,
267*05b00f60SXin Li * 'h->len' is the length of the packet off the wire, and 'h->caplen'
268*05b00f60SXin Li * is the number of bytes actually captured. It is quite similar
269*05b00f60SXin Li * to the non-Linux style printer except that Linux doesn't ever
270*05b00f60SXin Li * supply packets that look like exception frames, it always supplies
271*05b00f60SXin Li * reassembled packets rather than raw frames, and headers have an
272*05b00f60SXin Li * extra "offset" field between the src/dest and packet type.
273*05b00f60SXin Li */
274*05b00f60SXin Li void
arcnet_linux_if_print(netdissect_options * ndo,const struct pcap_pkthdr * h,const u_char * p)275*05b00f60SXin Li arcnet_linux_if_print(netdissect_options *ndo, const struct pcap_pkthdr *h, const u_char *p)
276*05b00f60SXin Li {
277*05b00f60SXin Li u_int caplen = h->caplen;
278*05b00f60SXin Li u_int length = h->len;
279*05b00f60SXin Li const struct arc_linux_header *ap;
280*05b00f60SXin Li
281*05b00f60SXin Li int archdrlen = 0;
282*05b00f60SXin Li u_char arc_type;
283*05b00f60SXin Li
284*05b00f60SXin Li ndo->ndo_protocol = "arcnet_linux";
285*05b00f60SXin Li if (caplen < ARC_LINUX_HDRLEN) {
286*05b00f60SXin Li ndo->ndo_ll_hdr_len += caplen;
287*05b00f60SXin Li nd_trunc_longjmp(ndo);
288*05b00f60SXin Li }
289*05b00f60SXin Li
290*05b00f60SXin Li ap = (const struct arc_linux_header *)p;
291*05b00f60SXin Li arc_type = GET_U_1(ap->arc_type);
292*05b00f60SXin Li
293*05b00f60SXin Li switch (arc_type) {
294*05b00f60SXin Li default:
295*05b00f60SXin Li archdrlen = ARC_LINUX_HDRNEWLEN;
296*05b00f60SXin Li if (caplen < ARC_LINUX_HDRNEWLEN) {
297*05b00f60SXin Li ndo->ndo_ll_hdr_len += caplen;
298*05b00f60SXin Li nd_trunc_longjmp(ndo);
299*05b00f60SXin Li }
300*05b00f60SXin Li break;
301*05b00f60SXin Li case ARCTYPE_IP_OLD:
302*05b00f60SXin Li case ARCTYPE_ARP_OLD:
303*05b00f60SXin Li case ARCTYPE_DIAGNOSE:
304*05b00f60SXin Li archdrlen = ARC_LINUX_HDRLEN;
305*05b00f60SXin Li break;
306*05b00f60SXin Li }
307*05b00f60SXin Li
308*05b00f60SXin Li if (ndo->ndo_eflag)
309*05b00f60SXin Li arcnet_print(ndo, p, length, 0, 0, 0);
310*05b00f60SXin Li
311*05b00f60SXin Li /*
312*05b00f60SXin Li * Go past the ARCNET header.
313*05b00f60SXin Li */
314*05b00f60SXin Li length -= archdrlen;
315*05b00f60SXin Li caplen -= archdrlen;
316*05b00f60SXin Li p += archdrlen;
317*05b00f60SXin Li
318*05b00f60SXin Li if (!arcnet_encap_print(ndo, arc_type, p, length, caplen))
319*05b00f60SXin Li ND_DEFAULTPRINT(p, caplen);
320*05b00f60SXin Li
321*05b00f60SXin Li ndo->ndo_ll_hdr_len += archdrlen;
322*05b00f60SXin Li }
323*05b00f60SXin Li
324*05b00f60SXin Li /*
325*05b00f60SXin Li * Prints the packet encapsulated in an ARCnet data field,
326*05b00f60SXin Li * given the ARCnet system code.
327*05b00f60SXin Li *
328*05b00f60SXin Li * Returns non-zero if it can do so, zero if the system code is unknown.
329*05b00f60SXin Li */
330*05b00f60SXin Li
331*05b00f60SXin Li
332*05b00f60SXin Li static int
arcnet_encap_print(netdissect_options * ndo,u_char arctype,const u_char * p,u_int length,u_int caplen)333*05b00f60SXin Li arcnet_encap_print(netdissect_options *ndo, u_char arctype, const u_char *p,
334*05b00f60SXin Li u_int length, u_int caplen)
335*05b00f60SXin Li {
336*05b00f60SXin Li switch (arctype) {
337*05b00f60SXin Li
338*05b00f60SXin Li case ARCTYPE_IP_OLD:
339*05b00f60SXin Li case ARCTYPE_IP:
340*05b00f60SXin Li ip_print(ndo, p, length);
341*05b00f60SXin Li return (1);
342*05b00f60SXin Li
343*05b00f60SXin Li case ARCTYPE_INET6:
344*05b00f60SXin Li ip6_print(ndo, p, length);
345*05b00f60SXin Li return (1);
346*05b00f60SXin Li
347*05b00f60SXin Li case ARCTYPE_ARP_OLD:
348*05b00f60SXin Li case ARCTYPE_ARP:
349*05b00f60SXin Li case ARCTYPE_REVARP:
350*05b00f60SXin Li arp_print(ndo, p, length, caplen);
351*05b00f60SXin Li return (1);
352*05b00f60SXin Li
353*05b00f60SXin Li case ARCTYPE_ATALK: /* XXX was this ever used? */
354*05b00f60SXin Li if (ndo->ndo_vflag)
355*05b00f60SXin Li ND_PRINT("et1 ");
356*05b00f60SXin Li atalk_print(ndo, p, length);
357*05b00f60SXin Li return (1);
358*05b00f60SXin Li
359*05b00f60SXin Li case ARCTYPE_IPX:
360*05b00f60SXin Li ipx_print(ndo, p, length);
361*05b00f60SXin Li return (1);
362*05b00f60SXin Li
363*05b00f60SXin Li default:
364*05b00f60SXin Li return (0);
365*05b00f60SXin Li }
366*05b00f60SXin Li }
367