aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorClemens Ladisch <clemens@ladisch.de>2010-03-01 03:12:50 -0500
committerGreg Kroah-Hartman <gregkh@suse.de>2010-04-01 19:01:36 -0400
commit25f3bec9fc514a21be373817f2306726340e5b5a (patch)
tree0c42e43468b3c4c93d55579061b023897086569c
parent55fe2557940e0cd1d88aa89ef580089698a7dac1 (diff)
USB: EHCI: fix ITD list order
commit 92bc3648e6027384479852b770a542722fadee7c upstream. When isochronous URBs are shorter than one frame and when more than one ITD in a frame has been completed before the interrupt can be handled, scan_periodic() completes the URBs in the order in which they are found in the descriptor list. Therefore, the descriptor list must contain the ITDs in the correct order, i.e., a new ITD must be linked in after any previous ITDs of the same endpoint. This should fix garbled capture data in the USB audio drivers. Signed-off-by: Clemens Ladisch <clemens@ladisch.de> Reported-by: Colin Fletcher <colin.m.fletcher@googlemail.com> Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
-rw-r--r--drivers/usb/host/ehci-sched.c24
1 files changed, 19 insertions, 5 deletions
diff --git a/drivers/usb/host/ehci-sched.c b/drivers/usb/host/ehci-sched.c
index 1e391e624c8a..dfb8e09fbac5 100644
--- a/drivers/usb/host/ehci-sched.c
+++ b/drivers/usb/host/ehci-sched.c
@@ -1563,13 +1563,27 @@ itd_patch(
1563static inline void 1563static inline void
1564itd_link (struct ehci_hcd *ehci, unsigned frame, struct ehci_itd *itd) 1564itd_link (struct ehci_hcd *ehci, unsigned frame, struct ehci_itd *itd)
1565{ 1565{
1566 /* always prepend ITD/SITD ... only QH tree is order-sensitive */ 1566 union ehci_shadow *prev = &ehci->pshadow[frame];
1567 itd->itd_next = ehci->pshadow [frame]; 1567 __hc32 *hw_p = &ehci->periodic[frame];
1568 itd->hw_next = ehci->periodic [frame]; 1568 union ehci_shadow here = *prev;
1569 ehci->pshadow [frame].itd = itd; 1569 __hc32 type = 0;
1570
1571 /* skip any iso nodes which might belong to previous microframes */
1572 while (here.ptr) {
1573 type = Q_NEXT_TYPE(ehci, *hw_p);
1574 if (type == cpu_to_hc32(ehci, Q_TYPE_QH))
1575 break;
1576 prev = periodic_next_shadow(ehci, prev, type);
1577 hw_p = shadow_next_periodic(ehci, &here, type);
1578 here = *prev;
1579 }
1580
1581 itd->itd_next = here;
1582 itd->hw_next = *hw_p;
1583 prev->itd = itd;
1570 itd->frame = frame; 1584 itd->frame = frame;
1571 wmb (); 1585 wmb ();
1572 ehci->periodic[frame] = cpu_to_hc32(ehci, itd->itd_dma | Q_TYPE_ITD); 1586 *hw_p = cpu_to_hc32(ehci, itd->itd_dma | Q_TYPE_ITD);
1573} 1587}
1574 1588
1575/* fit urb's itds into the selected schedule slot; activate as needed */ 1589/* fit urb's itds into the selected schedule slot; activate as needed */