Commit b40e43fc authored by Alan Stern's avatar Alan Stern Committed by Greg Kroah-Hartman

USB: EHCI: fix bug in Iso scheduling

This patch (as1098) changes the way ehci-hcd schedules its periodic
Iso transfers.  That the current scheduling code is wrong is clear on
the face of it: Sometimes it returns -EL2NSYNC (meaning that an URB
couldn't be scheduled because it was submitted too late), but it does
this even when the URB_ISO_ASAP flag is set (meaning the URB should be
scheduled as soon as possible).

The new code properly implements as-soon-as-possible scheduling,
assigning the next unexpired slot as the URB's starting point.  It
also is more careful about checking for Iso URB completion: It doesn't
bother to check for activity during frames that are already over,
and it allows for the possibility that some of the URB's packets may
have raced the hardware when they were submitted and so never got used
(the packet status is set to -EXDEV).

This fixes problems several people have experienced with USB video
applications.
Signed-off-by: default avatarAlan Stern <stern@rowland.harvard.edu>
Acked-by: default avatarDavid Brownell <dbrownell@users.sourceforge.net>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@suse.de>
parent d1f114d1
...@@ -1349,19 +1349,28 @@ iso_stream_schedule ( ...@@ -1349,19 +1349,28 @@ iso_stream_schedule (
/* when's the last uframe this urb could start? */ /* when's the last uframe this urb could start? */
max = now + mod; max = now + mod;
/* typical case: reuse current schedule. stream is still active, /* Typical case: reuse current schedule, stream is still active.
* and no gaps from host falling behind (irq delays etc) * Hopefully there are no gaps from the host falling behind
* (irq delays etc), but if there are we'll take the next
* slot in the schedule, implicitly assuming URB_ISO_ASAP.
*/ */
if (likely (!list_empty (&stream->td_list))) { if (likely (!list_empty (&stream->td_list))) {
start = stream->next_uframe; start = stream->next_uframe;
if (start < now) if (start < now)
start += mod; start += mod;
if (likely ((start + sched->span) < max))
goto ready; /* Fell behind (by up to twice the slop amount)? */
/* else fell behind; someday, try to reschedule */ if (start >= max - 2 * 8 * SCHEDULE_SLOP)
status = -EL2NSYNC; start += stream->interval * DIV_ROUND_UP(
max - start, stream->interval) - mod;
/* Tried to schedule too far into the future? */
if (unlikely((start + sched->span) >= max)) {
status = -EFBIG;
goto fail; goto fail;
} }
goto ready;
}
/* need to schedule; when's the next (u)frame we could start? /* need to schedule; when's the next (u)frame we could start?
* this is bigger than ehci->i_thresh allows; scheduling itself * this is bigger than ehci->i_thresh allows; scheduling itself
...@@ -1613,6 +1622,9 @@ itd_complete ( ...@@ -1613,6 +1622,9 @@ itd_complete (
} else if (likely ((t & EHCI_ISOC_ACTIVE) == 0)) { } else if (likely ((t & EHCI_ISOC_ACTIVE) == 0)) {
desc->status = 0; desc->status = 0;
desc->actual_length = EHCI_ITD_LENGTH (t); desc->actual_length = EHCI_ITD_LENGTH (t);
} else {
/* URB was too late */
desc->status = -EXDEV;
} }
} }
...@@ -2095,7 +2107,7 @@ done: ...@@ -2095,7 +2107,7 @@ done:
static void static void
scan_periodic (struct ehci_hcd *ehci) scan_periodic (struct ehci_hcd *ehci)
{ {
unsigned frame, clock, now_uframe, mod; unsigned now_uframe, frame, clock, clock_frame, mod;
unsigned modified; unsigned modified;
mod = ehci->periodic_size << 3; mod = ehci->periodic_size << 3;
...@@ -2111,6 +2123,7 @@ scan_periodic (struct ehci_hcd *ehci) ...@@ -2111,6 +2123,7 @@ scan_periodic (struct ehci_hcd *ehci)
else else
clock = now_uframe + mod - 1; clock = now_uframe + mod - 1;
clock %= mod; clock %= mod;
clock_frame = clock >> 3;
for (;;) { for (;;) {
union ehci_shadow q, *q_p; union ehci_shadow q, *q_p;
...@@ -2157,12 +2170,17 @@ restart: ...@@ -2157,12 +2170,17 @@ restart:
case Q_TYPE_ITD: case Q_TYPE_ITD:
/* If this ITD is still active, leave it for /* If this ITD is still active, leave it for
* later processing ... check the next entry. * later processing ... check the next entry.
* No need to check for activity unless the
* frame is current.
*/ */
rmb (); if (frame == clock_frame && live) {
for (uf = 0; uf < 8 && live; uf++) { rmb();
if (0 == (q.itd->hw_transaction [uf] for (uf = 0; uf < 8; uf++) {
& ITD_ACTIVE(ehci))) if (q.itd->hw_transaction[uf] &
continue; ITD_ACTIVE(ehci))
break;
}
if (uf < 8) {
incomplete = true; incomplete = true;
q_p = &q.itd->itd_next; q_p = &q.itd->itd_next;
hw_p = &q.itd->hw_next; hw_p = &q.itd->hw_next;
...@@ -2171,8 +2189,7 @@ restart: ...@@ -2171,8 +2189,7 @@ restart:
q = *q_p; q = *q_p;
break; break;
} }
if (uf < 8 && live) }
break;
/* Take finished ITDs out of the schedule /* Take finished ITDs out of the schedule
* and process them: recycle, maybe report * and process them: recycle, maybe report
...@@ -2189,9 +2206,12 @@ restart: ...@@ -2189,9 +2206,12 @@ restart:
case Q_TYPE_SITD: case Q_TYPE_SITD:
/* If this SITD is still active, leave it for /* If this SITD is still active, leave it for
* later processing ... check the next entry. * later processing ... check the next entry.
* No need to check for activity unless the
* frame is current.
*/ */
if ((q.sitd->hw_results & SITD_ACTIVE(ehci)) if (frame == clock_frame && live &&
&& live) { (q.sitd->hw_results &
SITD_ACTIVE(ehci))) {
incomplete = true; incomplete = true;
q_p = &q.sitd->sitd_next; q_p = &q.sitd->sitd_next;
hw_p = &q.sitd->hw_next; hw_p = &q.sitd->hw_next;
...@@ -2260,6 +2280,7 @@ restart: ...@@ -2260,6 +2280,7 @@ restart:
/* rescan the rest of this frame, then ... */ /* rescan the rest of this frame, then ... */
clock = now; clock = now;
clock_frame = clock >> 3;
} else { } else {
now_uframe++; now_uframe++;
now_uframe %= mod; now_uframe %= mod;
......
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