qemu_blockjob.c 8.7 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
/*
 * 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
 * <http://www.gnu.org/licenses/>.
 */

#include <config.h>

#include "internal.h"

#include "qemu_blockjob.h"
27
#include "qemu_block.h"
28 29 30 31 32 33 34
#include "qemu_domain.h"

#include "conf/domain_conf.h"
#include "conf/domain_event.h"

#include "virlog.h"
#include "virstoragefile.h"
35 36
#include "virthread.h"
#include "virtime.h"
37
#include "locking/domain_lock.h"
38
#include "viralloc.h"
39 40 41 42 43

#define VIR_FROM_THIS VIR_FROM_QEMU

VIR_LOG_INIT("qemu.qemu_blockjob");

44 45 46 47 48

/**
 * qemuBlockJobUpdate:
 * @vm: domain
 * @disk: domain disk
49
 * @error: error (output parameter)
50 51 52 53 54 55 56
 *
 * 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
57
qemuBlockJobUpdate(virDomainObjPtr vm,
58
                   qemuDomainAsyncJob asyncJob,
59 60
                   virDomainDiskDefPtr disk,
                   char **error)
61 62
{
    qemuDomainDiskPrivatePtr diskPriv = QEMU_DOMAIN_DISK_PRIVATE(disk);
63
    qemuDomainObjPrivatePtr priv = vm->privateData;
64 65
    int status = diskPriv->blockJobStatus;

66 67 68
    if (error)
        *error = NULL;

69
    if (status != -1) {
70
        qemuBlockJobEventProcess(priv->driver, vm, disk, asyncJob,
71 72 73
                                 diskPriv->blockJobType,
                                 diskPriv->blockJobStatus);
        diskPriv->blockJobStatus = -1;
74 75 76 77
        if (error)
            VIR_STEAL_PTR(*error, diskPriv->blockJobError);
        else
            VIR_FREE(diskPriv->blockJobError);
78 79 80 81 82 83
    }

    return status;
}


84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99
/**
 * qemuBlockJobEventProcess:
 * @driver: qemu driver
 * @vm: domain
 * @disk: domain disk
 * @type: block job type
 * @status: block job status
 *
 * 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.
 */
void
qemuBlockJobEventProcess(virQEMUDriverPtr driver,
                         virDomainObjPtr vm,
                         virDomainDiskDefPtr disk,
100
                         qemuDomainAsyncJob asyncJob,
101 102 103 104 105 106 107 108
                         int type,
                         int status)
{
    virObjectEventPtr event = NULL;
    virObjectEventPtr event2 = NULL;
    const char *path;
    virQEMUDriverConfigPtr cfg = virQEMUDriverGetConfig(driver);
    virDomainDiskDefPtr persistDisk = NULL;
109
    qemuDomainDiskPrivatePtr diskPriv = QEMU_DOMAIN_DISK_PRIVATE(disk);
110

111 112 113 114 115 116
    VIR_DEBUG("disk=%s, mirrorState=%s, type=%d, status=%d",
              disk->dst,
              NULLSTR(virDomainDiskMirrorStateTypeToString(disk->mirrorState)),
              type,
              status);

117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133
    /* Have to generate two variants of the event for old vs. new
     * client callbacks */
    if (type == VIR_DOMAIN_BLOCK_JOB_TYPE_COMMIT &&
        disk->mirrorJob == VIR_DOMAIN_BLOCK_JOB_TYPE_ACTIVE_COMMIT)
        type = disk->mirrorJob;
    path = virDomainDiskGetSource(disk);
    event = virDomainEventBlockJobNewFromObj(vm, path, type, status);
    event2 = virDomainEventBlockJob2NewFromObj(vm, disk->dst, type, status);

    /* If we completed a block pull or commit, then update the XML
     * to match.  */
    switch ((virConnectDomainEventBlockJobStatus) status) {
    case VIR_DOMAIN_BLOCK_JOB_COMPLETED:
        if (disk->mirrorState == VIR_DOMAIN_DISK_MIRROR_STATE_PIVOT) {
            if (vm->newDef) {
                virStorageSourcePtr copy = NULL;

134 135
                if ((persistDisk = virDomainDiskByName(vm->newDef,
                                                       disk->dst, false))) {
136
                    copy = virStorageSourceCopy(disk->mirror, false);
137 138
                    if (!copy ||
                        virStorageSourceInitChainElement(copy,
139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154
                                                         persistDisk->src,
                                                         true) < 0) {
                        VIR_WARN("Unable to update persistent definition "
                                 "on vm %s after block job",
                                 vm->def->name);
                        virStorageSourceFree(copy);
                        copy = NULL;
                        persistDisk = NULL;
                    }
                }
                if (copy) {
                    virStorageSourceFree(persistDisk->src);
                    persistDisk->src = copy;
                }
            }

155 156 157 158 159 160
            /* 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);
161 162 163
            virStorageSourceFree(disk->src);
            disk->src = disk->mirror;
        } else {
164 165 166 167
            if (disk->mirror) {
                virDomainLockImageDetach(driver->lockManager, vm, disk->mirror);
                virStorageSourceFree(disk->mirror);
            }
168 169 170 171 172 173 174 175 176
        }

        /* 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;
177
        disk->src->id = 0;
178 179
        ignore_value(qemuDomainDetermineDiskChain(driver, vm, disk,
                                                  true, true));
180
        ignore_value(qemuBlockNodeNamesDetect(driver, vm, asyncJob));
181
        diskPriv->blockjob = false;
182 183 184 185 186 187 188 189
        break;

    case VIR_DOMAIN_BLOCK_JOB_READY:
        disk->mirrorState = VIR_DOMAIN_DISK_MIRROR_STATE_READY;
        break;

    case VIR_DOMAIN_BLOCK_JOB_FAILED:
    case VIR_DOMAIN_BLOCK_JOB_CANCELED:
190 191 192 193 194
        if (disk->mirror) {
            virDomainLockImageDetach(driver->lockManager, vm, disk->mirror);
            virStorageSourceFree(disk->mirror);
            disk->mirror = NULL;
        }
195
        disk->mirrorState = VIR_DOMAIN_DISK_MIRROR_STATE_NONE;
196
        disk->mirrorJob = VIR_DOMAIN_BLOCK_JOB_TYPE_UNKNOWN;
197
        diskPriv->blockjob = false;
198 199 200 201 202 203
        break;

    case VIR_DOMAIN_BLOCK_JOB_LAST:
        break;
    }

204 205 206
    if (virDomainSaveStatus(driver->xmlopt, cfg->stateDir, vm, driver->caps) < 0)
        VIR_WARN("Unable to save status on vm %s after block job", vm->def->name);

207 208 209 210 211
    if (status == VIR_DOMAIN_BLOCK_JOB_COMPLETED && vm->newDef) {
        if (virDomainSaveConfig(cfg->configDir, driver->caps, vm->newDef) < 0)
            VIR_WARN("Unable to update persistent definition on vm %s "
                     "after block job", vm->def->name);
    }
212

213 214
    virObjectEventStateQueue(driver->domainEventState, event);
    virObjectEventStateQueue(driver->domainEventState, event2);
215 216 217

    virObjectUnref(cfg);
}
218 219 220 221 222 223 224 225 226 227 228 229


/**
 * qemuBlockJobSyncBegin:
 * @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
230 231
 * processed only when qemuBlockJobUpdate or qemuBlockJobSyncEnd
 * is called.
232 233 234 235
 */
void
qemuBlockJobSyncBegin(virDomainDiskDefPtr disk)
{
236 237
    qemuDomainDiskPrivatePtr diskPriv = QEMU_DOMAIN_DISK_PRIVATE(disk);

238
    VIR_DEBUG("disk=%s", disk->dst);
239
    diskPriv->blockJobSync = true;
240
    diskPriv->blockJobStatus = -1;
241 242 243 244 245 246 247 248 249
}


/**
 * qemuBlockJobSyncEnd:
 * @vm: domain
 * @disk: domain disk
 *
 * End a synchronous block job for @disk. Any pending block job event
250
 * for the disk is processed.
251 252
 */
void
253
qemuBlockJobSyncEnd(virDomainObjPtr vm,
254
                    qemuDomainAsyncJob asyncJob,
255
                    virDomainDiskDefPtr disk)
256
{
257
    VIR_DEBUG("disk=%s", disk->dst);
258
    qemuBlockJobUpdate(vm, asyncJob, disk, NULL);
259
    QEMU_DOMAIN_DISK_PRIVATE(disk)->blockJobSync = false;
260
}