2019-09-19 14:49:21 +00:00
|
|
|
/*
|
|
|
|
* qemu_checkpoint.c: checkpoint related implementation
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library. If not, see
|
|
|
|
* <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
|
|
|
|
#include "qemu_checkpoint.h"
|
|
|
|
#include "qemu_capabilities.h"
|
|
|
|
#include "qemu_monitor.h"
|
|
|
|
#include "qemu_domain.h"
|
|
|
|
|
|
|
|
#include "virerror.h"
|
|
|
|
#include "virlog.h"
|
|
|
|
#include "datatypes.h"
|
|
|
|
#include "viralloc.h"
|
|
|
|
#include "domain_conf.h"
|
|
|
|
#include "libvirt_internal.h"
|
|
|
|
#include "virxml.h"
|
|
|
|
#include "virstring.h"
|
|
|
|
#include "virdomaincheckpointobjlist.h"
|
|
|
|
#include "virdomainsnapshotobjlist.h"
|
|
|
|
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_QEMU
|
|
|
|
|
|
|
|
VIR_LOG_INIT("qemu.qemu_checkpoint");
|
|
|
|
|
|
|
|
/* Looks up the domain object from checkpoint and unlocks the
|
|
|
|
* driver. The returned domain object is locked and ref'd and the
|
|
|
|
* caller must call virDomainObjEndAPI() on it. */
|
|
|
|
virDomainObjPtr
|
|
|
|
qemuDomObjFromCheckpoint(virDomainCheckpointPtr checkpoint)
|
|
|
|
{
|
|
|
|
return qemuDomainObjFromDomain(checkpoint->domain);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Looks up checkpoint object from VM and name */
|
|
|
|
virDomainMomentObjPtr
|
|
|
|
qemuCheckpointObjFromName(virDomainObjPtr vm,
|
|
|
|
const char *name)
|
|
|
|
{
|
|
|
|
virDomainMomentObjPtr chk = NULL;
|
|
|
|
chk = virDomainCheckpointFindByName(vm->checkpoints, name);
|
|
|
|
if (!chk)
|
|
|
|
virReportError(VIR_ERR_NO_DOMAIN_CHECKPOINT,
|
|
|
|
_("no domain checkpoint with matching name '%s'"),
|
|
|
|
name);
|
|
|
|
|
|
|
|
return chk;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Looks up checkpoint object from VM and checkpointPtr */
|
|
|
|
virDomainMomentObjPtr
|
|
|
|
qemuCheckpointObjFromCheckpoint(virDomainObjPtr vm,
|
|
|
|
virDomainCheckpointPtr checkpoint)
|
|
|
|
{
|
|
|
|
return qemuCheckpointObjFromName(vm, checkpoint->name);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-09-20 11:47:04 +00:00
|
|
|
static int
|
|
|
|
qemuCheckpointWriteMetadata(virDomainObjPtr vm,
|
|
|
|
virDomainMomentObjPtr checkpoint,
|
|
|
|
virDomainXMLOptionPtr xmlopt,
|
|
|
|
const char *checkpointDir)
|
|
|
|
{
|
|
|
|
unsigned int flags = VIR_DOMAIN_CHECKPOINT_FORMAT_SECURE;
|
|
|
|
virDomainCheckpointDefPtr def = virDomainCheckpointObjGetDef(checkpoint);
|
2019-10-15 13:16:31 +00:00
|
|
|
g_autofree char *newxml = NULL;
|
|
|
|
g_autofree char *chkDir = NULL;
|
|
|
|
g_autofree char *chkFile = NULL;
|
2019-09-20 11:47:04 +00:00
|
|
|
|
2019-11-27 13:10:21 +00:00
|
|
|
newxml = virDomainCheckpointDefFormat(def, xmlopt, flags);
|
2019-09-20 11:47:04 +00:00
|
|
|
if (newxml == NULL)
|
|
|
|
return -1;
|
|
|
|
|
2019-10-22 13:26:14 +00:00
|
|
|
chkDir = g_strdup_printf("%s/%s", checkpointDir, vm->def->name);
|
2019-09-20 11:47:04 +00:00
|
|
|
if (virFileMakePath(chkDir) < 0) {
|
|
|
|
virReportSystemError(errno, _("cannot create checkpoint directory '%s'"),
|
|
|
|
chkDir);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2019-10-22 13:26:14 +00:00
|
|
|
chkFile = g_strdup_printf("%s/%s.xml", chkDir, def->parent.name);
|
2019-09-20 11:47:04 +00:00
|
|
|
|
|
|
|
return virXMLSaveFile(chkFile, NULL, "checkpoint-edit", newxml);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
qemuCheckpointDiscard(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainMomentObjPtr chk,
|
|
|
|
bool update_parent,
|
|
|
|
bool metadata_only)
|
|
|
|
{
|
|
|
|
virDomainMomentObjPtr parent = NULL;
|
|
|
|
virDomainMomentObjPtr moment;
|
|
|
|
virDomainCheckpointDefPtr parentdef = NULL;
|
|
|
|
size_t i, j;
|
2019-10-15 12:47:50 +00:00
|
|
|
g_autoptr(virQEMUDriverConfig) cfg = virQEMUDriverGetConfig(driver);
|
2019-10-15 13:16:31 +00:00
|
|
|
g_autofree char *chkFile = NULL;
|
2019-09-20 11:47:04 +00:00
|
|
|
|
|
|
|
if (!metadata_only && !virDomainObjIsActive(vm)) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
|
|
|
_("cannot remove checkpoint from inactive domain"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2019-10-22 13:26:14 +00:00
|
|
|
chkFile = g_strdup_printf("%s/%s/%s.xml", cfg->checkpointDir, vm->def->name,
|
|
|
|
chk->def->name);
|
2019-09-20 11:47:04 +00:00
|
|
|
|
|
|
|
if (!metadata_only) {
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
bool search_parents;
|
|
|
|
virDomainCheckpointDefPtr chkdef = virDomainCheckpointObjGetDef(chk);
|
2019-09-26 14:56:16 +00:00
|
|
|
int rc;
|
|
|
|
g_autoptr(virJSONValue) actions = NULL;
|
|
|
|
|
|
|
|
if (!(actions = virJSONValueNewArray()))
|
|
|
|
return -1;
|
2019-09-20 11:47:04 +00:00
|
|
|
|
|
|
|
parent = virDomainCheckpointFindByName(vm->checkpoints,
|
|
|
|
chk->def->parent_name);
|
|
|
|
for (i = 0; i < chkdef->ndisks; i++) {
|
|
|
|
virDomainCheckpointDiskDef *disk = &chkdef->disks[i];
|
|
|
|
const char *node;
|
|
|
|
|
|
|
|
if (disk->type != VIR_DOMAIN_CHECKPOINT_TYPE_BITMAP)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
node = qemuDomainDiskNodeFormatLookup(vm, disk->name);
|
|
|
|
/* If any ancestor checkpoint has a bitmap for the same
|
|
|
|
* disk, then this bitmap must be merged to the
|
|
|
|
* ancestor. */
|
|
|
|
search_parents = true;
|
|
|
|
for (moment = parent;
|
|
|
|
search_parents && moment;
|
|
|
|
moment = virDomainCheckpointFindByName(vm->checkpoints,
|
|
|
|
parentdef->parent.parent_name)) {
|
|
|
|
parentdef = virDomainCheckpointObjGetDef(moment);
|
|
|
|
for (j = 0; j < parentdef->ndisks; j++) {
|
|
|
|
virDomainCheckpointDiskDef *disk2;
|
2019-10-15 12:47:50 +00:00
|
|
|
g_autoptr(virJSONValue) arr = NULL;
|
2019-09-20 11:47:04 +00:00
|
|
|
|
|
|
|
disk2 = &parentdef->disks[j];
|
|
|
|
if (STRNEQ(disk->name, disk2->name) ||
|
|
|
|
disk2->type != VIR_DOMAIN_CHECKPOINT_TYPE_BITMAP)
|
|
|
|
continue;
|
|
|
|
search_parents = false;
|
|
|
|
|
2019-09-26 14:56:16 +00:00
|
|
|
if (!(arr = virJSONValueNewArray()))
|
|
|
|
return -1;
|
|
|
|
|
2019-10-07 14:33:49 +00:00
|
|
|
if (qemuMonitorTransactionBitmapMergeSourceAddBitmap(arr, node, disk->bitmap) < 0)
|
2019-09-26 14:56:16 +00:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (chk == virDomainCheckpointGetCurrent(vm->checkpoints)) {
|
|
|
|
if (qemuMonitorTransactionBitmapEnable(actions, node, disk2->bitmap) < 0)
|
|
|
|
return -1;
|
2019-09-20 11:47:04 +00:00
|
|
|
}
|
2019-09-26 14:56:16 +00:00
|
|
|
|
|
|
|
if (qemuMonitorTransactionBitmapMerge(actions, node, disk2->bitmap, &arr) < 0)
|
|
|
|
return -1;
|
2019-09-20 11:47:04 +00:00
|
|
|
}
|
|
|
|
}
|
2019-09-26 14:56:16 +00:00
|
|
|
|
|
|
|
if (qemuMonitorTransactionBitmapRemove(actions, node, disk->bitmap) < 0)
|
|
|
|
return -1;
|
2019-09-20 11:47:04 +00:00
|
|
|
}
|
2019-09-26 14:56:16 +00:00
|
|
|
|
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
|
|
|
rc = qemuMonitorTransaction(priv->mon, &actions);
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0 || rc < 0)
|
2019-09-20 11:47:04 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (chk == virDomainCheckpointGetCurrent(vm->checkpoints)) {
|
|
|
|
virDomainCheckpointSetCurrent(vm->checkpoints, NULL);
|
|
|
|
if (update_parent && parent) {
|
|
|
|
virDomainCheckpointSetCurrent(vm->checkpoints, parent);
|
2019-11-27 13:19:09 +00:00
|
|
|
if (qemuCheckpointWriteMetadata(vm, parent,
|
2019-09-20 11:47:04 +00:00
|
|
|
driver->xmlopt,
|
|
|
|
cfg->checkpointDir) < 0) {
|
|
|
|
VIR_WARN("failed to set parent checkpoint '%s' as current",
|
|
|
|
chk->def->parent_name);
|
|
|
|
virDomainCheckpointSetCurrent(vm->checkpoints, NULL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (unlink(chkFile) < 0)
|
|
|
|
VIR_WARN("Failed to unlink %s", chkFile);
|
|
|
|
if (update_parent)
|
|
|
|
virDomainMomentDropParent(chk);
|
|
|
|
virDomainCheckpointObjListRemove(vm->checkpoints, chk);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
qemuCheckpointDiscardAllMetadata(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm)
|
|
|
|
{
|
|
|
|
virQEMUMomentRemove rem = {
|
|
|
|
.driver = driver,
|
|
|
|
.vm = vm,
|
|
|
|
.metadata_only = true,
|
|
|
|
.momentDiscard = qemuCheckpointDiscard,
|
|
|
|
};
|
|
|
|
|
|
|
|
virDomainCheckpointForEach(vm->checkpoints, qemuDomainMomentDiscardAll,
|
|
|
|
&rem);
|
|
|
|
virDomainCheckpointObjListRemoveAll(vm->checkpoints);
|
|
|
|
|
|
|
|
return rem.err;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-09-19 14:49:21 +00:00
|
|
|
/* Called inside job lock */
|
|
|
|
static int
|
|
|
|
qemuCheckpointPrepare(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainCheckpointDefPtr def)
|
|
|
|
{
|
|
|
|
int ret = -1;
|
|
|
|
size_t i;
|
|
|
|
char *xml = NULL;
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
|
|
|
|
/* Easiest way to clone inactive portion of vm->def is via
|
|
|
|
* conversion in and back out of xml. */
|
|
|
|
if (!(xml = qemuDomainDefFormatLive(driver, priv->qemuCaps,
|
|
|
|
vm->def, priv->origCPU,
|
|
|
|
true, true)) ||
|
2019-11-27 12:29:21 +00:00
|
|
|
!(def->parent.dom = virDomainDefParseString(xml, driver->xmlopt,
|
2019-09-19 14:49:21 +00:00
|
|
|
priv->qemuCaps,
|
|
|
|
VIR_DOMAIN_DEF_PARSE_INACTIVE |
|
|
|
|
VIR_DOMAIN_DEF_PARSE_SKIP_VALIDATE)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (virDomainCheckpointAlignDisks(def) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
for (i = 0; i < def->ndisks; i++) {
|
|
|
|
virDomainCheckpointDiskDefPtr disk = &def->disks[i];
|
|
|
|
|
|
|
|
if (disk->type != VIR_DOMAIN_CHECKPOINT_TYPE_BITMAP)
|
|
|
|
continue;
|
|
|
|
|
2019-09-18 12:48:57 +00:00
|
|
|
if (STRNEQ(disk->bitmap, def->parent.name)) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("bitmap for disk '%s' must match checkpoint name '%s'"),
|
|
|
|
disk->name, def->parent.name);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2019-09-19 14:49:21 +00:00
|
|
|
if (vm->def->disks[i]->src->format != VIR_STORAGE_FILE_QCOW2) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("checkpoint for disk %s unsupported "
|
|
|
|
"for storage type %s"),
|
|
|
|
disk->name,
|
|
|
|
virStorageFileFormatTypeToString(
|
|
|
|
vm->def->disks[i]->src->format));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(xml);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
qemuCheckpointAddActions(virDomainObjPtr vm,
|
|
|
|
virJSONValuePtr actions,
|
|
|
|
virDomainMomentObjPtr old_current,
|
|
|
|
virDomainCheckpointDefPtr def)
|
|
|
|
{
|
|
|
|
size_t i, j;
|
|
|
|
virDomainCheckpointDefPtr olddef;
|
|
|
|
virDomainMomentObjPtr parent;
|
|
|
|
bool search_parents;
|
|
|
|
|
|
|
|
for (i = 0; i < def->ndisks; i++) {
|
|
|
|
virDomainCheckpointDiskDef *disk = &def->disks[i];
|
|
|
|
const char *node;
|
|
|
|
|
|
|
|
if (disk->type != VIR_DOMAIN_CHECKPOINT_TYPE_BITMAP)
|
|
|
|
continue;
|
|
|
|
node = qemuDomainDiskNodeFormatLookup(vm, disk->name);
|
2019-09-26 14:36:15 +00:00
|
|
|
if (qemuMonitorTransactionBitmapAdd(actions, node, disk->bitmap, true, false) < 0)
|
2019-09-19 14:49:21 +00:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
/* We only want one active bitmap for a disk along the
|
|
|
|
* checkpoint chain, then later differential backups will
|
|
|
|
* merge the bitmaps (only one active) between the bounding
|
|
|
|
* checkpoint and the leaf checkpoint. If the same disks are
|
|
|
|
* involved in each checkpoint, this search terminates in one
|
|
|
|
* iteration; but it is also possible to have to search
|
|
|
|
* further than the immediate parent to find another
|
|
|
|
* checkpoint with a bitmap on the same disk. */
|
|
|
|
search_parents = true;
|
|
|
|
for (parent = old_current; search_parents && parent;
|
|
|
|
parent = virDomainCheckpointFindByName(vm->checkpoints,
|
|
|
|
olddef->parent.parent_name)) {
|
|
|
|
olddef = virDomainCheckpointObjGetDef(parent);
|
|
|
|
for (j = 0; j < olddef->ndisks; j++) {
|
|
|
|
virDomainCheckpointDiskDef *disk2;
|
|
|
|
|
|
|
|
disk2 = &olddef->disks[j];
|
|
|
|
if (STRNEQ(disk->name, disk2->name) ||
|
|
|
|
disk2->type != VIR_DOMAIN_CHECKPOINT_TYPE_BITMAP)
|
|
|
|
continue;
|
2019-09-26 14:36:15 +00:00
|
|
|
if (qemuMonitorTransactionBitmapDisable(actions, node, disk2->bitmap) < 0)
|
2019-09-19 14:49:21 +00:00
|
|
|
return -1;
|
|
|
|
search_parents = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-10-01 13:04:33 +00:00
|
|
|
static virDomainMomentObjPtr
|
|
|
|
qemuCheckpointRedefine(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainCheckpointDefPtr *def,
|
|
|
|
bool *update_current)
|
|
|
|
{
|
|
|
|
virDomainMomentObjPtr chk = NULL;
|
|
|
|
|
|
|
|
if (virDomainCheckpointRedefinePrep(vm, def, &chk, driver->xmlopt,
|
|
|
|
update_current) < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
/* XXX Should we validate that the redefined checkpoint even
|
|
|
|
* makes sense, such as checking that qemu-img recognizes the
|
|
|
|
* checkpoint bitmap name in at least one of the domain's disks? */
|
|
|
|
|
|
|
|
if (chk)
|
|
|
|
return chk;
|
|
|
|
|
|
|
|
chk = virDomainCheckpointAssignDef(vm->checkpoints, *def);
|
|
|
|
*def = NULL;
|
|
|
|
return chk;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
qemuCheckpointCreateCommon(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainCheckpointDefPtr *def,
|
|
|
|
virJSONValuePtr *actions,
|
|
|
|
virDomainMomentObjPtr *chk)
|
|
|
|
{
|
|
|
|
g_autoptr(virJSONValue) tmpactions = NULL;
|
|
|
|
virDomainMomentObjPtr parent;
|
|
|
|
|
2019-11-27 13:19:09 +00:00
|
|
|
if (qemuCheckpointPrepare(driver, vm, *def) < 0)
|
2019-10-01 13:04:33 +00:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
if ((parent = virDomainCheckpointGetCurrent(vm->checkpoints)))
|
|
|
|
(*def)->parent.parent_name = g_strdup(parent->def->name);
|
|
|
|
|
|
|
|
if (!(tmpactions = virJSONValueNewArray()))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuCheckpointAddActions(vm, tmpactions, parent, *def) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (!(*chk = virDomainCheckpointAssignDef(vm->checkpoints, *def)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
*def = NULL;
|
|
|
|
|
|
|
|
*actions = g_steal_pointer(&tmpactions);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static virDomainMomentObjPtr
|
|
|
|
qemuCheckpointCreate(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainCheckpointDefPtr *def)
|
|
|
|
{
|
|
|
|
g_autoptr(virJSONValue) actions = NULL;
|
|
|
|
virDomainMomentObjPtr chk = NULL;
|
|
|
|
int rc;
|
|
|
|
|
2019-11-27 13:19:09 +00:00
|
|
|
if (qemuCheckpointCreateCommon(driver, vm, def, &actions, &chk) < 0)
|
2019-10-01 13:04:33 +00:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
|
|
|
rc = qemuMonitorTransaction(qemuDomainGetMonitor(vm), &actions);
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0 || rc < 0) {
|
|
|
|
virDomainCheckpointObjListRemove(vm->checkpoints, chk);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return chk;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-10-03 12:35:58 +00:00
|
|
|
int
|
|
|
|
qemuCheckpointCreateFinalize(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virQEMUDriverConfigPtr cfg,
|
|
|
|
virDomainMomentObjPtr chk,
|
|
|
|
bool update_current)
|
|
|
|
{
|
|
|
|
if (update_current)
|
|
|
|
virDomainCheckpointSetCurrent(vm->checkpoints, chk);
|
|
|
|
|
2019-11-27 13:19:09 +00:00
|
|
|
if (qemuCheckpointWriteMetadata(vm, chk,
|
2019-10-03 12:35:58 +00:00
|
|
|
driver->xmlopt,
|
|
|
|
cfg->checkpointDir) < 0) {
|
|
|
|
/* if writing of metadata fails, error out rather than trying
|
|
|
|
* to silently carry on without completing the checkpoint */
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("unable to save metadata for checkpoint %s"),
|
|
|
|
chk->def->name);
|
|
|
|
virDomainCheckpointObjListRemove(vm->checkpoints, chk);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
virDomainCheckpointLinkParent(vm->checkpoints, chk);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-09-19 14:49:21 +00:00
|
|
|
virDomainCheckpointPtr
|
|
|
|
qemuCheckpointCreateXML(virDomainPtr domain,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
const char *xmlDesc,
|
|
|
|
unsigned int flags)
|
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
virQEMUDriverPtr driver = priv->driver;
|
|
|
|
virDomainMomentObjPtr chk = NULL;
|
|
|
|
virDomainCheckpointPtr checkpoint = NULL;
|
|
|
|
bool update_current = true;
|
|
|
|
bool redefine = flags & VIR_DOMAIN_CHECKPOINT_CREATE_REDEFINE;
|
|
|
|
unsigned int parse_flags = 0;
|
2019-12-08 21:48:58 +00:00
|
|
|
g_autoptr(virQEMUDriverConfig) cfg = virQEMUDriverGetConfig(driver);
|
2019-10-15 12:47:50 +00:00
|
|
|
g_autoptr(virDomainCheckpointDef) def = NULL;
|
2019-09-19 14:49:21 +00:00
|
|
|
|
|
|
|
virCheckFlags(VIR_DOMAIN_CHECKPOINT_CREATE_REDEFINE, NULL);
|
|
|
|
|
|
|
|
if (redefine) {
|
|
|
|
parse_flags |= VIR_DOMAIN_CHECKPOINT_PARSE_REDEFINE;
|
|
|
|
update_current = false;
|
|
|
|
}
|
|
|
|
|
2019-09-26 11:25:40 +00:00
|
|
|
if (!virQEMUCapsGet(priv->qemuCaps, QEMU_CAPS_INCREMENTAL_BACKUP)) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
|
|
|
_("incremental backup is not supported yet"));
|
2019-09-26 11:19:40 +00:00
|
|
|
return NULL;
|
2019-09-19 14:49:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!virDomainObjIsActive(vm)) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
|
|
|
_("cannot create checkpoint for inactive domain"));
|
2019-09-26 11:19:40 +00:00
|
|
|
return NULL;
|
2019-09-19 14:49:21 +00:00
|
|
|
}
|
|
|
|
|
2019-11-27 13:10:21 +00:00
|
|
|
if (!(def = virDomainCheckpointDefParseString(xmlDesc, driver->xmlopt,
|
2019-09-19 14:49:21 +00:00
|
|
|
priv->qemuCaps, parse_flags)))
|
2019-09-26 11:19:40 +00:00
|
|
|
return NULL;
|
2019-09-19 14:49:21 +00:00
|
|
|
/* Unlike snapshots, the RNG schema already ensured a sane filename. */
|
|
|
|
|
|
|
|
/* We are going to modify the domain below. */
|
|
|
|
if (qemuDomainObjBeginJob(driver, vm, QEMU_JOB_MODIFY) < 0)
|
2019-09-26 11:19:40 +00:00
|
|
|
return NULL;
|
2019-09-19 14:49:21 +00:00
|
|
|
|
|
|
|
if (redefine) {
|
2019-10-01 13:04:33 +00:00
|
|
|
chk = qemuCheckpointRedefine(driver, vm, &def, &update_current);
|
|
|
|
} else {
|
2019-11-27 13:19:09 +00:00
|
|
|
chk = qemuCheckpointCreate(driver, vm, &def);
|
2019-09-19 14:49:21 +00:00
|
|
|
}
|
|
|
|
|
2019-10-01 13:04:33 +00:00
|
|
|
if (!chk)
|
|
|
|
goto endjob;
|
2019-09-19 14:49:21 +00:00
|
|
|
|
2019-10-03 12:35:58 +00:00
|
|
|
if (qemuCheckpointCreateFinalize(driver, vm, cfg, chk, update_current) < 0)
|
2019-10-01 13:04:33 +00:00
|
|
|
goto endjob;
|
2019-09-19 14:49:21 +00:00
|
|
|
|
|
|
|
/* If we fail after this point, there's not a whole lot we can do;
|
|
|
|
* we've successfully created the checkpoint, so we have to go
|
|
|
|
* forward the best we can.
|
|
|
|
*/
|
|
|
|
checkpoint = virGetDomainCheckpoint(domain, chk->def->name);
|
|
|
|
|
|
|
|
endjob:
|
|
|
|
qemuDomainObjEndJob(driver, vm);
|
|
|
|
|
|
|
|
return checkpoint;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
char *
|
|
|
|
qemuCheckpointGetXMLDesc(virDomainObjPtr vm,
|
|
|
|
virDomainCheckpointPtr checkpoint,
|
|
|
|
unsigned int flags)
|
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
virQEMUDriverPtr driver = priv->driver;
|
|
|
|
virDomainMomentObjPtr chk = NULL;
|
|
|
|
virDomainCheckpointDefPtr chkdef;
|
|
|
|
unsigned int format_flags;
|
|
|
|
|
|
|
|
virCheckFlags(VIR_DOMAIN_CHECKPOINT_XML_SECURE |
|
|
|
|
VIR_DOMAIN_CHECKPOINT_XML_NO_DOMAIN, NULL);
|
|
|
|
|
|
|
|
if (!(chk = qemuCheckpointObjFromCheckpoint(vm, checkpoint)))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
chkdef = virDomainCheckpointObjGetDef(chk);
|
|
|
|
|
|
|
|
format_flags = virDomainCheckpointFormatConvertXMLFlags(flags);
|
2019-11-27 13:10:21 +00:00
|
|
|
return virDomainCheckpointDefFormat(chkdef, driver->xmlopt,
|
2019-09-19 14:49:21 +00:00
|
|
|
format_flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
struct virQEMUCheckpointReparent {
|
|
|
|
const char *dir;
|
|
|
|
virDomainMomentObjPtr parent;
|
|
|
|
virDomainObjPtr vm;
|
|
|
|
virDomainXMLOptionPtr xmlopt;
|
|
|
|
int err;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
qemuCheckpointReparentChildren(void *payload,
|
2019-10-14 12:45:33 +00:00
|
|
|
const void *name G_GNUC_UNUSED,
|
2019-09-19 14:49:21 +00:00
|
|
|
void *data)
|
|
|
|
{
|
|
|
|
virDomainMomentObjPtr moment = payload;
|
|
|
|
struct virQEMUCheckpointReparent *rep = data;
|
|
|
|
|
|
|
|
if (rep->err < 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
VIR_FREE(moment->def->parent_name);
|
|
|
|
|
2019-10-20 11:49:46 +00:00
|
|
|
if (rep->parent->def)
|
|
|
|
moment->def->parent_name = g_strdup(rep->parent->def->name);
|
2019-09-19 14:49:21 +00:00
|
|
|
|
2019-11-27 13:19:09 +00:00
|
|
|
rep->err = qemuCheckpointWriteMetadata(rep->vm, moment,
|
2019-09-20 11:47:04 +00:00
|
|
|
rep->xmlopt, rep->dir);
|
2019-09-19 14:49:21 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
qemuCheckpointDelete(virDomainObjPtr vm,
|
|
|
|
virDomainCheckpointPtr checkpoint,
|
|
|
|
unsigned int flags)
|
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
virQEMUDriverPtr driver = priv->driver;
|
2019-10-15 12:47:50 +00:00
|
|
|
g_autoptr(virQEMUDriverConfig) cfg = virQEMUDriverGetConfig(driver);
|
2019-09-19 14:49:21 +00:00
|
|
|
int ret = -1;
|
|
|
|
virDomainMomentObjPtr chk = NULL;
|
|
|
|
virQEMUMomentRemove rem;
|
|
|
|
struct virQEMUCheckpointReparent rep;
|
|
|
|
bool metadata_only = !!(flags & VIR_DOMAIN_CHECKPOINT_DELETE_METADATA_ONLY);
|
|
|
|
|
|
|
|
virCheckFlags(VIR_DOMAIN_CHECKPOINT_DELETE_CHILDREN |
|
|
|
|
VIR_DOMAIN_CHECKPOINT_DELETE_METADATA_ONLY |
|
|
|
|
VIR_DOMAIN_CHECKPOINT_DELETE_CHILDREN_ONLY, -1);
|
|
|
|
|
|
|
|
if (qemuDomainObjBeginJob(driver, vm, QEMU_JOB_MODIFY) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (!metadata_only) {
|
2019-09-26 11:25:40 +00:00
|
|
|
if (!virQEMUCapsGet(priv->qemuCaps, QEMU_CAPS_INCREMENTAL_BACKUP)) {
|
2019-09-19 14:49:21 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
2019-09-26 11:25:40 +00:00
|
|
|
_("incremental backup is not supported yet"));
|
2019-09-19 14:49:21 +00:00
|
|
|
goto endjob;
|
|
|
|
}
|
2019-09-26 11:25:40 +00:00
|
|
|
|
|
|
|
if (!virDomainObjIsActive(vm)) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
|
|
|
_("cannot delete checkpoint for inactive domain"));
|
2019-09-19 14:49:21 +00:00
|
|
|
goto endjob;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(chk = qemuCheckpointObjFromCheckpoint(vm, checkpoint)))
|
|
|
|
goto endjob;
|
|
|
|
|
|
|
|
if (flags & (VIR_DOMAIN_CHECKPOINT_DELETE_CHILDREN |
|
|
|
|
VIR_DOMAIN_CHECKPOINT_DELETE_CHILDREN_ONLY)) {
|
|
|
|
rem.driver = driver;
|
|
|
|
rem.vm = vm;
|
|
|
|
rem.metadata_only = metadata_only;
|
|
|
|
rem.err = 0;
|
|
|
|
rem.current = virDomainCheckpointGetCurrent(vm->checkpoints);
|
|
|
|
rem.found = false;
|
2019-09-20 11:47:04 +00:00
|
|
|
rem.momentDiscard = qemuCheckpointDiscard;
|
2019-09-19 14:49:21 +00:00
|
|
|
virDomainMomentForEachDescendant(chk, qemuDomainMomentDiscardAll,
|
|
|
|
&rem);
|
|
|
|
if (rem.err < 0)
|
|
|
|
goto endjob;
|
|
|
|
if (rem.found) {
|
|
|
|
virDomainCheckpointSetCurrent(vm->checkpoints, chk);
|
|
|
|
if (flags & VIR_DOMAIN_CHECKPOINT_DELETE_CHILDREN_ONLY) {
|
2019-11-27 13:19:09 +00:00
|
|
|
if (qemuCheckpointWriteMetadata(vm, chk,
|
2019-09-20 11:47:04 +00:00
|
|
|
driver->xmlopt,
|
|
|
|
cfg->checkpointDir) < 0) {
|
2019-09-19 14:49:21 +00:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("failed to set checkpoint '%s' as current"),
|
|
|
|
chk->def->name);
|
|
|
|
virDomainCheckpointSetCurrent(vm->checkpoints, NULL);
|
|
|
|
goto endjob;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (chk->nchildren) {
|
|
|
|
rep.dir = cfg->checkpointDir;
|
|
|
|
rep.parent = chk->parent;
|
|
|
|
rep.vm = vm;
|
|
|
|
rep.err = 0;
|
|
|
|
rep.xmlopt = driver->xmlopt;
|
|
|
|
virDomainMomentForEachChild(chk, qemuCheckpointReparentChildren,
|
|
|
|
&rep);
|
|
|
|
if (rep.err < 0)
|
|
|
|
goto endjob;
|
|
|
|
virDomainMomentMoveChildren(chk, chk->parent);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (flags & VIR_DOMAIN_CHECKPOINT_DELETE_CHILDREN_ONLY) {
|
|
|
|
virDomainMomentDropChildren(chk);
|
|
|
|
ret = 0;
|
|
|
|
} else {
|
2019-09-20 11:47:04 +00:00
|
|
|
ret = qemuCheckpointDiscard(driver, vm, chk, true, metadata_only);
|
2019-09-19 14:49:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
endjob:
|
|
|
|
qemuDomainObjEndJob(driver, vm);
|
|
|
|
return ret;
|
|
|
|
}
|