Source-Changes-HG archive

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

[src/trunk]: src/sys/dev/hyperv vmbus(4): Do not call hyperv_dma_alloc() in i...



details:   https://anonhg.NetBSD.org/src/rev/cb20ae0b0d38
branches:  trunk
changeset: 1010480:cb20ae0b0d38
user:      nonaka <nonaka%NetBSD.org@localhost>
date:      Tue May 26 16:08:55 2020 +0000

description:
vmbus(4): Do not call hyperv_dma_alloc() in interrupt context.

The channel offer and rescind process is performed on another context.

diffstat:

 sys/dev/hyperv/vmbus.c    |  328 +++++++++++++++++++++++++++++++++++++--------
 sys/dev/hyperv/vmbusvar.h |   23 +++-
 2 files changed, 289 insertions(+), 62 deletions(-)

diffs (truncated from 569 to 300 lines):

diff -r ea7c574f221a -r cb20ae0b0d38 sys/dev/hyperv/vmbus.c
--- a/sys/dev/hyperv/vmbus.c    Tue May 26 16:00:06 2020 +0000
+++ b/sys/dev/hyperv/vmbus.c    Tue May 26 16:08:55 2020 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: vmbus.c,v 1.10 2020/05/26 16:00:06 nonaka Exp $        */
+/*     $NetBSD: vmbus.c,v 1.11 2020/05/26 16:08:55 nonaka Exp $        */
 /*     $OpenBSD: hyperv.c,v 1.43 2017/06/27 13:56:15 mikeb Exp $       */
 
 /*-
@@ -35,7 +35,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: vmbus.c,v 1.10 2020/05/26 16:00:06 nonaka Exp $");
+__KERNEL_RCSID(0, "$NetBSD: vmbus.c,v 1.11 2020/05/26 16:08:55 nonaka Exp $");
 
 #include <sys/param.h>
 #include <sys/systm.h>
@@ -103,10 +103,14 @@
 static void    vmbus_channel_pause(struct vmbus_channel *);
 static uint32_t        vmbus_channel_unpause(struct vmbus_channel *);
 static uint32_t        vmbus_channel_ready(struct vmbus_channel *);
+static void    vmbus_chevq_enqueue(struct vmbus_softc *, int, void *);
+static void    vmbus_process_chevq(void *);
+static void    vmbus_chevq_thread(void *);
 static void    vmbus_devq_enqueue(struct vmbus_softc *, int,
                    struct vmbus_channel *);
 static void    vmbus_process_devq(void *);
 static void    vmbus_devq_thread(void *);
+static void    vmbus_subchannel_devq_thread(void *);
 
 static struct vmbus_softc *vmbus_sc;
 
@@ -841,15 +845,33 @@
 static void
 vmbus_channel_offer(struct vmbus_softc *sc, struct vmbus_chanmsg_hdr *hdr)
 {
+       struct vmbus_chanmsg_choffer *co;
 
-       vmbus_process_offer(sc, (struct vmbus_chanmsg_choffer *)hdr);
+       co = kmem_intr_alloc(sizeof(*co), KM_NOSLEEP);
+       if (co == NULL) {
+               device_printf(sc->sc_dev,
+                   "failed to allocate an offer object\n");
+               return;
+       }
+
+       memcpy(co, hdr, sizeof(*co));
+       vmbus_chevq_enqueue(sc, VMBUS_CHEV_TYPE_OFFER, co);
 }
 
 static void
 vmbus_channel_rescind(struct vmbus_softc *sc, struct vmbus_chanmsg_hdr *hdr)
 {
+       struct vmbus_chanmsg_chrescind *cr;
+ 
+       cr = kmem_intr_alloc(sizeof(*cr), KM_NOSLEEP);
+       if (cr == NULL) {
+               device_printf(sc->sc_dev,
+                   "failed to allocate an rescind object\n");
+               return;
+       }
 
-       vmbus_process_rescind(sc, (struct vmbus_chanmsg_chrescind *)hdr);
+       memcpy(cr, hdr, sizeof(*cr));
+       vmbus_chevq_enqueue(sc, VMBUS_CHEV_TYPE_RESCIND, cr);
 }
 
 static void
@@ -906,16 +928,48 @@
        struct vmbus_chanmsg_hdr hdr;
        struct vmbus_chanmsg_choffer rsp;
 
+       TAILQ_INIT(&sc->sc_prichans);
+       mutex_init(&sc->sc_prichan_lock, MUTEX_DEFAULT, IPL_NET);
        TAILQ_INIT(&sc->sc_channels);
        mutex_init(&sc->sc_channel_lock, MUTEX_DEFAULT, IPL_NET);
 
+       /*
+        * This queue serializes vmbus channel offer and rescind messages.
+        */
+       SIMPLEQ_INIT(&sc->sc_chevq);
+       mutex_init(&sc->sc_chevq_lock, MUTEX_DEFAULT, IPL_NET);
+       cv_init(&sc->sc_chevq_cv, "hvchevcv");
+       if (kthread_create(PRI_NONE, KTHREAD_MPSAFE, NULL,
+           vmbus_chevq_thread, sc, NULL, "hvchevq") != 0) {
+               DPRINTF("%s: failed to create prich chevq thread\n",
+                   device_xname(sc->sc_dev));
+               return -1;
+       }
+
+       /*
+        * This queue serializes vmbus devices' attach and detach
+        * for channel offer and rescind messages.
+        */
        SIMPLEQ_INIT(&sc->sc_devq);
        mutex_init(&sc->sc_devq_lock, MUTEX_DEFAULT, IPL_NET);
        cv_init(&sc->sc_devq_cv, "hvdevqcv");
-
        if (kthread_create(PRI_NONE, KTHREAD_MPSAFE, NULL,
-           vmbus_devq_thread, sc, NULL, "hvoffer") != 0) {
-               DPRINTF("%s: failed to create offer thread\n",
+           vmbus_devq_thread, sc, NULL, "hvdevq") != 0) {
+               DPRINTF("%s: failed to create prich devq thread\n",
+                   device_xname(sc->sc_dev));
+               return -1;
+       }
+
+       /*
+        * This queue handles sub-channel detach, so that vmbus
+        * device's detach running in sc_devq can drain its sub-channels.
+        */
+       SIMPLEQ_INIT(&sc->sc_subch_devq);
+       mutex_init(&sc->sc_subch_devq_lock, MUTEX_DEFAULT, IPL_NET);
+       cv_init(&sc->sc_subch_devq_cv, "hvsdvqcv");
+       if (kthread_create(PRI_NONE, KTHREAD_MPSAFE, NULL,
+           vmbus_subchannel_devq_thread, sc, NULL, "hvsdevq") != 0) {
+               DPRINTF("%s: failed to create subch devq thread\n",
                    device_xname(sc->sc_dev));
                return -1;
        }
@@ -932,6 +986,9 @@
        while (!ISSET(sc->sc_flags, VMBUS_SCFLAG_OFFERS_DELIVERED))
                tsleep(&sc->sc_devq, PRIBIO, "hvscan", 1);
 
+       mutex_enter(&sc->sc_chevq_lock);
+       vmbus_process_chevq(sc);
+       mutex_exit(&sc->sc_chevq_lock);
        mutex_enter(&sc->sc_devq_lock);
        vmbus_process_devq(sc);
        mutex_exit(&sc->sc_devq_lock);
@@ -944,10 +1001,10 @@
 {
        struct vmbus_channel *ch;
 
-       ch = kmem_intr_zalloc(sizeof(*ch), KM_NOSLEEP);
+       ch = kmem_zalloc(sizeof(*ch), KM_SLEEP);
 
        ch->ch_monprm = hyperv_dma_alloc(sc->sc_dmat, &ch->ch_monprm_dma,
-           sizeof(*ch->ch_monprm), 8, 0, 1, HYPERV_DMA_NOSLEEP);
+           sizeof(*ch->ch_monprm), 8, 0, 1, HYPERV_DMA_SLEEPOK);
        if (ch->ch_monprm == NULL) {
                device_printf(sc->sc_dev, "monprm alloc failed\n");
                kmem_free(ch, sizeof(*ch));
@@ -1000,32 +1057,32 @@
                return EINVAL;
        }
 
-       mutex_enter(&sc->sc_channel_lock);
-       TAILQ_FOREACH(ch, &sc->sc_channels, ch_entry) {
+       mutex_enter(&sc->sc_prichan_lock);
+       TAILQ_FOREACH(ch, &sc->sc_prichans, ch_prientry) {
                if (!memcmp(&ch->ch_type, &nch->ch_type, sizeof(ch->ch_type)) &&
                    !memcmp(&ch->ch_inst, &nch->ch_inst, sizeof(ch->ch_inst)))
                        break;
        }
        if (VMBUS_CHAN_ISPRIMARY(nch)) {
                if (ch == NULL) {
-                       TAILQ_INSERT_TAIL(&sc->sc_channels, nch, ch_entry);
-                       mutex_exit(&sc->sc_channel_lock);
+                       TAILQ_INSERT_TAIL(&sc->sc_prichans, nch, ch_prientry);
+                       mutex_exit(&sc->sc_prichan_lock);
                        goto done;
                } else {
-                       mutex_exit(&sc->sc_channel_lock);
+                       mutex_exit(&sc->sc_prichan_lock);
                        device_printf(sc->sc_dev,
                            "duplicated primary channel%u\n", nch->ch_id);
                        return EINVAL;
                }
        } else {
                if (ch == NULL) {
-                       mutex_exit(&sc->sc_channel_lock);
+                       mutex_exit(&sc->sc_prichan_lock);
                        device_printf(sc->sc_dev, "no primary channel%u\n",
                            nch->ch_id);
                        return EINVAL;
                }
        }
-       mutex_exit(&sc->sc_channel_lock);
+       mutex_exit(&sc->sc_prichan_lock);
 
        KASSERT(!VMBUS_CHAN_ISPRIMARY(nch));
        KASSERT(ch != NULL);
@@ -1043,6 +1100,10 @@
        wakeup(ch);
 
 done:
+       mutex_enter(&sc->sc_channel_lock);
+       TAILQ_INSERT_TAIL(&sc->sc_channels, nch, ch_entry);
+       mutex_exit(&sc->sc_channel_lock);
+
        vmbus_channel_cpu_default(nch);
 
        return 0;
@@ -1094,7 +1155,6 @@
        return (ch->ch_flags & CHF_REVOKED) ? true : false;
 }
 
-
 static void
 vmbus_process_offer(struct vmbus_softc *sc, struct vmbus_chanmsg_choffer *co)
 {
@@ -1176,6 +1236,12 @@
        TAILQ_REMOVE(&sc->sc_channels, ch, ch_entry);
        mutex_exit(&sc->sc_channel_lock);
 
+       if (VMBUS_CHAN_ISPRIMARY(ch)) {
+               mutex_enter(&sc->sc_prichan_lock);
+               TAILQ_REMOVE(&sc->sc_prichans, ch, ch_prientry);
+               mutex_exit(&sc->sc_prichan_lock);
+       }
+
        KASSERTMSG(!(ch->ch_flags & CHF_REVOKED),
            "channel%u has already been revoked", ch->ch_id);
        atomic_or_uint(&ch->ch_flags, CHF_REVOKED);
@@ -1206,20 +1272,32 @@
 struct vmbus_channel **
 vmbus_subchannel_get(struct vmbus_channel *prich, int cnt)
 {
+       struct vmbus_softc *sc = prich->ch_sc;
        struct vmbus_channel **ret, *ch;
-       int i;
+       int i, s;
 
-       KASSERT(cnt > 0);
+       KASSERTMSG(cnt > 0, "invalid sub-channel count %d", cnt);
 
-       ret = kmem_alloc(sizeof(struct vmbus_channel *) * cnt,
-           cold ? KM_NOSLEEP : KM_SLEEP);
+       ret = kmem_zalloc(sizeof(struct vmbus_channel *) * cnt, KM_SLEEP);
 
        mutex_enter(&prich->ch_subchannel_lock);
 
-       while (prich->ch_subchannel_count < cnt)
-               /* XXX use condvar(9) instead of mtsleep */
-               mtsleep(prich, PRIBIO, "hvvmsubch", 0,
-                   &prich->ch_subchannel_lock);
+       while (prich->ch_subchannel_count < cnt) {
+               if (cold) {
+                       mutex_exit(&prich->ch_subchannel_lock);
+                       delay(1000);
+                       s = splnet();
+                       hyperv_intr();
+                       splx(s);
+                       mutex_enter(&sc->sc_chevq_lock);
+                       vmbus_process_chevq(sc);
+                       mutex_exit(&sc->sc_chevq_lock);
+                       mutex_enter(&prich->ch_subchannel_lock);
+               } else {
+                       mtsleep(prich, PRIBIO, "hvsubch", 1,
+                           &prich->ch_subchannel_lock);
+               }
+       }
 
        i = 0;
        TAILQ_FOREACH(ch, &prich->ch_subchannels, ch_subentry) {
@@ -1229,6 +1307,9 @@
                        break;
        }
 
+       KASSERTMSG(i == cnt, "invalid subchan count %d, should be %d",
+           prich->ch_subchannel_count, cnt);
+
        mutex_exit(&prich->ch_subchannel_lock);
 
        return ret;
@@ -1991,12 +2072,90 @@
        }
 }
 
+ static void
+vmbus_chevq_enqueue(struct vmbus_softc *sc, int type, void *arg)
+{
+       struct vmbus_chev *vce;
+ 
+       vce = kmem_intr_alloc(sizeof(*vce), KM_NOSLEEP);
+       if (vce == NULL) {
+               device_printf(sc->sc_dev, "failed to allocate chev\n");
+               return;
+       }
+
+       vce->vce_type = type;
+       vce->vce_arg = arg;
+
+       mutex_enter(&sc->sc_chevq_lock);
+       SIMPLEQ_INSERT_TAIL(&sc->sc_chevq, vce, vce_entry);
+       cv_broadcast(&sc->sc_chevq_cv);
+       mutex_exit(&sc->sc_chevq_lock);
+}
+
+static void
+vmbus_process_chevq(void *arg)
+{
+       struct vmbus_softc *sc = arg;
+       struct vmbus_chev *vce;
+       struct vmbus_chanmsg_choffer *co;
+       struct vmbus_chanmsg_chrescind *cr;
+
+       KASSERT(mutex_owned(&sc->sc_chevq_lock));



Home | Main Index | Thread Index | Old Index