Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/arch/x86/pci vmx(4) support if_transmit and Tx multiqueu...
details: https://anonhg.NetBSD.org/src/rev/ead265eb8480
branches: trunk
changeset: 1000396:ead265eb8480
user: knakahara <knakahara%NetBSD.org@localhost>
date: Wed Jul 24 10:13:14 2019 +0000
description:
vmx(4) support if_transmit and Tx multiqueue (1/2)
Implemented Tx processing only. Fix Tx interrupt handler later.
diffstat:
sys/arch/x86/pci/if_vmx.c | 105 +++++++++++++++++++++++++++++++++++++++++++++-
1 files changed, 103 insertions(+), 2 deletions(-)
diffs (174 lines):
diff -r 3211cdd39536 -r ead265eb8480 sys/arch/x86/pci/if_vmx.c
--- a/sys/arch/x86/pci/if_vmx.c Wed Jul 24 10:05:57 2019 +0000
+++ b/sys/arch/x86/pci/if_vmx.c Wed Jul 24 10:13:14 2019 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: if_vmx.c,v 1.37 2019/07/23 04:50:05 knakahara Exp $ */
+/* $NetBSD: if_vmx.c,v 1.38 2019/07/24 10:13:14 knakahara Exp $ */
/* $OpenBSD: if_vmx.c,v 1.16 2014/01/22 06:04:17 brad Exp $ */
/*
@@ -19,7 +19,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: if_vmx.c,v 1.37 2019/07/23 04:50:05 knakahara Exp $");
+__KERNEL_RCSID(0, "$NetBSD: if_vmx.c,v 1.38 2019/07/24 10:13:14 knakahara Exp $");
#include <sys/param.h>
#include <sys/cpu.h>
@@ -29,6 +29,7 @@
#include <sys/device.h>
#include <sys/mbuf.h>
#include <sys/sockio.h>
+#include <sys/pcq.h>
#include <net/bpf.h>
#include <net/if.h>
@@ -174,6 +175,7 @@
int vxtxq_id;
int vxtxq_intr_idx;
int vxtxq_watchdog;
+ pcq_t *vxtxq_interq;
struct vmxnet3_txring vxtxq_cmd_ring;
struct vmxnet3_comp_ring vxtxq_comp_ring;
struct vmxnet3_txq_stats vxtxq_stats;
@@ -371,6 +373,8 @@
int vmxnet3_txq_encap(struct vmxnet3_txqueue *, struct mbuf **);
void vmxnet3_start_locked(struct ifnet *);
void vmxnet3_start(struct ifnet *);
+void vmxnet3_transmit_locked(struct ifnet *, struct vmxnet3_txqueue *);
+int vmxnet3_transmit(struct ifnet *, struct mbuf *);
void vmxnet3_set_rxfilter(struct vmxnet3_softc *);
int vmxnet3_ioctl(struct ifnet *, u_long, void *);
@@ -1061,6 +1065,8 @@
txq->vxtxq_comp_ring.vxcr_ndesc = sc->vmx_ntxdescs;
+ txq->vxtxq_interq = pcq_create(sc->vmx_ntxdescs, KM_SLEEP);
+
return (0);
}
@@ -1134,12 +1140,17 @@
vmxnet3_destroy_txq(struct vmxnet3_txqueue *txq)
{
struct vmxnet3_txring *txr;
+ struct mbuf *m;
txr = &txq->vxtxq_cmd_ring;
txq->vxtxq_sc = NULL;
txq->vxtxq_id = -1;
+ while ((m = pcq_get(txq->vxtxq_interq)) != NULL)
+ m_freem(m);
+ pcq_destroy(txq->vxtxq_interq);
+
if (txr->vxtxr_txbuf != NULL) {
kmem_free(txr->vxtxr_txbuf,
txr->vxtxr_ndesc * sizeof(struct vmxnet3_txbuf));
@@ -1704,6 +1715,7 @@
ifp->if_extflags = IFEF_MPSAFE;
ifp->if_ioctl = vmxnet3_ioctl;
ifp->if_start = vmxnet3_start;
+ ifp->if_transmit = vmxnet3_transmit;
ifp->if_watchdog = NULL;
ifp->if_init = vmxnet3_init;
ifp->if_stop = vmxnet3_stop;
@@ -2805,6 +2817,95 @@
VMXNET3_TXQ_UNLOCK(txq);
}
+static int
+vmxnet3_select_txqueue(struct ifnet *ifp, struct mbuf *m __unused)
+{
+ struct vmxnet3_softc *sc;
+ u_int cpuid;
+
+ sc = ifp->if_softc;
+ cpuid = cpu_index(curcpu());
+ /*
+ * Furure work
+ * We should select txqueue to even up the load even if ncpu is
+ * different from sc->vmx_ntxqueues. Currently, the load is not
+ * even, that is, when ncpu is six and ntxqueues is four, the load
+ * of vmx_txq[0] and vmx_txq[1] is higher than vmx_txq[2] and
+ * vmx_txq[3] because CPU#4 always uses vmx_txq[0] and CPU#5 always
+ * uses vmx_txq[1].
+ * Furthermore, we should not use random value to select txqueue to
+ * avoid reordering. We should use flow information of mbuf.
+ */
+ return cpuid % sc->vmx_ntxqueues;
+}
+
+void
+vmxnet3_transmit_locked(struct ifnet *ifp, struct vmxnet3_txqueue *txq)
+{
+ struct vmxnet3_softc *sc;
+ struct vmxnet3_txring *txr;
+ struct mbuf *m_head;
+ int tx;
+
+ sc = ifp->if_softc;
+ txr = &txq->vxtxq_cmd_ring;
+ tx = 0;
+
+ VMXNET3_TXQ_LOCK_ASSERT(txq);
+
+ if ((ifp->if_flags & IFF_RUNNING) == 0 ||
+ sc->vmx_link_active == 0)
+ return;
+
+ for (;;) {
+ m_head = pcq_peek(txq->vxtxq_interq);
+ if (m_head == NULL)
+ break;
+
+ if (vmxnet3_txring_avail(txr) < VMXNET3_TX_MAXSEGS)
+ break;
+
+ m_head = pcq_get(txq->vxtxq_interq);
+ if (m_head == NULL)
+ break;
+
+ if (vmxnet3_txq_encap(txq, &m_head) != 0) {
+ if (m_head != NULL)
+ m_freem(m_head);
+ break;
+ }
+
+ tx++;
+ bpf_mtap(ifp, m_head, BPF_D_OUT);
+ }
+
+ if (tx > 0)
+ txq->vxtxq_watchdog = VMXNET3_WATCHDOG_TIMEOUT;
+}
+
+int
+vmxnet3_transmit(struct ifnet *ifp, struct mbuf *m)
+{
+ struct vmxnet3_softc *sc;
+ struct vmxnet3_txqueue *txq;
+ int qid;
+
+ qid = vmxnet3_select_txqueue(ifp, m);
+ sc = ifp->if_softc;
+ txq = &sc->vmx_txq[qid];
+
+ if (__predict_false(!pcq_put(txq->vxtxq_interq, m))) {
+ m_freem(m);
+ return ENOBUFS;
+ }
+
+ VMXNET3_TXQ_LOCK(txq);
+ vmxnet3_transmit_locked(ifp, txq);
+ VMXNET3_TXQ_UNLOCK(txq);
+
+ return 0;
+}
+
void
vmxnet3_set_rxfilter(struct vmxnet3_softc *sc)
{
Home |
Main Index |
Thread Index |
Old Index