xref: /aosp_15_r20/external/tcpdump/print.c (revision 05b00f6010a2396e3db2409989fc67270046269f)
1*05b00f60SXin Li /*
2*05b00f60SXin Li  * Copyright (c) 1988, 1989, 1990, 1991, 1992, 1993, 1994, 1995, 1996, 1997, 2000
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  * Support for splitting captures into multiple files with a maximum
22*05b00f60SXin Li  * file size:
23*05b00f60SXin Li  *
24*05b00f60SXin Li  * Copyright (c) 2001
25*05b00f60SXin Li  *	Seth Webster <[email protected]>
26*05b00f60SXin Li  */
27*05b00f60SXin Li 
28*05b00f60SXin Li #ifdef HAVE_CONFIG_H
29*05b00f60SXin Li #include <config.h>
30*05b00f60SXin Li #endif
31*05b00f60SXin Li 
32*05b00f60SXin Li #include <stdlib.h>
33*05b00f60SXin Li #include <string.h>
34*05b00f60SXin Li #include <setjmp.h>
35*05b00f60SXin Li 
36*05b00f60SXin Li #include "netdissect-stdinc.h"
37*05b00f60SXin Li 
38*05b00f60SXin Li #include "netdissect.h"
39*05b00f60SXin Li #include "addrtoname.h"
40*05b00f60SXin Li #include "print.h"
41*05b00f60SXin Li #include "netdissect-alloc.h"
42*05b00f60SXin Li 
43*05b00f60SXin Li #include "pcap-missing.h"
44*05b00f60SXin Li 
45*05b00f60SXin Li struct printer {
46*05b00f60SXin Li 	if_printer f;
47*05b00f60SXin Li 	int type;
48*05b00f60SXin Li };
49*05b00f60SXin Li 
50*05b00f60SXin Li static const struct printer printers[] = {
51*05b00f60SXin Li #ifdef DLT_APPLE_IP_OVER_IEEE1394
52*05b00f60SXin Li 	{ ap1394_if_print,	DLT_APPLE_IP_OVER_IEEE1394 },
53*05b00f60SXin Li #endif
54*05b00f60SXin Li 	{ arcnet_if_print,	DLT_ARCNET },
55*05b00f60SXin Li #ifdef DLT_ARCNET_LINUX
56*05b00f60SXin Li 	{ arcnet_linux_if_print, DLT_ARCNET_LINUX },
57*05b00f60SXin Li #endif
58*05b00f60SXin Li 	{ atm_if_print,		DLT_ATM_RFC1483 },
59*05b00f60SXin Li #ifdef DLT_DSA_TAG_BRCM
60*05b00f60SXin Li 	{ brcm_tag_if_print,	DLT_DSA_TAG_BRCM },
61*05b00f60SXin Li #endif
62*05b00f60SXin Li #ifdef DLT_DSA_TAG_BRCM_PREPEND
63*05b00f60SXin Li 	{ brcm_tag_prepend_if_print, DLT_DSA_TAG_BRCM_PREPEND },
64*05b00f60SXin Li #endif
65*05b00f60SXin Li #ifdef DLT_BLUETOOTH_HCI_H4_WITH_PHDR
66*05b00f60SXin Li 	{ bt_if_print,		DLT_BLUETOOTH_HCI_H4_WITH_PHDR},
67*05b00f60SXin Li #endif
68*05b00f60SXin Li #ifdef DLT_C_HDLC
69*05b00f60SXin Li 	{ chdlc_if_print,	DLT_C_HDLC },
70*05b00f60SXin Li #endif
71*05b00f60SXin Li #ifdef DLT_HDLC
72*05b00f60SXin Li 	{ chdlc_if_print,	DLT_HDLC },
73*05b00f60SXin Li #endif
74*05b00f60SXin Li #ifdef DLT_ATM_CLIP
75*05b00f60SXin Li 	{ cip_if_print,		DLT_ATM_CLIP },
76*05b00f60SXin Li #endif
77*05b00f60SXin Li #ifdef DLT_CIP
78*05b00f60SXin Li 	{ cip_if_print,		DLT_CIP },
79*05b00f60SXin Li #endif
80*05b00f60SXin Li #ifdef DLT_DSA_TAG_DSA
81*05b00f60SXin Li 	{ dsa_if_print,		DLT_DSA_TAG_DSA },
82*05b00f60SXin Li #endif
83*05b00f60SXin Li #ifdef DLT_DSA_TAG_EDSA
84*05b00f60SXin Li 	{ edsa_if_print,	DLT_DSA_TAG_EDSA },
85*05b00f60SXin Li #endif
86*05b00f60SXin Li #ifdef DLT_ENC
87*05b00f60SXin Li 	{ enc_if_print,		DLT_ENC },
88*05b00f60SXin Li #endif
89*05b00f60SXin Li 	{ ether_if_print,	DLT_EN10MB },
90*05b00f60SXin Li 	{ fddi_if_print,	DLT_FDDI },
91*05b00f60SXin Li #ifdef DLT_FR
92*05b00f60SXin Li 	{ fr_if_print,		DLT_FR },
93*05b00f60SXin Li #endif
94*05b00f60SXin Li #ifdef DLT_FRELAY
95*05b00f60SXin Li 	{ fr_if_print,		DLT_FRELAY },
96*05b00f60SXin Li #endif
97*05b00f60SXin Li #ifdef DLT_IEEE802_11
98*05b00f60SXin Li 	{ ieee802_11_if_print,	DLT_IEEE802_11},
99*05b00f60SXin Li #endif
100*05b00f60SXin Li #ifdef DLT_IEEE802_11_RADIO_AVS
101*05b00f60SXin Li 	{ ieee802_11_radio_avs_if_print, DLT_IEEE802_11_RADIO_AVS },
102*05b00f60SXin Li #endif
103*05b00f60SXin Li #ifdef DLT_IEEE802_11_RADIO
104*05b00f60SXin Li 	{ ieee802_11_radio_if_print,	DLT_IEEE802_11_RADIO },
105*05b00f60SXin Li #endif
106*05b00f60SXin Li #ifdef DLT_IEEE802_15_4
107*05b00f60SXin Li 	{ ieee802_15_4_if_print, DLT_IEEE802_15_4 },
108*05b00f60SXin Li #endif
109*05b00f60SXin Li #ifdef DLT_IEEE802_15_4_NOFCS
110*05b00f60SXin Li 	{ ieee802_15_4_if_print, DLT_IEEE802_15_4_NOFCS },
111*05b00f60SXin Li #endif
112*05b00f60SXin Li #ifdef DLT_IEEE802_15_4_TAP
113*05b00f60SXin Li 	{ ieee802_15_4_tap_if_print, DLT_IEEE802_15_4_TAP },
114*05b00f60SXin Li #endif
115*05b00f60SXin Li #ifdef DLT_IP_OVER_FC
116*05b00f60SXin Li 	{ ipfc_if_print,	DLT_IP_OVER_FC },
117*05b00f60SXin Li #endif
118*05b00f60SXin Li #ifdef DLT_IPNET
119*05b00f60SXin Li 	{ ipnet_if_print,	DLT_IPNET },
120*05b00f60SXin Li #endif
121*05b00f60SXin Li #ifdef DLT_IPOIB
122*05b00f60SXin Li 	{ ipoib_if_print,       DLT_IPOIB },
123*05b00f60SXin Li #endif
124*05b00f60SXin Li #ifdef DLT_JUNIPER_ATM1
125*05b00f60SXin Li 	{ juniper_atm1_if_print, DLT_JUNIPER_ATM1 },
126*05b00f60SXin Li #endif
127*05b00f60SXin Li #ifdef DLT_JUNIPER_ATM2
128*05b00f60SXin Li 	{ juniper_atm2_if_print, DLT_JUNIPER_ATM2 },
129*05b00f60SXin Li #endif
130*05b00f60SXin Li #ifdef DLT_JUNIPER_CHDLC
131*05b00f60SXin Li 	{ juniper_chdlc_if_print,	DLT_JUNIPER_CHDLC },
132*05b00f60SXin Li #endif
133*05b00f60SXin Li #ifdef DLT_JUNIPER_ES
134*05b00f60SXin Li 	{ juniper_es_if_print,	DLT_JUNIPER_ES },
135*05b00f60SXin Li #endif
136*05b00f60SXin Li #ifdef DLT_JUNIPER_ETHER
137*05b00f60SXin Li 	{ juniper_ether_if_print,	DLT_JUNIPER_ETHER },
138*05b00f60SXin Li #endif
139*05b00f60SXin Li #ifdef DLT_JUNIPER_FRELAY
140*05b00f60SXin Li 	{ juniper_frelay_if_print,	DLT_JUNIPER_FRELAY },
141*05b00f60SXin Li #endif
142*05b00f60SXin Li #ifdef DLT_JUNIPER_GGSN
143*05b00f60SXin Li 	{ juniper_ggsn_if_print, DLT_JUNIPER_GGSN },
144*05b00f60SXin Li #endif
145*05b00f60SXin Li #ifdef DLT_JUNIPER_MFR
146*05b00f60SXin Li 	{ juniper_mfr_if_print,	DLT_JUNIPER_MFR },
147*05b00f60SXin Li #endif
148*05b00f60SXin Li #ifdef DLT_JUNIPER_MLFR
149*05b00f60SXin Li 	{ juniper_mlfr_if_print, DLT_JUNIPER_MLFR },
150*05b00f60SXin Li #endif
151*05b00f60SXin Li #ifdef DLT_JUNIPER_MLPPP
152*05b00f60SXin Li 	{ juniper_mlppp_if_print, DLT_JUNIPER_MLPPP },
153*05b00f60SXin Li #endif
154*05b00f60SXin Li #ifdef DLT_JUNIPER_MONITOR
155*05b00f60SXin Li 	{ juniper_monitor_if_print, DLT_JUNIPER_MONITOR },
156*05b00f60SXin Li #endif
157*05b00f60SXin Li #ifdef DLT_JUNIPER_PPP
158*05b00f60SXin Li 	{ juniper_ppp_if_print,	DLT_JUNIPER_PPP },
159*05b00f60SXin Li #endif
160*05b00f60SXin Li #ifdef DLT_JUNIPER_PPPOE_ATM
161*05b00f60SXin Li 	{ juniper_pppoe_atm_if_print, DLT_JUNIPER_PPPOE_ATM },
162*05b00f60SXin Li #endif
163*05b00f60SXin Li #ifdef DLT_JUNIPER_PPPOE
164*05b00f60SXin Li 	{ juniper_pppoe_if_print, DLT_JUNIPER_PPPOE },
165*05b00f60SXin Li #endif
166*05b00f60SXin Li #ifdef DLT_JUNIPER_SERVICES
167*05b00f60SXin Li 	{ juniper_services_if_print, DLT_JUNIPER_SERVICES },
168*05b00f60SXin Li #endif
169*05b00f60SXin Li #ifdef DLT_LTALK
170*05b00f60SXin Li 	{ ltalk_if_print,	DLT_LTALK },
171*05b00f60SXin Li #endif
172*05b00f60SXin Li #ifdef DLT_MFR
173*05b00f60SXin Li 	{ mfr_if_print,		DLT_MFR },
174*05b00f60SXin Li #endif
175*05b00f60SXin Li #ifdef DLT_NETANALYZER
176*05b00f60SXin Li 	{ netanalyzer_if_print, DLT_NETANALYZER },
177*05b00f60SXin Li #endif
178*05b00f60SXin Li #ifdef DLT_NETANALYZER_TRANSPARENT
179*05b00f60SXin Li 	{ netanalyzer_transparent_if_print, DLT_NETANALYZER_TRANSPARENT },
180*05b00f60SXin Li #endif
181*05b00f60SXin Li #ifdef DLT_NFLOG
182*05b00f60SXin Li 	{ nflog_if_print,	DLT_NFLOG},
183*05b00f60SXin Li #endif
184*05b00f60SXin Li 	{ null_if_print,	DLT_NULL },
185*05b00f60SXin Li #ifdef DLT_LOOP
186*05b00f60SXin Li 	{ null_if_print,	DLT_LOOP },
187*05b00f60SXin Li #endif
188*05b00f60SXin Li #ifdef DLT_PFLOG
189*05b00f60SXin Li 	{ pflog_if_print,	DLT_PFLOG },
190*05b00f60SXin Li #endif
191*05b00f60SXin Li #ifdef DLT_PKTAP
192*05b00f60SXin Li 	{ pktap_if_print,	DLT_PKTAP },
193*05b00f60SXin Li #endif
194*05b00f60SXin Li #ifdef DLT_PPI
195*05b00f60SXin Li 	{ ppi_if_print,		DLT_PPI },
196*05b00f60SXin Li #endif
197*05b00f60SXin Li #ifdef DLT_PPP_BSDOS
198*05b00f60SXin Li 	{ ppp_bsdos_if_print,	DLT_PPP_BSDOS },
199*05b00f60SXin Li #endif
200*05b00f60SXin Li #ifdef DLT_PPP_SERIAL
201*05b00f60SXin Li 	{ ppp_hdlc_if_print,	DLT_PPP_SERIAL },
202*05b00f60SXin Li #endif
203*05b00f60SXin Li 	{ ppp_if_print,		DLT_PPP },
204*05b00f60SXin Li #ifdef DLT_PPP_PPPD
205*05b00f60SXin Li 	{ ppp_if_print,		DLT_PPP_PPPD },
206*05b00f60SXin Li #endif
207*05b00f60SXin Li #ifdef DLT_PPP_ETHER
208*05b00f60SXin Li 	{ pppoe_if_print,	DLT_PPP_ETHER },
209*05b00f60SXin Li #endif
210*05b00f60SXin Li #ifdef DLT_PRISM_HEADER
211*05b00f60SXin Li 	{ prism_if_print,	DLT_PRISM_HEADER },
212*05b00f60SXin Li #endif
213*05b00f60SXin Li 	{ raw_if_print,		DLT_RAW },
214*05b00f60SXin Li #ifdef DLT_IPV4
215*05b00f60SXin Li 	{ raw_if_print,		DLT_IPV4 },
216*05b00f60SXin Li #endif
217*05b00f60SXin Li #ifdef DLT_IPV6
218*05b00f60SXin Li 	{ raw_if_print,		DLT_IPV6 },
219*05b00f60SXin Li #endif
220*05b00f60SXin Li #ifdef DLT_SLIP_BSDOS
221*05b00f60SXin Li 	{ sl_bsdos_if_print,	DLT_SLIP_BSDOS },
222*05b00f60SXin Li #endif
223*05b00f60SXin Li 	{ sl_if_print,		DLT_SLIP },
224*05b00f60SXin Li #ifdef DLT_LINUX_SLL
225*05b00f60SXin Li 	{ sll_if_print,		DLT_LINUX_SLL },
226*05b00f60SXin Li #endif
227*05b00f60SXin Li #ifdef DLT_LINUX_SLL2
228*05b00f60SXin Li 	{ sll2_if_print,	DLT_LINUX_SLL2 },
229*05b00f60SXin Li #endif
230*05b00f60SXin Li #ifdef DLT_SUNATM
231*05b00f60SXin Li 	{ sunatm_if_print,	DLT_SUNATM },
232*05b00f60SXin Li #endif
233*05b00f60SXin Li #ifdef DLT_SYMANTEC_FIREWALL
234*05b00f60SXin Li 	{ symantec_if_print,	DLT_SYMANTEC_FIREWALL },
235*05b00f60SXin Li #endif
236*05b00f60SXin Li 	{ token_if_print,	DLT_IEEE802 },
237*05b00f60SXin Li #ifdef DLT_USB_LINUX
238*05b00f60SXin Li 	{ usb_linux_48_byte_if_print, DLT_USB_LINUX},
239*05b00f60SXin Li #endif /* DLT_USB_LINUX */
240*05b00f60SXin Li #ifdef DLT_USB_LINUX_MMAPPED
241*05b00f60SXin Li 	{ usb_linux_64_byte_if_print, DLT_USB_LINUX_MMAPPED},
242*05b00f60SXin Li #endif /* DLT_USB_LINUX_MMAPPED */
243*05b00f60SXin Li #ifdef DLT_VSOCK
244*05b00f60SXin Li 	{ vsock_if_print,	DLT_VSOCK },
245*05b00f60SXin Li #endif
246*05b00f60SXin Li 	{ NULL,                 0 },
247*05b00f60SXin Li };
248*05b00f60SXin Li 
249*05b00f60SXin Li static void	ndo_default_print(netdissect_options *ndo, const u_char *bp,
250*05b00f60SXin Li 		    u_int length);
251*05b00f60SXin Li 
252*05b00f60SXin Li static void NORETURN ndo_error(netdissect_options *ndo,
253*05b00f60SXin Li 		     status_exit_codes_t status,
254*05b00f60SXin Li 		     FORMAT_STRING(const char *fmt), ...)
255*05b00f60SXin Li 		     PRINTFLIKE(3, 4);
256*05b00f60SXin Li static void	ndo_warning(netdissect_options *ndo,
257*05b00f60SXin Li 		    FORMAT_STRING(const char *fmt), ...)
258*05b00f60SXin Li 		    PRINTFLIKE(2, 3);
259*05b00f60SXin Li 
260*05b00f60SXin Li static int	ndo_printf(netdissect_options *ndo,
261*05b00f60SXin Li 		     FORMAT_STRING(const char *fmt), ...)
262*05b00f60SXin Li 		     PRINTFLIKE(2, 3);
263*05b00f60SXin Li 
264*05b00f60SXin Li void
init_print(netdissect_options * ndo,uint32_t localnet,uint32_t mask)265*05b00f60SXin Li init_print(netdissect_options *ndo, uint32_t localnet, uint32_t mask)
266*05b00f60SXin Li {
267*05b00f60SXin Li 
268*05b00f60SXin Li 	init_addrtoname(ndo, localnet, mask);
269*05b00f60SXin Li 	init_checksum();
270*05b00f60SXin Li }
271*05b00f60SXin Li 
272*05b00f60SXin Li if_printer
lookup_printer(int type)273*05b00f60SXin Li lookup_printer(int type)
274*05b00f60SXin Li {
275*05b00f60SXin Li 	const struct printer *p;
276*05b00f60SXin Li 
277*05b00f60SXin Li 	for (p = printers; p->f; ++p)
278*05b00f60SXin Li 		if (type == p->type)
279*05b00f60SXin Li 			return p->f;
280*05b00f60SXin Li 
281*05b00f60SXin Li #if defined(DLT_USER2) && defined(DLT_PKTAP)
282*05b00f60SXin Li 	/*
283*05b00f60SXin Li 	 * Apple incorrectly chose to use DLT_USER2 for their PKTAP
284*05b00f60SXin Li 	 * header.
285*05b00f60SXin Li 	 *
286*05b00f60SXin Li 	 * We map DLT_PKTAP, whether it's DLT_USER2 as it is on Darwin-
287*05b00f60SXin Li 	 * based OSes or the same value as LINKTYPE_PKTAP as it is on
288*05b00f60SXin Li 	 * other OSes, to LINKTYPE_PKTAP, so files written with
289*05b00f60SXin Li 	 * this version of libpcap for a DLT_PKTAP capture have a link-
290*05b00f60SXin Li 	 * layer header type of LINKTYPE_PKTAP.
291*05b00f60SXin Li 	 *
292*05b00f60SXin Li 	 * However, files written on OS X Mavericks for a DLT_PKTAP
293*05b00f60SXin Li 	 * capture have a link-layer header type of LINKTYPE_USER2.
294*05b00f60SXin Li 	 * If we don't have a printer for DLT_USER2, and type is
295*05b00f60SXin Li 	 * DLT_USER2, we look up the printer for DLT_PKTAP and use
296*05b00f60SXin Li 	 * that.
297*05b00f60SXin Li 	 */
298*05b00f60SXin Li 	if (type == DLT_USER2) {
299*05b00f60SXin Li 		for (p = printers; p->f; ++p)
300*05b00f60SXin Li 			if (DLT_PKTAP == p->type)
301*05b00f60SXin Li 				return p->f;
302*05b00f60SXin Li 	}
303*05b00f60SXin Li #endif
304*05b00f60SXin Li 
305*05b00f60SXin Li 	return NULL;
306*05b00f60SXin Li 	/* NOTREACHED */
307*05b00f60SXin Li }
308*05b00f60SXin Li 
309*05b00f60SXin Li int
has_printer(int type)310*05b00f60SXin Li has_printer(int type)
311*05b00f60SXin Li {
312*05b00f60SXin Li 	return (lookup_printer(type) != NULL);
313*05b00f60SXin Li }
314*05b00f60SXin Li 
315*05b00f60SXin Li if_printer
get_if_printer(int type)316*05b00f60SXin Li get_if_printer(int type)
317*05b00f60SXin Li {
318*05b00f60SXin Li 	if_printer printer;
319*05b00f60SXin Li 
320*05b00f60SXin Li 	printer = lookup_printer(type);
321*05b00f60SXin Li 	if (printer == NULL)
322*05b00f60SXin Li 		printer = unsupported_if_print;
323*05b00f60SXin Li 	return printer;
324*05b00f60SXin Li }
325*05b00f60SXin Li 
326*05b00f60SXin Li void
pretty_print_packet(netdissect_options * ndo,const struct pcap_pkthdr * h,const u_char * sp,u_int packets_captured)327*05b00f60SXin Li pretty_print_packet(netdissect_options *ndo, const struct pcap_pkthdr *h,
328*05b00f60SXin Li 		    const u_char *sp, u_int packets_captured)
329*05b00f60SXin Li {
330*05b00f60SXin Li 	u_int hdrlen = 0;
331*05b00f60SXin Li 	int invalid_header = 0;
332*05b00f60SXin Li 
333*05b00f60SXin Li 	if (ndo->ndo_packet_number)
334*05b00f60SXin Li 		ND_PRINT("%5u  ", packets_captured);
335*05b00f60SXin Li 
336*05b00f60SXin Li 	/* Sanity checks on packet length / capture length */
337*05b00f60SXin Li 	if (h->caplen == 0) {
338*05b00f60SXin Li 		invalid_header = 1;
339*05b00f60SXin Li 		ND_PRINT("[Invalid header: caplen==0");
340*05b00f60SXin Li 	}
341*05b00f60SXin Li 	if (h->len == 0) {
342*05b00f60SXin Li 		if (!invalid_header) {
343*05b00f60SXin Li 			invalid_header = 1;
344*05b00f60SXin Li 			ND_PRINT("[Invalid header:");
345*05b00f60SXin Li 		} else
346*05b00f60SXin Li 			ND_PRINT(",");
347*05b00f60SXin Li 		ND_PRINT(" len==0");
348*05b00f60SXin Li 	} else if (h->len < h->caplen) {
349*05b00f60SXin Li 		if (!invalid_header) {
350*05b00f60SXin Li 			invalid_header = 1;
351*05b00f60SXin Li 			ND_PRINT("[Invalid header:");
352*05b00f60SXin Li 		} else
353*05b00f60SXin Li 			ND_PRINT(",");
354*05b00f60SXin Li 		ND_PRINT(" len(%u) < caplen(%u)", h->len, h->caplen);
355*05b00f60SXin Li 	}
356*05b00f60SXin Li 	if (h->caplen > MAXIMUM_SNAPLEN) {
357*05b00f60SXin Li 		if (!invalid_header) {
358*05b00f60SXin Li 			invalid_header = 1;
359*05b00f60SXin Li 			ND_PRINT("[Invalid header:");
360*05b00f60SXin Li 		} else
361*05b00f60SXin Li 			ND_PRINT(",");
362*05b00f60SXin Li 		ND_PRINT(" caplen(%u) > %u", h->caplen, MAXIMUM_SNAPLEN);
363*05b00f60SXin Li 	}
364*05b00f60SXin Li 	if (h->len > MAXIMUM_SNAPLEN) {
365*05b00f60SXin Li 		if (!invalid_header) {
366*05b00f60SXin Li 			invalid_header = 1;
367*05b00f60SXin Li 			ND_PRINT("[Invalid header:");
368*05b00f60SXin Li 		} else
369*05b00f60SXin Li 			ND_PRINT(",");
370*05b00f60SXin Li 		ND_PRINT(" len(%u) > %u", h->len, MAXIMUM_SNAPLEN);
371*05b00f60SXin Li 	}
372*05b00f60SXin Li 	if (invalid_header) {
373*05b00f60SXin Li 		ND_PRINT("]\n");
374*05b00f60SXin Li 		return;
375*05b00f60SXin Li 	}
376*05b00f60SXin Li 
377*05b00f60SXin Li 	/*
378*05b00f60SXin Li 	 * At this point:
379*05b00f60SXin Li 	 *   capture length != 0,
380*05b00f60SXin Li 	 *   packet length != 0,
381*05b00f60SXin Li 	 *   capture length <= MAXIMUM_SNAPLEN,
382*05b00f60SXin Li 	 *   packet length <= MAXIMUM_SNAPLEN,
383*05b00f60SXin Li 	 *   packet length >= capture length.
384*05b00f60SXin Li 	 *
385*05b00f60SXin Li 	 * Currently, there is no D-Bus printer, thus no need for
386*05b00f60SXin Li 	 * bigger lengths.
387*05b00f60SXin Li 	 */
388*05b00f60SXin Li 
389*05b00f60SXin Li 	/*
390*05b00f60SXin Li 	 * The header /usr/include/pcap/pcap.h in OpenBSD declares h->ts as
391*05b00f60SXin Li 	 * struct bpf_timeval, not struct timeval. The former comes from
392*05b00f60SXin Li 	 * /usr/include/net/bpf.h and uses 32-bit unsigned types instead of
393*05b00f60SXin Li 	 * the types used in struct timeval.
394*05b00f60SXin Li 	 */
395*05b00f60SXin Li 	struct timeval tvbuf;
396*05b00f60SXin Li 	tvbuf.tv_sec = h->ts.tv_sec;
397*05b00f60SXin Li 	tvbuf.tv_usec = h->ts.tv_usec;
398*05b00f60SXin Li 	ts_print(ndo, &tvbuf);
399*05b00f60SXin Li 
400*05b00f60SXin Li 	/*
401*05b00f60SXin Li 	 * Printers must check that they're not walking off the end of
402*05b00f60SXin Li 	 * the packet.
403*05b00f60SXin Li 	 * Rather than pass it all the way down, we set this member
404*05b00f60SXin Li 	 * of the netdissect_options structure.
405*05b00f60SXin Li 	 */
406*05b00f60SXin Li 	ndo->ndo_snapend = sp + h->caplen;
407*05b00f60SXin Li 	ndo->ndo_packetp = sp;
408*05b00f60SXin Li 
409*05b00f60SXin Li 	ndo->ndo_protocol = "";
410*05b00f60SXin Li 	ndo->ndo_ll_hdr_len = 0;
411*05b00f60SXin Li 	switch (setjmp(ndo->ndo_early_end)) {
412*05b00f60SXin Li 	case 0:
413*05b00f60SXin Li 		/* Print the packet. */
414*05b00f60SXin Li 		(ndo->ndo_if_printer)(ndo, h, sp);
415*05b00f60SXin Li 		break;
416*05b00f60SXin Li 	case ND_TRUNCATED:
417*05b00f60SXin Li 		/* A printer quit because the packet was truncated; report it */
418*05b00f60SXin Li 		nd_print_trunc(ndo);
419*05b00f60SXin Li 		/* Print the full packet */
420*05b00f60SXin Li 		ndo->ndo_ll_hdr_len = 0;
421*05b00f60SXin Li 		break;
422*05b00f60SXin Li 	}
423*05b00f60SXin Li 	hdrlen = ndo->ndo_ll_hdr_len;
424*05b00f60SXin Li 
425*05b00f60SXin Li 	/*
426*05b00f60SXin Li 	 * Empty the stack of packet information, freeing all pushed buffers;
427*05b00f60SXin Li 	 * if we got here by a printer quitting, we need to release anything
428*05b00f60SXin Li 	 * that didn't get released because we longjmped out of the code
429*05b00f60SXin Li 	 * before it popped the packet information.
430*05b00f60SXin Li 	 */
431*05b00f60SXin Li 	nd_pop_all_packet_info(ndo);
432*05b00f60SXin Li 
433*05b00f60SXin Li 	/*
434*05b00f60SXin Li 	 * Restore the original snapend, as a printer might have
435*05b00f60SXin Li 	 * changed it.
436*05b00f60SXin Li 	 */
437*05b00f60SXin Li 	ndo->ndo_snapend = sp + h->caplen;
438*05b00f60SXin Li 	if (ndo->ndo_Xflag) {
439*05b00f60SXin Li 		/*
440*05b00f60SXin Li 		 * Print the raw packet data in hex and ASCII.
441*05b00f60SXin Li 		 */
442*05b00f60SXin Li 		if (ndo->ndo_Xflag > 1) {
443*05b00f60SXin Li 			/*
444*05b00f60SXin Li 			 * Include the link-layer header.
445*05b00f60SXin Li 			 */
446*05b00f60SXin Li 			hex_and_ascii_print(ndo, "\n\t", sp, h->caplen);
447*05b00f60SXin Li 		} else {
448*05b00f60SXin Li 			/*
449*05b00f60SXin Li 			 * Don't include the link-layer header - and if
450*05b00f60SXin Li 			 * we have nothing past the link-layer header,
451*05b00f60SXin Li 			 * print nothing.
452*05b00f60SXin Li 			 */
453*05b00f60SXin Li 			if (h->caplen > hdrlen)
454*05b00f60SXin Li 				hex_and_ascii_print(ndo, "\n\t", sp + hdrlen,
455*05b00f60SXin Li 						    h->caplen - hdrlen);
456*05b00f60SXin Li 		}
457*05b00f60SXin Li 	} else if (ndo->ndo_xflag) {
458*05b00f60SXin Li 		/*
459*05b00f60SXin Li 		 * Print the raw packet data in hex.
460*05b00f60SXin Li 		 */
461*05b00f60SXin Li 		if (ndo->ndo_xflag > 1) {
462*05b00f60SXin Li 			/*
463*05b00f60SXin Li 			 * Include the link-layer header.
464*05b00f60SXin Li 			 */
465*05b00f60SXin Li 			hex_print(ndo, "\n\t", sp, h->caplen);
466*05b00f60SXin Li 		} else {
467*05b00f60SXin Li 			/*
468*05b00f60SXin Li 			 * Don't include the link-layer header - and if
469*05b00f60SXin Li 			 * we have nothing past the link-layer header,
470*05b00f60SXin Li 			 * print nothing.
471*05b00f60SXin Li 			 */
472*05b00f60SXin Li 			if (h->caplen > hdrlen)
473*05b00f60SXin Li 				hex_print(ndo, "\n\t", sp + hdrlen,
474*05b00f60SXin Li 					  h->caplen - hdrlen);
475*05b00f60SXin Li 		}
476*05b00f60SXin Li 	} else if (ndo->ndo_Aflag) {
477*05b00f60SXin Li 		/*
478*05b00f60SXin Li 		 * Print the raw packet data in ASCII.
479*05b00f60SXin Li 		 */
480*05b00f60SXin Li 		if (ndo->ndo_Aflag > 1) {
481*05b00f60SXin Li 			/*
482*05b00f60SXin Li 			 * Include the link-layer header.
483*05b00f60SXin Li 			 */
484*05b00f60SXin Li 			ascii_print(ndo, sp, h->caplen);
485*05b00f60SXin Li 		} else {
486*05b00f60SXin Li 			/*
487*05b00f60SXin Li 			 * Don't include the link-layer header - and if
488*05b00f60SXin Li 			 * we have nothing past the link-layer header,
489*05b00f60SXin Li 			 * print nothing.
490*05b00f60SXin Li 			 */
491*05b00f60SXin Li 			if (h->caplen > hdrlen)
492*05b00f60SXin Li 				ascii_print(ndo, sp + hdrlen, h->caplen - hdrlen);
493*05b00f60SXin Li 		}
494*05b00f60SXin Li 	}
495*05b00f60SXin Li 
496*05b00f60SXin Li 	ND_PRINT("\n");
497*05b00f60SXin Li 	nd_free_all(ndo);
498*05b00f60SXin Li }
499*05b00f60SXin Li 
500*05b00f60SXin Li /*
501*05b00f60SXin Li  * By default, print the specified data out in hex and ASCII.
502*05b00f60SXin Li  */
503*05b00f60SXin Li static void
ndo_default_print(netdissect_options * ndo,const u_char * bp,u_int length)504*05b00f60SXin Li ndo_default_print(netdissect_options *ndo, const u_char *bp, u_int length)
505*05b00f60SXin Li {
506*05b00f60SXin Li 	hex_and_ascii_print(ndo, "\n\t", bp, length); /* pass on lf and indentation string */
507*05b00f60SXin Li }
508*05b00f60SXin Li 
509*05b00f60SXin Li /* VARARGS */
510*05b00f60SXin Li static void
ndo_error(netdissect_options * ndo,status_exit_codes_t status,const char * fmt,...)511*05b00f60SXin Li ndo_error(netdissect_options *ndo, status_exit_codes_t status,
512*05b00f60SXin Li 	  const char *fmt, ...)
513*05b00f60SXin Li {
514*05b00f60SXin Li 	va_list ap;
515*05b00f60SXin Li 
516*05b00f60SXin Li 	if (ndo->program_name)
517*05b00f60SXin Li 		(void)fprintf(stderr, "%s: ", ndo->program_name);
518*05b00f60SXin Li 	va_start(ap, fmt);
519*05b00f60SXin Li 	(void)vfprintf(stderr, fmt, ap);
520*05b00f60SXin Li 	va_end(ap);
521*05b00f60SXin Li 	if (*fmt) {
522*05b00f60SXin Li 		fmt += strlen(fmt);
523*05b00f60SXin Li 		if (fmt[-1] != '\n')
524*05b00f60SXin Li 			(void)fputc('\n', stderr);
525*05b00f60SXin Li 	}
526*05b00f60SXin Li 	nd_cleanup();
527*05b00f60SXin Li 	exit(status);
528*05b00f60SXin Li 	/* NOTREACHED */
529*05b00f60SXin Li }
530*05b00f60SXin Li 
531*05b00f60SXin Li /* VARARGS */
532*05b00f60SXin Li static void
ndo_warning(netdissect_options * ndo,const char * fmt,...)533*05b00f60SXin Li ndo_warning(netdissect_options *ndo, const char *fmt, ...)
534*05b00f60SXin Li {
535*05b00f60SXin Li 	va_list ap;
536*05b00f60SXin Li 
537*05b00f60SXin Li 	if (ndo->program_name)
538*05b00f60SXin Li 		(void)fprintf(stderr, "%s: ", ndo->program_name);
539*05b00f60SXin Li 	(void)fprintf(stderr, "WARNING: ");
540*05b00f60SXin Li 	va_start(ap, fmt);
541*05b00f60SXin Li 	(void)vfprintf(stderr, fmt, ap);
542*05b00f60SXin Li 	va_end(ap);
543*05b00f60SXin Li 	if (*fmt) {
544*05b00f60SXin Li 		fmt += strlen(fmt);
545*05b00f60SXin Li 		if (fmt[-1] != '\n')
546*05b00f60SXin Li 			(void)fputc('\n', stderr);
547*05b00f60SXin Li 	}
548*05b00f60SXin Li }
549*05b00f60SXin Li 
550*05b00f60SXin Li static int
ndo_printf(netdissect_options * ndo,const char * fmt,...)551*05b00f60SXin Li ndo_printf(netdissect_options *ndo, const char *fmt, ...)
552*05b00f60SXin Li {
553*05b00f60SXin Li 	va_list args;
554*05b00f60SXin Li 	int ret;
555*05b00f60SXin Li 
556*05b00f60SXin Li 	va_start(args, fmt);
557*05b00f60SXin Li 	ret = vfprintf(stdout, fmt, args);
558*05b00f60SXin Li 	va_end(args);
559*05b00f60SXin Li 
560*05b00f60SXin Li 	if (ret < 0)
561*05b00f60SXin Li 		ndo_error(ndo, S_ERR_ND_WRITE_FILE,
562*05b00f60SXin Li 			  "Unable to write output: %s", pcap_strerror(errno));
563*05b00f60SXin Li 	return (ret);
564*05b00f60SXin Li }
565*05b00f60SXin Li 
566*05b00f60SXin Li void
ndo_set_function_pointers(netdissect_options * ndo)567*05b00f60SXin Li ndo_set_function_pointers(netdissect_options *ndo)
568*05b00f60SXin Li {
569*05b00f60SXin Li 	ndo->ndo_default_print=ndo_default_print;
570*05b00f60SXin Li 	ndo->ndo_printf=ndo_printf;
571*05b00f60SXin Li 	ndo->ndo_error=ndo_error;
572*05b00f60SXin Li 	ndo->ndo_warning=ndo_warning;
573*05b00f60SXin Li }
574