/* * qemu_backup.c: Implementation and handling of the backup jobs * * 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 "qemu_alias.h" #include "qemu_block.h" #include "qemu_conf.h" #include "qemu_capabilities.h" #include "qemu_monitor.h" #include "qemu_backup.h" #include "qemu_checkpoint.h" #include "qemu_command.h" #include "qemu_security.h" #include "storage_source.h" #include "storage_source_conf.h" #include "virerror.h" #include "virlog.h" #include "virbuffer.h" #include "backup_conf.h" #include "virdomaincheckpointobjlist.h" #define VIR_FROM_THIS VIR_FROM_QEMU VIR_LOG_INIT("qemu.qemu_backup"); static virDomainBackupDef * qemuDomainGetBackup(virDomainObj *vm) { qemuDomainObjPrivate *priv = vm->privateData; if (!priv->backup) { virReportError(VIR_ERR_NO_DOMAIN_BACKUP, "%s", _("no domain backup job present")); return NULL; } return priv->backup; } static int qemuBackupPrepare(virDomainBackupDef *def) { if (def->type == VIR_DOMAIN_BACKUP_TYPE_PULL) { if (!def->server) { def->server = g_new0(virStorageNetHostDef, 1); def->server->transport = VIR_STORAGE_NET_HOST_TRANS_TCP; def->server->name = g_strdup("localhost"); } switch ((virStorageNetHostTransport) def->server->transport) { case VIR_STORAGE_NET_HOST_TRANS_TCP: /* TODO: Update qemu.conf to provide a port range, * probably starting at 10809, for obtaining automatic * port via virPortAllocatorAcquire, as well as store * somewhere if we need to call virPortAllocatorRelease * during BackupEnd. Until then, user must provide port */ if (!def->server->port) { virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s", _(" must specify TCP port for now")); return -1; } break; case VIR_STORAGE_NET_HOST_TRANS_UNIX: /* TODO: Do we need to mess with selinux? */ break; case VIR_STORAGE_NET_HOST_TRANS_RDMA: case VIR_STORAGE_NET_HOST_TRANS_LAST: virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("unexpected transport in ")); return -1; } } return 0; } struct qemuBackupDiskData { virDomainBackupDiskDef *backupdisk; virDomainDiskDef *domdisk; qemuBlockJobData *blockjob; virStorageSource *store; virStorageSource *terminator; virStorageSource *backingStore; char *incrementalBitmap; const char *domdiskIncrementalBitmap; /* name of temporary bitmap installed on disk source */ qemuBlockStorageSourceChainData *crdata; bool labelled; bool initialized; bool created; bool added; bool started; bool done; }; static void qemuBackupDiskDataCleanupOne(virDomainObj *vm, struct qemuBackupDiskData *dd) { qemuDomainObjPrivate *priv = vm->privateData; if (!dd->started) { if (dd->added) { qemuDomainObjEnterMonitor(vm); qemuBlockStorageSourceAttachRollback(priv->mon, dd->crdata->srcdata[0]); qemuDomainObjExitMonitor(vm); } if (dd->created) { if (virStorageSourceUnlink(dd->store) < 0) VIR_WARN("Unable to remove just-created %s", NULLSTR(dd->store->path)); } if (dd->labelled) qemuDomainStorageSourceAccessRevoke(priv->driver, vm, dd->store); } if (dd->initialized) virStorageSourceDeinit(dd->store); if (dd->blockjob) qemuBlockJobStartupFinalize(vm, dd->blockjob); qemuBlockStorageSourceChainDataFree(dd->crdata); virObjectUnref(dd->terminator); g_free(dd->incrementalBitmap); } static void qemuBackupDiskDataCleanup(virDomainObj *vm, struct qemuBackupDiskData *dd, size_t ndd) { virErrorPtr orig_err; size_t i; if (!dd) return; virErrorPreserveLast(&orig_err); for (i = 0; i < ndd; i++) qemuBackupDiskDataCleanupOne(vm, dd + i); g_free(dd); virErrorRestore(&orig_err); } int qemuBackupDiskPrepareOneBitmapsChain(virStorageSource *backingChain, virStorageSource *targetsrc, const char *targetbitmap, const char *incremental, virJSONValue *actions, GHashTable *blockNamedNodeData) { g_autoptr(virJSONValue) tmpactions = NULL; if (qemuBlockGetBitmapMergeActions(backingChain, NULL, targetsrc, incremental, targetbitmap, NULL, &tmpactions, blockNamedNodeData) < 0) return -1; if (tmpactions && virJSONValueArrayConcat(actions, tmpactions) < 0) return -1; return 0; } static int qemuBackupDiskPrepareOneBitmaps(struct qemuBackupDiskData *dd, virJSONValue *actions, bool pull, GHashTable *blockNamedNodeData) { if (!qemuBlockBitmapChainIsValid(dd->domdisk->src, dd->backupdisk->incremental, blockNamedNodeData)) { virReportError(VIR_ERR_CHECKPOINT_INCONSISTENT, _("missing or broken bitmap '%s' for disk '%s'"), dd->backupdisk->incremental, dd->domdisk->dst); return -1; } /* For pull-mode backup, we need the bitmap to be present in the scratch * file as that will be exported. For push-mode backup the bitmap is * actually required on top of the image backing the disk */ if (pull) { if (qemuBackupDiskPrepareOneBitmapsChain(dd->domdisk->src, dd->store, dd->incrementalBitmap, dd->backupdisk->incremental, actions, blockNamedNodeData) < 0) return -1; } else { if (qemuBackupDiskPrepareOneBitmapsChain(dd->domdisk->src, dd->domdisk->src, dd->incrementalBitmap, dd->backupdisk->incremental, actions, blockNamedNodeData) < 0) return -1; dd->domdiskIncrementalBitmap = dd->incrementalBitmap; } return 0; } static int qemuBackupDiskPrepareDataOne(virDomainObj *vm, virDomainBackupDiskDef *backupdisk, struct qemuBackupDiskData *dd, virJSONValue *actions, bool pull, GHashTable *blockNamedNodeData, virQEMUDriverConfig *cfg) { qemuDomainObjPrivate *priv = vm->privateData; /* set data structure */ dd->backupdisk = backupdisk; dd->store = dd->backupdisk->store; if (!(dd->domdisk = virDomainDiskByTarget(vm->def, dd->backupdisk->name))) { virReportError(VIR_ERR_INVALID_ARG, _("no disk named '%s'"), dd->backupdisk->name); return -1; } if (!qemuDomainDiskBlockJobIsSupported(dd->domdisk)) return -1; if (dd->store->format == VIR_STORAGE_FILE_NONE) { dd->store->format = VIR_STORAGE_FILE_QCOW2; } else if (dd->store->format != VIR_STORAGE_FILE_QCOW2) { if (pull) { virReportError(VIR_ERR_CONFIG_UNSUPPORTED, _("pull mode backup for disk '%s' requires qcow2 driver"), dd->backupdisk->name); return -1; } } /* calculate backing store to use: * push mode: * full backups: no backing store * incremental: original disk if format supports backing store * pull mode: * both: original disk */ if (pull || (dd->backupdisk->incremental && dd->store->format >= VIR_STORAGE_FILE_BACKING)) { dd->backingStore = dd->domdisk->src; } else { dd->backingStore = dd->terminator = virStorageSourceNew(); } if (qemuDomainPrepareStorageSourceBlockdev(NULL, dd->store, priv, cfg) < 0) return -1; if (dd->backupdisk->incremental) { /* We deliberately don't check the config of the disk in the checkpoint * definition as it's not guaranteed that the disks still correspond. * We just verify that a checkpoint exists and later on that the disk * has corresponding bitmap. */ if (!virDomainCheckpointFindByName(vm->checkpoints, dd->backupdisk->incremental)) { virReportError(VIR_ERR_NO_DOMAIN_CHECKPOINT, _("Checkpoint '%s' for incremental backup of disk '%s' not found"), dd->backupdisk->incremental, dd->backupdisk->name); return -1; } if (dd->backupdisk->exportbitmap) dd->incrementalBitmap = g_strdup(dd->backupdisk->exportbitmap); else dd->incrementalBitmap = g_strdup_printf("backup-%s", dd->domdisk->dst); if (qemuBackupDiskPrepareOneBitmaps(dd, actions, pull, blockNamedNodeData) < 0) return -1; } if (!(dd->blockjob = qemuBlockJobDiskNewBackup(vm, dd->domdisk, dd->store, dd->domdiskIncrementalBitmap))) return -1; /* use original disk as backing to prevent opening the backing chain */ if (!(dd->crdata = qemuBuildStorageSourceChainAttachPrepareBlockdevTop(dd->store, dd->backingStore))) return -1; return 0; } static int qemuBackupDiskPrepareDataOnePush(virJSONValue *actions, struct qemuBackupDiskData *dd) { qemuMonitorTransactionBackupSyncMode syncmode = QEMU_MONITOR_TRANSACTION_BACKUP_SYNC_MODE_FULL; if (dd->incrementalBitmap) syncmode = QEMU_MONITOR_TRANSACTION_BACKUP_SYNC_MODE_INCREMENTAL; if (qemuMonitorTransactionBackup(actions, dd->domdisk->src->nodeformat, dd->blockjob->name, dd->store->nodeformat, dd->incrementalBitmap, syncmode) < 0) return -1; return 0; } static int qemuBackupDiskPrepareDataOnePull(virJSONValue *actions, struct qemuBackupDiskData *dd) { if (!dd->backupdisk->exportbitmap && dd->incrementalBitmap) dd->backupdisk->exportbitmap = g_strdup(dd->incrementalBitmap); if (qemuMonitorTransactionBackup(actions, dd->domdisk->src->nodeformat, dd->blockjob->name, dd->store->nodeformat, NULL, QEMU_MONITOR_TRANSACTION_BACKUP_SYNC_MODE_NONE) < 0) return -1; return 0; } static ssize_t qemuBackupDiskPrepareData(virDomainObj *vm, virDomainBackupDef *def, GHashTable *blockNamedNodeData, virJSONValue *actions, virQEMUDriverConfig *cfg, struct qemuBackupDiskData **rdd) { struct qemuBackupDiskData *disks = NULL; ssize_t ndisks = 0; size_t i; bool pull = def->type == VIR_DOMAIN_BACKUP_TYPE_PULL; disks = g_new0(struct qemuBackupDiskData, def->ndisks); for (i = 0; i < def->ndisks; i++) { virDomainBackupDiskDef *backupdisk = &def->disks[i]; struct qemuBackupDiskData *dd = disks + ndisks; if (!backupdisk->store) continue; ndisks++; if (qemuBackupDiskPrepareDataOne(vm, backupdisk, dd, actions, pull, blockNamedNodeData, cfg) < 0) goto error; if (pull) { if (qemuBackupDiskPrepareDataOnePull(actions, dd) < 0) goto error; } else { if (qemuBackupDiskPrepareDataOnePush(actions, dd) < 0) goto error; } } *rdd = g_steal_pointer(&disks); return ndisks; error: qemuBackupDiskDataCleanup(vm, disks, ndisks); return -1; } static int qemuBackupDiskPrepareOneStorage(virDomainObj *vm, GHashTable *blockNamedNodeData, struct qemuBackupDiskData *dd, bool reuse_external) { qemuDomainObjPrivate *priv = vm->privateData; int rc; if (!reuse_external && dd->store->type == VIR_STORAGE_TYPE_FILE && virStorageSourceSupportsCreate(dd->store)) { if (virFileExists(dd->store->path)) { virReportError(VIR_ERR_INVALID_ARG, _("store '%s' for backup of '%s' exists"), dd->store->path, dd->domdisk->dst); return -1; } if (qemuDomainStorageFileInit(priv->driver, vm, dd->store, dd->domdisk->src) < 0) return -1; dd->initialized = true; if (virStorageSourceCreate(dd->store) < 0) { virReportSystemError(errno, _("failed to create image file '%s'"), NULLSTR(dd->store->path)); return -1; } dd->created = true; } if (qemuDomainStorageSourceAccessAllow(priv->driver, vm, dd->store, false, true, true) < 0) return -1; dd->labelled = true; if (!reuse_external) { if (qemuBlockStorageSourceCreateDetectSize(blockNamedNodeData, dd->store, dd->domdisk->src) < 0) return -1; if (qemuBlockStorageSourceCreate(vm, dd->store, dd->backingStore, NULL, dd->crdata->srcdata[0], VIR_ASYNC_JOB_BACKUP) < 0) return -1; } else { if (qemuDomainObjEnterMonitorAsync(vm, VIR_ASYNC_JOB_BACKUP) < 0) return -1; rc = qemuBlockStorageSourceAttachApply(priv->mon, dd->crdata->srcdata[0]); qemuDomainObjExitMonitor(vm); if (rc < 0) return -1; } dd->added = true; return 0; } static int qemuBackupDiskPrepareStorage(virDomainObj *vm, struct qemuBackupDiskData *disks, size_t ndisks, GHashTable *blockNamedNodeData, bool reuse_external) { size_t i; for (i = 0; i < ndisks; i++) { if (qemuBackupDiskPrepareOneStorage(vm, blockNamedNodeData, disks + i, reuse_external) < 0) return -1; } return 0; } static void qemuBackupDiskStarted(virDomainObj *vm, struct qemuBackupDiskData *dd, size_t ndd) { size_t i; for (i = 0; i < ndd; i++) { dd[i].started = true; dd[i].backupdisk->state = VIR_DOMAIN_BACKUP_DISK_STATE_RUNNING; qemuBlockJobStarted(dd[i].blockjob, vm); } } /** * qemuBackupBeginPullExportDisks: * @vm: domain object * @disks: backup disk data list * @ndisks: number of valid disks in @disks * * Exports all disks from @dd when doing a pull backup in the NBD server. This * function must be called while in the monitor context. */ static int qemuBackupBeginPullExportDisks(virDomainObj *vm, struct qemuBackupDiskData *disks, size_t ndisks) { size_t i; for (i = 0; i < ndisks; i++) { struct qemuBackupDiskData *dd = disks + i; if (!dd->backupdisk->exportname) dd->backupdisk->exportname = g_strdup(dd->domdisk->dst); if (qemuBlockExportAddNBD(vm, dd->store, dd->backupdisk->exportname, false, dd->incrementalBitmap) < 0) return -1; } return 0; } void qemuBackupJobTerminate(virDomainObj *vm, virDomainJobStatus jobstatus) { qemuDomainObjPrivate *priv = vm->privateData; g_autoptr(virQEMUDriverConfig) cfg = NULL; size_t i; for (i = 0; i < priv->backup->ndisks; i++) { virDomainBackupDiskDef *backupdisk = priv->backup->disks + i; if (!backupdisk->store) continue; /* restore security label on the images in case the blockjob finishing * handler didn't do so, such as when the VM was destroyed */ if (backupdisk->state == VIR_DOMAIN_BACKUP_DISK_STATE_RUNNING || backupdisk->state == VIR_DOMAIN_BACKUP_DISK_STATE_NONE) { if (qemuSecurityRestoreImageLabel(priv->driver, vm, backupdisk->store, false) < 0) VIR_WARN("Unable to restore security label on %s", NULLSTR(backupdisk->store->path)); } /* delete unneeded images created by libvirt */ if (backupdisk->store->type == VIR_STORAGE_TYPE_FILE && !(priv->backup->apiFlags & VIR_DOMAIN_BACKUP_BEGIN_REUSE_EXTERNAL) && (priv->backup->type == VIR_DOMAIN_BACKUP_TYPE_PULL || (priv->backup->type == VIR_DOMAIN_BACKUP_TYPE_PUSH && jobstatus != VIR_DOMAIN_JOB_STATUS_COMPLETED))) { uid_t uid; gid_t gid; if (!cfg) cfg = virQEMUDriverGetConfig(priv->driver); qemuDomainGetImageIds(cfg, vm, backupdisk->store, NULL, &uid, &gid); if (virFileRemove(backupdisk->store->path, uid, gid) < 0) VIR_WARN("failed to remove scratch file '%s'", backupdisk->store->path); } } if (priv->job.current) { qemuDomainJobDataPrivate *privData = NULL; qemuDomainJobDataUpdateTime(priv->job.current); g_clear_pointer(&priv->job.completed, virDomainJobDataFree); priv->job.completed = virDomainJobDataCopy(priv->job.current); privData = priv->job.completed->privateData; privData->stats.backup.total = priv->backup->push_total; privData->stats.backup.transferred = priv->backup->push_transferred; privData->stats.backup.tmp_used = priv->backup->pull_tmp_used; privData->stats.backup.tmp_total = priv->backup->pull_tmp_total; priv->job.completed->status = jobstatus; priv->job.completed->errmsg = g_strdup(priv->backup->errmsg); qemuDomainEventEmitJobCompleted(priv->driver, vm); } g_clear_pointer(&priv->backup, virDomainBackupDefFree); if (priv->job.asyncJob == VIR_ASYNC_JOB_BACKUP) qemuDomainObjEndAsyncJob(vm); } /** * qemuBackupJobCancelBlockjobs: * @vm: domain object * @backup: backup definition * @terminatebackup: flag whether to terminate and unregister the backup * @asyncJob: currently used qemu asynchronous job type * * Sends all active blockjobs which are part of @backup of @vm a signal to * cancel. If @terminatebackup is true qemuBackupJobTerminate is also called * if there are no outstanding active blockjobs. */ void qemuBackupJobCancelBlockjobs(virDomainObj *vm, virDomainBackupDef *backup, bool terminatebackup, int asyncJob) { qemuDomainObjPrivate *priv = vm->privateData; size_t i; int rc = 0; bool has_active = false; if (!backup) return; for (i = 0; i < backup->ndisks; i++) { virDomainBackupDiskDef *backupdisk = backup->disks + i; virDomainDiskDef *disk; g_autoptr(qemuBlockJobData) job = NULL; if (!backupdisk->store) continue; /* Look up corresponding disk as backupdisk->idx is no longer reliable */ if (!(disk = virDomainDiskByTarget(vm->def, backupdisk->name))) continue; if (!(job = qemuBlockJobDiskGetJob(disk))) continue; if (backupdisk->state != VIR_DOMAIN_BACKUP_DISK_STATE_RUNNING && backupdisk->state != VIR_DOMAIN_BACKUP_DISK_STATE_CANCELLING) continue; has_active = true; if (backupdisk->state != VIR_DOMAIN_BACKUP_DISK_STATE_RUNNING) continue; if (qemuDomainObjEnterMonitorAsync(vm, asyncJob) < 0) return; rc = qemuMonitorBlockJobCancel(priv->mon, job->name, true); qemuDomainObjExitMonitor(vm); if (rc == 0) { backupdisk->state = VIR_DOMAIN_BACKUP_DISK_STATE_CANCELLING; job->state = QEMU_BLOCKJOB_STATE_ABORTING; } } if (terminatebackup && !has_active) qemuBackupJobTerminate(vm, VIR_DOMAIN_JOB_STATUS_CANCELED); } #define QEMU_BACKUP_TLS_ALIAS_BASE "libvirt_backup" static int qemuBackupBeginPrepareTLS(virDomainObj *vm, virQEMUDriverConfig *cfg, virDomainBackupDef *def, virJSONValue **tlsProps, virJSONValue **tlsSecretProps) { qemuDomainObjPrivate *priv = vm->privateData; g_autofree char *tlsObjAlias = qemuAliasTLSObjFromSrcAlias(QEMU_BACKUP_TLS_ALIAS_BASE); g_autoptr(qemuDomainSecretInfo) secinfo = NULL; const char *tlsKeySecretAlias = NULL; if (def->tls != VIR_TRISTATE_BOOL_YES) return 0; if (!cfg->backupTLSx509certdir) { virReportError(VIR_ERR_OPERATION_INVALID, "%s", _("backup TLS directory not configured")); return -1; } if (cfg->backupTLSx509secretUUID) { if (!(secinfo = qemuDomainSecretInfoTLSNew(priv, tlsObjAlias, cfg->backupTLSx509secretUUID))) return -1; if (qemuBuildSecretInfoProps(secinfo, tlsSecretProps) < 0) return -1; tlsKeySecretAlias = secinfo->alias; } if (qemuBuildTLSx509BackendProps(cfg->backupTLSx509certdir, true, cfg->backupTLSx509verify, tlsObjAlias, tlsKeySecretAlias, tlsProps) < 0) return -1; return 0; } int qemuBackupBegin(virDomainObj *vm, const char *backupXML, const char *checkpointXML, unsigned int flags) { qemuDomainObjPrivate *priv = vm->privateData; g_autoptr(virQEMUDriverConfig) cfg = virQEMUDriverGetConfig(priv->driver); g_autoptr(virDomainBackupDef) def = NULL; g_autofree char *suffix = NULL; bool pull = false; virDomainMomentObj *chk = NULL; g_autoptr(virDomainCheckpointDef) chkdef = NULL; g_autoptr(virJSONValue) actions = NULL; g_autoptr(virJSONValue) tlsProps = NULL; g_autofree char *tlsAlias = NULL; g_autoptr(virJSONValue) tlsSecretProps = NULL; g_autofree char *tlsSecretAlias = NULL; struct qemuBackupDiskData *dd = NULL; ssize_t ndd = 0; g_autoptr(GHashTable) blockNamedNodeData = NULL; bool job_started = false; bool nbd_running = false; bool reuse = (flags & VIR_DOMAIN_BACKUP_BEGIN_REUSE_EXTERNAL); int rc = 0; int ret = -1; virCheckFlags(VIR_DOMAIN_BACKUP_BEGIN_REUSE_EXTERNAL, -1); if (!(def = virDomainBackupDefParseString(backupXML, priv->driver->xmlopt, 0))) return -1; if (checkpointXML) { if (!(chkdef = virDomainCheckpointDefParseString(checkpointXML, priv->driver->xmlopt, priv->qemuCaps, 0))) return -1; suffix = g_strdup(chkdef->parent.name); } else { gint64 now_us = g_get_real_time(); suffix = g_strdup_printf("%lld", (long long)now_us/(1000*1000)); } if (def->type == VIR_DOMAIN_BACKUP_TYPE_PULL) pull = true; def->apiFlags = flags; /* we'll treat this kind of backup job as an asyncjob as it uses some of the * infrastructure for async jobs. We'll allow standard modify-type jobs * as the interlocking of conflicting operations is handled on the block * job level */ if (qemuDomainObjBeginAsyncJob(vm, VIR_ASYNC_JOB_BACKUP, VIR_DOMAIN_JOB_OPERATION_BACKUP, flags) < 0) return -1; qemuDomainObjSetAsyncJobMask(vm, (VIR_JOB_DEFAULT_MASK | JOB_MASK(VIR_JOB_SUSPEND) | JOB_MASK(VIR_JOB_MODIFY))); qemuDomainJobSetStatsType(priv->job.current, QEMU_DOMAIN_JOB_STATS_TYPE_BACKUP); if (!virDomainObjIsActive(vm)) { virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s", _("cannot perform disk backup for inactive domain")); goto endjob; } if (virDomainBackupAlignDisks(def, vm->def, suffix) < 0) goto endjob; if (!virQEMUCapsGet(priv->qemuCaps, QEMU_CAPS_INCREMENTAL_BACKUP)) { size_t i; if (chkdef) { virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s", _("creating checkpoint for incremental backup is not supported yet")); goto endjob; } for (i = 0; i < def->ndisks; i++) { if (def->disks[i].backupmode == VIR_DOMAIN_BACKUP_DISK_BACKUP_MODE_INCREMENTAL) { virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s", _("incremental backup is not supported yet")); goto endjob; } } } if (priv->backup) { virReportError(VIR_ERR_OPERATION_INVALID, "%s", _("another backup job is already running")); goto endjob; } if (qemuBackupPrepare(def) < 0) goto endjob; if (qemuBackupBeginPrepareTLS(vm, cfg, def, &tlsProps, &tlsSecretProps) < 0) goto endjob; actions = virJSONValueNewArray(); /* The 'chk' checkpoint must be rolled back if the transaction command * which creates it on disk is not executed or fails */ if (chkdef) { if (qemuCheckpointCreateCommon(priv->driver, vm, &chkdef, &actions, &chk) < 0) goto endjob; } if (!(blockNamedNodeData = qemuBlockGetNamedNodeData(vm, VIR_ASYNC_JOB_BACKUP))) goto endjob; if ((ndd = qemuBackupDiskPrepareData(vm, def, blockNamedNodeData, actions, cfg, &dd)) <= 0) { if (ndd == 0) { virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s", _("no disks selected for backup")); } goto endjob; } if (qemuBackupDiskPrepareStorage(vm, dd, ndd, blockNamedNodeData, reuse) < 0) goto endjob; priv->backup = g_steal_pointer(&def); if (qemuDomainObjEnterMonitorAsync(vm, VIR_ASYNC_JOB_BACKUP) < 0) goto endjob; if (pull) { if (tlsSecretProps) rc = qemuMonitorAddObject(priv->mon, &tlsSecretProps, &tlsSecretAlias); if (rc == 0 && tlsProps) rc = qemuMonitorAddObject(priv->mon, &tlsProps, &tlsAlias); if (rc == 0) { if ((rc = qemuMonitorNBDServerStart(priv->mon, priv->backup->server, tlsAlias)) == 0) nbd_running = true; } } if (rc == 0) rc = qemuMonitorTransaction(priv->mon, &actions); qemuDomainObjExitMonitor(vm); if (rc < 0) goto endjob; job_started = true; priv->backup->tlsAlias = g_steal_pointer(&tlsAlias); priv->backup->tlsSecretAlias = g_steal_pointer(&tlsSecretAlias); /* qemuBackupDiskStarted saves the status XML */ qemuBackupDiskStarted(vm, dd, ndd); if (chk) { virDomainMomentObj *tmpchk = g_steal_pointer(&chk); if (qemuCheckpointCreateFinalize(priv->driver, vm, cfg, tmpchk, true) < 0) goto endjob; } if (pull) { if (qemuDomainObjEnterMonitorAsync(vm, VIR_ASYNC_JOB_BACKUP) < 0) goto endjob; /* note that if the export fails we've already created the checkpoint * and we will not delete it */ rc = qemuBackupBeginPullExportDisks(vm, dd, ndd); qemuDomainObjExitMonitor(vm); if (rc < 0) { qemuBackupJobCancelBlockjobs(vm, priv->backup, false, VIR_ASYNC_JOB_BACKUP); goto endjob; } } ret = 0; endjob: qemuBackupDiskDataCleanup(vm, dd, ndd); /* if 'chk' is non-NULL here it's a failure and it must be rolled back */ qemuCheckpointRollbackMetadata(vm, chk); if (!job_started && (nbd_running || tlsAlias || tlsSecretAlias) && qemuDomainObjEnterMonitorAsync(vm, VIR_ASYNC_JOB_BACKUP) == 0) { if (nbd_running) ignore_value(qemuMonitorNBDServerStop(priv->mon)); if (tlsAlias) ignore_value(qemuMonitorDelObject(priv->mon, tlsAlias, false)); if (tlsSecretAlias) ignore_value(qemuMonitorDelObject(priv->mon, tlsSecretAlias, false)); qemuDomainObjExitMonitor(vm); } if (ret < 0 && !job_started && priv->backup) def = g_steal_pointer(&priv->backup); if (ret == 0) qemuDomainObjReleaseAsyncJob(vm); else qemuDomainObjEndAsyncJob(vm); return ret; } char * qemuBackupGetXMLDesc(virDomainObj *vm, unsigned int flags) { qemuDomainObjPrivate *priv = vm->privateData; g_auto(virBuffer) buf = VIR_BUFFER_INITIALIZER; virDomainBackupDef *backup; virCheckFlags(0, NULL); if (!(backup = qemuDomainGetBackup(vm))) return NULL; if (virDomainBackupDefFormat(&buf, backup, false, priv->driver->xmlopt) < 0) return NULL; return virBufferContentAndReset(&buf); } void qemuBackupNotifyBlockjobEnd(virDomainObj *vm, virDomainDiskDef *disk, qemuBlockjobState state, const char *errmsg, unsigned long long cur, unsigned long long end, int asyncJob) { qemuDomainObjPrivate *priv = vm->privateData; bool has_running = false; bool has_cancelling = false; bool has_cancelled = false; bool has_failed = false; virDomainJobStatus jobstatus = VIR_DOMAIN_JOB_STATUS_COMPLETED; virDomainBackupDef *backup = priv->backup; size_t i; VIR_DEBUG("vm: '%s', disk:'%s', state:'%d' errmsg:'%s'", vm->def->name, disk->dst, state, NULLSTR(errmsg)); if (!backup) return; if (backup->type == VIR_DOMAIN_BACKUP_TYPE_PULL) { if (qemuDomainObjEnterMonitorAsync(vm, asyncJob) < 0) return; ignore_value(qemuMonitorNBDServerStop(priv->mon)); if (backup->tlsAlias) ignore_value(qemuMonitorDelObject(priv->mon, backup->tlsAlias, false)); if (backup->tlsSecretAlias) ignore_value(qemuMonitorDelObject(priv->mon, backup->tlsSecretAlias, false)); qemuDomainObjExitMonitor(vm); /* update the final statistics with the current job's data */ backup->pull_tmp_used += cur; backup->pull_tmp_total += end; } else { backup->push_transferred += cur; backup->push_total += end; } /* record first error message */ if (!backup->errmsg) backup->errmsg = g_strdup(errmsg); for (i = 0; i < backup->ndisks; i++) { virDomainBackupDiskDef *backupdisk = backup->disks + i; if (!backupdisk->store) continue; if (STREQ(disk->dst, backupdisk->name)) { switch (state) { case QEMU_BLOCKJOB_STATE_COMPLETED: backupdisk->state = VIR_DOMAIN_BACKUP_DISK_STATE_COMPLETE; break; case QEMU_BLOCKJOB_STATE_CONCLUDED: case QEMU_BLOCKJOB_STATE_FAILED: backupdisk->state = VIR_DOMAIN_BACKUP_DISK_STATE_FAILED; break; case QEMU_BLOCKJOB_STATE_CANCELLED: backupdisk->state = VIR_DOMAIN_BACKUP_DISK_STATE_CANCELLED; break; case QEMU_BLOCKJOB_STATE_READY: case QEMU_BLOCKJOB_STATE_NEW: case QEMU_BLOCKJOB_STATE_RUNNING: case QEMU_BLOCKJOB_STATE_ABORTING: case QEMU_BLOCKJOB_STATE_PIVOTING: case QEMU_BLOCKJOB_STATE_LAST: default: break; } } switch (backupdisk->state) { case VIR_DOMAIN_BACKUP_DISK_STATE_COMPLETE: break; case VIR_DOMAIN_BACKUP_DISK_STATE_RUNNING: has_running = true; break; case VIR_DOMAIN_BACKUP_DISK_STATE_CANCELLING: has_cancelling = true; break; case VIR_DOMAIN_BACKUP_DISK_STATE_FAILED: has_failed = true; break; case VIR_DOMAIN_BACKUP_DISK_STATE_CANCELLED: has_cancelled = true; break; case VIR_DOMAIN_BACKUP_DISK_STATE_NONE: case VIR_DOMAIN_BACKUP_DISK_STATE_LAST: break; } } if (has_running && (has_failed || has_cancelled)) { /* cancel the rest of the jobs */ qemuBackupJobCancelBlockjobs(vm, backup, false, asyncJob); } else if (!has_running && !has_cancelling) { /* all sub-jobs have stopped */ if (has_failed) jobstatus = VIR_DOMAIN_JOB_STATUS_FAILED; else if (has_cancelled && backup->type == VIR_DOMAIN_BACKUP_TYPE_PUSH) jobstatus = VIR_DOMAIN_JOB_STATUS_CANCELED; qemuBackupJobTerminate(vm, jobstatus); } /* otherwise we must wait for the jobs to end */ } static void qemuBackupGetJobInfoStatsUpdateOne(virDomainObj *vm, bool push, const char *diskdst, qemuDomainBackupStats *stats, qemuMonitorJobInfo **blockjobs, size_t nblockjobs) { virDomainDiskDef *domdisk; qemuMonitorJobInfo *monblockjob = NULL; g_autoptr(qemuBlockJobData) diskblockjob = NULL; size_t i; /* it's just statistics so let's not worry so much about errors */ if (!(domdisk = virDomainDiskByTarget(vm->def, diskdst))) return; if (!(diskblockjob = qemuBlockJobDiskGetJob(domdisk))) return; for (i = 0; i < nblockjobs; i++) { if (STREQ_NULLABLE(blockjobs[i]->id, diskblockjob->name)) { monblockjob = blockjobs[i]; break; } } if (!monblockjob) return; if (push) { stats->total += monblockjob->progressTotal; stats->transferred += monblockjob->progressCurrent; } else { stats->tmp_used += monblockjob->progressCurrent; stats->tmp_total += monblockjob->progressTotal; } } int qemuBackupGetJobInfoStats(virDomainObj *vm, virDomainJobData *jobData) { qemuDomainJobDataPrivate *privJob = jobData->privateData; qemuDomainBackupStats *stats = &privJob->stats.backup; qemuDomainObjPrivate *priv = vm->privateData; qemuMonitorJobInfo **blockjobs = NULL; size_t nblockjobs = 0; size_t i; int rc; int ret = -1; if (!priv->backup) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("backup job data missing")); return -1; } if (qemuDomainJobDataUpdateTime(jobData) < 0) return -1; jobData->status = VIR_DOMAIN_JOB_STATUS_ACTIVE; qemuDomainObjEnterMonitor(vm); rc = qemuMonitorGetJobInfo(priv->mon, &blockjobs, &nblockjobs); qemuDomainObjExitMonitor(vm); if (rc < 0) goto cleanup; /* count in completed jobs */ stats->total = priv->backup->push_total; stats->transferred = priv->backup->push_transferred; stats->tmp_used = priv->backup->pull_tmp_used; stats->tmp_total = priv->backup->pull_tmp_total; for (i = 0; i < priv->backup->ndisks; i++) { if (priv->backup->disks[i].state != VIR_DOMAIN_BACKUP_DISK_STATE_RUNNING) continue; qemuBackupGetJobInfoStatsUpdateOne(vm, priv->backup->type == VIR_DOMAIN_BACKUP_TYPE_PUSH, priv->backup->disks[i].name, stats, blockjobs, nblockjobs); } ret = 0; cleanup: for (i = 0; i < nblockjobs; i++) qemuMonitorJobInfoFree(blockjobs[i]); g_free(blockjobs); return ret; }