Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/dev/pci mcx: Add if_transmit callback to enable more tha...
details: https://anonhg.NetBSD.org/src/rev/aab2c74d76d8
branches: trunk
changeset: 1018499:aab2c74d76d8
user: jmcneill <jmcneill%NetBSD.org@localhost>
date: Fri Feb 05 22:23:30 2021 +0000
description:
mcx: Add if_transmit callback to enable more than one tx queue
diffstat:
sys/dev/pci/if_mcx.c | 85 ++++++++++++++++++++++++++++++++++++++++++++++++---
1 files changed, 79 insertions(+), 6 deletions(-)
diffs (171 lines):
diff -r 8d662fd06d77 -r aab2c74d76d8 sys/dev/pci/if_mcx.c
--- a/sys/dev/pci/if_mcx.c Fri Feb 05 22:15:44 2021 +0000
+++ b/sys/dev/pci/if_mcx.c Fri Feb 05 22:23:30 2021 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: if_mcx.c,v 1.15 2021/01/30 21:26:32 jmcneill Exp $ */
+/* $NetBSD: if_mcx.c,v 1.16 2021/02/05 22:23:30 jmcneill Exp $ */
/* $OpenBSD: if_mcx.c,v 1.98 2021/01/27 07:46:11 dlg Exp $ */
/*
@@ -23,7 +23,7 @@
#endif
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: if_mcx.c,v 1.15 2021/01/30 21:26:32 jmcneill Exp $");
+__KERNEL_RCSID(0, "$NetBSD: if_mcx.c,v 1.16 2021/02/05 22:23:30 jmcneill Exp $");
#include <sys/param.h>
#include <sys/systm.h>
@@ -41,6 +41,8 @@
#include <sys/kmem.h>
#include <sys/bus.h>
#include <sys/interrupt.h>
+#include <sys/pcq.h>
+#include <sys/cpu.h>
#include <machine/intr.h>
@@ -70,6 +72,8 @@
#define CALLOUT_FLAGS 0
/* #endif */
+#define MCX_TXQ_NUM 2048
+
#define BUS_DMASYNC_PRERW (BUS_DMASYNC_PREREAD|BUS_DMASYNC_PREWRITE)
#define BUS_DMASYNC_POSTRW (BUS_DMASYNC_POSTREAD|BUS_DMASYNC_POSTWRITE)
@@ -2318,6 +2322,9 @@
struct mcx_tx {
struct mcx_softc *tx_softc;
+ kmutex_t tx_lock;
+ pcq_t *tx_pcq;
+ void *tx_softint;
int tx_uar;
int tx_sqn;
@@ -2624,6 +2631,8 @@
static void mcx_stop(struct ifnet *, int);
static int mcx_ioctl(struct ifnet *, u_long, void *);
static void mcx_start(struct ifnet *);
+static int mcx_transmit(struct ifnet *, struct mbuf *);
+static void mcx_deferred_transmit(void *);
static void mcx_watchdog(struct ifnet *);
static void mcx_media_add_types(struct mcx_softc *);
static void mcx_media_status(struct ifnet *, struct ifmediareq *);
@@ -2954,6 +2963,9 @@
ifp->if_stop = mcx_stop;
ifp->if_ioctl = mcx_ioctl;
ifp->if_start = mcx_start;
+ if (sc->sc_nqueues > 1) {
+ ifp->if_transmit = mcx_transmit;
+ }
ifp->if_watchdog = mcx_watchdog;
ifp->if_mtu = sc->sc_hardmtu;
ifp->if_capabilities = IFCAP_CSUM_IPv4_Rx | IFCAP_CSUM_IPv4_Tx |
@@ -3009,6 +3021,10 @@
callout_setfunc(&rx->rx_refill, mcx_refill, rx);
tx->tx_softc = sc;
+ mutex_init(&tx->tx_lock, MUTEX_DEFAULT, IPL_NET);
+ tx->tx_pcq = pcq_create(MCX_TXQ_NUM, KM_SLEEP);
+ tx->tx_softint = softint_establish(SOFTINT_NET|SOFTINT_MPSAFE,
+ mcx_deferred_transmit, tx);
snprintf(intrxname, sizeof(intrxname), "%s queue %d",
DEVNAME(sc), i);
@@ -7763,11 +7779,9 @@
}
static void
-mcx_start(struct ifnet *ifp)
+mcx_send_common_locked(struct ifnet *ifp, struct mcx_tx *tx, bool is_transmit)
{
struct mcx_softc *sc = ifp->if_softc;
- /* mcx_start() always uses TX ring[0] */
- struct mcx_tx *tx = &sc->sc_queues[0].q_tx;
struct mcx_sq_entry *sq, *sqe;
struct mcx_sq_entry_seg *sqs;
struct mcx_slot *ms;
@@ -7779,6 +7793,11 @@
size_t bf_base;
int i, seg, nseg;
+ KASSERT(mutex_owned(&tx->tx_lock));
+
+ if ((ifp->if_flags & IFF_RUNNING) == 0)
+ return;
+
bf_base = (tx->tx_uar * MCX_PAGE_SIZE) + MCX_UAR_BF;
idx = tx->tx_prod % (1 << MCX_LOG_SQ_SIZE);
@@ -7798,7 +7817,11 @@
break;
}
- IFQ_DEQUEUE(&ifp->if_snd, m);
+ if (is_transmit) {
+ m = pcq_get(tx->tx_pcq);
+ } else {
+ IFQ_DEQUEUE(&ifp->if_snd, m);
+ }
if (m == NULL) {
break;
}
@@ -7917,6 +7940,56 @@
}
static void
+mcx_start(struct ifnet *ifp)
+{
+ struct mcx_softc *sc = ifp->if_softc;
+ /* mcx_start() always uses TX ring[0] */
+ struct mcx_tx *tx = &sc->sc_queues[0].q_tx;
+
+ mutex_enter(&tx->tx_lock);
+ if (!ISSET(ifp->if_flags, IFF_OACTIVE)) {
+ mcx_send_common_locked(ifp, tx, false);
+ }
+ mutex_exit(&tx->tx_lock);
+}
+
+static int
+mcx_transmit(struct ifnet *ifp, struct mbuf *m)
+{
+ struct mcx_softc *sc = ifp->if_softc;
+ struct mcx_tx *tx;
+
+ tx = &sc->sc_queues[cpu_index(curcpu()) % sc->sc_nqueues].q_tx;
+ if (__predict_false(!pcq_put(tx->tx_pcq, m))) {
+ m_freem(m);
+ return ENOBUFS;
+ }
+
+ if (mutex_tryenter(&tx->tx_lock)) {
+ mcx_send_common_locked(ifp, tx, true);
+ mutex_exit(&tx->tx_lock);
+ } else {
+ softint_schedule(tx->tx_softint);
+ }
+
+ return 0;
+}
+
+static void
+mcx_deferred_transmit(void *arg)
+{
+ struct mcx_tx *tx = arg;
+ struct mcx_softc *sc = tx->tx_softc;
+ struct ifnet *ifp = &sc->sc_ec.ec_if;
+
+ mutex_enter(&tx->tx_lock);
+ if (pcq_peek(tx->tx_pcq) != NULL) {
+ mcx_send_common_locked(ifp, tx, true);
+ }
+ mutex_exit(&tx->tx_lock);
+}
+
+static void
mcx_watchdog(struct ifnet *ifp)
{
}
Home |
Main Index |
Thread Index |
Old Index