peterdelevoryas / rpms / qemu

Forked from rpms/qemu 2 years ago
Clone

Blame 0309-ehci-Properly-report-completed-but-not-yet-processed.patch

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