xref: /freebsd/tools/tools/npe/npestats/npestats.c (revision b3e7694832e81d7a904a10f525f8797b753bf0d3)
1fc4c0664SSam Leffler /*-
2fc4c0664SSam Leffler  * Copyright (c) 2009 Sam Leffler, Errno Consulting
3fc4c0664SSam Leffler  * All rights reserved.
4fc4c0664SSam Leffler  *
5fc4c0664SSam Leffler  * Redistribution and use in source and binary forms, with or without
6fc4c0664SSam Leffler  * modification, are permitted provided that the following conditions
7fc4c0664SSam Leffler  * are met:
8fc4c0664SSam Leffler  * 1. Redistributions of source code must retain the above copyright
9fc4c0664SSam Leffler  *    notice, this list of conditions and the following disclaimer,
10fc4c0664SSam Leffler  *    without modification.
11fc4c0664SSam Leffler  * 2. Redistributions in binary form must reproduce at minimum a disclaimer
12fc4c0664SSam Leffler  *    similar to the "NO WARRANTY" disclaimer below ("Disclaimer") and any
13fc4c0664SSam Leffler  *    redistribution must be conditioned upon including a substantially
14fc4c0664SSam Leffler  *    similar Disclaimer requirement for further binary redistribution.
15fc4c0664SSam Leffler  *
16fc4c0664SSam Leffler  * NO WARRANTY
17fc4c0664SSam Leffler  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
18fc4c0664SSam Leffler  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
19fc4c0664SSam Leffler  * LIMITED TO, THE IMPLIED WARRANTIES OF NONINFRINGEMENT, MERCHANTIBILITY
20fc4c0664SSam Leffler  * AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
21fc4c0664SSam Leffler  * THE COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY,
22fc4c0664SSam Leffler  * OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23fc4c0664SSam Leffler  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
24fc4c0664SSam Leffler  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER
25fc4c0664SSam Leffler  * IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
26fc4c0664SSam Leffler  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
27fc4c0664SSam Leffler  * THE POSSIBILITY OF SUCH DAMAGES.
28fc4c0664SSam Leffler  */
29fc4c0664SSam Leffler 
30fc4c0664SSam Leffler /*
31fc4c0664SSam Leffler  * npe statistics class.
32fc4c0664SSam Leffler  */
33*60caf0c9SCraig Rodrigues #include <sys/param.h>
34fc4c0664SSam Leffler #include <sys/sysctl.h>
35fc4c0664SSam Leffler 
36*60caf0c9SCraig Rodrigues #include <err.h>
37*60caf0c9SCraig Rodrigues #include <signal.h>
38fc4c0664SSam Leffler #include <stdio.h>
39fc4c0664SSam Leffler #include <stdlib.h>
40fc4c0664SSam Leffler #include <string.h>
41fc4c0664SSam Leffler #include <unistd.h>
42fc4c0664SSam Leffler 
43fc4c0664SSam Leffler #include "npestats.h"
44fc4c0664SSam Leffler 
45fc4c0664SSam Leffler #define	AFTER(prev)	((prev)+1)
46fc4c0664SSam Leffler 
47fc4c0664SSam Leffler static const struct fmt npestats[] = {
48fc4c0664SSam Leffler #define	S_ALIGN		0
49fc4c0664SSam Leffler 	{ 7,	"align",	"align",	"alignment errors" },
50fc4c0664SSam Leffler #define	S_FCS		AFTER(S_ALIGN)
51fc4c0664SSam Leffler 	{ 7,	"fcs",		"fcs",		"FCS errors" },
52fc4c0664SSam Leffler #define	S_MACRX		AFTER(S_FCS)
53fc4c0664SSam Leffler 	{ 7,	"macrx",	"macrx",	"internal MAC rx errors" },
54fc4c0664SSam Leffler #define	S_RXORN		AFTER(S_MACRX)
55fc4c0664SSam Leffler 	{ 6,	"overrun",	"overrun",	"rx overrun discards" },
56fc4c0664SSam Leffler #define	S_LEARN		AFTER(S_RXORN)
57fc4c0664SSam Leffler 	{ 5,	"learn",	"learn",	"rx learned entry discards" },
58fc4c0664SSam Leffler #define	S_LARGE		AFTER(S_LEARN)
59fc4c0664SSam Leffler 	{ 5,	"large",	"large",	"rx large frame discards" },
60fc4c0664SSam Leffler #define	S_STP		AFTER(S_LARGE)
61fc4c0664SSam Leffler 	{ 5,	"stp",		"stp",		"rx STP blocked discards" },
62fc4c0664SSam Leffler #define	S_RX_VLAN_TYPE	AFTER(S_STP)
63fc4c0664SSam Leffler 	{ 5,	"rx_vlan_type",	"rx_vlant",	"rx VLAN type filter discards" },
64fc4c0664SSam Leffler #define	S_RX_VLAN_ID	AFTER(S_RX_VLAN_TYPE)
65fc4c0664SSam Leffler 	{ 5,	"rx_vlan_id",	"rx_vlani",	"rx VLAN Id filter discards" },
66fc4c0664SSam Leffler #define	S_BADSRC	AFTER(S_RX_VLAN_ID)
67fc4c0664SSam Leffler 	{ 5,	"badsrc",	"badsrc",	"rx invalid source discards" },
68fc4c0664SSam Leffler #define	S_BLACKLIST	AFTER(S_BADSRC)
69fc4c0664SSam Leffler 	{ 5,	"blacklist",	"blacklist",	"rx black list discards" },
70fc4c0664SSam Leffler #define	S_WHITELIST	AFTER(S_BLACKLIST)
71fc4c0664SSam Leffler 	{ 5,	"whitelist",	"whitelist",	"rx white list discards" },
72fc4c0664SSam Leffler #define	S_UNDERFLOW	AFTER(S_WHITELIST)
73fc4c0664SSam Leffler 	{ 5,	"underflow",	"underflow",	"rx underflow entry discards" },
74fc4c0664SSam Leffler #define	S_COLL_SINGLE	AFTER(S_UNDERFLOW)
75fc4c0664SSam Leffler 	{ 5,	"collision1",	"collision1",	"single collision frames" },
76fc4c0664SSam Leffler #define	S_COLL_MULTI	AFTER(S_COLL_SINGLE)
77fc4c0664SSam Leffler 	{ 5,	"collisionM",	"collisionM",	"multiple collision frames" },
78fc4c0664SSam Leffler #define	S_DEFERRED	AFTER(S_COLL_MULTI)
79fc4c0664SSam Leffler 	{ 5,	"deferred",	"deferred",	"deferred transmissions" },
80fc4c0664SSam Leffler #define	S_LATE		AFTER(S_DEFERRED)
81fc4c0664SSam Leffler 	{ 5,	"late",		"late",		"late collisions" },
82fc4c0664SSam Leffler #define	S_EXCESSIVE	AFTER(S_LATE)
83fc4c0664SSam Leffler 	{ 5,	"excessive",	"excessive",	"excessive collisions" },
84fc4c0664SSam Leffler #define	S_MACTX		AFTER(S_EXCESSIVE)
85fc4c0664SSam Leffler 	{ 7,	"mactx",	"mactx",	"internal MAC tx errors" },
86fc4c0664SSam Leffler #define	S_CARRIER	AFTER(S_MACTX)
87fc4c0664SSam Leffler 	{ 7,	"carrier",	"carrier",	"carrier sense errors" },
88fc4c0664SSam Leffler #define	S_TOOBIG	AFTER(S_CARRIER)
89fc4c0664SSam Leffler 	{ 7,	"toobig",	"toobig",	"tx large frame discards" },
90fc4c0664SSam Leffler #define	S_TX_VLAN_ID	AFTER(S_TOOBIG)
91fc4c0664SSam Leffler 	{ 7,	"tx_vlan_id",	"tx_vlani",	"tx VLAN Id filter discards" },
92fc4c0664SSam Leffler };
93fc4c0664SSam Leffler #define	S_LAST		S_TX_VLAN_ID
94fc4c0664SSam Leffler 
95fc4c0664SSam Leffler /*
96fc4c0664SSam Leffler  * Stat block returned by NPE with NPE_GETSTATS msg.
97fc4c0664SSam Leffler  */
98fc4c0664SSam Leffler struct npestats {
99fc4c0664SSam Leffler 	uint32_t dot3StatsAlignmentErrors;
100fc4c0664SSam Leffler 	uint32_t dot3StatsFCSErrors;
101fc4c0664SSam Leffler 	uint32_t dot3StatsInternalMacReceiveErrors;
102fc4c0664SSam Leffler 	uint32_t RxOverrunDiscards;
103fc4c0664SSam Leffler 	uint32_t RxLearnedEntryDiscards;
104fc4c0664SSam Leffler 	uint32_t RxLargeFramesDiscards;
105fc4c0664SSam Leffler 	uint32_t RxSTPBlockedDiscards;
106fc4c0664SSam Leffler 	uint32_t RxVLANTypeFilterDiscards;
107fc4c0664SSam Leffler 	uint32_t RxVLANIdFilterDiscards;
108fc4c0664SSam Leffler 	uint32_t RxInvalidSourceDiscards;
109fc4c0664SSam Leffler 	uint32_t RxBlackListDiscards;
110fc4c0664SSam Leffler 	uint32_t RxWhiteListDiscards;
111fc4c0664SSam Leffler 	uint32_t RxUnderflowEntryDiscards;
112fc4c0664SSam Leffler 	uint32_t dot3StatsSingleCollisionFrames;
113fc4c0664SSam Leffler 	uint32_t dot3StatsMultipleCollisionFrames;
114fc4c0664SSam Leffler 	uint32_t dot3StatsDeferredTransmissions;
115fc4c0664SSam Leffler 	uint32_t dot3StatsLateCollisions;
116fc4c0664SSam Leffler 	uint32_t dot3StatsExcessiveCollisions;
117fc4c0664SSam Leffler 	uint32_t dot3StatsInternalMacTransmitErrors;
118fc4c0664SSam Leffler 	uint32_t dot3StatsCarrierSenseErrors;
119fc4c0664SSam Leffler 	uint32_t TxLargeFrameDiscards;
120fc4c0664SSam Leffler 	uint32_t TxVLANIdFilterDiscards;
121fc4c0664SSam Leffler };
122fc4c0664SSam Leffler 
123fc4c0664SSam Leffler struct npestatfoo_p {
124fc4c0664SSam Leffler 	struct npestatfoo base;
125fc4c0664SSam Leffler 	char oid[80];
126fc4c0664SSam Leffler 	int mib[4];
127fc4c0664SSam Leffler 	struct npestats cur;
128fc4c0664SSam Leffler 	struct npestats total;
129fc4c0664SSam Leffler };
130fc4c0664SSam Leffler 
131fc4c0664SSam Leffler static void
npe_setifname(struct npestatfoo * wf0,const char * ifname)132fc4c0664SSam Leffler npe_setifname(struct npestatfoo *wf0, const char *ifname)
133fc4c0664SSam Leffler {
134fc4c0664SSam Leffler 	struct npestatfoo_p *wf = (struct npestatfoo_p *) wf0;
135fc4c0664SSam Leffler 	size_t len;
136fc4c0664SSam Leffler 
137fc4c0664SSam Leffler 	snprintf(wf->oid, sizeof(wf->oid), "dev.npe.%s.stats", ifname+3);
138fc4c0664SSam Leffler 	len = 4;
139fc4c0664SSam Leffler 	if (sysctlnametomib(wf->oid, wf->mib, &len) < 0)
140fc4c0664SSam Leffler 		err(1, "sysctlnametomib: %s", wf->oid);
141fc4c0664SSam Leffler }
142fc4c0664SSam Leffler 
143fc4c0664SSam Leffler static void
npe_collect(struct npestatfoo_p * wf,struct npestats * stats)144fc4c0664SSam Leffler npe_collect(struct npestatfoo_p *wf, struct npestats *stats)
145fc4c0664SSam Leffler {
146fc4c0664SSam Leffler 	size_t len = sizeof(struct npestats);
147fc4c0664SSam Leffler 	if (sysctl(wf->mib, 4, stats, &len, NULL, 0) < 0)
148fc4c0664SSam Leffler 		err(1, "sysctl: %s", wf->oid);
149fc4c0664SSam Leffler }
150fc4c0664SSam Leffler 
151fc4c0664SSam Leffler static void
npe_collect_cur(struct bsdstat * sf)15203b1adbcSAdrian Chadd npe_collect_cur(struct bsdstat *sf)
153fc4c0664SSam Leffler {
154fc4c0664SSam Leffler 	struct npestatfoo_p *wf = (struct npestatfoo_p *) sf;
155fc4c0664SSam Leffler 
156fc4c0664SSam Leffler 	npe_collect(wf, &wf->cur);
157fc4c0664SSam Leffler }
158fc4c0664SSam Leffler 
159fc4c0664SSam Leffler static void
npe_collect_tot(struct bsdstat * sf)16003b1adbcSAdrian Chadd npe_collect_tot(struct bsdstat *sf)
161fc4c0664SSam Leffler {
162fc4c0664SSam Leffler 	struct npestatfoo_p *wf = (struct npestatfoo_p *) sf;
163fc4c0664SSam Leffler 
164fc4c0664SSam Leffler 	npe_collect(wf, &wf->total);
165fc4c0664SSam Leffler }
166fc4c0664SSam Leffler 
167fc4c0664SSam Leffler static void
npe_update_tot(struct bsdstat * sf)16803b1adbcSAdrian Chadd npe_update_tot(struct bsdstat *sf)
169fc4c0664SSam Leffler {
170fc4c0664SSam Leffler 	struct npestatfoo_p *wf = (struct npestatfoo_p *) sf;
171fc4c0664SSam Leffler 
172fc4c0664SSam Leffler 	wf->total = wf->cur;
173fc4c0664SSam Leffler }
174fc4c0664SSam Leffler 
175fc4c0664SSam Leffler static int
npe_get_curstat(struct bsdstat * sf,int s,char b[],size_t bs)17603b1adbcSAdrian Chadd npe_get_curstat(struct bsdstat *sf, int s, char b[], size_t bs)
177fc4c0664SSam Leffler {
178fc4c0664SSam Leffler 	struct npestatfoo_p *wf = (struct npestatfoo_p *) sf;
179fc4c0664SSam Leffler #define	STAT(x) \
180fc4c0664SSam Leffler 	snprintf(b, bs, "%u", wf->cur.x - wf->total.x); return 1
181fc4c0664SSam Leffler 
182fc4c0664SSam Leffler 	switch (s) {
183fc4c0664SSam Leffler 	case S_ALIGN:		STAT(dot3StatsAlignmentErrors);
184fc4c0664SSam Leffler 	case S_FCS:		STAT(dot3StatsFCSErrors);
185fc4c0664SSam Leffler 	case S_MACRX:		STAT(dot3StatsInternalMacReceiveErrors);
186fc4c0664SSam Leffler 	case S_RXORN:		STAT(RxOverrunDiscards);
187fc4c0664SSam Leffler 	case S_LEARN:		STAT(RxLearnedEntryDiscards);
188fc4c0664SSam Leffler 	case S_LARGE:		STAT(RxLargeFramesDiscards);
189fc4c0664SSam Leffler 	case S_STP:		STAT(RxSTPBlockedDiscards);
190fc4c0664SSam Leffler 	case S_RX_VLAN_TYPE:	STAT(RxVLANTypeFilterDiscards);
191fc4c0664SSam Leffler 	case S_RX_VLAN_ID:	STAT(RxVLANIdFilterDiscards);
192fc4c0664SSam Leffler 	case S_BADSRC:		STAT(RxInvalidSourceDiscards);
193fc4c0664SSam Leffler 	case S_BLACKLIST:	STAT(RxBlackListDiscards);
194fc4c0664SSam Leffler 	case S_WHITELIST:	STAT(RxWhiteListDiscards);
195fc4c0664SSam Leffler 	case S_UNDERFLOW:	STAT(RxUnderflowEntryDiscards);
196fc4c0664SSam Leffler 	case S_COLL_SINGLE:	STAT(dot3StatsSingleCollisionFrames);
197fc4c0664SSam Leffler 	case S_COLL_MULTI:	STAT(dot3StatsMultipleCollisionFrames);
198fc4c0664SSam Leffler 	case S_DEFERRED:	STAT(dot3StatsDeferredTransmissions);
199fc4c0664SSam Leffler 	case S_LATE:		STAT(dot3StatsLateCollisions);
200fc4c0664SSam Leffler 	case S_EXCESSIVE:	STAT(dot3StatsExcessiveCollisions);
201fc4c0664SSam Leffler 	case S_MACTX:		STAT(dot3StatsInternalMacTransmitErrors);
202fc4c0664SSam Leffler 	case S_CARRIER:		STAT(dot3StatsCarrierSenseErrors);
203fc4c0664SSam Leffler 	case S_TOOBIG:		STAT(TxLargeFrameDiscards);
204fc4c0664SSam Leffler 	case S_TX_VLAN_ID:	STAT(TxVLANIdFilterDiscards);
205fc4c0664SSam Leffler 	}
206fc4c0664SSam Leffler 	b[0] = '\0';
207fc4c0664SSam Leffler 	return 0;
208fc4c0664SSam Leffler #undef STAT
209fc4c0664SSam Leffler }
210fc4c0664SSam Leffler 
211fc4c0664SSam Leffler static int
npe_get_totstat(struct bsdstat * sf,int s,char b[],size_t bs)21203b1adbcSAdrian Chadd npe_get_totstat(struct bsdstat *sf, int s, char b[], size_t bs)
213fc4c0664SSam Leffler {
214fc4c0664SSam Leffler 	struct npestatfoo_p *wf = (struct npestatfoo_p *) sf;
215fc4c0664SSam Leffler #define	STAT(x) \
216fc4c0664SSam Leffler 	snprintf(b, bs, "%u", wf->total.x); return 1
217fc4c0664SSam Leffler 
218fc4c0664SSam Leffler 	switch (s) {
219fc4c0664SSam Leffler 	case S_ALIGN:		STAT(dot3StatsAlignmentErrors);
220fc4c0664SSam Leffler 	case S_FCS:		STAT(dot3StatsFCSErrors);
221fc4c0664SSam Leffler 	case S_MACRX:		STAT(dot3StatsInternalMacReceiveErrors);
222fc4c0664SSam Leffler 	case S_RXORN:		STAT(RxOverrunDiscards);
223fc4c0664SSam Leffler 	case S_LEARN:		STAT(RxLearnedEntryDiscards);
224fc4c0664SSam Leffler 	case S_LARGE:		STAT(RxLargeFramesDiscards);
225fc4c0664SSam Leffler 	case S_STP:		STAT(RxSTPBlockedDiscards);
226fc4c0664SSam Leffler 	case S_RX_VLAN_TYPE:	STAT(RxVLANTypeFilterDiscards);
227fc4c0664SSam Leffler 	case S_RX_VLAN_ID:	STAT(RxVLANIdFilterDiscards);
228fc4c0664SSam Leffler 	case S_BADSRC:		STAT(RxInvalidSourceDiscards);
229fc4c0664SSam Leffler 	case S_BLACKLIST:	STAT(RxBlackListDiscards);
230fc4c0664SSam Leffler 	case S_WHITELIST:	STAT(RxWhiteListDiscards);
231fc4c0664SSam Leffler 	case S_UNDERFLOW:	STAT(RxUnderflowEntryDiscards);
232fc4c0664SSam Leffler 	case S_COLL_SINGLE:	STAT(dot3StatsSingleCollisionFrames);
233fc4c0664SSam Leffler 	case S_COLL_MULTI:	STAT(dot3StatsMultipleCollisionFrames);
234fc4c0664SSam Leffler 	case S_DEFERRED:	STAT(dot3StatsDeferredTransmissions);
235fc4c0664SSam Leffler 	case S_LATE:		STAT(dot3StatsLateCollisions);
236fc4c0664SSam Leffler 	case S_EXCESSIVE:	STAT(dot3StatsExcessiveCollisions);
237fc4c0664SSam Leffler 	case S_MACTX:		STAT(dot3StatsInternalMacTransmitErrors);
238fc4c0664SSam Leffler 	case S_CARRIER:		STAT(dot3StatsCarrierSenseErrors);
239fc4c0664SSam Leffler 	case S_TOOBIG:		STAT(TxLargeFrameDiscards);
240fc4c0664SSam Leffler 	case S_TX_VLAN_ID:	STAT(TxVLANIdFilterDiscards);
241fc4c0664SSam Leffler 	}
242fc4c0664SSam Leffler 	b[0] = '\0';
243fc4c0664SSam Leffler 	return 0;
244fc4c0664SSam Leffler #undef STAT
245fc4c0664SSam Leffler }
246fc4c0664SSam Leffler 
BSDSTAT_DEFINE_BOUNCE(npestatfoo)24703b1adbcSAdrian Chadd BSDSTAT_DEFINE_BOUNCE(npestatfoo)
248fc4c0664SSam Leffler 
249fc4c0664SSam Leffler struct npestatfoo *
250fc4c0664SSam Leffler npestats_new(const char *ifname, const char *fmtstring)
251fc4c0664SSam Leffler {
252fc4c0664SSam Leffler 	struct npestatfoo_p *wf;
253fc4c0664SSam Leffler 
254fc4c0664SSam Leffler 	wf = calloc(1, sizeof(struct npestatfoo_p));
255fc4c0664SSam Leffler 	if (wf != NULL) {
256*60caf0c9SCraig Rodrigues 		bsdstat_init(&wf->base.base, "npestats", npestats,
257*60caf0c9SCraig Rodrigues 		    nitems(npestats));
258fc4c0664SSam Leffler 		/* override base methods */
259fc4c0664SSam Leffler 		wf->base.base.collect_cur = npe_collect_cur;
260fc4c0664SSam Leffler 		wf->base.base.collect_tot = npe_collect_tot;
261fc4c0664SSam Leffler 		wf->base.base.get_curstat = npe_get_curstat;
262fc4c0664SSam Leffler 		wf->base.base.get_totstat = npe_get_totstat;
263fc4c0664SSam Leffler 		wf->base.base.update_tot = npe_update_tot;
264fc4c0664SSam Leffler 
265fc4c0664SSam Leffler 		/* setup bounce functions for public methods */
26603b1adbcSAdrian Chadd 		BSDSTAT_BOUNCE(wf, npestatfoo);
267fc4c0664SSam Leffler 
268fc4c0664SSam Leffler 		/* setup our public methods */
269fc4c0664SSam Leffler 		wf->base.setifname = npe_setifname;
270fc4c0664SSam Leffler 
271fc4c0664SSam Leffler 		npe_setifname(&wf->base, ifname);
272fc4c0664SSam Leffler 		wf->base.setfmt(&wf->base, fmtstring);
273fc4c0664SSam Leffler 	}
274fc4c0664SSam Leffler 	return &wf->base;
275fc4c0664SSam Leffler }
276