vfio-pci: Move devices to D0 on reset

Guests may leave devices in a low power state at reboot, but we expect
devices to be woken up for the next boot.  Make this happen.

Signed-off-by: Alex Williamson <alex.williamson@redhat.com>
This commit is contained in:
Alex Williamson 2013-04-01 13:35:08 -06:00
parent 82ca891283
commit ba66181828

View File

@ -160,6 +160,7 @@ typedef struct VFIODevice {
uint32_t features; uint32_t features;
#define VFIO_FEATURE_ENABLE_VGA_BIT 0 #define VFIO_FEATURE_ENABLE_VGA_BIT 0
#define VFIO_FEATURE_ENABLE_VGA (1 << VFIO_FEATURE_ENABLE_VGA_BIT) #define VFIO_FEATURE_ENABLE_VGA (1 << VFIO_FEATURE_ENABLE_VGA_BIT)
uint8_t pm_cap;
bool reset_works; bool reset_works;
bool has_vga; bool has_vga;
} VFIODevice; } VFIODevice;
@ -2534,6 +2535,8 @@ static int vfio_add_std_cap(VFIODevice *vdev, uint8_t pos)
case PCI_CAP_ID_MSIX: case PCI_CAP_ID_MSIX:
ret = vfio_setup_msix(vdev, pos); ret = vfio_setup_msix(vdev, pos);
break; break;
case PCI_CAP_ID_PM:
vdev->pm_cap = pos;
default: default:
ret = pci_add_capability(pdev, cap_id, pos, size); ret = pci_add_capability(pdev, cap_id, pos, size);
break; break;
@ -3108,6 +3111,26 @@ static void vfio_pci_reset(DeviceState *dev)
vfio_disable_interrupts(vdev); vfio_disable_interrupts(vdev);
/* Make sure the device is in D0 */
if (vdev->pm_cap) {
uint16_t pmcsr;
uint8_t state;
pmcsr = vfio_pci_read_config(pdev, vdev->pm_cap + PCI_PM_CTRL, 2);
state = pmcsr & PCI_PM_CTRL_STATE_MASK;
if (state) {
pmcsr &= ~PCI_PM_CTRL_STATE_MASK;
vfio_pci_write_config(pdev, vdev->pm_cap + PCI_PM_CTRL, pmcsr, 2);
/* vfio handles the necessary delay here */
pmcsr = vfio_pci_read_config(pdev, vdev->pm_cap + PCI_PM_CTRL, 2);
state = pmcsr & PCI_PM_CTRL_STATE_MASK;
if (state) {
error_report("vfio: Unable to power on device, stuck in D%d\n",
state);
}
}
}
/* /*
* Stop any ongoing DMA by disconecting I/O, MMIO, and bus master. * Stop any ongoing DMA by disconecting I/O, MMIO, and bus master.
* Also put INTx Disable in known state. * Also put INTx Disable in known state.