xref: /aosp_15_r20/external/tcpdump/print-arcnet.c (revision 05b00f6010a2396e3db2409989fc67270046269f)
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