Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/sys/dev/pci Functionize the same code related to ctrl vq in ...



details:   https://anonhg.NetBSD.org/src/rev/b61dfd01cea9
branches:  trunk
changeset: 995999:b61dfd01cea9
user:      yamaguchi <yamaguchi%NetBSD.org@localhost>
date:      Mon Jan 14 14:52:57 2019 +0000

description:
Functionize the same code related to ctrl vq in vioif(4)

diffstat:

 sys/dev/pci/if_vioif.c |  256 ++++++++++++++++++++++++++----------------------
 1 files changed, 140 insertions(+), 116 deletions(-)

diffs (truncated from 348 to 300 lines):

diff -r 850d9c7241e0 -r b61dfd01cea9 sys/dev/pci/if_vioif.c
--- a/sys/dev/pci/if_vioif.c    Mon Jan 14 14:35:52 2019 +0000
+++ b/sys/dev/pci/if_vioif.c    Mon Jan 14 14:52:57 2019 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: if_vioif.c,v 1.43 2019/01/14 14:35:52 yamaguchi Exp $  */
+/*     $NetBSD: if_vioif.c,v 1.44 2019/01/14 14:52:57 yamaguchi Exp $  */
 
 /*
  * Copyright (c) 2010 Minoura Makoto.
@@ -26,7 +26,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: if_vioif.c,v 1.43 2019/01/14 14:35:52 yamaguchi Exp $");
+__KERNEL_RCSID(0, "$NetBSD: if_vioif.c,v 1.44 2019/01/14 14:52:57 yamaguchi Exp $");
 
 #ifdef _KERNEL_OPT
 #include "opt_net_mpsafe.h"
@@ -171,6 +171,11 @@
        uint16_t id;
 } __packed;
 
+struct vioif_ctrl_cmdspec {
+       bus_dmamap_t    dmamap;
+       void            *buf;
+       bus_size_t      bufsize;
+};
 
 /*
  * if_vioifvar.h:
@@ -222,6 +227,7 @@
        }                               ctrlq_inuse;
        kcondvar_t                      ctrlq_wait;
        kmutex_t                        ctrlq_wait_lock;
+       struct lwp                      *ctrlq_owner;
 
        struct virtio_net_ctrl_cmd      *ctrlq_cmd;
        struct virtio_net_ctrl_status   *ctrlq_status;
@@ -1279,33 +1285,96 @@
  * Control vq
  */
 /* issue a VIRTIO_NET_CTRL_RX class command and wait for completion */
-static int
-vioif_ctrl_rx(struct vioif_softc *sc, int cmd, bool onoff)
+static void
+vioif_ctrl_acquire(struct vioif_softc *sc)
 {
-       struct virtio_softc *vsc = sc->sc_virtio;
        struct vioif_ctrlqueue *ctrlq = &sc->sc_ctrlq;
-       struct virtqueue *vq = ctrlq->ctrlq_vq;
-       int r, slot;
-
-       if (!sc->sc_has_ctrl)
-               return ENOTSUP;
 
        mutex_enter(&ctrlq->ctrlq_wait_lock);
        while (ctrlq->ctrlq_inuse != FREE)
                cv_wait(&ctrlq->ctrlq_wait, &ctrlq->ctrlq_wait_lock);
        ctrlq->ctrlq_inuse = INUSE;
+       ctrlq->ctrlq_owner = curlwp;
        mutex_exit(&ctrlq->ctrlq_wait_lock);
+}
+
+static void
+vioif_ctrl_release(struct vioif_softc *sc)
+{
+       struct vioif_ctrlqueue *ctrlq = &sc->sc_ctrlq;
+
+       KASSERT(ctrlq->ctrlq_inuse != FREE);
+       KASSERT(ctrlq->ctrlq_owner == curlwp);
+
+       mutex_enter(&ctrlq->ctrlq_wait_lock);
+       ctrlq->ctrlq_inuse = FREE;
+       cv_signal(&ctrlq->ctrlq_wait);
+       mutex_exit(&ctrlq->ctrlq_wait_lock);
+}
+
+static int
+vioif_ctrl_load_cmdspec(struct vioif_softc *sc,
+    struct vioif_ctrl_cmdspec *specs, int nspecs)
+{
+       struct virtio_softc *vsc = sc->sc_virtio;
+       int i, r, loaded;
 
-       ctrlq->ctrlq_cmd->class = VIRTIO_NET_CTRL_RX;
+       loaded = 0;
+       for (i = 0; i < nspecs; i++) {
+               r = bus_dmamap_load(virtio_dmat(vsc),
+                   specs[i].dmamap, specs[i].buf, specs[i].bufsize,
+                   NULL, BUS_DMA_WRITE|BUS_DMA_NOWAIT);
+               if (r) {
+                       printf("%s: control command dmamap load failed, "
+                              "error code %d\n", device_xname(sc->sc_dev), r);
+                       goto err;
+               }
+               loaded++;
+
+       }
+
+       return r;
+
+err:
+       for (i = 0; i < loaded; i++) {
+               bus_dmamap_unload(virtio_dmat(vsc), specs[i].dmamap);
+       }
+
+       return r;
+}
+
+static void
+vioif_ctrl_unload_cmdspec(struct vioif_softc *sc,
+    struct vioif_ctrl_cmdspec *specs, int nspecs)
+{
+       struct virtio_softc *vsc = sc->sc_virtio;
+       int i;
+
+       for (i = 0; i < nspecs; i++) {
+               bus_dmamap_unload(virtio_dmat(vsc), specs[i].dmamap);
+       }
+}
+
+static int
+vioif_ctrl_send_command(struct vioif_softc *sc, uint8_t class, uint8_t cmd,
+    struct vioif_ctrl_cmdspec *specs, int nspecs)
+{
+       struct vioif_ctrlqueue *ctrlq = &sc->sc_ctrlq;
+       struct virtqueue *vq = ctrlq->ctrlq_vq;
+       struct virtio_softc *vsc = sc->sc_virtio;
+       int i, r, slot;
+
+       ctrlq->ctrlq_cmd->class = class;
        ctrlq->ctrlq_cmd->command = cmd;
-       ctrlq->ctrlq_rx->onoff = onoff;
 
        bus_dmamap_sync(virtio_dmat(vsc), ctrlq->ctrlq_cmd_dmamap,
                        0, sizeof(struct virtio_net_ctrl_cmd),
                        BUS_DMASYNC_PREWRITE);
-       bus_dmamap_sync(virtio_dmat(vsc), ctrlq->ctrlq_rx_dmamap,
-                       0, sizeof(struct virtio_net_ctrl_rx),
-                       BUS_DMASYNC_PREWRITE);
+       for (i = 0; i < nspecs; i++) {
+               bus_dmamap_sync(virtio_dmat(vsc), specs[i].dmamap,
+                               0, specs[i].bufsize,
+                               BUS_DMASYNC_PREWRITE);
+       }
        bus_dmamap_sync(virtio_dmat(vsc), ctrlq->ctrlq_status_dmamap,
                        0, sizeof(struct virtio_net_ctrl_status),
                        BUS_DMASYNC_PREREAD);
@@ -1313,11 +1382,13 @@
        r = virtio_enqueue_prep(vsc, vq, &slot);
        if (r != 0)
                panic("%s: control vq busy!?", device_xname(sc->sc_dev));
-       r = virtio_enqueue_reserve(vsc, vq, slot, 3);
+       r = virtio_enqueue_reserve(vsc, vq, slot, nspecs + 2);
        if (r != 0)
                panic("%s: control vq busy!?", device_xname(sc->sc_dev));
        virtio_enqueue(vsc, vq, slot, ctrlq->ctrlq_cmd_dmamap, true);
-       virtio_enqueue(vsc, vq, slot, ctrlq->ctrlq_rx_dmamap, true);
+       for (i = 0; i < nspecs; i++) {
+               virtio_enqueue(vsc, vq, slot, specs[i].dmamap, true);
+       }
        virtio_enqueue(vsc, vq, slot, ctrlq->ctrlq_status_dmamap, false);
        virtio_enqueue_commit(vsc, vq, slot, true);
 
@@ -1331,9 +1402,11 @@
        bus_dmamap_sync(virtio_dmat(vsc), ctrlq->ctrlq_cmd_dmamap, 0,
                        sizeof(struct virtio_net_ctrl_cmd),
                        BUS_DMASYNC_POSTWRITE);
-       bus_dmamap_sync(virtio_dmat(vsc), ctrlq->ctrlq_rx_dmamap, 0,
-                       sizeof(struct virtio_net_ctrl_rx),
-                       BUS_DMASYNC_POSTWRITE);
+       for (i = 0; i < nspecs; i++) {
+               bus_dmamap_sync(virtio_dmat(vsc), specs[i].dmamap, 0,
+                               specs[i].bufsize,
+                               BUS_DMASYNC_POSTWRITE);
+       }
        bus_dmamap_sync(virtio_dmat(vsc), ctrlq->ctrlq_status_dmamap, 0,
                        sizeof(struct virtio_net_ctrl_status),
                        BUS_DMASYNC_POSTREAD);
@@ -1346,11 +1419,30 @@
                r = EIO;
        }
 
-       mutex_enter(&ctrlq->ctrlq_wait_lock);
-       ctrlq->ctrlq_inuse = FREE;
-       cv_signal(&ctrlq->ctrlq_wait);
-       mutex_exit(&ctrlq->ctrlq_wait_lock);
+       return r;
+}
+
+static int
+vioif_ctrl_rx(struct vioif_softc *sc, int cmd, bool onoff)
+{
+       struct virtio_net_ctrl_rx *rx = sc->sc_ctrlq.ctrlq_rx;
+       struct vioif_ctrl_cmdspec specs[1];
+       int r;
 
+       if (!sc->sc_has_ctrl)
+               return ENOTSUP;
+
+       vioif_ctrl_acquire(sc);
+
+       rx->onoff = onoff;
+       specs[0].dmamap = sc->sc_ctrlq.ctrlq_rx_dmamap;
+       specs[0].buf = rx;
+       specs[0].bufsize = sizeof(*rx);
+
+       r = vioif_ctrl_send_command(sc, VIRTIO_NET_CTRL_RX, cmd,
+           specs, __arraycount(specs));
+
+       vioif_ctrl_release(sc);
        return r;
 }
 
@@ -1379,109 +1471,41 @@
 vioif_set_rx_filter(struct vioif_softc *sc)
 {
        /* filter already set in ctrlq->ctrlq_mac_tbl */
-       struct virtio_softc *vsc = sc->sc_virtio;
-       struct vioif_ctrlqueue *ctrlq = &sc->sc_ctrlq;
-       struct virtqueue *vq = ctrlq->ctrlq_vq;
-       int r, slot;
+       struct virtio_net_ctrl_mac_tbl *mac_tbl_uc, *mac_tbl_mc;
+       struct vioif_ctrl_cmdspec specs[2];
+       int nspecs = __arraycount(specs);
+       int r;
+
+       mac_tbl_uc = sc->sc_ctrlq.ctrlq_mac_tbl_uc;
+       mac_tbl_mc = sc->sc_ctrlq.ctrlq_mac_tbl_mc;
 
        if (!sc->sc_has_ctrl)
                return ENOTSUP;
 
-       mutex_enter(&ctrlq->ctrlq_wait_lock);
-       while (ctrlq->ctrlq_inuse != FREE)
-               cv_wait(&ctrlq->ctrlq_wait, &ctrlq->ctrlq_wait_lock);
-       ctrlq->ctrlq_inuse = INUSE;
-       mutex_exit(&ctrlq->ctrlq_wait_lock);
-
-       ctrlq->ctrlq_cmd->class = VIRTIO_NET_CTRL_MAC;
-       ctrlq->ctrlq_cmd->command = VIRTIO_NET_CTRL_MAC_TABLE_SET;
+       vioif_ctrl_acquire(sc);
 
-       r = bus_dmamap_load(virtio_dmat(vsc), ctrlq->ctrlq_tbl_uc_dmamap,
-                           ctrlq->ctrlq_mac_tbl_uc,
-                           (sizeof(struct virtio_net_ctrl_mac_tbl)
-                         + ETHER_ADDR_LEN * ctrlq->ctrlq_mac_tbl_uc->nentries),
-                           NULL, BUS_DMA_WRITE|BUS_DMA_NOWAIT);
-       if (r) {
-               printf("%s: control command dmamap load failed, "
-                      "error code %d\n", device_xname(sc->sc_dev), r);
-               goto out;
-       }
-       r = bus_dmamap_load(virtio_dmat(vsc), ctrlq->ctrlq_tbl_mc_dmamap,
-                           ctrlq->ctrlq_mac_tbl_mc,
-                           (sizeof(struct virtio_net_ctrl_mac_tbl)
-                         + ETHER_ADDR_LEN * ctrlq->ctrlq_mac_tbl_mc->nentries),
-                           NULL, BUS_DMA_WRITE|BUS_DMA_NOWAIT);
-       if (r) {
-               printf("%s: control command dmamap load failed, "
-                      "error code %d\n", device_xname(sc->sc_dev), r);
-               bus_dmamap_unload(virtio_dmat(vsc), ctrlq->ctrlq_tbl_uc_dmamap);
-               goto out;
-       }
+       specs[0].dmamap = sc->sc_ctrlq.ctrlq_tbl_uc_dmamap;
+       specs[0].buf = mac_tbl_uc;
+       specs[0].bufsize = sizeof(*mac_tbl_uc)
+           + (ETHER_ADDR_LEN * mac_tbl_uc->nentries);
 
-       bus_dmamap_sync(virtio_dmat(vsc), ctrlq->ctrlq_cmd_dmamap,
-                       0, sizeof(struct virtio_net_ctrl_cmd),
-                       BUS_DMASYNC_PREWRITE);
-       bus_dmamap_sync(virtio_dmat(vsc), ctrlq->ctrlq_tbl_uc_dmamap, 0,
-                       (sizeof(struct virtio_net_ctrl_mac_tbl)
-                        + ETHER_ADDR_LEN * ctrlq->ctrlq_mac_tbl_uc->nentries),
-                       BUS_DMASYNC_PREWRITE);
-       bus_dmamap_sync(virtio_dmat(vsc), ctrlq->ctrlq_tbl_mc_dmamap, 0,
-                       (sizeof(struct virtio_net_ctrl_mac_tbl)
-                        + ETHER_ADDR_LEN * ctrlq->ctrlq_mac_tbl_mc->nentries),
-                       BUS_DMASYNC_PREWRITE);
-       bus_dmamap_sync(virtio_dmat(vsc), ctrlq->ctrlq_status_dmamap,
-                       0, sizeof(struct virtio_net_ctrl_status),
-                       BUS_DMASYNC_PREREAD);
-
-       r = virtio_enqueue_prep(vsc, vq, &slot);
-       if (r != 0)
-               panic("%s: control vq busy!?", device_xname(sc->sc_dev));
-       r = virtio_enqueue_reserve(vsc, vq, slot, 4);
-       if (r != 0)
-               panic("%s: control vq busy!?", device_xname(sc->sc_dev));
-       virtio_enqueue(vsc, vq, slot, ctrlq->ctrlq_cmd_dmamap, true);
-       virtio_enqueue(vsc, vq, slot, ctrlq->ctrlq_tbl_uc_dmamap, true);
-       virtio_enqueue(vsc, vq, slot, ctrlq->ctrlq_tbl_mc_dmamap, true);
-       virtio_enqueue(vsc, vq, slot, ctrlq->ctrlq_status_dmamap, false);
-       virtio_enqueue_commit(vsc, vq, slot, true);
+       specs[1].dmamap = sc->sc_ctrlq.ctrlq_tbl_mc_dmamap;
+       specs[1].buf = mac_tbl_mc;
+       specs[1].bufsize = sizeof(*mac_tbl_mc)
+           + (ETHER_ADDR_LEN * mac_tbl_mc->nentries);
 



Home | Main Index | Thread Index | Old Index