Commit 044818a6 authored by Lukas Wunner's avatar Lukas Wunner Committed by Greg Kroah-Hartman

xhci: Set DESI bits in ERDP register correctly

When using more than one Event Ring segment (ERSTSZ > 1), software shall
set the DESI bits in the ERDP register to the number of the segment to
which the upper ERDP bits are pointing.  The xHC may use the DESI bits
as a shortcut to determine whether it needs to check for an Event Ring
Full condition:  If it's enqueueing events in a different segment, it
need not compare its internal Enqueue Pointer with the Dequeue Pointer
in the upper bits of the ERDP register (sec 5.5.2.3.3).

Not setting the DESI bits correctly can result in the xHC enqueueing
events past the Dequeue Pointer.  On Renesas uPD720201 host controllers,
incorrect DESI bits cause an interrupt storm.  For comparison, VIA VL805
host controllers do not exhibit such problems.  Perhaps they do not take
advantage of the optimization afforded by the DESI bits.

To fix the issue, assign the segment number to each struct xhci_segment
in xhci_segment_alloc().  When advancing the Dequeue Pointer in
xhci_update_erst_dequeue(), write the segment number to the DESI bits.

On driver probe, set the DESI bits to zero in xhci_set_hc_event_deq() as
processing starts in segment 0.  Likewise on driver teardown, clear the
DESI bits to zero in xhci_free_interrupter() when clearing the upper
bits of the ERDP register.  Previously those functions (incorrectly)
treated the DESI bits as if they're declared RsvdP.
Signed-off-by: default avatarLukas Wunner <lukas@wunner.de>
Signed-off-by: default avatarMathias Nyman <mathias.nyman@linux.intel.com>
Link: https://lore.kernel.org/r/20231019102924.2797346-5-mathias.nyman@linux.intel.comSigned-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent e2d3ac9c
...@@ -29,6 +29,7 @@ ...@@ -29,6 +29,7 @@
static struct xhci_segment *xhci_segment_alloc(struct xhci_hcd *xhci, static struct xhci_segment *xhci_segment_alloc(struct xhci_hcd *xhci,
unsigned int cycle_state, unsigned int cycle_state,
unsigned int max_packet, unsigned int max_packet,
unsigned int num,
gfp_t flags) gfp_t flags)
{ {
struct xhci_segment *seg; struct xhci_segment *seg;
...@@ -60,6 +61,7 @@ static struct xhci_segment *xhci_segment_alloc(struct xhci_hcd *xhci, ...@@ -60,6 +61,7 @@ static struct xhci_segment *xhci_segment_alloc(struct xhci_hcd *xhci,
for (i = 0; i < TRBS_PER_SEGMENT; i++) for (i = 0; i < TRBS_PER_SEGMENT; i++)
seg->trbs[i].link.control = cpu_to_le32(TRB_CYCLE); seg->trbs[i].link.control = cpu_to_le32(TRB_CYCLE);
} }
seg->num = num;
seg->dma = dma; seg->dma = dma;
seg->next = NULL; seg->next = NULL;
...@@ -324,6 +326,7 @@ static int xhci_alloc_segments_for_ring(struct xhci_hcd *xhci, ...@@ -324,6 +326,7 @@ static int xhci_alloc_segments_for_ring(struct xhci_hcd *xhci,
enum xhci_ring_type type, unsigned int max_packet, gfp_t flags) enum xhci_ring_type type, unsigned int max_packet, gfp_t flags)
{ {
struct xhci_segment *prev; struct xhci_segment *prev;
unsigned int num = 0;
bool chain_links; bool chain_links;
/* Set chain bit for 0.95 hosts, and for isoc rings on AMD 0.96 host */ /* Set chain bit for 0.95 hosts, and for isoc rings on AMD 0.96 host */
...@@ -331,16 +334,17 @@ static int xhci_alloc_segments_for_ring(struct xhci_hcd *xhci, ...@@ -331,16 +334,17 @@ static int xhci_alloc_segments_for_ring(struct xhci_hcd *xhci,
(type == TYPE_ISOC && (type == TYPE_ISOC &&
(xhci->quirks & XHCI_AMD_0x96_HOST))); (xhci->quirks & XHCI_AMD_0x96_HOST)));
prev = xhci_segment_alloc(xhci, cycle_state, max_packet, flags); prev = xhci_segment_alloc(xhci, cycle_state, max_packet, num, flags);
if (!prev) if (!prev)
return -ENOMEM; return -ENOMEM;
num_segs--; num++;
*first = prev; *first = prev;
while (num_segs > 0) { while (num < num_segs) {
struct xhci_segment *next; struct xhci_segment *next;
next = xhci_segment_alloc(xhci, cycle_state, max_packet, flags); next = xhci_segment_alloc(xhci, cycle_state, max_packet, num,
flags);
if (!next) { if (!next) {
prev = *first; prev = *first;
while (prev) { while (prev) {
...@@ -353,7 +357,7 @@ static int xhci_alloc_segments_for_ring(struct xhci_hcd *xhci, ...@@ -353,7 +357,7 @@ static int xhci_alloc_segments_for_ring(struct xhci_hcd *xhci,
xhci_link_segments(prev, next, type, chain_links); xhci_link_segments(prev, next, type, chain_links);
prev = next; prev = next;
num_segs--; num++;
} }
xhci_link_segments(prev, *first, type, chain_links); xhci_link_segments(prev, *first, type, chain_links);
*last = prev; *last = prev;
...@@ -1801,7 +1805,6 @@ xhci_free_interrupter(struct xhci_hcd *xhci, struct xhci_interrupter *ir) ...@@ -1801,7 +1805,6 @@ xhci_free_interrupter(struct xhci_hcd *xhci, struct xhci_interrupter *ir)
{ {
struct device *dev = xhci_to_hcd(xhci)->self.sysdev; struct device *dev = xhci_to_hcd(xhci)->self.sysdev;
size_t erst_size; size_t erst_size;
u64 tmp64;
u32 tmp; u32 tmp;
if (!ir) if (!ir)
...@@ -1824,9 +1827,7 @@ xhci_free_interrupter(struct xhci_hcd *xhci, struct xhci_interrupter *ir) ...@@ -1824,9 +1827,7 @@ xhci_free_interrupter(struct xhci_hcd *xhci, struct xhci_interrupter *ir)
tmp &= ERST_SIZE_MASK; tmp &= ERST_SIZE_MASK;
writel(tmp, &ir->ir_set->erst_size); writel(tmp, &ir->ir_set->erst_size);
tmp64 = xhci_read_64(xhci, &ir->ir_set->erst_dequeue); xhci_write_64(xhci, ERST_EHB, &ir->ir_set->erst_dequeue);
tmp64 &= (u64) ERST_PTR_MASK;
xhci_write_64(xhci, tmp64, &ir->ir_set->erst_dequeue);
} }
/* free interrrupter event ring */ /* free interrrupter event ring */
...@@ -1933,7 +1934,6 @@ void xhci_mem_cleanup(struct xhci_hcd *xhci) ...@@ -1933,7 +1934,6 @@ void xhci_mem_cleanup(struct xhci_hcd *xhci)
static void xhci_set_hc_event_deq(struct xhci_hcd *xhci, struct xhci_interrupter *ir) static void xhci_set_hc_event_deq(struct xhci_hcd *xhci, struct xhci_interrupter *ir)
{ {
u64 temp;
dma_addr_t deq; dma_addr_t deq;
deq = xhci_trb_virt_to_dma(ir->event_ring->deq_seg, deq = xhci_trb_virt_to_dma(ir->event_ring->deq_seg,
...@@ -1941,15 +1941,12 @@ static void xhci_set_hc_event_deq(struct xhci_hcd *xhci, struct xhci_interrupter ...@@ -1941,15 +1941,12 @@ static void xhci_set_hc_event_deq(struct xhci_hcd *xhci, struct xhci_interrupter
if (!deq) if (!deq)
xhci_warn(xhci, "WARN something wrong with SW event ring dequeue ptr.\n"); xhci_warn(xhci, "WARN something wrong with SW event ring dequeue ptr.\n");
/* Update HC event ring dequeue pointer */ /* Update HC event ring dequeue pointer */
temp = xhci_read_64(xhci, &ir->ir_set->erst_dequeue);
temp &= ERST_PTR_MASK;
/* Don't clear the EHB bit (which is RW1C) because /* Don't clear the EHB bit (which is RW1C) because
* there might be more events to service. * there might be more events to service.
*/ */
temp &= ~ERST_EHB;
xhci_dbg_trace(xhci, trace_xhci_dbg_init, xhci_dbg_trace(xhci, trace_xhci_dbg_init,
"// Write event ring dequeue pointer, preserving EHB bit"); "// Write event ring dequeue pointer, preserving EHB bit");
xhci_write_64(xhci, ((u64) deq & (u64) ~ERST_PTR_MASK) | temp, xhci_write_64(xhci, ((u64) deq & (u64) ~ERST_PTR_MASK),
&ir->ir_set->erst_dequeue); &ir->ir_set->erst_dequeue);
} }
......
...@@ -3018,7 +3018,7 @@ static void xhci_update_erst_dequeue(struct xhci_hcd *xhci, ...@@ -3018,7 +3018,7 @@ static void xhci_update_erst_dequeue(struct xhci_hcd *xhci,
return; return;
/* Update HC event ring dequeue pointer */ /* Update HC event ring dequeue pointer */
temp_64 &= ERST_DESI_MASK; temp_64 = ir->event_ring->deq_seg->num & ERST_DESI_MASK;
temp_64 |= ((u64) deq & (u64) ~ERST_PTR_MASK); temp_64 |= ((u64) deq & (u64) ~ERST_PTR_MASK);
} }
......
...@@ -1518,6 +1518,7 @@ struct xhci_segment { ...@@ -1518,6 +1518,7 @@ struct xhci_segment {
union xhci_trb *trbs; union xhci_trb *trbs;
/* private to HCD */ /* private to HCD */
struct xhci_segment *next; struct xhci_segment *next;
unsigned int num;
dma_addr_t dma; dma_addr_t dma;
/* Max packet sized bounce buffer for td-fragmant alignment */ /* Max packet sized bounce buffer for td-fragmant alignment */
dma_addr_t bounce_dma; dma_addr_t bounce_dma;
......
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