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) can select workqueue for packet proc...



details:   https://anonhg.NetBSD.org/src/rev/6ce666cada01
branches:  trunk
changeset: 462271:6ce666cada01
user:      knakahara <knakahara%NetBSD.org@localhost>
date:      Tue Jul 30 11:16:15 2019 +0000

description:
vmx(4) can select workqueue for packet processing like ixg(4).

diffstat:

 sys/arch/x86/pci/if_vmx.c |  60 +++++++++++++++++++++++++++++++++++++++++-----
 1 files changed, 53 insertions(+), 7 deletions(-)

diffs (177 lines):

diff -r c4ddd71f8f79 -r 6ce666cada01 sys/arch/x86/pci/if_vmx.c
--- a/sys/arch/x86/pci/if_vmx.c Tue Jul 30 11:11:15 2019 +0000
+++ b/sys/arch/x86/pci/if_vmx.c Tue Jul 30 11:16:15 2019 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: if_vmx.c,v 1.44 2019/07/29 10:28:57 knakahara Exp $    */
+/*     $NetBSD: if_vmx.c,v 1.45 2019/07/30 11:16:15 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.44 2019/07/29 10:28:57 knakahara Exp $");
+__KERNEL_RCSID(0, "$NetBSD: if_vmx.c,v 1.45 2019/07/30 11:16:15 knakahara Exp $");
 
 #include <sys/param.h>
 #include <sys/cpu.h>
@@ -30,6 +30,7 @@
 #include <sys/mbuf.h>
 #include <sys/sockio.h>
 #include <sys/pcq.h>
+#include <sys/workqueue.h>
 
 #include <net/bpf.h>
 #include <net/if.h>
@@ -96,6 +97,8 @@
 #define VMXNET3_RX_PROCESS_LIMIT 256
 #define VMXNET3_TX_PROCESS_LIMIT 256
 
+#define VMXNET3_WORKQUEUE_PRI PRI_SOFTNET
+
 /*
  * IP protocols that we can perform Tx checksum offloading of.
  */
@@ -225,6 +228,8 @@
        struct vmxnet3_rxqueue vxq_rxqueue;
 
        void *vxq_si;
+       bool vxq_workqueue;
+       struct work vxq_wq_cookie;
 };
 
 struct vmxnet3_statistics {
@@ -291,8 +296,10 @@
        u_int vmx_tx_intr_process_limit;
        u_int vmx_rx_process_limit;
        u_int vmx_tx_process_limit;
-
        struct sysctllog *vmx_sysctllog;
+
+       bool vmx_txrx_workqueue;
+       struct workqueue *vmx_queue_wq;
 };
 
 #define VMXNET3_STAT
@@ -378,6 +385,7 @@
 int vmxnet3_legacy_intr(void *);
 int vmxnet3_txrxq_intr(void *);
 void vmxnet3_handle_queue(void *);
+void vmxnet3_handle_queue_work(struct work *, void *);
 int vmxnet3_event_intr(void *);
 
 void vmxnet3_txstop(struct vmxnet3_softc *, struct vmxnet3_txqueue *);
@@ -849,6 +857,7 @@
        pci_chipset_tag_t pc = sc->vmx_pc;
        int i;
 
+       workqueue_destroy(sc->vmx_queue_wq);
        for (i = 0; i < sc->vmx_nintrs; i++) {
                struct vmxnet3_queue *vmxq =  &sc->vmx_queue[i];
 
@@ -866,7 +875,7 @@
        struct vmxnet3_queue *vmxq;
        pci_intr_handle_t *intr;
        void **ihs;
-       int intr_idx, i, use_queues;
+       int intr_idx, i, use_queues, error;
        const char *intrstr;
        char intrbuf[PCI_INTRSTR_LEN];
        char xnamebuf[32];
@@ -904,6 +913,15 @@
 
                vmxq->vxq_intr_idx = intr_idx;
        }
+       snprintf(xnamebuf, MAXCOMLEN, "%s_tx_rx", device_xname(sc->vmx_dev));
+       error = workqueue_create(&sc->vmx_queue_wq, xnamebuf,
+           vmxnet3_handle_queue_work, sc, VMXNET3_WORKQUEUE_PRI, IPL_NET,
+           WQ_PERCPU | WQ_MPSAFE);
+       if (error) {
+               aprint_error_dev(sc->vmx_dev, "workqueue_create failed\n");
+               return (-1);
+       }
+       sc->vmx_txrx_workqueue = false;
 
        intrstr = pci_intr_string(pc, *intr, intrbuf, sizeof(intrbuf));
 
@@ -1839,6 +1857,12 @@
            NULL, 0, NULL, 0, CTL_HW, CTL_CREATE, CTL_EOL);
        if (error)
                goto out;
+       error = sysctl_createv(log, 0, &rnode, NULL,
+           CTLFLAG_READWRITE, CTLTYPE_BOOL, "txrx_workqueue",
+           SYSCTL_DESCR("Use workqueue for packet processing"),
+           NULL, 0, &sc->vmx_txrx_workqueue, 0, CTL_CREATE, CTL_EOL);
+       if (error)
+               goto out;
 
        error = sysctl_createv(log, 0, &rnode, &rxnode,
            0, CTLTYPE_NODE, "rx",
@@ -2305,6 +2329,18 @@
        return more;
 }
 
+static inline void
+vmxnet3_sched_handle_queue(struct vmxnet3_softc *sc, struct vmxnet3_queue *vmxq)
+{
+
+       if (vmxq->vxq_workqueue) {
+               workqueue_enqueue(sc->vmx_queue_wq, &vmxq->vxq_wq_cookie,
+                   curcpu());
+       } else {
+               softint_schedule(vmxq->vxq_si);
+       }
+}
+
 int
 vmxnet3_legacy_intr(void *xsc)
 {
@@ -2339,7 +2375,7 @@
        VMXNET3_TXQ_UNLOCK(txq);
 
        if (txmore || rxmore) {
-               softint_schedule(sc->vmx_queue[0].vxq_si);
+               vmxnet3_sched_handle_queue(sc, &sc->vmx_queue[0]);
        } else {
                if_schedule_deferred_start(&sc->vmx_ethercom.ec_if);
                vmxnet3_enable_all_intrs(sc);
@@ -2363,6 +2399,7 @@
        sc = txq->vxtxq_sc;
        txlimit = sc->vmx_tx_intr_process_limit;
        rxlimit = sc->vmx_rx_intr_process_limit;
+       vmxq->vxq_workqueue = sc->vmx_txrx_workqueue;
 
        if (sc->vmx_intr_mask_mode == VMXNET3_IMM_ACTIVE)
                vmxnet3_disable_intr(sc, vmxq->vxq_intr_idx);
@@ -2376,7 +2413,7 @@
        VMXNET3_RXQ_UNLOCK(rxq);
 
        if (txmore || rxmore) {
-               softint_schedule(vmxq->vxq_si);
+               vmxnet3_sched_handle_queue(sc, vmxq);
        } else {
                /* for ALTQ */
                if (vmxq->vxq_id == 0)
@@ -2419,11 +2456,20 @@
        VMXNET3_RXQ_UNLOCK(rxq);
 
        if (txmore || rxmore)
-               softint_schedule(vmxq->vxq_si);
+               vmxnet3_sched_handle_queue(sc, vmxq);
        else
                vmxnet3_enable_intr(sc, vmxq->vxq_intr_idx);
 }
 
+void
+vmxnet3_handle_queue_work(struct work *wk, void *context)
+{
+       struct vmxnet3_queue *vmxq;
+
+       vmxq = container_of(wk, struct vmxnet3_queue, vxq_wq_cookie);
+       vmxnet3_handle_queue(vmxq);
+}
+
 int
 vmxnet3_event_intr(void *xsc)
 {



Home | Main Index | Thread Index | Old Index