mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-22 12:35:17 +00:00
qemu: Introduce a wrapper over virFileWrapperFdClose
https://bugzilla.redhat.com/show_bug.cgi?id=1448268 When migrating to a file (e.g. when doing 'virsh save file'), couple of things are happening in the thread that is executing the API: 1) the domain obj is locked 2) iohelper is spawned as a separate process to handle all I/O 3) the thread waits for iohelper to finish 4) the domain obj is unlocked Now, the problem is that while the thread waits in step 3 for iohelper to finish this may take ages because iohelper calls fdatasync(). And unfortunately, we are waiting the whole time with the domain locked. So if another thread wants to jump in and say copy the domain name ('virsh list' for instance), they are stuck. The solution is to unlock the domain whenever waiting for I/O and lock it back again when it finished. Signed-off-by: Michal Privoznik <mprivozn@redhat.com> Reviewed-by: John Ferlan <jferlan@redhat.com>
This commit is contained in:
parent
ed2a741e48
commit
92524d3e6e
@ -3216,6 +3216,35 @@ qemuOpenFileAs(uid_t fallback_uid, gid_t fallback_gid,
|
|||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int
|
||||||
|
qemuFileWrapperFDClose(virDomainObjPtr vm,
|
||||||
|
virFileWrapperFdPtr fd)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
/* virFileWrapperFd uses iohelper to write data onto disk.
|
||||||
|
* However, iohelper calls fdatasync() which may take ages to
|
||||||
|
* finish. Therefore, we shouldn't be waiting with the domain
|
||||||
|
* object locked. */
|
||||||
|
|
||||||
|
/* XXX Currently, this function is intended for *Save() only
|
||||||
|
* as restore needs some reworking before it's ready for
|
||||||
|
* this. */
|
||||||
|
|
||||||
|
virObjectUnlock(vm);
|
||||||
|
ret = virFileWrapperFdClose(fd);
|
||||||
|
virObjectLock(vm);
|
||||||
|
if (!virDomainObjIsActive(vm)) {
|
||||||
|
if (!virGetLastError())
|
||||||
|
virReportError(VIR_ERR_OPERATION_FAILED, "%s",
|
||||||
|
_("domain is no longer running"));
|
||||||
|
ret = -1;
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/* Helper function to execute a migration to file with a correct save header
|
/* Helper function to execute a migration to file with a correct save header
|
||||||
* the caller needs to make sure that the processors are stopped and do all other
|
* the caller needs to make sure that the processors are stopped and do all other
|
||||||
* actions besides saving memory */
|
* actions besides saving memory */
|
||||||
@ -3276,7 +3305,7 @@ qemuDomainSaveMemory(virQEMUDriverPtr driver,
|
|||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (virFileWrapperFdClose(wrapperFd) < 0)
|
if (qemuFileWrapperFDClose(vm, wrapperFd) < 0)
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
|
|
||||||
if ((fd = qemuOpenFile(driver, vm, path, O_WRONLY, NULL, NULL)) < 0 ||
|
if ((fd = qemuOpenFile(driver, vm, path, O_WRONLY, NULL, NULL)) < 0 ||
|
||||||
@ -3827,7 +3856,7 @@ doCoreDump(virQEMUDriverPtr driver,
|
|||||||
path);
|
path);
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
if (virFileWrapperFdClose(wrapperFd) < 0)
|
if (qemuFileWrapperFDClose(vm, wrapperFd) < 0)
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
|
|
||||||
ret = 0;
|
ret = 0;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user