提交 d93ae5b6 编写于 作者: P Peter Maydell

Merge remote-tracking branch 'remotes/awilliam/tags/vfio-update-20151110.0' into staging

VFIO updates 2015-11-10

 - Make Windows happy with vfio-pci devices exposed on conventional
   PCI buses on q35 by hiding PCIe capability (Alex Williamson)
 - Convert to g_new() where appropriate (Markus Armbruster)

# gpg: Signature made Tue 10 Nov 2015 19:46:41 GMT using RSA key ID 3BB08B22
# gpg: Good signature from "Alex Williamson <alex.williamson@redhat.com>"
# gpg:                 aka "Alex Williamson <alex@shazbot.org>"
# gpg:                 aka "Alex Williamson <alwillia@redhat.com>"
# gpg:                 aka "Alex Williamson <alex.l.williamson@gmail.com>"

* remotes/awilliam/tags/vfio-update-20151110.0:
  vfio: Use g_new() & friends where that makes obvious sense
  vfio/pci: Hide device PCIe capability on non-express buses for PCIe VMs
Signed-off-by: NPeter Maydell <peter.maydell@linaro.org>
...@@ -284,7 +284,7 @@ static void vfio_vga_probe_ati_3c3_quirk(VFIOPCIDevice *vdev) ...@@ -284,7 +284,7 @@ static void vfio_vga_probe_ati_3c3_quirk(VFIOPCIDevice *vdev)
} }
quirk = g_malloc0(sizeof(*quirk)); quirk = g_malloc0(sizeof(*quirk));
quirk->mem = g_malloc0(sizeof(MemoryRegion)); quirk->mem = g_new0(MemoryRegion, 1);
quirk->nr_mem = 1; quirk->nr_mem = 1;
memory_region_init_io(quirk->mem, OBJECT(vdev), &vfio_ati_3c3_quirk, vdev, memory_region_init_io(quirk->mem, OBJECT(vdev), &vfio_ati_3c3_quirk, vdev,
...@@ -319,7 +319,7 @@ static void vfio_probe_ati_bar4_quirk(VFIOPCIDevice *vdev, int nr) ...@@ -319,7 +319,7 @@ static void vfio_probe_ati_bar4_quirk(VFIOPCIDevice *vdev, int nr)
} }
quirk = g_malloc0(sizeof(*quirk)); quirk = g_malloc0(sizeof(*quirk));
quirk->mem = g_malloc0(sizeof(MemoryRegion) * 2); quirk->mem = g_new0(MemoryRegion, 2);
quirk->nr_mem = 2; quirk->nr_mem = 2;
window = quirk->data = g_malloc0(sizeof(*window) + window = quirk->data = g_malloc0(sizeof(*window) +
sizeof(VFIOConfigWindowMatch)); sizeof(VFIOConfigWindowMatch));
...@@ -368,7 +368,7 @@ static void vfio_probe_ati_bar2_quirk(VFIOPCIDevice *vdev, int nr) ...@@ -368,7 +368,7 @@ static void vfio_probe_ati_bar2_quirk(VFIOPCIDevice *vdev, int nr)
quirk = g_malloc0(sizeof(*quirk)); quirk = g_malloc0(sizeof(*quirk));
mirror = quirk->data = g_malloc0(sizeof(*mirror)); mirror = quirk->data = g_malloc0(sizeof(*mirror));
mirror->mem = quirk->mem = g_malloc0(sizeof(MemoryRegion)); mirror->mem = quirk->mem = g_new0(MemoryRegion, 1);
quirk->nr_mem = 1; quirk->nr_mem = 1;
mirror->vdev = vdev; mirror->vdev = vdev;
mirror->offset = 0x4000; mirror->offset = 0x4000;
...@@ -544,7 +544,7 @@ static void vfio_vga_probe_nvidia_3d0_quirk(VFIOPCIDevice *vdev) ...@@ -544,7 +544,7 @@ static void vfio_vga_probe_nvidia_3d0_quirk(VFIOPCIDevice *vdev)
quirk = g_malloc0(sizeof(*quirk)); quirk = g_malloc0(sizeof(*quirk));
quirk->data = data = g_malloc0(sizeof(*data)); quirk->data = data = g_malloc0(sizeof(*data));
quirk->mem = g_malloc0(sizeof(MemoryRegion) * 2); quirk->mem = g_new0(MemoryRegion, 2);
quirk->nr_mem = 2; quirk->nr_mem = 2;
data->vdev = vdev; data->vdev = vdev;
...@@ -661,7 +661,7 @@ static void vfio_probe_nvidia_bar5_quirk(VFIOPCIDevice *vdev, int nr) ...@@ -661,7 +661,7 @@ static void vfio_probe_nvidia_bar5_quirk(VFIOPCIDevice *vdev, int nr)
} }
quirk = g_malloc0(sizeof(*quirk)); quirk = g_malloc0(sizeof(*quirk));
quirk->mem = g_malloc0(sizeof(MemoryRegion) * 4); quirk->mem = g_new0(MemoryRegion, 4);
quirk->nr_mem = 4; quirk->nr_mem = 4;
bar5 = quirk->data = g_malloc0(sizeof(*bar5) + bar5 = quirk->data = g_malloc0(sizeof(*bar5) +
(sizeof(VFIOConfigWindowMatch) * 2)); (sizeof(VFIOConfigWindowMatch) * 2));
...@@ -756,7 +756,7 @@ static void vfio_probe_nvidia_bar0_quirk(VFIOPCIDevice *vdev, int nr) ...@@ -756,7 +756,7 @@ static void vfio_probe_nvidia_bar0_quirk(VFIOPCIDevice *vdev, int nr)
quirk = g_malloc0(sizeof(*quirk)); quirk = g_malloc0(sizeof(*quirk));
mirror = quirk->data = g_malloc0(sizeof(*mirror)); mirror = quirk->data = g_malloc0(sizeof(*mirror));
mirror->mem = quirk->mem = g_malloc0(sizeof(MemoryRegion)); mirror->mem = quirk->mem = g_new0(MemoryRegion, 1);
quirk->nr_mem = 1; quirk->nr_mem = 1;
mirror->vdev = vdev; mirror->vdev = vdev;
mirror->offset = 0x88000; mirror->offset = 0x88000;
...@@ -775,7 +775,7 @@ static void vfio_probe_nvidia_bar0_quirk(VFIOPCIDevice *vdev, int nr) ...@@ -775,7 +775,7 @@ static void vfio_probe_nvidia_bar0_quirk(VFIOPCIDevice *vdev, int nr)
if (vdev->has_vga) { if (vdev->has_vga) {
quirk = g_malloc0(sizeof(*quirk)); quirk = g_malloc0(sizeof(*quirk));
mirror = quirk->data = g_malloc0(sizeof(*mirror)); mirror = quirk->data = g_malloc0(sizeof(*mirror));
mirror->mem = quirk->mem = g_malloc0(sizeof(MemoryRegion)); mirror->mem = quirk->mem = g_new0(MemoryRegion, 1);
quirk->nr_mem = 1; quirk->nr_mem = 1;
mirror->vdev = vdev; mirror->vdev = vdev;
mirror->offset = 0x1800; mirror->offset = 0x1800;
...@@ -938,7 +938,7 @@ static void vfio_probe_rtl8168_bar2_quirk(VFIOPCIDevice *vdev, int nr) ...@@ -938,7 +938,7 @@ static void vfio_probe_rtl8168_bar2_quirk(VFIOPCIDevice *vdev, int nr)
} }
quirk = g_malloc0(sizeof(*quirk)); quirk = g_malloc0(sizeof(*quirk));
quirk->mem = g_malloc0(sizeof(MemoryRegion) * 2); quirk->mem = g_new0(MemoryRegion, 2);
quirk->nr_mem = 2; quirk->nr_mem = 2;
quirk->data = rtl = g_malloc0(sizeof(*rtl)); quirk->data = rtl = g_malloc0(sizeof(*rtl));
rtl->vdev = vdev; rtl->vdev = vdev;
......
...@@ -28,6 +28,7 @@ ...@@ -28,6 +28,7 @@
#include "config.h" #include "config.h"
#include "hw/pci/msi.h" #include "hw/pci/msi.h"
#include "hw/pci/msix.h" #include "hw/pci/msix.h"
#include "hw/pci/pci_bridge.h"
#include "qemu/error-report.h" #include "qemu/error-report.h"
#include "qemu/range.h" #include "qemu/range.h"
#include "sysemu/kvm.h" #include "sysemu/kvm.h"
...@@ -586,7 +587,7 @@ static void vfio_msix_enable(VFIOPCIDevice *vdev) ...@@ -586,7 +587,7 @@ static void vfio_msix_enable(VFIOPCIDevice *vdev)
{ {
vfio_disable_interrupts(vdev); vfio_disable_interrupts(vdev);
vdev->msi_vectors = g_malloc0(vdev->msix->entries * sizeof(VFIOMSIVector)); vdev->msi_vectors = g_new0(VFIOMSIVector, vdev->msix->entries);
vdev->interrupt = VFIO_INT_MSIX; vdev->interrupt = VFIO_INT_MSIX;
...@@ -622,7 +623,7 @@ static void vfio_msi_enable(VFIOPCIDevice *vdev) ...@@ -622,7 +623,7 @@ static void vfio_msi_enable(VFIOPCIDevice *vdev)
vdev->nr_vectors = msi_nr_vectors_allocated(&vdev->pdev); vdev->nr_vectors = msi_nr_vectors_allocated(&vdev->pdev);
retry: retry:
vdev->msi_vectors = g_malloc0(vdev->nr_vectors * sizeof(VFIOMSIVector)); vdev->msi_vectors = g_new0(VFIOMSIVector, vdev->nr_vectors);
for (i = 0; i < vdev->nr_vectors; i++) { for (i = 0; i < vdev->nr_vectors; i++) {
VFIOMSIVector *vector = &vdev->msi_vectors[i]; VFIOMSIVector *vector = &vdev->msi_vectors[i];
...@@ -1524,13 +1525,38 @@ static int vfio_setup_pcie_cap(VFIOPCIDevice *vdev, int pos, uint8_t size) ...@@ -1524,13 +1525,38 @@ static int vfio_setup_pcie_cap(VFIOPCIDevice *vdev, int pos, uint8_t size)
} }
if (!pci_bus_is_express(vdev->pdev.bus)) { if (!pci_bus_is_express(vdev->pdev.bus)) {
PCIBus *bus = vdev->pdev.bus;
PCIDevice *bridge;
/* /*
* Use express capability as-is on PCI bus. It doesn't make much * Traditionally PCI device assignment exposes the PCIe capability
* sense to even expose, but some drivers (ex. tg3) depend on it * as-is on non-express buses. The reason being that some drivers
* and guests don't seem to be particular about it. We'll need * simply assume that it's there, for example tg3. However when
* to revist this or force express devices to express buses if we * we're running on a native PCIe machine type, like Q35, we need
* ever expose an IOMMU to the guest. * to hide the PCIe capability. The reason for this is twofold;
* first Windows guests get a Code 10 error when the PCIe capability
* is exposed in this configuration. Therefore express devices won't
* work at all unless they're attached to express buses in the VM.
* Second, a native PCIe machine introduces the possibility of fine
* granularity IOMMUs supporting both translation and isolation.
* Guest code to discover the IOMMU visibility of a device, such as
* IOMMU grouping code on Linux, is very aware of device types and
* valid transitions between bus types. An express device on a non-
* express bus is not a valid combination on bare metal systems.
*
* Drivers that require a PCIe capability to make the device
* functional are simply going to need to have their devices placed
* on a PCIe bus in the VM.
*/ */
while (!pci_bus_is_root(bus)) {
bridge = pci_bridge_get_device(bus);
bus = bridge->bus;
}
if (pci_bus_is_express(bus)) {
return 0;
}
} else if (pci_bus_is_root(vdev->pdev.bus)) { } else if (pci_bus_is_root(vdev->pdev.bus)) {
/* /*
* On a Root Complex bus Endpoints become Root Complex Integrated * On a Root Complex bus Endpoints become Root Complex Integrated
......
...@@ -478,7 +478,7 @@ static int vfio_populate_device(VFIODevice *vbasedev) ...@@ -478,7 +478,7 @@ static int vfio_populate_device(VFIODevice *vbasedev)
struct vfio_region_info reg_info = { .argsz = sizeof(reg_info) }; struct vfio_region_info reg_info = { .argsz = sizeof(reg_info) };
VFIORegion *ptr; VFIORegion *ptr;
vdev->regions[i] = g_malloc0(sizeof(VFIORegion)); vdev->regions[i] = g_new0(VFIORegion, 1);
ptr = vdev->regions[i]; ptr = vdev->regions[i];
reg_info.index = i; reg_info.index = i;
ret = ioctl(vbasedev->fd, VFIO_DEVICE_GET_REGION_INFO, &reg_info); ret = ioctl(vbasedev->fd, VFIO_DEVICE_GET_REGION_INFO, &reg_info);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册