/* * qemu_blockjob.c: helper functions for QEMU block jobs * * Copyright (C) 2006-2015 Red Hat, Inc. * Copyright (C) 2006 Daniel P. Berrange * * 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 * . */ #include #include "internal.h" #include "qemu_blockjob.h" #include "qemu_block.h" #include "qemu_domain.h" #include "qemu_alias.h" #include "qemu_backup.h" #include "conf/domain_conf.h" #include "conf/domain_event.h" #include "virlog.h" #include "virstoragefile.h" #include "virthread.h" #include "virtime.h" #include "locking/domain_lock.h" #include "viralloc.h" #include "virstring.h" #include "qemu_security.h" #define VIR_FROM_THIS VIR_FROM_QEMU VIR_LOG_INIT("qemu.qemu_blockjob"); /* Note that qemuBlockjobState and qemuBlockjobType values are formatted into * the status XML */ VIR_ENUM_IMPL(qemuBlockjobState, QEMU_BLOCKJOB_STATE_LAST, "completed", "failed", "cancelled", "ready", "new", "running", "concluded", "aborting", "pivoting"); VIR_ENUM_IMPL(qemuBlockjob, QEMU_BLOCKJOB_TYPE_LAST, "", "pull", "copy", "commit", "active-commit", "backup", "", "create", "broken"); static virClassPtr qemuBlockJobDataClass; static void qemuBlockJobDataDisposeJobdata(qemuBlockJobDataPtr job) { if (job->type == QEMU_BLOCKJOB_TYPE_CREATE) virObjectUnref(job->data.create.src); if (job->type == QEMU_BLOCKJOB_TYPE_BACKUP) { virObjectUnref(job->data.backup.store); g_free(job->data.backup.bitmap); } } static void qemuBlockJobDataDispose(void *obj) { qemuBlockJobDataPtr job = obj; virObjectUnref(job->chain); virObjectUnref(job->mirrorChain); qemuBlockJobDataDisposeJobdata(job); g_free(job->name); g_free(job->errmsg); } static int qemuBlockJobDataOnceInit(void) { if (!VIR_CLASS_NEW(qemuBlockJobData, virClassForObject())) return -1; return 0; } VIR_ONCE_GLOBAL_INIT(qemuBlockJobData); qemuBlockJobDataPtr qemuBlockJobDataNew(qemuBlockJobType type, const char *name) { g_autoptr(qemuBlockJobData) job = NULL; if (qemuBlockJobDataInitialize() < 0) return NULL; if (!(job = virObjectNew(qemuBlockJobDataClass))) return NULL; job->name = g_strdup(name); job->state = QEMU_BLOCKJOB_STATE_NEW; job->newstate = -1; job->type = type; return g_steal_pointer(&job); } /** * qemuBlockJobMarkBroken: * @job: job to mark as broken * * In case when we are unable to parse the block job data from the XML * successfully we'll need to mark the job as broken and then attempt to abort * it. This function marks the job as broken. */ static void qemuBlockJobMarkBroken(qemuBlockJobDataPtr job) { qemuBlockJobDataDisposeJobdata(job); job->brokentype = job->type; job->type = QEMU_BLOCKJOB_TYPE_BROKEN; } /** * qemuBlockJobRegister: * @job: job to register * @vm: domain to register @job with * @disk: disk to register @job with * @savestatus: save the status XML after registering * * This function registers @job with @disk and @vm and records it into the status * xml (if @savestatus is true). * * Note that if @job also references a separate chain e.g. for disk mirroring, * then job->mirrorchain needs to be set manually. */ int qemuBlockJobRegister(qemuBlockJobDataPtr job, virDomainObjPtr vm, virDomainDiskDefPtr disk, bool savestatus) { qemuDomainObjPrivatePtr priv = vm->privateData; if (disk && QEMU_DOMAIN_DISK_PRIVATE(disk)->blockjob) { virReportError(VIR_ERR_INTERNAL_ERROR, _("disk '%s' has a blockjob assigned"), disk->dst); return -1; } if (virHashAddEntry(priv->blockjobs, job->name, virObjectRef(job)) < 0) { virObjectUnref(job); return -1; } if (disk) { job->disk = disk; job->chain = virObjectRef(disk->src); QEMU_DOMAIN_DISK_PRIVATE(disk)->blockjob = virObjectRef(job); } if (savestatus) qemuDomainSaveStatus(vm); return 0; } static void qemuBlockJobUnregister(qemuBlockJobDataPtr job, virDomainObjPtr vm) { qemuDomainObjPrivatePtr priv = vm->privateData; qemuDomainDiskPrivatePtr diskPriv; if (job->disk) { diskPriv = QEMU_DOMAIN_DISK_PRIVATE(job->disk); if (job == diskPriv->blockjob) { virObjectUnref(diskPriv->blockjob); diskPriv->blockjob = NULL; } job->disk = NULL; } /* this may remove the last reference of 'job' */ virHashRemoveEntry(priv->blockjobs, job->name); qemuDomainSaveStatus(vm); } /** * qemuBlockJobDiskNew: * @disk: disk definition * * Start/associate a new blockjob with @disk. * * Returns 0 on success and -1 on failure. */ qemuBlockJobDataPtr qemuBlockJobDiskNew(virDomainObjPtr vm, virDomainDiskDefPtr disk, qemuBlockJobType type, const char *jobname) { g_autoptr(qemuBlockJobData) job = NULL; if (!(job = qemuBlockJobDataNew(type, jobname))) return NULL; if (qemuBlockJobRegister(job, vm, disk, true) < 0) return NULL; return g_steal_pointer(&job); } qemuBlockJobDataPtr qemuBlockJobDiskNewPull(virDomainObjPtr vm, virDomainDiskDefPtr disk, virStorageSourcePtr base) { qemuDomainObjPrivatePtr priv = vm->privateData; g_autoptr(qemuBlockJobData) job = NULL; g_autofree char *jobname = NULL; if (virQEMUCapsGet(priv->qemuCaps, QEMU_CAPS_BLOCKDEV)) { jobname = g_strdup_printf("pull-%s-%s", disk->dst, disk->src->nodeformat); } else { if (!(jobname = qemuAliasDiskDriveFromDisk(disk))) return NULL; } if (!(job = qemuBlockJobDataNew(QEMU_BLOCKJOB_TYPE_PULL, jobname))) return NULL; job->data.pull.base = base; if (qemuBlockJobRegister(job, vm, disk, true) < 0) return NULL; return g_steal_pointer(&job); } qemuBlockJobDataPtr qemuBlockJobDiskNewCommit(virDomainObjPtr vm, virDomainDiskDefPtr disk, virStorageSourcePtr topparent, virStorageSourcePtr top, virStorageSourcePtr base, bool delete_imgs) { qemuDomainObjPrivatePtr priv = vm->privateData; g_autoptr(qemuBlockJobData) job = NULL; g_autofree char *jobname = NULL; qemuBlockJobType jobtype = QEMU_BLOCKJOB_TYPE_COMMIT; if (topparent == NULL) jobtype = QEMU_BLOCKJOB_TYPE_ACTIVE_COMMIT; if (virQEMUCapsGet(priv->qemuCaps, QEMU_CAPS_BLOCKDEV)) { jobname = g_strdup_printf("commit-%s-%s", disk->dst, top->nodeformat); } else { if (!(jobname = qemuAliasDiskDriveFromDisk(disk))) return NULL; } if (!(job = qemuBlockJobDataNew(jobtype, jobname))) return NULL; job->data.commit.topparent = topparent; job->data.commit.top = top; job->data.commit.base = base; job->data.commit.deleteCommittedImages = delete_imgs; if (qemuBlockJobRegister(job, vm, disk, true) < 0) return NULL; return g_steal_pointer(&job); } qemuBlockJobDataPtr qemuBlockJobNewCreate(virDomainObjPtr vm, virStorageSourcePtr src, virStorageSourcePtr chain, bool storage) { g_autoptr(qemuBlockJobData) job = NULL; g_autofree char *jobname = NULL; const char *nodename = src->nodeformat; if (storage) nodename = src->nodestorage; jobname = g_strdup_printf("create-%s", nodename); if (!(job = qemuBlockJobDataNew(QEMU_BLOCKJOB_TYPE_CREATE, jobname))) return NULL; if (virStorageSourceIsBacking(chain)) job->chain = virObjectRef(chain); job->data.create.src = virObjectRef(src); if (qemuBlockJobRegister(job, vm, NULL, true) < 0) return NULL; return g_steal_pointer(&job); } qemuBlockJobDataPtr qemuBlockJobDiskNewCopy(virDomainObjPtr vm, virDomainDiskDefPtr disk, virStorageSourcePtr mirror, bool shallow, bool reuse) { qemuDomainObjPrivatePtr priv = vm->privateData; g_autoptr(qemuBlockJobData) job = NULL; g_autofree char *jobname = NULL; if (virQEMUCapsGet(priv->qemuCaps, QEMU_CAPS_BLOCKDEV)) { jobname = g_strdup_printf("copy-%s-%s", disk->dst, disk->src->nodeformat); } else { if (!(jobname = qemuAliasDiskDriveFromDisk(disk))) return NULL; } if (!(job = qemuBlockJobDataNew(QEMU_BLOCKJOB_TYPE_COPY, jobname))) return NULL; job->mirrorChain = virObjectRef(mirror); if (shallow && !reuse) job->data.copy.shallownew = true; if (qemuBlockJobRegister(job, vm, disk, true) < 0) return NULL; return g_steal_pointer(&job); } qemuBlockJobDataPtr qemuBlockJobDiskNewBackup(virDomainObjPtr vm, virDomainDiskDefPtr disk, virStorageSourcePtr store, bool deleteStore, const char *bitmap) { g_autoptr(qemuBlockJobData) job = NULL; g_autofree char *jobname = NULL; jobname = g_strdup_printf("backup-%s-%s", disk->dst, disk->src->nodeformat); if (!(job = qemuBlockJobDataNew(QEMU_BLOCKJOB_TYPE_BACKUP, jobname))) return NULL; job->data.backup.bitmap = g_strdup(bitmap); job->data.backup.store = virObjectRef(store); job->data.backup.deleteStore = deleteStore; /* backup jobs are usually started in bulk by transaction so the caller * shall save the status XML */ if (qemuBlockJobRegister(job, vm, disk, false) < 0) return NULL; return g_steal_pointer(&job); } /** * qemuBlockJobDiskGetJob: * @disk: disk definition * * Get a reference to the block job data object associated with @disk. */ qemuBlockJobDataPtr qemuBlockJobDiskGetJob(virDomainDiskDefPtr disk) { qemuBlockJobDataPtr job = QEMU_DOMAIN_DISK_PRIVATE(disk)->blockjob; if (!job) return NULL; return virObjectRef(job); } /** * qemuBlockJobStarted: * @job: job data * * Mark @job as started in qemu. */ void qemuBlockJobStarted(qemuBlockJobDataPtr job, virDomainObjPtr vm) { if (job->state == QEMU_BLOCKJOB_STATE_NEW) job->state = QEMU_BLOCKJOB_STATE_RUNNING; qemuDomainSaveStatus(vm); } /** * qemuBlockJobStartupFinalize: * @job: job being started * * Cancels and clears the job private data if the job was not started with * qemu (see qemuBlockJobStarted) or just clears up the local reference * to @job if it was started. */ void qemuBlockJobStartupFinalize(virDomainObjPtr vm, qemuBlockJobDataPtr job) { if (!job) return; if (job->state == QEMU_BLOCKJOB_STATE_NEW) qemuBlockJobUnregister(job, vm); virObjectUnref(job); } bool qemuBlockJobIsRunning(qemuBlockJobDataPtr job) { return job->state == QEMU_BLOCKJOB_STATE_RUNNING || job->state == QEMU_BLOCKJOB_STATE_READY || job->state == QEMU_BLOCKJOB_STATE_ABORTING || job->state == QEMU_BLOCKJOB_STATE_PIVOTING; } /* returns 1 for a job we didn't reconnect to */ static int qemuBlockJobRefreshJobsFindInactive(const void *payload, const void *name G_GNUC_UNUSED, const void *data G_GNUC_UNUSED) { const qemuBlockJobData *job = payload; return !job->reconnected; } int qemuBlockJobRefreshJobs(virQEMUDriverPtr driver, virDomainObjPtr vm) { qemuDomainObjPrivatePtr priv = vm->privateData; qemuMonitorJobInfoPtr *jobinfo = NULL; size_t njobinfo = 0; qemuBlockJobDataPtr job = NULL; int newstate; size_t i; int ret = -1; int rc; qemuDomainObjEnterMonitor(driver, vm); rc = qemuMonitorGetJobInfo(priv->mon, &jobinfo, &njobinfo); if (qemuDomainObjExitMonitor(driver, vm) < 0 || rc < 0) goto cleanup; for (i = 0; i < njobinfo; i++) { if (!(job = virHashLookup(priv->blockjobs, jobinfo[i]->id))) { VIR_DEBUG("ignoring untracked job '%s'", jobinfo[i]->id); continue; } /* try cancelling invalid jobs - this works only if the job is not * concluded. In such case it will fail. We'll leave such job linger * in qemu and just forget about it in libvirt because there's not much * we coud do besides killing the VM */ if (job->invalidData) { qemuBlockJobMarkBroken(job); qemuDomainObjEnterMonitor(driver, vm); rc = qemuMonitorJobCancel(priv->mon, job->name, true); if (rc == -1 && jobinfo[i]->status == QEMU_MONITOR_JOB_STATUS_CONCLUDED) VIR_WARN("can't cancel job '%s' with invalid data", job->name); if (qemuDomainObjExitMonitor(driver, vm) < 0) goto cleanup; if (rc < 0) qemuBlockJobUnregister(job, vm); else job->reconnected = true; continue; } if ((newstate = qemuBlockjobConvertMonitorStatus(jobinfo[i]->status)) < 0) continue; if (newstate != job->state) { if ((job->state == QEMU_BLOCKJOB_STATE_FAILED || job->state == QEMU_BLOCKJOB_STATE_COMPLETED)) { /* preserve the old state but allow the job to be bumped to * execute the finishing steps */ job->newstate = job->state; } else if (newstate == QEMU_BLOCKJOB_STATE_CONCLUDED) { job->errmsg = g_strdup(jobinfo[i]->error); if (job->errmsg) job->newstate = QEMU_BLOCKJOB_STATE_FAILED; else job->newstate = QEMU_BLOCKJOB_STATE_COMPLETED; } else if (newstate == QEMU_BLOCKJOB_STATE_READY) { /* Apply _READY state only if it was not applied before */ if (job->state == QEMU_BLOCKJOB_STATE_NEW || job->state == QEMU_BLOCKJOB_STATE_RUNNING) job->newstate = newstate; } /* don't update the job otherwise */ } job->reconnected = true; if (job->newstate != -1) qemuBlockJobUpdate(vm, job, QEMU_ASYNC_JOB_NONE); } /* remove data for job which qemu didn't report (the algorithm is * inefficient, but the possibility of such jobs is very low */ while ((job = virHashSearch(priv->blockjobs, qemuBlockJobRefreshJobsFindInactive, NULL, NULL))) { VIR_WARN("dropping blockjob '%s' untracked by qemu", job->name); qemuBlockJobUnregister(job, vm); } ret = 0; cleanup: for (i = 0; i < njobinfo; i++) qemuMonitorJobInfoFree(jobinfo[i]); VIR_FREE(jobinfo); return ret; } /** * qemuBlockJobEmitEvents: * * Emits the VIR_DOMAIN_EVENT_ID_BLOCK_JOB and VIR_DOMAIN_EVENT_ID_BLOCK_JOB_2 * for a block job. The former event is emitted only for local disks. */ static void qemuBlockJobEmitEvents(virQEMUDriverPtr driver, virDomainObjPtr vm, virDomainDiskDefPtr disk, virDomainBlockJobType type, virConnectDomainEventBlockJobStatus status) { virObjectEventPtr event = NULL; virObjectEventPtr event2 = NULL; /* don't emit events for jobs without disk */ if (!disk) return; /* don't emit events for internal jobs and states */ if (type >= VIR_DOMAIN_BLOCK_JOB_TYPE_LAST || status >= VIR_DOMAIN_BLOCK_JOB_LAST) return; if (virStorageSourceIsLocalStorage(disk->src) && !virStorageSourceIsEmpty(disk->src)) { event = virDomainEventBlockJobNewFromObj(vm, virDomainDiskGetSource(disk), type, status); virObjectEventStateQueue(driver->domainEventState, event); } event2 = virDomainEventBlockJob2NewFromObj(vm, disk->dst, type, status); virObjectEventStateQueue(driver->domainEventState, event2); } /** * qemuBlockJobCleanStorageSourceRuntime: * @src: storage source to clean from runtime data * * Remove all runtime related data from the storage source. */ static void qemuBlockJobCleanStorageSourceRuntime(virStorageSourcePtr src) { src->id = 0; src->detected = false; VIR_FREE(src->relPath); VIR_FREE(src->backingStoreRaw); VIR_FREE(src->nodestorage); VIR_FREE(src->nodeformat); VIR_FREE(src->tlsAlias); VIR_FREE(src->tlsCertdir); } /** * qemuBlockJobRewriteConfigDiskSource: * @vm: domain object * @disk: live definition disk * @newsrc: new source which should be also considered for the new disk * * For block jobs which modify the running disk source it is required that we * try our best to update the config XML's disk source as well in most cases. * * This helper finds the disk from the persistent definition corresponding to * @disk and updates its source to @newsrc. */ static void qemuBlockJobRewriteConfigDiskSource(virDomainObjPtr vm, virDomainDiskDefPtr disk, virStorageSourcePtr newsrc) { virDomainDiskDefPtr persistDisk = NULL; g_autoptr(virStorageSource) copy = NULL; virStorageSourcePtr n; if (!vm->newDef) return; if (!(persistDisk = virDomainDiskByTarget(vm->newDef, disk->dst))) return; if (!virStorageSourceIsSameLocation(disk->src, persistDisk->src)) return; if (!(copy = virStorageSourceCopy(newsrc, true)) || virStorageSourceInitChainElement(copy, persistDisk->src, true) < 0) { VIR_WARN("Unable to update persistent definition on vm %s after block job", vm->def->name); return; } for (n = copy; virStorageSourceIsBacking(n); n = n->backingStore) { qemuBlockJobCleanStorageSourceRuntime(n); /* discard any detected backing store */ if (virStorageSourceIsBacking(n->backingStore) && n->backingStore->detected) { virObjectUnref(n->backingStore); n->backingStore = NULL; break; } } virObjectUnref(persistDisk->src); persistDisk->src = g_steal_pointer(©); } static void qemuBlockJobEventProcessLegacyCompleted(virQEMUDriverPtr driver, virDomainObjPtr vm, qemuBlockJobDataPtr job, int asyncJob) { virDomainDiskDefPtr disk = job->disk; if (!disk) return; if (disk->mirrorState == VIR_DOMAIN_DISK_MIRROR_STATE_PIVOT) { qemuBlockJobRewriteConfigDiskSource(vm, disk, disk->mirror); /* XXX We want to revoke security labels as well as audit that * revocation, before dropping the original source. But it gets * tricky if both source and mirror share common backing files (we * want to only revoke the non-shared portion of the chain); so for * now, we leak the access to the original. */ virDomainLockImageDetach(driver->lockManager, vm, disk->src); /* Move secret driver metadata */ if (qemuSecurityMoveImageMetadata(driver, vm, disk->src, disk->mirror) < 0) { VIR_WARN("Unable to move disk metadata on " "vm %s from %s to %s (disk target %s)", vm->def->name, NULLSTR(disk->src->path), NULLSTR(disk->mirror->path), disk->dst); } virObjectUnref(disk->src); disk->src = disk->mirror; } else { if (disk->mirror) { virDomainLockImageDetach(driver->lockManager, vm, disk->mirror); /* Ideally, we would restore seclabels on the backing chain here * but we don't know if somebody else is not using parts of it. * Remove security driver metadata so that they are not leaked. */ qemuBlockRemoveImageMetadata(driver, vm, disk->dst, disk->mirror); virObjectUnref(disk->mirror); } qemuBlockRemoveImageMetadata(driver, vm, disk->dst, disk->src); } /* Recompute the cached backing chain to match our * updates. Better would be storing the chain ourselves * rather than reprobing, but we haven't quite completed * that conversion to use our XML tracking. */ disk->mirror = NULL; disk->mirrorState = VIR_DOMAIN_DISK_MIRROR_STATE_NONE; disk->mirrorJob = VIR_DOMAIN_BLOCK_JOB_TYPE_UNKNOWN; disk->src->id = 0; virStorageSourceBackingStoreClear(disk->src); ignore_value(qemuDomainDetermineDiskChain(driver, vm, disk, NULL, true)); ignore_value(qemuBlockNodeNamesDetect(driver, vm, asyncJob)); qemuBlockJobUnregister(job, vm); qemuDomainSaveConfig(vm); } /** * qemuBlockJobEventProcessLegacy: * @driver: qemu driver * @vm: domain * @job: job to process events for * * Update disk's mirror state in response to a block job event * from QEMU. For mirror state's that must survive libvirt * restart, also update the domain's status XML. */ static void qemuBlockJobEventProcessLegacy(virQEMUDriverPtr driver, virDomainObjPtr vm, qemuBlockJobDataPtr job, int asyncJob) { virDomainDiskDefPtr disk = job->disk; VIR_DEBUG("disk=%s, mirrorState=%s, type=%d, state=%d, newstate=%d", disk->dst, NULLSTR(virDomainDiskMirrorStateTypeToString(disk->mirrorState)), job->type, job->state, job->newstate); if (job->newstate == -1) return; qemuBlockJobEmitEvents(driver, vm, disk, job->type, job->newstate); job->state = job->newstate; job->newstate = -1; /* If we completed a block pull or commit, then update the XML * to match. */ switch ((virConnectDomainEventBlockJobStatus) job->state) { case VIR_DOMAIN_BLOCK_JOB_COMPLETED: qemuBlockJobEventProcessLegacyCompleted(driver, vm, job, asyncJob); break; case VIR_DOMAIN_BLOCK_JOB_READY: disk->mirrorState = VIR_DOMAIN_DISK_MIRROR_STATE_READY; qemuDomainSaveStatus(vm); break; case VIR_DOMAIN_BLOCK_JOB_FAILED: case VIR_DOMAIN_BLOCK_JOB_CANCELED: if (disk->mirror) { virDomainLockImageDetach(driver->lockManager, vm, disk->mirror); /* Ideally, we would restore seclabels on the backing chain here * but we don't know if somebody else is not using parts of it. * Remove security driver metadata so that they are not leaked. */ qemuBlockRemoveImageMetadata(driver, vm, disk->dst, disk->mirror); virObjectUnref(disk->mirror); disk->mirror = NULL; } disk->mirrorState = VIR_DOMAIN_DISK_MIRROR_STATE_NONE; disk->mirrorJob = VIR_DOMAIN_BLOCK_JOB_TYPE_UNKNOWN; qemuBlockJobUnregister(job, vm); break; case VIR_DOMAIN_BLOCK_JOB_LAST: break; } } static void qemuBlockJobEventProcessConcludedRemoveChain(virQEMUDriverPtr driver, virDomainObjPtr vm, qemuDomainAsyncJob asyncJob, virStorageSourcePtr chain) { g_autoptr(qemuBlockStorageSourceChainData) data = NULL; if (!(data = qemuBlockStorageSourceChainDetachPrepareBlockdev(chain))) return; if (qemuDomainObjEnterMonitorAsync(driver, vm, asyncJob) < 0) return; qemuBlockStorageSourceChainDetach(qemuDomainGetMonitor(vm), data); if (qemuDomainObjExitMonitor(driver, vm) < 0) return; qemuDomainStorageSourceChainAccessRevoke(driver, vm, chain); } /** * qemuBlockJobGetConfigDisk: * @vm: domain object * @disk: disk from the running definition * @diskChainBottom: the last element of backing chain of @disk which is relevant * * Finds and returns the disk corresponding to @disk in the inactive definition. * The inactive disk must have the backing chain starting from the source until * @@diskChainBottom identical. If @diskChainBottom is NULL the whole backing * chains of both @disk and the persistent config definition equivalent must * be identical. */ static virDomainDiskDefPtr qemuBlockJobGetConfigDisk(virDomainObjPtr vm, virDomainDiskDefPtr disk, virStorageSourcePtr diskChainBottom) { virStorageSourcePtr disksrc = NULL; virStorageSourcePtr cfgsrc = NULL; virDomainDiskDefPtr ret = NULL; if (!vm->newDef || !disk) return NULL; disksrc = disk->src; if (!(ret = virDomainDiskByTarget(vm->newDef, disk->dst))) return NULL; cfgsrc = ret->src; while (disksrc && cfgsrc) { if (!virStorageSourceIsSameLocation(disksrc, cfgsrc)) return NULL; if (diskChainBottom && diskChainBottom == disksrc) return ret; disksrc = disksrc->backingStore; cfgsrc = cfgsrc->backingStore; } if (disksrc || cfgsrc) return NULL; return ret; } /** * qemuBlockJobClearConfigChain: * @vm: domain object * @disk: disk object from running definition of @vm * * In cases when the backing chain definitions of the live disk differ from * the definition for the next start config and the backing chain would touch * it we'd not be able to restore the chain in the next start config properly. * * This function checks that the source of the running disk definition and the * config disk definition are the same and if such it clears the backing chain * data. */ static void qemuBlockJobClearConfigChain(virDomainObjPtr vm, virDomainDiskDefPtr disk) { virDomainDiskDefPtr cfgdisk = NULL; if (!vm->newDef || !disk) return; if (!(cfgdisk = virDomainDiskByTarget(vm->newDef, disk->dst))) return; if (!virStorageSourceIsSameLocation(disk->src, cfgdisk->src)) return; virObjectUnref(cfgdisk->src->backingStore); cfgdisk->src->backingStore = NULL; } /** * qemuBlockJobProcessEventCompletedPull: * @driver: qemu driver object * @vm: domain object * @job: job data * @asyncJob: qemu asynchronous job type (for monitor interaction) * * This function executes the finalizing steps after a successful block pull job * (block-stream in qemu terminology. The pull job copies all the data from the * images in the backing chain up to the 'base' image. The 'base' image becomes * the backing store of the active top level image. If 'base' was not used * everything is pulled into the top level image and the top level image will * cease to have backing store. All intermediate images between the active image * and base image are no longer required and can be unplugged. */ static void qemuBlockJobProcessEventCompletedPull(virQEMUDriverPtr driver, virDomainObjPtr vm, qemuBlockJobDataPtr job, qemuDomainAsyncJob asyncJob) { virStorageSourcePtr baseparent = NULL; virDomainDiskDefPtr cfgdisk = NULL; virStorageSourcePtr cfgbase = NULL; virStorageSourcePtr cfgbaseparent = NULL; virStorageSourcePtr n; virStorageSourcePtr tmp; VIR_DEBUG("pull job '%s' on VM '%s' completed", job->name, vm->def->name); /* if the job isn't associated with a disk there's nothing to do */ if (!job->disk) return; if ((cfgdisk = qemuBlockJobGetConfigDisk(vm, job->disk, job->data.pull.base))) cfgbase = cfgdisk->src->backingStore; if (!cfgdisk) qemuBlockJobClearConfigChain(vm, job->disk); /* when pulling if 'base' is right below the top image we don't have to modify it */ if (job->disk->src->backingStore == job->data.pull.base) return; if (job->data.pull.base) { for (n = job->disk->src->backingStore; n && n != job->data.pull.base; n = n->backingStore) { /* find the image on top of 'base' */ if (cfgbase) { cfgbaseparent = cfgbase; cfgbase = cfgbase->backingStore; } baseparent = n; } } tmp = job->disk->src->backingStore; job->disk->src->backingStore = job->data.pull.base; if (baseparent) baseparent->backingStore = NULL; qemuBlockJobEventProcessConcludedRemoveChain(driver, vm, asyncJob, tmp); virObjectUnref(tmp); if (cfgdisk) { tmp = cfgdisk->src->backingStore; cfgdisk->src->backingStore = cfgbase; if (cfgbaseparent) cfgbaseparent->backingStore = NULL; virObjectUnref(tmp); } } /** * qemuBlockJobDeleteImages: * @driver: qemu driver object * @vm: domain object * @disk: disk object that the chain to be deleted is associated with * @top: top snapshot of the chain to be deleted * * Helper for removing snapshot images. Intended for callers like * qemuBlockJobProcessEventCompletedCommit() and * qemuBlockJobProcessEventCompletedActiveCommit() as it relies on adjustments * these functions perform on the 'backingStore' chain to function correctly. * * TODO look into removing backing store for non-local snapshots too */ static void qemuBlockJobDeleteImages(virQEMUDriverPtr driver, virDomainObjPtr vm, virDomainDiskDefPtr disk, virStorageSourcePtr top) { virStorageSourcePtr p = top; g_autoptr(virQEMUDriverConfig) cfg = virQEMUDriverGetConfig(driver); uid_t uid; gid_t gid; for (; p != NULL; p = p->backingStore) { if (virStorageSourceGetActualType(p) == VIR_STORAGE_TYPE_FILE) { qemuDomainGetImageIds(cfg, vm, p, disk->src, &uid, &gid); if (virFileRemove(p->path, uid, gid) < 0) { VIR_WARN("Unable to remove snapshot image file '%s' (%s)", p->path, g_strerror(errno)); } } } } /** * qemuBlockJobProcessEventCompletedCommit: * @driver: qemu driver object * @vm: domain object * @job: job data * @asyncJob: qemu asynchronous job type (for monitor interaction) * * This function executes the finalizing steps after a successful block commit * job. The commit job moves the blocks from backing chain images starting from * 'top' into the 'base' image. The overlay of the 'top' image ('topparent') * then directly references the 'base' image. All intermediate images can be * removed/deleted. */ static void qemuBlockJobProcessEventCompletedCommit(virQEMUDriverPtr driver, virDomainObjPtr vm, qemuBlockJobDataPtr job, qemuDomainAsyncJob asyncJob) { virStorageSourcePtr baseparent = NULL; virDomainDiskDefPtr cfgdisk = NULL; virStorageSourcePtr cfgnext = NULL; virStorageSourcePtr cfgtopparent = NULL; virStorageSourcePtr cfgtop = NULL; virStorageSourcePtr cfgbase = NULL; virStorageSourcePtr cfgbaseparent = NULL; virStorageSourcePtr n; VIR_DEBUG("commit job '%s' on VM '%s' completed", job->name, vm->def->name); /* if the job isn't associated with a disk there's nothing to do */ if (!job->disk) return; if ((cfgdisk = qemuBlockJobGetConfigDisk(vm, job->disk, job->data.commit.base))) cfgnext = cfgdisk->src; if (!cfgdisk) qemuBlockJobClearConfigChain(vm, job->disk); for (n = job->disk->src; n && n != job->data.commit.base; n = n->backingStore) { if (cfgnext) { if (n == job->data.commit.topparent) cfgtopparent = cfgnext; if (n == job->data.commit.top) cfgtop = cfgnext; cfgbaseparent = cfgnext; cfgnext = cfgnext->backingStore; } baseparent = n; } if (!n) return; /* revert access to images */ qemuDomainStorageSourceAccessAllow(driver, vm, job->data.commit.base, true, false); if (job->data.commit.topparent != job->disk->src) qemuDomainStorageSourceAccessAllow(driver, vm, job->data.commit.topparent, true, false); baseparent->backingStore = NULL; job->data.commit.topparent->backingStore = job->data.commit.base; qemuBlockJobEventProcessConcludedRemoveChain(driver, vm, asyncJob, job->data.commit.top); if (job->data.commit.deleteCommittedImages) qemuBlockJobDeleteImages(driver, vm, job->disk, job->data.commit.top); virObjectUnref(job->data.commit.top); job->data.commit.top = NULL; if (cfgbaseparent) { cfgbase = cfgbaseparent->backingStore; cfgbaseparent->backingStore = NULL; if (cfgtopparent) cfgtopparent->backingStore = cfgbase; else cfgdisk->src = cfgbase; virObjectUnref(cfgtop); } } /** * qemuBlockJobProcessEventCompletedActiveCommit: * @driver: qemu driver object * @vm: domain object * @job: job data * @asyncJob: qemu asynchronous job type (for monitor interaction) * * This function executes the finalizing steps after a successful active layer * block commit job. The commit job moves the blocks from backing chain images * starting from the active disk source image into the 'base' image. The disk * source then changes to the 'base' image. All intermediate images can be * removed/deleted. */ static void qemuBlockJobProcessEventCompletedActiveCommit(virQEMUDriverPtr driver, virDomainObjPtr vm, qemuBlockJobDataPtr job, qemuDomainAsyncJob asyncJob) { virStorageSourcePtr baseparent = NULL; virDomainDiskDefPtr cfgdisk = NULL; virStorageSourcePtr cfgnext = NULL; virStorageSourcePtr cfgtop = NULL; virStorageSourcePtr cfgbase = NULL; virStorageSourcePtr cfgbaseparent = NULL; virStorageSourcePtr n; VIR_DEBUG("active commit job '%s' on VM '%s' completed", job->name, vm->def->name); /* if the job isn't associated with a disk there's nothing to do */ if (!job->disk) return; if ((cfgdisk = qemuBlockJobGetConfigDisk(vm, job->disk, job->data.commit.base))) cfgnext = cfgdisk->src; for (n = job->disk->src; n && n != job->data.commit.base; n = n->backingStore) { if (cfgnext) { if (n == job->data.commit.top) cfgtop = cfgnext; cfgbaseparent = cfgnext; cfgnext = cfgnext->backingStore; } baseparent = n; } if (!n) return; if (!cfgdisk) { /* in case when the config disk chain didn't match but the disk top seems * to be identical we need to modify the disk source since the active * commit makes the top level image invalid. */ qemuBlockJobRewriteConfigDiskSource(vm, job->disk, job->data.commit.base); } else { cfgbase = cfgbaseparent->backingStore; cfgbaseparent->backingStore = NULL; cfgdisk->src = cfgbase; cfgdisk->src->readonly = cfgtop->readonly; virObjectUnref(cfgtop); } /* Move security driver metadata */ if (qemuSecurityMoveImageMetadata(driver, vm, job->disk->src, job->data.commit.base) < 0) VIR_WARN("Unable to move disk metadata on vm %s", vm->def->name); baseparent->backingStore = NULL; job->disk->src = job->data.commit.base; job->disk->src->readonly = job->data.commit.top->readonly; qemuBlockJobEventProcessConcludedRemoveChain(driver, vm, asyncJob, job->data.commit.top); if (job->data.commit.deleteCommittedImages) qemuBlockJobDeleteImages(driver, vm, job->disk, job->data.commit.top); virObjectUnref(job->data.commit.top); job->data.commit.top = NULL; /* the mirror element does not serve functional purpose for the commit job */ virObjectUnref(job->disk->mirror); job->disk->mirror = NULL; } static void qemuBlockJobProcessEventConcludedCopyPivot(virQEMUDriverPtr driver, virDomainObjPtr vm, qemuBlockJobDataPtr job, qemuDomainAsyncJob asyncJob) { VIR_DEBUG("copy job '%s' on VM '%s' pivoted", job->name, vm->def->name); /* mirror may be NULL for copy job corresponding to migration */ if (!job->disk || !job->disk->mirror) return; /* for shallow copy without reusing external image the user can either not * specify the backing chain in which case libvirt will open and use the * chain the user provided or not specify a chain in which case we'll * inherit the rest of the chain */ if (job->data.copy.shallownew && !virStorageSourceIsBacking(job->disk->mirror->backingStore)) job->disk->mirror->backingStore = g_steal_pointer(&job->disk->src->backingStore); qemuBlockJobRewriteConfigDiskSource(vm, job->disk, job->disk->mirror); qemuBlockJobEventProcessConcludedRemoveChain(driver, vm, asyncJob, job->disk->src); virObjectUnref(job->disk->src); job->disk->src = g_steal_pointer(&job->disk->mirror); } static void qemuBlockJobProcessEventConcludedCopyAbort(virQEMUDriverPtr driver, virDomainObjPtr vm, qemuBlockJobDataPtr job, qemuDomainAsyncJob asyncJob) { VIR_DEBUG("copy job '%s' on VM '%s' aborted", job->name, vm->def->name); /* mirror may be NULL for copy job corresponding to migration */ if (!job->disk || !job->disk->mirror) return; qemuBlockJobEventProcessConcludedRemoveChain(driver, vm, asyncJob, job->disk->mirror); virObjectUnref(job->disk->mirror); job->disk->mirror = NULL; } static void qemuBlockJobProcessEventFailedActiveCommit(virQEMUDriverPtr driver, virDomainObjPtr vm, qemuBlockJobDataPtr job) { virDomainDiskDefPtr disk = job->disk; VIR_DEBUG("active commit job '%s' on VM '%s' failed", job->name, vm->def->name); if (!disk) return; /* Ideally, we would make the backing chain read only again (yes, SELinux * can do that using different labels). But that is not implemented yet and * not leaking security driver metadata is more important. */ qemuBlockRemoveImageMetadata(driver, vm, disk->dst, disk->mirror); virObjectUnref(disk->mirror); disk->mirror = NULL; } static void qemuBlockJobProcessEventConcludedCreate(virQEMUDriverPtr driver, virDomainObjPtr vm, qemuBlockJobDataPtr job, qemuDomainAsyncJob asyncJob) { g_autoptr(qemuBlockStorageSourceAttachData) backend = NULL; /* if there is a synchronous client waiting for this job that means that * it will handle further hotplug of the created volume and also that * the 'chain' which was registered is under their control */ if (job->synchronous) { virObjectUnref(job->chain); job->chain = NULL; return; } if (!job->data.create.src) return; if (!(backend = qemuBlockStorageSourceDetachPrepare(job->data.create.src, NULL))) return; /* the format node part was not attached yet, so we don't need to detach it */ backend->formatAttached = false; if (job->data.create.storage) { backend->storageAttached = false; VIR_FREE(backend->encryptsecretAlias); } if (qemuDomainObjEnterMonitorAsync(driver, vm, asyncJob) < 0) return; qemuBlockStorageSourceAttachRollback(qemuDomainGetMonitor(vm), backend); if (qemuDomainObjExitMonitor(driver, vm) < 0) return; qemuDomainStorageSourceAccessRevoke(driver, vm, job->data.create.src); } static void qemuBlockJobProcessEventConcludedBackup(virQEMUDriverPtr driver, virDomainObjPtr vm, qemuBlockJobDataPtr job, qemuDomainAsyncJob asyncJob, qemuBlockjobState newstate, unsigned long long progressCurrent, unsigned long long progressTotal) { g_autoptr(virQEMUDriverConfig) cfg = virQEMUDriverGetConfig(driver); uid_t uid; gid_t gid; g_autoptr(qemuBlockStorageSourceAttachData) backend = NULL; g_autoptr(virJSONValue) actions = NULL; qemuBackupNotifyBlockjobEnd(vm, job->disk, newstate, progressCurrent, progressTotal); if (job->data.backup.store && !(backend = qemuBlockStorageSourceDetachPrepare(job->data.backup.store, NULL))) return; if (job->data.backup.bitmap) { if (!(actions = virJSONValueNewArray())) return; if (qemuMonitorTransactionBitmapRemove(actions, job->disk->src->nodeformat, job->data.backup.bitmap) < 0) return; } if (qemuDomainObjEnterMonitorAsync(driver, vm, asyncJob) < 0) return; if (backend) qemuBlockStorageSourceAttachRollback(qemuDomainGetMonitor(vm), backend); if (actions) qemuMonitorTransaction(qemuDomainGetMonitor(vm), &actions); if (qemuDomainObjExitMonitor(driver, vm) < 0) return; if (job->data.backup.store) { qemuDomainStorageSourceAccessRevoke(driver, vm, job->data.backup.store); if (job->data.backup.deleteStore && job->data.backup.store->type == VIR_STORAGE_TYPE_FILE) { qemuDomainGetImageIds(cfg, vm, job->data.backup.store, NULL, &uid, &gid); if (virFileRemove(job->data.backup.store->path, uid, gid) < 0) VIR_WARN("failed to remove scratch file '%s'", job->data.backup.store->path); } } } static void qemuBlockJobEventProcessConcludedTransition(qemuBlockJobDataPtr job, virQEMUDriverPtr driver, virDomainObjPtr vm, qemuDomainAsyncJob asyncJob, unsigned long long progressCurrent, unsigned long long progressTotal) { bool success = job->newstate == QEMU_BLOCKJOB_STATE_COMPLETED; switch ((qemuBlockJobType) job->type) { case QEMU_BLOCKJOB_TYPE_PULL: if (success) qemuBlockJobProcessEventCompletedPull(driver, vm, job, asyncJob); break; case QEMU_BLOCKJOB_TYPE_COMMIT: if (success) qemuBlockJobProcessEventCompletedCommit(driver, vm, job, asyncJob); break; case QEMU_BLOCKJOB_TYPE_ACTIVE_COMMIT: if (success) qemuBlockJobProcessEventCompletedActiveCommit(driver, vm, job, asyncJob); else qemuBlockJobProcessEventFailedActiveCommit(driver, vm, job); break; case QEMU_BLOCKJOB_TYPE_CREATE: qemuBlockJobProcessEventConcludedCreate(driver, vm, job, asyncJob); break; case QEMU_BLOCKJOB_TYPE_COPY: if (job->state == QEMU_BLOCKJOB_STATE_PIVOTING && success) qemuBlockJobProcessEventConcludedCopyPivot(driver, vm, job, asyncJob); else qemuBlockJobProcessEventConcludedCopyAbort(driver, vm, job, asyncJob); break; case QEMU_BLOCKJOB_TYPE_BACKUP: qemuBlockJobProcessEventConcludedBackup(driver, vm, job, asyncJob, job->newstate, progressCurrent, progressTotal); break; case QEMU_BLOCKJOB_TYPE_BROKEN: case QEMU_BLOCKJOB_TYPE_NONE: case QEMU_BLOCKJOB_TYPE_INTERNAL: case QEMU_BLOCKJOB_TYPE_LAST: default: break; } qemuBlockJobEmitEvents(driver, vm, job->disk, job->type, job->newstate); job->state = job->newstate; job->newstate = -1; } static void qemuBlockJobEventProcessConcluded(qemuBlockJobDataPtr job, virQEMUDriverPtr driver, virDomainObjPtr vm, qemuDomainAsyncJob asyncJob) { qemuMonitorJobInfoPtr *jobinfo = NULL; size_t njobinfo = 0; size_t i; bool refreshed = false; unsigned long long progressCurrent = 0; unsigned long long progressTotal = 0; if (qemuDomainObjEnterMonitorAsync(driver, vm, asyncJob) < 0) goto cleanup; /* we need to fetch the error state as the event does not propagate it */ if (job->newstate == QEMU_BLOCKJOB_STATE_CONCLUDED && qemuMonitorGetJobInfo(qemuDomainGetMonitor(vm), &jobinfo, &njobinfo) == 0) { for (i = 0; i < njobinfo; i++) { if (STRNEQ_NULLABLE(job->name, jobinfo[i]->id)) continue; progressCurrent = jobinfo[i]->progressCurrent; progressTotal = jobinfo[i]->progressTotal; job->errmsg = g_strdup(jobinfo[i]->error); if (job->errmsg) job->newstate = QEMU_BLOCKJOB_STATE_FAILED; else job->newstate = QEMU_BLOCKJOB_STATE_COMPLETED; refreshed = true; break; } if (i == njobinfo) VIR_WARN("failed to refresh job '%s'", job->name); } /* dismiss job in qemu */ ignore_value(qemuMonitorJobDismiss(qemuDomainGetMonitor(vm), job->name)); if (qemuDomainObjExitMonitor(driver, vm) < 0) goto cleanup; if ((job->newstate == QEMU_BLOCKJOB_STATE_COMPLETED || job->newstate == QEMU_BLOCKJOB_STATE_FAILED) && job->state == QEMU_BLOCKJOB_STATE_ABORTING) job->newstate = QEMU_BLOCKJOB_STATE_CANCELLED; if (refreshed) qemuDomainSaveStatus(vm); VIR_DEBUG("handling job '%s' state '%d' newstate '%d'", job->name, job->state, job->newstate); qemuBlockJobEventProcessConcludedTransition(job, driver, vm, asyncJob, progressCurrent, progressTotal); /* unplug the backing chains in case the job inherited them */ if (!job->disk) { if (job->chain) qemuBlockJobEventProcessConcludedRemoveChain(driver, vm, asyncJob, job->chain); if (job->mirrorChain) qemuBlockJobEventProcessConcludedRemoveChain(driver, vm, asyncJob, job->mirrorChain); } cleanup: qemuBlockJobUnregister(job, vm); qemuDomainSaveConfig(vm); for (i = 0; i < njobinfo; i++) qemuMonitorJobInfoFree(jobinfo[i]); VIR_FREE(jobinfo); } static void qemuBlockJobEventProcess(virQEMUDriverPtr driver, virDomainObjPtr vm, qemuBlockJobDataPtr job, qemuDomainAsyncJob asyncJob) { switch ((qemuBlockjobState) job->newstate) { case QEMU_BLOCKJOB_STATE_COMPLETED: case QEMU_BLOCKJOB_STATE_FAILED: case QEMU_BLOCKJOB_STATE_CANCELLED: case QEMU_BLOCKJOB_STATE_CONCLUDED: qemuBlockJobEventProcessConcluded(job, driver, vm, asyncJob); break; case QEMU_BLOCKJOB_STATE_READY: /* mirror may be NULL for copy job corresponding to migration */ if (job->disk) { job->disk->mirrorState = VIR_DOMAIN_DISK_MIRROR_STATE_READY; qemuBlockJobEmitEvents(driver, vm, job->disk, job->type, job->newstate); } job->state = job->newstate; job->newstate = -1; qemuDomainSaveStatus(vm); break; case QEMU_BLOCKJOB_STATE_NEW: case QEMU_BLOCKJOB_STATE_RUNNING: case QEMU_BLOCKJOB_STATE_LAST: /* these are never processed as 'newstate' */ case QEMU_BLOCKJOB_STATE_ABORTING: case QEMU_BLOCKJOB_STATE_PIVOTING: default: job->newstate = -1; } } /** * qemuBlockJobUpdate: * @vm: domain * @job: job data * @asyncJob: current qemu asynchronous job type * * Update disk's mirror state in response to a block job event stored in * blockJobStatus by qemuProcessHandleBlockJob event handler. * * Returns the block job event processed or -1 if there was no pending event. */ int qemuBlockJobUpdate(virDomainObjPtr vm, qemuBlockJobDataPtr job, int asyncJob) { qemuDomainObjPrivatePtr priv = vm->privateData; if (job->newstate == -1) return -1; if (virQEMUCapsGet(priv->qemuCaps, QEMU_CAPS_BLOCKDEV)) qemuBlockJobEventProcess(priv->driver, vm, job, asyncJob); else qemuBlockJobEventProcessLegacy(priv->driver, vm, job, asyncJob); return job->state; } /** * qemuBlockJobSyncBegin: * @job: block job data * @disk: domain disk * * Begin a new synchronous block job for @disk. The synchronous * block job is ended by a call to qemuBlockJobSyncEnd, or by * the guest quitting. * * During a synchronous block job, a block job event for @disk * will not be processed asynchronously. Instead, it will be * processed only when qemuBlockJobUpdate or qemuBlockJobSyncEnd * is called. */ void qemuBlockJobSyncBegin(qemuBlockJobDataPtr job) { const char *diskdst = NULL; if (job->disk) diskdst = job->disk->dst; VIR_DEBUG("disk=%s", NULLSTR(diskdst)); job->synchronous = true; } /** * qemuBlockJobSyncEnd: * @vm: domain * @disk: domain disk * * End a synchronous block job for @disk. Any pending block job event * for the disk is processed. Note that it's not necessary to call this function * in case the block job was not started successfully if * qemuBlockJobStartupFinalize will be called. */ void qemuBlockJobSyncEnd(virDomainObjPtr vm, qemuBlockJobDataPtr job, int asyncJob) { const char *diskdst = NULL; if (job->disk) diskdst = job->disk->dst; VIR_DEBUG("disk=%s", NULLSTR(diskdst)); job->synchronous = false; qemuBlockJobUpdate(vm, job, asyncJob); } qemuBlockJobDataPtr qemuBlockJobGetByDisk(virDomainDiskDefPtr disk) { qemuBlockJobDataPtr job = QEMU_DOMAIN_DISK_PRIVATE(disk)->blockjob; if (!job) return NULL; return virObjectRef(job); } /** * @monitorstatus: Status of the blockjob from qemu monitor (qemuMonitorJobStatus) * * Converts the block job status from the monitor to the one used by * qemuBlockJobData. If the status is unknown or does not require any handling * QEMU_BLOCKJOB_TYPE_LAST is returned. */ qemuBlockjobState qemuBlockjobConvertMonitorStatus(int monitorstatus) { qemuBlockjobState ret = QEMU_BLOCKJOB_STATE_LAST; switch ((qemuMonitorJobStatus) monitorstatus) { case QEMU_MONITOR_JOB_STATUS_READY: ret = QEMU_BLOCKJOB_STATE_READY; break; case QEMU_MONITOR_JOB_STATUS_CONCLUDED: ret = QEMU_BLOCKJOB_STATE_CONCLUDED; break; case QEMU_MONITOR_JOB_STATUS_UNKNOWN: case QEMU_MONITOR_JOB_STATUS_CREATED: case QEMU_MONITOR_JOB_STATUS_RUNNING: case QEMU_MONITOR_JOB_STATUS_PAUSED: case QEMU_MONITOR_JOB_STATUS_STANDBY: case QEMU_MONITOR_JOB_STATUS_WAITING: case QEMU_MONITOR_JOB_STATUS_PENDING: case QEMU_MONITOR_JOB_STATUS_ABORTING: case QEMU_MONITOR_JOB_STATUS_UNDEFINED: case QEMU_MONITOR_JOB_STATUS_NULL: case QEMU_MONITOR_JOB_STATUS_LAST: default: break; } return ret; }