Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/dev/pci Introduce packet handling in softint or kthread ...
details: https://anonhg.NetBSD.org/src/rev/1c460094d1dc
branches: trunk
changeset: 972379:1c460094d1dc
user: yamaguchi <yamaguchi%NetBSD.org@localhost>
date: Mon May 25 08:16:23 2020 +0000
description:
Introduce packet handling in softint or kthread for vioif(4)
diffstat:
sys/dev/pci/if_vioif.c | 543 +++++++++++++++++++++++++++++++++++++++++++-----
1 files changed, 481 insertions(+), 62 deletions(-)
diffs (truncated from 812 to 300 lines):
diff -r e2b1e2a8d87e -r 1c460094d1dc sys/dev/pci/if_vioif.c
--- a/sys/dev/pci/if_vioif.c Mon May 25 07:52:16 2020 +0000
+++ b/sys/dev/pci/if_vioif.c Mon May 25 08:16:23 2020 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: if_vioif.c,v 1.54 2020/05/25 07:52:16 yamaguchi Exp $ */
+/* $NetBSD: if_vioif.c,v 1.55 2020/05/25 08:16:23 yamaguchi Exp $ */
/*
* Copyright (c) 2010 Minoura Makoto.
@@ -26,7 +26,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: if_vioif.c,v 1.54 2020/05/25 07:52:16 yamaguchi Exp $");
+__KERNEL_RCSID(0, "$NetBSD: if_vioif.c,v 1.55 2020/05/25 08:16:23 yamaguchi Exp $");
#ifdef _KERNEL_OPT
#include "opt_net_mpsafe.h"
@@ -35,6 +35,7 @@
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/kernel.h>
+#include <sys/atomic.h>
#include <sys/bus.h>
#include <sys/condvar.h>
#include <sys/device.h>
@@ -46,6 +47,7 @@
#include <sys/cpu.h>
#include <sys/module.h>
#include <sys/pcq.h>
+#include <sys/workqueue.h>
#include <dev/pci/virtioreg.h>
#include <dev/pci/virtiovar.h>
@@ -205,6 +207,13 @@
* - txq_lock or rxq_lock cannot be held along with ctrlq_wait_lock
*/
+struct vioif_work {
+ struct work cookie;
+ void (*func)(void *);
+ void *arg;
+ unsigned int added;
+};
+
struct vioif_txqueue {
kmutex_t *txq_lock; /* lock for tx operations */
@@ -220,6 +229,10 @@
bus_dmamap_t *txq_dmamaps;
void *txq_deferred_transmit;
+ void *txq_handle_si;
+ struct vioif_work txq_work;
+ bool txq_workqueue;
+ bool txq_active;
};
struct vioif_rxqueue {
@@ -235,6 +248,10 @@
bus_dmamap_t *rxq_dmamaps;
void *rxq_softint;
+ void *rxq_handle_si;
+ struct vioif_work rxq_work;
+ bool rxq_workqueue;
+ bool rxq_active;
};
struct vioif_ctrlqueue {
@@ -263,6 +280,7 @@
struct vioif_softc {
device_t sc_dev;
+ struct sysctllog *sc_sysctllog;
struct virtio_softc *sc_virtio;
struct virtqueue *sc_vqs;
@@ -287,14 +305,29 @@
void *sc_kmem;
void *sc_ctl_softint;
+
+ struct workqueue *sc_txrx_workqueue;
+ bool sc_txrx_workqueue_sysctl;
+ u_int sc_tx_intr_process_limit;
+ u_int sc_tx_process_limit;
+ u_int sc_rx_intr_process_limit;
+ u_int sc_rx_process_limit;
};
#define VIRTIO_NET_TX_MAXNSEGS (16) /* XXX */
#define VIRTIO_NET_CTRL_MAC_MAXENTRIES (64) /* XXX */
+#define VIOIF_TX_INTR_PROCESS_LIMIT 256
+#define VIOIF_TX_PROCESS_LIMIT 256
+#define VIOIF_RX_INTR_PROCESS_LIMIT 0U
+#define VIOIF_RX_PROCESS_LIMIT 256
+
+#define VIOIF_WORKQUEUE_PRI PRI_SOFTNET
+
/* cfattach interface functions */
static int vioif_match(device_t, cfdata_t, void *);
static void vioif_attach(device_t, device_t, void *);
static void vioif_deferred_init(device_t);
+static int vioif_finalize_teardown(device_t);
/* ifnet interface functions */
static int vioif_init(struct ifnet *);
@@ -311,18 +344,36 @@
static void vioif_free_rx_mbuf(struct vioif_rxqueue *, int);
static void vioif_populate_rx_mbufs(struct vioif_rxqueue *);
static void vioif_populate_rx_mbufs_locked(struct vioif_rxqueue *);
-static int vioif_rx_deq(struct vioif_rxqueue *);
-static int vioif_rx_deq_locked(struct vioif_rxqueue *);
+static void vioif_rx_queue_clear(struct vioif_rxqueue *);
+static bool vioif_rx_deq_locked(struct vioif_softc *, struct virtio_softc *,
+ struct vioif_rxqueue *, u_int);
static int vioif_rx_intr(void *);
+static void vioif_rx_handle(void *);
+static void vioif_rx_sched_handle(struct vioif_softc *,
+ struct vioif_rxqueue *);
static void vioif_rx_softint(void *);
static void vioif_rx_drain(struct vioif_rxqueue *);
/* tx */
static int vioif_tx_intr(void *);
-static int vioif_tx_deq_locked(struct virtqueue *);
+static void vioif_tx_handle(void *);
+static void vioif_tx_sched_handle(struct vioif_softc *,
+ struct vioif_txqueue *);
+static void vioif_tx_queue_clear(struct vioif_txqueue *);
+static bool vioif_tx_deq_locked(struct vioif_softc *, struct virtio_softc *,
+ struct vioif_txqueue *, u_int);
static void vioif_tx_drain(struct vioif_txqueue *);
static void vioif_deferred_transmit(void *);
+/* workqueue */
+static struct workqueue*
+ vioif_workq_create(const char *, pri_t, int, int);
+static void vioif_workq_destroy(struct workqueue *);
+static void vioif_workq_work(struct work *, void *);
+static void vioif_work_set(struct vioif_work *, void(*)(void *), void *);
+static void vioif_work_add(struct workqueue *, struct vioif_work *);
+static void vioif_work_wait(struct workqueue *, struct vioif_work *);
+
/* other control */
static bool vioif_is_link_up(struct vioif_softc *);
static void vioif_update_link_status(struct vioif_softc *);
@@ -337,6 +388,7 @@
static int vioif_ctrl_mq_vq_pairs_set(struct vioif_softc *, int);
static void vioif_enable_interrupt_vqpairs(struct vioif_softc *);
static void vioif_disable_interrupt_vqpairs(struct vioif_softc *);
+static int vioif_setup_sysctl(struct vioif_softc *);
CFATTACH_DECL_NEW(vioif, sizeof(struct vioif_softc),
vioif_match, vioif_attach, NULL, NULL);
@@ -672,6 +724,7 @@
struct ifnet *ifp = &sc->sc_ethercom.ec_if;
u_int softint_flags;
int r, i, nvqs=0, req_flags;
+ char xnamebuf[MAXCOMLEN];
if (virtio_child(vsc) != NULL) {
aprint_normal(": child already attached for %s; "
@@ -686,6 +739,17 @@
sc->sc_max_nvq_pairs = 1;
sc->sc_req_nvq_pairs = 1;
sc->sc_act_nvq_pairs = 1;
+ sc->sc_txrx_workqueue_sysctl = true;
+ sc->sc_tx_intr_process_limit = VIOIF_TX_INTR_PROCESS_LIMIT;
+ sc->sc_tx_process_limit = VIOIF_TX_PROCESS_LIMIT;
+ sc->sc_rx_intr_process_limit = VIOIF_RX_INTR_PROCESS_LIMIT;
+ sc->sc_rx_process_limit = VIOIF_RX_PROCESS_LIMIT;
+
+ snprintf(xnamebuf, sizeof(xnamebuf), "%s_txrx", device_xname(self));
+ sc->sc_txrx_workqueue = vioif_workq_create(xnamebuf, VIOIF_WORKQUEUE_PRI,
+ IPL_NET, WQ_PERCPU | WQ_MPSAFE);
+ if (sc->sc_txrx_workqueue == NULL)
+ goto err;
req_flags = 0;
@@ -774,6 +838,13 @@
aprint_error_dev(self, "cannot establish rx softint\n");
goto err;
}
+ rxq->rxq_handle_si = softint_establish(softint_flags,
+ vioif_rx_handle, rxq);
+ if (rxq->rxq_handle_si == NULL) {
+ aprint_error_dev(self, "cannot establish rx softint\n");
+ goto err;
+ }
+
snprintf(qname, sizeof(qname), "rx%d", i);
r = virtio_alloc_vq(vsc, rxq->rxq_vq, nvqs,
MCLBYTES+sizeof(struct virtio_net_hdr), nvqs, qname);
@@ -783,14 +854,23 @@
rxq->rxq_vq->vq_intrhand = vioif_rx_intr;
rxq->rxq_vq->vq_intrhand_arg = (void *)rxq;
rxq->rxq_stopping = true;
+ vioif_work_set(&rxq->rxq_work, vioif_rx_handle, rxq);
txq->txq_lock = mutex_obj_alloc(MUTEX_DEFAULT, IPL_NET);
+
txq->txq_deferred_transmit = softint_establish(softint_flags,
vioif_deferred_transmit, txq);
if (txq->txq_deferred_transmit == NULL) {
aprint_error_dev(self, "cannot establish tx softint\n");
goto err;
}
+ txq->txq_handle_si = softint_establish(softint_flags,
+ vioif_tx_handle, txq);
+ if (txq->txq_handle_si == NULL) {
+ aprint_error_dev(self, "cannot establish tx softint\n");
+ goto err;
+ }
+
snprintf(qname, sizeof(qname), "tx%d", i);
r = virtio_alloc_vq(vsc, txq->txq_vq, nvqs,
sizeof(struct virtio_net_hdr)
@@ -804,6 +884,7 @@
txq->txq_link_active = sc->sc_link_active;
txq->txq_stopping = false;
txq->txq_intrq = pcq_create(txq->txq_vq->vq_num, KM_SLEEP);
+ vioif_work_set(&txq->txq_work, vioif_tx_handle, txq);
}
if (sc->sc_has_ctrl) {
@@ -840,6 +921,11 @@
if (virtio_child_attach_finish(vsc) != 0)
goto err;
+ if (vioif_setup_sysctl(sc) != 0) {
+ aprint_error_dev(self, "unable to create sysctl node\n");
+ /* continue */
+ }
+
strlcpy(ifp->if_xname, device_xname(self), IFNAMSIZ);
ifp->if_softc = sc;
ifp->if_flags = IFF_BROADCAST | IFF_SIMPLEX | IFF_MULTICAST;
@@ -881,11 +967,21 @@
rxq->rxq_softint = NULL;
}
+ if (rxq->rxq_handle_si) {
+ softint_disestablish(rxq->rxq_handle_si);
+ rxq->rxq_handle_si = NULL;
+ }
+
if (txq->txq_lock) {
mutex_obj_free(txq->txq_lock);
txq->txq_lock = NULL;
}
+ if (txq->txq_handle_si) {
+ softint_disestablish(txq->txq_handle_si);
+ txq->txq_handle_si = NULL;
+ }
+
if (txq->txq_deferred_transmit) {
softint_disestablish(txq->txq_deferred_transmit);
txq->txq_deferred_transmit = NULL;
@@ -906,11 +1002,25 @@
virtio_free_vq(vsc, &sc->sc_vqs[--nvqs]);
vioif_free_queues(sc);
-
virtio_child_attach_failed(vsc);
+ config_finalize_register(self, vioif_finalize_teardown);
+
return;
}
+static int
+vioif_finalize_teardown(device_t self)
+{
+ struct vioif_softc *sc = device_private(self);
+
+ if (sc->sc_txrx_workqueue != NULL) {
+ vioif_workq_destroy(sc->sc_txrx_workqueue);
+ sc->sc_txrx_workqueue = NULL;
+ }
+
+ return 0;
+}
+
/* we need interrupts to make promiscuous mode off */
static void
vioif_deferred_init(device_t self)
@@ -1049,8 +1159,26 @@
/* only way to stop I/O and DMA is resetting... */
virtio_reset(vsc);
- for (i = 0; i < sc->sc_act_nvq_pairs; i++)
- vioif_rx_deq(&sc->sc_rxq[i]);
+
+ /* rendezvous for finish of handlers */
+ for (i = 0; i < sc->sc_act_nvq_pairs; i++) {
+ txq = &sc->sc_txq[i];
+ rxq = &sc->sc_rxq[i];
Home |
Main Index |
Thread Index |
Old Index