mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-11 07:17:44 +00:00
qemu_migration: Introduce qemuMigrationStartNBDServer()
We need to start NBD server and feed it with all non-<shared/>, RW and source-full disks. Moreover, with new virPortAllocator we must ensure the borrowed port for NBD server will be returned if either migration completes or qemu process is torn down.
This commit is contained in:
parent
f1748e34e2
commit
86d90b3abd
@ -35,6 +35,7 @@
|
|||||||
#include "qemu_domain.h"
|
#include "qemu_domain.h"
|
||||||
#include "qemu_process.h"
|
#include "qemu_process.h"
|
||||||
#include "qemu_capabilities.h"
|
#include "qemu_capabilities.h"
|
||||||
|
#include "qemu_command.h"
|
||||||
#include "qemu_cgroup.h"
|
#include "qemu_cgroup.h"
|
||||||
|
|
||||||
#include "domain_audit.h"
|
#include "domain_audit.h"
|
||||||
@ -1088,6 +1089,72 @@ error:
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* qemuMigrationStartNBDServer:
|
||||||
|
* @driver: qemu driver
|
||||||
|
* @vm: domain
|
||||||
|
*
|
||||||
|
* Starts NBD server. This is a newer method to copy
|
||||||
|
* storage during migration than using 'blk' and 'inc'
|
||||||
|
* arguments in 'migrate' monitor command.
|
||||||
|
* Error is reported here.
|
||||||
|
*
|
||||||
|
* Returns 0 on success, -1 otherwise.
|
||||||
|
*/
|
||||||
|
static int
|
||||||
|
qemuMigrationStartNBDServer(virQEMUDriverPtr driver,
|
||||||
|
virDomainObjPtr vm)
|
||||||
|
{
|
||||||
|
int ret = -1;
|
||||||
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
||||||
|
unsigned short port = 0;
|
||||||
|
const char *listenAddr = "0.0.0.0";
|
||||||
|
char *diskAlias = NULL;
|
||||||
|
size_t i;
|
||||||
|
|
||||||
|
for (i = 0; i < vm->def->ndisks; i++) {
|
||||||
|
virDomainDiskDefPtr disk = vm->def->disks[i];
|
||||||
|
|
||||||
|
/* skip shared, RO and source-less disks */
|
||||||
|
if (disk->shared || disk->readonly || !disk->src)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
VIR_FREE(diskAlias);
|
||||||
|
if (virAsprintf(&diskAlias, "%s%s",
|
||||||
|
QEMU_DRIVE_HOST_PREFIX, disk->info.alias) < 0) {
|
||||||
|
virReportOOMError();
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (qemuDomainObjEnterMonitorAsync(driver, vm,
|
||||||
|
QEMU_ASYNC_JOB_MIGRATION_IN) < 0)
|
||||||
|
goto cleanup;
|
||||||
|
|
||||||
|
if (!port &&
|
||||||
|
((virPortAllocatorAcquire(driver->remotePorts, &port) < 0) ||
|
||||||
|
(qemuMonitorNBDServerStart(priv->mon, listenAddr, port) < 0))) {
|
||||||
|
qemuDomainObjExitMonitor(driver, vm);
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (qemuMonitorNBDServerAdd(priv->mon, diskAlias, true) < 0) {
|
||||||
|
qemuDomainObjExitMonitor(driver, vm);
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
qemuDomainObjExitMonitor(driver, vm);
|
||||||
|
}
|
||||||
|
|
||||||
|
priv->nbdPort = port;
|
||||||
|
ret = 0;
|
||||||
|
|
||||||
|
cleanup:
|
||||||
|
VIR_FREE(diskAlias);
|
||||||
|
if ((ret < 0) && port)
|
||||||
|
virPortAllocatorRelease(driver->remotePorts, port);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/* Validate whether the domain is safe to migrate. If vm is NULL,
|
/* Validate whether the domain is safe to migrate. If vm is NULL,
|
||||||
* then this is being run in the v2 Prepare stage on the destination
|
* then this is being run in the v2 Prepare stage on the destination
|
||||||
* (where we only have the target xml); if vm is provided, then this
|
* (where we only have the target xml); if vm is provided, then this
|
||||||
@ -1864,8 +1931,11 @@ done:
|
|||||||
if (flags & VIR_MIGRATE_TUNNELLED)
|
if (flags & VIR_MIGRATE_TUNNELLED)
|
||||||
VIR_DEBUG("NBD in tunnelled migration is currently not supported");
|
VIR_DEBUG("NBD in tunnelled migration is currently not supported");
|
||||||
else {
|
else {
|
||||||
|
if (qemuMigrationStartNBDServer(driver, vm) < 0) {
|
||||||
|
/* error already reported */
|
||||||
|
goto endjob;
|
||||||
|
}
|
||||||
cookieFlags |= QEMU_MIGRATION_COOKIE_NBD;
|
cookieFlags |= QEMU_MIGRATION_COOKIE_NBD;
|
||||||
priv->nbdPort = 0;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1912,6 +1982,10 @@ cleanup:
|
|||||||
virObjectUnlock(vm);
|
virObjectUnlock(vm);
|
||||||
else
|
else
|
||||||
qemuDomainRemoveInactive(driver, vm);
|
qemuDomainRemoveInactive(driver, vm);
|
||||||
|
if (ret < 0 && priv->nbdPort) {
|
||||||
|
virPortAllocatorRelease(driver->remotePorts, priv->nbdPort);
|
||||||
|
priv->nbdPort = 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (event)
|
if (event)
|
||||||
qemuDomainEventQueue(driver, event);
|
qemuDomainEventQueue(driver, event);
|
||||||
|
@ -4202,6 +4202,11 @@ void qemuProcessStop(virQEMUDriverPtr driver,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (priv->nbdPort) {
|
||||||
|
virPortAllocatorRelease(driver->remotePorts, priv->nbdPort);
|
||||||
|
priv->nbdPort = 0;
|
||||||
|
}
|
||||||
|
|
||||||
if (priv->agent) {
|
if (priv->agent) {
|
||||||
qemuAgentClose(priv->agent);
|
qemuAgentClose(priv->agent);
|
||||||
priv->agent = NULL;
|
priv->agent = NULL;
|
||||||
|
Loading…
Reference in New Issue
Block a user