2010-06-02 18:48:27 +02:00
|
|
|
/*
|
|
|
|
* QEMU host block devices
|
|
|
|
*
|
|
|
|
* Copyright (c) 2003-2008 Fabrice Bellard
|
|
|
|
*
|
|
|
|
* This work is licensed under the terms of the GNU GPL, version 2 or
|
|
|
|
* later. See the COPYING file in the top-level directory.
|
2013-03-14 13:59:53 +01:00
|
|
|
*
|
|
|
|
* This file incorporates work covered by the following copyright and
|
|
|
|
* permission notice:
|
|
|
|
*
|
|
|
|
* Copyright (c) 2003-2008 Fabrice Bellard
|
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
|
|
* in the Software without restriction, including without limitation the rights
|
|
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
|
|
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
* THE SOFTWARE.
|
2010-06-02 18:48:27 +02:00
|
|
|
*/
|
|
|
|
|
2012-12-17 18:20:04 +01:00
|
|
|
#include "sysemu/blockdev.h"
|
2013-02-05 17:06:20 +01:00
|
|
|
#include "hw/block/block.h"
|
2012-12-17 18:19:44 +01:00
|
|
|
#include "block/blockjob.h"
|
2012-12-17 18:19:49 +01:00
|
|
|
#include "monitor/monitor.h"
|
2012-12-17 18:19:43 +01:00
|
|
|
#include "qapi/qmp/qerror.h"
|
2012-12-17 18:20:00 +01:00
|
|
|
#include "qemu/option.h"
|
|
|
|
#include "qemu/config-file.h"
|
2012-12-17 18:19:43 +01:00
|
|
|
#include "qapi/qmp/types.h"
|
2013-09-23 15:26:03 +02:00
|
|
|
#include "qapi-visit.h"
|
|
|
|
#include "qapi/qmp-output-visitor.h"
|
2012-12-17 18:20:04 +01:00
|
|
|
#include "sysemu/sysemu.h"
|
2012-12-17 18:19:44 +01:00
|
|
|
#include "block/block_int.h"
|
2011-11-23 13:28:21 -02:00
|
|
|
#include "qmp-commands.h"
|
2012-01-18 14:40:46 +00:00
|
|
|
#include "trace.h"
|
2012-12-17 18:20:04 +01:00
|
|
|
#include "sysemu/arch_init.h"
|
2010-06-02 18:48:27 +02:00
|
|
|
|
2010-06-02 18:55:22 +02:00
|
|
|
static QTAILQ_HEAD(drivelist, DriveInfo) drives = QTAILQ_HEAD_INITIALIZER(drives);
|
2010-06-02 18:48:27 +02:00
|
|
|
|
2011-01-28 11:21:39 +01:00
|
|
|
static const char *const if_name[IF_COUNT] = {
|
|
|
|
[IF_NONE] = "none",
|
|
|
|
[IF_IDE] = "ide",
|
|
|
|
[IF_SCSI] = "scsi",
|
|
|
|
[IF_FLOPPY] = "floppy",
|
|
|
|
[IF_PFLASH] = "pflash",
|
|
|
|
[IF_MTD] = "mtd",
|
|
|
|
[IF_SD] = "sd",
|
|
|
|
[IF_VIRTIO] = "virtio",
|
|
|
|
[IF_XEN] = "xen",
|
|
|
|
};
|
|
|
|
|
|
|
|
static const int if_max_devs[IF_COUNT] = {
|
2011-01-28 11:21:40 +01:00
|
|
|
/*
|
|
|
|
* Do not change these numbers! They govern how drive option
|
|
|
|
* index maps to unit and bus. That mapping is ABI.
|
|
|
|
*
|
|
|
|
* All controllers used to imlement if=T drives need to support
|
|
|
|
* if_max_devs[T] units, for any T with if_max_devs[T] != 0.
|
|
|
|
* Otherwise, some index values map to "impossible" bus, unit
|
|
|
|
* values.
|
|
|
|
*
|
|
|
|
* For instance, if you change [IF_SCSI] to 255, -drive
|
|
|
|
* if=scsi,index=12 no longer means bus=1,unit=5, but
|
|
|
|
* bus=0,unit=12. With an lsi53c895a controller (7 units max),
|
|
|
|
* the drive can't be set up. Regression.
|
|
|
|
*/
|
|
|
|
[IF_IDE] = 2,
|
|
|
|
[IF_SCSI] = 7,
|
2011-01-28 11:21:39 +01:00
|
|
|
};
|
|
|
|
|
2010-06-25 08:09:10 +02:00
|
|
|
/*
|
|
|
|
* We automatically delete the drive when a device using it gets
|
|
|
|
* unplugged. Questionable feature, but we can't just drop it.
|
|
|
|
* Device models call blockdev_mark_auto_del() to schedule the
|
|
|
|
* automatic deletion, and generic qdev code calls blockdev_auto_del()
|
|
|
|
* when deletion is actually safe.
|
|
|
|
*/
|
|
|
|
void blockdev_mark_auto_del(BlockDriverState *bs)
|
|
|
|
{
|
|
|
|
DriveInfo *dinfo = drive_get_by_blockdev(bs);
|
|
|
|
|
2013-09-18 15:14:47 +02:00
|
|
|
if (dinfo && !dinfo->enable_auto_del) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-03-30 13:17:10 +02:00
|
|
|
if (bs->job) {
|
|
|
|
block_job_cancel(bs->job);
|
|
|
|
}
|
2010-12-08 10:05:00 -06:00
|
|
|
if (dinfo) {
|
|
|
|
dinfo->auto_del = 1;
|
|
|
|
}
|
2010-06-25 08:09:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void blockdev_auto_del(BlockDriverState *bs)
|
|
|
|
{
|
|
|
|
DriveInfo *dinfo = drive_get_by_blockdev(bs);
|
|
|
|
|
2010-12-08 10:05:00 -06:00
|
|
|
if (dinfo && dinfo->auto_del) {
|
2011-01-26 12:12:32 -02:00
|
|
|
drive_put_ref(dinfo);
|
2010-06-25 08:09:10 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-01-28 11:21:43 +01:00
|
|
|
static int drive_index_to_bus_id(BlockInterfaceType type, int index)
|
|
|
|
{
|
|
|
|
int max_devs = if_max_devs[type];
|
|
|
|
return max_devs ? index / max_devs : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int drive_index_to_unit_id(BlockInterfaceType type, int index)
|
|
|
|
{
|
|
|
|
int max_devs = if_max_devs[type];
|
|
|
|
return max_devs ? index % max_devs : index;
|
|
|
|
}
|
|
|
|
|
2011-01-28 11:21:41 +01:00
|
|
|
QemuOpts *drive_def(const char *optstr)
|
|
|
|
{
|
|
|
|
return qemu_opts_parse(qemu_find_opts("drive"), optstr, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
QemuOpts *drive_add(BlockInterfaceType type, int index, const char *file,
|
2011-01-31 11:50:09 +01:00
|
|
|
const char *optstr)
|
2010-06-02 18:48:27 +02:00
|
|
|
{
|
|
|
|
QemuOpts *opts;
|
2011-01-28 11:21:41 +01:00
|
|
|
char buf[32];
|
2010-06-02 18:48:27 +02:00
|
|
|
|
2011-01-28 11:21:41 +01:00
|
|
|
opts = drive_def(optstr);
|
2010-06-02 18:48:27 +02:00
|
|
|
if (!opts) {
|
|
|
|
return NULL;
|
|
|
|
}
|
2011-01-28 11:21:41 +01:00
|
|
|
if (type != IF_DEFAULT) {
|
|
|
|
qemu_opt_set(opts, "if", if_name[type]);
|
|
|
|
}
|
|
|
|
if (index >= 0) {
|
|
|
|
snprintf(buf, sizeof(buf), "%d", index);
|
|
|
|
qemu_opt_set(opts, "index", buf);
|
|
|
|
}
|
2010-06-02 18:48:27 +02:00
|
|
|
if (file)
|
|
|
|
qemu_opt_set(opts, "file", file);
|
|
|
|
return opts;
|
|
|
|
}
|
|
|
|
|
|
|
|
DriveInfo *drive_get(BlockInterfaceType type, int bus, int unit)
|
|
|
|
{
|
|
|
|
DriveInfo *dinfo;
|
|
|
|
|
|
|
|
/* seek interface, bus and unit */
|
|
|
|
|
|
|
|
QTAILQ_FOREACH(dinfo, &drives, next) {
|
|
|
|
if (dinfo->type == type &&
|
|
|
|
dinfo->bus == bus &&
|
|
|
|
dinfo->unit == unit)
|
|
|
|
return dinfo;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2011-01-28 11:21:44 +01:00
|
|
|
DriveInfo *drive_get_by_index(BlockInterfaceType type, int index)
|
|
|
|
{
|
|
|
|
return drive_get(type,
|
|
|
|
drive_index_to_bus_id(type, index),
|
|
|
|
drive_index_to_unit_id(type, index));
|
|
|
|
}
|
|
|
|
|
2010-06-02 18:48:27 +02:00
|
|
|
int drive_get_max_bus(BlockInterfaceType type)
|
|
|
|
{
|
|
|
|
int max_bus;
|
|
|
|
DriveInfo *dinfo;
|
|
|
|
|
|
|
|
max_bus = -1;
|
|
|
|
QTAILQ_FOREACH(dinfo, &drives, next) {
|
|
|
|
if(dinfo->type == type &&
|
|
|
|
dinfo->bus > max_bus)
|
|
|
|
max_bus = dinfo->bus;
|
|
|
|
}
|
|
|
|
return max_bus;
|
|
|
|
}
|
|
|
|
|
2011-01-28 11:21:37 +01:00
|
|
|
/* Get a block device. This should only be used for single-drive devices
|
|
|
|
(e.g. SD/Floppy/MTD). Multi-disk devices (scsi/ide) should use the
|
|
|
|
appropriate bus. */
|
|
|
|
DriveInfo *drive_get_next(BlockInterfaceType type)
|
|
|
|
{
|
|
|
|
static int next_block_unit[IF_COUNT];
|
|
|
|
|
|
|
|
return drive_get(type, 0, next_block_unit[type]++);
|
|
|
|
}
|
|
|
|
|
2010-06-24 17:25:32 +02:00
|
|
|
DriveInfo *drive_get_by_blockdev(BlockDriverState *bs)
|
2010-06-02 18:48:27 +02:00
|
|
|
{
|
|
|
|
DriveInfo *dinfo;
|
|
|
|
|
|
|
|
QTAILQ_FOREACH(dinfo, &drives, next) {
|
2010-06-24 17:25:32 +02:00
|
|
|
if (dinfo->bdrv == bs) {
|
|
|
|
return dinfo;
|
|
|
|
}
|
2010-06-02 18:48:27 +02:00
|
|
|
}
|
2010-06-24 17:25:32 +02:00
|
|
|
return NULL;
|
2010-06-02 18:48:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void bdrv_format_print(void *opaque, const char *name)
|
|
|
|
{
|
2011-01-17 19:31:27 +01:00
|
|
|
error_printf(" %s", name);
|
2010-06-02 18:48:27 +02:00
|
|
|
}
|
|
|
|
|
2011-01-26 12:12:32 -02:00
|
|
|
static void drive_uninit(DriveInfo *dinfo)
|
2010-06-02 18:48:27 +02:00
|
|
|
{
|
2013-09-10 12:01:20 +02:00
|
|
|
if (dinfo->opts) {
|
|
|
|
qemu_opts_del(dinfo->opts);
|
|
|
|
}
|
|
|
|
|
2013-08-23 09:14:47 +08:00
|
|
|
bdrv_unref(dinfo->bdrv);
|
2011-08-20 22:09:37 -05:00
|
|
|
g_free(dinfo->id);
|
2010-06-02 18:48:27 +02:00
|
|
|
QTAILQ_REMOVE(&drives, dinfo, next);
|
2013-03-15 10:35:06 +01:00
|
|
|
g_free(dinfo->serial);
|
2011-08-20 22:09:37 -05:00
|
|
|
g_free(dinfo);
|
2010-06-02 18:48:27 +02:00
|
|
|
}
|
|
|
|
|
2011-01-26 12:12:32 -02:00
|
|
|
void drive_put_ref(DriveInfo *dinfo)
|
|
|
|
{
|
|
|
|
assert(dinfo->refcount);
|
|
|
|
if (--dinfo->refcount == 0) {
|
|
|
|
drive_uninit(dinfo);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void drive_get_ref(DriveInfo *dinfo)
|
|
|
|
{
|
|
|
|
dinfo->refcount++;
|
|
|
|
}
|
|
|
|
|
2012-01-18 14:40:50 +00:00
|
|
|
typedef struct {
|
|
|
|
QEMUBH *bh;
|
2013-08-23 09:14:51 +08:00
|
|
|
BlockDriverState *bs;
|
|
|
|
} BDRVPutRefBH;
|
2012-01-18 14:40:50 +00:00
|
|
|
|
2013-08-23 09:14:51 +08:00
|
|
|
static void bdrv_put_ref_bh(void *opaque)
|
2012-01-18 14:40:50 +00:00
|
|
|
{
|
2013-08-23 09:14:51 +08:00
|
|
|
BDRVPutRefBH *s = opaque;
|
2012-01-18 14:40:50 +00:00
|
|
|
|
2013-08-23 09:14:51 +08:00
|
|
|
bdrv_unref(s->bs);
|
2012-01-18 14:40:50 +00:00
|
|
|
qemu_bh_delete(s->bh);
|
|
|
|
g_free(s);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2013-08-23 09:14:51 +08:00
|
|
|
* Release a BDS reference in a BH
|
2012-01-18 14:40:50 +00:00
|
|
|
*
|
2013-08-23 09:14:51 +08:00
|
|
|
* It is not safe to use bdrv_unref() from a callback function when the callers
|
|
|
|
* still need the BlockDriverState. In such cases we schedule a BH to release
|
|
|
|
* the reference.
|
2012-01-18 14:40:50 +00:00
|
|
|
*/
|
2013-08-23 09:14:51 +08:00
|
|
|
static void bdrv_put_ref_bh_schedule(BlockDriverState *bs)
|
2012-01-18 14:40:50 +00:00
|
|
|
{
|
2013-08-23 09:14:51 +08:00
|
|
|
BDRVPutRefBH *s;
|
2012-01-18 14:40:50 +00:00
|
|
|
|
2013-08-23 09:14:51 +08:00
|
|
|
s = g_new(BDRVPutRefBH, 1);
|
|
|
|
s->bh = qemu_bh_new(bdrv_put_ref_bh, s);
|
|
|
|
s->bs = bs;
|
2012-01-18 14:40:50 +00:00
|
|
|
qemu_bh_schedule(s->bh);
|
|
|
|
}
|
|
|
|
|
2013-09-20 11:33:11 +02:00
|
|
|
static int parse_block_error_action(const char *buf, bool is_read, Error **errp)
|
2010-06-02 18:48:27 +02:00
|
|
|
{
|
|
|
|
if (!strcmp(buf, "ignore")) {
|
2012-09-28 17:22:55 +02:00
|
|
|
return BLOCKDEV_ON_ERROR_IGNORE;
|
2010-06-02 18:48:27 +02:00
|
|
|
} else if (!is_read && !strcmp(buf, "enospc")) {
|
2012-09-28 17:22:55 +02:00
|
|
|
return BLOCKDEV_ON_ERROR_ENOSPC;
|
2010-06-02 18:48:27 +02:00
|
|
|
} else if (!strcmp(buf, "stop")) {
|
2012-09-28 17:22:55 +02:00
|
|
|
return BLOCKDEV_ON_ERROR_STOP;
|
2010-06-02 18:48:27 +02:00
|
|
|
} else if (!strcmp(buf, "report")) {
|
2012-09-28 17:22:55 +02:00
|
|
|
return BLOCKDEV_ON_ERROR_REPORT;
|
2010-06-02 18:48:27 +02:00
|
|
|
} else {
|
2013-09-20 11:33:11 +02:00
|
|
|
error_setg(errp, "'%s' invalid %s error action",
|
|
|
|
buf, is_read ? "read" : "write");
|
2010-06-02 18:48:27 +02:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-02 14:14:39 +02:00
|
|
|
static bool check_throttle_config(ThrottleConfig *cfg, Error **errp)
|
2011-11-03 16:57:25 +08:00
|
|
|
{
|
2013-09-02 14:14:39 +02:00
|
|
|
if (throttle_conflicting(cfg)) {
|
|
|
|
error_setg(errp, "bps/iops/max total values and read/write values"
|
|
|
|
" cannot be used at the same time");
|
2011-11-03 16:57:25 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2013-09-02 14:14:39 +02:00
|
|
|
if (!throttle_is_valid(cfg)) {
|
|
|
|
error_setg(errp, "bps/iops/maxs values must be 0 or greater");
|
2013-02-13 16:53:43 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2011-11-03 16:57:25 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-08-28 17:00:13 +02:00
|
|
|
typedef enum { MEDIA_DISK, MEDIA_CDROM } DriveMediaType;
|
|
|
|
|
2013-09-10 12:01:20 +02:00
|
|
|
/* Takes the ownership of bs_opts */
|
|
|
|
static DriveInfo *blockdev_init(QDict *bs_opts,
|
2013-09-20 11:33:11 +02:00
|
|
|
BlockInterfaceType type,
|
|
|
|
Error **errp)
|
2010-06-02 18:48:27 +02:00
|
|
|
{
|
|
|
|
const char *buf;
|
|
|
|
const char *file = NULL;
|
|
|
|
const char *serial;
|
|
|
|
int ro = 0;
|
|
|
|
int bdrv_flags = 0;
|
|
|
|
int on_read_error, on_write_error;
|
|
|
|
DriveInfo *dinfo;
|
2013-09-02 14:14:39 +02:00
|
|
|
ThrottleConfig cfg;
|
2010-06-02 18:48:27 +02:00
|
|
|
int snapshot = 0;
|
2011-11-17 13:40:32 +00:00
|
|
|
bool copy_on_read;
|
2010-06-02 18:48:27 +02:00
|
|
|
int ret;
|
2013-02-13 16:53:42 +01:00
|
|
|
Error *error = NULL;
|
2013-03-15 10:35:07 +01:00
|
|
|
QemuOpts *opts;
|
|
|
|
const char *id;
|
2013-07-09 11:09:02 +02:00
|
|
|
bool has_driver_specific_opts;
|
2013-08-08 10:45:16 -04:00
|
|
|
BlockDriver *drv = NULL;
|
2010-06-02 18:48:27 +02:00
|
|
|
|
2013-09-10 12:01:20 +02:00
|
|
|
/* Check common options by copying from bs_opts to opts, all other options
|
|
|
|
* stay in bs_opts for processing by bdrv_open(). */
|
|
|
|
id = qdict_get_try_str(bs_opts, "id");
|
2013-03-15 10:35:07 +01:00
|
|
|
opts = qemu_opts_create(&qemu_common_drive_opts, id, 1, &error);
|
|
|
|
if (error_is_set(&error)) {
|
2013-09-20 11:33:11 +02:00
|
|
|
error_propagate(errp, error);
|
2013-03-15 10:35:07 +01:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
qemu_opts_absorb_qdict(opts, bs_opts, &error);
|
|
|
|
if (error_is_set(&error)) {
|
2013-09-20 11:33:11 +02:00
|
|
|
error_propagate(errp, error);
|
2013-10-30 14:54:30 +01:00
|
|
|
goto early_err;
|
2013-03-15 10:35:07 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (id) {
|
|
|
|
qdict_del(bs_opts, "id");
|
|
|
|
}
|
|
|
|
|
2013-07-09 11:09:02 +02:00
|
|
|
has_driver_specific_opts = !!qdict_size(bs_opts);
|
|
|
|
|
2010-06-02 18:48:27 +02:00
|
|
|
/* extract parameters */
|
|
|
|
snapshot = qemu_opt_get_bool(opts, "snapshot", 0);
|
2013-07-19 20:07:29 +02:00
|
|
|
ro = qemu_opt_get_bool(opts, "read-only", 0);
|
2011-11-17 13:40:32 +00:00
|
|
|
copy_on_read = qemu_opt_get_bool(opts, "copy-on-read", false);
|
2010-06-02 18:48:27 +02:00
|
|
|
|
|
|
|
file = qemu_opt_get(opts, "file");
|
|
|
|
serial = qemu_opt_get(opts, "serial");
|
|
|
|
|
2013-02-08 14:06:12 +01:00
|
|
|
if ((buf = qemu_opt_get(opts, "discard")) != NULL) {
|
|
|
|
if (bdrv_parse_discard_flags(buf, &bdrv_flags) != 0) {
|
2013-09-20 11:33:11 +02:00
|
|
|
error_setg(errp, "invalid discard option");
|
2013-10-30 14:54:30 +01:00
|
|
|
goto early_err;
|
2013-02-08 14:06:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-18 16:31:25 +02:00
|
|
|
if (qemu_opt_get_bool(opts, "cache.writeback", true)) {
|
|
|
|
bdrv_flags |= BDRV_O_CACHE_WB;
|
|
|
|
}
|
|
|
|
if (qemu_opt_get_bool(opts, "cache.direct", false)) {
|
|
|
|
bdrv_flags |= BDRV_O_NOCACHE;
|
|
|
|
}
|
2013-09-19 18:48:53 +02:00
|
|
|
if (qemu_opt_get_bool(opts, "cache.no-flush", false)) {
|
2013-07-18 16:31:25 +02:00
|
|
|
bdrv_flags |= BDRV_O_NO_FLUSH;
|
2010-06-02 18:48:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef CONFIG_LINUX_AIO
|
|
|
|
if ((buf = qemu_opt_get(opts, "aio")) != NULL) {
|
|
|
|
if (!strcmp(buf, "native")) {
|
|
|
|
bdrv_flags |= BDRV_O_NATIVE_AIO;
|
|
|
|
} else if (!strcmp(buf, "threads")) {
|
|
|
|
/* this is the default */
|
|
|
|
} else {
|
2013-09-20 11:33:11 +02:00
|
|
|
error_setg(errp, "invalid aio option");
|
2013-10-30 14:54:30 +01:00
|
|
|
goto early_err;
|
2010-06-02 18:48:27 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if ((buf = qemu_opt_get(opts, "format")) != NULL) {
|
2012-08-02 13:45:54 +01:00
|
|
|
if (is_help_option(buf)) {
|
|
|
|
error_printf("Supported formats:");
|
|
|
|
bdrv_iterate_format(bdrv_format_print, NULL);
|
|
|
|
error_printf("\n");
|
2013-10-30 14:54:30 +01:00
|
|
|
goto early_err;
|
2010-06-02 18:48:27 +02:00
|
|
|
}
|
2013-07-09 11:09:02 +02:00
|
|
|
|
2013-10-10 11:45:55 +02:00
|
|
|
drv = bdrv_find_format(buf);
|
2013-08-08 10:45:16 -04:00
|
|
|
if (!drv) {
|
2013-09-20 11:33:11 +02:00
|
|
|
error_setg(errp, "'%s' invalid format", buf);
|
2013-10-30 14:54:30 +01:00
|
|
|
goto early_err;
|
2013-08-08 10:45:16 -04:00
|
|
|
}
|
2010-06-02 18:48:27 +02:00
|
|
|
}
|
|
|
|
|
2011-11-03 16:57:25 +08:00
|
|
|
/* disk I/O throttling */
|
2013-09-02 14:14:39 +02:00
|
|
|
memset(&cfg, 0, sizeof(cfg));
|
|
|
|
cfg.buckets[THROTTLE_BPS_TOTAL].avg =
|
2013-07-17 14:41:54 +02:00
|
|
|
qemu_opt_get_number(opts, "throttling.bps-total", 0);
|
2013-09-02 14:14:39 +02:00
|
|
|
cfg.buckets[THROTTLE_BPS_READ].avg =
|
2013-07-17 14:41:54 +02:00
|
|
|
qemu_opt_get_number(opts, "throttling.bps-read", 0);
|
2013-09-02 14:14:39 +02:00
|
|
|
cfg.buckets[THROTTLE_BPS_WRITE].avg =
|
2013-07-17 14:41:54 +02:00
|
|
|
qemu_opt_get_number(opts, "throttling.bps-write", 0);
|
2013-09-02 14:14:39 +02:00
|
|
|
cfg.buckets[THROTTLE_OPS_TOTAL].avg =
|
2013-07-17 14:41:54 +02:00
|
|
|
qemu_opt_get_number(opts, "throttling.iops-total", 0);
|
2013-09-02 14:14:39 +02:00
|
|
|
cfg.buckets[THROTTLE_OPS_READ].avg =
|
2013-07-17 14:41:54 +02:00
|
|
|
qemu_opt_get_number(opts, "throttling.iops-read", 0);
|
2013-09-02 14:14:39 +02:00
|
|
|
cfg.buckets[THROTTLE_OPS_WRITE].avg =
|
2013-07-17 14:41:54 +02:00
|
|
|
qemu_opt_get_number(opts, "throttling.iops-write", 0);
|
2011-11-03 16:57:25 +08:00
|
|
|
|
2013-09-02 14:14:40 +02:00
|
|
|
cfg.buckets[THROTTLE_BPS_TOTAL].max =
|
|
|
|
qemu_opt_get_number(opts, "throttling.bps-total-max", 0);
|
|
|
|
cfg.buckets[THROTTLE_BPS_READ].max =
|
|
|
|
qemu_opt_get_number(opts, "throttling.bps-read-max", 0);
|
|
|
|
cfg.buckets[THROTTLE_BPS_WRITE].max =
|
|
|
|
qemu_opt_get_number(opts, "throttling.bps-write-max", 0);
|
|
|
|
cfg.buckets[THROTTLE_OPS_TOTAL].max =
|
|
|
|
qemu_opt_get_number(opts, "throttling.iops-total-max", 0);
|
|
|
|
cfg.buckets[THROTTLE_OPS_READ].max =
|
|
|
|
qemu_opt_get_number(opts, "throttling.iops-read-max", 0);
|
|
|
|
cfg.buckets[THROTTLE_OPS_WRITE].max =
|
|
|
|
qemu_opt_get_number(opts, "throttling.iops-write-max", 0);
|
2013-09-02 14:14:39 +02:00
|
|
|
|
2013-09-02 14:14:41 +02:00
|
|
|
cfg.op_size = qemu_opt_get_number(opts, "throttling.iops-size", 0);
|
2013-09-02 14:14:39 +02:00
|
|
|
|
|
|
|
if (!check_throttle_config(&cfg, &error)) {
|
2013-09-20 11:33:11 +02:00
|
|
|
error_propagate(errp, error);
|
2013-10-30 14:54:30 +01:00
|
|
|
goto early_err;
|
2011-11-03 16:57:25 +08:00
|
|
|
}
|
|
|
|
|
2012-09-28 17:22:55 +02:00
|
|
|
on_write_error = BLOCKDEV_ON_ERROR_ENOSPC;
|
2010-06-02 18:48:27 +02:00
|
|
|
if ((buf = qemu_opt_get(opts, "werror")) != NULL) {
|
|
|
|
if (type != IF_IDE && type != IF_SCSI && type != IF_VIRTIO && type != IF_NONE) {
|
2013-09-20 11:33:11 +02:00
|
|
|
error_setg(errp, "werror is not supported by this bus type");
|
2013-10-30 14:54:30 +01:00
|
|
|
goto early_err;
|
2010-06-02 18:48:27 +02:00
|
|
|
}
|
|
|
|
|
2013-09-20 11:33:11 +02:00
|
|
|
on_write_error = parse_block_error_action(buf, 0, &error);
|
|
|
|
if (error_is_set(&error)) {
|
|
|
|
error_propagate(errp, error);
|
2013-10-30 14:54:30 +01:00
|
|
|
goto early_err;
|
2010-06-02 18:48:27 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-09-28 17:22:55 +02:00
|
|
|
on_read_error = BLOCKDEV_ON_ERROR_REPORT;
|
2010-06-02 18:48:27 +02:00
|
|
|
if ((buf = qemu_opt_get(opts, "rerror")) != NULL) {
|
2010-10-25 14:52:21 +02:00
|
|
|
if (type != IF_IDE && type != IF_VIRTIO && type != IF_SCSI && type != IF_NONE) {
|
2011-01-17 19:31:27 +01:00
|
|
|
error_report("rerror is not supported by this bus type");
|
2013-10-30 14:54:30 +01:00
|
|
|
goto early_err;
|
2010-06-02 18:48:27 +02:00
|
|
|
}
|
|
|
|
|
2013-09-20 11:33:11 +02:00
|
|
|
on_read_error = parse_block_error_action(buf, 1, &error);
|
|
|
|
if (error_is_set(&error)) {
|
|
|
|
error_propagate(errp, error);
|
2013-10-30 14:54:30 +01:00
|
|
|
goto early_err;
|
2010-06-02 18:48:27 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-11 12:52:34 +02:00
|
|
|
/* init */
|
|
|
|
dinfo = g_malloc0(sizeof(*dinfo));
|
|
|
|
dinfo->id = g_strdup(qemu_opts_id(opts));
|
2010-06-02 18:48:27 +02:00
|
|
|
dinfo->bdrv = bdrv_new(dinfo->id);
|
2012-09-21 08:50:22 +09:30
|
|
|
dinfo->bdrv->open_flags = snapshot ? BDRV_O_SNAPSHOT : 0;
|
|
|
|
dinfo->bdrv->read_only = ro;
|
2010-06-02 18:48:27 +02:00
|
|
|
dinfo->type = type;
|
2011-01-26 12:12:32 -02:00
|
|
|
dinfo->refcount = 1;
|
2013-03-15 10:35:06 +01:00
|
|
|
if (serial != NULL) {
|
|
|
|
dinfo->serial = g_strdup(serial);
|
|
|
|
}
|
2010-06-02 18:48:27 +02:00
|
|
|
QTAILQ_INSERT_TAIL(&drives, dinfo, next);
|
|
|
|
|
2010-06-02 18:55:17 +02:00
|
|
|
bdrv_set_on_error(dinfo->bdrv, on_read_error, on_write_error);
|
|
|
|
|
2011-11-03 16:57:25 +08:00
|
|
|
/* disk I/O throttling */
|
2013-09-02 14:14:39 +02:00
|
|
|
if (throttle_enabled(&cfg)) {
|
|
|
|
bdrv_io_limits_enable(dinfo->bdrv);
|
|
|
|
bdrv_set_io_limits(dinfo->bdrv, &cfg);
|
|
|
|
}
|
2011-11-03 16:57:25 +08:00
|
|
|
|
2010-05-11 15:36:46 +02:00
|
|
|
if (!file || !*file) {
|
2013-07-09 11:09:02 +02:00
|
|
|
if (has_driver_specific_opts) {
|
2013-03-18 16:40:51 +01:00
|
|
|
file = NULL;
|
|
|
|
} else {
|
2013-10-30 14:54:30 +01:00
|
|
|
QDECREF(bs_opts);
|
|
|
|
qemu_opts_del(opts);
|
2013-03-18 16:40:51 +01:00
|
|
|
return dinfo;
|
|
|
|
}
|
2010-06-02 18:48:27 +02:00
|
|
|
}
|
|
|
|
if (snapshot) {
|
|
|
|
/* always use cache=unsafe with snapshot */
|
|
|
|
bdrv_flags &= ~BDRV_O_CACHE_MASK;
|
|
|
|
bdrv_flags |= (BDRV_O_SNAPSHOT|BDRV_O_CACHE_WB|BDRV_O_NO_FLUSH);
|
|
|
|
}
|
|
|
|
|
2011-11-17 13:40:32 +00:00
|
|
|
if (copy_on_read) {
|
|
|
|
bdrv_flags |= BDRV_O_COPY_ON_READ;
|
|
|
|
}
|
|
|
|
|
2012-03-23 08:36:50 +01:00
|
|
|
if (runstate_check(RUN_STATE_INMIGRATE)) {
|
|
|
|
bdrv_flags |= BDRV_O_INCOMING;
|
|
|
|
}
|
|
|
|
|
2010-06-02 18:48:27 +02:00
|
|
|
bdrv_flags |= ro ? 0 : BDRV_O_RDWR;
|
|
|
|
|
2013-07-09 11:09:02 +02:00
|
|
|
QINCREF(bs_opts);
|
2013-09-05 14:45:29 +02:00
|
|
|
ret = bdrv_open(dinfo->bdrv, file, bs_opts, bdrv_flags, drv, &error);
|
2013-03-15 10:35:07 +01:00
|
|
|
|
2010-06-02 18:48:27 +02:00
|
|
|
if (ret < 0) {
|
2013-09-20 11:33:11 +02:00
|
|
|
error_setg(errp, "could not open disk image %s: %s",
|
|
|
|
file ?: dinfo->id, error_get_pretty(error));
|
|
|
|
error_free(error);
|
2011-02-08 15:12:39 +01:00
|
|
|
goto err;
|
2010-06-02 18:48:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (bdrv_key_required(dinfo->bdrv))
|
|
|
|
autostart = 0;
|
2013-03-15 10:35:07 +01:00
|
|
|
|
2013-07-09 11:09:02 +02:00
|
|
|
QDECREF(bs_opts);
|
2013-03-15 10:35:07 +01:00
|
|
|
qemu_opts_del(opts);
|
|
|
|
|
2010-06-02 18:48:27 +02:00
|
|
|
return dinfo;
|
2011-02-08 15:12:39 +01:00
|
|
|
|
|
|
|
err:
|
2013-08-23 09:14:47 +08:00
|
|
|
bdrv_unref(dinfo->bdrv);
|
2011-08-20 22:09:37 -05:00
|
|
|
g_free(dinfo->id);
|
2011-02-08 15:12:39 +01:00
|
|
|
QTAILQ_REMOVE(&drives, dinfo, next);
|
2011-08-20 22:09:37 -05:00
|
|
|
g_free(dinfo);
|
2013-10-30 14:54:30 +01:00
|
|
|
early_err:
|
|
|
|
QDECREF(bs_opts);
|
|
|
|
qemu_opts_del(opts);
|
2011-02-08 15:12:39 +01:00
|
|
|
return NULL;
|
2010-06-02 18:48:27 +02:00
|
|
|
}
|
|
|
|
|
2013-07-17 14:41:54 +02:00
|
|
|
static void qemu_opt_rename(QemuOpts *opts, const char *from, const char *to)
|
|
|
|
{
|
|
|
|
const char *value;
|
|
|
|
|
|
|
|
value = qemu_opt_get(opts, from);
|
|
|
|
if (value) {
|
|
|
|
qemu_opt_set(opts, to, value);
|
|
|
|
qemu_opt_unset(opts, from);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-28 17:00:13 +02:00
|
|
|
QemuOptsList qemu_legacy_drive_opts = {
|
|
|
|
.name = "drive",
|
|
|
|
.head = QTAILQ_HEAD_INITIALIZER(qemu_legacy_drive_opts.head),
|
|
|
|
.desc = {
|
|
|
|
{
|
2013-09-10 15:48:13 +02:00
|
|
|
.name = "bus",
|
|
|
|
.type = QEMU_OPT_NUMBER,
|
|
|
|
.help = "bus number",
|
|
|
|
},{
|
|
|
|
.name = "unit",
|
|
|
|
.type = QEMU_OPT_NUMBER,
|
|
|
|
.help = "unit number (i.e. lun for scsi)",
|
|
|
|
},{
|
|
|
|
.name = "index",
|
|
|
|
.type = QEMU_OPT_NUMBER,
|
|
|
|
.help = "index number",
|
|
|
|
},{
|
2013-08-28 17:00:13 +02:00
|
|
|
.name = "media",
|
|
|
|
.type = QEMU_OPT_STRING,
|
|
|
|
.help = "media type (disk, cdrom)",
|
2013-08-28 17:24:51 +02:00
|
|
|
},{
|
|
|
|
.name = "if",
|
|
|
|
.type = QEMU_OPT_STRING,
|
|
|
|
.help = "interface (ide, scsi, sd, mtd, floppy, pflash, virtio)",
|
2013-09-09 16:49:49 +02:00
|
|
|
},{
|
|
|
|
.name = "cyls",
|
|
|
|
.type = QEMU_OPT_NUMBER,
|
|
|
|
.help = "number of cylinders (ide disk geometry)",
|
|
|
|
},{
|
|
|
|
.name = "heads",
|
|
|
|
.type = QEMU_OPT_NUMBER,
|
|
|
|
.help = "number of heads (ide disk geometry)",
|
|
|
|
},{
|
|
|
|
.name = "secs",
|
|
|
|
.type = QEMU_OPT_NUMBER,
|
|
|
|
.help = "number of sectors (ide disk geometry)",
|
|
|
|
},{
|
|
|
|
.name = "trans",
|
|
|
|
.type = QEMU_OPT_STRING,
|
|
|
|
.help = "chs translation (auto, lba, none)",
|
2013-09-09 17:01:03 +02:00
|
|
|
},{
|
|
|
|
.name = "boot",
|
|
|
|
.type = QEMU_OPT_BOOL,
|
|
|
|
.help = "(deprecated, ignored)",
|
2013-09-13 14:09:17 +02:00
|
|
|
},{
|
|
|
|
.name = "addr",
|
|
|
|
.type = QEMU_OPT_STRING,
|
|
|
|
.help = "pci address (virtio only)",
|
2013-08-28 17:00:13 +02:00
|
|
|
},
|
2013-09-19 15:12:18 +02:00
|
|
|
|
|
|
|
/* Options that are passed on, but have special semantics with -drive */
|
|
|
|
{
|
|
|
|
.name = "read-only",
|
|
|
|
.type = QEMU_OPT_BOOL,
|
|
|
|
.help = "open drive file as read-only",
|
|
|
|
},{
|
|
|
|
.name = "copy-on-read",
|
|
|
|
.type = QEMU_OPT_BOOL,
|
|
|
|
.help = "copy read data from backing file into image file",
|
|
|
|
},
|
|
|
|
|
2013-08-28 17:00:13 +02:00
|
|
|
{ /* end of list */ }
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2013-07-17 14:41:54 +02:00
|
|
|
DriveInfo *drive_init(QemuOpts *all_opts, BlockInterfaceType block_default_type)
|
|
|
|
{
|
2013-07-18 16:31:25 +02:00
|
|
|
const char *value;
|
2013-08-28 17:00:13 +02:00
|
|
|
DriveInfo *dinfo = NULL;
|
2013-09-10 12:01:20 +02:00
|
|
|
QDict *bs_opts;
|
2013-08-28 17:00:13 +02:00
|
|
|
QemuOpts *legacy_opts;
|
|
|
|
DriveMediaType media = MEDIA_DISK;
|
2013-08-28 17:24:51 +02:00
|
|
|
BlockInterfaceType type;
|
2013-09-09 16:49:49 +02:00
|
|
|
int cyls, heads, secs, translation;
|
2013-09-10 15:48:13 +02:00
|
|
|
int max_devs, bus_id, unit_id, index;
|
2013-09-13 14:09:17 +02:00
|
|
|
const char *devaddr;
|
2013-10-15 17:45:50 +08:00
|
|
|
bool read_only = false;
|
|
|
|
bool copy_on_read;
|
2013-08-28 17:00:13 +02:00
|
|
|
Error *local_err = NULL;
|
2013-07-18 16:31:25 +02:00
|
|
|
|
2013-07-17 14:41:54 +02:00
|
|
|
/* Change legacy command line options into QMP ones */
|
|
|
|
qemu_opt_rename(all_opts, "iops", "throttling.iops-total");
|
|
|
|
qemu_opt_rename(all_opts, "iops_rd", "throttling.iops-read");
|
|
|
|
qemu_opt_rename(all_opts, "iops_wr", "throttling.iops-write");
|
|
|
|
|
|
|
|
qemu_opt_rename(all_opts, "bps", "throttling.bps-total");
|
|
|
|
qemu_opt_rename(all_opts, "bps_rd", "throttling.bps-read");
|
|
|
|
qemu_opt_rename(all_opts, "bps_wr", "throttling.bps-write");
|
|
|
|
|
2013-09-02 14:14:40 +02:00
|
|
|
qemu_opt_rename(all_opts, "iops_max", "throttling.iops-total-max");
|
|
|
|
qemu_opt_rename(all_opts, "iops_rd_max", "throttling.iops-read-max");
|
|
|
|
qemu_opt_rename(all_opts, "iops_wr_max", "throttling.iops-write-max");
|
|
|
|
|
|
|
|
qemu_opt_rename(all_opts, "bps_max", "throttling.bps-total-max");
|
|
|
|
qemu_opt_rename(all_opts, "bps_rd_max", "throttling.bps-read-max");
|
|
|
|
qemu_opt_rename(all_opts, "bps_wr_max", "throttling.bps-write-max");
|
|
|
|
|
2013-09-02 14:14:41 +02:00
|
|
|
qemu_opt_rename(all_opts,
|
|
|
|
"iops_size", "throttling.iops-size");
|
|
|
|
|
2013-07-19 20:07:29 +02:00
|
|
|
qemu_opt_rename(all_opts, "readonly", "read-only");
|
|
|
|
|
2013-07-18 16:31:25 +02:00
|
|
|
value = qemu_opt_get(all_opts, "cache");
|
|
|
|
if (value) {
|
|
|
|
int flags = 0;
|
|
|
|
|
|
|
|
if (bdrv_parse_cache_flags(value, &flags) != 0) {
|
|
|
|
error_report("invalid cache option");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Specific options take precedence */
|
|
|
|
if (!qemu_opt_get(all_opts, "cache.writeback")) {
|
|
|
|
qemu_opt_set_bool(all_opts, "cache.writeback",
|
|
|
|
!!(flags & BDRV_O_CACHE_WB));
|
|
|
|
}
|
|
|
|
if (!qemu_opt_get(all_opts, "cache.direct")) {
|
|
|
|
qemu_opt_set_bool(all_opts, "cache.direct",
|
|
|
|
!!(flags & BDRV_O_NOCACHE));
|
|
|
|
}
|
|
|
|
if (!qemu_opt_get(all_opts, "cache.no-flush")) {
|
|
|
|
qemu_opt_set_bool(all_opts, "cache.no-flush",
|
|
|
|
!!(flags & BDRV_O_NO_FLUSH));
|
|
|
|
}
|
|
|
|
qemu_opt_unset(all_opts, "cache");
|
|
|
|
}
|
|
|
|
|
2013-09-10 12:01:20 +02:00
|
|
|
/* Get a QDict for processing the options */
|
|
|
|
bs_opts = qdict_new();
|
|
|
|
qemu_opts_to_qdict(all_opts, bs_opts);
|
|
|
|
|
2014-01-01 18:49:17 -08:00
|
|
|
legacy_opts = qemu_opts_create(&qemu_legacy_drive_opts, NULL, 0,
|
|
|
|
&error_abort);
|
2013-08-28 17:00:13 +02:00
|
|
|
qemu_opts_absorb_qdict(legacy_opts, bs_opts, &local_err);
|
|
|
|
if (error_is_set(&local_err)) {
|
|
|
|
qerror_report_err(local_err);
|
|
|
|
error_free(local_err);
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
2013-09-09 17:01:03 +02:00
|
|
|
/* Deprecated option boot=[on|off] */
|
|
|
|
if (qemu_opt_get(legacy_opts, "boot") != NULL) {
|
|
|
|
fprintf(stderr, "qemu-kvm: boot=on|off is deprecated and will be "
|
|
|
|
"ignored. Future versions will reject this parameter. Please "
|
|
|
|
"update your scripts.\n");
|
|
|
|
}
|
|
|
|
|
2013-08-28 17:00:13 +02:00
|
|
|
/* Media type */
|
|
|
|
value = qemu_opt_get(legacy_opts, "media");
|
|
|
|
if (value) {
|
|
|
|
if (!strcmp(value, "disk")) {
|
|
|
|
media = MEDIA_DISK;
|
|
|
|
} else if (!strcmp(value, "cdrom")) {
|
|
|
|
media = MEDIA_CDROM;
|
2013-10-15 17:45:50 +08:00
|
|
|
read_only = true;
|
2013-08-28 17:00:13 +02:00
|
|
|
} else {
|
|
|
|
error_report("'%s' invalid media", value);
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-19 15:12:18 +02:00
|
|
|
/* copy-on-read is disabled with a warning for read-only devices */
|
2013-10-15 17:45:50 +08:00
|
|
|
read_only |= qemu_opt_get_bool(legacy_opts, "read-only", false);
|
2013-09-19 15:12:18 +02:00
|
|
|
copy_on_read = qemu_opt_get_bool(legacy_opts, "copy-on-read", false);
|
|
|
|
|
|
|
|
if (read_only && copy_on_read) {
|
|
|
|
error_report("warning: disabling copy-on-read on read-only drive");
|
|
|
|
copy_on_read = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
qdict_put(bs_opts, "read-only",
|
|
|
|
qstring_from_str(read_only ? "on" : "off"));
|
|
|
|
qdict_put(bs_opts, "copy-on-read",
|
|
|
|
qstring_from_str(copy_on_read ? "on" :"off"));
|
|
|
|
|
2013-08-28 17:24:51 +02:00
|
|
|
/* Controller type */
|
|
|
|
value = qemu_opt_get(legacy_opts, "if");
|
|
|
|
if (value) {
|
|
|
|
for (type = 0;
|
|
|
|
type < IF_COUNT && strcmp(value, if_name[type]);
|
|
|
|
type++) {
|
|
|
|
}
|
|
|
|
if (type == IF_COUNT) {
|
|
|
|
error_report("unsupported bus type '%s'", value);
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
type = block_default_type;
|
|
|
|
}
|
|
|
|
|
2013-09-09 16:49:49 +02:00
|
|
|
/* Geometry */
|
|
|
|
cyls = qemu_opt_get_number(legacy_opts, "cyls", 0);
|
|
|
|
heads = qemu_opt_get_number(legacy_opts, "heads", 0);
|
|
|
|
secs = qemu_opt_get_number(legacy_opts, "secs", 0);
|
|
|
|
|
|
|
|
if (cyls || heads || secs) {
|
|
|
|
if (cyls < 1) {
|
|
|
|
error_report("invalid physical cyls number");
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
if (heads < 1) {
|
|
|
|
error_report("invalid physical heads number");
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
if (secs < 1) {
|
|
|
|
error_report("invalid physical secs number");
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
translation = BIOS_ATA_TRANSLATION_AUTO;
|
|
|
|
value = qemu_opt_get(legacy_opts, "trans");
|
|
|
|
if (value != NULL) {
|
|
|
|
if (!cyls) {
|
|
|
|
error_report("'%s' trans must be used with cyls, heads and secs",
|
|
|
|
value);
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
if (!strcmp(value, "none")) {
|
|
|
|
translation = BIOS_ATA_TRANSLATION_NONE;
|
|
|
|
} else if (!strcmp(value, "lba")) {
|
|
|
|
translation = BIOS_ATA_TRANSLATION_LBA;
|
|
|
|
} else if (!strcmp(value, "auto")) {
|
|
|
|
translation = BIOS_ATA_TRANSLATION_AUTO;
|
|
|
|
} else {
|
|
|
|
error_report("'%s' invalid translation type", value);
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (media == MEDIA_CDROM) {
|
|
|
|
if (cyls || secs || heads) {
|
|
|
|
error_report("CHS can't be set with media=cdrom");
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-10 15:48:13 +02:00
|
|
|
/* Device address specified by bus/unit or index.
|
|
|
|
* If none was specified, try to find the first free one. */
|
|
|
|
bus_id = qemu_opt_get_number(legacy_opts, "bus", 0);
|
|
|
|
unit_id = qemu_opt_get_number(legacy_opts, "unit", -1);
|
|
|
|
index = qemu_opt_get_number(legacy_opts, "index", -1);
|
|
|
|
|
|
|
|
max_devs = if_max_devs[type];
|
|
|
|
|
|
|
|
if (index != -1) {
|
|
|
|
if (bus_id != 0 || unit_id != -1) {
|
|
|
|
error_report("index cannot be used with bus and unit");
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
bus_id = drive_index_to_bus_id(type, index);
|
|
|
|
unit_id = drive_index_to_unit_id(type, index);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (unit_id == -1) {
|
|
|
|
unit_id = 0;
|
|
|
|
while (drive_get(type, bus_id, unit_id) != NULL) {
|
|
|
|
unit_id++;
|
|
|
|
if (max_devs && unit_id >= max_devs) {
|
|
|
|
unit_id -= max_devs;
|
|
|
|
bus_id++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (max_devs && unit_id >= max_devs) {
|
|
|
|
error_report("unit %d too big (max is %d)", unit_id, max_devs - 1);
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (drive_get(type, bus_id, unit_id) != NULL) {
|
|
|
|
error_report("drive with bus=%d, unit=%d (index=%d) exists",
|
|
|
|
bus_id, unit_id, index);
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* no id supplied -> create one */
|
|
|
|
if (qemu_opts_id(all_opts) == NULL) {
|
|
|
|
char *new_id;
|
|
|
|
const char *mediastr = "";
|
|
|
|
if (type == IF_IDE || type == IF_SCSI) {
|
|
|
|
mediastr = (media == MEDIA_CDROM) ? "-cd" : "-hd";
|
|
|
|
}
|
|
|
|
if (max_devs) {
|
|
|
|
new_id = g_strdup_printf("%s%i%s%i", if_name[type], bus_id,
|
|
|
|
mediastr, unit_id);
|
|
|
|
} else {
|
|
|
|
new_id = g_strdup_printf("%s%s%i", if_name[type],
|
|
|
|
mediastr, unit_id);
|
|
|
|
}
|
|
|
|
qdict_put(bs_opts, "id", qstring_from_str(new_id));
|
|
|
|
g_free(new_id);
|
|
|
|
}
|
|
|
|
|
2013-09-13 14:09:17 +02:00
|
|
|
/* Add virtio block device */
|
|
|
|
devaddr = qemu_opt_get(legacy_opts, "addr");
|
|
|
|
if (devaddr && type != IF_VIRTIO) {
|
|
|
|
error_report("addr is not supported by this bus type");
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (type == IF_VIRTIO) {
|
|
|
|
QemuOpts *devopts;
|
2014-01-01 18:49:17 -08:00
|
|
|
devopts = qemu_opts_create(qemu_find_opts("device"), NULL, 0,
|
|
|
|
&error_abort);
|
2013-09-13 14:09:17 +02:00
|
|
|
if (arch_type == QEMU_ARCH_S390X) {
|
|
|
|
qemu_opt_set(devopts, "driver", "virtio-blk-s390");
|
|
|
|
} else {
|
|
|
|
qemu_opt_set(devopts, "driver", "virtio-blk-pci");
|
|
|
|
}
|
|
|
|
qemu_opt_set(devopts, "drive", qdict_get_str(bs_opts, "id"));
|
|
|
|
if (devaddr) {
|
|
|
|
qemu_opt_set(devopts, "addr", devaddr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-18 15:14:47 +02:00
|
|
|
/* Actual block device init: Functionality shared with blockdev-add */
|
2013-09-20 11:33:11 +02:00
|
|
|
dinfo = blockdev_init(bs_opts, type, &local_err);
|
2013-09-18 15:14:47 +02:00
|
|
|
if (dinfo == NULL) {
|
2013-09-20 11:33:11 +02:00
|
|
|
if (error_is_set(&local_err)) {
|
|
|
|
qerror_report_err(local_err);
|
|
|
|
error_free(local_err);
|
|
|
|
}
|
2013-09-18 15:14:47 +02:00
|
|
|
goto fail;
|
2013-09-20 11:33:11 +02:00
|
|
|
} else {
|
|
|
|
assert(!error_is_set(&local_err));
|
2013-09-18 15:14:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Set legacy DriveInfo fields */
|
|
|
|
dinfo->enable_auto_del = true;
|
2013-09-10 12:01:20 +02:00
|
|
|
dinfo->opts = all_opts;
|
2013-09-18 15:14:47 +02:00
|
|
|
|
2013-09-09 16:49:49 +02:00
|
|
|
dinfo->cyls = cyls;
|
|
|
|
dinfo->heads = heads;
|
|
|
|
dinfo->secs = secs;
|
|
|
|
dinfo->trans = translation;
|
|
|
|
|
2013-09-10 15:48:13 +02:00
|
|
|
dinfo->bus = bus_id;
|
|
|
|
dinfo->unit = unit_id;
|
2013-09-13 14:09:17 +02:00
|
|
|
dinfo->devaddr = devaddr;
|
2013-09-10 15:48:13 +02:00
|
|
|
|
2013-09-19 14:24:10 +02:00
|
|
|
switch(type) {
|
|
|
|
case IF_IDE:
|
|
|
|
case IF_SCSI:
|
|
|
|
case IF_XEN:
|
|
|
|
case IF_NONE:
|
|
|
|
dinfo->media_cd = media == MEDIA_CDROM;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2013-09-18 15:14:47 +02:00
|
|
|
fail:
|
2013-08-28 17:00:13 +02:00
|
|
|
qemu_opts_del(legacy_opts);
|
2013-09-18 15:14:47 +02:00
|
|
|
return dinfo;
|
2013-07-17 14:41:54 +02:00
|
|
|
}
|
|
|
|
|
2010-06-02 18:48:27 +02:00
|
|
|
void do_commit(Monitor *mon, const QDict *qdict)
|
|
|
|
{
|
|
|
|
const char *device = qdict_get_str(qdict, "device");
|
2010-06-02 18:55:18 +02:00
|
|
|
BlockDriverState *bs;
|
2012-03-05 18:10:11 +00:00
|
|
|
int ret;
|
2010-06-02 18:48:27 +02:00
|
|
|
|
2010-06-02 18:55:18 +02:00
|
|
|
if (!strcmp(device, "all")) {
|
2012-03-05 18:10:11 +00:00
|
|
|
ret = bdrv_commit_all();
|
2010-06-02 18:55:18 +02:00
|
|
|
} else {
|
|
|
|
bs = bdrv_find(device);
|
2010-06-02 18:55:19 +02:00
|
|
|
if (!bs) {
|
2013-01-18 12:45:35 -05:00
|
|
|
monitor_printf(mon, "Device '%s' not found\n", device);
|
2010-06-02 18:55:19 +02:00
|
|
|
return;
|
2010-06-02 18:55:18 +02:00
|
|
|
}
|
2012-01-18 14:40:41 +00:00
|
|
|
ret = bdrv_commit(bs);
|
2013-01-18 12:45:35 -05:00
|
|
|
}
|
|
|
|
if (ret < 0) {
|
|
|
|
monitor_printf(mon, "'commit' error for '%s': %s\n", device,
|
|
|
|
strerror(-ret));
|
2010-06-02 18:48:27 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-06 18:55:59 +01:00
|
|
|
static void blockdev_do_action(int kind, void *data, Error **errp)
|
|
|
|
{
|
2013-05-13 10:43:43 +02:00
|
|
|
TransactionAction action;
|
|
|
|
TransactionActionList list;
|
2012-03-06 18:55:59 +01:00
|
|
|
|
|
|
|
action.kind = kind;
|
|
|
|
action.data = data;
|
|
|
|
list.value = &action;
|
|
|
|
list.next = NULL;
|
|
|
|
qmp_transaction(&list, errp);
|
|
|
|
}
|
|
|
|
|
2011-11-25 16:15:19 -02:00
|
|
|
void qmp_blockdev_snapshot_sync(const char *device, const char *snapshot_file,
|
|
|
|
bool has_format, const char *format,
|
2012-03-06 18:55:59 +01:00
|
|
|
bool has_mode, enum NewImageMode mode,
|
2011-11-25 16:15:19 -02:00
|
|
|
Error **errp)
|
2010-12-16 13:52:16 +01:00
|
|
|
{
|
2012-03-06 18:55:59 +01:00
|
|
|
BlockdevSnapshot snapshot = {
|
|
|
|
.device = (char *) device,
|
|
|
|
.snapshot_file = (char *) snapshot_file,
|
|
|
|
.has_format = has_format,
|
|
|
|
.format = (char *) format,
|
|
|
|
.has_mode = has_mode,
|
|
|
|
.mode = mode,
|
|
|
|
};
|
2013-05-13 10:43:43 +02:00
|
|
|
blockdev_do_action(TRANSACTION_ACTION_KIND_BLOCKDEV_SNAPSHOT_SYNC,
|
|
|
|
&snapshot, errp);
|
2010-12-16 13:52:16 +01:00
|
|
|
}
|
|
|
|
|
2013-09-11 14:04:35 +08:00
|
|
|
void qmp_blockdev_snapshot_internal_sync(const char *device,
|
|
|
|
const char *name,
|
|
|
|
Error **errp)
|
|
|
|
{
|
|
|
|
BlockdevSnapshotInternal snapshot = {
|
|
|
|
.device = (char *) device,
|
|
|
|
.name = (char *) name
|
|
|
|
};
|
|
|
|
|
|
|
|
blockdev_do_action(TRANSACTION_ACTION_KIND_BLOCKDEV_SNAPSHOT_INTERNAL_SYNC,
|
|
|
|
&snapshot, errp);
|
|
|
|
}
|
|
|
|
|
2013-09-11 14:04:36 +08:00
|
|
|
SnapshotInfo *qmp_blockdev_snapshot_delete_internal_sync(const char *device,
|
|
|
|
bool has_id,
|
|
|
|
const char *id,
|
|
|
|
bool has_name,
|
|
|
|
const char *name,
|
|
|
|
Error **errp)
|
|
|
|
{
|
|
|
|
BlockDriverState *bs = bdrv_find(device);
|
|
|
|
QEMUSnapshotInfo sn;
|
|
|
|
Error *local_err = NULL;
|
|
|
|
SnapshotInfo *info = NULL;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (!bs) {
|
|
|
|
error_set(errp, QERR_DEVICE_NOT_FOUND, device);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!has_id) {
|
|
|
|
id = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!has_name) {
|
|
|
|
name = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!id && !name) {
|
|
|
|
error_setg(errp, "Name or id must be provided");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = bdrv_snapshot_find_by_id_and_name(bs, id, name, &sn, &local_err);
|
|
|
|
if (error_is_set(&local_err)) {
|
|
|
|
error_propagate(errp, local_err);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
if (!ret) {
|
|
|
|
error_setg(errp,
|
|
|
|
"Snapshot with id '%s' and name '%s' does not exist on "
|
|
|
|
"device '%s'",
|
|
|
|
STR_OR_NULL(id), STR_OR_NULL(name), device);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
bdrv_snapshot_delete(bs, id, name, &local_err);
|
|
|
|
if (error_is_set(&local_err)) {
|
|
|
|
error_propagate(errp, local_err);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
info = g_malloc0(sizeof(SnapshotInfo));
|
|
|
|
info->id = g_strdup(sn.id_str);
|
|
|
|
info->name = g_strdup(sn.name);
|
|
|
|
info->date_nsec = sn.date_nsec;
|
|
|
|
info->date_sec = sn.date_sec;
|
|
|
|
info->vm_state_size = sn.vm_state_size;
|
|
|
|
info->vm_clock_nsec = sn.vm_clock_nsec % 1000000000;
|
|
|
|
info->vm_clock_sec = sn.vm_clock_nsec / 1000000000;
|
|
|
|
|
|
|
|
return info;
|
|
|
|
}
|
2012-02-28 15:54:06 -05:00
|
|
|
|
|
|
|
/* New and old BlockDriverState structs for group snapshots */
|
2013-05-08 18:25:16 +08:00
|
|
|
|
2013-06-24 17:13:15 +02:00
|
|
|
typedef struct BlkTransactionState BlkTransactionState;
|
2013-05-08 18:25:16 +08:00
|
|
|
|
|
|
|
/* Only prepare() may fail. In a single transaction, only one of commit() or
|
|
|
|
abort() will be called, clean() will always be called if it present. */
|
|
|
|
typedef struct BdrvActionOps {
|
|
|
|
/* Size of state struct, in bytes. */
|
|
|
|
size_t instance_size;
|
|
|
|
/* Prepare the work, must NOT be NULL. */
|
2013-06-24 17:13:15 +02:00
|
|
|
void (*prepare)(BlkTransactionState *common, Error **errp);
|
2013-06-24 17:13:16 +02:00
|
|
|
/* Commit the changes, can be NULL. */
|
2013-06-24 17:13:15 +02:00
|
|
|
void (*commit)(BlkTransactionState *common);
|
2013-05-08 18:25:16 +08:00
|
|
|
/* Abort the changes on fail, can be NULL. */
|
2013-06-24 17:13:15 +02:00
|
|
|
void (*abort)(BlkTransactionState *common);
|
2013-05-08 18:25:16 +08:00
|
|
|
/* Clean up resource in the end, can be NULL. */
|
2013-06-24 17:13:15 +02:00
|
|
|
void (*clean)(BlkTransactionState *common);
|
2013-05-08 18:25:16 +08:00
|
|
|
} BdrvActionOps;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This structure must be arranged as first member in child type, assuming
|
|
|
|
* that compiler will also arrange it to the same address with parent instance.
|
|
|
|
* Later it will be used in free().
|
|
|
|
*/
|
2013-06-24 17:13:15 +02:00
|
|
|
struct BlkTransactionState {
|
2013-05-13 10:43:43 +02:00
|
|
|
TransactionAction *action;
|
2013-05-08 18:25:16 +08:00
|
|
|
const BdrvActionOps *ops;
|
2013-06-24 17:13:15 +02:00
|
|
|
QSIMPLEQ_ENTRY(BlkTransactionState) entry;
|
2013-05-08 18:25:16 +08:00
|
|
|
};
|
|
|
|
|
2013-09-11 14:04:34 +08:00
|
|
|
/* internal snapshot private data */
|
|
|
|
typedef struct InternalSnapshotState {
|
|
|
|
BlkTransactionState common;
|
|
|
|
BlockDriverState *bs;
|
|
|
|
QEMUSnapshotInfo sn;
|
|
|
|
} InternalSnapshotState;
|
|
|
|
|
|
|
|
static void internal_snapshot_prepare(BlkTransactionState *common,
|
|
|
|
Error **errp)
|
|
|
|
{
|
|
|
|
const char *device;
|
|
|
|
const char *name;
|
|
|
|
BlockDriverState *bs;
|
|
|
|
QEMUSnapshotInfo old_sn, *sn;
|
|
|
|
bool ret;
|
|
|
|
qemu_timeval tv;
|
|
|
|
BlockdevSnapshotInternal *internal;
|
|
|
|
InternalSnapshotState *state;
|
|
|
|
int ret1;
|
|
|
|
|
|
|
|
g_assert(common->action->kind ==
|
|
|
|
TRANSACTION_ACTION_KIND_BLOCKDEV_SNAPSHOT_INTERNAL_SYNC);
|
|
|
|
internal = common->action->blockdev_snapshot_internal_sync;
|
|
|
|
state = DO_UPCAST(InternalSnapshotState, common, common);
|
|
|
|
|
|
|
|
/* 1. parse input */
|
|
|
|
device = internal->device;
|
|
|
|
name = internal->name;
|
|
|
|
|
|
|
|
/* 2. check for validation */
|
|
|
|
bs = bdrv_find(device);
|
|
|
|
if (!bs) {
|
|
|
|
error_set(errp, QERR_DEVICE_NOT_FOUND, device);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!bdrv_is_inserted(bs)) {
|
|
|
|
error_set(errp, QERR_DEVICE_HAS_NO_MEDIUM, device);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bdrv_is_read_only(bs)) {
|
|
|
|
error_set(errp, QERR_DEVICE_IS_READ_ONLY, device);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!bdrv_can_snapshot(bs)) {
|
|
|
|
error_set(errp, QERR_BLOCK_FORMAT_FEATURE_NOT_SUPPORTED,
|
|
|
|
bs->drv->format_name, device, "internal snapshot");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!strlen(name)) {
|
|
|
|
error_setg(errp, "Name is empty");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* check whether a snapshot with name exist */
|
|
|
|
ret = bdrv_snapshot_find_by_id_and_name(bs, NULL, name, &old_sn, errp);
|
|
|
|
if (error_is_set(errp)) {
|
|
|
|
return;
|
|
|
|
} else if (ret) {
|
|
|
|
error_setg(errp,
|
|
|
|
"Snapshot with name '%s' already exists on device '%s'",
|
|
|
|
name, device);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* 3. take the snapshot */
|
|
|
|
sn = &state->sn;
|
|
|
|
pstrcpy(sn->name, sizeof(sn->name), name);
|
|
|
|
qemu_gettimeofday(&tv);
|
|
|
|
sn->date_sec = tv.tv_sec;
|
|
|
|
sn->date_nsec = tv.tv_usec * 1000;
|
|
|
|
sn->vm_clock_nsec = qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL);
|
|
|
|
|
|
|
|
ret1 = bdrv_snapshot_create(bs, sn);
|
|
|
|
if (ret1 < 0) {
|
|
|
|
error_setg_errno(errp, -ret1,
|
|
|
|
"Failed to create snapshot '%s' on device '%s'",
|
|
|
|
name, device);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* 4. succeed, mark a snapshot is created */
|
|
|
|
state->bs = bs;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void internal_snapshot_abort(BlkTransactionState *common)
|
|
|
|
{
|
|
|
|
InternalSnapshotState *state =
|
|
|
|
DO_UPCAST(InternalSnapshotState, common, common);
|
|
|
|
BlockDriverState *bs = state->bs;
|
|
|
|
QEMUSnapshotInfo *sn = &state->sn;
|
|
|
|
Error *local_error = NULL;
|
|
|
|
|
|
|
|
if (!bs) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bdrv_snapshot_delete(bs, sn->id_str, sn->name, &local_error) < 0) {
|
|
|
|
error_report("Failed to delete snapshot with id '%s' and name '%s' on "
|
|
|
|
"device '%s' in abort: %s",
|
|
|
|
sn->id_str,
|
|
|
|
sn->name,
|
|
|
|
bdrv_get_device_name(bs),
|
|
|
|
error_get_pretty(local_error));
|
|
|
|
error_free(local_error);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-08 18:25:16 +08:00
|
|
|
/* external snapshot private data */
|
2013-06-24 17:13:15 +02:00
|
|
|
typedef struct ExternalSnapshotState {
|
|
|
|
BlkTransactionState common;
|
2012-02-28 15:54:06 -05:00
|
|
|
BlockDriverState *old_bs;
|
|
|
|
BlockDriverState *new_bs;
|
2013-06-24 17:13:15 +02:00
|
|
|
} ExternalSnapshotState;
|
2012-02-28 15:54:06 -05:00
|
|
|
|
2013-06-24 17:13:15 +02:00
|
|
|
static void external_snapshot_prepare(BlkTransactionState *common,
|
2013-05-08 18:25:12 +08:00
|
|
|
Error **errp)
|
|
|
|
{
|
|
|
|
BlockDriver *drv;
|
|
|
|
int flags, ret;
|
|
|
|
Error *local_err = NULL;
|
2013-05-08 18:25:13 +08:00
|
|
|
const char *device;
|
|
|
|
const char *new_image_file;
|
|
|
|
const char *format = "qcow2";
|
|
|
|
enum NewImageMode mode = NEW_IMAGE_MODE_ABSOLUTE_PATHS;
|
2013-06-24 17:13:15 +02:00
|
|
|
ExternalSnapshotState *state =
|
|
|
|
DO_UPCAST(ExternalSnapshotState, common, common);
|
2013-05-13 10:43:43 +02:00
|
|
|
TransactionAction *action = common->action;
|
2013-05-08 18:25:12 +08:00
|
|
|
|
2013-05-08 18:25:13 +08:00
|
|
|
/* get parameters */
|
2013-05-13 10:43:43 +02:00
|
|
|
g_assert(action->kind == TRANSACTION_ACTION_KIND_BLOCKDEV_SNAPSHOT_SYNC);
|
2013-05-08 18:25:13 +08:00
|
|
|
|
|
|
|
device = action->blockdev_snapshot_sync->device;
|
|
|
|
new_image_file = action->blockdev_snapshot_sync->snapshot_file;
|
|
|
|
if (action->blockdev_snapshot_sync->has_format) {
|
|
|
|
format = action->blockdev_snapshot_sync->format;
|
|
|
|
}
|
|
|
|
if (action->blockdev_snapshot_sync->has_mode) {
|
|
|
|
mode = action->blockdev_snapshot_sync->mode;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* start processing */
|
2013-05-08 18:25:12 +08:00
|
|
|
drv = bdrv_find_format(format);
|
|
|
|
if (!drv) {
|
|
|
|
error_set(errp, QERR_INVALID_BLOCK_FORMAT, format);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-06-24 17:13:15 +02:00
|
|
|
state->old_bs = bdrv_find(device);
|
|
|
|
if (!state->old_bs) {
|
2013-05-08 18:25:12 +08:00
|
|
|
error_set(errp, QERR_DEVICE_NOT_FOUND, device);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-06-24 17:13:15 +02:00
|
|
|
if (!bdrv_is_inserted(state->old_bs)) {
|
2013-05-08 18:25:12 +08:00
|
|
|
error_set(errp, QERR_DEVICE_HAS_NO_MEDIUM, device);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-06-24 17:13:15 +02:00
|
|
|
if (bdrv_in_use(state->old_bs)) {
|
2013-05-08 18:25:12 +08:00
|
|
|
error_set(errp, QERR_DEVICE_IN_USE, device);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-06-24 17:13:15 +02:00
|
|
|
if (!bdrv_is_read_only(state->old_bs)) {
|
|
|
|
if (bdrv_flush(state->old_bs)) {
|
2013-05-08 18:25:12 +08:00
|
|
|
error_set(errp, QERR_IO_ERROR);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-02 14:33:48 +02:00
|
|
|
if (bdrv_check_ext_snapshot(state->old_bs) != EXT_SNAPSHOT_ALLOWED) {
|
|
|
|
error_set(errp, QERR_FEATURE_DISABLED, "snapshot");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-06-24 17:13:15 +02:00
|
|
|
flags = state->old_bs->open_flags;
|
2013-05-08 18:25:12 +08:00
|
|
|
|
|
|
|
/* create new image w/backing file */
|
|
|
|
if (mode != NEW_IMAGE_MODE_EXISTING) {
|
|
|
|
bdrv_img_create(new_image_file, format,
|
2013-06-24 17:13:15 +02:00
|
|
|
state->old_bs->filename,
|
|
|
|
state->old_bs->drv->format_name,
|
2013-05-08 18:25:12 +08:00
|
|
|
NULL, -1, flags, &local_err, false);
|
|
|
|
if (error_is_set(&local_err)) {
|
|
|
|
error_propagate(errp, local_err);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We will manually add the backing_hd field to the bs later */
|
2013-06-24 17:13:15 +02:00
|
|
|
state->new_bs = bdrv_new("");
|
2013-05-08 18:25:12 +08:00
|
|
|
/* TODO Inherit bs->options or only take explicit options with an
|
|
|
|
* extended QMP command? */
|
2013-06-24 17:13:15 +02:00
|
|
|
ret = bdrv_open(state->new_bs, new_image_file, NULL,
|
2013-09-05 14:45:29 +02:00
|
|
|
flags | BDRV_O_NO_BACKING, drv, &local_err);
|
2013-05-08 18:25:12 +08:00
|
|
|
if (ret != 0) {
|
2013-09-05 14:45:29 +02:00
|
|
|
error_propagate(errp, local_err);
|
2013-05-08 18:25:12 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-24 17:13:15 +02:00
|
|
|
static void external_snapshot_commit(BlkTransactionState *common)
|
2013-05-08 18:25:14 +08:00
|
|
|
{
|
2013-06-24 17:13:15 +02:00
|
|
|
ExternalSnapshotState *state =
|
|
|
|
DO_UPCAST(ExternalSnapshotState, common, common);
|
2013-05-08 18:25:16 +08:00
|
|
|
|
2013-06-24 17:13:15 +02:00
|
|
|
/* This removes our old bs and adds the new bs */
|
|
|
|
bdrv_append(state->new_bs, state->old_bs);
|
2013-05-08 18:25:14 +08:00
|
|
|
/* We don't need (or want) to use the transactional
|
|
|
|
* bdrv_reopen_multiple() across all the entries at once, because we
|
|
|
|
* don't want to abort all of them if one of them fails the reopen */
|
2013-06-24 17:13:15 +02:00
|
|
|
bdrv_reopen(state->new_bs, state->new_bs->open_flags & ~BDRV_O_RDWR,
|
2013-05-08 18:25:14 +08:00
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
|
2013-06-24 17:13:15 +02:00
|
|
|
static void external_snapshot_abort(BlkTransactionState *common)
|
2013-05-08 18:25:15 +08:00
|
|
|
{
|
2013-06-24 17:13:15 +02:00
|
|
|
ExternalSnapshotState *state =
|
|
|
|
DO_UPCAST(ExternalSnapshotState, common, common);
|
|
|
|
if (state->new_bs) {
|
2013-08-23 09:14:47 +08:00
|
|
|
bdrv_unref(state->new_bs);
|
2013-05-08 18:25:15 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
blockdev: add DriveBackup transaction
This patch adds a transactional version of the drive-backup QMP command.
It allows atomic snapshots of multiple drives along with automatic
cleanup if there is a failure to start one of the backup jobs.
Note that QMP events are emitted for block job completion/cancellation
and the block job will be listed by query-block-jobs.
@device: the name of the device whose writes should be mirrored.
@target: the target of the new image. If the file exists, or if it
is a device, the existing file/device will be used as the new
destination. If it does not exist, a new file will be created.
@format: #optional the format of the new destination, default is to
probe if @mode is 'existing', else the format of the source
@mode: #optional whether and how QEMU should create a new image, default is
'absolute-paths'.
@speed: #optional the maximum speed, in bytes per second
@on-source-error: #optional the action to take on an error on the source,
default 'report'. 'stop' and 'enospc' can only be used
if the block device supports io-status (see BlockInfo).
@on-target-error: #optional the action to take on an error on the target,
default 'report' (no limitations, since this applies to
a different block device than @device).
Reviewed-by: Eric Blake <eblake@redhat.com>
Reviewed-by: Kevin Wolf <kwolf@redhat.com>
Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
2013-06-24 17:13:17 +02:00
|
|
|
typedef struct DriveBackupState {
|
|
|
|
BlkTransactionState common;
|
|
|
|
BlockDriverState *bs;
|
|
|
|
BlockJob *job;
|
|
|
|
} DriveBackupState;
|
|
|
|
|
|
|
|
static void drive_backup_prepare(BlkTransactionState *common, Error **errp)
|
|
|
|
{
|
|
|
|
DriveBackupState *state = DO_UPCAST(DriveBackupState, common, common);
|
|
|
|
DriveBackup *backup;
|
|
|
|
Error *local_err = NULL;
|
|
|
|
|
|
|
|
assert(common->action->kind == TRANSACTION_ACTION_KIND_DRIVE_BACKUP);
|
|
|
|
backup = common->action->drive_backup;
|
|
|
|
|
|
|
|
qmp_drive_backup(backup->device, backup->target,
|
|
|
|
backup->has_format, backup->format,
|
2013-06-26 14:11:57 +02:00
|
|
|
backup->sync,
|
blockdev: add DriveBackup transaction
This patch adds a transactional version of the drive-backup QMP command.
It allows atomic snapshots of multiple drives along with automatic
cleanup if there is a failure to start one of the backup jobs.
Note that QMP events are emitted for block job completion/cancellation
and the block job will be listed by query-block-jobs.
@device: the name of the device whose writes should be mirrored.
@target: the target of the new image. If the file exists, or if it
is a device, the existing file/device will be used as the new
destination. If it does not exist, a new file will be created.
@format: #optional the format of the new destination, default is to
probe if @mode is 'existing', else the format of the source
@mode: #optional whether and how QEMU should create a new image, default is
'absolute-paths'.
@speed: #optional the maximum speed, in bytes per second
@on-source-error: #optional the action to take on an error on the source,
default 'report'. 'stop' and 'enospc' can only be used
if the block device supports io-status (see BlockInfo).
@on-target-error: #optional the action to take on an error on the target,
default 'report' (no limitations, since this applies to
a different block device than @device).
Reviewed-by: Eric Blake <eblake@redhat.com>
Reviewed-by: Kevin Wolf <kwolf@redhat.com>
Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
2013-06-24 17:13:17 +02:00
|
|
|
backup->has_mode, backup->mode,
|
|
|
|
backup->has_speed, backup->speed,
|
|
|
|
backup->has_on_source_error, backup->on_source_error,
|
|
|
|
backup->has_on_target_error, backup->on_target_error,
|
|
|
|
&local_err);
|
|
|
|
if (error_is_set(&local_err)) {
|
|
|
|
error_propagate(errp, local_err);
|
|
|
|
state->bs = NULL;
|
|
|
|
state->job = NULL;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
state->bs = bdrv_find(backup->device);
|
|
|
|
state->job = state->bs->job;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void drive_backup_abort(BlkTransactionState *common)
|
|
|
|
{
|
|
|
|
DriveBackupState *state = DO_UPCAST(DriveBackupState, common, common);
|
|
|
|
BlockDriverState *bs = state->bs;
|
|
|
|
|
|
|
|
/* Only cancel if it's the job we started */
|
|
|
|
if (bs && bs->job && bs->job == state->job) {
|
|
|
|
block_job_cancel_sync(bs->job);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-24 17:13:18 +02:00
|
|
|
static void abort_prepare(BlkTransactionState *common, Error **errp)
|
|
|
|
{
|
|
|
|
error_setg(errp, "Transaction aborted using Abort action");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void abort_commit(BlkTransactionState *common)
|
|
|
|
{
|
2013-07-25 18:21:28 +02:00
|
|
|
g_assert_not_reached(); /* this action never succeeds */
|
2013-06-24 17:13:18 +02:00
|
|
|
}
|
|
|
|
|
2013-05-08 18:25:16 +08:00
|
|
|
static const BdrvActionOps actions[] = {
|
2013-05-13 10:43:43 +02:00
|
|
|
[TRANSACTION_ACTION_KIND_BLOCKDEV_SNAPSHOT_SYNC] = {
|
2013-06-24 17:13:15 +02:00
|
|
|
.instance_size = sizeof(ExternalSnapshotState),
|
2013-05-08 18:25:16 +08:00
|
|
|
.prepare = external_snapshot_prepare,
|
|
|
|
.commit = external_snapshot_commit,
|
|
|
|
.abort = external_snapshot_abort,
|
|
|
|
},
|
blockdev: add DriveBackup transaction
This patch adds a transactional version of the drive-backup QMP command.
It allows atomic snapshots of multiple drives along with automatic
cleanup if there is a failure to start one of the backup jobs.
Note that QMP events are emitted for block job completion/cancellation
and the block job will be listed by query-block-jobs.
@device: the name of the device whose writes should be mirrored.
@target: the target of the new image. If the file exists, or if it
is a device, the existing file/device will be used as the new
destination. If it does not exist, a new file will be created.
@format: #optional the format of the new destination, default is to
probe if @mode is 'existing', else the format of the source
@mode: #optional whether and how QEMU should create a new image, default is
'absolute-paths'.
@speed: #optional the maximum speed, in bytes per second
@on-source-error: #optional the action to take on an error on the source,
default 'report'. 'stop' and 'enospc' can only be used
if the block device supports io-status (see BlockInfo).
@on-target-error: #optional the action to take on an error on the target,
default 'report' (no limitations, since this applies to
a different block device than @device).
Reviewed-by: Eric Blake <eblake@redhat.com>
Reviewed-by: Kevin Wolf <kwolf@redhat.com>
Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
2013-06-24 17:13:17 +02:00
|
|
|
[TRANSACTION_ACTION_KIND_DRIVE_BACKUP] = {
|
|
|
|
.instance_size = sizeof(DriveBackupState),
|
|
|
|
.prepare = drive_backup_prepare,
|
|
|
|
.abort = drive_backup_abort,
|
|
|
|
},
|
2013-06-24 17:13:18 +02:00
|
|
|
[TRANSACTION_ACTION_KIND_ABORT] = {
|
|
|
|
.instance_size = sizeof(BlkTransactionState),
|
|
|
|
.prepare = abort_prepare,
|
|
|
|
.commit = abort_commit,
|
|
|
|
},
|
2013-09-11 14:04:34 +08:00
|
|
|
[TRANSACTION_ACTION_KIND_BLOCKDEV_SNAPSHOT_INTERNAL_SYNC] = {
|
|
|
|
.instance_size = sizeof(InternalSnapshotState),
|
|
|
|
.prepare = internal_snapshot_prepare,
|
|
|
|
.abort = internal_snapshot_abort,
|
|
|
|
},
|
2013-05-08 18:25:16 +08:00
|
|
|
};
|
|
|
|
|
2012-02-28 15:54:06 -05:00
|
|
|
/*
|
|
|
|
* 'Atomic' group snapshots. The snapshots are taken as a set, and if any fail
|
|
|
|
* then we do not pivot any of the devices in the group, and abandon the
|
|
|
|
* snapshots
|
|
|
|
*/
|
2013-05-13 10:43:43 +02:00
|
|
|
void qmp_transaction(TransactionActionList *dev_list, Error **errp)
|
2012-02-28 15:54:06 -05:00
|
|
|
{
|
2013-05-13 10:43:43 +02:00
|
|
|
TransactionActionList *dev_entry = dev_list;
|
2013-06-24 17:13:15 +02:00
|
|
|
BlkTransactionState *state, *next;
|
2012-11-30 10:52:07 -02:00
|
|
|
Error *local_err = NULL;
|
2012-02-28 15:54:06 -05:00
|
|
|
|
2013-06-24 17:13:15 +02:00
|
|
|
QSIMPLEQ_HEAD(snap_bdrv_states, BlkTransactionState) snap_bdrv_states;
|
2012-02-28 15:54:06 -05:00
|
|
|
QSIMPLEQ_INIT(&snap_bdrv_states);
|
|
|
|
|
|
|
|
/* drain all i/o before any snapshots */
|
|
|
|
bdrv_drain_all();
|
|
|
|
|
|
|
|
/* We don't do anything in this loop that commits us to the snapshot */
|
|
|
|
while (NULL != dev_entry) {
|
2013-05-13 10:43:43 +02:00
|
|
|
TransactionAction *dev_info = NULL;
|
2013-05-08 18:25:16 +08:00
|
|
|
const BdrvActionOps *ops;
|
2012-03-06 18:55:57 +01:00
|
|
|
|
2012-02-28 15:54:06 -05:00
|
|
|
dev_info = dev_entry->value;
|
|
|
|
dev_entry = dev_entry->next;
|
|
|
|
|
2013-05-08 18:25:16 +08:00
|
|
|
assert(dev_info->kind < ARRAY_SIZE(actions));
|
|
|
|
|
|
|
|
ops = &actions[dev_info->kind];
|
2013-09-12 14:57:27 +02:00
|
|
|
assert(ops->instance_size > 0);
|
|
|
|
|
2013-06-24 17:13:15 +02:00
|
|
|
state = g_malloc0(ops->instance_size);
|
|
|
|
state->ops = ops;
|
|
|
|
state->action = dev_info;
|
|
|
|
QSIMPLEQ_INSERT_TAIL(&snap_bdrv_states, state, entry);
|
2012-02-28 15:54:06 -05:00
|
|
|
|
2013-06-24 17:13:15 +02:00
|
|
|
state->ops->prepare(state, &local_err);
|
2013-05-08 18:25:16 +08:00
|
|
|
if (error_is_set(&local_err)) {
|
|
|
|
error_propagate(errp, local_err);
|
|
|
|
goto delete_and_fail;
|
2012-02-28 15:54:06 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-24 17:13:15 +02:00
|
|
|
QSIMPLEQ_FOREACH(state, &snap_bdrv_states, entry) {
|
2013-06-24 17:13:16 +02:00
|
|
|
if (state->ops->commit) {
|
|
|
|
state->ops->commit(state);
|
|
|
|
}
|
2012-02-28 15:54:06 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/* success */
|
|
|
|
goto exit;
|
|
|
|
|
|
|
|
delete_and_fail:
|
|
|
|
/*
|
|
|
|
* failure, and it is all-or-none; abandon each new bs, and keep using
|
|
|
|
* the original bs for all images
|
|
|
|
*/
|
2013-06-24 17:13:15 +02:00
|
|
|
QSIMPLEQ_FOREACH(state, &snap_bdrv_states, entry) {
|
|
|
|
if (state->ops->abort) {
|
|
|
|
state->ops->abort(state);
|
2013-05-08 18:25:16 +08:00
|
|
|
}
|
2012-02-28 15:54:06 -05:00
|
|
|
}
|
|
|
|
exit:
|
2013-06-24 17:13:15 +02:00
|
|
|
QSIMPLEQ_FOREACH_SAFE(state, &snap_bdrv_states, entry, next) {
|
|
|
|
if (state->ops->clean) {
|
|
|
|
state->ops->clean(state);
|
2013-05-08 18:25:16 +08:00
|
|
|
}
|
2013-06-24 17:13:15 +02:00
|
|
|
g_free(state);
|
2012-02-28 15:54:06 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-12-07 15:47:23 -02:00
|
|
|
static void eject_device(BlockDriverState *bs, int force, Error **errp)
|
2010-06-02 18:48:27 +02:00
|
|
|
{
|
2012-01-18 14:40:41 +00:00
|
|
|
if (bdrv_in_use(bs)) {
|
|
|
|
error_set(errp, QERR_DEVICE_IN_USE, bdrv_get_device_name(bs));
|
|
|
|
return;
|
|
|
|
}
|
block: Clean up remaining users of "removable"
BlockDriverState member removable is a confused mess. It is true when
an ide-cd, scsi-cd or floppy qdev is attached, or when the
BlockDriverState was created with -drive if={floppy,sd} or -drive
if={ide,scsi,xen,none},media=cdrom ("created removable"), except when
an ide-hd, scsi-hd, scsi-generic or virtio-blk qdev is attached.
Three users remain:
1. eject_device(), via bdrv_is_removable() uses it to determine
whether a block device can eject media.
2. bdrv_info() is monitor command "info block". QMP documentation
says "true if the device is removable, false otherwise". From the
monitor user's point of view, the only sensible interpretation of
"is removable" is "can eject media with monitor commands eject and
change".
A block device can eject media unless a device is attached that
doesn't support it. Switch the two users over to new
bdrv_dev_has_removable_media() that returns exactly that.
3. bdrv_getlength() uses to suppress its length cache when media can
change (see commit 46a4e4e6). Media change is either monitor
command change (updates the length cache), monitor command eject
(doesn't update the length cache, easily fixable), or physical
media change (invalidates length cache, not so easily fixable).
I'm refraining from improving anything here, because this series is
long enough already. Instead, I simply switch it over to
bdrv_dev_has_removable_media() as well.
This changes the behavior of the length cache and of monitor commands
eject and change in two cases:
a. drive not created removable, no device attached
The commit makes the drive removable, and defeats the length cache.
Example: -drive if=none
b. drive created removable, but the attached drive is non-removable,
and doesn't call bdrv_set_removable(..., 0) (most devices don't)
The commit makes the drive non-removable, and enables the length
cache.
Example: -drive if=xen,media=cdrom -M xenpv
The other non-removable devices that don't call
bdrv_set_removable() can't currently use a drive created removable,
either because they aren't qdevified, or because they lack a drive
property. Won't stay that way.
Signed-off-by: Markus Armbruster <armbru@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
2011-09-06 18:58:51 +02:00
|
|
|
if (!bdrv_dev_has_removable_media(bs)) {
|
2011-12-07 15:47:23 -02:00
|
|
|
error_set(errp, QERR_DEVICE_NOT_REMOVABLE, bdrv_get_device_name(bs));
|
|
|
|
return;
|
2011-07-20 18:23:35 +02:00
|
|
|
}
|
2011-12-07 15:47:23 -02:00
|
|
|
|
2011-11-07 17:50:13 +01:00
|
|
|
if (bdrv_dev_is_medium_locked(bs) && !bdrv_dev_is_tray_open(bs)) {
|
|
|
|
bdrv_dev_eject_request(bs, force);
|
|
|
|
if (!force) {
|
2011-12-07 15:47:23 -02:00
|
|
|
error_set(errp, QERR_DEVICE_LOCKED, bdrv_get_device_name(bs));
|
|
|
|
return;
|
2011-11-07 17:50:13 +01:00
|
|
|
}
|
2010-06-02 18:48:27 +02:00
|
|
|
}
|
2011-12-07 15:47:23 -02:00
|
|
|
|
2010-06-01 19:12:19 -03:00
|
|
|
bdrv_close(bs);
|
2010-06-02 18:48:27 +02:00
|
|
|
}
|
|
|
|
|
2011-12-07 16:02:36 -02:00
|
|
|
void qmp_eject(const char *device, bool has_force, bool force, Error **errp)
|
2010-06-02 18:48:27 +02:00
|
|
|
{
|
|
|
|
BlockDriverState *bs;
|
|
|
|
|
2011-12-07 16:02:36 -02:00
|
|
|
bs = bdrv_find(device);
|
2010-06-02 18:48:27 +02:00
|
|
|
if (!bs) {
|
2011-12-07 16:02:36 -02:00
|
|
|
error_set(errp, QERR_DEVICE_NOT_FOUND, device);
|
|
|
|
return;
|
2011-12-07 15:47:23 -02:00
|
|
|
}
|
|
|
|
|
2011-12-07 16:02:36 -02:00
|
|
|
eject_device(bs, force, errp);
|
2010-06-02 18:48:27 +02:00
|
|
|
}
|
|
|
|
|
2011-11-23 13:28:21 -02:00
|
|
|
void qmp_block_passwd(const char *device, const char *password, Error **errp)
|
2010-06-02 18:48:27 +02:00
|
|
|
{
|
|
|
|
BlockDriverState *bs;
|
|
|
|
int err;
|
|
|
|
|
2011-11-23 13:28:21 -02:00
|
|
|
bs = bdrv_find(device);
|
2010-06-02 18:48:27 +02:00
|
|
|
if (!bs) {
|
2011-11-23 13:28:21 -02:00
|
|
|
error_set(errp, QERR_DEVICE_NOT_FOUND, device);
|
|
|
|
return;
|
2010-06-02 18:48:27 +02:00
|
|
|
}
|
|
|
|
|
2011-11-23 13:28:21 -02:00
|
|
|
err = bdrv_set_key(bs, password);
|
2010-06-02 18:48:27 +02:00
|
|
|
if (err == -EINVAL) {
|
2011-11-23 13:28:21 -02:00
|
|
|
error_set(errp, QERR_DEVICE_NOT_ENCRYPTED, bdrv_get_device_name(bs));
|
|
|
|
return;
|
2010-06-02 18:48:27 +02:00
|
|
|
} else if (err < 0) {
|
2011-11-23 13:28:21 -02:00
|
|
|
error_set(errp, QERR_INVALID_PASSWORD);
|
|
|
|
return;
|
2010-06-02 18:48:27 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-08 11:13:50 -02:00
|
|
|
static void qmp_bdrv_open_encrypted(BlockDriverState *bs, const char *filename,
|
|
|
|
int bdrv_flags, BlockDriver *drv,
|
|
|
|
const char *password, Error **errp)
|
|
|
|
{
|
2013-09-05 14:45:29 +02:00
|
|
|
Error *local_err = NULL;
|
2013-06-07 14:33:48 -04:00
|
|
|
int ret;
|
|
|
|
|
2013-09-05 14:45:29 +02:00
|
|
|
ret = bdrv_open(bs, filename, NULL, bdrv_flags, drv, &local_err);
|
2013-06-07 14:33:48 -04:00
|
|
|
if (ret < 0) {
|
2013-09-05 14:45:29 +02:00
|
|
|
error_propagate(errp, local_err);
|
2011-12-08 11:13:50 -02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bdrv_key_required(bs)) {
|
|
|
|
if (password) {
|
|
|
|
if (bdrv_set_key(bs, password) < 0) {
|
|
|
|
error_set(errp, QERR_INVALID_PASSWORD);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
error_set(errp, QERR_DEVICE_ENCRYPTED, bdrv_get_device_name(bs),
|
|
|
|
bdrv_get_encrypted_filename(bs));
|
|
|
|
}
|
|
|
|
} else if (password) {
|
|
|
|
error_set(errp, QERR_DEVICE_NOT_ENCRYPTED, bdrv_get_device_name(bs));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void qmp_change_blockdev(const char *device, const char *filename,
|
2013-12-01 22:23:37 +01:00
|
|
|
const char *format, Error **errp)
|
2010-06-02 18:48:27 +02:00
|
|
|
{
|
|
|
|
BlockDriverState *bs;
|
|
|
|
BlockDriver *drv = NULL;
|
|
|
|
int bdrv_flags;
|
2011-12-07 15:47:23 -02:00
|
|
|
Error *err = NULL;
|
2010-06-02 18:48:27 +02:00
|
|
|
|
|
|
|
bs = bdrv_find(device);
|
|
|
|
if (!bs) {
|
2011-12-08 11:13:50 -02:00
|
|
|
error_set(errp, QERR_DEVICE_NOT_FOUND, device);
|
|
|
|
return;
|
2010-06-02 18:48:27 +02:00
|
|
|
}
|
2011-12-08 11:13:50 -02:00
|
|
|
|
|
|
|
if (format) {
|
2013-05-29 19:35:40 +08:00
|
|
|
drv = bdrv_find_whitelisted_format(format, bs->read_only);
|
2010-06-02 18:48:27 +02:00
|
|
|
if (!drv) {
|
2011-12-08 11:13:50 -02:00
|
|
|
error_set(errp, QERR_INVALID_BLOCK_FORMAT, format);
|
|
|
|
return;
|
2010-06-02 18:48:27 +02:00
|
|
|
}
|
|
|
|
}
|
2011-12-08 11:13:50 -02:00
|
|
|
|
2011-12-07 15:47:23 -02:00
|
|
|
eject_device(bs, 0, &err);
|
|
|
|
if (error_is_set(&err)) {
|
2011-12-08 11:13:50 -02:00
|
|
|
error_propagate(errp, err);
|
|
|
|
return;
|
2010-06-02 18:48:27 +02:00
|
|
|
}
|
2011-12-08 11:13:50 -02:00
|
|
|
|
2010-06-25 15:26:48 +02:00
|
|
|
bdrv_flags = bdrv_is_read_only(bs) ? 0 : BDRV_O_RDWR;
|
2010-07-25 20:49:34 +00:00
|
|
|
bdrv_flags |= bdrv_is_snapshot(bs) ? BDRV_O_SNAPSHOT : 0;
|
2011-12-08 11:13:50 -02:00
|
|
|
|
|
|
|
qmp_bdrv_open_encrypted(bs, filename, bdrv_flags, drv, NULL, errp);
|
2010-06-02 18:48:27 +02:00
|
|
|
}
|
2010-11-12 11:07:13 -06:00
|
|
|
|
2011-11-08 13:00:31 +08:00
|
|
|
/* throttling disk I/O limits */
|
2011-12-14 16:49:14 -02:00
|
|
|
void qmp_block_set_io_throttle(const char *device, int64_t bps, int64_t bps_rd,
|
2013-09-02 14:14:40 +02:00
|
|
|
int64_t bps_wr,
|
|
|
|
int64_t iops,
|
|
|
|
int64_t iops_rd,
|
|
|
|
int64_t iops_wr,
|
|
|
|
bool has_bps_max,
|
|
|
|
int64_t bps_max,
|
|
|
|
bool has_bps_rd_max,
|
|
|
|
int64_t bps_rd_max,
|
|
|
|
bool has_bps_wr_max,
|
|
|
|
int64_t bps_wr_max,
|
|
|
|
bool has_iops_max,
|
|
|
|
int64_t iops_max,
|
|
|
|
bool has_iops_rd_max,
|
|
|
|
int64_t iops_rd_max,
|
|
|
|
bool has_iops_wr_max,
|
2013-09-02 14:14:41 +02:00
|
|
|
int64_t iops_wr_max,
|
|
|
|
bool has_iops_size,
|
|
|
|
int64_t iops_size, Error **errp)
|
2011-11-08 13:00:31 +08:00
|
|
|
{
|
2013-09-02 14:14:39 +02:00
|
|
|
ThrottleConfig cfg;
|
2011-11-08 13:00:31 +08:00
|
|
|
BlockDriverState *bs;
|
|
|
|
|
2011-12-14 16:49:14 -02:00
|
|
|
bs = bdrv_find(device);
|
2011-11-08 13:00:31 +08:00
|
|
|
if (!bs) {
|
2011-12-14 16:49:14 -02:00
|
|
|
error_set(errp, QERR_DEVICE_NOT_FOUND, device);
|
|
|
|
return;
|
2011-11-08 13:00:31 +08:00
|
|
|
}
|
|
|
|
|
2013-09-02 14:14:39 +02:00
|
|
|
memset(&cfg, 0, sizeof(cfg));
|
|
|
|
cfg.buckets[THROTTLE_BPS_TOTAL].avg = bps;
|
|
|
|
cfg.buckets[THROTTLE_BPS_READ].avg = bps_rd;
|
|
|
|
cfg.buckets[THROTTLE_BPS_WRITE].avg = bps_wr;
|
|
|
|
|
|
|
|
cfg.buckets[THROTTLE_OPS_TOTAL].avg = iops;
|
|
|
|
cfg.buckets[THROTTLE_OPS_READ].avg = iops_rd;
|
|
|
|
cfg.buckets[THROTTLE_OPS_WRITE].avg = iops_wr;
|
|
|
|
|
2013-09-02 14:14:40 +02:00
|
|
|
if (has_bps_max) {
|
|
|
|
cfg.buckets[THROTTLE_BPS_TOTAL].max = bps_max;
|
|
|
|
}
|
|
|
|
if (has_bps_rd_max) {
|
|
|
|
cfg.buckets[THROTTLE_BPS_READ].max = bps_rd_max;
|
|
|
|
}
|
|
|
|
if (has_bps_wr_max) {
|
|
|
|
cfg.buckets[THROTTLE_BPS_WRITE].max = bps_wr_max;
|
|
|
|
}
|
|
|
|
if (has_iops_max) {
|
|
|
|
cfg.buckets[THROTTLE_OPS_TOTAL].max = iops_max;
|
|
|
|
}
|
|
|
|
if (has_iops_rd_max) {
|
|
|
|
cfg.buckets[THROTTLE_OPS_READ].max = iops_rd_max;
|
|
|
|
}
|
|
|
|
if (has_iops_wr_max) {
|
|
|
|
cfg.buckets[THROTTLE_OPS_WRITE].max = iops_wr_max;
|
|
|
|
}
|
2011-11-08 13:00:31 +08:00
|
|
|
|
2013-09-02 14:14:41 +02:00
|
|
|
if (has_iops_size) {
|
|
|
|
cfg.op_size = iops_size;
|
|
|
|
}
|
2013-09-02 14:14:39 +02:00
|
|
|
|
|
|
|
if (!check_throttle_config(&cfg, errp)) {
|
2011-12-14 16:49:14 -02:00
|
|
|
return;
|
2011-11-08 13:00:31 +08:00
|
|
|
}
|
|
|
|
|
2013-09-02 14:14:39 +02:00
|
|
|
if (!bs->io_limits_enabled && throttle_enabled(&cfg)) {
|
2011-11-08 13:00:31 +08:00
|
|
|
bdrv_io_limits_enable(bs);
|
2013-09-02 14:14:39 +02:00
|
|
|
} else if (bs->io_limits_enabled && !throttle_enabled(&cfg)) {
|
2011-11-08 13:00:31 +08:00
|
|
|
bdrv_io_limits_disable(bs);
|
2013-09-02 14:14:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (bs->io_limits_enabled) {
|
|
|
|
bdrv_set_io_limits(bs, &cfg);
|
2011-11-08 13:00:31 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-11-12 11:07:13 -06:00
|
|
|
int do_drive_del(Monitor *mon, const QDict *qdict, QObject **ret_data)
|
|
|
|
{
|
|
|
|
const char *id = qdict_get_str(qdict, "id");
|
|
|
|
BlockDriverState *bs;
|
|
|
|
|
|
|
|
bs = bdrv_find(id);
|
|
|
|
if (!bs) {
|
|
|
|
qerror_report(QERR_DEVICE_NOT_FOUND, id);
|
|
|
|
return -1;
|
|
|
|
}
|
2011-01-26 12:12:35 -02:00
|
|
|
if (bdrv_in_use(bs)) {
|
|
|
|
qerror_report(QERR_DEVICE_IN_USE, id);
|
|
|
|
return -1;
|
|
|
|
}
|
2010-11-12 11:07:13 -06:00
|
|
|
|
|
|
|
/* quiesce block driver; prevent further io */
|
2011-11-30 12:23:43 +00:00
|
|
|
bdrv_drain_all();
|
2010-11-12 11:07:13 -06:00
|
|
|
bdrv_flush(bs);
|
|
|
|
bdrv_close(bs);
|
|
|
|
|
2011-08-03 15:07:40 +02:00
|
|
|
/* if we have a device attached to this BlockDriverState
|
Do not delete BlockDriverState when deleting the drive
When removing a drive from the host-side via drive_del we currently have
the following path:
drive_del
qemu_aio_flush()
bdrv_close() // zaps bs->drv, which makes any subsequent I/O get
// dropped. Works as designed
drive_uninit()
bdrv_delete() // frees the bs. Since the device is still connected to
// bs, any subsequent I/O is a use-after-free.
The value of bs->drv becomes unpredictable on free. As long as it
remains null, I/O still gets dropped, however it could become non-null
at any point after the free resulting SEGVs or other QEMU state
corruption.
To resolve this issue as simply as possible, we can chose to not
actually delete the BlockDriverState pointer. Since bdrv_close()
handles setting the drv pointer to NULL, we just need to remove the
BlockDriverState from the QLIST that is used to enumerate the block
devices. This is currently handled within bdrv_delete, so move this
into its own function, bdrv_make_anon().
The result is that we can now invoke drive_del, this closes the file
descriptors and sets BlockDriverState->drv to NULL which prevents futher
IO to the device, and since we do not free BlockDriverState, we don't
have to worry about the copy retained in the block devices.
We also don't attempt to remove the qdev property since we are no longer
deleting the BlockDriverState on drives with associated drives. This
also allows for removing Drives with no devices associated either.
Reported-by: Markus Armbruster <armbru@redhat.com>
Signed-off-by: Ryan Harper <ryanh@us.ibm.com>
Acked-by: Markus Armbruster <armbru@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
2011-03-29 20:51:47 -05:00
|
|
|
* then we need to make the drive anonymous until the device
|
|
|
|
* can be removed. If this is a drive with no device backing
|
|
|
|
* then we can just get rid of the block driver state right here.
|
|
|
|
*/
|
2011-08-03 15:07:40 +02:00
|
|
|
if (bdrv_get_attached_dev(bs)) {
|
Do not delete BlockDriverState when deleting the drive
When removing a drive from the host-side via drive_del we currently have
the following path:
drive_del
qemu_aio_flush()
bdrv_close() // zaps bs->drv, which makes any subsequent I/O get
// dropped. Works as designed
drive_uninit()
bdrv_delete() // frees the bs. Since the device is still connected to
// bs, any subsequent I/O is a use-after-free.
The value of bs->drv becomes unpredictable on free. As long as it
remains null, I/O still gets dropped, however it could become non-null
at any point after the free resulting SEGVs or other QEMU state
corruption.
To resolve this issue as simply as possible, we can chose to not
actually delete the BlockDriverState pointer. Since bdrv_close()
handles setting the drv pointer to NULL, we just need to remove the
BlockDriverState from the QLIST that is used to enumerate the block
devices. This is currently handled within bdrv_delete, so move this
into its own function, bdrv_make_anon().
The result is that we can now invoke drive_del, this closes the file
descriptors and sets BlockDriverState->drv to NULL which prevents futher
IO to the device, and since we do not free BlockDriverState, we don't
have to worry about the copy retained in the block devices.
We also don't attempt to remove the qdev property since we are no longer
deleting the BlockDriverState on drives with associated drives. This
also allows for removing Drives with no devices associated either.
Reported-by: Markus Armbruster <armbru@redhat.com>
Signed-off-by: Ryan Harper <ryanh@us.ibm.com>
Acked-by: Markus Armbruster <armbru@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
2011-03-29 20:51:47 -05:00
|
|
|
bdrv_make_anon(bs);
|
2013-06-05 10:33:14 +02:00
|
|
|
|
|
|
|
/* Further I/O must not pause the guest */
|
|
|
|
bdrv_set_on_error(bs, BLOCKDEV_ON_ERROR_REPORT,
|
|
|
|
BLOCKDEV_ON_ERROR_REPORT);
|
Do not delete BlockDriverState when deleting the drive
When removing a drive from the host-side via drive_del we currently have
the following path:
drive_del
qemu_aio_flush()
bdrv_close() // zaps bs->drv, which makes any subsequent I/O get
// dropped. Works as designed
drive_uninit()
bdrv_delete() // frees the bs. Since the device is still connected to
// bs, any subsequent I/O is a use-after-free.
The value of bs->drv becomes unpredictable on free. As long as it
remains null, I/O still gets dropped, however it could become non-null
at any point after the free resulting SEGVs or other QEMU state
corruption.
To resolve this issue as simply as possible, we can chose to not
actually delete the BlockDriverState pointer. Since bdrv_close()
handles setting the drv pointer to NULL, we just need to remove the
BlockDriverState from the QLIST that is used to enumerate the block
devices. This is currently handled within bdrv_delete, so move this
into its own function, bdrv_make_anon().
The result is that we can now invoke drive_del, this closes the file
descriptors and sets BlockDriverState->drv to NULL which prevents futher
IO to the device, and since we do not free BlockDriverState, we don't
have to worry about the copy retained in the block devices.
We also don't attempt to remove the qdev property since we are no longer
deleting the BlockDriverState on drives with associated drives. This
also allows for removing Drives with no devices associated either.
Reported-by: Markus Armbruster <armbru@redhat.com>
Signed-off-by: Ryan Harper <ryanh@us.ibm.com>
Acked-by: Markus Armbruster <armbru@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
2011-03-29 20:51:47 -05:00
|
|
|
} else {
|
|
|
|
drive_uninit(drive_get_by_blockdev(bs));
|
2010-11-12 11:07:13 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2011-01-24 13:32:33 +01:00
|
|
|
|
2011-11-25 14:57:10 -02:00
|
|
|
void qmp_block_resize(const char *device, int64_t size, Error **errp)
|
2011-01-24 13:32:33 +01:00
|
|
|
{
|
|
|
|
BlockDriverState *bs;
|
2013-05-02 15:32:55 +02:00
|
|
|
int ret;
|
2011-01-24 13:32:33 +01:00
|
|
|
|
|
|
|
bs = bdrv_find(device);
|
|
|
|
if (!bs) {
|
2011-11-25 14:57:10 -02:00
|
|
|
error_set(errp, QERR_DEVICE_NOT_FOUND, device);
|
|
|
|
return;
|
2011-01-24 13:32:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (size < 0) {
|
2012-01-04 22:23:34 +00:00
|
|
|
error_set(errp, QERR_INVALID_PARAMETER_VALUE, "size", "a >0 size");
|
2011-11-25 14:57:10 -02:00
|
|
|
return;
|
2011-01-24 13:32:33 +01:00
|
|
|
}
|
|
|
|
|
2013-03-11 11:04:24 +01:00
|
|
|
/* complete all in-flight operations before resizing the device */
|
|
|
|
bdrv_drain_all();
|
|
|
|
|
2013-05-02 15:32:55 +02:00
|
|
|
ret = bdrv_truncate(bs, size);
|
|
|
|
switch (ret) {
|
2012-01-04 22:23:34 +00:00
|
|
|
case 0:
|
|
|
|
break;
|
|
|
|
case -ENOMEDIUM:
|
|
|
|
error_set(errp, QERR_DEVICE_HAS_NO_MEDIUM, device);
|
|
|
|
break;
|
|
|
|
case -ENOTSUP:
|
|
|
|
error_set(errp, QERR_UNSUPPORTED);
|
|
|
|
break;
|
|
|
|
case -EACCES:
|
|
|
|
error_set(errp, QERR_DEVICE_IS_READ_ONLY, device);
|
|
|
|
break;
|
|
|
|
case -EBUSY:
|
|
|
|
error_set(errp, QERR_DEVICE_IN_USE, device);
|
|
|
|
break;
|
|
|
|
default:
|
2013-05-02 15:32:55 +02:00
|
|
|
error_setg_errno(errp, -ret, "Could not resize");
|
2012-01-04 22:23:34 +00:00
|
|
|
break;
|
2011-01-24 13:32:33 +01:00
|
|
|
}
|
|
|
|
}
|
2012-01-18 14:40:46 +00:00
|
|
|
|
2012-09-27 13:29:14 -04:00
|
|
|
static void block_job_cb(void *opaque, int ret)
|
2012-01-18 14:40:46 +00:00
|
|
|
{
|
|
|
|
BlockDriverState *bs = opaque;
|
|
|
|
QObject *obj;
|
|
|
|
|
2012-09-27 13:29:14 -04:00
|
|
|
trace_block_job_cb(bs, bs->job, ret);
|
2012-01-18 14:40:46 +00:00
|
|
|
|
|
|
|
assert(bs->job);
|
|
|
|
obj = qobject_from_block_job(bs->job);
|
|
|
|
if (ret < 0) {
|
|
|
|
QDict *dict = qobject_to_qdict(obj);
|
|
|
|
qdict_put(dict, "error", qstring_from_str(strerror(-ret)));
|
|
|
|
}
|
|
|
|
|
2012-01-18 14:40:48 +00:00
|
|
|
if (block_job_is_cancelled(bs->job)) {
|
|
|
|
monitor_protocol_event(QEVENT_BLOCK_JOB_CANCELLED, obj);
|
|
|
|
} else {
|
|
|
|
monitor_protocol_event(QEVENT_BLOCK_JOB_COMPLETED, obj);
|
|
|
|
}
|
2012-01-18 14:40:46 +00:00
|
|
|
qobject_decref(obj);
|
2012-01-18 14:40:50 +00:00
|
|
|
|
2013-08-23 09:14:51 +08:00
|
|
|
bdrv_put_ref_bh_schedule(bs);
|
2012-01-18 14:40:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void qmp_block_stream(const char *device, bool has_base,
|
2012-09-28 17:22:59 +02:00
|
|
|
const char *base, bool has_speed, int64_t speed,
|
|
|
|
bool has_on_error, BlockdevOnError on_error,
|
|
|
|
Error **errp)
|
2012-01-18 14:40:46 +00:00
|
|
|
{
|
|
|
|
BlockDriverState *bs;
|
2012-01-18 14:40:53 +00:00
|
|
|
BlockDriverState *base_bs = NULL;
|
2012-04-25 16:51:00 +01:00
|
|
|
Error *local_err = NULL;
|
2012-01-18 14:40:46 +00:00
|
|
|
|
2012-09-28 17:22:59 +02:00
|
|
|
if (!has_on_error) {
|
|
|
|
on_error = BLOCKDEV_ON_ERROR_REPORT;
|
|
|
|
}
|
|
|
|
|
2012-01-18 14:40:46 +00:00
|
|
|
bs = bdrv_find(device);
|
|
|
|
if (!bs) {
|
|
|
|
error_set(errp, QERR_DEVICE_NOT_FOUND, device);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (base) {
|
2012-01-18 14:40:53 +00:00
|
|
|
base_bs = bdrv_find_backing_image(bs, base);
|
|
|
|
if (base_bs == NULL) {
|
|
|
|
error_set(errp, QERR_BASE_NOT_FOUND, base);
|
|
|
|
return;
|
|
|
|
}
|
2012-01-18 14:40:46 +00:00
|
|
|
}
|
|
|
|
|
2012-04-25 16:51:03 +01:00
|
|
|
stream_start(bs, base_bs, base, has_speed ? speed : 0,
|
2012-09-28 17:22:59 +02:00
|
|
|
on_error, block_job_cb, bs, &local_err);
|
2012-04-25 16:51:00 +01:00
|
|
|
if (error_is_set(&local_err)) {
|
|
|
|
error_propagate(errp, local_err);
|
|
|
|
return;
|
2012-01-18 14:40:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
trace_qmp_block_stream(bs, bs->job);
|
|
|
|
}
|
2012-01-18 14:40:47 +00:00
|
|
|
|
QAPI: add command for live block commit, 'block-commit'
The command for live block commit is added, which has the following
arguments:
device: the block device to perform the commit on (mandatory)
base: the base image to commit into; optional (if not specified,
it is the underlying original image)
top: the top image of the commit - all data from inside top down
to base will be committed into base (mandatory for now; see
note, below)
speed: maximum speed, in bytes/sec
Note: Eventually this command will support merging down the active layer,
but that code is not yet complete. If the active layer is passed
in as top, then an error will be returned. Once merging down the
active layer is supported, the 'top' argument may become optional,
and default to the active layer.
The is done as a block job, so upon completion a BLOCK_JOB_COMPLETED will
be emitted.
Signed-off-by: Jeff Cody <jcody@redhat.com>
Reviewed-by: Eric Blake <eblake@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
2012-09-27 13:29:16 -04:00
|
|
|
void qmp_block_commit(const char *device,
|
|
|
|
bool has_base, const char *base, const char *top,
|
|
|
|
bool has_speed, int64_t speed,
|
|
|
|
Error **errp)
|
|
|
|
{
|
|
|
|
BlockDriverState *bs;
|
|
|
|
BlockDriverState *base_bs, *top_bs;
|
|
|
|
Error *local_err = NULL;
|
|
|
|
/* This will be part of the QMP command, if/when the
|
|
|
|
* BlockdevOnError change for blkmirror makes it in
|
|
|
|
*/
|
2012-09-28 17:22:55 +02:00
|
|
|
BlockdevOnError on_error = BLOCKDEV_ON_ERROR_REPORT;
|
QAPI: add command for live block commit, 'block-commit'
The command for live block commit is added, which has the following
arguments:
device: the block device to perform the commit on (mandatory)
base: the base image to commit into; optional (if not specified,
it is the underlying original image)
top: the top image of the commit - all data from inside top down
to base will be committed into base (mandatory for now; see
note, below)
speed: maximum speed, in bytes/sec
Note: Eventually this command will support merging down the active layer,
but that code is not yet complete. If the active layer is passed
in as top, then an error will be returned. Once merging down the
active layer is supported, the 'top' argument may become optional,
and default to the active layer.
The is done as a block job, so upon completion a BLOCK_JOB_COMPLETED will
be emitted.
Signed-off-by: Jeff Cody <jcody@redhat.com>
Reviewed-by: Eric Blake <eblake@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
2012-09-27 13:29:16 -04:00
|
|
|
|
|
|
|
/* drain all i/o before commits */
|
|
|
|
bdrv_drain_all();
|
|
|
|
|
|
|
|
bs = bdrv_find(device);
|
|
|
|
if (!bs) {
|
|
|
|
error_set(errp, QERR_DEVICE_NOT_FOUND, device);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* default top_bs is the active layer */
|
|
|
|
top_bs = bs;
|
|
|
|
|
|
|
|
if (top) {
|
|
|
|
if (strcmp(bs->filename, top) != 0) {
|
|
|
|
top_bs = bdrv_find_backing_image(bs, top);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (top_bs == NULL) {
|
|
|
|
error_setg(errp, "Top image file %s not found", top ? top : "NULL");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-10-16 15:49:10 -04:00
|
|
|
if (has_base && base) {
|
|
|
|
base_bs = bdrv_find_backing_image(top_bs, base);
|
|
|
|
} else {
|
|
|
|
base_bs = bdrv_find_base(top_bs);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (base_bs == NULL) {
|
|
|
|
error_set(errp, QERR_BASE_NOT_FOUND, base ? base : "NULL");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-12-16 14:45:31 +08:00
|
|
|
if (top_bs == bs) {
|
|
|
|
commit_active_start(bs, base_bs, speed, on_error, block_job_cb,
|
|
|
|
bs, &local_err);
|
|
|
|
} else {
|
|
|
|
commit_start(bs, base_bs, top_bs, speed, on_error, block_job_cb, bs,
|
|
|
|
&local_err);
|
|
|
|
}
|
QAPI: add command for live block commit, 'block-commit'
The command for live block commit is added, which has the following
arguments:
device: the block device to perform the commit on (mandatory)
base: the base image to commit into; optional (if not specified,
it is the underlying original image)
top: the top image of the commit - all data from inside top down
to base will be committed into base (mandatory for now; see
note, below)
speed: maximum speed, in bytes/sec
Note: Eventually this command will support merging down the active layer,
but that code is not yet complete. If the active layer is passed
in as top, then an error will be returned. Once merging down the
active layer is supported, the 'top' argument may become optional,
and default to the active layer.
The is done as a block job, so upon completion a BLOCK_JOB_COMPLETED will
be emitted.
Signed-off-by: Jeff Cody <jcody@redhat.com>
Reviewed-by: Eric Blake <eblake@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
2012-09-27 13:29:16 -04:00
|
|
|
if (local_err != NULL) {
|
|
|
|
error_propagate(errp, local_err);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
block: add drive-backup QMP command
@drive-backup
Start a point-in-time copy of a block device to a new destination. The
status of ongoing drive-backup operations can be checked with
query-block-jobs where the BlockJobInfo.type field has the value 'backup'.
The operation can be stopped before it has completed using the
block-job-cancel command.
@device: the name of the device which should be copied.
@target: the target of the new image. If the file exists, or if it
is a device, the existing file/device will be used as the new
destination. If it does not exist, a new file will be created.
@format: #optional the format of the new destination, default is to
probe if @mode is 'existing', else the format of the source
@mode: #optional whether and how QEMU should create a new image, default is
'absolute-paths'.
@speed: #optional the maximum speed, in bytes per second
@on-source-error: #optional the action to take on an error on the source,
default 'report'. 'stop' and 'enospc' can only be used
if the block device supports io-status (see BlockInfo).
@on-target-error: #optional the action to take on an error on the target,
default 'report' (no limitations, since this applies to
a different block device than @device).
Note that @on-source-error and @on-target-error only affect background I/O.
If an error occurs during a guest write request, the device's rerror/werror
actions will be used.
Returns: nothing on success
If @device is not a valid block device, DeviceNotFound
Since 1.6
Reviewed-by: Eric Blake <eblake@redhat.com>
Reviewed-by: Kevin Wolf <kwolf@redhat.com>
Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
2013-06-24 17:13:14 +02:00
|
|
|
void qmp_drive_backup(const char *device, const char *target,
|
|
|
|
bool has_format, const char *format,
|
2013-06-26 14:11:57 +02:00
|
|
|
enum MirrorSyncMode sync,
|
block: add drive-backup QMP command
@drive-backup
Start a point-in-time copy of a block device to a new destination. The
status of ongoing drive-backup operations can be checked with
query-block-jobs where the BlockJobInfo.type field has the value 'backup'.
The operation can be stopped before it has completed using the
block-job-cancel command.
@device: the name of the device which should be copied.
@target: the target of the new image. If the file exists, or if it
is a device, the existing file/device will be used as the new
destination. If it does not exist, a new file will be created.
@format: #optional the format of the new destination, default is to
probe if @mode is 'existing', else the format of the source
@mode: #optional whether and how QEMU should create a new image, default is
'absolute-paths'.
@speed: #optional the maximum speed, in bytes per second
@on-source-error: #optional the action to take on an error on the source,
default 'report'. 'stop' and 'enospc' can only be used
if the block device supports io-status (see BlockInfo).
@on-target-error: #optional the action to take on an error on the target,
default 'report' (no limitations, since this applies to
a different block device than @device).
Note that @on-source-error and @on-target-error only affect background I/O.
If an error occurs during a guest write request, the device's rerror/werror
actions will be used.
Returns: nothing on success
If @device is not a valid block device, DeviceNotFound
Since 1.6
Reviewed-by: Eric Blake <eblake@redhat.com>
Reviewed-by: Kevin Wolf <kwolf@redhat.com>
Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
2013-06-24 17:13:14 +02:00
|
|
|
bool has_mode, enum NewImageMode mode,
|
|
|
|
bool has_speed, int64_t speed,
|
|
|
|
bool has_on_source_error, BlockdevOnError on_source_error,
|
|
|
|
bool has_on_target_error, BlockdevOnError on_target_error,
|
|
|
|
Error **errp)
|
|
|
|
{
|
|
|
|
BlockDriverState *bs;
|
|
|
|
BlockDriverState *target_bs;
|
2013-07-26 11:39:04 -07:00
|
|
|
BlockDriverState *source = NULL;
|
block: add drive-backup QMP command
@drive-backup
Start a point-in-time copy of a block device to a new destination. The
status of ongoing drive-backup operations can be checked with
query-block-jobs where the BlockJobInfo.type field has the value 'backup'.
The operation can be stopped before it has completed using the
block-job-cancel command.
@device: the name of the device which should be copied.
@target: the target of the new image. If the file exists, or if it
is a device, the existing file/device will be used as the new
destination. If it does not exist, a new file will be created.
@format: #optional the format of the new destination, default is to
probe if @mode is 'existing', else the format of the source
@mode: #optional whether and how QEMU should create a new image, default is
'absolute-paths'.
@speed: #optional the maximum speed, in bytes per second
@on-source-error: #optional the action to take on an error on the source,
default 'report'. 'stop' and 'enospc' can only be used
if the block device supports io-status (see BlockInfo).
@on-target-error: #optional the action to take on an error on the target,
default 'report' (no limitations, since this applies to
a different block device than @device).
Note that @on-source-error and @on-target-error only affect background I/O.
If an error occurs during a guest write request, the device's rerror/werror
actions will be used.
Returns: nothing on success
If @device is not a valid block device, DeviceNotFound
Since 1.6
Reviewed-by: Eric Blake <eblake@redhat.com>
Reviewed-by: Kevin Wolf <kwolf@redhat.com>
Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
2013-06-24 17:13:14 +02:00
|
|
|
BlockDriver *drv = NULL;
|
|
|
|
Error *local_err = NULL;
|
|
|
|
int flags;
|
|
|
|
int64_t size;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (!has_speed) {
|
|
|
|
speed = 0;
|
|
|
|
}
|
|
|
|
if (!has_on_source_error) {
|
|
|
|
on_source_error = BLOCKDEV_ON_ERROR_REPORT;
|
|
|
|
}
|
|
|
|
if (!has_on_target_error) {
|
|
|
|
on_target_error = BLOCKDEV_ON_ERROR_REPORT;
|
|
|
|
}
|
|
|
|
if (!has_mode) {
|
|
|
|
mode = NEW_IMAGE_MODE_ABSOLUTE_PATHS;
|
|
|
|
}
|
|
|
|
|
|
|
|
bs = bdrv_find(device);
|
|
|
|
if (!bs) {
|
|
|
|
error_set(errp, QERR_DEVICE_NOT_FOUND, device);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!bdrv_is_inserted(bs)) {
|
|
|
|
error_set(errp, QERR_DEVICE_HAS_NO_MEDIUM, device);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!has_format) {
|
|
|
|
format = mode == NEW_IMAGE_MODE_EXISTING ? NULL : bs->drv->format_name;
|
|
|
|
}
|
|
|
|
if (format) {
|
|
|
|
drv = bdrv_find_format(format);
|
|
|
|
if (!drv) {
|
|
|
|
error_set(errp, QERR_INVALID_BLOCK_FORMAT, format);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bdrv_in_use(bs)) {
|
|
|
|
error_set(errp, QERR_DEVICE_IN_USE, device);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
flags = bs->open_flags | BDRV_O_RDWR;
|
|
|
|
|
2013-07-26 11:39:04 -07:00
|
|
|
/* See if we have a backing HD we can use to create our new image
|
|
|
|
* on top of. */
|
|
|
|
if (sync == MIRROR_SYNC_MODE_TOP) {
|
|
|
|
source = bs->backing_hd;
|
|
|
|
if (!source) {
|
|
|
|
sync = MIRROR_SYNC_MODE_FULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (sync == MIRROR_SYNC_MODE_NONE) {
|
|
|
|
source = bs;
|
|
|
|
}
|
|
|
|
|
block: add drive-backup QMP command
@drive-backup
Start a point-in-time copy of a block device to a new destination. The
status of ongoing drive-backup operations can be checked with
query-block-jobs where the BlockJobInfo.type field has the value 'backup'.
The operation can be stopped before it has completed using the
block-job-cancel command.
@device: the name of the device which should be copied.
@target: the target of the new image. If the file exists, or if it
is a device, the existing file/device will be used as the new
destination. If it does not exist, a new file will be created.
@format: #optional the format of the new destination, default is to
probe if @mode is 'existing', else the format of the source
@mode: #optional whether and how QEMU should create a new image, default is
'absolute-paths'.
@speed: #optional the maximum speed, in bytes per second
@on-source-error: #optional the action to take on an error on the source,
default 'report'. 'stop' and 'enospc' can only be used
if the block device supports io-status (see BlockInfo).
@on-target-error: #optional the action to take on an error on the target,
default 'report' (no limitations, since this applies to
a different block device than @device).
Note that @on-source-error and @on-target-error only affect background I/O.
If an error occurs during a guest write request, the device's rerror/werror
actions will be used.
Returns: nothing on success
If @device is not a valid block device, DeviceNotFound
Since 1.6
Reviewed-by: Eric Blake <eblake@redhat.com>
Reviewed-by: Kevin Wolf <kwolf@redhat.com>
Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
2013-06-24 17:13:14 +02:00
|
|
|
size = bdrv_getlength(bs);
|
|
|
|
if (size < 0) {
|
|
|
|
error_setg_errno(errp, -size, "bdrv_getlength failed");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mode != NEW_IMAGE_MODE_EXISTING) {
|
|
|
|
assert(format && drv);
|
2013-07-26 11:39:04 -07:00
|
|
|
if (source) {
|
|
|
|
bdrv_img_create(target, format, source->filename,
|
|
|
|
source->drv->format_name, NULL,
|
|
|
|
size, flags, &local_err, false);
|
|
|
|
} else {
|
|
|
|
bdrv_img_create(target, format, NULL, NULL, NULL,
|
|
|
|
size, flags, &local_err, false);
|
|
|
|
}
|
block: add drive-backup QMP command
@drive-backup
Start a point-in-time copy of a block device to a new destination. The
status of ongoing drive-backup operations can be checked with
query-block-jobs where the BlockJobInfo.type field has the value 'backup'.
The operation can be stopped before it has completed using the
block-job-cancel command.
@device: the name of the device which should be copied.
@target: the target of the new image. If the file exists, or if it
is a device, the existing file/device will be used as the new
destination. If it does not exist, a new file will be created.
@format: #optional the format of the new destination, default is to
probe if @mode is 'existing', else the format of the source
@mode: #optional whether and how QEMU should create a new image, default is
'absolute-paths'.
@speed: #optional the maximum speed, in bytes per second
@on-source-error: #optional the action to take on an error on the source,
default 'report'. 'stop' and 'enospc' can only be used
if the block device supports io-status (see BlockInfo).
@on-target-error: #optional the action to take on an error on the target,
default 'report' (no limitations, since this applies to
a different block device than @device).
Note that @on-source-error and @on-target-error only affect background I/O.
If an error occurs during a guest write request, the device's rerror/werror
actions will be used.
Returns: nothing on success
If @device is not a valid block device, DeviceNotFound
Since 1.6
Reviewed-by: Eric Blake <eblake@redhat.com>
Reviewed-by: Kevin Wolf <kwolf@redhat.com>
Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
2013-06-24 17:13:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (error_is_set(&local_err)) {
|
|
|
|
error_propagate(errp, local_err);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
target_bs = bdrv_new("");
|
2013-09-05 14:45:29 +02:00
|
|
|
ret = bdrv_open(target_bs, target, NULL, flags, drv, &local_err);
|
block: add drive-backup QMP command
@drive-backup
Start a point-in-time copy of a block device to a new destination. The
status of ongoing drive-backup operations can be checked with
query-block-jobs where the BlockJobInfo.type field has the value 'backup'.
The operation can be stopped before it has completed using the
block-job-cancel command.
@device: the name of the device which should be copied.
@target: the target of the new image. If the file exists, or if it
is a device, the existing file/device will be used as the new
destination. If it does not exist, a new file will be created.
@format: #optional the format of the new destination, default is to
probe if @mode is 'existing', else the format of the source
@mode: #optional whether and how QEMU should create a new image, default is
'absolute-paths'.
@speed: #optional the maximum speed, in bytes per second
@on-source-error: #optional the action to take on an error on the source,
default 'report'. 'stop' and 'enospc' can only be used
if the block device supports io-status (see BlockInfo).
@on-target-error: #optional the action to take on an error on the target,
default 'report' (no limitations, since this applies to
a different block device than @device).
Note that @on-source-error and @on-target-error only affect background I/O.
If an error occurs during a guest write request, the device's rerror/werror
actions will be used.
Returns: nothing on success
If @device is not a valid block device, DeviceNotFound
Since 1.6
Reviewed-by: Eric Blake <eblake@redhat.com>
Reviewed-by: Kevin Wolf <kwolf@redhat.com>
Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
2013-06-24 17:13:14 +02:00
|
|
|
if (ret < 0) {
|
2013-08-23 09:14:47 +08:00
|
|
|
bdrv_unref(target_bs);
|
2013-09-05 14:45:29 +02:00
|
|
|
error_propagate(errp, local_err);
|
block: add drive-backup QMP command
@drive-backup
Start a point-in-time copy of a block device to a new destination. The
status of ongoing drive-backup operations can be checked with
query-block-jobs where the BlockJobInfo.type field has the value 'backup'.
The operation can be stopped before it has completed using the
block-job-cancel command.
@device: the name of the device which should be copied.
@target: the target of the new image. If the file exists, or if it
is a device, the existing file/device will be used as the new
destination. If it does not exist, a new file will be created.
@format: #optional the format of the new destination, default is to
probe if @mode is 'existing', else the format of the source
@mode: #optional whether and how QEMU should create a new image, default is
'absolute-paths'.
@speed: #optional the maximum speed, in bytes per second
@on-source-error: #optional the action to take on an error on the source,
default 'report'. 'stop' and 'enospc' can only be used
if the block device supports io-status (see BlockInfo).
@on-target-error: #optional the action to take on an error on the target,
default 'report' (no limitations, since this applies to
a different block device than @device).
Note that @on-source-error and @on-target-error only affect background I/O.
If an error occurs during a guest write request, the device's rerror/werror
actions will be used.
Returns: nothing on success
If @device is not a valid block device, DeviceNotFound
Since 1.6
Reviewed-by: Eric Blake <eblake@redhat.com>
Reviewed-by: Kevin Wolf <kwolf@redhat.com>
Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
2013-06-24 17:13:14 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-07-26 11:39:04 -07:00
|
|
|
backup_start(bs, target_bs, speed, sync, on_source_error, on_target_error,
|
block: add drive-backup QMP command
@drive-backup
Start a point-in-time copy of a block device to a new destination. The
status of ongoing drive-backup operations can be checked with
query-block-jobs where the BlockJobInfo.type field has the value 'backup'.
The operation can be stopped before it has completed using the
block-job-cancel command.
@device: the name of the device which should be copied.
@target: the target of the new image. If the file exists, or if it
is a device, the existing file/device will be used as the new
destination. If it does not exist, a new file will be created.
@format: #optional the format of the new destination, default is to
probe if @mode is 'existing', else the format of the source
@mode: #optional whether and how QEMU should create a new image, default is
'absolute-paths'.
@speed: #optional the maximum speed, in bytes per second
@on-source-error: #optional the action to take on an error on the source,
default 'report'. 'stop' and 'enospc' can only be used
if the block device supports io-status (see BlockInfo).
@on-target-error: #optional the action to take on an error on the target,
default 'report' (no limitations, since this applies to
a different block device than @device).
Note that @on-source-error and @on-target-error only affect background I/O.
If an error occurs during a guest write request, the device's rerror/werror
actions will be used.
Returns: nothing on success
If @device is not a valid block device, DeviceNotFound
Since 1.6
Reviewed-by: Eric Blake <eblake@redhat.com>
Reviewed-by: Kevin Wolf <kwolf@redhat.com>
Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
2013-06-24 17:13:14 +02:00
|
|
|
block_job_cb, bs, &local_err);
|
|
|
|
if (local_err != NULL) {
|
2013-08-23 09:14:47 +08:00
|
|
|
bdrv_unref(target_bs);
|
block: add drive-backup QMP command
@drive-backup
Start a point-in-time copy of a block device to a new destination. The
status of ongoing drive-backup operations can be checked with
query-block-jobs where the BlockJobInfo.type field has the value 'backup'.
The operation can be stopped before it has completed using the
block-job-cancel command.
@device: the name of the device which should be copied.
@target: the target of the new image. If the file exists, or if it
is a device, the existing file/device will be used as the new
destination. If it does not exist, a new file will be created.
@format: #optional the format of the new destination, default is to
probe if @mode is 'existing', else the format of the source
@mode: #optional whether and how QEMU should create a new image, default is
'absolute-paths'.
@speed: #optional the maximum speed, in bytes per second
@on-source-error: #optional the action to take on an error on the source,
default 'report'. 'stop' and 'enospc' can only be used
if the block device supports io-status (see BlockInfo).
@on-target-error: #optional the action to take on an error on the target,
default 'report' (no limitations, since this applies to
a different block device than @device).
Note that @on-source-error and @on-target-error only affect background I/O.
If an error occurs during a guest write request, the device's rerror/werror
actions will be used.
Returns: nothing on success
If @device is not a valid block device, DeviceNotFound
Since 1.6
Reviewed-by: Eric Blake <eblake@redhat.com>
Reviewed-by: Kevin Wolf <kwolf@redhat.com>
Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
2013-06-24 17:13:14 +02:00
|
|
|
error_propagate(errp, local_err);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-22 09:03:13 +01:00
|
|
|
#define DEFAULT_MIRROR_BUF_SIZE (10 << 20)
|
|
|
|
|
2012-10-18 16:49:24 +02:00
|
|
|
void qmp_drive_mirror(const char *device, const char *target,
|
|
|
|
bool has_format, const char *format,
|
|
|
|
enum MirrorSyncMode sync,
|
|
|
|
bool has_mode, enum NewImageMode mode,
|
2012-10-18 16:49:28 +02:00
|
|
|
bool has_speed, int64_t speed,
|
2013-01-21 17:09:46 +01:00
|
|
|
bool has_granularity, uint32_t granularity,
|
2013-01-22 09:03:13 +01:00
|
|
|
bool has_buf_size, int64_t buf_size,
|
2012-10-18 16:49:28 +02:00
|
|
|
bool has_on_source_error, BlockdevOnError on_source_error,
|
|
|
|
bool has_on_target_error, BlockdevOnError on_target_error,
|
|
|
|
Error **errp)
|
2012-10-18 16:49:24 +02:00
|
|
|
{
|
|
|
|
BlockDriverState *bs;
|
|
|
|
BlockDriverState *source, *target_bs;
|
|
|
|
BlockDriver *drv = NULL;
|
|
|
|
Error *local_err = NULL;
|
|
|
|
int flags;
|
2013-06-24 17:13:13 +02:00
|
|
|
int64_t size;
|
2012-10-18 16:49:24 +02:00
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (!has_speed) {
|
|
|
|
speed = 0;
|
|
|
|
}
|
2012-10-18 16:49:28 +02:00
|
|
|
if (!has_on_source_error) {
|
|
|
|
on_source_error = BLOCKDEV_ON_ERROR_REPORT;
|
|
|
|
}
|
|
|
|
if (!has_on_target_error) {
|
|
|
|
on_target_error = BLOCKDEV_ON_ERROR_REPORT;
|
|
|
|
}
|
2012-10-18 16:49:24 +02:00
|
|
|
if (!has_mode) {
|
|
|
|
mode = NEW_IMAGE_MODE_ABSOLUTE_PATHS;
|
|
|
|
}
|
2013-01-21 17:09:46 +01:00
|
|
|
if (!has_granularity) {
|
|
|
|
granularity = 0;
|
|
|
|
}
|
2013-01-22 09:03:13 +01:00
|
|
|
if (!has_buf_size) {
|
|
|
|
buf_size = DEFAULT_MIRROR_BUF_SIZE;
|
|
|
|
}
|
|
|
|
|
2013-01-21 17:09:46 +01:00
|
|
|
if (granularity != 0 && (granularity < 512 || granularity > 1048576 * 64)) {
|
|
|
|
error_set(errp, QERR_INVALID_PARAMETER, device);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (granularity & (granularity - 1)) {
|
|
|
|
error_set(errp, QERR_INVALID_PARAMETER, device);
|
|
|
|
return;
|
|
|
|
}
|
2012-10-18 16:49:24 +02:00
|
|
|
|
|
|
|
bs = bdrv_find(device);
|
|
|
|
if (!bs) {
|
|
|
|
error_set(errp, QERR_DEVICE_NOT_FOUND, device);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!bdrv_is_inserted(bs)) {
|
|
|
|
error_set(errp, QERR_DEVICE_HAS_NO_MEDIUM, device);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!has_format) {
|
|
|
|
format = mode == NEW_IMAGE_MODE_EXISTING ? NULL : bs->drv->format_name;
|
|
|
|
}
|
|
|
|
if (format) {
|
|
|
|
drv = bdrv_find_format(format);
|
|
|
|
if (!drv) {
|
|
|
|
error_set(errp, QERR_INVALID_BLOCK_FORMAT, format);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bdrv_in_use(bs)) {
|
|
|
|
error_set(errp, QERR_DEVICE_IN_USE, device);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
flags = bs->open_flags | BDRV_O_RDWR;
|
|
|
|
source = bs->backing_hd;
|
|
|
|
if (!source && sync == MIRROR_SYNC_MODE_TOP) {
|
|
|
|
sync = MIRROR_SYNC_MODE_FULL;
|
|
|
|
}
|
block/drive-mirror: Reuse backing HD for sync=none
For "none" sync mode in "absolute-paths" mode, the current image should
be used as the backing file for the newly created image.
The current behavior is:
a) If the image to be mirrored has a backing file, use that (which is
wrong, since the operations recorded by "none" are applied to the
image itself, not to its backing file).
b) If the image to be mirrored lacks a backing file, the target doesn't
have one either (which is not really wrong, but not really right,
either; "none" records a set of operations executed on the image
file, therefore having no backing file to apply these operations on
seems rather pointless).
For a, this is clearly a bugfix. For b, it is still a bugfix, although
it might break existing API - but since that case crashed qemu just
three weeks ago (before 1452686495922b81d6cf43edf025c1aef15965c0), we
can safely assume there is no such API relying on that case yet.
Suggested-by: Paolo Bonzini <pbonzini@redhat.com>
Signed-off-by: Max Reitz <mreitz@redhat.com>
Reviewed-by: Paolo Bonzini <pbonzini@redhat.com>
Reviewed-by: Kevin Wolf <kwolf@redhat.com>
Reviewed-by: Eric Blake <eblake@redhat.com>
Message-id: 1385407736-13941-2-git-send-email-mreitz@redhat.com
Signed-off-by: Anthony Liguori <aliguori@amazon.com>
2013-11-25 20:28:55 +01:00
|
|
|
if (sync == MIRROR_SYNC_MODE_NONE) {
|
|
|
|
source = bs;
|
|
|
|
}
|
2012-10-18 16:49:24 +02:00
|
|
|
|
2013-06-24 17:13:13 +02:00
|
|
|
size = bdrv_getlength(bs);
|
|
|
|
if (size < 0) {
|
|
|
|
error_setg_errno(errp, -size, "bdrv_getlength failed");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-11-06 19:50:44 +01:00
|
|
|
if ((sync == MIRROR_SYNC_MODE_FULL || !source)
|
|
|
|
&& mode != NEW_IMAGE_MODE_EXISTING)
|
|
|
|
{
|
2012-10-18 16:49:24 +02:00
|
|
|
/* create new image w/o backing file */
|
|
|
|
assert(format && drv);
|
2012-11-30 10:52:08 -02:00
|
|
|
bdrv_img_create(target, format,
|
2013-02-13 09:09:40 +01:00
|
|
|
NULL, NULL, NULL, size, flags, &local_err, false);
|
2012-10-18 16:49:24 +02:00
|
|
|
} else {
|
|
|
|
switch (mode) {
|
|
|
|
case NEW_IMAGE_MODE_EXISTING:
|
|
|
|
break;
|
|
|
|
case NEW_IMAGE_MODE_ABSOLUTE_PATHS:
|
|
|
|
/* create new image with backing file */
|
2012-11-30 10:52:08 -02:00
|
|
|
bdrv_img_create(target, format,
|
|
|
|
source->filename,
|
|
|
|
source->drv->format_name,
|
2013-02-13 09:09:40 +01:00
|
|
|
NULL, size, flags, &local_err, false);
|
2012-10-18 16:49:24 +02:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-11-30 10:52:08 -02:00
|
|
|
if (error_is_set(&local_err)) {
|
|
|
|
error_propagate(errp, local_err);
|
2012-10-18 16:49:24 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-01-21 17:09:43 +01:00
|
|
|
/* Mirroring takes care of copy-on-write using the source's backing
|
|
|
|
* file.
|
|
|
|
*/
|
2012-10-18 16:49:24 +02:00
|
|
|
target_bs = bdrv_new("");
|
2013-09-05 14:45:29 +02:00
|
|
|
ret = bdrv_open(target_bs, target, NULL, flags | BDRV_O_NO_BACKING, drv,
|
|
|
|
&local_err);
|
2012-10-18 16:49:24 +02:00
|
|
|
if (ret < 0) {
|
2013-08-23 09:14:47 +08:00
|
|
|
bdrv_unref(target_bs);
|
2013-09-05 14:45:29 +02:00
|
|
|
error_propagate(errp, local_err);
|
2012-10-18 16:49:24 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-01-22 09:03:13 +01:00
|
|
|
mirror_start(bs, target_bs, speed, granularity, buf_size, sync,
|
2013-01-21 17:09:46 +01:00
|
|
|
on_source_error, on_target_error,
|
2012-10-18 16:49:28 +02:00
|
|
|
block_job_cb, bs, &local_err);
|
2012-10-18 16:49:24 +02:00
|
|
|
if (local_err != NULL) {
|
2013-08-23 09:14:47 +08:00
|
|
|
bdrv_unref(target_bs);
|
2012-10-18 16:49:24 +02:00
|
|
|
error_propagate(errp, local_err);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-18 14:40:47 +00:00
|
|
|
static BlockJob *find_block_job(const char *device)
|
|
|
|
{
|
|
|
|
BlockDriverState *bs;
|
|
|
|
|
|
|
|
bs = bdrv_find(device);
|
|
|
|
if (!bs || !bs->job) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
return bs->job;
|
|
|
|
}
|
|
|
|
|
2012-04-25 16:51:02 +01:00
|
|
|
void qmp_block_job_set_speed(const char *device, int64_t speed, Error **errp)
|
2012-01-18 14:40:47 +00:00
|
|
|
{
|
|
|
|
BlockJob *job = find_block_job(device);
|
|
|
|
|
|
|
|
if (!job) {
|
2012-09-28 17:22:44 +02:00
|
|
|
error_set(errp, QERR_BLOCK_JOB_NOT_ACTIVE, device);
|
2012-01-18 14:40:47 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-04-25 16:51:02 +01:00
|
|
|
block_job_set_speed(job, speed, errp);
|
2012-01-18 14:40:47 +00:00
|
|
|
}
|
2012-01-18 14:40:48 +00:00
|
|
|
|
2012-09-28 17:22:51 +02:00
|
|
|
void qmp_block_job_cancel(const char *device,
|
|
|
|
bool has_force, bool force, Error **errp)
|
2012-01-18 14:40:48 +00:00
|
|
|
{
|
|
|
|
BlockJob *job = find_block_job(device);
|
|
|
|
|
2012-09-28 17:22:51 +02:00
|
|
|
if (!has_force) {
|
|
|
|
force = false;
|
|
|
|
}
|
|
|
|
|
2012-01-18 14:40:48 +00:00
|
|
|
if (!job) {
|
2012-09-28 17:22:44 +02:00
|
|
|
error_set(errp, QERR_BLOCK_JOB_NOT_ACTIVE, device);
|
2012-01-18 14:40:48 +00:00
|
|
|
return;
|
|
|
|
}
|
2012-09-28 17:22:51 +02:00
|
|
|
if (job->paused && !force) {
|
2012-09-28 17:22:50 +02:00
|
|
|
error_set(errp, QERR_BLOCK_JOB_PAUSED, device);
|
|
|
|
return;
|
|
|
|
}
|
2012-01-18 14:40:48 +00:00
|
|
|
|
|
|
|
trace_qmp_block_job_cancel(job);
|
|
|
|
block_job_cancel(job);
|
|
|
|
}
|
2012-01-18 14:40:49 +00:00
|
|
|
|
2012-09-28 17:22:51 +02:00
|
|
|
void qmp_block_job_pause(const char *device, Error **errp)
|
|
|
|
{
|
|
|
|
BlockJob *job = find_block_job(device);
|
|
|
|
|
|
|
|
if (!job) {
|
|
|
|
error_set(errp, QERR_BLOCK_JOB_NOT_ACTIVE, device);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
trace_qmp_block_job_pause(job);
|
|
|
|
block_job_pause(job);
|
|
|
|
}
|
|
|
|
|
|
|
|
void qmp_block_job_resume(const char *device, Error **errp)
|
|
|
|
{
|
|
|
|
BlockJob *job = find_block_job(device);
|
|
|
|
|
|
|
|
if (!job) {
|
|
|
|
error_set(errp, QERR_BLOCK_JOB_NOT_ACTIVE, device);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
trace_qmp_block_job_resume(job);
|
|
|
|
block_job_resume(job);
|
|
|
|
}
|
|
|
|
|
2012-10-18 16:49:21 +02:00
|
|
|
void qmp_block_job_complete(const char *device, Error **errp)
|
|
|
|
{
|
|
|
|
BlockJob *job = find_block_job(device);
|
|
|
|
|
|
|
|
if (!job) {
|
|
|
|
error_set(errp, QERR_BLOCK_JOB_NOT_ACTIVE, device);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
trace_qmp_block_job_complete(job);
|
|
|
|
block_job_complete(job, errp);
|
|
|
|
}
|
|
|
|
|
2013-09-23 15:26:03 +02:00
|
|
|
void qmp_blockdev_add(BlockdevOptions *options, Error **errp)
|
|
|
|
{
|
|
|
|
QmpOutputVisitor *ov = qmp_output_visitor_new();
|
|
|
|
QObject *obj;
|
|
|
|
QDict *qdict;
|
|
|
|
Error *local_err = NULL;
|
|
|
|
|
|
|
|
/* Require an ID in the top level */
|
|
|
|
if (!options->has_id) {
|
|
|
|
error_setg(errp, "Block device needs an ID");
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* TODO Sort it out in raw-posix and drive_init: Reject aio=native with
|
|
|
|
* cache.direct=false instead of silently switching to aio=threads, except
|
|
|
|
* if called from drive_init.
|
|
|
|
*
|
|
|
|
* For now, simply forbidding the combination for all drivers will do. */
|
|
|
|
if (options->has_aio && options->aio == BLOCKDEV_AIO_OPTIONS_NATIVE) {
|
|
|
|
bool direct = options->cache->has_direct && options->cache->direct;
|
|
|
|
if (!options->has_cache && !direct) {
|
|
|
|
error_setg(errp, "aio=native requires cache.direct=true");
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
visit_type_BlockdevOptions(qmp_output_get_visitor(ov),
|
|
|
|
&options, NULL, &local_err);
|
|
|
|
if (error_is_set(&local_err)) {
|
|
|
|
error_propagate(errp, local_err);
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
obj = qmp_output_get_qobject(ov);
|
|
|
|
qdict = qobject_to_qdict(obj);
|
|
|
|
|
|
|
|
qdict_flatten(qdict);
|
|
|
|
|
2013-09-20 11:33:11 +02:00
|
|
|
blockdev_init(qdict, IF_NONE, &local_err);
|
|
|
|
if (error_is_set(&local_err)) {
|
|
|
|
error_propagate(errp, local_err);
|
2013-09-23 15:26:03 +02:00
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
fail:
|
|
|
|
qmp_output_visitor_cleanup(ov);
|
|
|
|
}
|
|
|
|
|
2012-01-18 14:40:49 +00:00
|
|
|
static void do_qmp_query_block_jobs_one(void *opaque, BlockDriverState *bs)
|
|
|
|
{
|
|
|
|
BlockJobInfoList **prev = opaque;
|
|
|
|
BlockJob *job = bs->job;
|
|
|
|
|
|
|
|
if (job) {
|
2012-09-28 17:22:48 +02:00
|
|
|
BlockJobInfoList *elem = g_new0(BlockJobInfoList, 1);
|
|
|
|
elem->value = block_job_query(bs->job);
|
2012-01-18 14:40:49 +00:00
|
|
|
(*prev)->next = elem;
|
|
|
|
*prev = elem;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
BlockJobInfoList *qmp_query_block_jobs(Error **errp)
|
|
|
|
{
|
|
|
|
/* Dummy is a fake list element for holding the head pointer */
|
|
|
|
BlockJobInfoList dummy = {};
|
|
|
|
BlockJobInfoList *prev = &dummy;
|
|
|
|
bdrv_iterate(do_qmp_query_block_jobs_one, &prev);
|
|
|
|
return dummy.next;
|
|
|
|
}
|
2012-11-26 16:03:42 +01:00
|
|
|
|
2013-03-15 10:35:07 +01:00
|
|
|
QemuOptsList qemu_common_drive_opts = {
|
2012-11-26 16:03:42 +01:00
|
|
|
.name = "drive",
|
2013-03-15 10:35:07 +01:00
|
|
|
.head = QTAILQ_HEAD_INITIALIZER(qemu_common_drive_opts.head),
|
2012-11-26 16:03:42 +01:00
|
|
|
.desc = {
|
|
|
|
{
|
|
|
|
.name = "snapshot",
|
|
|
|
.type = QEMU_OPT_BOOL,
|
|
|
|
.help = "enable/disable snapshot mode",
|
|
|
|
},{
|
|
|
|
.name = "file",
|
|
|
|
.type = QEMU_OPT_STRING,
|
|
|
|
.help = "disk image",
|
2013-02-08 14:06:12 +01:00
|
|
|
},{
|
|
|
|
.name = "discard",
|
|
|
|
.type = QEMU_OPT_STRING,
|
|
|
|
.help = "discard operation (ignore/off, unmap/on)",
|
2012-11-26 16:03:42 +01:00
|
|
|
},{
|
2013-07-18 16:31:25 +02:00
|
|
|
.name = "cache.writeback",
|
|
|
|
.type = QEMU_OPT_BOOL,
|
|
|
|
.help = "enables writeback mode for any caches",
|
|
|
|
},{
|
|
|
|
.name = "cache.direct",
|
|
|
|
.type = QEMU_OPT_BOOL,
|
|
|
|
.help = "enables use of O_DIRECT (bypass the host page cache)",
|
|
|
|
},{
|
|
|
|
.name = "cache.no-flush",
|
|
|
|
.type = QEMU_OPT_BOOL,
|
|
|
|
.help = "ignore any flush requests for the device",
|
2012-11-26 16:03:42 +01:00
|
|
|
},{
|
|
|
|
.name = "aio",
|
|
|
|
.type = QEMU_OPT_STRING,
|
|
|
|
.help = "host AIO implementation (threads, native)",
|
|
|
|
},{
|
|
|
|
.name = "format",
|
|
|
|
.type = QEMU_OPT_STRING,
|
|
|
|
.help = "disk format (raw, qcow2, ...)",
|
|
|
|
},{
|
|
|
|
.name = "serial",
|
|
|
|
.type = QEMU_OPT_STRING,
|
|
|
|
.help = "disk serial number",
|
|
|
|
},{
|
|
|
|
.name = "rerror",
|
|
|
|
.type = QEMU_OPT_STRING,
|
|
|
|
.help = "read error action",
|
|
|
|
},{
|
|
|
|
.name = "werror",
|
|
|
|
.type = QEMU_OPT_STRING,
|
|
|
|
.help = "write error action",
|
|
|
|
},{
|
2013-07-19 20:07:29 +02:00
|
|
|
.name = "read-only",
|
2012-11-26 16:03:42 +01:00
|
|
|
.type = QEMU_OPT_BOOL,
|
|
|
|
.help = "open drive file as read-only",
|
|
|
|
},{
|
2013-07-17 14:41:54 +02:00
|
|
|
.name = "throttling.iops-total",
|
2012-11-26 16:03:42 +01:00
|
|
|
.type = QEMU_OPT_NUMBER,
|
|
|
|
.help = "limit total I/O operations per second",
|
|
|
|
},{
|
2013-07-17 14:41:54 +02:00
|
|
|
.name = "throttling.iops-read",
|
2012-11-26 16:03:42 +01:00
|
|
|
.type = QEMU_OPT_NUMBER,
|
|
|
|
.help = "limit read operations per second",
|
|
|
|
},{
|
2013-07-17 14:41:54 +02:00
|
|
|
.name = "throttling.iops-write",
|
2012-11-26 16:03:42 +01:00
|
|
|
.type = QEMU_OPT_NUMBER,
|
|
|
|
.help = "limit write operations per second",
|
|
|
|
},{
|
2013-07-17 14:41:54 +02:00
|
|
|
.name = "throttling.bps-total",
|
2012-11-26 16:03:42 +01:00
|
|
|
.type = QEMU_OPT_NUMBER,
|
|
|
|
.help = "limit total bytes per second",
|
|
|
|
},{
|
2013-07-17 14:41:54 +02:00
|
|
|
.name = "throttling.bps-read",
|
2012-11-26 16:03:42 +01:00
|
|
|
.type = QEMU_OPT_NUMBER,
|
|
|
|
.help = "limit read bytes per second",
|
|
|
|
},{
|
2013-07-17 14:41:54 +02:00
|
|
|
.name = "throttling.bps-write",
|
2012-11-26 16:03:42 +01:00
|
|
|
.type = QEMU_OPT_NUMBER,
|
|
|
|
.help = "limit write bytes per second",
|
2013-09-02 14:14:40 +02:00
|
|
|
},{
|
|
|
|
.name = "throttling.iops-total-max",
|
|
|
|
.type = QEMU_OPT_NUMBER,
|
|
|
|
.help = "I/O operations burst",
|
|
|
|
},{
|
|
|
|
.name = "throttling.iops-read-max",
|
|
|
|
.type = QEMU_OPT_NUMBER,
|
|
|
|
.help = "I/O operations read burst",
|
|
|
|
},{
|
|
|
|
.name = "throttling.iops-write-max",
|
|
|
|
.type = QEMU_OPT_NUMBER,
|
|
|
|
.help = "I/O operations write burst",
|
|
|
|
},{
|
|
|
|
.name = "throttling.bps-total-max",
|
|
|
|
.type = QEMU_OPT_NUMBER,
|
|
|
|
.help = "total bytes burst",
|
|
|
|
},{
|
|
|
|
.name = "throttling.bps-read-max",
|
|
|
|
.type = QEMU_OPT_NUMBER,
|
|
|
|
.help = "total bytes read burst",
|
|
|
|
},{
|
|
|
|
.name = "throttling.bps-write-max",
|
|
|
|
.type = QEMU_OPT_NUMBER,
|
|
|
|
.help = "total bytes write burst",
|
2013-09-02 14:14:41 +02:00
|
|
|
},{
|
|
|
|
.name = "throttling.iops-size",
|
|
|
|
.type = QEMU_OPT_NUMBER,
|
|
|
|
.help = "when limiting by iops max size of an I/O in bytes",
|
2012-11-26 16:03:42 +01:00
|
|
|
},{
|
|
|
|
.name = "copy-on-read",
|
|
|
|
.type = QEMU_OPT_BOOL,
|
|
|
|
.help = "copy read data from backing file into image file",
|
|
|
|
},
|
|
|
|
{ /* end of list */ }
|
|
|
|
},
|
|
|
|
};
|
2013-03-15 10:35:07 +01:00
|
|
|
|
|
|
|
QemuOptsList qemu_drive_opts = {
|
|
|
|
.name = "drive",
|
|
|
|
.head = QTAILQ_HEAD_INITIALIZER(qemu_drive_opts.head),
|
|
|
|
.desc = {
|
2013-06-19 13:44:17 +02:00
|
|
|
/*
|
|
|
|
* no elements => accept any params
|
|
|
|
* validation will happen later
|
|
|
|
*/
|
2013-03-15 10:35:07 +01:00
|
|
|
{ /* end of list */ }
|
|
|
|
},
|
|
|
|
};
|