mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-08 22:15:21 +00:00
qemu_snapshot: implement deletion of external snapshot
When deleting snapshot we are starting block-commit job over all disks that are part of the snapshot. This operation may fail as it writes data changes to the backing qcow2 image so we need to wait for all the disks to finish the operation and wait for correct signal from QEMU. If deleting active snapshot we will get `ready` signal and for inactive snapshots we need to disable autofinalize in order to get `pending` signal. At this point if commit for any disk fails for some reason and we abort the VM is still in consistent state and user can fix the reason why the deletion failed. After that we do `pivot` or `finalize` if it's active snapshot or not to finish the block job. It still may fail but there is nothing else we can do about it. Signed-off-by: Pavel Hrdina <phrdina@redhat.com> Reviewed-by: Peter Krempa <pkrempa@redhat.com>
This commit is contained in:
parent
4a4d89a925
commit
e27f081967
@ -2410,6 +2410,199 @@ qemuSnapshotChildrenReparent(void *payload,
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* Deleting external snapshot is started by running qemu block-commit job.
|
||||||
|
* We need to wait for all block-commit jobs to be 'ready' or 'pending' to
|
||||||
|
* continue with external snapshot deletion. */
|
||||||
|
static int
|
||||||
|
qemuSnapshotDeleteBlockJobIsRunning(qemuBlockjobState state)
|
||||||
|
{
|
||||||
|
switch (state) {
|
||||||
|
case QEMU_BLOCKJOB_STATE_NEW:
|
||||||
|
case QEMU_BLOCKJOB_STATE_RUNNING:
|
||||||
|
case QEMU_BLOCKJOB_STATE_ABORTING:
|
||||||
|
case QEMU_BLOCKJOB_STATE_PIVOTING:
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
case QEMU_BLOCKJOB_STATE_COMPLETED:
|
||||||
|
case QEMU_BLOCKJOB_STATE_FAILED:
|
||||||
|
case QEMU_BLOCKJOB_STATE_CANCELLED:
|
||||||
|
case QEMU_BLOCKJOB_STATE_READY:
|
||||||
|
case QEMU_BLOCKJOB_STATE_CONCLUDED:
|
||||||
|
case QEMU_BLOCKJOB_STATE_PENDING:
|
||||||
|
case QEMU_BLOCKJOB_STATE_LAST:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* When finishing or aborting qemu blockjob we only need to know if the
|
||||||
|
* job is still active or not. */
|
||||||
|
static int
|
||||||
|
qemuSnapshotDeleteBlockJobIsActive(qemuBlockjobState state)
|
||||||
|
{
|
||||||
|
switch (state) {
|
||||||
|
case QEMU_BLOCKJOB_STATE_READY:
|
||||||
|
case QEMU_BLOCKJOB_STATE_NEW:
|
||||||
|
case QEMU_BLOCKJOB_STATE_RUNNING:
|
||||||
|
case QEMU_BLOCKJOB_STATE_ABORTING:
|
||||||
|
case QEMU_BLOCKJOB_STATE_PENDING:
|
||||||
|
case QEMU_BLOCKJOB_STATE_PIVOTING:
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
case QEMU_BLOCKJOB_STATE_COMPLETED:
|
||||||
|
case QEMU_BLOCKJOB_STATE_FAILED:
|
||||||
|
case QEMU_BLOCKJOB_STATE_CANCELLED:
|
||||||
|
case QEMU_BLOCKJOB_STATE_CONCLUDED:
|
||||||
|
case QEMU_BLOCKJOB_STATE_LAST:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* Wait for qemu blockjob to finish 'block-commit' operation until it is
|
||||||
|
* ready to be finished by calling 'block-pivot' or 'block-finalize'. */
|
||||||
|
static int
|
||||||
|
qemuSnapshotDeleteBlockJobRunning(virDomainObj *vm,
|
||||||
|
qemuBlockJobData *job)
|
||||||
|
{
|
||||||
|
int rc;
|
||||||
|
qemuBlockJobUpdate(vm, job, VIR_ASYNC_JOB_SNAPSHOT);
|
||||||
|
|
||||||
|
while ((rc = qemuSnapshotDeleteBlockJobIsRunning(job->state)) > 0) {
|
||||||
|
if (qemuDomainObjWait(vm) < 0)
|
||||||
|
return -1;
|
||||||
|
qemuBlockJobUpdate(vm, job, VIR_ASYNC_JOB_SNAPSHOT);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (rc < 0)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* Wait for qemu blockjob to be done after 'block-pivot' or 'block-finalize'
|
||||||
|
* was started. */
|
||||||
|
static int
|
||||||
|
qemuSnapshotDeleteBlockJobFinishing(virDomainObj *vm,
|
||||||
|
qemuBlockJobData *job)
|
||||||
|
{
|
||||||
|
int rc;
|
||||||
|
qemuBlockJobUpdate(vm, job, VIR_ASYNC_JOB_SNAPSHOT);
|
||||||
|
|
||||||
|
while ((rc = qemuSnapshotDeleteBlockJobIsActive(job->state)) > 0) {
|
||||||
|
if (qemuDomainObjWait(vm) < 0)
|
||||||
|
return -1;
|
||||||
|
qemuBlockJobUpdate(vm, job, VIR_ASYNC_JOB_SNAPSHOT);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (rc < 0)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int
|
||||||
|
qemuSnapshotDiscardExternal(virDomainObj *vm,
|
||||||
|
GSList *externalData)
|
||||||
|
{
|
||||||
|
GSList *cur = NULL;
|
||||||
|
|
||||||
|
for (cur = externalData; cur; cur = g_slist_next(cur)) {
|
||||||
|
qemuSnapshotDeleteExternalData *data = cur->data;
|
||||||
|
virTristateBool autofinalize = VIR_TRISTATE_BOOL_NO;
|
||||||
|
unsigned int commitFlags = VIR_DOMAIN_BLOCK_COMMIT_DELETE;
|
||||||
|
|
||||||
|
if (data->domDisk->src == data->diskSrc) {
|
||||||
|
commitFlags |= VIR_DOMAIN_BLOCK_COMMIT_ACTIVE;
|
||||||
|
autofinalize = VIR_TRISTATE_BOOL_YES;
|
||||||
|
}
|
||||||
|
|
||||||
|
data->job = qemuBlockCommit(vm,
|
||||||
|
data->domDisk,
|
||||||
|
data->parentDiskSrc,
|
||||||
|
data->diskSrc,
|
||||||
|
data->prevDiskSrc,
|
||||||
|
0,
|
||||||
|
VIR_ASYNC_JOB_SNAPSHOT,
|
||||||
|
autofinalize,
|
||||||
|
commitFlags);
|
||||||
|
|
||||||
|
if (!data->job)
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (cur = externalData; cur; cur = g_slist_next(cur)) {
|
||||||
|
qemuSnapshotDeleteExternalData *data = cur->data;
|
||||||
|
|
||||||
|
if (qemuSnapshotDeleteBlockJobRunning(vm, data->job) < 0)
|
||||||
|
goto error;
|
||||||
|
|
||||||
|
if (data->job->state == QEMU_BLOCKJOB_STATE_FAILED) {
|
||||||
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
||||||
|
_("block commit failed while deleting disk '%s' snapshot: '%s'"),
|
||||||
|
data->snapDisk->name, data->job->errmsg);
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (cur = externalData; cur; cur = g_slist_next(cur)) {
|
||||||
|
qemuSnapshotDeleteExternalData *data = cur->data;
|
||||||
|
|
||||||
|
if (data->job->state == QEMU_BLOCKJOB_STATE_READY) {
|
||||||
|
if (qemuBlockPivot(vm, data->job, VIR_ASYNC_JOB_SNAPSHOT, NULL) < 0)
|
||||||
|
goto error;
|
||||||
|
} else if (data->job->state == QEMU_BLOCKJOB_STATE_PENDING) {
|
||||||
|
if (qemuBlockFinalize(vm, data->job, VIR_ASYNC_JOB_SNAPSHOT) < 0)
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (qemuSnapshotDeleteBlockJobFinishing(vm, data->job) < 0)
|
||||||
|
goto error;
|
||||||
|
|
||||||
|
if (data->job->state == QEMU_BLOCKJOB_STATE_FAILED) {
|
||||||
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
||||||
|
_("finishing block job failed while deleting disk '%s' snapshot: '%s'"),
|
||||||
|
data->snapDisk->name, data->job->errmsg);
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
|
||||||
|
qemuBlockJobSyncEnd(vm, data->job, VIR_ASYNC_JOB_SNAPSHOT);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
error:
|
||||||
|
for (cur = externalData; cur; cur = g_slist_next(cur)) {
|
||||||
|
qemuDomainObjPrivate *priv = vm->privateData;
|
||||||
|
qemuSnapshotDeleteExternalData *data = cur->data;
|
||||||
|
|
||||||
|
if (!data->job)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
qemuBlockJobUpdate(vm, data->job, VIR_ASYNC_JOB_SNAPSHOT);
|
||||||
|
|
||||||
|
if (qemuSnapshotDeleteBlockJobIsActive(data->job->state)) {
|
||||||
|
if (qemuDomainObjEnterMonitorAsync(vm, VIR_ASYNC_JOB_SNAPSHOT) == 0) {
|
||||||
|
ignore_value(qemuMonitorBlockJobCancel(priv->mon, data->job->name, false));
|
||||||
|
qemuDomainObjExitMonitor(vm);
|
||||||
|
|
||||||
|
data->job->state = QEMU_BLOCKJOB_STATE_ABORTING;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
qemuBlockJobSyncEnd(vm, data->job, VIR_ASYNC_JOB_SNAPSHOT);
|
||||||
|
}
|
||||||
|
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
static int
|
static int
|
||||||
qemuSnapshotDiscardMetadata(virDomainObj *vm,
|
qemuSnapshotDiscardMetadata(virDomainObj *vm,
|
||||||
virDomainMomentObj *snap,
|
virDomainMomentObj *snap,
|
||||||
@ -2476,11 +2669,12 @@ qemuSnapshotDiscardMetadata(virDomainObj *vm,
|
|||||||
|
|
||||||
/* Discard one snapshot (or its metadata), without reparenting any children. */
|
/* Discard one snapshot (or its metadata), without reparenting any children. */
|
||||||
static int
|
static int
|
||||||
qemuSnapshotDiscard(virQEMUDriver *driver,
|
qemuSnapshotDiscardImpl(virQEMUDriver *driver,
|
||||||
virDomainObj *vm,
|
virDomainObj *vm,
|
||||||
virDomainMomentObj *snap,
|
virDomainMomentObj *snap,
|
||||||
bool update_parent,
|
GSList *externalData,
|
||||||
bool metadata_only)
|
bool update_parent,
|
||||||
|
bool metadata_only)
|
||||||
{
|
{
|
||||||
if (!metadata_only) {
|
if (!metadata_only) {
|
||||||
if (!virDomainObjIsActive(vm)) {
|
if (!virDomainObjIsActive(vm)) {
|
||||||
@ -2500,18 +2694,28 @@ qemuSnapshotDiscard(virQEMUDriver *driver,
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (qemuDomainSnapshotForEachQcow2(driver, def, snap, "-d", true) < 0)
|
if (virDomainSnapshotIsExternal(snap)) {
|
||||||
return -1;
|
if (qemuSnapshotDiscardExternal(vm, externalData) < 0)
|
||||||
|
return -1;
|
||||||
|
} else {
|
||||||
|
if (qemuDomainSnapshotForEachQcow2(driver, def, snap, "-d", true) < 0)
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
/* Similarly as internal snapshot creation we would use a regular job
|
if (virDomainSnapshotIsExternal(snap)) {
|
||||||
* here so set a mask to forbid any other job. */
|
if (qemuSnapshotDiscardExternal(vm, externalData) < 0)
|
||||||
qemuDomainObjSetAsyncJobMask(vm, VIR_JOB_NONE);
|
return -1;
|
||||||
if (qemuDomainObjEnterMonitorAsync(vm, VIR_ASYNC_JOB_SNAPSHOT) < 0)
|
} else {
|
||||||
return -1;
|
/* Similarly as internal snapshot creation we would use a regular job
|
||||||
/* we continue on even in the face of error */
|
* here so set a mask to forbid any other job. */
|
||||||
qemuMonitorDeleteSnapshot(qemuDomainGetMonitor(vm), snap->def->name);
|
qemuDomainObjSetAsyncJobMask(vm, VIR_JOB_NONE);
|
||||||
qemuDomainObjExitMonitor(vm);
|
if (qemuDomainObjEnterMonitorAsync(vm, VIR_ASYNC_JOB_SNAPSHOT) < 0)
|
||||||
qemuDomainObjSetAsyncJobMask(vm, VIR_JOB_DEFAULT_MASK);
|
return -1;
|
||||||
|
/* we continue on even in the face of error */
|
||||||
|
qemuMonitorDeleteSnapshot(qemuDomainGetMonitor(vm), snap->def->name);
|
||||||
|
qemuDomainObjExitMonitor(vm);
|
||||||
|
qemuDomainObjSetAsyncJobMask(vm, VIR_JOB_DEFAULT_MASK);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2522,6 +2726,17 @@ qemuSnapshotDiscard(virQEMUDriver *driver,
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int
|
||||||
|
qemuSnapshotDiscard(virQEMUDriver *driver,
|
||||||
|
virDomainObj *vm,
|
||||||
|
virDomainMomentObj *snap,
|
||||||
|
bool update_parent,
|
||||||
|
bool metadata_only)
|
||||||
|
{
|
||||||
|
return qemuSnapshotDiscardImpl(driver, vm, snap, NULL, update_parent, metadata_only);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
int
|
int
|
||||||
qemuSnapshotDiscardAllMetadata(virQEMUDriver *driver,
|
qemuSnapshotDiscardAllMetadata(virQEMUDriver *driver,
|
||||||
virDomainObj *vm)
|
virDomainObj *vm)
|
||||||
@ -2543,12 +2758,13 @@ qemuSnapshotDiscardAllMetadata(virQEMUDriver *driver,
|
|||||||
static int
|
static int
|
||||||
qemuSnapshotDeleteSingle(virDomainObj *vm,
|
qemuSnapshotDeleteSingle(virDomainObj *vm,
|
||||||
virDomainMomentObj *snap,
|
virDomainMomentObj *snap,
|
||||||
|
GSList *externalData,
|
||||||
bool metadata_only)
|
bool metadata_only)
|
||||||
{
|
{
|
||||||
qemuDomainObjPrivate *priv = vm->privateData;
|
qemuDomainObjPrivate *priv = vm->privateData;
|
||||||
virQEMUDriver *driver = priv->driver;
|
virQEMUDriver *driver = priv->driver;
|
||||||
|
|
||||||
return qemuSnapshotDiscard(driver, vm, snap, true, metadata_only);
|
return qemuSnapshotDiscardImpl(driver, vm, snap, externalData, true, metadata_only);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -2568,7 +2784,7 @@ qemuSnapshotDeleteAllHelper(void *payload,
|
|||||||
virDomainMomentObj *snap = payload;
|
virDomainMomentObj *snap = payload;
|
||||||
struct qemuSnapshotDeleteAllData *data = opaque;
|
struct qemuSnapshotDeleteAllData *data = opaque;
|
||||||
|
|
||||||
error = qemuSnapshotDeleteSingle(data->vm, snap, data->metadata_only);
|
error = qemuSnapshotDeleteSingle(data->vm, snap, NULL, data->metadata_only);
|
||||||
|
|
||||||
if (error != 0)
|
if (error != 0)
|
||||||
data->error = error;
|
data->error = error;
|
||||||
@ -2608,7 +2824,7 @@ qemuSnapshotDeleteChildren(virDomainObj *vm,
|
|||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
if (!children_only &&
|
if (!children_only &&
|
||||||
qemuSnapshotDeleteSingle(vm, snap, metadata_only) < 0) {
|
qemuSnapshotDeleteSingle(vm, snap, NULL, metadata_only) < 0) {
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2763,7 +2979,7 @@ qemuSnapshotDelete(virDomainObj *vm,
|
|||||||
bool children_only = !!(flags & VIR_DOMAIN_SNAPSHOT_DELETE_CHILDREN_ONLY);
|
bool children_only = !!(flags & VIR_DOMAIN_SNAPSHOT_DELETE_CHILDREN_ONLY);
|
||||||
ret = qemuSnapshotDeleteChildren(vm, snap, metadata_only, children_only);
|
ret = qemuSnapshotDeleteChildren(vm, snap, metadata_only, children_only);
|
||||||
} else {
|
} else {
|
||||||
ret = qemuSnapshotDeleteSingle(vm, snap, metadata_only);
|
ret = qemuSnapshotDeleteSingle(vm, snap, externalData, metadata_only);
|
||||||
}
|
}
|
||||||
|
|
||||||
endjob:
|
endjob:
|
||||||
|
Loading…
Reference in New Issue
Block a user