dcavalca / rpms / qemu

Forked from rpms/qemu a year ago
Clone

Blame 0353-xhci-kill-xhci_mem_-read-write-dispatcher-functions.patch

c8dfc6
From e5295db184d2b78a6a779aac019acbf58ed3da5e Mon Sep 17 00:00:00 2001
c8dfc6
From: Gerd Hoffmann <kraxel@redhat.com>
c8dfc6
Date: Tue, 4 Sep 2012 14:42:20 +0200
c8dfc6
Subject: [PATCH 353/366] xhci: kill xhci_mem_{read,write} dispatcher
c8dfc6
 functions
c8dfc6
c8dfc6
... and register subregions instead, so we offload the dispatching
c8dfc6
to the the memory subsystem which is designed to handle it.
c8dfc6
c8dfc6
Signed-off-by: Gerd Hoffmann <kraxel@redhat.com>
c8dfc6
---
c8dfc6
 hw/usb/hcd-xhci.c | 140 +++++++++++++++++++++++++++++-------------------------
c8dfc6
 1 file changed, 75 insertions(+), 65 deletions(-)
c8dfc6
c8dfc6
diff --git a/hw/usb/hcd-xhci.c b/hw/usb/hcd-xhci.c
c8dfc6
index 55e31ec..500892d 100644
c8dfc6
--- a/hw/usb/hcd-xhci.c
c8dfc6
+++ b/hw/usb/hcd-xhci.c
c8dfc6
@@ -404,6 +404,10 @@ struct XHCIState {
c8dfc6
     USBBus bus;
c8dfc6
     qemu_irq irq;
c8dfc6
     MemoryRegion mem;
c8dfc6
+    MemoryRegion mem_cap;
c8dfc6
+    MemoryRegion mem_oper;
c8dfc6
+    MemoryRegion mem_runtime;
c8dfc6
+    MemoryRegion mem_doorbell;
c8dfc6
     const char *name;
c8dfc6
     unsigned int devaddr;
c8dfc6
 
c8dfc6
@@ -2345,8 +2349,9 @@ static void xhci_reset(DeviceState *dev)
c8dfc6
     xhci_mfwrap_update(xhci);
c8dfc6
 }
c8dfc6
 
c8dfc6
-static uint32_t xhci_cap_read(XHCIState *xhci, uint32_t reg)
c8dfc6
+static uint64_t xhci_cap_read(void *ptr, target_phys_addr_t reg, unsigned size)
c8dfc6
 {
c8dfc6
+    XHCIState *xhci = ptr;
c8dfc6
     uint32_t ret;
c8dfc6
 
c8dfc6
     switch (reg) {
c8dfc6
@@ -2403,7 +2408,7 @@ static uint32_t xhci_cap_read(XHCIState *xhci, uint32_t reg)
c8dfc6
         ret = 0x00000000; /* reserved */
c8dfc6
         break;
c8dfc6
     default:
c8dfc6
-        fprintf(stderr, "xhci_cap_read: reg %d unimplemented\n", reg);
c8dfc6
+        fprintf(stderr, "xhci_cap_read: reg %d unimplemented\n", (int)reg);
c8dfc6
         ret = 0;
c8dfc6
     }
c8dfc6
 
c8dfc6
@@ -2484,8 +2489,9 @@ static void xhci_port_write(XHCIState *xhci, uint32_t reg, uint32_t val)
c8dfc6
     }
c8dfc6
 }
c8dfc6
 
c8dfc6
-static uint32_t xhci_oper_read(XHCIState *xhci, uint32_t reg)
c8dfc6
+static uint64_t xhci_oper_read(void *ptr, target_phys_addr_t reg, unsigned size)
c8dfc6
 {
c8dfc6
+    XHCIState *xhci = ptr;
c8dfc6
     uint32_t ret;
c8dfc6
 
c8dfc6
     if (reg >= 0x400) {
c8dfc6
@@ -2521,7 +2527,7 @@ static uint32_t xhci_oper_read(XHCIState *xhci, uint32_t reg)
c8dfc6
         ret = xhci->config;
c8dfc6
         break;
c8dfc6
     default:
c8dfc6
-        fprintf(stderr, "xhci_oper_read: reg 0x%x unimplemented\n", reg);
c8dfc6
+        fprintf(stderr, "xhci_oper_read: reg 0x%x unimplemented\n", (int)reg);
c8dfc6
         ret = 0;
c8dfc6
     }
c8dfc6
 
c8dfc6
@@ -2529,8 +2535,11 @@ static uint32_t xhci_oper_read(XHCIState *xhci, uint32_t reg)
c8dfc6
     return ret;
c8dfc6
 }
c8dfc6
 
c8dfc6
-static void xhci_oper_write(XHCIState *xhci, uint32_t reg, uint32_t val)
c8dfc6
+static void xhci_oper_write(void *ptr, target_phys_addr_t reg,
c8dfc6
+                            uint64_t val, unsigned size)
c8dfc6
 {
c8dfc6
+    XHCIState *xhci = ptr;
c8dfc6
+
c8dfc6
     if (reg >= 0x400) {
c8dfc6
         xhci_port_write(xhci, reg - 0x400, val);
c8dfc6
         return;
c8dfc6
@@ -2588,12 +2597,14 @@ static void xhci_oper_write(XHCIState *xhci, uint32_t reg, uint32_t val)
c8dfc6
         xhci->config = val & 0xff;
c8dfc6
         break;
c8dfc6
     default:
c8dfc6
-        fprintf(stderr, "xhci_oper_write: reg 0x%x unimplemented\n", reg);
c8dfc6
+        fprintf(stderr, "xhci_oper_write: reg 0x%x unimplemented\n", (int)reg);
c8dfc6
     }
c8dfc6
 }
c8dfc6
 
c8dfc6
-static uint32_t xhci_runtime_read(XHCIState *xhci, uint32_t reg)
c8dfc6
+static uint64_t xhci_runtime_read(void *ptr, target_phys_addr_t reg,
c8dfc6
+                                  unsigned size)
c8dfc6
 {
c8dfc6
+    XHCIState *xhci = ptr;
c8dfc6
     uint32_t ret = 0;
c8dfc6
 
c8dfc6
     if (reg < 0x20) {
c8dfc6
@@ -2602,7 +2613,8 @@ static uint32_t xhci_runtime_read(XHCIState *xhci, uint32_t reg)
c8dfc6
             ret = xhci_mfindex_get(xhci) & 0x3fff;
c8dfc6
             break;
c8dfc6
         default:
c8dfc6
-            fprintf(stderr, "xhci_runtime_read: reg 0x%x unimplemented\n", reg);
c8dfc6
+            fprintf(stderr, "xhci_runtime_read: reg 0x%x unimplemented\n",
c8dfc6
+                    (int)reg);
c8dfc6
             break;
c8dfc6
         }
c8dfc6
     } else {
c8dfc6
@@ -2637,14 +2649,16 @@ static uint32_t xhci_runtime_read(XHCIState *xhci, uint32_t reg)
c8dfc6
     return ret;
c8dfc6
 }
c8dfc6
 
c8dfc6
-static void xhci_runtime_write(XHCIState *xhci, uint32_t reg, uint32_t val)
c8dfc6
+static void xhci_runtime_write(void *ptr, target_phys_addr_t reg,
c8dfc6
+                               uint64_t val, unsigned size)
c8dfc6
 {
c8dfc6
+    XHCIState *xhci = ptr;
c8dfc6
     int v = (reg - 0x20) / 0x20;
c8dfc6
     XHCIInterrupter *intr = &xhci->intr[v];
c8dfc6
     trace_usb_xhci_runtime_write(reg, val);
c8dfc6
 
c8dfc6
     if (reg < 0x20) {
c8dfc6
-        fprintf(stderr, "xhci_oper_write: reg 0x%x unimplemented\n", reg);
c8dfc6
+        fprintf(stderr, "xhci_oper_write: reg 0x%x unimplemented\n", (int)reg);
c8dfc6
         return;
c8dfc6
     }
c8dfc6
 
c8dfc6
@@ -2686,19 +2700,24 @@ static void xhci_runtime_write(XHCIState *xhci, uint32_t reg, uint32_t val)
c8dfc6
         xhci_events_update(xhci, v);
c8dfc6
         break;
c8dfc6
     default:
c8dfc6
-        fprintf(stderr, "xhci_oper_write: reg 0x%x unimplemented\n", reg);
c8dfc6
+        fprintf(stderr, "xhci_oper_write: reg 0x%x unimplemented\n",
c8dfc6
+                (int)reg);
c8dfc6
     }
c8dfc6
 }
c8dfc6
 
c8dfc6
-static uint32_t xhci_doorbell_read(XHCIState *xhci, uint32_t reg)
c8dfc6
+static uint64_t xhci_doorbell_read(void *ptr, target_phys_addr_t reg,
c8dfc6
+                                   unsigned size)
c8dfc6
 {
c8dfc6
     /* doorbells always read as 0 */
c8dfc6
     trace_usb_xhci_doorbell_read(reg, 0);
c8dfc6
     return 0;
c8dfc6
 }
c8dfc6
 
c8dfc6
-static void xhci_doorbell_write(XHCIState *xhci, uint32_t reg, uint32_t val)
c8dfc6
+static void xhci_doorbell_write(void *ptr, target_phys_addr_t reg,
c8dfc6
+                                uint64_t val, unsigned size)
c8dfc6
 {
c8dfc6
+    XHCIState *xhci = ptr;
c8dfc6
+
c8dfc6
     trace_usb_xhci_doorbell_write(reg, val);
c8dfc6
 
c8dfc6
     if (!xhci_running(xhci)) {
c8dfc6
@@ -2712,69 +2731,47 @@ static void xhci_doorbell_write(XHCIState *xhci, uint32_t reg, uint32_t val)
c8dfc6
         if (val == 0) {
c8dfc6
             xhci_process_commands(xhci);
c8dfc6
         } else {
c8dfc6
-            fprintf(stderr, "xhci: bad doorbell 0 write: 0x%x\n", val);
c8dfc6
+            fprintf(stderr, "xhci: bad doorbell 0 write: 0x%x\n",
c8dfc6
+                    (uint32_t)val);
c8dfc6
         }
c8dfc6
     } else {
c8dfc6
         if (reg > MAXSLOTS) {
c8dfc6
-            fprintf(stderr, "xhci: bad doorbell %d\n", reg);
c8dfc6
+            fprintf(stderr, "xhci: bad doorbell %d\n", (int)reg);
c8dfc6
         } else if (val > 31) {
c8dfc6
-            fprintf(stderr, "xhci: bad doorbell %d write: 0x%x\n", reg, val);
c8dfc6
+            fprintf(stderr, "xhci: bad doorbell %d write: 0x%x\n",
c8dfc6
+                    (int)reg, (uint32_t)val);
c8dfc6
         } else {
c8dfc6
             xhci_kick_ep(xhci, reg, val);
c8dfc6
         }
c8dfc6
     }
c8dfc6
 }
c8dfc6
 
c8dfc6
-static uint64_t xhci_mem_read(void *ptr, target_phys_addr_t addr,
c8dfc6
-                              unsigned size)
c8dfc6
-{
c8dfc6
-    XHCIState *xhci = ptr;
c8dfc6
-
c8dfc6
-    /* Only aligned reads are allowed on xHCI */
c8dfc6
-    if (addr & 3) {
c8dfc6
-        fprintf(stderr, "xhci_mem_read: Mis-aligned read\n");
c8dfc6
-        return 0;
c8dfc6
-    }
c8dfc6
-
c8dfc6
-    if (addr < LEN_CAP) {
c8dfc6
-        return xhci_cap_read(xhci, addr);
c8dfc6
-    } else if (addr >= OFF_OPER && addr < (OFF_OPER + LEN_OPER)) {
c8dfc6
-        return xhci_oper_read(xhci, addr - OFF_OPER);
c8dfc6
-    } else if (addr >= OFF_RUNTIME && addr < (OFF_RUNTIME + LEN_RUNTIME)) {
c8dfc6
-        return xhci_runtime_read(xhci, addr - OFF_RUNTIME);
c8dfc6
-    } else if (addr >= OFF_DOORBELL && addr < (OFF_DOORBELL + LEN_DOORBELL)) {
c8dfc6
-        return xhci_doorbell_read(xhci, addr - OFF_DOORBELL);
c8dfc6
-    } else {
c8dfc6
-        fprintf(stderr, "xhci_mem_read: Bad offset %x\n", (int)addr);
c8dfc6
-        return 0;
c8dfc6
-    }
c8dfc6
-}
c8dfc6
-
c8dfc6
-static void xhci_mem_write(void *ptr, target_phys_addr_t addr,
c8dfc6
-                           uint64_t val, unsigned size)
c8dfc6
-{
c8dfc6
-    XHCIState *xhci = ptr;
c8dfc6
+static const MemoryRegionOps xhci_cap_ops = {
c8dfc6
+    .read = xhci_cap_read,
c8dfc6
+    .valid.min_access_size = 4,
c8dfc6
+    .valid.max_access_size = 4,
c8dfc6
+    .endianness = DEVICE_LITTLE_ENDIAN,
c8dfc6
+};
c8dfc6
 
c8dfc6
-    /* Only aligned writes are allowed on xHCI */
c8dfc6
-    if (addr & 3) {
c8dfc6
-        fprintf(stderr, "xhci_mem_write: Mis-aligned write\n");
c8dfc6
-        return;
c8dfc6
-    }
c8dfc6
+static const MemoryRegionOps xhci_oper_ops = {
c8dfc6
+    .read = xhci_oper_read,
c8dfc6
+    .write = xhci_oper_write,
c8dfc6
+    .valid.min_access_size = 4,
c8dfc6
+    .valid.max_access_size = 4,
c8dfc6
+    .endianness = DEVICE_LITTLE_ENDIAN,
c8dfc6
+};
c8dfc6
 
c8dfc6
-    if (addr >= OFF_OPER && addr < (OFF_OPER + LEN_OPER)) {
c8dfc6
-        xhci_oper_write(xhci, addr - OFF_OPER, val);
c8dfc6
-    } else if (addr >= OFF_RUNTIME && addr < (OFF_RUNTIME + LEN_RUNTIME)) {
c8dfc6
-        xhci_runtime_write(xhci, addr - OFF_RUNTIME, val);
c8dfc6
-    } else if (addr >= OFF_DOORBELL && addr < (OFF_DOORBELL + LEN_DOORBELL)) {
c8dfc6
-        xhci_doorbell_write(xhci, addr - OFF_DOORBELL, val);
c8dfc6
-    } else {
c8dfc6
-        fprintf(stderr, "xhci_mem_write: Bad offset %x\n", (int)addr);
c8dfc6
-    }
c8dfc6
-}
c8dfc6
+static const MemoryRegionOps xhci_runtime_ops = {
c8dfc6
+    .read = xhci_runtime_read,
c8dfc6
+    .write = xhci_runtime_write,
c8dfc6
+    .valid.min_access_size = 4,
c8dfc6
+    .valid.max_access_size = 4,
c8dfc6
+    .endianness = DEVICE_LITTLE_ENDIAN,
c8dfc6
+};
c8dfc6
 
c8dfc6
-static const MemoryRegionOps xhci_mem_ops = {
c8dfc6
-    .read = xhci_mem_read,
c8dfc6
-    .write = xhci_mem_write,
c8dfc6
+static const MemoryRegionOps xhci_doorbell_ops = {
c8dfc6
+    .read = xhci_doorbell_read,
c8dfc6
+    .write = xhci_doorbell_write,
c8dfc6
     .valid.min_access_size = 4,
c8dfc6
     .valid.max_access_size = 4,
c8dfc6
     .endianness = DEVICE_LITTLE_ENDIAN,
c8dfc6
@@ -2940,8 +2937,21 @@ static int usb_xhci_initfn(struct PCIDevice *dev)
c8dfc6
 
c8dfc6
     xhci->irq = xhci->pci_dev.irq[0];
c8dfc6
 
c8dfc6
-    memory_region_init_io(&xhci->mem, &xhci_mem_ops, xhci,
c8dfc6
-                          "xhci", LEN_REGS);
c8dfc6
+    memory_region_init(&xhci->mem, "xhci", LEN_REGS);
c8dfc6
+    memory_region_init_io(&xhci->mem_cap, &xhci_cap_ops, xhci,
c8dfc6
+                          "capabilities", LEN_CAP);
c8dfc6
+    memory_region_init_io(&xhci->mem_oper, &xhci_oper_ops, xhci,
c8dfc6
+                          "operational", 0x400 + 0x10 * xhci->numports);
c8dfc6
+    memory_region_init_io(&xhci->mem_runtime, &xhci_runtime_ops, xhci,
c8dfc6
+                          "runtime", LEN_RUNTIME);
c8dfc6
+    memory_region_init_io(&xhci->mem_doorbell, &xhci_doorbell_ops, xhci,
c8dfc6
+                          "doorbell", LEN_DOORBELL);
c8dfc6
+
c8dfc6
+    memory_region_add_subregion(&xhci->mem, 0,            &xhci->mem_cap);
c8dfc6
+    memory_region_add_subregion(&xhci->mem, OFF_OPER,     &xhci->mem_oper);
c8dfc6
+    memory_region_add_subregion(&xhci->mem, OFF_RUNTIME,  &xhci->mem_runtime);
c8dfc6
+    memory_region_add_subregion(&xhci->mem, OFF_DOORBELL, &xhci->mem_doorbell);
c8dfc6
+
c8dfc6
     pci_register_bar(&xhci->pci_dev, 0,
c8dfc6
                      PCI_BASE_ADDRESS_SPACE_MEMORY|PCI_BASE_ADDRESS_MEM_TYPE_64,
c8dfc6
                      &xhci->mem);
c8dfc6
-- 
c8dfc6
1.7.12
c8dfc6