|
|
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 |
|