summaryrefslogtreecommitdiff
path: root/sys/dev/pci/if_myx.c
diff options
context:
space:
mode:
authorDavid Gwynne <dlg@cvs.openbsd.org>2020-07-17 02:58:15 +0000
committerDavid Gwynne <dlg@cvs.openbsd.org>2020-07-17 02:58:15 +0000
commite631141509a5f73fcd5725b4feb2d55056875dca (patch)
treed29e5280d1e2bce3acf03fd968a554c1beb938cc /sys/dev/pci/if_myx.c
parent1b54105f6a84816f0d15212caa903c46b79ffa1f (diff)
add kstats to myx.
myx is unusually minimal, so there's not a lot of information that the chip provides. the most interesting is the number of packets the chip drops cos of a lack of space on the rx rings.
Diffstat (limited to 'sys/dev/pci/if_myx.c')
-rw-r--r--sys/dev/pci/if_myx.c207
1 files changed, 206 insertions, 1 deletions
diff --git a/sys/dev/pci/if_myx.c b/sys/dev/pci/if_myx.c
index c2885f88eb7..c155cea092a 100644
--- a/sys/dev/pci/if_myx.c
+++ b/sys/dev/pci/if_myx.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: if_myx.c,v 1.109 2020/07/10 13:26:38 patrick Exp $ */
+/* $OpenBSD: if_myx.c,v 1.110 2020/07/17 02:58:14 dlg Exp $ */
/*
* Copyright (c) 2007 Reyk Floeter <reyk@openbsd.org>
@@ -21,6 +21,7 @@
*/
#include "bpfilter.h"
+#include "kstat.h"
#include <sys/param.h>
#include <sys/systm.h>
@@ -35,6 +36,7 @@
#include <sys/proc.h>
#include <sys/queue.h>
#include <sys/rwlock.h>
+#include <sys/kstat.h>
#include <machine/bus.h>
#include <machine/intr.h>
@@ -159,6 +161,12 @@ struct myx_softc {
volatile u_int8_t sc_linkdown;
struct rwlock sc_sff_lock;
+
+#if NKSTAT > 0
+ struct mutex sc_kstat_mtx;
+ struct timeout sc_kstat_tmo;
+ struct kstat *sc_kstat;
+#endif
};
#define MYX_RXSMALL_SIZE MCLBYTES
@@ -230,6 +238,12 @@ void myx_tx_free(struct myx_softc *);
void myx_refill(void *);
+#if NKSTAT > 0
+void myx_kstat_attach(struct myx_softc *);
+void myx_kstat_start(struct myx_softc *);
+void myx_kstat_stop(struct myx_softc *);
+#endif
+
struct cfdriver myx_cd = {
NULL, "myx", DV_IFNET
};
@@ -504,6 +518,10 @@ myx_attachhook(struct device *self)
goto freecmd;
}
+#if NKSTAT > 0
+ myx_kstat_attach(sc);
+#endif
+
ifp->if_softc = sc;
ifp->if_flags = IFF_BROADCAST | IFF_SIMPLEX | IFF_MULTICAST;
ifp->if_xflags = IFXF_MPSAFE;
@@ -1235,6 +1253,10 @@ myx_up(struct myx_softc *sc)
SET(ifp->if_flags, IFF_RUNNING);
ifq_restart(&ifp->if_snd);
+#if NKSTAT > 0
+ timeout_add_sec(&sc->sc_kstat_tmo, 1);
+#endif
+
return;
empty_rx_ring_big:
@@ -1402,6 +1424,11 @@ myx_down(struct myx_softc *sc)
myx_tx_empty(sc);
myx_tx_free(sc);
+#if NKSTAT > 0
+ myx_kstat_stop(sc);
+ sc->sc_sts = NULL;
+#endif
+
/* the sleep shizz above already synced this dmamem */
myx_dmamem_free(sc, &sc->sc_sts_dma);
@@ -2063,3 +2090,181 @@ myx_tx_free(struct myx_softc *sc)
free(sc->sc_tx_slots, M_DEVBUF, sizeof(*ms) * sc->sc_tx_ring_count);
}
+
+#if NKSTAT > 0
+enum myx_counters {
+ myx_stat_dropped_pause,
+ myx_stat_dropped_ucast_filtered,
+ myx_stat_dropped_bad_crc32,
+ myx_stat_dropped_bad_phy,
+ myx_stat_dropped_mcast_filtered,
+ myx_stat_send_done,
+ myx_stat_dropped_link_overflow,
+ myx_stat_dropped_link,
+ myx_stat_dropped_runt,
+ myx_stat_dropped_overrun,
+ myx_stat_dropped_no_small_bufs,
+ myx_stat_dropped_no_large_bufs,
+
+ myx_ncounters,
+};
+
+struct myx_counter {
+ const char *mc_name;
+ unsigned int mc_offset;
+};
+
+#define MYX_C_OFF(_f) offsetof(struct myx_status, _f)
+
+static const struct myx_counter myx_counters[myx_ncounters] = {
+ { "pause drops", MYX_C_OFF(ms_dropped_pause), },
+ { "ucast filtered", MYX_C_OFF(ms_dropped_unicast), },
+ { "bad crc32", MYX_C_OFF(ms_dropped_pause), },
+ { "bad phy", MYX_C_OFF(ms_dropped_phyerr), },
+ { "mcast filtered", MYX_C_OFF(ms_dropped_mcast), },
+ { "tx done", MYX_C_OFF(ms_txdonecnt), },
+ { "rx discards", MYX_C_OFF(ms_dropped_linkoverflow), },
+ { "rx errors", MYX_C_OFF(ms_dropped_linkerror), },
+ { "rx undersize", MYX_C_OFF(ms_dropped_runt), },
+ { "rx oversize", MYX_C_OFF(ms_dropped_overrun), },
+ { "small discards", MYX_C_OFF(ms_dropped_smallbufunderrun), },
+ { "large discards", MYX_C_OFF(ms_dropped_bigbufunderrun), },
+};
+
+struct myx_kstats {
+ struct kstat_kv mk_counters[myx_ncounters];
+ struct kstat_kv mk_rdma_tags_available;
+};
+
+struct myx_kstat_cache {
+ uint32_t mkc_counters[myx_ncounters];
+};
+
+struct myx_kstat_state {
+ struct myx_kstat_cache mks_caches[2];
+ unsigned int mks_gen;
+};
+
+int
+myx_kstat_read(struct kstat *ks)
+{
+ struct myx_softc *sc = ks->ks_softc;
+ struct myx_kstats *mk = ks->ks_data;
+ struct myx_kstat_state *mks = ks->ks_ptr;
+ unsigned int gen = (mks->mks_gen++ & 1);
+ struct myx_kstat_cache *omkc = &mks->mks_caches[gen];
+ struct myx_kstat_cache *nmkc = &mks->mks_caches[!gen];
+ unsigned int i = 0;
+
+ volatile struct myx_status *sts = sc->sc_sts;
+ bus_dmamap_t map = sc->sc_sts_dma.mxm_map;
+
+ if (sc->sc_sts == NULL)
+ return (0); /* counters are valid, just not updated */
+
+ getnanouptime(&ks->ks_updated);
+
+ bus_dmamap_sync(sc->sc_dmat, map, 0, map->dm_mapsize,
+ BUS_DMASYNC_POSTREAD);
+ for (i = 0; i < myx_ncounters; i++) {
+ const struct myx_counter *mc = &myx_counters[i];
+ nmkc->mkc_counters[i] =
+ bemtoh32((uint32_t *)((uint8_t *)sts + mc->mc_offset));
+ }
+
+ kstat_kv_u32(&mk->mk_rdma_tags_available) =
+ bemtoh32(&sts->ms_rdmatags_available);
+ bus_dmamap_sync(sc->sc_dmat, map, 0, map->dm_mapsize,
+ BUS_DMASYNC_PREREAD);
+
+ for (i = 0; i < myx_ncounters; i++) {
+ kstat_kv_u64(&mk->mk_counters[i]) +=
+ nmkc->mkc_counters[i] - omkc->mkc_counters[i];
+ }
+
+ return (0);
+}
+
+void
+myx_kstat_tick(void *arg)
+{
+ struct myx_softc *sc = arg;
+
+ if (!ISSET(sc->sc_ac.ac_if.if_flags, IFF_RUNNING))
+ return;
+
+ timeout_add_sec(&sc->sc_kstat_tmo, 4);
+
+ if (!mtx_enter_try(&sc->sc_kstat_mtx))
+ return;
+
+ myx_kstat_read(sc->sc_kstat);
+
+ mtx_leave(&sc->sc_kstat_mtx);
+}
+
+void
+myx_kstat_start(struct myx_softc *sc)
+{
+ if (sc->sc_kstat == NULL)
+ return;
+
+ myx_kstat_tick(sc);
+}
+
+void
+myx_kstat_stop(struct myx_softc *sc)
+{
+ struct myx_kstat_state *mks;
+
+ if (sc->sc_kstat == NULL)
+ return;
+
+ timeout_del_barrier(&sc->sc_kstat_tmo);
+
+ mks = sc->sc_kstat->ks_ptr;
+
+ mtx_enter(&sc->sc_kstat_mtx);
+ memset(mks, 0, sizeof(*mks));
+ mtx_leave(&sc->sc_kstat_mtx);
+}
+
+void
+myx_kstat_attach(struct myx_softc *sc)
+{
+ struct kstat *ks;
+ struct myx_kstats *mk;
+ struct myx_kstat_state *mks;
+ unsigned int i;
+
+ mtx_init(&sc->sc_kstat_mtx, IPL_SOFTCLOCK);
+ timeout_set(&sc->sc_kstat_tmo, myx_kstat_tick, sc);
+
+ ks = kstat_create(DEVNAME(sc), 0, "myx-stats", 0, KSTAT_T_KV, 0);
+ if (ks == NULL)
+ return;
+
+ mk = malloc(sizeof(*mk), M_DEVBUF, M_WAITOK|M_ZERO);
+ for (i = 0; i < myx_ncounters; i++) {
+ const struct myx_counter *mc = &myx_counters[i];
+
+ kstat_kv_unit_init(&mk->mk_counters[i], mc->mc_name,
+ KSTAT_KV_T_COUNTER64, KSTAT_KV_U_PACKETS);
+ }
+ kstat_kv_init(&mk->mk_rdma_tags_available, "rdma tags free",
+ KSTAT_KV_T_UINT32);
+
+ mks = malloc(sizeof(*mks), M_DEVBUF, M_WAITOK|M_ZERO);
+ /* these start at 0 */
+
+ kstat_set_mutex(ks, &sc->sc_kstat_mtx);
+ ks->ks_data = mk;
+ ks->ks_datalen = sizeof(*mk);
+ ks->ks_read = myx_kstat_read;
+ ks->ks_ptr = mks;
+
+ ks->ks_softc = sc;
+ sc->sc_kstat = ks;
+ kstat_install(ks);
+}
+#endif /* NKSTAT > 0 */