diff options
author | Clemens Ladisch <clemens@ladisch.de> | 2010-03-01 03:12:50 -0500 |
---|---|---|
committer | Greg Kroah-Hartman <gregkh@suse.de> | 2010-04-01 19:01:36 -0400 |
commit | 25f3bec9fc514a21be373817f2306726340e5b5a (patch) | |
tree | 0c42e43468b3c4c93d55579061b023897086569c | |
parent | 55fe2557940e0cd1d88aa89ef580089698a7dac1 (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.c | 24 |
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( | |||
1563 | static inline void | 1563 | static inline void |
1564 | itd_link (struct ehci_hcd *ehci, unsigned frame, struct ehci_itd *itd) | 1564 | itd_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 */ |