|
|
c8dfc6 |
From 584606864261092041b46806030e4889b747ad41 Mon Sep 17 00:00:00 2001
|
|
|
c8dfc6 |
From: Hans de Goede <hdegoede@redhat.com>
|
|
|
c8dfc6 |
Date: Thu, 30 Aug 2012 15:18:24 +0200
|
|
|
c8dfc6 |
Subject: [PATCH 309/366] ehci: Properly report completed but not yet
|
|
|
c8dfc6 |
processed packets to the guest
|
|
|
c8dfc6 |
|
|
|
c8dfc6 |
Reported packets which have completed before being cancelled as such to the
|
|
|
c8dfc6 |
host. Note that the new code path this patch adds is untested since it I've
|
|
|
c8dfc6 |
been unable to actually trigger the race which needs this code path.
|
|
|
c8dfc6 |
|
|
|
c8dfc6 |
Signed-off-by: Hans de Goede <hdegoede@redhat.com>
|
|
|
c8dfc6 |
---
|
|
|
c8dfc6 |
hw/usb/hcd-ehci.c | 13 +++++++++++++
|
|
|
c8dfc6 |
1 file changed, 13 insertions(+)
|
|
|
c8dfc6 |
|
|
|
c8dfc6 |
diff --git a/hw/usb/hcd-ehci.c b/hw/usb/hcd-ehci.c
|
|
|
c8dfc6 |
index 4fe85c8..0a6c9ef 100644
|
|
|
c8dfc6 |
--- a/hw/usb/hcd-ehci.c
|
|
|
c8dfc6 |
+++ b/hw/usb/hcd-ehci.c
|
|
|
c8dfc6 |
@@ -489,6 +489,9 @@ static const char *ehci_mmio_names[] = {
|
|
|
c8dfc6 |
[CONFIGFLAG] = "CONFIGFLAG",
|
|
|
c8dfc6 |
};
|
|
|
c8dfc6 |
|
|
|
c8dfc6 |
+static int ehci_state_executing(EHCIQueue *q);
|
|
|
c8dfc6 |
+static int ehci_state_writeback(EHCIQueue *q);
|
|
|
c8dfc6 |
+
|
|
|
c8dfc6 |
static const char *nr2str(const char **n, size_t len, uint32_t nr)
|
|
|
c8dfc6 |
{
|
|
|
c8dfc6 |
if (nr < len && n[nr] != NULL) {
|
|
|
c8dfc6 |
@@ -750,6 +753,16 @@ static void ehci_free_packet(EHCIPacket *p)
|
|
|
c8dfc6 |
usb_packet_unmap(&p->packet, &p->sgl);
|
|
|
c8dfc6 |
qemu_sglist_destroy(&p->sgl);
|
|
|
c8dfc6 |
}
|
|
|
c8dfc6 |
+ if (p->async == EHCI_ASYNC_FINISHED) {
|
|
|
c8dfc6 |
+ int state = ehci_get_state(p->queue->ehci, p->queue->async);
|
|
|
c8dfc6 |
+ /* This is a normal, but rare condition (cancel racing completion) */
|
|
|
c8dfc6 |
+ fprintf(stderr, "EHCI: Warning packet completed but not processed\n");
|
|
|
c8dfc6 |
+ ehci_state_executing(p->queue);
|
|
|
c8dfc6 |
+ ehci_state_writeback(p->queue);
|
|
|
c8dfc6 |
+ ehci_set_state(p->queue->ehci, p->queue->async, state);
|
|
|
c8dfc6 |
+ /* state_writeback recurses into us with async == EHCI_ASYNC_NONE!! */
|
|
|
c8dfc6 |
+ return;
|
|
|
c8dfc6 |
+ }
|
|
|
c8dfc6 |
QTAILQ_REMOVE(&p->queue->packets, p, next);
|
|
|
c8dfc6 |
usb_packet_cleanup(&p->packet);
|
|
|
c8dfc6 |
g_free(p);
|
|
|
c8dfc6 |
--
|
|
|
c8dfc6 |
1.7.12
|
|
|
c8dfc6 |
|