diff options
author | Alan Stern <stern@rowland.harvard.edu> | 2013-03-22 13:30:56 -0400 |
---|---|---|
committer | Greg Kroah-Hartman <gregkh@linuxfoundation.org> | 2013-03-25 16:35:05 -0400 |
commit | 79bcf7b02ba3d45bafe81a2753cedb8ef49548e3 (patch) | |
tree | da06b30b0fdd12dfb3286bed6a940bd2e4e0dfe0 /drivers/usb/host/ehci-sched.c | |
parent | c1fdb68e3d73741630ca16695cf9176c233be7ed (diff) |
USB: EHCI: change return value of qh_completions()
This patch (as1658) cleans up the usage of qh_completions() in
ehci-hcd. Currently the function's return value indicates whether any
URBs were given back; the idea was that the caller can scan the QH
over again to handle any URBs that were dequeued by a completion
handler. This is not necessary; when qh_completions() is ready to
give back dequeued URBs, it does its own rescanning.
Therefore the new return value will be a flag indicating whether the
caller needs to unlink the QH. This is more convenient than forcing
the caller to check qh->needs_rescan, and it makes a lot more sense --
why should "needs_rescan" imply that an unlink is needed? The callers
are also changed to remove the unneeded rescans.
Lastly, the check for whether qh->qtd_list is non-empty is removed
from the start of qh_completions(). Two of the callers have to make
this test anyway, so the same test can simply be added to the other
two callers.
Signed-off-by: Alan Stern <stern@rowland.harvard.edu>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Diffstat (limited to 'drivers/usb/host/ehci-sched.c')
-rw-r--r-- | drivers/usb/host/ehci-sched.c | 12 |
1 files changed, 5 insertions, 7 deletions
diff --git a/drivers/usb/host/ehci-sched.c b/drivers/usb/host/ehci-sched.c index 66259dc7822e..5c82bbab9a48 100644 --- a/drivers/usb/host/ehci-sched.c +++ b/drivers/usb/host/ehci-sched.c | |||
@@ -649,7 +649,8 @@ static void end_unlink_intr(struct ehci_hcd *ehci, struct ehci_qh *qh) | |||
649 | qh->qh_state = QH_STATE_IDLE; | 649 | qh->qh_state = QH_STATE_IDLE; |
650 | hw->hw_next = EHCI_LIST_END(ehci); | 650 | hw->hw_next = EHCI_LIST_END(ehci); |
651 | 651 | ||
652 | qh_completions(ehci, qh); | 652 | if (!list_empty(&qh->qtd_list)) |
653 | qh_completions(ehci, qh); | ||
653 | 654 | ||
654 | /* reschedule QH iff another request is queued */ | 655 | /* reschedule QH iff another request is queued */ |
655 | if (!list_empty(&qh->qtd_list) && ehci->rh_state == EHCI_RH_RUNNING) { | 656 | if (!list_empty(&qh->qtd_list) && ehci->rh_state == EHCI_RH_RUNNING) { |
@@ -914,7 +915,7 @@ static void scan_intr(struct ehci_hcd *ehci) | |||
914 | 915 | ||
915 | list_for_each_entry_safe(qh, ehci->qh_scan_next, &ehci->intr_qh_list, | 916 | list_for_each_entry_safe(qh, ehci->qh_scan_next, &ehci->intr_qh_list, |
916 | intr_node) { | 917 | intr_node) { |
917 | rescan: | 918 | |
918 | /* clean any finished work for this qh */ | 919 | /* clean any finished work for this qh */ |
919 | if (!list_empty(&qh->qtd_list)) { | 920 | if (!list_empty(&qh->qtd_list)) { |
920 | int temp; | 921 | int temp; |
@@ -927,12 +928,9 @@ static void scan_intr(struct ehci_hcd *ehci) | |||
927 | * in qh_unlink_periodic(). | 928 | * in qh_unlink_periodic(). |
928 | */ | 929 | */ |
929 | temp = qh_completions(ehci, qh); | 930 | temp = qh_completions(ehci, qh); |
930 | if (unlikely(qh->needs_rescan || | 931 | if (unlikely(temp || (list_empty(&qh->qtd_list) && |
931 | (list_empty(&qh->qtd_list) && | 932 | qh->qh_state == QH_STATE_LINKED))) |
932 | qh->qh_state == QH_STATE_LINKED))) | ||
933 | start_unlink_intr(ehci, qh); | 933 | start_unlink_intr(ehci, qh); |
934 | else if (temp != 0) | ||
935 | goto rescan; | ||
936 | } | 934 | } |
937 | } | 935 | } |
938 | } | 936 | } |