From f1debecd6554f1b1a6eeb01c1369ded1a6214b0a Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Wed, 23 Jun 2021 23:46:19 +0000 Subject: [PATCH] ehci: Replace bogus polling during suspend by taking bus lock. The system is not at high IPL or limited to a single CPU at this point; it is running essentially normally, just with some devices suspended. Other threads might be trying to touch EHCI registers, which might have bad outcomes while we're suspending stuff. Enabling polling only makes sense if there is a single thread and single CPU running, such as when we enter ddb on crash. --- sys/dev/usb/ehci.c | 16 ++++++---------- 1 file changed, 6 insertions(+), 10 deletions(-) diff --git a/sys/dev/usb/ehci.c b/sys/dev/usb/ehci.c index 13a1cea3ccbb..e7a2f44f043a 100644 --- a/sys/dev/usb/ehci.c +++ b/sys/dev/usb/ehci.c @@ -1384,10 +1384,6 @@ ehci_activate(device_t self, enum devact act) /* * Handle suspend/resume. * - * We need to switch to polling mode here, because this routine is - * called from an interrupt context. This is all right since we - * are almost suspended anyway. - * * Note that this power handler isn't to be registered directly; the * bus glue needs to call out to it. */ @@ -1400,9 +1396,7 @@ ehci_suspend(device_t dv, const pmf_qual_t *qual) EHCIHIST_FUNC(); EHCIHIST_CALLED(); - mutex_spin_enter(&sc->sc_intr_lock); - sc->sc_bus.ub_usepolling++; - mutex_spin_exit(&sc->sc_intr_lock); + mutex_enter(&sc->sc_lock); for (i = 1; i <= sc->sc_noport; i++) { cmd = EOREAD4(sc, EHCI_PORTSC(i)) & ~EHCI_PS_CLEAR; @@ -1438,9 +1432,7 @@ ehci_suspend(device_t dv, const pmf_qual_t *qual) if (hcr != EHCI_STS_HCH) printf("%s: config timeout\n", device_xname(dv)); - mutex_spin_enter(&sc->sc_intr_lock); - sc->sc_bus.ub_usepolling--; - mutex_spin_exit(&sc->sc_intr_lock); + mutex_exit(&sc->sc_lock); return true; } @@ -1454,6 +1446,8 @@ ehci_resume(device_t dv, const pmf_qual_t *qual) EHCIHIST_FUNC(); EHCIHIST_CALLED(); + mutex_enter(&sc->sc_lock); + /* restore things in case the bios sucks */ EOWRITE4(sc, EHCI_CTRLDSSEGMENT, 0); EOWRITE4(sc, EHCI_PERIODICLISTBASE, DMAADDR(&sc->sc_fldma, 0)); @@ -1499,6 +1493,8 @@ ehci_resume(device_t dv, const pmf_qual_t *qual) if (hcr == EHCI_STS_HCH) printf("%s: config timeout\n", device_xname(dv)); + mutex_exit(&sc->sc_lock); + return true; }