virDomainSnapshotRedefineValidate: Don't modify the snapshot definition

It is not expected that a function with 'Validate' in the name actually
modifies the validated object, even worse when it even modifies another
object and the ultimatively worst bit is that it doesn't undo the mess
if the validation fails midway.

Move the stealing of the domain definition from the definition of a
snapshot being redefined into the caller along with the call to
virDomainSnapshotAlignDisks.

Signed-off-by: Peter Krempa <pkrempa@redhat.com>
Reviewed-by: Ján Tomko <jtomko@redhat.com>
This commit is contained in:
Peter Krempa 2022-01-12 15:08:25 +01:00
parent 504b108cb9
commit bec2a922bd
1 changed files with 24 additions and 32 deletions

View File

@ -463,9 +463,7 @@ virDomainSnapshotDefParseString(const char *xmlStr,
} }
/* Perform sanity checking on a redefined snapshot definition. If /* Perform sanity checking on a redefined snapshot definition. */
* @other is non-NULL, this may include swapping def->parent.dom from other
* into def. */
static int static int
virDomainSnapshotRedefineValidate(virDomainSnapshotDef *def, virDomainSnapshotRedefineValidate(virDomainSnapshotDef *def,
const unsigned char *domain_uuid, const unsigned char *domain_uuid,
@ -473,10 +471,6 @@ virDomainSnapshotRedefineValidate(virDomainSnapshotDef *def,
virDomainXMLOption *xmlopt, virDomainXMLOption *xmlopt,
unsigned int flags) unsigned int flags)
{ {
virDomainSnapshotLocation align_location = VIR_DOMAIN_SNAPSHOT_LOCATION_INTERNAL;
bool external = def->state == VIR_DOMAIN_SNAPSHOT_DISK_SNAPSHOT ||
virDomainSnapshotDefIsExternal(def);
if ((flags & VIR_DOMAIN_SNAPSHOT_CREATE_DISK_ONLY) && if ((flags & VIR_DOMAIN_SNAPSHOT_CREATE_DISK_ONLY) &&
def->state != VIR_DOMAIN_SNAPSHOT_DISK_SNAPSHOT) { def->state != VIR_DOMAIN_SNAPSHOT_DISK_SNAPSHOT) {
virReportError(VIR_ERR_INVALID_ARG, virReportError(VIR_ERR_INVALID_ARG,
@ -524,22 +518,10 @@ virDomainSnapshotRedefineValidate(virDomainSnapshotDef *def,
if (!virDomainDefCheckABIStability(otherdef->parent.dom, if (!virDomainDefCheckABIStability(otherdef->parent.dom,
def->parent.dom, xmlopt)) def->parent.dom, xmlopt))
return -1; return -1;
} else {
/* Transfer the domain def */
def->parent.dom = g_steal_pointer(&otherdef->parent.dom);
} }
} }
} }
if (def->parent.dom) {
if (external)
align_location = VIR_DOMAIN_SNAPSHOT_LOCATION_EXTERNAL;
if (virDomainSnapshotAlignDisks(def, NULL, align_location, true) < 0)
return -1;
}
return 0; return 0;
} }
@ -982,28 +964,38 @@ virDomainSnapshotRedefinePrep(virDomainObj *vm,
{ {
virDomainSnapshotDef *snapdef = *defptr; virDomainSnapshotDef *snapdef = *defptr;
virDomainMomentObj *other; virDomainMomentObj *other;
virDomainSnapshotDef *otherdef = NULL; virDomainSnapshotDef *otherSnapDef = NULL;
bool check_if_stolen; virDomainDef *otherDomDef = NULL;
virDomainSnapshotLocation align_location = VIR_DOMAIN_SNAPSHOT_LOCATION_INTERNAL;
if (virDomainSnapshotCheckCycles(vm->snapshots, snapdef, vm->def->name) < 0) if (virDomainSnapshotCheckCycles(vm->snapshots, snapdef, vm->def->name) < 0)
return -1; return -1;
other = virDomainSnapshotFindByName(vm->snapshots, snapdef->parent.name); if ((other = virDomainSnapshotFindByName(vm->snapshots, snapdef->parent.name))) {
if (other) otherSnapDef = virDomainSnapshotObjGetDef(other);
otherdef = virDomainSnapshotObjGetDef(other); otherDomDef = otherSnapDef->parent.dom;
check_if_stolen = other && otherdef->parent.dom;
if (virDomainSnapshotRedefineValidate(snapdef, vm->def->uuid, other, xmlopt,
flags) < 0) {
/* revert any stealing of the snapshot domain definition */
if (check_if_stolen && snapdef->parent.dom && !otherdef->parent.dom)
otherdef->parent.dom = g_steal_pointer(&snapdef->parent.dom);
return -1;
} }
if (virDomainSnapshotRedefineValidate(snapdef, vm->def->uuid, other, xmlopt, flags) < 0)
return -1;
if (snapdef->state == VIR_DOMAIN_SNAPSHOT_DISK_SNAPSHOT ||
virDomainSnapshotDefIsExternal(snapdef))
align_location = VIR_DOMAIN_SNAPSHOT_LOCATION_EXTERNAL;
if (virDomainSnapshotAlignDisks(snapdef, otherDomDef, align_location, true) < 0)
return -1;
if (other) { if (other) {
/* steal the domain definition if redefining an existing snapshot which
* with a snapshot definition lacking the domain definition */
if (!snapdef->parent.dom)
snapdef->parent.dom = g_steal_pointer(&otherSnapDef->parent.dom);
/* Drop and rebuild the parent relationship, but keep all /* Drop and rebuild the parent relationship, but keep all
* child relations by reusing snap. */ * child relations by reusing snap. */
virDomainMomentDropParent(other); virDomainMomentDropParent(other);
virObjectUnref(otherdef); virObjectUnref(otherSnapDef);
other->def = &(*defptr)->parent; other->def = &(*defptr)->parent;
*defptr = NULL; *defptr = NULL;
*snap = other; *snap = other;