mirror of
https://github.com/Motorhead1991/qemu.git
synced 2025-08-11 03:24:58 -06:00
vfio-pci: preserve MSI
Save the MSI message area as part of vfio-pci vmstate, and preserve the interrupt and notifier eventfd's. migrate_incoming loads the MSI data, then the vfio-pci post_load handler finds the eventfds in CPR state, rebuilds vector data structures, and attaches the interrupts to the new KVM instance. Signed-off-by: Steve Sistare <steven.sistare@oracle.com> Reviewed-by: Cédric Le Goater <clg@redhat.com> Link: https://lore.kernel.org/qemu-devel/1751493538-202042-2-git-send-email-steven.sistare@oracle.com Signed-off-by: Cédric Le Goater <clg@redhat.com>
This commit is contained in:
parent
3f2a36d0ef
commit
30edcb4d4e
4 changed files with 157 additions and 2 deletions
|
@ -9,6 +9,8 @@
|
||||||
#include "hw/vfio/vfio-device.h"
|
#include "hw/vfio/vfio-device.h"
|
||||||
#include "hw/vfio/vfio-cpr.h"
|
#include "hw/vfio/vfio-cpr.h"
|
||||||
#include "hw/vfio/pci.h"
|
#include "hw/vfio/pci.h"
|
||||||
|
#include "hw/pci/msix.h"
|
||||||
|
#include "hw/pci/msi.h"
|
||||||
#include "migration/cpr.h"
|
#include "migration/cpr.h"
|
||||||
#include "qapi/error.h"
|
#include "qapi/error.h"
|
||||||
#include "system/runstate.h"
|
#include "system/runstate.h"
|
||||||
|
@ -40,6 +42,69 @@ void vfio_cpr_unregister_container(VFIOContainerBase *bcontainer)
|
||||||
migration_remove_notifier(&bcontainer->cpr_reboot_notifier);
|
migration_remove_notifier(&bcontainer->cpr_reboot_notifier);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#define STRDUP_VECTOR_FD_NAME(vdev, name) \
|
||||||
|
g_strdup_printf("%s_%s", (vdev)->vbasedev.name, (name))
|
||||||
|
|
||||||
|
void vfio_cpr_save_vector_fd(VFIOPCIDevice *vdev, const char *name, int nr,
|
||||||
|
int fd)
|
||||||
|
{
|
||||||
|
g_autofree char *fdname = STRDUP_VECTOR_FD_NAME(vdev, name);
|
||||||
|
cpr_save_fd(fdname, nr, fd);
|
||||||
|
}
|
||||||
|
|
||||||
|
int vfio_cpr_load_vector_fd(VFIOPCIDevice *vdev, const char *name, int nr)
|
||||||
|
{
|
||||||
|
g_autofree char *fdname = STRDUP_VECTOR_FD_NAME(vdev, name);
|
||||||
|
return cpr_find_fd(fdname, nr);
|
||||||
|
}
|
||||||
|
|
||||||
|
void vfio_cpr_delete_vector_fd(VFIOPCIDevice *vdev, const char *name, int nr)
|
||||||
|
{
|
||||||
|
g_autofree char *fdname = STRDUP_VECTOR_FD_NAME(vdev, name);
|
||||||
|
cpr_delete_fd(fdname, nr);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void vfio_cpr_claim_vectors(VFIOPCIDevice *vdev, int nr_vectors,
|
||||||
|
bool msix)
|
||||||
|
{
|
||||||
|
int i, fd;
|
||||||
|
bool pending = false;
|
||||||
|
PCIDevice *pdev = &vdev->pdev;
|
||||||
|
|
||||||
|
vdev->nr_vectors = nr_vectors;
|
||||||
|
vdev->msi_vectors = g_new0(VFIOMSIVector, nr_vectors);
|
||||||
|
vdev->interrupt = msix ? VFIO_INT_MSIX : VFIO_INT_MSI;
|
||||||
|
|
||||||
|
vfio_pci_prepare_kvm_msi_virq_batch(vdev);
|
||||||
|
|
||||||
|
for (i = 0; i < nr_vectors; i++) {
|
||||||
|
VFIOMSIVector *vector = &vdev->msi_vectors[i];
|
||||||
|
|
||||||
|
fd = vfio_cpr_load_vector_fd(vdev, "interrupt", i);
|
||||||
|
if (fd >= 0) {
|
||||||
|
vfio_pci_vector_init(vdev, i);
|
||||||
|
vfio_pci_msi_set_handler(vdev, i);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (vfio_cpr_load_vector_fd(vdev, "kvm_interrupt", i) >= 0) {
|
||||||
|
vfio_pci_add_kvm_msi_virq(vdev, vector, i, msix);
|
||||||
|
} else {
|
||||||
|
vdev->msi_vectors[i].virq = -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (msix && msix_is_pending(pdev, i) && msix_is_masked(pdev, i)) {
|
||||||
|
set_bit(i, vdev->msix->pending);
|
||||||
|
pending = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
vfio_pci_commit_kvm_msi_virq_batch(vdev);
|
||||||
|
|
||||||
|
if (msix) {
|
||||||
|
memory_region_set_enabled(&pdev->msix_pba_mmio, pending);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The kernel may change non-emulated config bits. Exclude them from the
|
* The kernel may change non-emulated config bits. Exclude them from the
|
||||||
* changed-bits check in get_pci_config_device.
|
* changed-bits check in get_pci_config_device.
|
||||||
|
@ -58,13 +123,45 @@ static int vfio_cpr_pci_pre_load(void *opaque)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int vfio_cpr_pci_post_load(void *opaque, int version_id)
|
||||||
|
{
|
||||||
|
VFIOPCIDevice *vdev = opaque;
|
||||||
|
PCIDevice *pdev = &vdev->pdev;
|
||||||
|
int nr_vectors;
|
||||||
|
|
||||||
|
if (msix_enabled(pdev)) {
|
||||||
|
vfio_pci_msix_set_notifiers(vdev);
|
||||||
|
nr_vectors = vdev->msix->entries;
|
||||||
|
vfio_cpr_claim_vectors(vdev, nr_vectors, true);
|
||||||
|
|
||||||
|
} else if (msi_enabled(pdev)) {
|
||||||
|
nr_vectors = msi_nr_vectors_allocated(pdev);
|
||||||
|
vfio_cpr_claim_vectors(vdev, nr_vectors, false);
|
||||||
|
|
||||||
|
} else if (vfio_pci_read_config(pdev, PCI_INTERRUPT_PIN, 1)) {
|
||||||
|
g_assert_not_reached(); /* completed in a subsequent patch */
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool pci_msix_present(void *opaque, int version_id)
|
||||||
|
{
|
||||||
|
PCIDevice *pdev = opaque;
|
||||||
|
|
||||||
|
return msix_present(pdev);
|
||||||
|
}
|
||||||
|
|
||||||
const VMStateDescription vfio_cpr_pci_vmstate = {
|
const VMStateDescription vfio_cpr_pci_vmstate = {
|
||||||
.name = "vfio-cpr-pci",
|
.name = "vfio-cpr-pci",
|
||||||
.version_id = 0,
|
.version_id = 0,
|
||||||
.minimum_version_id = 0,
|
.minimum_version_id = 0,
|
||||||
.pre_load = vfio_cpr_pci_pre_load,
|
.pre_load = vfio_cpr_pci_pre_load,
|
||||||
|
.post_load = vfio_cpr_pci_post_load,
|
||||||
.needed = cpr_incoming_needed,
|
.needed = cpr_incoming_needed,
|
||||||
.fields = (VMStateField[]) {
|
.fields = (VMStateField[]) {
|
||||||
|
VMSTATE_PCI_DEVICE(pdev, VFIOPCIDevice),
|
||||||
|
VMSTATE_MSIX_TEST(pdev, VFIOPCIDevice, pci_msix_present),
|
||||||
VMSTATE_END_OF_LIST()
|
VMSTATE_END_OF_LIST()
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
|
@ -29,6 +29,7 @@
|
||||||
#include "hw/pci/pci_bridge.h"
|
#include "hw/pci/pci_bridge.h"
|
||||||
#include "hw/qdev-properties.h"
|
#include "hw/qdev-properties.h"
|
||||||
#include "hw/qdev-properties-system.h"
|
#include "hw/qdev-properties-system.h"
|
||||||
|
#include "hw/vfio/vfio-cpr.h"
|
||||||
#include "migration/vmstate.h"
|
#include "migration/vmstate.h"
|
||||||
#include "migration/cpr.h"
|
#include "migration/cpr.h"
|
||||||
#include "qobject/qdict.h"
|
#include "qobject/qdict.h"
|
||||||
|
@ -57,20 +58,33 @@ static void vfio_disable_interrupts(VFIOPCIDevice *vdev);
|
||||||
static void vfio_mmap_set_enabled(VFIOPCIDevice *vdev, bool enabled);
|
static void vfio_mmap_set_enabled(VFIOPCIDevice *vdev, bool enabled);
|
||||||
static void vfio_msi_disable_common(VFIOPCIDevice *vdev);
|
static void vfio_msi_disable_common(VFIOPCIDevice *vdev);
|
||||||
|
|
||||||
|
/* Create new or reuse existing eventfd */
|
||||||
static bool vfio_notifier_init(VFIOPCIDevice *vdev, EventNotifier *e,
|
static bool vfio_notifier_init(VFIOPCIDevice *vdev, EventNotifier *e,
|
||||||
const char *name, int nr, Error **errp)
|
const char *name, int nr, Error **errp)
|
||||||
{
|
{
|
||||||
int ret = event_notifier_init(e, 0);
|
int fd, ret;
|
||||||
|
|
||||||
|
fd = vfio_cpr_load_vector_fd(vdev, name, nr);
|
||||||
|
if (fd >= 0) {
|
||||||
|
event_notifier_init_fd(e, fd);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = event_notifier_init(e, 0);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
error_setg_errno(errp, -ret, "vfio_notifier_init %s failed", name);
|
error_setg_errno(errp, -ret, "vfio_notifier_init %s failed", name);
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
return !ret;
|
|
||||||
|
fd = event_notifier_get_fd(e);
|
||||||
|
vfio_cpr_save_vector_fd(vdev, name, nr, fd);
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void vfio_notifier_cleanup(VFIOPCIDevice *vdev, EventNotifier *e,
|
static void vfio_notifier_cleanup(VFIOPCIDevice *vdev, EventNotifier *e,
|
||||||
const char *name, int nr)
|
const char *name, int nr)
|
||||||
{
|
{
|
||||||
|
vfio_cpr_delete_vector_fd(vdev, name, nr);
|
||||||
event_notifier_cleanup(e);
|
event_notifier_cleanup(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -394,6 +408,14 @@ static void vfio_msi_interrupt(void *opaque)
|
||||||
notify(&vdev->pdev, nr);
|
notify(&vdev->pdev, nr);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void vfio_pci_msi_set_handler(VFIOPCIDevice *vdev, int nr)
|
||||||
|
{
|
||||||
|
VFIOMSIVector *vector = &vdev->msi_vectors[nr];
|
||||||
|
int fd = event_notifier_get_fd(&vector->interrupt);
|
||||||
|
|
||||||
|
qemu_set_fd_handler(fd, vfio_msi_interrupt, NULL, vector);
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Get MSI-X enabled, but no vector enabled, by setting vector 0 with an invalid
|
* Get MSI-X enabled, but no vector enabled, by setting vector 0 with an invalid
|
||||||
* fd to kernel.
|
* fd to kernel.
|
||||||
|
@ -656,6 +678,15 @@ static int vfio_msix_vector_do_use(PCIDevice *pdev, unsigned int nr,
|
||||||
static int vfio_msix_vector_use(PCIDevice *pdev,
|
static int vfio_msix_vector_use(PCIDevice *pdev,
|
||||||
unsigned int nr, MSIMessage msg)
|
unsigned int nr, MSIMessage msg)
|
||||||
{
|
{
|
||||||
|
/*
|
||||||
|
* Ignore the callback from msix_set_vector_notifiers during resume.
|
||||||
|
* The necessary subset of these actions is called from
|
||||||
|
* vfio_cpr_claim_vectors during post load.
|
||||||
|
*/
|
||||||
|
if (cpr_is_incoming()) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
return vfio_msix_vector_do_use(pdev, nr, &msg, vfio_msi_interrupt);
|
return vfio_msix_vector_do_use(pdev, nr, &msg, vfio_msi_interrupt);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -686,6 +717,12 @@ static void vfio_msix_vector_release(PCIDevice *pdev, unsigned int nr)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void vfio_pci_msix_set_notifiers(VFIOPCIDevice *vdev)
|
||||||
|
{
|
||||||
|
msix_set_vector_notifiers(&vdev->pdev, vfio_msix_vector_use,
|
||||||
|
vfio_msix_vector_release, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
void vfio_pci_prepare_kvm_msi_virq_batch(VFIOPCIDevice *vdev)
|
void vfio_pci_prepare_kvm_msi_virq_batch(VFIOPCIDevice *vdev)
|
||||||
{
|
{
|
||||||
assert(!vdev->defer_kvm_irq_routing);
|
assert(!vdev->defer_kvm_irq_routing);
|
||||||
|
@ -2965,6 +3002,11 @@ void vfio_pci_register_err_notifier(VFIOPCIDevice *vdev)
|
||||||
fd = event_notifier_get_fd(&vdev->err_notifier);
|
fd = event_notifier_get_fd(&vdev->err_notifier);
|
||||||
qemu_set_fd_handler(fd, vfio_err_notifier_handler, NULL, vdev);
|
qemu_set_fd_handler(fd, vfio_err_notifier_handler, NULL, vdev);
|
||||||
|
|
||||||
|
/* Do not alter irq_signaling during vfio_realize for cpr */
|
||||||
|
if (cpr_is_incoming()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (!vfio_device_irq_set_signaling(&vdev->vbasedev, VFIO_PCI_ERR_IRQ_INDEX, 0,
|
if (!vfio_device_irq_set_signaling(&vdev->vbasedev, VFIO_PCI_ERR_IRQ_INDEX, 0,
|
||||||
VFIO_IRQ_SET_ACTION_TRIGGER, fd, &err)) {
|
VFIO_IRQ_SET_ACTION_TRIGGER, fd, &err)) {
|
||||||
error_reportf_err(err, VFIO_MSG_PREFIX, vdev->vbasedev.name);
|
error_reportf_err(err, VFIO_MSG_PREFIX, vdev->vbasedev.name);
|
||||||
|
@ -3032,6 +3074,12 @@ void vfio_pci_register_req_notifier(VFIOPCIDevice *vdev)
|
||||||
fd = event_notifier_get_fd(&vdev->req_notifier);
|
fd = event_notifier_get_fd(&vdev->req_notifier);
|
||||||
qemu_set_fd_handler(fd, vfio_req_notifier_handler, NULL, vdev);
|
qemu_set_fd_handler(fd, vfio_req_notifier_handler, NULL, vdev);
|
||||||
|
|
||||||
|
/* Do not alter irq_signaling during vfio_realize for cpr */
|
||||||
|
if (cpr_is_incoming()) {
|
||||||
|
vdev->req_enabled = true;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (!vfio_device_irq_set_signaling(&vdev->vbasedev, VFIO_PCI_REQ_IRQ_INDEX, 0,
|
if (!vfio_device_irq_set_signaling(&vdev->vbasedev, VFIO_PCI_REQ_IRQ_INDEX, 0,
|
||||||
VFIO_IRQ_SET_ACTION_TRIGGER, fd, &err)) {
|
VFIO_IRQ_SET_ACTION_TRIGGER, fd, &err)) {
|
||||||
error_reportf_err(err, VFIO_MSG_PREFIX, vdev->vbasedev.name);
|
error_reportf_err(err, VFIO_MSG_PREFIX, vdev->vbasedev.name);
|
||||||
|
|
|
@ -218,6 +218,8 @@ void vfio_pci_add_kvm_msi_virq(VFIOPCIDevice *vdev, VFIOMSIVector *vector,
|
||||||
void vfio_pci_prepare_kvm_msi_virq_batch(VFIOPCIDevice *vdev);
|
void vfio_pci_prepare_kvm_msi_virq_batch(VFIOPCIDevice *vdev);
|
||||||
void vfio_pci_commit_kvm_msi_virq_batch(VFIOPCIDevice *vdev);
|
void vfio_pci_commit_kvm_msi_virq_batch(VFIOPCIDevice *vdev);
|
||||||
bool vfio_pci_intx_enable(VFIOPCIDevice *vdev, Error **errp);
|
bool vfio_pci_intx_enable(VFIOPCIDevice *vdev, Error **errp);
|
||||||
|
void vfio_pci_msix_set_notifiers(VFIOPCIDevice *vdev);
|
||||||
|
void vfio_pci_msi_set_handler(VFIOPCIDevice *vdev, int nr);
|
||||||
|
|
||||||
uint32_t vfio_pci_read_config(PCIDevice *pdev, uint32_t addr, int len);
|
uint32_t vfio_pci_read_config(PCIDevice *pdev, uint32_t addr, int len);
|
||||||
void vfio_pci_write_config(PCIDevice *pdev,
|
void vfio_pci_write_config(PCIDevice *pdev,
|
||||||
|
|
|
@ -15,6 +15,7 @@
|
||||||
struct VFIOContainer;
|
struct VFIOContainer;
|
||||||
struct VFIOContainerBase;
|
struct VFIOContainerBase;
|
||||||
struct VFIOGroup;
|
struct VFIOGroup;
|
||||||
|
struct VFIOPCIDevice;
|
||||||
|
|
||||||
typedef int (*dma_map_fn)(const struct VFIOContainerBase *bcontainer,
|
typedef int (*dma_map_fn)(const struct VFIOContainerBase *bcontainer,
|
||||||
hwaddr iova, ram_addr_t size, void *vaddr,
|
hwaddr iova, ram_addr_t size, void *vaddr,
|
||||||
|
@ -53,6 +54,13 @@ void vfio_cpr_giommu_remap(struct VFIOContainerBase *bcontainer,
|
||||||
bool vfio_cpr_ram_discard_register_listener(
|
bool vfio_cpr_ram_discard_register_listener(
|
||||||
struct VFIOContainerBase *bcontainer, MemoryRegionSection *section);
|
struct VFIOContainerBase *bcontainer, MemoryRegionSection *section);
|
||||||
|
|
||||||
|
void vfio_cpr_save_vector_fd(struct VFIOPCIDevice *vdev, const char *name,
|
||||||
|
int nr, int fd);
|
||||||
|
int vfio_cpr_load_vector_fd(struct VFIOPCIDevice *vdev, const char *name,
|
||||||
|
int nr);
|
||||||
|
void vfio_cpr_delete_vector_fd(struct VFIOPCIDevice *vdev, const char *name,
|
||||||
|
int nr);
|
||||||
|
|
||||||
extern const VMStateDescription vfio_cpr_pci_vmstate;
|
extern const VMStateDescription vfio_cpr_pci_vmstate;
|
||||||
|
|
||||||
#endif /* HW_VFIO_VFIO_CPR_H */
|
#endif /* HW_VFIO_VFIO_CPR_H */
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue