xref: /aosp_15_r20/external/tcpdump/print-sflow.c (revision 05b00f6010a2396e3db2409989fc67270046269f)
1*05b00f60SXin Li /*
2*05b00f60SXin Li  * Copyright (c) 1998-2007 The TCPDUMP project
3*05b00f60SXin Li  *
4*05b00f60SXin Li  * Redistribution and use in source and binary forms, with or without
5*05b00f60SXin Li  * modification, are permitted provided that: (1) source code
6*05b00f60SXin Li  * distributions retain the above copyright notice and this paragraph
7*05b00f60SXin Li  * in its entirety, and (2) distributions including binary code include
8*05b00f60SXin Li  * the above copyright notice and this paragraph in its entirety in
9*05b00f60SXin Li  * the documentation or other materials provided with the distribution.
10*05b00f60SXin Li  * THIS SOFTWARE IS PROVIDED ``AS IS'' AND
11*05b00f60SXin Li  * WITHOUT ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, WITHOUT
12*05b00f60SXin Li  * LIMITATION, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
13*05b00f60SXin Li  * FOR A PARTICULAR PURPOSE.
14*05b00f60SXin Li  *
15*05b00f60SXin Li  * Original code by Carles Kishimoto <[email protected]>
16*05b00f60SXin Li  *
17*05b00f60SXin Li  * Expansion and refactoring by Rick Jones <[email protected]>
18*05b00f60SXin Li  */
19*05b00f60SXin Li 
20*05b00f60SXin Li /* \summary: sFlow protocol printer */
21*05b00f60SXin Li 
22*05b00f60SXin Li /* specification: https://sflow.org/developers/specifications.php */
23*05b00f60SXin Li 
24*05b00f60SXin Li #ifdef HAVE_CONFIG_H
25*05b00f60SXin Li #include <config.h>
26*05b00f60SXin Li #endif
27*05b00f60SXin Li 
28*05b00f60SXin Li #include "netdissect-stdinc.h"
29*05b00f60SXin Li 
30*05b00f60SXin Li #define ND_LONGJMP_FROM_TCHECK
31*05b00f60SXin Li #include "netdissect.h"
32*05b00f60SXin Li #include "extract.h"
33*05b00f60SXin Li #include "addrtoname.h"
34*05b00f60SXin Li 
35*05b00f60SXin Li /*
36*05b00f60SXin Li  * sFlow datagram
37*05b00f60SXin Li  *
38*05b00f60SXin Li  * 0                   1                   2                   3
39*05b00f60SXin Li  * 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
40*05b00f60SXin Li  * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
41*05b00f60SXin Li  * |                     Sflow version (2,4,5)                     |
42*05b00f60SXin Li  * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
43*05b00f60SXin Li  * |               IP version (1 for IPv4 | 2 for IPv6)            |
44*05b00f60SXin Li  * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
45*05b00f60SXin Li  * |                     IP Address AGENT (4 or 16 bytes)          |
46*05b00f60SXin Li  * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
47*05b00f60SXin Li  * |                          Sub agent ID                         |
48*05b00f60SXin Li  * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
49*05b00f60SXin Li  * |                      Datagram sequence number                 |
50*05b00f60SXin Li  * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
51*05b00f60SXin Li  * |                      Switch uptime in ms                      |
52*05b00f60SXin Li  * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
53*05b00f60SXin Li  * |                    num samples in datagram                    |
54*05b00f60SXin Li  * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
55*05b00f60SXin Li  *
56*05b00f60SXin Li  */
57*05b00f60SXin Li 
58*05b00f60SXin Li struct sflow_datagram_t {
59*05b00f60SXin Li     nd_uint32_t version;
60*05b00f60SXin Li     nd_uint32_t ip_version;
61*05b00f60SXin Li     nd_ipv4	agent;
62*05b00f60SXin Li     nd_uint32_t	agent_id;
63*05b00f60SXin Li     nd_uint32_t	seqnum;
64*05b00f60SXin Li     nd_uint32_t	uptime;
65*05b00f60SXin Li     nd_uint32_t	samples;
66*05b00f60SXin Li };
67*05b00f60SXin Li 
68*05b00f60SXin Li struct sflow_v6_datagram_t {
69*05b00f60SXin Li     nd_uint32_t version;
70*05b00f60SXin Li     nd_uint32_t ip_version;
71*05b00f60SXin Li     nd_ipv6     agent;
72*05b00f60SXin Li     nd_uint32_t	agent_id;
73*05b00f60SXin Li     nd_uint32_t	seqnum;
74*05b00f60SXin Li     nd_uint32_t	uptime;
75*05b00f60SXin Li     nd_uint32_t	samples;
76*05b00f60SXin Li };
77*05b00f60SXin Li 
78*05b00f60SXin Li struct sflow_sample_header {
79*05b00f60SXin Li     nd_uint32_t	format;
80*05b00f60SXin Li     nd_uint32_t	len;
81*05b00f60SXin Li };
82*05b00f60SXin Li 
83*05b00f60SXin Li #define		SFLOW_FLOW_SAMPLE		1
84*05b00f60SXin Li #define		SFLOW_COUNTER_SAMPLE		2
85*05b00f60SXin Li #define		SFLOW_EXPANDED_FLOW_SAMPLE	3
86*05b00f60SXin Li #define		SFLOW_EXPANDED_COUNTER_SAMPLE	4
87*05b00f60SXin Li 
88*05b00f60SXin Li static const struct tok sflow_format_values[] = {
89*05b00f60SXin Li     { SFLOW_FLOW_SAMPLE, "flow sample" },
90*05b00f60SXin Li     { SFLOW_COUNTER_SAMPLE, "counter sample" },
91*05b00f60SXin Li     { SFLOW_EXPANDED_FLOW_SAMPLE, "expanded flow sample" },
92*05b00f60SXin Li     { SFLOW_EXPANDED_COUNTER_SAMPLE, "expanded counter sample" },
93*05b00f60SXin Li     { 0, NULL}
94*05b00f60SXin Li };
95*05b00f60SXin Li 
96*05b00f60SXin Li struct sflow_flow_sample_t {
97*05b00f60SXin Li     nd_uint32_t seqnum;
98*05b00f60SXin Li     nd_uint8_t  type;
99*05b00f60SXin Li     nd_uint24_t index;
100*05b00f60SXin Li     nd_uint32_t rate;
101*05b00f60SXin Li     nd_uint32_t pool;
102*05b00f60SXin Li     nd_uint32_t drops;
103*05b00f60SXin Li     nd_uint32_t in_interface;
104*05b00f60SXin Li     nd_uint32_t out_interface;
105*05b00f60SXin Li     nd_uint32_t records;
106*05b00f60SXin Li 
107*05b00f60SXin Li };
108*05b00f60SXin Li 
109*05b00f60SXin Li struct sflow_expanded_flow_sample_t {
110*05b00f60SXin Li     nd_uint32_t seqnum;
111*05b00f60SXin Li     nd_uint32_t type;
112*05b00f60SXin Li     nd_uint32_t index;
113*05b00f60SXin Li     nd_uint32_t rate;
114*05b00f60SXin Li     nd_uint32_t pool;
115*05b00f60SXin Li     nd_uint32_t drops;
116*05b00f60SXin Li     nd_uint32_t in_interface_format;
117*05b00f60SXin Li     nd_uint32_t in_interface_value;
118*05b00f60SXin Li     nd_uint32_t out_interface_format;
119*05b00f60SXin Li     nd_uint32_t out_interface_value;
120*05b00f60SXin Li     nd_uint32_t records;
121*05b00f60SXin Li };
122*05b00f60SXin Li 
123*05b00f60SXin Li #define	SFLOW_FLOW_RAW_PACKET			1
124*05b00f60SXin Li #define	SFLOW_FLOW_ETHERNET_FRAME		2
125*05b00f60SXin Li #define	SFLOW_FLOW_IPV4_DATA			3
126*05b00f60SXin Li #define	SFLOW_FLOW_IPV6_DATA			4
127*05b00f60SXin Li #define	SFLOW_FLOW_EXTENDED_SWITCH_DATA		1001
128*05b00f60SXin Li #define	SFLOW_FLOW_EXTENDED_ROUTER_DATA		1002
129*05b00f60SXin Li #define	SFLOW_FLOW_EXTENDED_GATEWAY_DATA	1003
130*05b00f60SXin Li #define	SFLOW_FLOW_EXTENDED_USER_DATA		1004
131*05b00f60SXin Li #define	SFLOW_FLOW_EXTENDED_URL_DATA		1005
132*05b00f60SXin Li #define	SFLOW_FLOW_EXTENDED_MPLS_DATA		1006
133*05b00f60SXin Li #define	SFLOW_FLOW_EXTENDED_NAT_DATA		1007
134*05b00f60SXin Li #define	SFLOW_FLOW_EXTENDED_MPLS_TUNNEL		1008
135*05b00f60SXin Li #define	SFLOW_FLOW_EXTENDED_MPLS_VC		1009
136*05b00f60SXin Li #define	SFLOW_FLOW_EXTENDED_MPLS_FEC		1010
137*05b00f60SXin Li #define	SFLOW_FLOW_EXTENDED_MPLS_LVP_FEC	1011
138*05b00f60SXin Li #define	SFLOW_FLOW_EXTENDED_VLAN_TUNNEL		1012
139*05b00f60SXin Li 
140*05b00f60SXin Li static const struct tok sflow_flow_type_values[] = {
141*05b00f60SXin Li     { SFLOW_FLOW_RAW_PACKET, "Raw packet"},
142*05b00f60SXin Li     { SFLOW_FLOW_ETHERNET_FRAME, "Ethernet frame"},
143*05b00f60SXin Li     { SFLOW_FLOW_IPV4_DATA, "IPv4 Data"},
144*05b00f60SXin Li     { SFLOW_FLOW_IPV6_DATA, "IPv6 Data"},
145*05b00f60SXin Li     { SFLOW_FLOW_EXTENDED_SWITCH_DATA, "Extended Switch data"},
146*05b00f60SXin Li     { SFLOW_FLOW_EXTENDED_ROUTER_DATA, "Extended Router data"},
147*05b00f60SXin Li     { SFLOW_FLOW_EXTENDED_GATEWAY_DATA, "Extended Gateway data"},
148*05b00f60SXin Li     { SFLOW_FLOW_EXTENDED_USER_DATA, "Extended User data"},
149*05b00f60SXin Li     { SFLOW_FLOW_EXTENDED_URL_DATA, "Extended URL data"},
150*05b00f60SXin Li     { SFLOW_FLOW_EXTENDED_MPLS_DATA, "Extended MPLS data"},
151*05b00f60SXin Li     { SFLOW_FLOW_EXTENDED_NAT_DATA, "Extended NAT data"},
152*05b00f60SXin Li     { SFLOW_FLOW_EXTENDED_MPLS_TUNNEL, "Extended MPLS tunnel"},
153*05b00f60SXin Li     { SFLOW_FLOW_EXTENDED_MPLS_VC, "Extended MPLS VC"},
154*05b00f60SXin Li     { SFLOW_FLOW_EXTENDED_MPLS_FEC, "Extended MPLS FEC"},
155*05b00f60SXin Li     { SFLOW_FLOW_EXTENDED_MPLS_LVP_FEC, "Extended MPLS LVP FEC"},
156*05b00f60SXin Li     { SFLOW_FLOW_EXTENDED_VLAN_TUNNEL, "Extended VLAN Tunnel"},
157*05b00f60SXin Li     { 0, NULL}
158*05b00f60SXin Li };
159*05b00f60SXin Li 
160*05b00f60SXin Li #define		SFLOW_HEADER_PROTOCOL_ETHERNET	1
161*05b00f60SXin Li #define		SFLOW_HEADER_PROTOCOL_IPV4	11
162*05b00f60SXin Li #define		SFLOW_HEADER_PROTOCOL_IPV6	12
163*05b00f60SXin Li 
164*05b00f60SXin Li static const struct tok sflow_flow_raw_protocol_values[] = {
165*05b00f60SXin Li     { SFLOW_HEADER_PROTOCOL_ETHERNET, "Ethernet"},
166*05b00f60SXin Li     { SFLOW_HEADER_PROTOCOL_IPV4, "IPv4"},
167*05b00f60SXin Li     { SFLOW_HEADER_PROTOCOL_IPV6, "IPv6"},
168*05b00f60SXin Li     { 0, NULL}
169*05b00f60SXin Li };
170*05b00f60SXin Li 
171*05b00f60SXin Li struct sflow_expanded_flow_raw_t {
172*05b00f60SXin Li     nd_uint32_t protocol;
173*05b00f60SXin Li     nd_uint32_t length;
174*05b00f60SXin Li     nd_uint32_t stripped_bytes;
175*05b00f60SXin Li     nd_uint32_t header_size;
176*05b00f60SXin Li };
177*05b00f60SXin Li 
178*05b00f60SXin Li struct sflow_ethernet_frame_t {
179*05b00f60SXin Li     nd_uint32_t length;
180*05b00f60SXin Li     nd_byte     src_mac[8];
181*05b00f60SXin Li     nd_byte     dst_mac[8];
182*05b00f60SXin Li     nd_uint32_t type;
183*05b00f60SXin Li };
184*05b00f60SXin Li 
185*05b00f60SXin Li struct sflow_extended_switch_data_t {
186*05b00f60SXin Li     nd_uint32_t src_vlan;
187*05b00f60SXin Li     nd_uint32_t src_pri;
188*05b00f60SXin Li     nd_uint32_t dst_vlan;
189*05b00f60SXin Li     nd_uint32_t dst_pri;
190*05b00f60SXin Li };
191*05b00f60SXin Li 
192*05b00f60SXin Li struct sflow_counter_record_t {
193*05b00f60SXin Li     nd_uint32_t    format;
194*05b00f60SXin Li     nd_uint32_t    length;
195*05b00f60SXin Li };
196*05b00f60SXin Li 
197*05b00f60SXin Li struct sflow_flow_record_t {
198*05b00f60SXin Li     nd_uint32_t    format;
199*05b00f60SXin Li     nd_uint32_t    length;
200*05b00f60SXin Li };
201*05b00f60SXin Li 
202*05b00f60SXin Li struct sflow_counter_sample_t {
203*05b00f60SXin Li     nd_uint32_t    seqnum;
204*05b00f60SXin Li     nd_uint8_t     type;
205*05b00f60SXin Li     nd_uint24_t    index;
206*05b00f60SXin Li     nd_uint32_t    records;
207*05b00f60SXin Li };
208*05b00f60SXin Li 
209*05b00f60SXin Li struct sflow_expanded_counter_sample_t {
210*05b00f60SXin Li     nd_uint32_t    seqnum;
211*05b00f60SXin Li     nd_uint32_t    type;
212*05b00f60SXin Li     nd_uint32_t    index;
213*05b00f60SXin Li     nd_uint32_t    records;
214*05b00f60SXin Li };
215*05b00f60SXin Li 
216*05b00f60SXin Li #define         SFLOW_COUNTER_GENERIC           1
217*05b00f60SXin Li #define         SFLOW_COUNTER_ETHERNET          2
218*05b00f60SXin Li #define         SFLOW_COUNTER_TOKEN_RING        3
219*05b00f60SXin Li #define         SFLOW_COUNTER_BASEVG            4
220*05b00f60SXin Li #define         SFLOW_COUNTER_VLAN              5
221*05b00f60SXin Li #define         SFLOW_COUNTER_PROCESSOR         1001
222*05b00f60SXin Li 
223*05b00f60SXin Li static const struct tok sflow_counter_type_values[] = {
224*05b00f60SXin Li     { SFLOW_COUNTER_GENERIC, "Generic counter"},
225*05b00f60SXin Li     { SFLOW_COUNTER_ETHERNET, "Ethernet counter"},
226*05b00f60SXin Li     { SFLOW_COUNTER_TOKEN_RING, "Token ring counter"},
227*05b00f60SXin Li     { SFLOW_COUNTER_BASEVG, "100 BaseVG counter"},
228*05b00f60SXin Li     { SFLOW_COUNTER_VLAN, "Vlan counter"},
229*05b00f60SXin Li     { SFLOW_COUNTER_PROCESSOR, "Processor counter"},
230*05b00f60SXin Li     { 0, NULL}
231*05b00f60SXin Li };
232*05b00f60SXin Li 
233*05b00f60SXin Li #define		SFLOW_IFACE_DIRECTION_UNKNOWN		0
234*05b00f60SXin Li #define		SFLOW_IFACE_DIRECTION_FULLDUPLEX	1
235*05b00f60SXin Li #define		SFLOW_IFACE_DIRECTION_HALFDUPLEX	2
236*05b00f60SXin Li #define		SFLOW_IFACE_DIRECTION_IN		3
237*05b00f60SXin Li #define		SFLOW_IFACE_DIRECTION_OUT		4
238*05b00f60SXin Li 
239*05b00f60SXin Li static const struct tok sflow_iface_direction_values[] = {
240*05b00f60SXin Li     { SFLOW_IFACE_DIRECTION_UNKNOWN, "unknown"},
241*05b00f60SXin Li     { SFLOW_IFACE_DIRECTION_FULLDUPLEX, "full-duplex"},
242*05b00f60SXin Li     { SFLOW_IFACE_DIRECTION_HALFDUPLEX, "half-duplex"},
243*05b00f60SXin Li     { SFLOW_IFACE_DIRECTION_IN, "in"},
244*05b00f60SXin Li     { SFLOW_IFACE_DIRECTION_OUT, "out"},
245*05b00f60SXin Li     { 0, NULL}
246*05b00f60SXin Li };
247*05b00f60SXin Li 
248*05b00f60SXin Li struct sflow_generic_counter_t {
249*05b00f60SXin Li     nd_uint32_t    ifindex;
250*05b00f60SXin Li     nd_uint32_t    iftype;
251*05b00f60SXin Li     nd_uint64_t    ifspeed;
252*05b00f60SXin Li     nd_uint32_t    ifdirection;
253*05b00f60SXin Li     nd_uint32_t    ifstatus;
254*05b00f60SXin Li     nd_uint64_t    ifinoctets;
255*05b00f60SXin Li     nd_uint32_t    ifinunicastpkts;
256*05b00f60SXin Li     nd_uint32_t    ifinmulticastpkts;
257*05b00f60SXin Li     nd_uint32_t    ifinbroadcastpkts;
258*05b00f60SXin Li     nd_uint32_t    ifindiscards;
259*05b00f60SXin Li     nd_uint32_t    ifinerrors;
260*05b00f60SXin Li     nd_uint32_t    ifinunkownprotos;
261*05b00f60SXin Li     nd_uint64_t    ifoutoctets;
262*05b00f60SXin Li     nd_uint32_t    ifoutunicastpkts;
263*05b00f60SXin Li     nd_uint32_t    ifoutmulticastpkts;
264*05b00f60SXin Li     nd_uint32_t    ifoutbroadcastpkts;
265*05b00f60SXin Li     nd_uint32_t    ifoutdiscards;
266*05b00f60SXin Li     nd_uint32_t    ifouterrors;
267*05b00f60SXin Li     nd_uint32_t    ifpromiscmode;
268*05b00f60SXin Li };
269*05b00f60SXin Li 
270*05b00f60SXin Li struct sflow_ethernet_counter_t {
271*05b00f60SXin Li     nd_uint32_t    alignerrors;
272*05b00f60SXin Li     nd_uint32_t    fcserrors;
273*05b00f60SXin Li     nd_uint32_t    single_collision_frames;
274*05b00f60SXin Li     nd_uint32_t    multiple_collision_frames;
275*05b00f60SXin Li     nd_uint32_t    test_errors;
276*05b00f60SXin Li     nd_uint32_t    deferred_transmissions;
277*05b00f60SXin Li     nd_uint32_t    late_collisions;
278*05b00f60SXin Li     nd_uint32_t    excessive_collisions;
279*05b00f60SXin Li     nd_uint32_t    mac_transmit_errors;
280*05b00f60SXin Li     nd_uint32_t    carrier_sense_errors;
281*05b00f60SXin Li     nd_uint32_t    frame_too_longs;
282*05b00f60SXin Li     nd_uint32_t    mac_receive_errors;
283*05b00f60SXin Li     nd_uint32_t    symbol_errors;
284*05b00f60SXin Li };
285*05b00f60SXin Li 
286*05b00f60SXin Li struct sflow_100basevg_counter_t {
287*05b00f60SXin Li     nd_uint32_t    in_highpriority_frames;
288*05b00f60SXin Li     nd_uint64_t    in_highpriority_octets;
289*05b00f60SXin Li     nd_uint32_t    in_normpriority_frames;
290*05b00f60SXin Li     nd_uint64_t    in_normpriority_octets;
291*05b00f60SXin Li     nd_uint32_t    in_ipmerrors;
292*05b00f60SXin Li     nd_uint32_t    in_oversized;
293*05b00f60SXin Li     nd_uint32_t    in_data_errors;
294*05b00f60SXin Li     nd_uint32_t    in_null_addressed_frames;
295*05b00f60SXin Li     nd_uint32_t    out_highpriority_frames;
296*05b00f60SXin Li     nd_uint64_t    out_highpriority_octets;
297*05b00f60SXin Li     nd_uint32_t    transitioninto_frames;
298*05b00f60SXin Li     nd_uint64_t    hc_in_highpriority_octets;
299*05b00f60SXin Li     nd_uint64_t    hc_in_normpriority_octets;
300*05b00f60SXin Li     nd_uint64_t    hc_out_highpriority_octets;
301*05b00f60SXin Li };
302*05b00f60SXin Li 
303*05b00f60SXin Li struct sflow_vlan_counter_t {
304*05b00f60SXin Li     nd_uint32_t    vlan_id;
305*05b00f60SXin Li     nd_uint64_t    octets;
306*05b00f60SXin Li     nd_uint32_t    unicast_pkt;
307*05b00f60SXin Li     nd_uint32_t    multicast_pkt;
308*05b00f60SXin Li     nd_uint32_t    broadcast_pkt;
309*05b00f60SXin Li     nd_uint32_t    discards;
310*05b00f60SXin Li };
311*05b00f60SXin Li 
312*05b00f60SXin Li static int
print_sflow_counter_generic(netdissect_options * ndo,const u_char * pointer,u_int len)313*05b00f60SXin Li print_sflow_counter_generic(netdissect_options *ndo,
314*05b00f60SXin Li                             const u_char *pointer, u_int len)
315*05b00f60SXin Li {
316*05b00f60SXin Li     const struct sflow_generic_counter_t *sflow_gen_counter;
317*05b00f60SXin Li 
318*05b00f60SXin Li     if (len < sizeof(struct sflow_generic_counter_t))
319*05b00f60SXin Li 	return 1;
320*05b00f60SXin Li 
321*05b00f60SXin Li     sflow_gen_counter = (const struct sflow_generic_counter_t *)pointer;
322*05b00f60SXin Li     ND_PRINT("\n\t      ifindex %u, iftype %u, ifspeed %" PRIu64 ", ifdirection %u (%s)",
323*05b00f60SXin Li 	   GET_BE_U_4(sflow_gen_counter->ifindex),
324*05b00f60SXin Li 	   GET_BE_U_4(sflow_gen_counter->iftype),
325*05b00f60SXin Li 	   GET_BE_U_8(sflow_gen_counter->ifspeed),
326*05b00f60SXin Li 	   GET_BE_U_4(sflow_gen_counter->ifdirection),
327*05b00f60SXin Li 	   tok2str(sflow_iface_direction_values, "Unknown",
328*05b00f60SXin Li 	   GET_BE_U_4(sflow_gen_counter->ifdirection)));
329*05b00f60SXin Li     ND_PRINT("\n\t      ifstatus %u, adminstatus: %s, operstatus: %s",
330*05b00f60SXin Li 	   GET_BE_U_4(sflow_gen_counter->ifstatus),
331*05b00f60SXin Li 	   GET_BE_U_4(sflow_gen_counter->ifstatus)&1 ? "up" : "down",
332*05b00f60SXin Li 	   (GET_BE_U_4(sflow_gen_counter->ifstatus)>>1)&1 ? "up" : "down");
333*05b00f60SXin Li     ND_PRINT("\n\t      In octets %" PRIu64
334*05b00f60SXin Li 	   ", unicast pkts %u, multicast pkts %u, broadcast pkts %u, discards %u",
335*05b00f60SXin Li 	   GET_BE_U_8(sflow_gen_counter->ifinoctets),
336*05b00f60SXin Li 	   GET_BE_U_4(sflow_gen_counter->ifinunicastpkts),
337*05b00f60SXin Li 	   GET_BE_U_4(sflow_gen_counter->ifinmulticastpkts),
338*05b00f60SXin Li 	   GET_BE_U_4(sflow_gen_counter->ifinbroadcastpkts),
339*05b00f60SXin Li 	   GET_BE_U_4(sflow_gen_counter->ifindiscards));
340*05b00f60SXin Li     ND_PRINT("\n\t      In errors %u, unknown protos %u",
341*05b00f60SXin Li 	   GET_BE_U_4(sflow_gen_counter->ifinerrors),
342*05b00f60SXin Li 	   GET_BE_U_4(sflow_gen_counter->ifinunkownprotos));
343*05b00f60SXin Li     ND_PRINT("\n\t      Out octets %" PRIu64
344*05b00f60SXin Li 	   ", unicast pkts %u, multicast pkts %u, broadcast pkts %u, discards %u",
345*05b00f60SXin Li 	   GET_BE_U_8(sflow_gen_counter->ifoutoctets),
346*05b00f60SXin Li 	   GET_BE_U_4(sflow_gen_counter->ifoutunicastpkts),
347*05b00f60SXin Li 	   GET_BE_U_4(sflow_gen_counter->ifoutmulticastpkts),
348*05b00f60SXin Li 	   GET_BE_U_4(sflow_gen_counter->ifoutbroadcastpkts),
349*05b00f60SXin Li 	   GET_BE_U_4(sflow_gen_counter->ifoutdiscards));
350*05b00f60SXin Li     ND_PRINT("\n\t      Out errors %u, promisc mode %u",
351*05b00f60SXin Li 	   GET_BE_U_4(sflow_gen_counter->ifouterrors),
352*05b00f60SXin Li 	   GET_BE_U_4(sflow_gen_counter->ifpromiscmode));
353*05b00f60SXin Li 
354*05b00f60SXin Li     return 0;
355*05b00f60SXin Li }
356*05b00f60SXin Li 
357*05b00f60SXin Li static int
print_sflow_counter_ethernet(netdissect_options * ndo,const u_char * pointer,u_int len)358*05b00f60SXin Li print_sflow_counter_ethernet(netdissect_options *ndo,
359*05b00f60SXin Li                              const u_char *pointer, u_int len)
360*05b00f60SXin Li {
361*05b00f60SXin Li     const struct sflow_ethernet_counter_t *sflow_eth_counter;
362*05b00f60SXin Li 
363*05b00f60SXin Li     if (len < sizeof(struct sflow_ethernet_counter_t))
364*05b00f60SXin Li 	return 1;
365*05b00f60SXin Li 
366*05b00f60SXin Li     sflow_eth_counter = (const struct sflow_ethernet_counter_t *)pointer;
367*05b00f60SXin Li     ND_PRINT("\n\t      align errors %u, fcs errors %u, single collision %u, multiple collision %u, test error %u",
368*05b00f60SXin Li 	   GET_BE_U_4(sflow_eth_counter->alignerrors),
369*05b00f60SXin Li 	   GET_BE_U_4(sflow_eth_counter->fcserrors),
370*05b00f60SXin Li 	   GET_BE_U_4(sflow_eth_counter->single_collision_frames),
371*05b00f60SXin Li 	   GET_BE_U_4(sflow_eth_counter->multiple_collision_frames),
372*05b00f60SXin Li 	   GET_BE_U_4(sflow_eth_counter->test_errors));
373*05b00f60SXin Li     ND_PRINT("\n\t      deferred %u, late collision %u, excessive collision %u, mac trans error %u",
374*05b00f60SXin Li 	   GET_BE_U_4(sflow_eth_counter->deferred_transmissions),
375*05b00f60SXin Li 	   GET_BE_U_4(sflow_eth_counter->late_collisions),
376*05b00f60SXin Li 	   GET_BE_U_4(sflow_eth_counter->excessive_collisions),
377*05b00f60SXin Li 	   GET_BE_U_4(sflow_eth_counter->mac_transmit_errors));
378*05b00f60SXin Li     ND_PRINT("\n\t      carrier error %u, frames too long %u, mac receive errors %u, symbol errors %u",
379*05b00f60SXin Li 	   GET_BE_U_4(sflow_eth_counter->carrier_sense_errors),
380*05b00f60SXin Li 	   GET_BE_U_4(sflow_eth_counter->frame_too_longs),
381*05b00f60SXin Li 	   GET_BE_U_4(sflow_eth_counter->mac_receive_errors),
382*05b00f60SXin Li 	   GET_BE_U_4(sflow_eth_counter->symbol_errors));
383*05b00f60SXin Li 
384*05b00f60SXin Li     return 0;
385*05b00f60SXin Li }
386*05b00f60SXin Li 
387*05b00f60SXin Li static int
print_sflow_counter_token_ring(netdissect_options * ndo _U_,const u_char * pointer _U_,u_int len _U_)388*05b00f60SXin Li print_sflow_counter_token_ring(netdissect_options *ndo _U_,
389*05b00f60SXin Li                                const u_char *pointer _U_, u_int len _U_)
390*05b00f60SXin Li {
391*05b00f60SXin Li     return 0;
392*05b00f60SXin Li }
393*05b00f60SXin Li 
394*05b00f60SXin Li static int
print_sflow_counter_basevg(netdissect_options * ndo,const u_char * pointer,u_int len)395*05b00f60SXin Li print_sflow_counter_basevg(netdissect_options *ndo,
396*05b00f60SXin Li                            const u_char *pointer, u_int len)
397*05b00f60SXin Li {
398*05b00f60SXin Li     const struct sflow_100basevg_counter_t *sflow_100basevg_counter;
399*05b00f60SXin Li 
400*05b00f60SXin Li     if (len < sizeof(struct sflow_100basevg_counter_t))
401*05b00f60SXin Li 	return 1;
402*05b00f60SXin Li 
403*05b00f60SXin Li     sflow_100basevg_counter = (const struct sflow_100basevg_counter_t *)pointer;
404*05b00f60SXin Li     ND_PRINT("\n\t      in high prio frames %u, in high prio octets %" PRIu64,
405*05b00f60SXin Li 	   GET_BE_U_4(sflow_100basevg_counter->in_highpriority_frames),
406*05b00f60SXin Li 	   GET_BE_U_8(sflow_100basevg_counter->in_highpriority_octets));
407*05b00f60SXin Li     ND_PRINT("\n\t      in norm prio frames %u, in norm prio octets %" PRIu64,
408*05b00f60SXin Li 	   GET_BE_U_4(sflow_100basevg_counter->in_normpriority_frames),
409*05b00f60SXin Li 	   GET_BE_U_8(sflow_100basevg_counter->in_normpriority_octets));
410*05b00f60SXin Li     ND_PRINT("\n\t      in ipm errors %u, oversized %u, in data errors %u, null addressed frames %u",
411*05b00f60SXin Li 	   GET_BE_U_4(sflow_100basevg_counter->in_ipmerrors),
412*05b00f60SXin Li 	   GET_BE_U_4(sflow_100basevg_counter->in_oversized),
413*05b00f60SXin Li 	   GET_BE_U_4(sflow_100basevg_counter->in_data_errors),
414*05b00f60SXin Li 	   GET_BE_U_4(sflow_100basevg_counter->in_null_addressed_frames));
415*05b00f60SXin Li     ND_PRINT("\n\t      out high prio frames %u, out high prio octets %" PRIu64
416*05b00f60SXin Li 	   ", trans into frames %u",
417*05b00f60SXin Li 	   GET_BE_U_4(sflow_100basevg_counter->out_highpriority_frames),
418*05b00f60SXin Li 	   GET_BE_U_8(sflow_100basevg_counter->out_highpriority_octets),
419*05b00f60SXin Li 	   GET_BE_U_4(sflow_100basevg_counter->transitioninto_frames));
420*05b00f60SXin Li     ND_PRINT("\n\t      in hc high prio octets %" PRIu64
421*05b00f60SXin Li 	   ", in hc norm prio octets %" PRIu64
422*05b00f60SXin Li 	   ", out hc high prio octets %" PRIu64,
423*05b00f60SXin Li 	   GET_BE_U_8(sflow_100basevg_counter->hc_in_highpriority_octets),
424*05b00f60SXin Li 	   GET_BE_U_8(sflow_100basevg_counter->hc_in_normpriority_octets),
425*05b00f60SXin Li 	   GET_BE_U_8(sflow_100basevg_counter->hc_out_highpriority_octets));
426*05b00f60SXin Li 
427*05b00f60SXin Li     return 0;
428*05b00f60SXin Li }
429*05b00f60SXin Li 
430*05b00f60SXin Li static int
print_sflow_counter_vlan(netdissect_options * ndo,const u_char * pointer,u_int len)431*05b00f60SXin Li print_sflow_counter_vlan(netdissect_options *ndo,
432*05b00f60SXin Li                          const u_char *pointer, u_int len)
433*05b00f60SXin Li {
434*05b00f60SXin Li     const struct sflow_vlan_counter_t *sflow_vlan_counter;
435*05b00f60SXin Li 
436*05b00f60SXin Li     if (len < sizeof(struct sflow_vlan_counter_t))
437*05b00f60SXin Li 	return 1;
438*05b00f60SXin Li 
439*05b00f60SXin Li     sflow_vlan_counter = (const struct sflow_vlan_counter_t *)pointer;
440*05b00f60SXin Li     ND_PRINT("\n\t      vlan_id %u, octets %" PRIu64
441*05b00f60SXin Li 	   ", unicast_pkt %u, multicast_pkt %u, broadcast_pkt %u, discards %u",
442*05b00f60SXin Li 	   GET_BE_U_4(sflow_vlan_counter->vlan_id),
443*05b00f60SXin Li 	   GET_BE_U_8(sflow_vlan_counter->octets),
444*05b00f60SXin Li 	   GET_BE_U_4(sflow_vlan_counter->unicast_pkt),
445*05b00f60SXin Li 	   GET_BE_U_4(sflow_vlan_counter->multicast_pkt),
446*05b00f60SXin Li 	   GET_BE_U_4(sflow_vlan_counter->broadcast_pkt),
447*05b00f60SXin Li 	   GET_BE_U_4(sflow_vlan_counter->discards));
448*05b00f60SXin Li 
449*05b00f60SXin Li     return 0;
450*05b00f60SXin Li }
451*05b00f60SXin Li 
452*05b00f60SXin Li struct sflow_processor_counter_t {
453*05b00f60SXin Li     nd_uint32_t five_sec_util;
454*05b00f60SXin Li     nd_uint32_t one_min_util;
455*05b00f60SXin Li     nd_uint32_t five_min_util;
456*05b00f60SXin Li     nd_uint64_t total_memory;
457*05b00f60SXin Li     nd_uint64_t free_memory;
458*05b00f60SXin Li };
459*05b00f60SXin Li 
460*05b00f60SXin Li static int
print_sflow_counter_processor(netdissect_options * ndo,const u_char * pointer,u_int len)461*05b00f60SXin Li print_sflow_counter_processor(netdissect_options *ndo,
462*05b00f60SXin Li                               const u_char *pointer, u_int len)
463*05b00f60SXin Li {
464*05b00f60SXin Li     const struct sflow_processor_counter_t *sflow_processor_counter;
465*05b00f60SXin Li 
466*05b00f60SXin Li     if (len < sizeof(struct sflow_processor_counter_t))
467*05b00f60SXin Li 	return 1;
468*05b00f60SXin Li 
469*05b00f60SXin Li     sflow_processor_counter = (const struct sflow_processor_counter_t *)pointer;
470*05b00f60SXin Li     ND_PRINT("\n\t      5sec %u, 1min %u, 5min %u, total_mem %" PRIu64
471*05b00f60SXin Li 	   ", total_mem %" PRIu64,
472*05b00f60SXin Li 	   GET_BE_U_4(sflow_processor_counter->five_sec_util),
473*05b00f60SXin Li 	   GET_BE_U_4(sflow_processor_counter->one_min_util),
474*05b00f60SXin Li 	   GET_BE_U_4(sflow_processor_counter->five_min_util),
475*05b00f60SXin Li 	   GET_BE_U_8(sflow_processor_counter->total_memory),
476*05b00f60SXin Li 	   GET_BE_U_8(sflow_processor_counter->free_memory));
477*05b00f60SXin Li 
478*05b00f60SXin Li     return 0;
479*05b00f60SXin Li }
480*05b00f60SXin Li 
481*05b00f60SXin Li static int
sflow_print_counter_records(netdissect_options * ndo,const u_char * pointer,u_int len,u_int records)482*05b00f60SXin Li sflow_print_counter_records(netdissect_options *ndo,
483*05b00f60SXin Li                             const u_char *pointer, u_int len, u_int records)
484*05b00f60SXin Li {
485*05b00f60SXin Li     u_int nrecords;
486*05b00f60SXin Li     const u_char *tptr;
487*05b00f60SXin Li     u_int tlen;
488*05b00f60SXin Li     u_int counter_type;
489*05b00f60SXin Li     u_int counter_len;
490*05b00f60SXin Li     u_int enterprise;
491*05b00f60SXin Li     const struct sflow_counter_record_t *sflow_counter_record;
492*05b00f60SXin Li 
493*05b00f60SXin Li     nrecords = records;
494*05b00f60SXin Li     tptr = pointer;
495*05b00f60SXin Li     tlen = len;
496*05b00f60SXin Li 
497*05b00f60SXin Li     while (nrecords > 0) {
498*05b00f60SXin Li 	/* do we have the "header?" */
499*05b00f60SXin Li 	if (tlen < sizeof(struct sflow_counter_record_t))
500*05b00f60SXin Li 	    return 1;
501*05b00f60SXin Li 	sflow_counter_record = (const struct sflow_counter_record_t *)tptr;
502*05b00f60SXin Li 
503*05b00f60SXin Li 	enterprise = GET_BE_U_4(sflow_counter_record->format);
504*05b00f60SXin Li 	counter_type = enterprise & 0x0FFF;
505*05b00f60SXin Li 	enterprise = enterprise >> 20;
506*05b00f60SXin Li 	counter_len  = GET_BE_U_4(sflow_counter_record->length);
507*05b00f60SXin Li 	ND_PRINT("\n\t    enterprise %u, %s (%u) length %u",
508*05b00f60SXin Li 	       enterprise,
509*05b00f60SXin Li 	       (enterprise == 0) ? tok2str(sflow_counter_type_values,"Unknown",counter_type) : "Unknown",
510*05b00f60SXin Li 	       counter_type,
511*05b00f60SXin Li 	       counter_len);
512*05b00f60SXin Li 
513*05b00f60SXin Li 	tptr += sizeof(struct sflow_counter_record_t);
514*05b00f60SXin Li 	tlen -= sizeof(struct sflow_counter_record_t);
515*05b00f60SXin Li 
516*05b00f60SXin Li 	if (tlen < counter_len)
517*05b00f60SXin Li 	    return 1;
518*05b00f60SXin Li 	if (enterprise == 0) {
519*05b00f60SXin Li 	    switch (counter_type) {
520*05b00f60SXin Li 	    case SFLOW_COUNTER_GENERIC:
521*05b00f60SXin Li 		if (print_sflow_counter_generic(ndo, tptr, tlen))
522*05b00f60SXin Li 		    return 1;
523*05b00f60SXin Li 		break;
524*05b00f60SXin Li 	    case SFLOW_COUNTER_ETHERNET:
525*05b00f60SXin Li 		if (print_sflow_counter_ethernet(ndo, tptr, tlen))
526*05b00f60SXin Li 		    return 1;
527*05b00f60SXin Li 		break;
528*05b00f60SXin Li 	    case SFLOW_COUNTER_TOKEN_RING:
529*05b00f60SXin Li 		if (print_sflow_counter_token_ring(ndo, tptr,tlen))
530*05b00f60SXin Li 		    return 1;
531*05b00f60SXin Li 		break;
532*05b00f60SXin Li 	    case SFLOW_COUNTER_BASEVG:
533*05b00f60SXin Li 		if (print_sflow_counter_basevg(ndo, tptr, tlen))
534*05b00f60SXin Li 		    return 1;
535*05b00f60SXin Li 		break;
536*05b00f60SXin Li 	    case SFLOW_COUNTER_VLAN:
537*05b00f60SXin Li 		if (print_sflow_counter_vlan(ndo, tptr, tlen))
538*05b00f60SXin Li 		    return 1;
539*05b00f60SXin Li 		break;
540*05b00f60SXin Li 	    case SFLOW_COUNTER_PROCESSOR:
541*05b00f60SXin Li 		if (print_sflow_counter_processor(ndo, tptr, tlen))
542*05b00f60SXin Li 		    return 1;
543*05b00f60SXin Li 		break;
544*05b00f60SXin Li 	    default:
545*05b00f60SXin Li 		if (ndo->ndo_vflag <= 1)
546*05b00f60SXin Li 		    print_unknown_data(ndo, tptr, "\n\t\t", counter_len);
547*05b00f60SXin Li 		break;
548*05b00f60SXin Li 	    }
549*05b00f60SXin Li 	}
550*05b00f60SXin Li 	tptr += counter_len;
551*05b00f60SXin Li 	tlen -= counter_len;
552*05b00f60SXin Li 	nrecords--;
553*05b00f60SXin Li 
554*05b00f60SXin Li     }
555*05b00f60SXin Li 
556*05b00f60SXin Li     return 0;
557*05b00f60SXin Li }
558*05b00f60SXin Li 
559*05b00f60SXin Li static int
sflow_print_counter_sample(netdissect_options * ndo,const u_char * pointer,u_int len)560*05b00f60SXin Li sflow_print_counter_sample(netdissect_options *ndo,
561*05b00f60SXin Li                            const u_char *pointer, u_int len)
562*05b00f60SXin Li {
563*05b00f60SXin Li     const struct sflow_counter_sample_t *sflow_counter_sample;
564*05b00f60SXin Li     u_int           nrecords;
565*05b00f60SXin Li 
566*05b00f60SXin Li     if (len < sizeof(struct sflow_counter_sample_t))
567*05b00f60SXin Li 	return 1;
568*05b00f60SXin Li 
569*05b00f60SXin Li     sflow_counter_sample = (const struct sflow_counter_sample_t *)pointer;
570*05b00f60SXin Li 
571*05b00f60SXin Li     nrecords   = GET_BE_U_4(sflow_counter_sample->records);
572*05b00f60SXin Li 
573*05b00f60SXin Li     ND_PRINT(" seqnum %u, type %u, idx %u, records %u",
574*05b00f60SXin Li 	   GET_BE_U_4(sflow_counter_sample->seqnum),
575*05b00f60SXin Li 	   GET_U_1(sflow_counter_sample->type),
576*05b00f60SXin Li 	   GET_BE_U_3(sflow_counter_sample->index),
577*05b00f60SXin Li 	   nrecords);
578*05b00f60SXin Li 
579*05b00f60SXin Li     return sflow_print_counter_records(ndo, pointer + sizeof(struct sflow_counter_sample_t),
580*05b00f60SXin Li 				       len - sizeof(struct sflow_counter_sample_t),
581*05b00f60SXin Li 				       nrecords);
582*05b00f60SXin Li }
583*05b00f60SXin Li 
584*05b00f60SXin Li static int
sflow_print_expanded_counter_sample(netdissect_options * ndo,const u_char * pointer,u_int len)585*05b00f60SXin Li sflow_print_expanded_counter_sample(netdissect_options *ndo,
586*05b00f60SXin Li                                     const u_char *pointer, u_int len)
587*05b00f60SXin Li {
588*05b00f60SXin Li     const struct sflow_expanded_counter_sample_t *sflow_expanded_counter_sample;
589*05b00f60SXin Li     u_int           nrecords;
590*05b00f60SXin Li 
591*05b00f60SXin Li 
592*05b00f60SXin Li     if (len < sizeof(struct sflow_expanded_counter_sample_t))
593*05b00f60SXin Li 	return 1;
594*05b00f60SXin Li 
595*05b00f60SXin Li     sflow_expanded_counter_sample = (const struct sflow_expanded_counter_sample_t *)pointer;
596*05b00f60SXin Li 
597*05b00f60SXin Li     nrecords = GET_BE_U_4(sflow_expanded_counter_sample->records);
598*05b00f60SXin Li 
599*05b00f60SXin Li     ND_PRINT(" seqnum %u, type %u, idx %u, records %u",
600*05b00f60SXin Li 	   GET_BE_U_4(sflow_expanded_counter_sample->seqnum),
601*05b00f60SXin Li 	   GET_BE_U_4(sflow_expanded_counter_sample->type),
602*05b00f60SXin Li 	   GET_BE_U_4(sflow_expanded_counter_sample->index),
603*05b00f60SXin Li 	   nrecords);
604*05b00f60SXin Li 
605*05b00f60SXin Li     return sflow_print_counter_records(ndo, pointer + sizeof(struct sflow_expanded_counter_sample_t),
606*05b00f60SXin Li 				       len - sizeof(struct sflow_expanded_counter_sample_t),
607*05b00f60SXin Li 				       nrecords);
608*05b00f60SXin Li }
609*05b00f60SXin Li 
610*05b00f60SXin Li static int
print_sflow_raw_packet(netdissect_options * ndo,const u_char * pointer,u_int len)611*05b00f60SXin Li print_sflow_raw_packet(netdissect_options *ndo,
612*05b00f60SXin Li                        const u_char *pointer, u_int len)
613*05b00f60SXin Li {
614*05b00f60SXin Li     const struct sflow_expanded_flow_raw_t *sflow_flow_raw;
615*05b00f60SXin Li 
616*05b00f60SXin Li     if (len < sizeof(struct sflow_expanded_flow_raw_t))
617*05b00f60SXin Li 	return 1;
618*05b00f60SXin Li 
619*05b00f60SXin Li     sflow_flow_raw = (const struct sflow_expanded_flow_raw_t *)pointer;
620*05b00f60SXin Li     ND_PRINT("\n\t      protocol %s (%u), length %u, stripped bytes %u, header_size %u",
621*05b00f60SXin Li 	   tok2str(sflow_flow_raw_protocol_values,"Unknown",GET_BE_U_4(sflow_flow_raw->protocol)),
622*05b00f60SXin Li 	   GET_BE_U_4(sflow_flow_raw->protocol),
623*05b00f60SXin Li 	   GET_BE_U_4(sflow_flow_raw->length),
624*05b00f60SXin Li 	   GET_BE_U_4(sflow_flow_raw->stripped_bytes),
625*05b00f60SXin Li 	   GET_BE_U_4(sflow_flow_raw->header_size));
626*05b00f60SXin Li 
627*05b00f60SXin Li     /* QUESTION - should we attempt to print the raw header itself?
628*05b00f60SXin Li        assuming of course there is enough data present to do so... */
629*05b00f60SXin Li 
630*05b00f60SXin Li     return 0;
631*05b00f60SXin Li }
632*05b00f60SXin Li 
633*05b00f60SXin Li static int
print_sflow_ethernet_frame(netdissect_options * ndo,const u_char * pointer,u_int len)634*05b00f60SXin Li print_sflow_ethernet_frame(netdissect_options *ndo,
635*05b00f60SXin Li                            const u_char *pointer, u_int len)
636*05b00f60SXin Li {
637*05b00f60SXin Li     const struct sflow_ethernet_frame_t *sflow_ethernet_frame;
638*05b00f60SXin Li 
639*05b00f60SXin Li     if (len < sizeof(struct sflow_ethernet_frame_t))
640*05b00f60SXin Li 	return 1;
641*05b00f60SXin Li 
642*05b00f60SXin Li     sflow_ethernet_frame = (const struct sflow_ethernet_frame_t *)pointer;
643*05b00f60SXin Li 
644*05b00f60SXin Li     ND_PRINT("\n\t      frame len %u, type %u",
645*05b00f60SXin Li 	   GET_BE_U_4(sflow_ethernet_frame->length),
646*05b00f60SXin Li 	   GET_BE_U_4(sflow_ethernet_frame->type));
647*05b00f60SXin Li 
648*05b00f60SXin Li     return 0;
649*05b00f60SXin Li }
650*05b00f60SXin Li 
651*05b00f60SXin Li static int
print_sflow_extended_switch_data(netdissect_options * ndo,const u_char * pointer,u_int len)652*05b00f60SXin Li print_sflow_extended_switch_data(netdissect_options *ndo,
653*05b00f60SXin Li                                  const u_char *pointer, u_int len)
654*05b00f60SXin Li {
655*05b00f60SXin Li     const struct sflow_extended_switch_data_t *sflow_extended_sw_data;
656*05b00f60SXin Li 
657*05b00f60SXin Li     if (len < sizeof(struct sflow_extended_switch_data_t))
658*05b00f60SXin Li 	return 1;
659*05b00f60SXin Li 
660*05b00f60SXin Li     sflow_extended_sw_data = (const struct sflow_extended_switch_data_t *)pointer;
661*05b00f60SXin Li     ND_PRINT("\n\t      src vlan %u, src pri %u, dst vlan %u, dst pri %u",
662*05b00f60SXin Li 	   GET_BE_U_4(sflow_extended_sw_data->src_vlan),
663*05b00f60SXin Li 	   GET_BE_U_4(sflow_extended_sw_data->src_pri),
664*05b00f60SXin Li 	   GET_BE_U_4(sflow_extended_sw_data->dst_vlan),
665*05b00f60SXin Li 	   GET_BE_U_4(sflow_extended_sw_data->dst_pri));
666*05b00f60SXin Li 
667*05b00f60SXin Li     return 0;
668*05b00f60SXin Li }
669*05b00f60SXin Li 
670*05b00f60SXin Li static int
sflow_print_flow_records(netdissect_options * ndo,const u_char * pointer,u_int len,u_int records)671*05b00f60SXin Li sflow_print_flow_records(netdissect_options *ndo,
672*05b00f60SXin Li                          const u_char *pointer, u_int len, u_int records)
673*05b00f60SXin Li {
674*05b00f60SXin Li     u_int nrecords;
675*05b00f60SXin Li     const u_char *tptr;
676*05b00f60SXin Li     u_int tlen;
677*05b00f60SXin Li     u_int flow_type;
678*05b00f60SXin Li     u_int enterprise;
679*05b00f60SXin Li     u_int flow_len;
680*05b00f60SXin Li     const struct sflow_flow_record_t *sflow_flow_record;
681*05b00f60SXin Li 
682*05b00f60SXin Li     nrecords = records;
683*05b00f60SXin Li     tptr = pointer;
684*05b00f60SXin Li     tlen = len;
685*05b00f60SXin Li 
686*05b00f60SXin Li     while (nrecords > 0) {
687*05b00f60SXin Li 	/* do we have the "header?" */
688*05b00f60SXin Li 	if (tlen < sizeof(struct sflow_flow_record_t))
689*05b00f60SXin Li 	    return 1;
690*05b00f60SXin Li 
691*05b00f60SXin Li 	sflow_flow_record = (const struct sflow_flow_record_t *)tptr;
692*05b00f60SXin Li 
693*05b00f60SXin Li 	/* so, the funky encoding means we cannot blythly mask-off
694*05b00f60SXin Li 	   bits, we must also check the enterprise. */
695*05b00f60SXin Li 
696*05b00f60SXin Li 	enterprise = GET_BE_U_4(sflow_flow_record->format);
697*05b00f60SXin Li 	flow_type = enterprise & 0x0FFF;
698*05b00f60SXin Li 	enterprise = enterprise >> 12;
699*05b00f60SXin Li 	flow_len  = GET_BE_U_4(sflow_flow_record->length);
700*05b00f60SXin Li 	ND_PRINT("\n\t    enterprise %u %s (%u) length %u",
701*05b00f60SXin Li 	       enterprise,
702*05b00f60SXin Li 	       (enterprise == 0) ? tok2str(sflow_flow_type_values,"Unknown",flow_type) : "Unknown",
703*05b00f60SXin Li 	       flow_type,
704*05b00f60SXin Li 	       flow_len);
705*05b00f60SXin Li 
706*05b00f60SXin Li 	tptr += sizeof(struct sflow_flow_record_t);
707*05b00f60SXin Li 	tlen -= sizeof(struct sflow_flow_record_t);
708*05b00f60SXin Li 
709*05b00f60SXin Li 	if (tlen < flow_len)
710*05b00f60SXin Li 	    return 1;
711*05b00f60SXin Li 
712*05b00f60SXin Li 	if (enterprise == 0) {
713*05b00f60SXin Li 	    switch (flow_type) {
714*05b00f60SXin Li 	    case SFLOW_FLOW_RAW_PACKET:
715*05b00f60SXin Li 		if (print_sflow_raw_packet(ndo, tptr, tlen))
716*05b00f60SXin Li 		    return 1;
717*05b00f60SXin Li 		break;
718*05b00f60SXin Li 	    case SFLOW_FLOW_EXTENDED_SWITCH_DATA:
719*05b00f60SXin Li 		if (print_sflow_extended_switch_data(ndo, tptr, tlen))
720*05b00f60SXin Li 		    return 1;
721*05b00f60SXin Li 		break;
722*05b00f60SXin Li 	    case SFLOW_FLOW_ETHERNET_FRAME:
723*05b00f60SXin Li 		if (print_sflow_ethernet_frame(ndo, tptr, tlen))
724*05b00f60SXin Li 		    return 1;
725*05b00f60SXin Li 		break;
726*05b00f60SXin Li 		/* FIXME these need a decoder */
727*05b00f60SXin Li 	    case SFLOW_FLOW_IPV4_DATA:
728*05b00f60SXin Li 	    case SFLOW_FLOW_IPV6_DATA:
729*05b00f60SXin Li 	    case SFLOW_FLOW_EXTENDED_ROUTER_DATA:
730*05b00f60SXin Li 	    case SFLOW_FLOW_EXTENDED_GATEWAY_DATA:
731*05b00f60SXin Li 	    case SFLOW_FLOW_EXTENDED_USER_DATA:
732*05b00f60SXin Li 	    case SFLOW_FLOW_EXTENDED_URL_DATA:
733*05b00f60SXin Li 	    case SFLOW_FLOW_EXTENDED_MPLS_DATA:
734*05b00f60SXin Li 	    case SFLOW_FLOW_EXTENDED_NAT_DATA:
735*05b00f60SXin Li 	    case SFLOW_FLOW_EXTENDED_MPLS_TUNNEL:
736*05b00f60SXin Li 	    case SFLOW_FLOW_EXTENDED_MPLS_VC:
737*05b00f60SXin Li 	    case SFLOW_FLOW_EXTENDED_MPLS_FEC:
738*05b00f60SXin Li 	    case SFLOW_FLOW_EXTENDED_MPLS_LVP_FEC:
739*05b00f60SXin Li 	    case SFLOW_FLOW_EXTENDED_VLAN_TUNNEL:
740*05b00f60SXin Li 		break;
741*05b00f60SXin Li 	    default:
742*05b00f60SXin Li 		if (ndo->ndo_vflag <= 1)
743*05b00f60SXin Li 		    print_unknown_data(ndo, tptr, "\n\t\t", flow_len);
744*05b00f60SXin Li 		break;
745*05b00f60SXin Li 	    }
746*05b00f60SXin Li 	}
747*05b00f60SXin Li 	tptr += flow_len;
748*05b00f60SXin Li 	tlen -= flow_len;
749*05b00f60SXin Li 	nrecords--;
750*05b00f60SXin Li 
751*05b00f60SXin Li     }
752*05b00f60SXin Li 
753*05b00f60SXin Li     return 0;
754*05b00f60SXin Li }
755*05b00f60SXin Li 
756*05b00f60SXin Li static int
sflow_print_flow_sample(netdissect_options * ndo,const u_char * pointer,u_int len)757*05b00f60SXin Li sflow_print_flow_sample(netdissect_options *ndo,
758*05b00f60SXin Li                         const u_char *pointer, u_int len)
759*05b00f60SXin Li {
760*05b00f60SXin Li     const struct sflow_flow_sample_t *sflow_flow_sample;
761*05b00f60SXin Li     u_int          nrecords;
762*05b00f60SXin Li 
763*05b00f60SXin Li     if (len < sizeof(struct sflow_flow_sample_t))
764*05b00f60SXin Li 	return 1;
765*05b00f60SXin Li 
766*05b00f60SXin Li     sflow_flow_sample = (const struct sflow_flow_sample_t *)pointer;
767*05b00f60SXin Li 
768*05b00f60SXin Li     nrecords = GET_BE_U_4(sflow_flow_sample->records);
769*05b00f60SXin Li 
770*05b00f60SXin Li     ND_PRINT(" seqnum %u, type %u, idx %u, rate %u, pool %u, drops %u, input %u output %u records %u",
771*05b00f60SXin Li 	   GET_BE_U_4(sflow_flow_sample->seqnum),
772*05b00f60SXin Li 	   GET_U_1(sflow_flow_sample->type),
773*05b00f60SXin Li 	   GET_BE_U_3(sflow_flow_sample->index),
774*05b00f60SXin Li 	   GET_BE_U_4(sflow_flow_sample->rate),
775*05b00f60SXin Li 	   GET_BE_U_4(sflow_flow_sample->pool),
776*05b00f60SXin Li 	   GET_BE_U_4(sflow_flow_sample->drops),
777*05b00f60SXin Li 	   GET_BE_U_4(sflow_flow_sample->in_interface),
778*05b00f60SXin Li 	   GET_BE_U_4(sflow_flow_sample->out_interface),
779*05b00f60SXin Li 	   nrecords);
780*05b00f60SXin Li 
781*05b00f60SXin Li     return sflow_print_flow_records(ndo, pointer + sizeof(struct sflow_flow_sample_t),
782*05b00f60SXin Li 				    len - sizeof(struct sflow_flow_sample_t),
783*05b00f60SXin Li 				    nrecords);
784*05b00f60SXin Li }
785*05b00f60SXin Li 
786*05b00f60SXin Li static int
sflow_print_expanded_flow_sample(netdissect_options * ndo,const u_char * pointer,u_int len)787*05b00f60SXin Li sflow_print_expanded_flow_sample(netdissect_options *ndo,
788*05b00f60SXin Li                                  const u_char *pointer, u_int len)
789*05b00f60SXin Li {
790*05b00f60SXin Li     const struct sflow_expanded_flow_sample_t *sflow_expanded_flow_sample;
791*05b00f60SXin Li     u_int nrecords;
792*05b00f60SXin Li 
793*05b00f60SXin Li     if (len < sizeof(struct sflow_expanded_flow_sample_t))
794*05b00f60SXin Li 	return 1;
795*05b00f60SXin Li 
796*05b00f60SXin Li     sflow_expanded_flow_sample = (const struct sflow_expanded_flow_sample_t *)pointer;
797*05b00f60SXin Li 
798*05b00f60SXin Li     nrecords = GET_BE_U_4(sflow_expanded_flow_sample->records);
799*05b00f60SXin Li 
800*05b00f60SXin Li     ND_PRINT(" seqnum %u, type %u, idx %u, rate %u, pool %u, drops %u, records %u",
801*05b00f60SXin Li 	   GET_BE_U_4(sflow_expanded_flow_sample->seqnum),
802*05b00f60SXin Li 	   GET_BE_U_4(sflow_expanded_flow_sample->type),
803*05b00f60SXin Li 	   GET_BE_U_4(sflow_expanded_flow_sample->index),
804*05b00f60SXin Li 	   GET_BE_U_4(sflow_expanded_flow_sample->rate),
805*05b00f60SXin Li 	   GET_BE_U_4(sflow_expanded_flow_sample->pool),
806*05b00f60SXin Li 	   GET_BE_U_4(sflow_expanded_flow_sample->drops),
807*05b00f60SXin Li 	   nrecords);
808*05b00f60SXin Li 
809*05b00f60SXin Li     return sflow_print_flow_records(ndo, pointer + sizeof(struct sflow_expanded_flow_sample_t),
810*05b00f60SXin Li 				    len - sizeof(struct sflow_expanded_flow_sample_t),
811*05b00f60SXin Li 				    nrecords);
812*05b00f60SXin Li }
813*05b00f60SXin Li 
814*05b00f60SXin Li void
sflow_print(netdissect_options * ndo,const u_char * pptr,u_int len)815*05b00f60SXin Li sflow_print(netdissect_options *ndo,
816*05b00f60SXin Li             const u_char *pptr, u_int len)
817*05b00f60SXin Li {
818*05b00f60SXin Li     const struct sflow_datagram_t *sflow_datagram;
819*05b00f60SXin Li     const struct sflow_v6_datagram_t *sflow_v6_datagram;
820*05b00f60SXin Li     const struct sflow_sample_header *sflow_sample;
821*05b00f60SXin Li 
822*05b00f60SXin Li     const u_char *tptr;
823*05b00f60SXin Li     u_int tlen;
824*05b00f60SXin Li     uint32_t sflow_sample_type, sflow_sample_len;
825*05b00f60SXin Li     uint32_t nsamples;
826*05b00f60SXin Li     uint32_t ip_version;
827*05b00f60SXin Li 
828*05b00f60SXin Li     ndo->ndo_protocol = "sflow";
829*05b00f60SXin Li     tptr = pptr;
830*05b00f60SXin Li     tlen = len;
831*05b00f60SXin Li     sflow_datagram = (const struct sflow_datagram_t *)pptr;
832*05b00f60SXin Li     sflow_v6_datagram = (const struct sflow_v6_datagram_t *)pptr;
833*05b00f60SXin Li     ip_version = GET_BE_U_4(sflow_datagram->ip_version);
834*05b00f60SXin Li 
835*05b00f60SXin Li     if ((len < sizeof(struct sflow_datagram_t) && (ip_version == 1)) ||
836*05b00f60SXin Li         (len < sizeof(struct sflow_v6_datagram_t) && (ip_version == 2))) {
837*05b00f60SXin Li         ND_PRINT("sFlowv%u", GET_BE_U_4(sflow_datagram->version));
838*05b00f60SXin Li         ND_PRINT(" [length %u < %zu]", len, sizeof(struct sflow_datagram_t));
839*05b00f60SXin Li         nd_print_invalid(ndo);
840*05b00f60SXin Li         return;
841*05b00f60SXin Li     }
842*05b00f60SXin Li     ND_TCHECK_SIZE(sflow_datagram);
843*05b00f60SXin Li 
844*05b00f60SXin Li     /*
845*05b00f60SXin Li      * Sanity checking of the header.
846*05b00f60SXin Li      */
847*05b00f60SXin Li     if (GET_BE_U_4(sflow_datagram->version) != 5) {
848*05b00f60SXin Li         ND_PRINT("sFlow version %u packet not supported",
849*05b00f60SXin Li                GET_BE_U_4(sflow_datagram->version));
850*05b00f60SXin Li         return;
851*05b00f60SXin Li     }
852*05b00f60SXin Li 
853*05b00f60SXin Li     if (ndo->ndo_vflag < 1) {
854*05b00f60SXin Li         ND_PRINT("sFlowv%u, %s agent %s, agent-id %u, length %u",
855*05b00f60SXin Li                GET_BE_U_4(sflow_datagram->version),
856*05b00f60SXin Li                ip_version == 1 ? "IPv4" : "IPv6",
857*05b00f60SXin Li                ip_version == 1 ? GET_IPADDR_STRING(sflow_datagram->agent) :
858*05b00f60SXin Li                                  GET_IP6ADDR_STRING( sflow_v6_datagram->agent),
859*05b00f60SXin Li                ip_version == 1 ? GET_BE_U_4(sflow_datagram->agent_id) :
860*05b00f60SXin Li                                  GET_BE_U_4(sflow_v6_datagram->agent_id),
861*05b00f60SXin Li                len);
862*05b00f60SXin Li         return;
863*05b00f60SXin Li     }
864*05b00f60SXin Li 
865*05b00f60SXin Li     /* ok they seem to want to know everything - lets fully decode it */
866*05b00f60SXin Li     if (ip_version == 1) {
867*05b00f60SXin Li         nsamples=GET_BE_U_4(sflow_datagram->samples);
868*05b00f60SXin Li         ND_PRINT("sFlowv%u, %s agent %s, agent-id %u, seqnum %u, uptime %u, samples %u, length %u",
869*05b00f60SXin Li                GET_BE_U_4(sflow_datagram->version),
870*05b00f60SXin Li                "IPv4",
871*05b00f60SXin Li                GET_IPADDR_STRING(sflow_datagram->agent),
872*05b00f60SXin Li                GET_BE_U_4(sflow_datagram->agent_id),
873*05b00f60SXin Li                GET_BE_U_4(sflow_datagram->seqnum),
874*05b00f60SXin Li                GET_BE_U_4(sflow_datagram->uptime),
875*05b00f60SXin Li                nsamples,
876*05b00f60SXin Li                len);
877*05b00f60SXin Li 
878*05b00f60SXin Li         /* skip Common header */
879*05b00f60SXin Li         ND_LCHECK_ZU(tlen, sizeof(struct sflow_datagram_t));
880*05b00f60SXin Li         tptr += sizeof(struct sflow_datagram_t);
881*05b00f60SXin Li         tlen -= sizeof(struct sflow_datagram_t);
882*05b00f60SXin Li     } else {
883*05b00f60SXin Li         nsamples=GET_BE_U_4(sflow_v6_datagram->samples);
884*05b00f60SXin Li         ND_PRINT("sFlowv%u, %s agent %s, agent-id %u, seqnum %u, uptime %u, samples %u, length %u",
885*05b00f60SXin Li                GET_BE_U_4(sflow_v6_datagram->version),
886*05b00f60SXin Li                "IPv6",
887*05b00f60SXin Li                GET_IP6ADDR_STRING(sflow_v6_datagram->agent),
888*05b00f60SXin Li                GET_BE_U_4(sflow_v6_datagram->agent_id),
889*05b00f60SXin Li                GET_BE_U_4(sflow_v6_datagram->seqnum),
890*05b00f60SXin Li                GET_BE_U_4(sflow_v6_datagram->uptime),
891*05b00f60SXin Li                nsamples,
892*05b00f60SXin Li                len);
893*05b00f60SXin Li 
894*05b00f60SXin Li         /* skip Common header */
895*05b00f60SXin Li         ND_LCHECK_ZU(tlen, sizeof(struct sflow_v6_datagram_t));
896*05b00f60SXin Li         tptr += sizeof(struct sflow_v6_datagram_t);
897*05b00f60SXin Li         tlen -= sizeof(struct sflow_v6_datagram_t);
898*05b00f60SXin Li     }
899*05b00f60SXin Li     while (nsamples > 0 && tlen > 0) {
900*05b00f60SXin Li         sflow_sample = (const struct sflow_sample_header *)tptr;
901*05b00f60SXin Li 
902*05b00f60SXin Li         sflow_sample_type = (GET_BE_U_4(sflow_sample->format)&0x0FFF);
903*05b00f60SXin Li         sflow_sample_len = GET_BE_U_4(sflow_sample->len);
904*05b00f60SXin Li 
905*05b00f60SXin Li 	if (tlen < sizeof(struct sflow_sample_header))
906*05b00f60SXin Li 	    goto invalid;
907*05b00f60SXin Li 
908*05b00f60SXin Li         tptr += sizeof(struct sflow_sample_header);
909*05b00f60SXin Li         tlen -= sizeof(struct sflow_sample_header);
910*05b00f60SXin Li 
911*05b00f60SXin Li         ND_PRINT("\n\t%s (%u), length %u,",
912*05b00f60SXin Li                tok2str(sflow_format_values, "Unknown", sflow_sample_type),
913*05b00f60SXin Li                sflow_sample_type,
914*05b00f60SXin Li                sflow_sample_len);
915*05b00f60SXin Li 
916*05b00f60SXin Li         /* basic sanity check */
917*05b00f60SXin Li         if (sflow_sample_type == 0 || sflow_sample_len ==0) {
918*05b00f60SXin Li             return;
919*05b00f60SXin Li         }
920*05b00f60SXin Li 
921*05b00f60SXin Li 	if (tlen < sflow_sample_len)
922*05b00f60SXin Li 	    goto invalid;
923*05b00f60SXin Li 
924*05b00f60SXin Li         /* did we capture enough for fully decoding the sample ? */
925*05b00f60SXin Li         ND_TCHECK_LEN(tptr, sflow_sample_len);
926*05b00f60SXin Li 
927*05b00f60SXin Li 	switch(sflow_sample_type) {
928*05b00f60SXin Li         case SFLOW_FLOW_SAMPLE:
929*05b00f60SXin Li 	    if (sflow_print_flow_sample(ndo, tptr, tlen))
930*05b00f60SXin Li 		goto invalid;
931*05b00f60SXin Li             break;
932*05b00f60SXin Li 
933*05b00f60SXin Li         case SFLOW_COUNTER_SAMPLE:
934*05b00f60SXin Li 	    if (sflow_print_counter_sample(ndo, tptr,tlen))
935*05b00f60SXin Li 		goto invalid;
936*05b00f60SXin Li             break;
937*05b00f60SXin Li 
938*05b00f60SXin Li         case SFLOW_EXPANDED_FLOW_SAMPLE:
939*05b00f60SXin Li 	    if (sflow_print_expanded_flow_sample(ndo, tptr, tlen))
940*05b00f60SXin Li 		goto invalid;
941*05b00f60SXin Li 	    break;
942*05b00f60SXin Li 
943*05b00f60SXin Li         case SFLOW_EXPANDED_COUNTER_SAMPLE:
944*05b00f60SXin Li 	    if (sflow_print_expanded_counter_sample(ndo, tptr,tlen))
945*05b00f60SXin Li 		goto invalid;
946*05b00f60SXin Li 	    break;
947*05b00f60SXin Li 
948*05b00f60SXin Li         default:
949*05b00f60SXin Li             if (ndo->ndo_vflag <= 1)
950*05b00f60SXin Li                 print_unknown_data(ndo, tptr, "\n\t    ", sflow_sample_len);
951*05b00f60SXin Li             break;
952*05b00f60SXin Li         }
953*05b00f60SXin Li         tptr += sflow_sample_len;
954*05b00f60SXin Li         tlen -= sflow_sample_len;
955*05b00f60SXin Li         nsamples--;
956*05b00f60SXin Li     }
957*05b00f60SXin Li     return;
958*05b00f60SXin Li 
959*05b00f60SXin Li invalid:
960*05b00f60SXin Li     nd_print_invalid(ndo);
961*05b00f60SXin Li     ND_TCHECK_LEN(tptr, tlen);
962*05b00f60SXin Li }
963