dcavalca / rpms / qemu

Forked from rpms/qemu a year ago
Clone

Blame 0135-usb-ehci-Fix-and-simplify-nakcnt-handling.patch

1b1995
From 2d9b6cb9bd00ede47635dc4db413f647143d5a1d Mon Sep 17 00:00:00 2001
1b1995
From: Hans de Goede <hdegoede@redhat.com>
1b1995
Date: Thu, 1 Mar 2012 23:55:11 +0100
1b1995
Subject: [PATCH 135/140] usb-ehci: Fix and simplify nakcnt handling
1b1995
1b1995
The nakcnt code in ehci_execute_complete() marked transactions as finished
1b1995
when a packet completed with a result of USB_RET_NAK, but USB_RET_NAK
1b1995
means that the device cannot receive / send data at that time and that
1b1995
the transaction should be retried later, which is also what the usb-uhci
1b1995
and usb-ohci code does.
1b1995
1b1995
Note that there already was some special code in place to handle this
1b1995
for interrupt endpoints in the form of doing a return from
1b1995
ehci_execute_complete() when reload == 0, but that for bulk transactions
1b1995
this was not handled correctly (where as for example the usb-ccid device does
1b1995
return USB_RET_NAK for bulk packets).
1b1995
1b1995
Besides that the code in ehci_execute_complete() decrement nakcnt by 1
1b1995
on a packet result of USB_RET_NAK, but
1b1995
-since the transaction got marked as finished,
1b1995
 nakcnt would never be decremented again
1b1995
-there is no code checking for nakcnt becoming 0
1b1995
-there is no use in re-trying the transaction within the same usb frame /
1b1995
 usb-ehci frame-timer call, since the status of emulated devices won't change
1b1995
 as long as the usb-ehci frame-timer is running
1b1995
So we should simply set the nakcnt to 0 when we get a USB_RET_NAK, thus
1b1995
claiming that we've tried reload times (or as many times as possible if
1b1995
reload is 0).
1b1995
1b1995
Besides the code in ehci_execute_complete() handling USB_RET_NAK there
1b1995
was also code handling it in ehci_state_executing(), which calls
1b1995
ehci_execute_complete(), and then does its own handling on top of the handling
1b1995
in ehci_execute_complete(), this code would decrement nakcnt *again* (if not
1b1995
already 0), or restore the reload value (which was never changed) on success.
1b1995
1b1995
Since the double decrement was wrong to begin with, and is no longer needed
1b1995
now that we set nakcnt directly to 0 on USB_RET_NAK, and the restore of reload
1b1995
is not needed either, this patch simply removes all nakcnt handling from
1b1995
ehci_state_executing().
1b1995
1b1995
Signed-off-by: Hans de Goede <hdegoede@redhat.com>
1b1995
---
1b1995
 hw/usb-ehci.c |   32 ++++----------------------------
1b1995
 1 file changed, 4 insertions(+), 28 deletions(-)
1b1995
1b1995
diff --git a/hw/usb-ehci.c b/hw/usb-ehci.c
1b1995
index 92cdf2a..aa6fae5 100644
1b1995
--- a/hw/usb-ehci.c
1b1995
+++ b/hw/usb-ehci.c
1b1995
@@ -1269,8 +1269,6 @@ static void ehci_async_complete_packet(USBPort *port, USBPacket *packet)
1b1995
 
1b1995
 static void ehci_execute_complete(EHCIQueue *q)
1b1995
 {
1b1995
-    int reload;
1b1995
-
1b1995
     assert(q->async != EHCI_ASYNC_INFLIGHT);
1b1995
     q->async = EHCI_ASYNC_NONE;
1b1995
 
1b1995
@@ -1289,16 +1287,8 @@ static void ehci_execute_complete(EHCIQueue *q)
1b1995
             ehci_record_interrupt(q->ehci, USBSTS_ERRINT);
1b1995
             break;
1b1995
         case USB_RET_NAK:
1b1995
-            /* 4.10.3 */
1b1995
-            reload = get_field(q->qh.epchar, QH_EPCHAR_RL);
1b1995
-            if ((q->pid == USB_TOKEN_IN) && reload) {
1b1995
-                int nakcnt = get_field(q->qh.altnext_qtd, QH_ALTNEXT_NAKCNT);
1b1995
-                nakcnt--;
1b1995
-                set_field(&q->qh.altnext_qtd, nakcnt, QH_ALTNEXT_NAKCNT);
1b1995
-            } else if (!reload) {
1b1995
-                return;
1b1995
-            }
1b1995
-            break;
1b1995
+            set_field(&q->qh.altnext_qtd, 0, QH_ALTNEXT_NAKCNT);
1b1995
+            return; /* We're not done yet with this transaction */
1b1995
         case USB_RET_BABBLE:
1b1995
             q->qh.token |= (QTD_TOKEN_HALT | QTD_TOKEN_BABBLE);
1b1995
             ehci_record_interrupt(q->ehci, USBSTS_ERRINT);
1b1995
@@ -1331,7 +1321,7 @@ static void ehci_execute_complete(EHCIQueue *q)
1b1995
     q->qh.token ^= QTD_TOKEN_DTOGGLE;
1b1995
     q->qh.token &= ~QTD_TOKEN_ACTIVE;
1b1995
 
1b1995
-    if ((q->usb_status != USB_RET_NAK) && (q->qh.token & QTD_TOKEN_IOC)) {
1b1995
+    if (q->qh.token & QTD_TOKEN_IOC) {
1b1995
         ehci_record_interrupt(q->ehci, USBSTS_INT);
1b1995
     }
1b1995
 }
1b1995
@@ -1905,7 +1895,6 @@ out:
1b1995
 static int ehci_state_executing(EHCIQueue *q, int async)
1b1995
 {
1b1995
     int again = 0;
1b1995
-    int reload, nakcnt;
1b1995
 
1b1995
     ehci_execute_complete(q);
1b1995
     if (q->usb_status == USB_RET_ASYNC) {
1b1995
@@ -1925,21 +1914,8 @@ static int ehci_state_executing(EHCIQueue *q, int async)
1b1995
         // counter decrements to 0
1b1995
     }
1b1995
 
1b1995
-    reload = get_field(q->qh.epchar, QH_EPCHAR_RL);
1b1995
-    if (reload) {
1b1995
-        nakcnt = get_field(q->qh.altnext_qtd, QH_ALTNEXT_NAKCNT);
1b1995
-        if (q->usb_status == USB_RET_NAK) {
1b1995
-            if (nakcnt) {
1b1995
-                nakcnt--;
1b1995
-            }
1b1995
-        } else {
1b1995
-            nakcnt = reload;
1b1995
-        }
1b1995
-        set_field(&q->qh.altnext_qtd, nakcnt, QH_ALTNEXT_NAKCNT);
1b1995
-    }
1b1995
-
1b1995
     /* 4.10.5 */
1b1995
-    if ((q->usb_status == USB_RET_NAK) || (q->qh.token & QTD_TOKEN_ACTIVE)) {
1b1995
+    if (q->usb_status == USB_RET_NAK) {
1b1995
         ehci_set_state(q->ehci, async, EST_HORIZONTALQH);
1b1995
     } else {
1b1995
         ehci_set_state(q->ehci, async, EST_WRITEBACK);
1b1995
-- 
1b1995
1.7.9.3
1b1995