Commit 21b224d7 authored by Mathias Nyman's avatar Mathias Nyman Committed by Greg Kroah-Hartman

xhci: rework xhci internal endpoint halt state detection.

When xhci_requires_manual_halt_cleanup() was written it wasn't clear
that the xhci internal endpoint halt state always needs to be cleared
with a reset endpoint command. Functional stall cases additionally halt
the device side endpoint which requires class driver to clear the device
side halt with a CLEAR_FEATURE(ENDPOINT_HALT) request as well.

Clean up, rename, and make sure the new function always return true
when internal endpoint state is halted, including stall cases.

Based on related cleanup suggestion code by Niklas Neronin

cc: Niklas Neronin <niklas.neronin@linux.intel.com>
Signed-off-by: default avatarMathias Nyman <mathias.nyman@linux.intel.com>
Link: https://lore.kernel.org/r/20240626124835.1023046-15-mathias.nyman@linux.intel.comSigned-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent bde66d2d
...@@ -2130,28 +2130,34 @@ static void xhci_clear_hub_tt_buffer(struct xhci_hcd *xhci, struct xhci_td *td, ...@@ -2130,28 +2130,34 @@ static void xhci_clear_hub_tt_buffer(struct xhci_hcd *xhci, struct xhci_td *td,
} }
} }
/* Check if an error has halted the endpoint ring. The class driver will /*
* cleanup the halt for a non-default control endpoint if we indicate a stall. * Check if xhci internal endpoint state has gone to a "halt" state due to an
* However, a babble and other errors also halt the endpoint ring, and the class * error or stall, including default control pipe protocol stall.
* driver won't clear the halt in that case, so we need to issue a Set Transfer * The internal halt needs to be cleared with a reset endpoint command.
* Ring Dequeue Pointer command manually. *
* External device side is also halted in functional stall cases. Class driver
* will clear the device halt with a CLEAR_FEATURE(ENDPOINT_HALT) request later.
*/ */
static int xhci_requires_manual_halt_cleanup(struct xhci_ep_ctx *ep_ctx, unsigned int trb_comp_code) static bool xhci_halted_host_endpoint(struct xhci_ep_ctx *ep_ctx, unsigned int comp_code)
{ {
/* TRB completion codes that may require a manual halt cleanup */ /* Stall halts both internal and device side endpoint */
if (trb_comp_code == COMP_USB_TRANSACTION_ERROR || if (comp_code == COMP_STALL_ERROR)
trb_comp_code == COMP_BABBLE_DETECTED_ERROR || return true;
trb_comp_code == COMP_SPLIT_TRANSACTION_ERROR)
/* The 0.95 spec says a babbling control endpoint /* TRB completion codes that may require internal halt cleanup */
* is not halted. The 0.96 spec says it is. Some HW if (comp_code == COMP_USB_TRANSACTION_ERROR ||
* claims to be 0.95 compliant, but it halts the control comp_code == COMP_BABBLE_DETECTED_ERROR ||
* endpoint anyway. Check if a babble halted the comp_code == COMP_SPLIT_TRANSACTION_ERROR)
* endpoint. /*
* The 0.95 spec says a babbling control endpoint is not halted.
* The 0.96 spec says it is. Some HW claims to be 0.95
* compliant, but it halts the control endpoint anyway.
* Check endpoint context if endpoint is halted.
*/ */
if (GET_EP_CTX_STATE(ep_ctx) == EP_STATE_HALTED) if (GET_EP_CTX_STATE(ep_ctx) == EP_STATE_HALTED)
return 1; return true;
return 0; return false;
} }
int xhci_is_vendor_info_code(struct xhci_hcd *xhci, unsigned int trb_comp_code) int xhci_is_vendor_info_code(struct xhci_hcd *xhci, unsigned int trb_comp_code)
...@@ -2321,7 +2327,7 @@ static int process_ctrl_td(struct xhci_hcd *xhci, struct xhci_virt_ep *ep, ...@@ -2321,7 +2327,7 @@ static int process_ctrl_td(struct xhci_hcd *xhci, struct xhci_virt_ep *ep,
case COMP_STOPPED_LENGTH_INVALID: case COMP_STOPPED_LENGTH_INVALID:
goto finish_td; goto finish_td;
default: default:
if (!xhci_requires_manual_halt_cleanup(ep_ctx, trb_comp_code)) if (!xhci_halted_host_endpoint(ep_ctx, trb_comp_code))
break; break;
xhci_dbg(xhci, "TRB error %u, halted endpoint index = %u\n", xhci_dbg(xhci, "TRB error %u, halted endpoint index = %u\n",
trb_comp_code, ep->ep_index); trb_comp_code, ep->ep_index);
...@@ -2791,11 +2797,9 @@ static int handle_tx_event(struct xhci_hcd *xhci, ...@@ -2791,11 +2797,9 @@ static int handle_tx_event(struct xhci_hcd *xhci,
xhci_dbg(xhci, "td_list is empty while skip flag set. Clear skip flag for slot %u ep %u.\n", xhci_dbg(xhci, "td_list is empty while skip flag set. Clear skip flag for slot %u ep %u.\n",
slot_id, ep_index); slot_id, ep_index);
} }
if (trb_comp_code == COMP_STALL_ERROR || if (xhci_halted_host_endpoint(ep_ctx, trb_comp_code))
xhci_requires_manual_halt_cleanup(ep_ctx, trb_comp_code)) { xhci_handle_halted_endpoint(xhci, ep, NULL, EP_HARD_RESET);
xhci_handle_halted_endpoint(xhci, ep, NULL,
EP_HARD_RESET);
}
return 0; return 0;
} }
...@@ -2911,10 +2915,8 @@ static int handle_tx_event(struct xhci_hcd *xhci, ...@@ -2911,10 +2915,8 @@ static int handle_tx_event(struct xhci_hcd *xhci,
*/ */
if (trb_is_noop(ep_trb)) { if (trb_is_noop(ep_trb)) {
if (trb_comp_code == COMP_STALL_ERROR || if (xhci_halted_host_endpoint(ep_ctx, trb_comp_code))
xhci_requires_manual_halt_cleanup(ep_ctx, trb_comp_code)) xhci_handle_halted_endpoint(xhci, ep, td, EP_HARD_RESET);
xhci_handle_halted_endpoint(xhci, ep, td,
EP_HARD_RESET);
} else { } else {
td->status = status; td->status = status;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment