Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/dev/pci Add multiqueue support, virtio(4)
details: https://anonhg.NetBSD.org/src/rev/bef47fe610b3
branches: trunk
changeset: 996000:bef47fe610b3
user: yamaguchi <yamaguchi%NetBSD.org@localhost>
date: Mon Jan 14 14:55:37 2019 +0000
description:
Add multiqueue support, virtio(4)
diffstat:
sys/dev/pci/virtio.c | 115 +++++++++++++++++++++++++++++++++++----
sys/dev/pci/virtio_pci.c | 137 +++++++++++++++++++++++++++++++++++++++-------
sys/dev/pci/virtiovar.h | 8 ++-
3 files changed, 226 insertions(+), 34 deletions(-)
diffs (truncated from 463 to 300 lines):
diff -r b61dfd01cea9 -r bef47fe610b3 sys/dev/pci/virtio.c
--- a/sys/dev/pci/virtio.c Mon Jan 14 14:52:57 2019 +0000
+++ b/sys/dev/pci/virtio.c Mon Jan 14 14:55:37 2019 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: virtio.c,v 1.36 2018/09/30 15:44:22 jmcneill Exp $ */
+/* $NetBSD: virtio.c,v 1.37 2019/01/14 14:55:37 yamaguchi Exp $ */
/*
* Copyright (c) 2010 Minoura Makoto.
@@ -26,7 +26,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: virtio.c,v 1.36 2018/09/30 15:44:22 jmcneill Exp $");
+__KERNEL_RCSID(0, "$NetBSD: virtio.c,v 1.37 2019/01/14 14:55:37 yamaguchi Exp $");
#include <sys/param.h>
#include <sys/systm.h>
@@ -235,6 +235,54 @@
ops);
}
+static void
+virtio_vq_soft_intr(void *arg)
+{
+ struct virtqueue *vq = arg;
+
+ KASSERT(vq->vq_intrhand != NULL);
+
+ (vq->vq_intrhand)(vq);
+}
+
+static int
+virtio_vq_softint_establish(struct virtio_softc *sc)
+{
+ struct virtqueue *vq;
+ int qid;
+ u_int flags;
+
+ flags = SOFTINT_NET;
+ if (sc->sc_flags & VIRTIO_F_PCI_INTR_MPSAFE)
+ flags |= SOFTINT_MPSAFE;
+
+ for (qid = 0; qid < sc->sc_nvqs; qid++) {
+ vq = &sc->sc_vqs[qid];
+ vq->vq_soft_ih =
+ softint_establish(flags, virtio_vq_soft_intr, vq);
+ if (vq->vq_soft_ih == NULL)
+ return -1;
+ }
+
+ return 0;
+}
+
+static void
+virtio_vq_softint_disestablish(struct virtio_softc *sc)
+{
+ struct virtqueue *vq;
+ int qid;
+
+ for (qid = 0; qid < sc->sc_nvqs; qid++) {
+ vq = &sc->sc_vqs[qid];
+ if (vq->vq_soft_ih == NULL)
+ continue;
+
+ softint_disestablish(vq->vq_soft_ih);
+ vq->vq_soft_ih = NULL;
+ }
+}
+
/*
* Can be used as sc_intrhand.
*/
@@ -242,6 +290,26 @@
* Scan vq, bus_dmamap_sync for the vqs (not for the payload),
* and calls (*vq_done)() if some entries are consumed.
*/
+static int
+virtio_vq_intr_common(struct virtqueue *vq)
+{
+ struct virtio_softc *sc = vq->vq_owner;
+ int r = 0;
+
+ if (vq->vq_queued) {
+ vq->vq_queued = 0;
+ vq_sync_aring(sc, vq, BUS_DMASYNC_POSTWRITE);
+ }
+ vq_sync_uring(sc, vq, BUS_DMASYNC_POSTREAD);
+ membar_consumer();
+ if (vq->vq_used_idx != vq->vq_used->idx) {
+ if (vq->vq_done)
+ r |= (vq->vq_done)(vq);
+ }
+
+ return r;
+}
+
int
virtio_vq_intr(struct virtio_softc *sc)
{
@@ -250,21 +318,19 @@
for (i = 0; i < sc->sc_nvqs; i++) {
vq = &sc->sc_vqs[i];
- if (vq->vq_queued) {
- vq->vq_queued = 0;
- vq_sync_aring(sc, vq, BUS_DMASYNC_POSTWRITE);
- }
- vq_sync_uring(sc, vq, BUS_DMASYNC_POSTREAD);
- membar_consumer();
- if (vq->vq_used_idx != vq->vq_used->idx) {
- if (vq->vq_done)
- r |= (vq->vq_done)(vq);
- }
+ r |= virtio_vq_intr_common(vq);
}
return r;
}
+static int
+virtio_vq_mq_intr(struct virtqueue *vq)
+{
+
+ return virtio_vq_intr_common(vq);
+}
+
/*
* Start/stop vq interrupt. No guarantee.
*/
@@ -409,6 +475,7 @@
/* remember addresses and offsets for later use */
vq->vq_owner = sc;
+ vq->vq_intrhand = virtio_vq_mq_intr;
vq->vq_num = vq_size;
vq->vq_index = index;
vq->vq_desc = vq->vq_vaddr;
@@ -844,6 +911,16 @@
aprint_naive("\n");
}
+void
+virtio_child_attach_set_vqs(struct virtio_softc *sc,
+ struct virtqueue *vqs, int nvq_pairs)
+{
+ if (nvq_pairs > 1)
+ sc->sc_child_mq = true;
+
+ sc->sc_vqs = vqs;
+}
+
int
virtio_child_attach_finish(struct virtio_softc *sc)
{
@@ -868,12 +945,26 @@
"failed to establish soft interrupt\n");
goto fail;
}
+
+ if (sc->sc_child_mq) {
+ r = virtio_vq_softint_establish(sc);
+ aprint_error_dev(sc->sc_dev,
+ "failed to establish softint interrupt\n");
+ goto fail;
+ }
}
virtio_set_status(sc, VIRTIO_CONFIG_DEVICE_STATUS_DRIVER_OK);
return 0;
fail:
+ if (sc->sc_soft_ih) {
+ softint_disestablish(sc->sc_soft_ih);
+ sc->sc_soft_ih = NULL;
+ }
+
+ virtio_vq_softint_disestablish(sc);
+
virtio_set_status(sc, VIRTIO_CONFIG_DEVICE_STATUS_FAILED);
return 1;
}
diff -r b61dfd01cea9 -r bef47fe610b3 sys/dev/pci/virtio_pci.c
--- a/sys/dev/pci/virtio_pci.c Mon Jan 14 14:52:57 2019 +0000
+++ b/sys/dev/pci/virtio_pci.c Mon Jan 14 14:55:37 2019 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: virtio_pci.c,v 1.5 2018/06/06 16:11:36 jakllsch Exp $ */
+/* $NetBSD: virtio_pci.c,v 1.6 2019/01/14 14:55:37 yamaguchi Exp $ */
/*
* Copyright (c) 2010 Minoura Makoto.
@@ -26,12 +26,13 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: virtio_pci.c,v 1.5 2018/06/06 16:11:36 jakllsch Exp $");
+__KERNEL_RCSID(0, "$NetBSD: virtio_pci.c,v 1.6 2019/01/14 14:55:37 yamaguchi Exp $");
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/kmem.h>
#include <sys/module.h>
+#include <sys/interrupt.h>
#include <sys/device.h>
@@ -79,6 +80,7 @@
static int virtio_pci_intr(void *arg);
static int virtio_pci_msix_queue_intr(void *);
+static int virtio_pci_msix_vq_intr(void *);
static int virtio_pci_msix_config_intr(void *);
static int virtio_pci_setup_msix_vectors(struct virtio_softc *);
static int virtio_pci_setup_msix_interrupts(struct virtio_softc *,
@@ -428,7 +430,7 @@
if (psc->sc_ihs_num > 1) {
int vec = VIRTIO_MSIX_QUEUE_VECTOR_INDEX;
- if (false) /* (for per-vq vectors) */
+ if (sc->sc_child_mq)
vec += idx;
bus_space_write_stream_2(psc->sc_iot, psc->sc_ioh,
VIRTIO_CONFIG_MSI_QUEUE_VECTOR, vec);
@@ -488,6 +490,9 @@
offset = VIRTIO_CONFIG_MSI_QUEUE_VECTOR;
vector = VIRTIO_MSIX_QUEUE_VECTOR_INDEX;
+ if (sc->sc_child_mq)
+ vector += qid;
+
bus_space_write_stream_2(psc->sc_iot, psc->sc_ioh, offset, vector);
ret = bus_space_read_stream_2(psc->sc_iot, psc->sc_ioh, offset);
aprint_debug_dev(sc->sc_dev, "expected=%d, actual=%d\n",
@@ -507,29 +512,57 @@
device_t self = sc->sc_dev;
pci_chipset_tag_t pc = pa->pa_pc;
char intrbuf[PCI_INTRSTR_LEN];
+ char intr_xname[INTRDEVNAMEBUF];
char const *intrstr;
- int idx;
+ int idx, qid, n;
idx = VIRTIO_MSIX_CONFIG_VECTOR_INDEX;
if (sc->sc_flags & VIRTIO_F_PCI_INTR_MPSAFE)
pci_intr_setattr(pc, &psc->sc_ihp[idx], PCI_INTR_MPSAFE, true);
+ snprintf(intr_xname, sizeof(intr_xname), "%s config",
+ device_xname(sc->sc_dev));
+
psc->sc_ihs[idx] = pci_intr_establish_xname(pc, psc->sc_ihp[idx],
- sc->sc_ipl, virtio_pci_msix_config_intr, sc, device_xname(sc->sc_dev));
+ sc->sc_ipl, virtio_pci_msix_config_intr, sc, intr_xname);
if (psc->sc_ihs[idx] == NULL) {
aprint_error_dev(self, "couldn't establish MSI-X for config\n");
goto error;
}
idx = VIRTIO_MSIX_QUEUE_VECTOR_INDEX;
- if (sc->sc_flags & VIRTIO_F_PCI_INTR_MPSAFE)
- pci_intr_setattr(pc, &psc->sc_ihp[idx], PCI_INTR_MPSAFE, true);
+ if (sc->sc_child_mq) {
+ for (qid = 0; qid < sc->sc_nvqs; qid++) {
+ n = idx + qid;
+
+ snprintf(intr_xname, sizeof(intr_xname), "%s vq#%d",
+ device_xname(sc->sc_dev), qid);
+
+ if (sc->sc_flags & VIRTIO_F_PCI_INTR_MPSAFE) {
+ pci_intr_setattr(pc, &psc->sc_ihp[n],
+ PCI_INTR_MPSAFE, true);
+ }
- psc->sc_ihs[idx] = pci_intr_establish_xname(pc, psc->sc_ihp[idx],
- sc->sc_ipl, virtio_pci_msix_queue_intr, sc, device_xname(sc->sc_dev));
- if (psc->sc_ihs[idx] == NULL) {
- aprint_error_dev(self, "couldn't establish MSI-X for queues\n");
- goto error;
+ psc->sc_ihs[n] = pci_intr_establish_xname(pc, psc->sc_ihp[n],
+ sc->sc_ipl, virtio_pci_msix_vq_intr, &sc->sc_vqs[qid],
+ intr_xname);
+ if (psc->sc_ihs[n] == NULL) {
+ aprint_error_dev(self, "couldn't establish MSI-X for a vq\n");
+ goto error;
+ }
+ }
+ } else {
+ if (sc->sc_flags & VIRTIO_F_PCI_INTR_MPSAFE)
+ pci_intr_setattr(pc, &psc->sc_ihp[idx], PCI_INTR_MPSAFE, true);
+
+ snprintf(intr_xname, sizeof(intr_xname), "%s queues",
+ device_xname(sc->sc_dev));
+ psc->sc_ihs[idx] = pci_intr_establish_xname(pc, psc->sc_ihp[idx],
+ sc->sc_ipl, virtio_pci_msix_queue_intr, sc, intr_xname);
+ if (psc->sc_ihs[idx] == NULL) {
+ aprint_error_dev(self, "couldn't establish MSI-X for queues\n");
+ goto error;
+ }
}
if (virtio_pci_setup_msix_vectors(sc) != 0) {
Home |
Main Index |
Thread Index |
Old Index