A vhost device is implicitly preserved across re-exec because its fd is not closed, and the value of the fd is specified on the command line for the new qemu to find. However, new qemu issues an VHOST_RESET_OWNER ioctl, which fails because the device already has an owner. To fix, reset the owner prior to exec.
Signed-off-by: Mark Kanda <mark.ka...@oracle.com> Signed-off-by: Steve Sistare <steven.sist...@oracle.com> --- hw/virtio/vhost.c | 17 +++++++++++++++++ include/hw/virtio/vhost.h | 1 + 2 files changed, 18 insertions(+) diff --git a/hw/virtio/vhost.c b/hw/virtio/vhost.c index dd3263d..efaa28c 100644 --- a/hw/virtio/vhost.c +++ b/hw/virtio/vhost.c @@ -23,6 +23,7 @@ #include "standard-headers/linux/vhost_types.h" #include "hw/virtio/virtio-bus.h" #include "hw/virtio/virtio-access.h" +#include "migration/cpr.h" #include "migration/blocker.h" #include "migration/qemu-file-types.h" #include "sysemu/dma.h" @@ -1306,6 +1307,17 @@ static void vhost_virtqueue_cleanup(struct vhost_virtqueue *vq) event_notifier_cleanup(&vq->masked_notifier); } +static void vhost_cpr_exec_notifier(Notifier *notifier, void *data) +{ + struct vhost_dev *dev = container_of(notifier, struct vhost_dev, + cpr_notifier); + int r = dev->vhost_ops->vhost_reset_device(dev); + + if (r < 0) { + VHOST_OPS_DEBUG(r, "vhost_reset_device failed"); + } +} + int vhost_dev_init(struct vhost_dev *hdev, void *opaque, VhostBackendType backend_type, uint32_t busyloop_timeout, Error **errp) @@ -1405,6 +1417,8 @@ int vhost_dev_init(struct vhost_dev *hdev, void *opaque, hdev->log_enabled = false; hdev->started = false; memory_listener_register(&hdev->memory_listener, &address_space_memory); + cpr_add_notifier(&hdev->cpr_notifier, vhost_cpr_exec_notifier, + CPR_NOTIFY_EXEC); QLIST_INSERT_HEAD(&vhost_devices, hdev, entry); if (used_memslots > hdev->vhost_ops->vhost_backend_memslots_limit(hdev)) { @@ -1444,6 +1458,9 @@ void vhost_dev_cleanup(struct vhost_dev *hdev) migrate_del_blocker(hdev->migration_blocker); error_free(hdev->migration_blocker); } + if (hdev->cpr_notifier.notify) { + cpr_remove_notifier(&hdev->cpr_notifier); + } g_free(hdev->mem); g_free(hdev->mem_sections); if (hdev->vhost_ops) { diff --git a/include/hw/virtio/vhost.h b/include/hw/virtio/vhost.h index b291fe4..1316b14 100644 --- a/include/hw/virtio/vhost.h +++ b/include/hw/virtio/vhost.h @@ -100,6 +100,7 @@ struct vhost_dev { QLIST_ENTRY(vhost_dev) entry; QLIST_HEAD(, vhost_iommu) iommu_list; IOMMUNotifier n; + Notifier cpr_notifier; const VhostDevConfigOps *config_ops; }; -- 1.8.3.1