2007-11-11 10:51:17 +08:00
|
|
|
#ifndef BLOCK_H
|
|
|
|
#define BLOCK_H
|
|
|
|
|
2012-12-18 01:19:44 +08:00
|
|
|
#include "block/aio.h"
|
2016-03-09 17:52:44 +08:00
|
|
|
#include "qemu/iov.h"
|
2012-12-18 01:20:00 +08:00
|
|
|
#include "qemu/option.h"
|
2015-09-01 21:48:02 +08:00
|
|
|
#include "qemu/coroutine.h"
|
2014-09-05 21:46:18 +08:00
|
|
|
#include "block/accounting.h"
|
2016-03-08 12:44:55 +08:00
|
|
|
#include "block/dirty-bitmap.h"
|
2012-12-18 01:19:43 +08:00
|
|
|
#include "qapi/qmp/qobject.h"
|
2012-09-28 23:22:47 +08:00
|
|
|
#include "qapi-types.h"
|
2016-03-08 12:44:53 +08:00
|
|
|
#include "qemu/hbitmap.h"
|
2008-09-23 03:17:18 +08:00
|
|
|
|
2007-11-11 10:51:17 +08:00
|
|
|
/* block.c */
|
|
|
|
typedef struct BlockDriver BlockDriver;
|
2012-09-28 23:22:47 +08:00
|
|
|
typedef struct BlockJob BlockJob;
|
2015-06-15 19:24:19 +08:00
|
|
|
typedef struct BdrvChild BdrvChild;
|
2015-04-08 19:43:47 +08:00
|
|
|
typedef struct BdrvChildRole BdrvChildRole;
|
2015-11-06 07:13:15 +08:00
|
|
|
typedef struct BlockJobTxn BlockJobTxn;
|
2007-11-11 10:51:17 +08:00
|
|
|
|
|
|
|
typedef struct BlockDriverInfo {
|
|
|
|
/* in bytes, 0 if irrelevant */
|
|
|
|
int cluster_size;
|
|
|
|
/* offset at which the VM state can be saved (0 if not possible) */
|
|
|
|
int64_t vm_state_offset;
|
2012-03-15 20:13:33 +08:00
|
|
|
bool is_dirty;
|
2013-10-24 18:06:53 +08:00
|
|
|
/*
|
|
|
|
* True if unallocated blocks read back as zeroes. This is equivalent
|
2015-08-26 19:17:13 +08:00
|
|
|
* to the LBPRZ flag in the SCSI logical block provisioning page.
|
2013-10-24 18:06:53 +08:00
|
|
|
*/
|
|
|
|
bool unallocated_blocks_are_zero;
|
|
|
|
/*
|
|
|
|
* True if the driver can optimize writing zeroes by unmapping
|
|
|
|
* sectors. This is equivalent to the BLKDISCARDZEROES ioctl in Linux
|
|
|
|
* with the difference that in qemu a discard is allowed to silently
|
2016-06-02 05:10:04 +08:00
|
|
|
* fail. Therefore we have to use bdrv_pwrite_zeroes with the
|
2013-10-24 18:06:53 +08:00
|
|
|
* BDRV_REQ_MAY_UNMAP flag for an optimized zero write with unmapping.
|
|
|
|
* After this call the driver has to guarantee that the contents read
|
|
|
|
* back as zero. It is additionally required that the block device is
|
|
|
|
* opened with BDRV_O_UNMAP flag for this to work.
|
|
|
|
*/
|
|
|
|
bool can_write_zeroes_with_unmap;
|
2014-05-06 21:08:43 +08:00
|
|
|
/*
|
|
|
|
* True if this block driver only supports compressed writes
|
|
|
|
*/
|
|
|
|
bool needs_compressed_writes;
|
2007-11-11 10:51:17 +08:00
|
|
|
} BlockDriverInfo;
|
|
|
|
|
2012-03-15 20:13:31 +08:00
|
|
|
typedef struct BlockFragInfo {
|
|
|
|
uint64_t allocated_clusters;
|
|
|
|
uint64_t total_clusters;
|
|
|
|
uint64_t fragmented_clusters;
|
2013-02-08 00:15:04 +08:00
|
|
|
uint64_t compressed_clusters;
|
2012-03-15 20:13:31 +08:00
|
|
|
} BlockFragInfo;
|
|
|
|
|
2013-10-24 18:06:50 +08:00
|
|
|
typedef enum {
|
2015-09-08 11:28:32 +08:00
|
|
|
BDRV_REQ_COPY_ON_READ = 0x1,
|
|
|
|
BDRV_REQ_ZERO_WRITE = 0x2,
|
2013-10-24 18:06:52 +08:00
|
|
|
/* The BDRV_REQ_MAY_UNMAP flag is used to indicate that the block driver
|
|
|
|
* is allowed to optimize a write zeroes request by unmapping (discarding)
|
|
|
|
* blocks if it is guaranteed that the result will read back as
|
|
|
|
* zeroes. The flag is only passed to the driver if the block device is
|
|
|
|
* opened with BDRV_O_UNMAP.
|
|
|
|
*/
|
2015-09-08 11:28:32 +08:00
|
|
|
BDRV_REQ_MAY_UNMAP = 0x4,
|
2015-12-01 17:36:28 +08:00
|
|
|
BDRV_REQ_NO_SERIALISING = 0x8,
|
2016-03-04 21:28:01 +08:00
|
|
|
BDRV_REQ_FUA = 0x10,
|
2016-06-14 02:56:35 +08:00
|
|
|
|
|
|
|
/* Mask of valid flags */
|
|
|
|
BDRV_REQ_MASK = 0x1f,
|
2013-10-24 18:06:50 +08:00
|
|
|
} BdrvRequestFlags;
|
|
|
|
|
2015-02-16 19:47:54 +08:00
|
|
|
typedef struct BlockSizes {
|
|
|
|
uint32_t phys;
|
|
|
|
uint32_t log;
|
|
|
|
} BlockSizes;
|
|
|
|
|
|
|
|
typedef struct HDGeometry {
|
|
|
|
uint32_t heads;
|
|
|
|
uint32_t sectors;
|
|
|
|
uint32_t cylinders;
|
|
|
|
} HDGeometry;
|
|
|
|
|
2007-11-11 10:51:17 +08:00
|
|
|
#define BDRV_O_RDWR 0x0002
|
|
|
|
#define BDRV_O_SNAPSHOT 0x0008 /* open the file read only and save writes in a snapshot */
|
2014-04-12 01:16:36 +08:00
|
|
|
#define BDRV_O_TEMPORARY 0x0010 /* delete the file after use */
|
2008-10-14 22:42:54 +08:00
|
|
|
#define BDRV_O_NOCACHE 0x0020 /* do not use the host page cache */
|
2009-08-20 22:58:35 +08:00
|
|
|
#define BDRV_O_NATIVE_AIO 0x0080 /* use native AIO instead of the thread pool */
|
2010-01-12 19:55:16 +08:00
|
|
|
#define BDRV_O_NO_BACKING 0x0100 /* don't open the backing file */
|
2010-05-26 23:51:49 +08:00
|
|
|
#define BDRV_O_NO_FLUSH 0x0200 /* disable flushing on this disk */
|
2011-11-29 00:08:47 +08:00
|
|
|
#define BDRV_O_COPY_ON_READ 0x0400 /* copy read backing sectors into image */
|
2016-01-13 22:56:06 +08:00
|
|
|
#define BDRV_O_INACTIVE 0x0800 /* consistency hint for migration handoff */
|
2012-08-09 20:05:56 +08:00
|
|
|
#define BDRV_O_CHECK 0x1000 /* open solely for consistency check */
|
block: correctly set the keep_read_only flag
I believe the bs->keep_read_only flag is supposed to reflect
the initial open state of the device. If the device is initially
opened R/O, then commit operations, or reopen operations changing
to R/W, are prohibited.
Currently, the keep_read_only flag is only accurate for the active
layer, and its backing file. Subsequent images end up always having
the keep_read_only flag set.
For instance, what happens now:
[ base ] kro = 1, ro = 1
|
v
[ snap-1 ] kro = 1, ro = 1
|
v
[ snap-2 ] kro = 0, ro = 1
|
v
[ active ] kro = 0, ro = 0
What we want:
[ base ] kro = 0, ro = 1
|
v
[ snap-1 ] kro = 0, ro = 1
|
v
[ snap-2 ] kro = 0, ro = 1
|
v
[ active ] kro = 0, ro = 0
Signed-off-by: Jeff Cody <jcody@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
2012-09-21 03:13:17 +08:00
|
|
|
#define BDRV_O_ALLOW_RDWR 0x2000 /* allow reopen to change from r/o to r/w */
|
2013-02-08 21:06:11 +08:00
|
|
|
#define BDRV_O_UNMAP 0x4000 /* execute guest UNMAP/TRIM operations */
|
2014-02-19 01:33:07 +08:00
|
|
|
#define BDRV_O_PROTOCOL 0x8000 /* if no block driver is explicitly given:
|
|
|
|
select an appropriate protocol driver,
|
|
|
|
ignoring the format layer */
|
2016-03-21 22:11:42 +08:00
|
|
|
#define BDRV_O_NO_IO 0x10000 /* don't initialize for I/O */
|
2008-10-14 22:42:54 +08:00
|
|
|
|
2016-03-19 00:46:45 +08:00
|
|
|
#define BDRV_O_CACHE_MASK (BDRV_O_NOCACHE | BDRV_O_NO_FLUSH)
|
2007-11-11 10:51:17 +08:00
|
|
|
|
2015-04-07 22:55:00 +08:00
|
|
|
|
|
|
|
/* Option names of options parsed by the block layer */
|
|
|
|
|
|
|
|
#define BDRV_OPT_CACHE_WB "cache.writeback"
|
|
|
|
#define BDRV_OPT_CACHE_DIRECT "cache.direct"
|
|
|
|
#define BDRV_OPT_CACHE_NO_FLUSH "cache.no-flush"
|
|
|
|
|
|
|
|
|
2009-12-01 01:21:19 +08:00
|
|
|
#define BDRV_SECTOR_BITS 9
|
2010-05-27 21:46:55 +08:00
|
|
|
#define BDRV_SECTOR_SIZE (1ULL << BDRV_SECTOR_BITS)
|
2010-05-01 14:23:32 +08:00
|
|
|
#define BDRV_SECTOR_MASK ~(BDRV_SECTOR_SIZE - 1)
|
2009-12-01 01:21:19 +08:00
|
|
|
|
2015-02-06 18:54:11 +08:00
|
|
|
#define BDRV_REQUEST_MAX_SECTORS MIN(SIZE_MAX >> BDRV_SECTOR_BITS, \
|
|
|
|
INT_MAX >> BDRV_SECTOR_BITS)
|
|
|
|
|
2014-11-10 17:10:38 +08:00
|
|
|
/*
|
|
|
|
* Allocation status flags
|
2016-01-26 11:58:48 +08:00
|
|
|
* BDRV_BLOCK_DATA: data is read from a file returned by bdrv_get_block_status.
|
2013-09-05 01:00:29 +08:00
|
|
|
* BDRV_BLOCK_ZERO: sectors read as zero
|
2016-01-26 11:58:48 +08:00
|
|
|
* BDRV_BLOCK_OFFSET_VALID: sector stored as raw data in a file returned by
|
|
|
|
* bdrv_get_block_status.
|
2014-05-06 21:25:36 +08:00
|
|
|
* BDRV_BLOCK_ALLOCATED: the content of the block is determined by this
|
|
|
|
* layer (as opposed to the backing file)
|
2013-10-08 20:43:14 +08:00
|
|
|
* BDRV_BLOCK_RAW: used internally to indicate that the request
|
|
|
|
* was answered by the raw driver and that one
|
|
|
|
* should look in bs->file directly.
|
2013-09-05 01:00:29 +08:00
|
|
|
*
|
|
|
|
* If BDRV_BLOCK_OFFSET_VALID is set, bits 9-62 represent the offset in
|
|
|
|
* bs->file where sector data can be read from as raw data.
|
|
|
|
*
|
|
|
|
* DATA == 0 && ZERO == 0 means that data is read from backing_hd if present.
|
|
|
|
*
|
|
|
|
* DATA ZERO OFFSET_VALID
|
|
|
|
* t t t sectors read as zero, bs->file is zero at offset
|
|
|
|
* t f t sectors read as valid from bs->file at offset
|
|
|
|
* f t t sectors preallocated, read as zero, bs->file not
|
|
|
|
* necessarily zero at offset
|
|
|
|
* f f t sectors preallocated but read from backing_hd,
|
|
|
|
* bs->file contains garbage at offset
|
|
|
|
* t t f sectors preallocated, read as zero, unknown offset
|
|
|
|
* t f f sectors read from unknown file or offset
|
|
|
|
* f t f not allocated or unknown offset, read as zero
|
|
|
|
* f f f not allocated or unknown offset, read from backing_hd
|
|
|
|
*/
|
2014-05-06 21:25:36 +08:00
|
|
|
#define BDRV_BLOCK_DATA 0x01
|
|
|
|
#define BDRV_BLOCK_ZERO 0x02
|
|
|
|
#define BDRV_BLOCK_OFFSET_VALID 0x04
|
|
|
|
#define BDRV_BLOCK_RAW 0x08
|
|
|
|
#define BDRV_BLOCK_ALLOCATED 0x10
|
2013-09-05 01:00:29 +08:00
|
|
|
#define BDRV_BLOCK_OFFSET_MASK BDRV_SECTOR_MASK
|
|
|
|
|
2012-09-21 03:13:19 +08:00
|
|
|
typedef QSIMPLEQ_HEAD(BlockReopenQueue, BlockReopenQueueEntry) BlockReopenQueue;
|
|
|
|
|
|
|
|
typedef struct BDRVReopenState {
|
|
|
|
BlockDriverState *bs;
|
|
|
|
int flags;
|
2015-04-10 23:50:50 +08:00
|
|
|
QDict *options;
|
2015-05-08 22:15:03 +08:00
|
|
|
QDict *explicit_options;
|
2012-09-21 03:13:19 +08:00
|
|
|
void *opaque;
|
|
|
|
} BDRVReopenState;
|
|
|
|
|
2014-05-23 21:29:41 +08:00
|
|
|
/*
|
|
|
|
* Block operation types
|
|
|
|
*/
|
|
|
|
typedef enum BlockOpType {
|
|
|
|
BLOCK_OP_TYPE_BACKUP_SOURCE,
|
|
|
|
BLOCK_OP_TYPE_BACKUP_TARGET,
|
|
|
|
BLOCK_OP_TYPE_CHANGE,
|
2014-09-11 13:14:00 +08:00
|
|
|
BLOCK_OP_TYPE_COMMIT_SOURCE,
|
|
|
|
BLOCK_OP_TYPE_COMMIT_TARGET,
|
2014-05-23 21:29:41 +08:00
|
|
|
BLOCK_OP_TYPE_DATAPLANE,
|
|
|
|
BLOCK_OP_TYPE_DRIVE_DEL,
|
|
|
|
BLOCK_OP_TYPE_EJECT,
|
|
|
|
BLOCK_OP_TYPE_EXTERNAL_SNAPSHOT,
|
|
|
|
BLOCK_OP_TYPE_INTERNAL_SNAPSHOT,
|
|
|
|
BLOCK_OP_TYPE_INTERNAL_SNAPSHOT_DELETE,
|
2015-12-24 12:45:02 +08:00
|
|
|
BLOCK_OP_TYPE_MIRROR_SOURCE,
|
2015-12-24 12:45:04 +08:00
|
|
|
BLOCK_OP_TYPE_MIRROR_TARGET,
|
2014-05-23 21:29:41 +08:00
|
|
|
BLOCK_OP_TYPE_RESIZE,
|
|
|
|
BLOCK_OP_TYPE_STREAM,
|
2014-06-28 00:25:25 +08:00
|
|
|
BLOCK_OP_TYPE_REPLACE,
|
2014-05-23 21:29:41 +08:00
|
|
|
BLOCK_OP_TYPE_MAX,
|
|
|
|
} BlockOpType;
|
2012-09-21 03:13:19 +08:00
|
|
|
|
2009-12-11 03:16:06 +08:00
|
|
|
void bdrv_info_print(Monitor *mon, const QObject *data);
|
|
|
|
void bdrv_info(Monitor *mon, QObject **ret_data);
|
2009-12-11 03:16:07 +08:00
|
|
|
void bdrv_stats_print(Monitor *mon, const QObject *data);
|
|
|
|
void bdrv_info_stats(Monitor *mon, QObject **ret_data);
|
2007-11-11 10:51:17 +08:00
|
|
|
|
2011-11-03 16:57:25 +08:00
|
|
|
/* disk I/O throttling */
|
2007-11-11 10:51:17 +08:00
|
|
|
void bdrv_init(void);
|
2009-10-28 01:41:44 +08:00
|
|
|
void bdrv_init_with_whitelist(void);
|
2016-03-21 22:11:48 +08:00
|
|
|
bool bdrv_uses_whitelist(void);
|
2013-07-10 21:47:39 +08:00
|
|
|
BlockDriver *bdrv_find_protocol(const char *filename,
|
2015-02-06 02:58:12 +08:00
|
|
|
bool allow_protocol_prefix,
|
|
|
|
Error **errp);
|
2007-11-11 10:51:17 +08:00
|
|
|
BlockDriver *bdrv_find_format(const char *format_name);
|
2009-05-18 22:42:10 +08:00
|
|
|
int bdrv_create(BlockDriver *drv, const char* filename,
|
2014-06-05 17:21:11 +08:00
|
|
|
QemuOpts *opts, Error **errp);
|
|
|
|
int bdrv_create_file(const char *filename, QemuOpts *opts, Error **errp);
|
2014-10-07 19:59:03 +08:00
|
|
|
BlockDriverState *bdrv_new(void);
|
2012-02-29 04:54:06 +08:00
|
|
|
void bdrv_append(BlockDriverState *bs_new, BlockDriverState *bs_top);
|
2015-09-15 17:58:23 +08:00
|
|
|
void bdrv_replace_in_backing_chain(BlockDriverState *old,
|
|
|
|
BlockDriverState *new);
|
|
|
|
|
2016-03-14 18:40:23 +08:00
|
|
|
int bdrv_parse_cache_mode(const char *mode, int *flags, bool *writethrough);
|
2013-02-08 21:06:11 +08:00
|
|
|
int bdrv_parse_discard_flags(const char *mode, int *flags);
|
2015-06-15 19:24:19 +08:00
|
|
|
BdrvChild *bdrv_open_child(const char *filename,
|
|
|
|
QDict *options, const char *bdref_key,
|
|
|
|
BlockDriverState* parent,
|
|
|
|
const BdrvChildRole *child_role,
|
|
|
|
bool allow_none, Error **errp);
|
2014-05-23 21:29:45 +08:00
|
|
|
void bdrv_set_backing_hd(BlockDriverState *bs, BlockDriverState *backing_hd);
|
2015-01-17 01:23:41 +08:00
|
|
|
int bdrv_open_backing_file(BlockDriverState *bs, QDict *parent_options,
|
|
|
|
const char *bdref_key, Error **errp);
|
2016-05-17 22:41:31 +08:00
|
|
|
BlockDriverState *bdrv_open(const char *filename, const char *reference,
|
|
|
|
QDict *options, int flags, Error **errp);
|
2012-09-21 03:13:19 +08:00
|
|
|
BlockReopenQueue *bdrv_reopen_queue(BlockReopenQueue *bs_queue,
|
2015-04-10 23:50:50 +08:00
|
|
|
BlockDriverState *bs,
|
|
|
|
QDict *options, int flags);
|
2012-09-21 03:13:19 +08:00
|
|
|
int bdrv_reopen_multiple(BlockReopenQueue *bs_queue, Error **errp);
|
|
|
|
int bdrv_reopen(BlockDriverState *bs, int bdrv_flags, Error **errp);
|
|
|
|
int bdrv_reopen_prepare(BDRVReopenState *reopen_state,
|
|
|
|
BlockReopenQueue *queue, Error **errp);
|
|
|
|
void bdrv_reopen_commit(BDRVReopenState *reopen_state);
|
|
|
|
void bdrv_reopen_abort(BDRVReopenState *reopen_state);
|
2016-05-30 22:48:35 +08:00
|
|
|
int bdrv_read(BdrvChild *child, int64_t sector_num,
|
2007-11-11 10:51:17 +08:00
|
|
|
uint8_t *buf, int nb_sectors);
|
2016-05-31 20:42:08 +08:00
|
|
|
int bdrv_write(BdrvChild *child, int64_t sector_num,
|
2007-11-11 10:51:17 +08:00
|
|
|
const uint8_t *buf, int nb_sectors);
|
2016-06-16 21:13:15 +08:00
|
|
|
int bdrv_pwrite_zeroes(BdrvChild *child, int64_t offset,
|
2016-06-02 05:10:04 +08:00
|
|
|
int count, BdrvRequestFlags flags);
|
2016-06-16 21:13:15 +08:00
|
|
|
int bdrv_make_zero(BdrvChild *child, BdrvRequestFlags flags);
|
2016-06-21 00:24:02 +08:00
|
|
|
int bdrv_pread(BdrvChild *child, int64_t offset, void *buf, int bytes);
|
|
|
|
int bdrv_preadv(BdrvChild *child, int64_t offset, QEMUIOVector *qiov);
|
2016-06-21 02:09:15 +08:00
|
|
|
int bdrv_pwrite(BdrvChild *child, int64_t offset, const void *buf, int bytes);
|
|
|
|
int bdrv_pwritev(BdrvChild *child, int64_t offset, QEMUIOVector *qiov);
|
|
|
|
int bdrv_pwrite_sync(BdrvChild *child, int64_t offset,
|
|
|
|
const void *buf, int count);
|
2016-05-24 23:21:22 +08:00
|
|
|
int coroutine_fn bdrv_co_readv(BdrvChild *child, int64_t sector_num,
|
|
|
|
int nb_sectors, QEMUIOVector *qiov);
|
2016-05-24 23:21:22 +08:00
|
|
|
int coroutine_fn bdrv_co_writev(BdrvChild *child, int64_t sector_num,
|
|
|
|
int nb_sectors, QEMUIOVector *qiov);
|
2012-02-07 21:27:25 +08:00
|
|
|
/*
|
|
|
|
* Efficiently zero a region of the disk image. Note that this is a regular
|
|
|
|
* I/O request like read or write and should have a reasonable size. This
|
|
|
|
* function is not suitable for zeroing the entire image in a single request
|
|
|
|
* because it may allocate memory for the entire region.
|
|
|
|
*/
|
2016-06-21 03:31:46 +08:00
|
|
|
int coroutine_fn bdrv_co_pwrite_zeroes(BdrvChild *child, int64_t offset,
|
|
|
|
int count, BdrvRequestFlags flags);
|
2012-01-18 22:40:51 +08:00
|
|
|
BlockDriverState *bdrv_find_backing_image(BlockDriverState *bs,
|
|
|
|
const char *backing_file);
|
2012-08-02 16:22:47 +08:00
|
|
|
int bdrv_get_backing_file_depth(BlockDriverState *bs);
|
2014-07-19 02:24:56 +08:00
|
|
|
void bdrv_refresh_filename(BlockDriverState *bs);
|
2007-11-11 10:51:17 +08:00
|
|
|
int bdrv_truncate(BlockDriverState *bs, int64_t offset);
|
2014-06-26 19:23:17 +08:00
|
|
|
int64_t bdrv_nb_sectors(BlockDriverState *bs);
|
2007-11-11 10:51:17 +08:00
|
|
|
int64_t bdrv_getlength(BlockDriverState *bs);
|
2011-07-12 19:56:39 +08:00
|
|
|
int64_t bdrv_get_allocated_file_size(BlockDriverState *bs);
|
2007-12-17 09:35:20 +08:00
|
|
|
void bdrv_get_geometry(BlockDriverState *bs, uint64_t *nb_sectors_ptr);
|
2014-07-16 23:48:16 +08:00
|
|
|
void bdrv_refresh_limits(BlockDriverState *bs, Error **errp);
|
2007-11-11 10:51:17 +08:00
|
|
|
int bdrv_commit(BlockDriverState *bs);
|
2010-01-12 19:55:17 +08:00
|
|
|
int bdrv_change_backing_file(BlockDriverState *bs,
|
|
|
|
const char *backing_file, const char *backing_fmt);
|
2009-05-10 06:03:42 +08:00
|
|
|
void bdrv_register(BlockDriver *bdrv);
|
2012-09-28 01:29:12 +08:00
|
|
|
int bdrv_drop_intermediate(BlockDriverState *active, BlockDriverState *top,
|
block: extend block-commit to accept a string for the backing file
On some image chains, QEMU may not always be able to resolve the
filenames properly, when updating the backing file of an image
after a block commit.
For instance, certain relative pathnames may fail, or drives may
have been specified originally by file descriptor (e.g. /dev/fd/???),
or a relative protocol pathname may have been used.
In these instances, QEMU may lack the information to be able to make
the correct choice, but the user or management layer most likely does
have that knowledge.
With this extension to the block-commit api, the user is able to change
the backing file of the overlay image as part of the block-commit
operation.
This allows the change to be 'safe', in the sense that if the attempt
to write the overlay image metadata fails, then the block-commit
operation returns failure, without disrupting the guest.
If the commit top is the active layer, then specifying the backing
file string will be treated as an error (there is no overlay image
to modify in that case).
If a backing file string is not specified in the command, the backing
file string to use is determined in the same manner as it was
previously.
Reviewed-by: Eric Blake <eblake@redhat.com>
Signed-off-by: Jeff Cody <jcody@redhat.com>
Reviewed-by: Kevin Wolf <kwolf@redhat.com>
Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
2014-06-26 03:40:10 +08:00
|
|
|
BlockDriverState *base,
|
|
|
|
const char *backing_file_str);
|
2012-09-28 01:29:12 +08:00
|
|
|
BlockDriverState *bdrv_find_overlay(BlockDriverState *active,
|
|
|
|
BlockDriverState *bs);
|
2012-09-28 01:29:15 +08:00
|
|
|
BlockDriverState *bdrv_find_base(BlockDriverState *bs);
|
2009-05-10 06:03:42 +08:00
|
|
|
|
2010-06-29 17:43:13 +08:00
|
|
|
|
|
|
|
typedef struct BdrvCheckResult {
|
|
|
|
int corruptions;
|
|
|
|
int leaks;
|
|
|
|
int check_errors;
|
2012-05-12 00:16:54 +08:00
|
|
|
int corruptions_fixed;
|
|
|
|
int leaks_fixed;
|
2013-01-28 19:59:46 +08:00
|
|
|
int64_t image_end_offset;
|
2012-03-15 20:13:31 +08:00
|
|
|
BlockFragInfo bfi;
|
2010-06-29 17:43:13 +08:00
|
|
|
} BdrvCheckResult;
|
|
|
|
|
2012-05-11 22:07:02 +08:00
|
|
|
typedef enum {
|
|
|
|
BDRV_FIX_LEAKS = 1,
|
|
|
|
BDRV_FIX_ERRORS = 2,
|
|
|
|
} BdrvCheckMode;
|
|
|
|
|
|
|
|
int bdrv_check(BlockDriverState *bs, BdrvCheckResult *res, BdrvCheckMode fix);
|
2010-06-29 17:43:13 +08:00
|
|
|
|
2014-10-27 18:12:50 +08:00
|
|
|
/* The units of offset and total_work_size may be chosen arbitrarily by the
|
|
|
|
* block driver; total_work_size may change during the course of the amendment
|
|
|
|
* operation */
|
|
|
|
typedef void BlockDriverAmendStatusCB(BlockDriverState *bs, int64_t offset,
|
2015-07-27 23:51:32 +08:00
|
|
|
int64_t total_work_size, void *opaque);
|
2014-10-27 18:12:50 +08:00
|
|
|
int bdrv_amend_options(BlockDriverState *bs_new, QemuOpts *opts,
|
2015-07-27 23:51:32 +08:00
|
|
|
BlockDriverAmendStatusCB *status_cb, void *cb_opaque);
|
2013-09-03 16:09:50 +08:00
|
|
|
|
2013-10-02 20:33:48 +08:00
|
|
|
/* external snapshots */
|
2014-01-24 04:31:36 +08:00
|
|
|
bool bdrv_recurse_is_first_non_filter(BlockDriverState *bs,
|
|
|
|
BlockDriverState *candidate);
|
|
|
|
bool bdrv_is_first_non_filter(BlockDriverState *candidate);
|
2013-10-02 20:33:48 +08:00
|
|
|
|
2014-06-28 00:25:25 +08:00
|
|
|
/* check if a named node can be replaced when doing drive-mirror */
|
2015-07-17 10:12:22 +08:00
|
|
|
BlockDriverState *check_to_replace_node(BlockDriverState *parent_bs,
|
|
|
|
const char *node_name, Error **errp);
|
2014-06-28 00:25:25 +08:00
|
|
|
|
2007-11-11 10:51:17 +08:00
|
|
|
/* async block I/O */
|
2016-05-30 18:06:33 +08:00
|
|
|
BlockAIOCB *bdrv_aio_readv(BdrvChild *child, int64_t sector_num,
|
2014-10-07 19:59:14 +08:00
|
|
|
QEMUIOVector *iov, int nb_sectors,
|
2014-10-07 19:59:15 +08:00
|
|
|
BlockCompletionFunc *cb, void *opaque);
|
2016-05-30 18:06:33 +08:00
|
|
|
BlockAIOCB *bdrv_aio_writev(BdrvChild *child, int64_t sector_num,
|
2014-10-07 19:59:14 +08:00
|
|
|
QEMUIOVector *iov, int nb_sectors,
|
2014-10-07 19:59:15 +08:00
|
|
|
BlockCompletionFunc *cb, void *opaque);
|
2014-10-07 19:59:14 +08:00
|
|
|
BlockAIOCB *bdrv_aio_flush(BlockDriverState *bs,
|
2014-10-07 19:59:15 +08:00
|
|
|
BlockCompletionFunc *cb, void *opaque);
|
2016-07-16 07:22:53 +08:00
|
|
|
BlockAIOCB *bdrv_aio_pdiscard(BlockDriverState *bs,
|
|
|
|
int64_t offset, int count,
|
|
|
|
BlockCompletionFunc *cb, void *opaque);
|
2014-10-07 19:59:14 +08:00
|
|
|
void bdrv_aio_cancel(BlockAIOCB *acb);
|
|
|
|
void bdrv_aio_cancel_async(BlockAIOCB *acb);
|
2007-11-11 10:51:17 +08:00
|
|
|
|
2009-03-13 03:57:08 +08:00
|
|
|
/* sg packet commands */
|
2009-03-29 01:28:41 +08:00
|
|
|
int bdrv_ioctl(BlockDriverState *bs, unsigned long int req, void *buf);
|
2014-10-07 19:59:14 +08:00
|
|
|
BlockAIOCB *bdrv_aio_ioctl(BlockDriverState *bs,
|
2009-03-29 01:28:41 +08:00
|
|
|
unsigned long int req, void *buf,
|
2014-10-07 19:59:15 +08:00
|
|
|
BlockCompletionFunc *cb, void *opaque);
|
2009-03-13 03:57:08 +08:00
|
|
|
|
2011-11-15 05:09:45 +08:00
|
|
|
/* Invalidate any cached metadata used by image formats */
|
2014-03-12 22:59:16 +08:00
|
|
|
void bdrv_invalidate_cache(BlockDriverState *bs, Error **errp);
|
|
|
|
void bdrv_invalidate_cache_all(Error **errp);
|
2015-12-22 21:07:08 +08:00
|
|
|
int bdrv_inactivate_all(void);
|
2011-11-15 05:09:45 +08:00
|
|
|
|
2007-11-11 10:51:17 +08:00
|
|
|
/* Ensure contents are flushed to disk. */
|
2010-10-21 22:43:43 +08:00
|
|
|
int bdrv_flush(BlockDriverState *bs);
|
2011-10-17 18:32:12 +08:00
|
|
|
int coroutine_fn bdrv_co_flush(BlockDriverState *bs);
|
2010-05-28 10:44:57 +08:00
|
|
|
void bdrv_close_all(void);
|
2014-10-21 19:03:55 +08:00
|
|
|
void bdrv_drain(BlockDriverState *bs);
|
2016-04-05 19:20:52 +08:00
|
|
|
void coroutine_fn bdrv_co_drain(BlockDriverState *bs);
|
2011-11-30 20:23:43 +08:00
|
|
|
void bdrv_drain_all(void);
|
2008-10-06 21:55:43 +08:00
|
|
|
|
2016-07-16 07:22:51 +08:00
|
|
|
int bdrv_pdiscard(BlockDriverState *bs, int64_t offset, int count);
|
2016-07-16 07:22:50 +08:00
|
|
|
int bdrv_co_pdiscard(BlockDriverState *bs, int64_t offset, int count);
|
2013-06-28 18:47:42 +08:00
|
|
|
int bdrv_has_zero_init_1(BlockDriverState *bs);
|
2010-04-14 23:30:35 +08:00
|
|
|
int bdrv_has_zero_init(BlockDriverState *bs);
|
2013-10-24 18:06:54 +08:00
|
|
|
bool bdrv_unallocated_blocks_are_zero(BlockDriverState *bs);
|
|
|
|
bool bdrv_can_write_zeroes_with_unmap(BlockDriverState *bs);
|
2013-09-05 01:00:28 +08:00
|
|
|
int64_t bdrv_get_block_status(BlockDriverState *bs, int64_t sector_num,
|
2016-01-26 11:58:48 +08:00
|
|
|
int nb_sectors, int *pnum,
|
|
|
|
BlockDriverState **file);
|
2015-06-08 13:56:07 +08:00
|
|
|
int64_t bdrv_get_block_status_above(BlockDriverState *bs,
|
|
|
|
BlockDriverState *base,
|
|
|
|
int64_t sector_num,
|
2016-01-26 11:58:48 +08:00
|
|
|
int nb_sectors, int *pnum,
|
|
|
|
BlockDriverState **file);
|
2008-06-06 05:53:49 +08:00
|
|
|
int bdrv_is_allocated(BlockDriverState *bs, int64_t sector_num, int nb_sectors,
|
2011-06-09 13:06:43 +08:00
|
|
|
int *pnum);
|
2013-02-13 16:09:39 +08:00
|
|
|
int bdrv_is_allocated_above(BlockDriverState *top, BlockDriverState *base,
|
|
|
|
int64_t sector_num, int nb_sectors, int *pnum);
|
2007-11-11 10:51:17 +08:00
|
|
|
|
2016-06-24 06:37:26 +08:00
|
|
|
bool bdrv_is_read_only(BlockDriverState *bs);
|
|
|
|
bool bdrv_is_sg(BlockDriverState *bs);
|
2015-10-19 23:53:11 +08:00
|
|
|
bool bdrv_is_inserted(BlockDriverState *bs);
|
2007-11-11 10:51:17 +08:00
|
|
|
int bdrv_media_changed(BlockDriverState *bs);
|
2011-09-07 00:58:47 +08:00
|
|
|
void bdrv_lock_medium(BlockDriverState *bs, bool locked);
|
2012-02-04 02:24:53 +08:00
|
|
|
void bdrv_eject(BlockDriverState *bs, bool eject_flag);
|
2012-06-13 16:11:48 +08:00
|
|
|
const char *bdrv_get_format_name(BlockDriverState *bs);
|
2014-01-24 04:31:32 +08:00
|
|
|
BlockDriverState *bdrv_find_node(const char *node_name);
|
2015-04-17 19:52:43 +08:00
|
|
|
BlockDeviceInfoList *bdrv_named_nodes_list(Error **errp);
|
2014-01-24 04:31:35 +08:00
|
|
|
BlockDriverState *bdrv_lookup_bs(const char *device,
|
|
|
|
const char *node_name,
|
|
|
|
Error **errp);
|
2014-06-26 03:40:09 +08:00
|
|
|
bool bdrv_chain_contains(BlockDriverState *top, BlockDriverState *base);
|
2014-10-31 11:32:54 +08:00
|
|
|
BlockDriverState *bdrv_next_node(BlockDriverState *bs);
|
2016-05-21 00:49:07 +08:00
|
|
|
|
|
|
|
typedef struct BdrvNextIterator {
|
|
|
|
enum {
|
|
|
|
BDRV_NEXT_BACKEND_ROOTS,
|
|
|
|
BDRV_NEXT_MONITOR_OWNED,
|
|
|
|
} phase;
|
|
|
|
BlockBackend *blk;
|
|
|
|
BlockDriverState *bs;
|
|
|
|
} BdrvNextIterator;
|
|
|
|
|
|
|
|
BlockDriverState *bdrv_first(BdrvNextIterator *it);
|
|
|
|
BlockDriverState *bdrv_next(BdrvNextIterator *it);
|
|
|
|
|
2016-03-17 02:54:41 +08:00
|
|
|
BlockDriverState *bdrv_next_monitor_owned(BlockDriverState *bs);
|
2016-06-24 06:37:26 +08:00
|
|
|
bool bdrv_is_encrypted(BlockDriverState *bs);
|
|
|
|
bool bdrv_key_required(BlockDriverState *bs);
|
2007-11-11 10:51:17 +08:00
|
|
|
int bdrv_set_key(BlockDriverState *bs, const char *key);
|
2015-01-29 17:37:00 +08:00
|
|
|
void bdrv_add_key(BlockDriverState *bs, const char *key, Error **errp);
|
2009-03-06 07:01:01 +08:00
|
|
|
int bdrv_query_missing_keys(void);
|
2007-11-11 10:51:17 +08:00
|
|
|
void bdrv_iterate_format(void (*it)(void *opaque, const char *name),
|
|
|
|
void *opaque);
|
2014-10-31 11:32:55 +08:00
|
|
|
const char *bdrv_get_node_name(const BlockDriverState *bs);
|
2014-10-07 19:59:11 +08:00
|
|
|
const char *bdrv_get_device_name(const BlockDriverState *bs);
|
2015-04-08 17:29:18 +08:00
|
|
|
const char *bdrv_get_device_or_node_name(const BlockDriverState *bs);
|
2012-06-05 22:49:24 +08:00
|
|
|
int bdrv_get_flags(BlockDriverState *bs);
|
2007-11-11 10:51:17 +08:00
|
|
|
int bdrv_write_compressed(BlockDriverState *bs, int64_t sector_num,
|
|
|
|
const uint8_t *buf, int nb_sectors);
|
|
|
|
int bdrv_get_info(BlockDriverState *bs, BlockDriverInfo *bdi);
|
2013-10-09 16:46:16 +08:00
|
|
|
ImageInfoSpecific *bdrv_get_specific_info(BlockDriverState *bs);
|
2016-06-02 17:41:52 +08:00
|
|
|
void bdrv_round_sectors_to_clusters(BlockDriverState *bs,
|
|
|
|
int64_t sector_num, int nb_sectors,
|
|
|
|
int64_t *cluster_sector_num,
|
|
|
|
int *cluster_nb_sectors);
|
2013-01-22 00:09:42 +08:00
|
|
|
void bdrv_round_to_clusters(BlockDriverState *bs,
|
2016-06-02 17:41:52 +08:00
|
|
|
int64_t offset, unsigned int bytes,
|
|
|
|
int64_t *cluster_offset,
|
|
|
|
unsigned int *cluster_bytes);
|
2007-11-11 10:51:17 +08:00
|
|
|
|
2009-03-06 07:00:48 +08:00
|
|
|
const char *bdrv_get_encrypted_filename(BlockDriverState *bs);
|
2007-11-11 10:51:17 +08:00
|
|
|
void bdrv_get_backing_filename(BlockDriverState *bs,
|
|
|
|
char *filename, int filename_size);
|
2012-05-08 22:51:50 +08:00
|
|
|
void bdrv_get_full_backing_filename(BlockDriverState *bs,
|
2014-11-27 00:20:26 +08:00
|
|
|
char *dest, size_t sz, Error **errp);
|
2014-11-27 00:20:25 +08:00
|
|
|
void bdrv_get_full_backing_filename_from_filename(const char *backed,
|
|
|
|
const char *backing,
|
2014-11-27 00:20:26 +08:00
|
|
|
char *dest, size_t sz,
|
|
|
|
Error **errp);
|
2010-07-26 04:49:34 +08:00
|
|
|
int bdrv_is_snapshot(BlockDriverState *bs);
|
2007-11-11 10:51:17 +08:00
|
|
|
|
2014-12-03 21:57:22 +08:00
|
|
|
int path_has_protocol(const char *path);
|
2007-11-11 10:51:17 +08:00
|
|
|
int path_is_absolute(const char *path);
|
|
|
|
void path_combine(char *dest, int dest_size,
|
|
|
|
const char *base_path,
|
|
|
|
const char *filename);
|
|
|
|
|
2016-06-09 22:50:16 +08:00
|
|
|
int bdrv_readv_vmstate(BlockDriverState *bs, QEMUIOVector *qiov, int64_t pos);
|
2013-04-06 03:27:53 +08:00
|
|
|
int bdrv_writev_vmstate(BlockDriverState *bs, QEMUIOVector *qiov, int64_t pos);
|
2009-07-11 05:11:57 +08:00
|
|
|
int bdrv_save_vmstate(BlockDriverState *bs, const uint8_t *buf,
|
|
|
|
int64_t pos, int size);
|
2009-04-06 03:10:55 +08:00
|
|
|
|
2009-07-11 05:11:57 +08:00
|
|
|
int bdrv_load_vmstate(BlockDriverState *bs, uint8_t *buf,
|
|
|
|
int64_t pos, int size);
|
2009-04-06 03:10:55 +08:00
|
|
|
|
2012-11-30 20:52:09 +08:00
|
|
|
void bdrv_img_create(const char *filename, const char *fmt,
|
|
|
|
const char *base_filename, const char *base_fmt,
|
2013-02-13 16:09:40 +08:00
|
|
|
char *options, uint64_t img_size, int flags,
|
|
|
|
Error **errp, bool quiet);
|
2010-12-16 20:52:15 +08:00
|
|
|
|
2013-11-28 17:23:32 +08:00
|
|
|
/* Returns the alignment in bytes that is required so that no bounce buffer
|
|
|
|
* is required throughout the stack */
|
2015-05-12 22:30:55 +08:00
|
|
|
size_t bdrv_min_mem_align(BlockDriverState *bs);
|
|
|
|
/* Returns optimal alignment in bytes for bounce buffer */
|
2013-11-28 17:23:32 +08:00
|
|
|
size_t bdrv_opt_mem_align(BlockDriverState *bs);
|
2011-08-03 21:08:19 +08:00
|
|
|
void *qemu_blockalign(BlockDriverState *bs, size_t size);
|
2014-10-22 20:09:27 +08:00
|
|
|
void *qemu_blockalign0(BlockDriverState *bs, size_t size);
|
2014-05-20 18:24:05 +08:00
|
|
|
void *qemu_try_blockalign(BlockDriverState *bs, size_t size);
|
2014-10-22 20:09:27 +08:00
|
|
|
void *qemu_try_blockalign0(BlockDriverState *bs, size_t size);
|
2013-01-11 23:41:27 +08:00
|
|
|
bool bdrv_qiov_is_aligned(BlockDriverState *bs, QEMUIOVector *qiov);
|
2011-08-03 21:08:19 +08:00
|
|
|
|
2011-11-29 00:08:47 +08:00
|
|
|
void bdrv_enable_copy_on_read(BlockDriverState *bs);
|
|
|
|
void bdrv_disable_copy_on_read(BlockDriverState *bs);
|
|
|
|
|
2013-08-23 09:14:46 +08:00
|
|
|
void bdrv_ref(BlockDriverState *bs);
|
|
|
|
void bdrv_unref(BlockDriverState *bs);
|
2015-06-15 19:51:04 +08:00
|
|
|
void bdrv_unref_child(BlockDriverState *parent, BdrvChild *child);
|
2016-05-10 15:36:38 +08:00
|
|
|
BdrvChild *bdrv_attach_child(BlockDriverState *parent_bs,
|
|
|
|
BlockDriverState *child_bs,
|
|
|
|
const char *child_name,
|
|
|
|
const BdrvChildRole *child_role);
|
2010-03-16 00:27:00 +08:00
|
|
|
|
2014-05-23 21:29:42 +08:00
|
|
|
bool bdrv_op_is_blocked(BlockDriverState *bs, BlockOpType op, Error **errp);
|
|
|
|
void bdrv_op_block(BlockDriverState *bs, BlockOpType op, Error *reason);
|
|
|
|
void bdrv_op_unblock(BlockDriverState *bs, BlockOpType op, Error *reason);
|
|
|
|
void bdrv_op_block_all(BlockDriverState *bs, Error *reason);
|
|
|
|
void bdrv_op_unblock_all(BlockDriverState *bs, Error *reason);
|
|
|
|
bool bdrv_op_blocker_is_empty(BlockDriverState *bs);
|
|
|
|
|
2015-06-16 20:19:22 +08:00
|
|
|
#define BLKDBG_EVENT(child, evt) \
|
|
|
|
do { \
|
|
|
|
if (child) { \
|
|
|
|
bdrv_debug_event(child->bs, evt); \
|
|
|
|
} \
|
|
|
|
} while (0)
|
|
|
|
|
2015-11-18 16:52:54 +08:00
|
|
|
void bdrv_debug_event(BlockDriverState *bs, BlkdebugEvent event);
|
2010-03-16 00:27:00 +08:00
|
|
|
|
2012-12-06 21:32:58 +08:00
|
|
|
int bdrv_debug_breakpoint(BlockDriverState *bs, const char *event,
|
|
|
|
const char *tag);
|
2013-11-20 10:01:54 +08:00
|
|
|
int bdrv_debug_remove_breakpoint(BlockDriverState *bs, const char *tag);
|
2012-12-06 21:32:58 +08:00
|
|
|
int bdrv_debug_resume(BlockDriverState *bs, const char *tag);
|
|
|
|
bool bdrv_debug_is_suspended(BlockDriverState *bs, const char *tag);
|
|
|
|
|
2014-05-15 19:22:05 +08:00
|
|
|
/**
|
|
|
|
* bdrv_get_aio_context:
|
|
|
|
*
|
|
|
|
* Returns: the currently bound #AioContext
|
|
|
|
*/
|
|
|
|
AioContext *bdrv_get_aio_context(BlockDriverState *bs);
|
|
|
|
|
2014-05-08 22:34:37 +08:00
|
|
|
/**
|
|
|
|
* bdrv_set_aio_context:
|
|
|
|
*
|
|
|
|
* Changes the #AioContext used for fd handlers, timers, and BHs by this
|
|
|
|
* BlockDriverState and all its children.
|
|
|
|
*
|
block: Forbid bdrv_set_aio_context outside BQL
Even if the caller has both the old and the new AioContext's, there can
be a deadlock, due to the leading bdrv_drain_all.
Suppose there are four io threads (A, B, A0, B0) with A and B owning a
BDS for each (bs_a, bs_b); Now A wants to move bs_a to iothread A0, and
B wants to move bs_b to B0, at the same time:
iothread A iothread B
--------------------------------------------------------------------------
aio_context_acquire(A0) /* OK */ aio_context_acquire(B0) /* OK */
bdrv_set_aio_context(bs_a, A0) bdrv_set_aio_context(bs_b, B0)
-> bdrv_drain_all() -> bdrv_drain_all()
-> acquire A /* OK */ -> acquire A /* blocked */
-> acquire B /* blocked */ -> acquire B
... ...
Deadlock happens because A is waiting for B, and B is waiting for A.
Signed-off-by: Fam Zheng <famz@redhat.com>
Reviewed-by: Paolo Bonzini <pbonzini@redhat.com>
Message-Id: <1423969591-23646-2-git-send-email-famz@redhat.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2015-02-15 11:06:30 +08:00
|
|
|
* This function must be called with iothread lock held.
|
2014-05-08 22:34:37 +08:00
|
|
|
*/
|
|
|
|
void bdrv_set_aio_context(BlockDriverState *bs, AioContext *new_context);
|
2015-02-16 19:47:54 +08:00
|
|
|
int bdrv_probe_blocksizes(BlockDriverState *bs, BlockSizes *bsz);
|
|
|
|
int bdrv_probe_geometry(BlockDriverState *bs, HDGeometry *geo);
|
2014-05-08 22:34:37 +08:00
|
|
|
|
2014-07-04 18:04:33 +08:00
|
|
|
void bdrv_io_plug(BlockDriverState *bs);
|
|
|
|
void bdrv_io_unplug(BlockDriverState *bs);
|
2016-04-08 00:33:34 +08:00
|
|
|
void bdrv_io_unplugged_begin(BlockDriverState *bs);
|
|
|
|
void bdrv_io_unplugged_end(BlockDriverState *bs);
|
2014-07-04 18:04:33 +08:00
|
|
|
|
2015-10-23 11:08:09 +08:00
|
|
|
/**
|
|
|
|
* bdrv_drained_begin:
|
|
|
|
*
|
|
|
|
* Begin a quiesced section for exclusive access to the BDS, by disabling
|
|
|
|
* external request sources including NBD server and device model. Note that
|
|
|
|
* this doesn't block timers or coroutines from submitting more requests, which
|
|
|
|
* means block_job_pause is still necessary.
|
|
|
|
*
|
|
|
|
* This function can be recursive.
|
|
|
|
*/
|
|
|
|
void bdrv_drained_begin(BlockDriverState *bs);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* bdrv_drained_end:
|
|
|
|
*
|
|
|
|
* End a quiescent section started by bdrv_drained_begin().
|
|
|
|
*/
|
|
|
|
void bdrv_drained_end(BlockDriverState *bs);
|
|
|
|
|
2016-05-10 15:36:37 +08:00
|
|
|
void bdrv_add_child(BlockDriverState *parent, BlockDriverState *child,
|
|
|
|
Error **errp);
|
|
|
|
void bdrv_del_child(BlockDriverState *parent, BdrvChild *child, Error **errp);
|
|
|
|
|
2012-07-10 17:12:40 +08:00
|
|
|
#endif
|