From d0b688a23659b6f60074a588f15d551de133bb47 Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Fri, 17 Jun 2022 10:36:40 +0000 Subject: [PATCH 1/3] xhci(4): Handle another race between abort and hardware completion. XXX Broken! --- sys/dev/usb/xhci.c | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/sys/dev/usb/xhci.c b/sys/dev/usb/xhci.c index 6379e436d0c8..183edffa25d3 100644 --- a/sys/dev/usb/xhci.c +++ b/sys/dev/usb/xhci.c @@ -2439,6 +2439,12 @@ xhci_event_transfer(struct xhci_softc * const sc, return; } + if (xfer->ux_pipe->up_aborting) { + DPRINTF("xfer %#jx pipe %#jx is aborting", + (uintptr_t)xfer, (uintptr_t)xfer->ux_pipe, 0, 0); + return; + } + const uint8_t xfertype = UE_GET_XFERTYPE(xfer->ux_pipe->up_endpoint->ue_edesc->bmAttributes); From 46a7d7f538ba10b508e423bb6df40214e1bf0269 Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Fri, 17 Jun 2022 10:37:55 +0000 Subject: [PATCH 2/3] Revert "xhci(4): Handle another race between abort and hardware completion." This reverts commit 771c90ed6d74c621638f1e35637582bb1f61a697. --- sys/dev/usb/xhci.c | 6 ------ 1 file changed, 6 deletions(-) diff --git a/sys/dev/usb/xhci.c b/sys/dev/usb/xhci.c index 183edffa25d3..6379e436d0c8 100644 --- a/sys/dev/usb/xhci.c +++ b/sys/dev/usb/xhci.c @@ -2439,12 +2439,6 @@ xhci_event_transfer(struct xhci_softc * const sc, return; } - if (xfer->ux_pipe->up_aborting) { - DPRINTF("xfer %#jx pipe %#jx is aborting", - (uintptr_t)xfer, (uintptr_t)xfer->ux_pipe, 0, 0); - return; - } - const uint8_t xfertype = UE_GET_XFERTYPE(xfer->ux_pipe->up_endpoint->ue_edesc->bmAttributes); From f3a5fd25ba61bbe147223b4f4f8fbf3e05055b37 Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Fri, 17 Jun 2022 10:45:52 +0000 Subject: [PATCH 3/3] xhci(4): Handle another race between abort and hardware completion. --- sys/dev/usb/xhci.c | 21 +++++++++++++++++++++ 1 file changed, 21 insertions(+) diff --git a/sys/dev/usb/xhci.c b/sys/dev/usb/xhci.c index 6379e436d0c8..b56bcc511aba 100644 --- a/sys/dev/usb/xhci.c +++ b/sys/dev/usb/xhci.c @@ -132,6 +132,7 @@ struct xhci_pipe { int16_t xp_isoc_next; /* next frame */ uint8_t xp_maxb; /* max burst */ uint8_t xp_mult; + bool xp_aborting; }; #define XHCI_COMMAND_RING_TRBS 256 @@ -2031,6 +2032,8 @@ xhci_open(struct usbd_pipe *pipe) if (sc->sc_dying) return USBD_IOERROR; + xpipe->xp_aborting = false; + /* Root Hub */ if (dev->ud_depth == 0 && dev->ud_powersrc->up_portno == 0) { switch (ed->bEndpointAddress) { @@ -2108,6 +2111,8 @@ xhci_close_pipe(struct usbd_pipe *pipe) XHCIHIST_FUNC(); + KASSERT(!xp->xp_aborting); + usb_rem_task_wait(pipe->up_dev, &xp->xp_async_task, USB_TASKQ_HC, &sc->sc_lock); @@ -2177,6 +2182,8 @@ xhci_abortx(struct usbd_xfer *xfer) { XHCIHIST_FUNC(); struct xhci_softc * const sc = XHCI_XFER2SC(xfer); + struct xhci_pipe * const xpipe = + container_of(xfer->ux_pipe, struct xhci_pipe, xp_pipe); XHCIHIST_CALLARGS("xfer %#jx pipe %#jx", (uintptr_t)xfer, (uintptr_t)xfer->ux_pipe, 0, 0); @@ -2186,8 +2193,14 @@ xhci_abortx(struct usbd_xfer *xfer) xfer->ux_status == USBD_TIMEOUT), "bad abort status: %d", xfer->ux_status); + KASSERT(!xpipe->xp_aborting); + xpipe->xp_aborting = true; + xhci_pipe_restart(xfer->ux_pipe); + KASSERT(xpipe->xp_aborting); + xpipe->xp_aborting = false; + DPRINTFN(14, "end", 0, 0, 0, 0); } @@ -2371,6 +2384,7 @@ xhci_event_transfer(struct xhci_softc * const sc, struct xhci_ring *xr; struct xhci_xfer *xx; struct usbd_xfer *xfer; + struct xhci_pipe *xpipe; usbd_status err; XHCIHIST_FUNC(); XHCIHIST_CALLED(); @@ -2439,6 +2453,13 @@ xhci_event_transfer(struct xhci_softc * const sc, return; } + xpipe = container_of(xfer->ux_pipe, struct xhci_pipe, xp_pipe); + if (xpipe->xp_aborting) { + DPRINTF("xfer %#jx pipe %#jx is aborting", + (uintptr_t)xfer, (uintptr_t)xfer->ux_pipe, 0, 0); + return; + } + const uint8_t xfertype = UE_GET_XFERTYPE(xfer->ux_pipe->up_endpoint->ue_edesc->bmAttributes);