From ccfd5ec803314c41cb4f9b817b44c65f6f25eab5 Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Wed, 7 Sep 2022 07:05:07 +0000 Subject: [PATCH 1/6] xhci(4): Suspend commands while detaching. XXX OOPS -- this will deadlock because there's nothing to make xhci_do_command_locked wake up if we actually commit to detaching. --- sys/dev/usb/xhci.c | 43 +++++++++++++++++++++++++++++-------------- 1 file changed, 29 insertions(+), 14 deletions(-) diff --git a/sys/dev/usb/xhci.c b/sys/dev/usb/xhci.c index 6379e436d0c8..bf4711203036 100644 --- a/sys/dev/usb/xhci.c +++ b/sys/dev/usb/xhci.c @@ -611,20 +611,32 @@ xhci_childdet(device_t self, device_t child) int xhci_detach(struct xhci_softc *sc, int flags) { - int rv = 0; + int error; - if (sc->sc_child2 != NULL) { - rv = config_detach(sc->sc_child2, flags); - if (rv != 0) - return rv; - KASSERT(sc->sc_child2 == NULL); - } + /* + * Stop heeding xhci interrupts and cause all xhci commands to + * fail. Suboptimal: if a child is still in use, it may lose + * data and/or end up in a broken state requiring detach and + * reattach anyway, so backing out if we fail to detach might + * not work. + */ + mutex_enter(&sc->sc_lock); + mutex_enter(&sc->sc_intr_lock); + KASSERT(!sc->sc_dying); + sc->sc_dying = true; + mutex_exit(&sc->sc_intr_lock); + cv_broadcast(&sc->sc_cmdbusy_cv); + mutex_exit(&sc->sc_lock); - if (sc->sc_child != NULL) { - rv = config_detach(sc->sc_child, flags); - if (rv != 0) - return rv; - KASSERT(sc->sc_child == NULL); + error = config_detach_children(sc->sc_dev, flags); + if (error) { + mutex_enter(&sc->sc_lock); + mutex_enter(&sc->sc_intr_lock); + KASSERT(sc->sc_dying); + sc->sc_dying = false; + mutex_exit(&sc->sc_intr_lock); + mutex_exit(&sc->sc_lock); + return error; } /* XXX unconfigure/free slots */ @@ -665,7 +677,7 @@ xhci_detach(struct xhci_softc *sc, int flags) pool_cache_destroy(sc->sc_xferpool); - return rv; + return 0; } int @@ -3197,8 +3209,11 @@ xhci_do_command_locked(struct xhci_softc * const sc, KASSERT(mutex_owned(&sc->sc_lock)); while (sc->sc_command_addr != 0 || - (sc->sc_suspender != NULL && sc->sc_suspender != curlwp)) + (sc->sc_suspender != NULL && sc->sc_suspender != curlwp)) { + if (sc->sc_dying) + return USBD_IOERROR; cv_wait(&sc->sc_cmdbusy_cv, &sc->sc_lock); + } /* * If enqueue pointer points at last of ring, it's Link TRB, From d03f2c4d70ea0c5aa075233c7a7431dcf7683b56 Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Wed, 7 Sep 2022 08:56:57 +0000 Subject: [PATCH 2/6] Revert "xhci(4): Resume commands even if USBSTS.SRE is set." This reverts commit 3130d9e2d4536c81ebe23447a1d6ef2306dc78f5. --- sys/dev/usb/xhci.c | 15 +++++---------- 1 file changed, 5 insertions(+), 10 deletions(-) diff --git a/sys/dev/usb/xhci.c b/sys/dev/usb/xhci.c index bf4711203036..64a1b7e6132e 100644 --- a/sys/dev/usb/xhci.c +++ b/sys/dev/usb/xhci.c @@ -1111,22 +1111,17 @@ xhci_resume(device_t self, const pmf_qual_t *qual) goto out; } - /* Success! */ - ok = true; - -out: /* - * Resume command issuance. If the hardware failed to resume, - * well, tough -- deadlocking because everything is held up on - * the suspension, with no opportunity to detach, isn't better - * than timing out waiting for dead hardware. - */ + /* Resume command issuance. */ mutex_enter(&sc->sc_lock); KASSERT(sc->sc_suspender); sc->sc_suspender = NULL; cv_broadcast(&sc->sc_cmdbusy_cv); mutex_exit(&sc->sc_lock); - mutex_exit(&sc->sc_rhlock); + /* Success! */ + ok = true; + +out: mutex_exit(&sc->sc_rhlock); return ok; } From 36be4c2e0c54867ce84e61766dbc71810736e339 Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Wed, 7 Sep 2022 13:53:18 +0000 Subject: [PATCH 3/6] xhci(4): Wait for CNR to clear before resuming. Linux does this; let's see if it helps with anything here. --- sys/dev/usb/xhci.c | 19 +++++++++++++++++++ 1 file changed, 19 insertions(+) diff --git a/sys/dev/usb/xhci.c b/sys/dev/usb/xhci.c index 64a1b7e6132e..c745dbea3cad 100644 --- a/sys/dev/usb/xhci.c +++ b/sys/dev/usb/xhci.c @@ -937,6 +937,25 @@ xhci_resume(device_t self, const pmf_qual_t *qual) */ mutex_enter(&sc->sc_rhlock); + /* + * Wait for Controller Not Ready to clear like in xhci_hc_reset + * on init. Not required in the spec; parroted from Linux. + */ + for (i = 0; i < XHCI_WAIT_CNR; i++) { + if ((xhci_op_read_4(sc, XHCI_USBSTS) & XHCI_STS_CNR) == 0) { + if (i > 0) { + aprint_debug_dev(self, + "waited %zums for CNR\n", i); + } + break; + } + usb_delay_ms(&sc->sc_bus, 1); + } + if (i >= XHCI_WAIT_CNR) { + device_printf(self, "controller not ready timeout\n"); + goto out; + } + /* * xHCI Requirements Specification 1.2, May 2019, Sec. 4.23.2: * xHCI Power Management, p. 343 From 585cabecdfe38956e164c63cedab2a386f73e7ef Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Wed, 7 Sep 2022 14:36:23 +0000 Subject: [PATCH 4/6] Revert "xhci(4): Wait for CNR to clear before resuming." This reverts commit df38e6cb0b8c73553ae7ba13a2336bdb33ff6c72. --- sys/dev/usb/xhci.c | 19 ------------------- 1 file changed, 19 deletions(-) diff --git a/sys/dev/usb/xhci.c b/sys/dev/usb/xhci.c index c745dbea3cad..64a1b7e6132e 100644 --- a/sys/dev/usb/xhci.c +++ b/sys/dev/usb/xhci.c @@ -937,25 +937,6 @@ xhci_resume(device_t self, const pmf_qual_t *qual) */ mutex_enter(&sc->sc_rhlock); - /* - * Wait for Controller Not Ready to clear like in xhci_hc_reset - * on init. Not required in the spec; parroted from Linux. - */ - for (i = 0; i < XHCI_WAIT_CNR; i++) { - if ((xhci_op_read_4(sc, XHCI_USBSTS) & XHCI_STS_CNR) == 0) { - if (i > 0) { - aprint_debug_dev(self, - "waited %zums for CNR\n", i); - } - break; - } - usb_delay_ms(&sc->sc_bus, 1); - } - if (i >= XHCI_WAIT_CNR) { - device_printf(self, "controller not ready timeout\n"); - goto out; - } - /* * xHCI Requirements Specification 1.2, May 2019, Sec. 4.23.2: * xHCI Power Management, p. 343 From 2a3dc77141ee28670cb7cb2d80cf08db4c3a8825 Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Wed, 7 Sep 2022 13:55:30 +0000 Subject: [PATCH 5/6] xhci(4): Try resetting if resume fails. --- sys/dev/usb/xhci.c | 25 +++++++++++++++++++++++-- sys/dev/usb/xhcivar.h | 1 + 2 files changed, 24 insertions(+), 2 deletions(-) diff --git a/sys/dev/usb/xhci.c b/sys/dev/usb/xhci.c index 64a1b7e6132e..c49dec83394c 100644 --- a/sys/dev/usb/xhci.c +++ b/sys/dev/usb/xhci.c @@ -139,6 +139,8 @@ struct xhci_pipe { #define XHCI_EVENT_RING_SEGMENTS 1 #define XHCI_TRB_3_ED_BIT XHCI_TRB_3_ISP_BIT +static int xhci_hc_reset(struct xhci_softc *); + static usbd_status xhci_open(struct usbd_pipe *); static void xhci_close_pipe(struct usbd_pipe *); static int xhci_intr1(struct xhci_softc * const); @@ -1108,7 +1110,21 @@ xhci_resume(device_t self, const pmf_qual_t *qual) */ if (xhci_op_read_4(sc, XHCI_USBSTS) & XHCI_STS_SRE) { device_printf(self, "resume error, USBSTS.SRE\n"); - goto out; + if (xhci_hc_reset(sc) != 0) + goto out; + /* copied from xhci_init */ + xhci_rt_write_4(sc, XHCI_ERSTSZ(0), XHCI_EVENT_RING_SEGMENTS); + xhci_rt_write_8(sc, XHCI_ERSTBA(0), + DMAADDR(&sc->sc_eventst_dma, 0)); + xhci_rt_write_8(sc, XHCI_ERDP(0), + xhci_ring_trbp(sc->sc_er, 0) | XHCI_ERDP_BUSY); + xhci_op_write_8(sc, XHCI_DCBAAP, + DMAADDR(&sc->sc_dcbaa_dma, 0)); + xhci_op_write_8(sc, XHCI_CRCR, xhci_ring_trbp(sc->sc_cr, 0) | + sc->sc_cr->xr_cs); + xhci_start(sc); + device_printf(self, "resuming after reset\n"); + sc->sc_resetgen++; } /* Resume command issuance. */ @@ -3194,6 +3210,7 @@ xhci_do_command_locked(struct xhci_softc * const sc, struct xhci_soft_trb * const trb, int timeout) { struct xhci_ring * const cr = sc->sc_cr; + uint32_t resetgen = sc->sc_resetgen; usbd_status err; XHCIHIST_FUNC(); @@ -3205,7 +3222,7 @@ xhci_do_command_locked(struct xhci_softc * const sc, while (sc->sc_command_addr != 0 || (sc->sc_suspender != NULL && sc->sc_suspender != curlwp)) { - if (sc->sc_dying) + if (sc->sc_dying || sc->sc_resetgen != resetgen) return USBD_IOERROR; cv_wait(&sc->sc_cmdbusy_cv, &sc->sc_lock); } @@ -3234,6 +3251,10 @@ xhci_do_command_locked(struct xhci_softc * const sc, err = USBD_TIMEOUT; goto timedout; } + if (sc->sc_resetgen != resetgen) { + err = USBD_IOERROR; + goto timedout; + } } trb->trb_0 = sc->sc_result_trb.trb_0; diff --git a/sys/dev/usb/xhcivar.h b/sys/dev/usb/xhcivar.h index 20ae54059332..32347a71592f 100644 --- a/sys/dev/usb/xhcivar.h +++ b/sys/dev/usb/xhcivar.h @@ -140,6 +140,7 @@ struct xhci_softc { bool sc_resultpending; bool sc_dying; + uint32_t sc_resetgen; struct lwp *sc_suspender; void (*sc_vendor_init)(struct xhci_softc *); From b318fd905bdda3e74a3605186feeb0295ac78cee Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Wed, 7 Sep 2022 13:58:50 +0000 Subject: [PATCH 6/6] Revert "xhci(4): Try resetting if resume fails." This reverts commit 0b5c36c2dd63a3a32f8b01ffdadf8b3099076c22. --- sys/dev/usb/xhci.c | 25 ++----------------------- sys/dev/usb/xhcivar.h | 1 - 2 files changed, 2 insertions(+), 24 deletions(-) diff --git a/sys/dev/usb/xhci.c b/sys/dev/usb/xhci.c index c49dec83394c..64a1b7e6132e 100644 --- a/sys/dev/usb/xhci.c +++ b/sys/dev/usb/xhci.c @@ -139,8 +139,6 @@ struct xhci_pipe { #define XHCI_EVENT_RING_SEGMENTS 1 #define XHCI_TRB_3_ED_BIT XHCI_TRB_3_ISP_BIT -static int xhci_hc_reset(struct xhci_softc *); - static usbd_status xhci_open(struct usbd_pipe *); static void xhci_close_pipe(struct usbd_pipe *); static int xhci_intr1(struct xhci_softc * const); @@ -1110,21 +1108,7 @@ xhci_resume(device_t self, const pmf_qual_t *qual) */ if (xhci_op_read_4(sc, XHCI_USBSTS) & XHCI_STS_SRE) { device_printf(self, "resume error, USBSTS.SRE\n"); - if (xhci_hc_reset(sc) != 0) - goto out; - /* copied from xhci_init */ - xhci_rt_write_4(sc, XHCI_ERSTSZ(0), XHCI_EVENT_RING_SEGMENTS); - xhci_rt_write_8(sc, XHCI_ERSTBA(0), - DMAADDR(&sc->sc_eventst_dma, 0)); - xhci_rt_write_8(sc, XHCI_ERDP(0), - xhci_ring_trbp(sc->sc_er, 0) | XHCI_ERDP_BUSY); - xhci_op_write_8(sc, XHCI_DCBAAP, - DMAADDR(&sc->sc_dcbaa_dma, 0)); - xhci_op_write_8(sc, XHCI_CRCR, xhci_ring_trbp(sc->sc_cr, 0) | - sc->sc_cr->xr_cs); - xhci_start(sc); - device_printf(self, "resuming after reset\n"); - sc->sc_resetgen++; + goto out; } /* Resume command issuance. */ @@ -3210,7 +3194,6 @@ xhci_do_command_locked(struct xhci_softc * const sc, struct xhci_soft_trb * const trb, int timeout) { struct xhci_ring * const cr = sc->sc_cr; - uint32_t resetgen = sc->sc_resetgen; usbd_status err; XHCIHIST_FUNC(); @@ -3222,7 +3205,7 @@ xhci_do_command_locked(struct xhci_softc * const sc, while (sc->sc_command_addr != 0 || (sc->sc_suspender != NULL && sc->sc_suspender != curlwp)) { - if (sc->sc_dying || sc->sc_resetgen != resetgen) + if (sc->sc_dying) return USBD_IOERROR; cv_wait(&sc->sc_cmdbusy_cv, &sc->sc_lock); } @@ -3251,10 +3234,6 @@ xhci_do_command_locked(struct xhci_softc * const sc, err = USBD_TIMEOUT; goto timedout; } - if (sc->sc_resetgen != resetgen) { - err = USBD_IOERROR; - goto timedout; - } } trb->trb_0 = sc->sc_result_trb.trb_0; diff --git a/sys/dev/usb/xhcivar.h b/sys/dev/usb/xhcivar.h index 32347a71592f..20ae54059332 100644 --- a/sys/dev/usb/xhcivar.h +++ b/sys/dev/usb/xhcivar.h @@ -140,7 +140,6 @@ struct xhci_softc { bool sc_resultpending; bool sc_dying; - uint32_t sc_resetgen; struct lwp *sc_suspender; void (*sc_vendor_init)(struct xhci_softc *);