[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]
Re: [Qemu-block] [Qemu-devel] [PATCH v2 05/16] block: Convert bs->file t
From: |
Fam Zheng |
Subject: |
Re: [Qemu-block] [Qemu-devel] [PATCH v2 05/16] block: Convert bs->file to BdrvChild |
Date: |
Thu, 8 Oct 2015 18:23:01 +0800 |
User-agent: |
Mutt/1.5.24 (2015-08-30) |
On Thu, 10/01 15:13, Kevin Wolf wrote:
> @@ -1935,6 +1932,11 @@ void bdrv_close(BlockDriverState *bs)
> bdrv_unref(backing_hd);
> }
>
> + if (bs->file != NULL) {
> + bdrv_unref_child(bs, bs->file);
> + bs->file = NULL;
> + }
> +
> QLIST_FOREACH_SAFE(child, &bs->children, next, next) {
> /* TODO Remove bdrv_unref() from drivers' close function and use
> * bdrv_unref_child() here */
> @@ -1946,7 +1948,6 @@ void bdrv_close(BlockDriverState *bs)
>
> g_free(bs->opaque);
> bs->opaque = NULL;
> - bs->drv = NULL;
> bs->copy_on_read = 0;
> bs->backing_file[0] = '\0';
> bs->backing_format[0] = '\0';
> @@ -1959,11 +1960,6 @@ void bdrv_close(BlockDriverState *bs)
> bs->options = NULL;
> QDECREF(bs->full_open_options);
> bs->full_open_options = NULL;
> -
> - if (bs->file != NULL) {
> - bdrv_unref(bs->file);
> - bs->file = NULL;
> - }
Why do you need to move them up? Changing bdrv_unref to bdrv_unref_child is not
enough?
> }
>
> if (bs->blk) {
> @@ -2566,7 +2562,7 @@ int64_t bdrv_get_allocated_file_size(BlockDriverState
> *bs)
> return drv->bdrv_get_allocated_file_size(bs);
> }
> if (bs->file) {
> - return bdrv_get_allocated_file_size(bs->file);
> + return bdrv_get_allocated_file_size(bs->file->bs);
> }
> return -ENOTSUP;
> }
> @@ -3048,7 +3044,7 @@ int bdrv_debug_breakpoint(BlockDriverState *bs, const
> char *event,
> const char *tag)
> {
> while (bs && bs->drv && !bs->drv->bdrv_debug_breakpoint) {
> - bs = bs->file;
> + bs = bs->file ? bs->file->bs : NULL;
> }
>
> if (bs && bs->drv && bs->drv->bdrv_debug_breakpoint) {
> @@ -3061,7 +3057,7 @@ int bdrv_debug_breakpoint(BlockDriverState *bs, const
> char *event,
> int bdrv_debug_remove_breakpoint(BlockDriverState *bs, const char *tag)
> {
> while (bs && bs->drv && !bs->drv->bdrv_debug_remove_breakpoint) {
> - bs = bs->file;
> + bs = bs->file ? bs->file->bs : NULL;
> }
>
> if (bs && bs->drv && bs->drv->bdrv_debug_remove_breakpoint) {
> @@ -3074,7 +3070,7 @@ int bdrv_debug_remove_breakpoint(BlockDriverState *bs,
> const char *tag)
> int bdrv_debug_resume(BlockDriverState *bs, const char *tag)
> {
> while (bs && (!bs->drv || !bs->drv->bdrv_debug_resume)) {
> - bs = bs->file;
> + bs = bs->file ? bs->file->bs : NULL;
> }
>
> if (bs && bs->drv && bs->drv->bdrv_debug_resume) {
> @@ -3087,7 +3083,7 @@ int bdrv_debug_resume(BlockDriverState *bs, const char
> *tag)
> bool bdrv_debug_is_suspended(BlockDriverState *bs, const char *tag)
> {
> while (bs && bs->drv && !bs->drv->bdrv_debug_is_suspended) {
> - bs = bs->file;
> + bs = bs->file ? bs->file->bs : NULL;
> }
>
> if (bs && bs->drv && bs->drv->bdrv_debug_is_suspended) {
> @@ -3209,7 +3205,7 @@ void bdrv_invalidate_cache(BlockDriverState *bs, Error
> **errp)
> if (bs->drv->bdrv_invalidate_cache) {
> bs->drv->bdrv_invalidate_cache(bs, &local_err);
> } else if (bs->file) {
> - bdrv_invalidate_cache(bs->file, &local_err);
> + bdrv_invalidate_cache(bs->file->bs, &local_err);
> }
> if (local_err) {
> error_propagate(errp, local_err);
> @@ -3939,7 +3935,7 @@ void bdrv_detach_aio_context(BlockDriverState *bs)
> bs->drv->bdrv_detach_aio_context(bs);
> }
> if (bs->file) {
> - bdrv_detach_aio_context(bs->file);
> + bdrv_detach_aio_context(bs->file->bs);
> }
> if (bs->backing_hd) {
> bdrv_detach_aio_context(bs->backing_hd);
> @@ -3963,7 +3959,7 @@ void bdrv_attach_aio_context(BlockDriverState *bs,
> bdrv_attach_aio_context(bs->backing_hd, new_context);
> }
> if (bs->file) {
> - bdrv_attach_aio_context(bs->file, new_context);
> + bdrv_attach_aio_context(bs->file->bs, new_context);
> }
> if (bs->drv->bdrv_attach_aio_context) {
> bs->drv->bdrv_attach_aio_context(bs, new_context);
> @@ -4175,7 +4171,7 @@ void bdrv_refresh_filename(BlockDriverState *bs)
> /* This BDS's file name will most probably depend on its file's name, so
> * refresh that first */
> if (bs->file) {
> - bdrv_refresh_filename(bs->file);
> + bdrv_refresh_filename(bs->file->bs);
> }
>
> if (drv->bdrv_refresh_filename) {
> @@ -4203,19 +4199,20 @@ void bdrv_refresh_filename(BlockDriverState *bs)
>
> /* If no specific options have been given for this BDS, the filename
> of
> * the underlying file should suffice for this one as well */
> - if (bs->file->exact_filename[0] && !has_open_options) {
> - strcpy(bs->exact_filename, bs->file->exact_filename);
> + if (bs->file->bs->exact_filename[0] && !has_open_options) {
> + strcpy(bs->exact_filename, bs->file->bs->exact_filename);
> }
> /* Reconstructing the full options QDict is simple for most format
> block
> * drivers, as long as the full options are known for the underlying
> * file BDS. The full options QDict of that file BDS should somehow
> * contain a representation of the filename, therefore the following
> * suffices without querying the (exact_)filename of this BDS. */
> - if (bs->file->full_open_options) {
> + if (bs->file->bs->full_open_options) {
> qdict_put_obj(opts, "driver",
> QOBJECT(qstring_from_str(drv->format_name)));
> - QINCREF(bs->file->full_open_options);
> - qdict_put_obj(opts, "file",
> QOBJECT(bs->file->full_open_options));
> + QINCREF(bs->file->bs->full_open_options);
> + qdict_put_obj(opts, "file",
> + QOBJECT(bs->file->bs->full_open_options));
>
> bs->full_open_options = opts;
> } else {
> diff --git a/block/blkdebug.c b/block/blkdebug.c
> index bc247f4..117fce6 100644
> --- a/block/blkdebug.c
> +++ b/block/blkdebug.c
> @@ -427,10 +427,10 @@ static int blkdebug_open(BlockDriverState *bs, QDict
> *options, int flags,
> s->state = 1;
>
> /* Open the backing file */
Isn't "backing file" a confusing term for bs->file given that we have
bs->backing_hd? :)
> - assert(bs->file == NULL);
> - ret = bdrv_open_image(&bs->file, qemu_opt_get(opts, "x-image"), options,
> "image",
> - bs, &child_file, false, &local_err);
> - if (ret < 0) {
Should we keep the assertion?
> + bs->file = bdrv_open_child(qemu_opt_get(opts, "x-image"), options,
> "image",
> + bs, &child_file, false, &local_err);
> + if (local_err) {
> + ret = -EINVAL;
> error_propagate(errp, local_err);
> goto out;
> }
> @@ -449,7 +449,7 @@ static int blkdebug_open(BlockDriverState *bs, QDict
> *options, int flags,
> goto out;
>
> fail_unref:
> - bdrv_unref(bs->file);
> + bdrv_unref(bs->file->bs);
> out:
> qemu_opts_del(opts);
> return ret;
> @@ -510,7 +510,8 @@ static BlockAIOCB *blkdebug_aio_readv(BlockDriverState
> *bs,
> return inject_error(bs, cb, opaque, rule);
> }
>
> - return bdrv_aio_readv(bs->file, sector_num, qiov, nb_sectors, cb,
> opaque);
> + return bdrv_aio_readv(bs->file->bs, sector_num, qiov, nb_sectors,
> + cb, opaque);
> }
>
> static BlockAIOCB *blkdebug_aio_writev(BlockDriverState *bs,
> @@ -532,7 +533,8 @@ static BlockAIOCB *blkdebug_aio_writev(BlockDriverState
> *bs,
> return inject_error(bs, cb, opaque, rule);
> }
>
> - return bdrv_aio_writev(bs->file, sector_num, qiov, nb_sectors, cb,
> opaque);
> + return bdrv_aio_writev(bs->file->bs, sector_num, qiov, nb_sectors,
> + cb, opaque);
> }
>
> static BlockAIOCB *blkdebug_aio_flush(BlockDriverState *bs,
> @@ -551,7 +553,7 @@ static BlockAIOCB *blkdebug_aio_flush(BlockDriverState
> *bs,
> return inject_error(bs, cb, opaque, rule);
> }
>
> - return bdrv_aio_flush(bs->file, cb, opaque);
> + return bdrv_aio_flush(bs->file->bs, cb, opaque);
> }
>
>
> @@ -716,12 +718,12 @@ static bool
> blkdebug_debug_is_suspended(BlockDriverState *bs, const char *tag)
>
> static int64_t blkdebug_getlength(BlockDriverState *bs)
> {
> - return bdrv_getlength(bs->file);
> + return bdrv_getlength(bs->file->bs);
> }
>
> static int blkdebug_truncate(BlockDriverState *bs, int64_t offset)
> {
> - return bdrv_truncate(bs->file, offset);
> + return bdrv_truncate(bs->file->bs, offset);
> }
>
> static void blkdebug_refresh_filename(BlockDriverState *bs)
> @@ -741,24 +743,24 @@ static void blkdebug_refresh_filename(BlockDriverState
> *bs)
> }
> }
>
> - if (force_json && !bs->file->full_open_options) {
> + if (force_json && !bs->file->bs->full_open_options) {
> /* The config file cannot be recreated, so creating a plain filename
> * is impossible */
> return;
> }
>
> - if (!force_json && bs->file->exact_filename[0]) {
> + if (!force_json && bs->file->bs->exact_filename[0]) {
> snprintf(bs->exact_filename, sizeof(bs->exact_filename),
> "blkdebug:%s:%s",
> qdict_get_try_str(bs->options, "config") ?: "",
> - bs->file->exact_filename);
> + bs->file->bs->exact_filename);
> }
>
> opts = qdict_new();
> qdict_put_obj(opts, "driver", QOBJECT(qstring_from_str("blkdebug")));
>
> - QINCREF(bs->file->full_open_options);
> - qdict_put_obj(opts, "image", QOBJECT(bs->file->full_open_options));
> + QINCREF(bs->file->bs->full_open_options);
> + qdict_put_obj(opts, "image", QOBJECT(bs->file->bs->full_open_options));
>
> for (e = qdict_first(bs->options); e; e = qdict_next(bs->options, e)) {
> if (strcmp(qdict_entry_key(e), "x-image") &&
> diff --git a/block/blkverify.c b/block/blkverify.c
> index 6b71622..f8655ad 100644
> --- a/block/blkverify.c
> +++ b/block/blkverify.c
> @@ -123,10 +123,10 @@ static int blkverify_open(BlockDriverState *bs, QDict
> *options, int flags,
> }
>
> /* Open the raw file */
> - assert(bs->file == NULL);
> - ret = bdrv_open_image(&bs->file, qemu_opt_get(opts, "x-raw"), options,
> - "raw", bs, &child_file, false, &local_err);
> - if (ret < 0) {
Same as above, should we keep the assert?
> + bs->file = bdrv_open_child(qemu_opt_get(opts, "x-raw"), options, "raw",
> + bs, &child_file, false, &local_err);
> + if (local_err) {
> + ret = -EINVAL;
> error_propagate(errp, local_err);
> goto fail;
> }
> @@ -238,13 +238,13 @@ static BlockAIOCB *blkverify_aio_readv(BlockDriverState
> *bs,
> nb_sectors, cb, opaque);
>
> acb->verify = blkverify_verify_readv;
> - acb->buf = qemu_blockalign(bs->file, qiov->size);
> + acb->buf = qemu_blockalign(bs->file->bs, qiov->size);
> qemu_iovec_init(&acb->raw_qiov, acb->qiov->niov);
> qemu_iovec_clone(&acb->raw_qiov, qiov, acb->buf);
>
> bdrv_aio_readv(s->test_file->bs, sector_num, qiov, nb_sectors,
> blkverify_aio_cb, acb);
> - bdrv_aio_readv(bs->file, sector_num, &acb->raw_qiov, nb_sectors,
> + bdrv_aio_readv(bs->file->bs, sector_num, &acb->raw_qiov, nb_sectors,
> blkverify_aio_cb, acb);
> return &acb->common;
> }
> @@ -259,7 +259,7 @@ static BlockAIOCB *blkverify_aio_writev(BlockDriverState
> *bs,
>
> bdrv_aio_writev(s->test_file->bs, sector_num, qiov, nb_sectors,
> blkverify_aio_cb, acb);
> - bdrv_aio_writev(bs->file, sector_num, qiov, nb_sectors,
> + bdrv_aio_writev(bs->file->bs, sector_num, qiov, nb_sectors,
> blkverify_aio_cb, acb);
> return &acb->common;
> }
> @@ -279,7 +279,7 @@ static bool
> blkverify_recurse_is_first_non_filter(BlockDriverState *bs,
> {
> BDRVBlkverifyState *s = bs->opaque;
>
> - bool perm = bdrv_recurse_is_first_non_filter(bs->file, candidate);
> + bool perm = bdrv_recurse_is_first_non_filter(bs->file->bs, candidate);
>
> if (perm) {
> return true;
> @@ -308,15 +308,17 @@ static void blkverify_refresh_filename(BlockDriverState
> *bs)
> {
> BDRVBlkverifyState *s = bs->opaque;
>
> - /* bs->file has already been refreshed */
> + /* bs->file->bs has already been refreshed */
> bdrv_refresh_filename(s->test_file->bs);
>
> - if (bs->file->full_open_options && s->test_file->bs->full_open_options) {
> + if (bs->file->bs->full_open_options
> + && s->test_file->bs->full_open_options)
> + {
> QDict *opts = qdict_new();
> qdict_put_obj(opts, "driver",
> QOBJECT(qstring_from_str("blkverify")));
>
> - QINCREF(bs->file->full_open_options);
> - qdict_put_obj(opts, "raw", QOBJECT(bs->file->full_open_options));
> + QINCREF(bs->file->bs->full_open_options);
> + qdict_put_obj(opts, "raw", QOBJECT(bs->file->bs->full_open_options));
> QINCREF(s->test_file->bs->full_open_options);
> qdict_put_obj(opts, "test",
> QOBJECT(s->test_file->bs->full_open_options));
> @@ -324,10 +326,13 @@ static void blkverify_refresh_filename(BlockDriverState
> *bs)
> bs->full_open_options = opts;
> }
>
> - if (bs->file->exact_filename[0] && s->test_file->bs->exact_filename[0]) {
> + if (bs->file->bs->exact_filename[0]
> + && s->test_file->bs->exact_filename[0])
> + {
> snprintf(bs->exact_filename, sizeof(bs->exact_filename),
> "blkverify:%s:%s",
> - bs->file->exact_filename, s->test_file->bs->exact_filename);
> + bs->file->bs->exact_filename,
> + s->test_file->bs->exact_filename);
> }
> }
>
> diff --git a/block/bochs.c b/block/bochs.c
> index 199ac2b..18949b9 100644
> --- a/block/bochs.c
> +++ b/block/bochs.c
> @@ -103,7 +103,7 @@ static int bochs_open(BlockDriverState *bs, QDict
> *options, int flags,
>
> bs->read_only = 1; // no write support yet
>
> - ret = bdrv_pread(bs->file, 0, &bochs, sizeof(bochs));
> + ret = bdrv_pread(bs->file->bs, 0, &bochs, sizeof(bochs));
> if (ret < 0) {
> return ret;
> }
> @@ -137,7 +137,7 @@ static int bochs_open(BlockDriverState *bs, QDict
> *options, int flags,
> return -ENOMEM;
> }
>
> - ret = bdrv_pread(bs->file, le32_to_cpu(bochs.header), s->catalog_bitmap,
> + ret = bdrv_pread(bs->file->bs, le32_to_cpu(bochs.header),
> s->catalog_bitmap,
> s->catalog_size * 4);
> if (ret < 0) {
> goto fail;
> @@ -206,7 +206,7 @@ static int64_t seek_to_sector(BlockDriverState *bs,
> int64_t sector_num)
> (s->extent_blocks + s->bitmap_blocks));
>
> /* read in bitmap for current extent */
> - ret = bdrv_pread(bs->file, bitmap_offset + (extent_offset / 8),
> + ret = bdrv_pread(bs->file->bs, bitmap_offset + (extent_offset / 8),
> &bitmap_entry, 1);
> if (ret < 0) {
> return ret;
> @@ -229,7 +229,7 @@ static int bochs_read(BlockDriverState *bs, int64_t
> sector_num,
> if (block_offset < 0) {
> return block_offset;
> } else if (block_offset > 0) {
> - ret = bdrv_pread(bs->file, block_offset, buf, 512);
> + ret = bdrv_pread(bs->file->bs, block_offset, buf, 512);
> if (ret < 0) {
> return ret;
> }
> diff --git a/block/cloop.c b/block/cloop.c
> index f328be0..4190ae0 100644
> --- a/block/cloop.c
> +++ b/block/cloop.c
> @@ -66,7 +66,7 @@ static int cloop_open(BlockDriverState *bs, QDict *options,
> int flags,
> bs->read_only = 1;
>
> /* read header */
> - ret = bdrv_pread(bs->file, 128, &s->block_size, 4);
> + ret = bdrv_pread(bs->file->bs, 128, &s->block_size, 4);
> if (ret < 0) {
> return ret;
> }
> @@ -92,7 +92,7 @@ static int cloop_open(BlockDriverState *bs, QDict *options,
> int flags,
> return -EINVAL;
> }
>
> - ret = bdrv_pread(bs->file, 128 + 4, &s->n_blocks, 4);
> + ret = bdrv_pread(bs->file->bs, 128 + 4, &s->n_blocks, 4);
> if (ret < 0) {
> return ret;
> }
> @@ -123,7 +123,7 @@ static int cloop_open(BlockDriverState *bs, QDict
> *options, int flags,
> return -ENOMEM;
> }
>
> - ret = bdrv_pread(bs->file, 128 + 4 + 4, s->offsets, offsets_size);
> + ret = bdrv_pread(bs->file->bs, 128 + 4 + 4, s->offsets, offsets_size);
> if (ret < 0) {
> goto fail;
> }
> @@ -203,8 +203,8 @@ static inline int cloop_read_block(BlockDriverState *bs,
> int block_num)
> int ret;
> uint32_t bytes = s->offsets[block_num + 1] - s->offsets[block_num];
>
> - ret = bdrv_pread(bs->file, s->offsets[block_num],
> s->compressed_block,
> - bytes);
> + ret = bdrv_pread(bs->file->bs, s->offsets[block_num],
> + s->compressed_block, bytes);
> if (ret != bytes) {
> return -1;
> }
> diff --git a/block/dmg.c b/block/dmg.c
> index 9f25281..546a6f5 100644
> --- a/block/dmg.c
> +++ b/block/dmg.c
> @@ -85,7 +85,7 @@ static int read_uint64(BlockDriverState *bs, int64_t
> offset, uint64_t *result)
> uint64_t buffer;
> int ret;
>
> - ret = bdrv_pread(bs->file, offset, &buffer, 8);
> + ret = bdrv_pread(bs->file->bs, offset, &buffer, 8);
> if (ret < 0) {
> return ret;
> }
> @@ -99,7 +99,7 @@ static int read_uint32(BlockDriverState *bs, int64_t
> offset, uint32_t *result)
> uint32_t buffer;
> int ret;
>
> - ret = bdrv_pread(bs->file, offset, &buffer, 4);
> + ret = bdrv_pread(bs->file->bs, offset, &buffer, 4);
> if (ret < 0) {
> return ret;
> }
> @@ -354,7 +354,7 @@ static int dmg_read_resource_fork(BlockDriverState *bs,
> DmgHeaderState *ds,
> offset += 4;
>
> buffer = g_realloc(buffer, count);
> - ret = bdrv_pread(bs->file, offset, buffer, count);
> + ret = bdrv_pread(bs->file->bs, offset, buffer, count);
> if (ret < 0) {
> goto fail;
> }
> @@ -391,7 +391,7 @@ static int dmg_read_plist_xml(BlockDriverState *bs,
> DmgHeaderState *ds,
>
> buffer = g_malloc(info_length + 1);
> buffer[info_length] = '\0';
> - ret = bdrv_pread(bs->file, info_begin, buffer, info_length);
> + ret = bdrv_pread(bs->file->bs, info_begin, buffer, info_length);
> if (ret != info_length) {
> ret = -EINVAL;
> goto fail;
> @@ -446,7 +446,7 @@ static int dmg_open(BlockDriverState *bs, QDict *options,
> int flags,
> ds.max_sectors_per_chunk = 1;
>
> /* locate the UDIF trailer */
> - offset = dmg_find_koly_offset(bs->file, errp);
> + offset = dmg_find_koly_offset(bs->file->bs, errp);
> if (offset < 0) {
> ret = offset;
> goto fail;
> @@ -514,9 +514,9 @@ static int dmg_open(BlockDriverState *bs, QDict *options,
> int flags,
> }
>
> /* initialize zlib engine */
> - s->compressed_chunk = qemu_try_blockalign(bs->file,
> + s->compressed_chunk = qemu_try_blockalign(bs->file->bs,
> ds.max_compressed_size + 1);
> - s->uncompressed_chunk = qemu_try_blockalign(bs->file,
> + s->uncompressed_chunk = qemu_try_blockalign(bs->file->bs,
> 512 *
> ds.max_sectors_per_chunk);
> if (s->compressed_chunk == NULL || s->uncompressed_chunk == NULL) {
> ret = -ENOMEM;
> @@ -592,7 +592,7 @@ static inline int dmg_read_chunk(BlockDriverState *bs,
> uint64_t sector_num)
> case 0x80000005: { /* zlib compressed */
> /* we need to buffer, because only the chunk as whole can be
> * inflated. */
> - ret = bdrv_pread(bs->file, s->offsets[chunk],
> + ret = bdrv_pread(bs->file->bs, s->offsets[chunk],
> s->compressed_chunk, s->lengths[chunk]);
> if (ret != s->lengths[chunk]) {
> return -1;
> @@ -616,7 +616,7 @@ static inline int dmg_read_chunk(BlockDriverState *bs,
> uint64_t sector_num)
> case 0x80000006: /* bzip2 compressed */
> /* we need to buffer, because only the chunk as whole can be
> * inflated. */
> - ret = bdrv_pread(bs->file, s->offsets[chunk],
> + ret = bdrv_pread(bs->file->bs, s->offsets[chunk],
> s->compressed_chunk, s->lengths[chunk]);
> if (ret != s->lengths[chunk]) {
> return -1;
> @@ -641,7 +641,7 @@ static inline int dmg_read_chunk(BlockDriverState *bs,
> uint64_t sector_num)
> break;
> #endif /* CONFIG_BZIP2 */
> case 1: /* copy */
> - ret = bdrv_pread(bs->file, s->offsets[chunk],
> + ret = bdrv_pread(bs->file->bs, s->offsets[chunk],
> s->uncompressed_chunk, s->lengths[chunk]);
> if (ret != s->lengths[chunk]) {
> return -1;
> diff --git a/block/io.c b/block/io.c
> index 94e18e6..15c676a 100644
> --- a/block/io.c
> +++ b/block/io.c
> @@ -156,15 +156,15 @@ void bdrv_refresh_limits(BlockDriverState *bs, Error
> **errp)
>
> /* Take some limits from the children as a default */
> if (bs->file) {
> - bdrv_refresh_limits(bs->file, &local_err);
> + bdrv_refresh_limits(bs->file->bs, &local_err);
> if (local_err) {
> error_propagate(errp, local_err);
> return;
> }
> - bs->bl.opt_transfer_length = bs->file->bl.opt_transfer_length;
> - bs->bl.max_transfer_length = bs->file->bl.max_transfer_length;
> - bs->bl.min_mem_alignment = bs->file->bl.min_mem_alignment;
> - bs->bl.opt_mem_alignment = bs->file->bl.opt_mem_alignment;
> + bs->bl.opt_transfer_length = bs->file->bs->bl.opt_transfer_length;
> + bs->bl.max_transfer_length = bs->file->bs->bl.max_transfer_length;
> + bs->bl.min_mem_alignment = bs->file->bs->bl.min_mem_alignment;
> + bs->bl.opt_mem_alignment = bs->file->bs->bl.opt_mem_alignment;
> } else {
> bs->bl.min_mem_alignment = 512;
> bs->bl.opt_mem_alignment = getpagesize();
> @@ -224,7 +224,7 @@ static bool bdrv_requests_pending(BlockDriverState *bs)
> if (!qemu_co_queue_empty(&bs->throttled_reqs[1])) {
> return true;
> }
> - if (bs->file && bdrv_requests_pending(bs->file)) {
> + if (bs->file && bdrv_requests_pending(bs->file->bs)) {
> return true;
> }
> if (bs->backing_hd && bdrv_requests_pending(bs->backing_hd)) {
> @@ -1137,13 +1137,13 @@ static int coroutine_fn
> bdrv_aligned_pwritev(BlockDriverState *bs,
> if (ret < 0) {
> /* Do nothing, write notifier decided to fail this request */
> } else if (flags & BDRV_REQ_ZERO_WRITE) {
> - BLKDBG_EVENT(bs, BLKDBG_PWRITEV_ZERO);
> + bdrv_debug_event(bs, BLKDBG_PWRITEV_ZERO);
> ret = bdrv_co_do_write_zeroes(bs, sector_num, nb_sectors, flags);
> } else {
> - BLKDBG_EVENT(bs, BLKDBG_PWRITEV);
> + bdrv_debug_event(bs, BLKDBG_PWRITEV);
> ret = drv->bdrv_co_writev(bs, sector_num, nb_sectors, qiov);
> }
> - BLKDBG_EVENT(bs, BLKDBG_PWRITEV_DONE);
> + bdrv_debug_event(bs, BLKDBG_PWRITEV_DONE);
>
> if (ret == 0 && !bs->enable_write_cache) {
> ret = bdrv_co_flush(bs);
> @@ -1192,13 +1192,13 @@ static int coroutine_fn
> bdrv_co_do_zero_pwritev(BlockDriverState *bs,
> /* RMW the unaligned part before head. */
> mark_request_serialising(req, align);
> wait_serialising_requests(req);
> - BLKDBG_EVENT(bs, BLKDBG_PWRITEV_RMW_HEAD);
> + bdrv_debug_event(bs, BLKDBG_PWRITEV_RMW_HEAD);
> ret = bdrv_aligned_preadv(bs, req, offset & ~(align - 1), align,
> align, &local_qiov, 0);
> if (ret < 0) {
> goto fail;
> }
> - BLKDBG_EVENT(bs, BLKDBG_PWRITEV_RMW_AFTER_HEAD);
> + bdrv_debug_event(bs, BLKDBG_PWRITEV_RMW_AFTER_HEAD);
>
> memset(buf + head_padding_bytes, 0, zero_bytes);
> ret = bdrv_aligned_pwritev(bs, req, offset & ~(align - 1), align,
> @@ -1230,13 +1230,13 @@ static int coroutine_fn
> bdrv_co_do_zero_pwritev(BlockDriverState *bs,
> /* RMW the unaligned part after tail. */
> mark_request_serialising(req, align);
> wait_serialising_requests(req);
> - BLKDBG_EVENT(bs, BLKDBG_PWRITEV_RMW_TAIL);
> + bdrv_debug_event(bs, BLKDBG_PWRITEV_RMW_TAIL);
> ret = bdrv_aligned_preadv(bs, req, offset, align,
> align, &local_qiov, 0);
> if (ret < 0) {
> goto fail;
> }
> - BLKDBG_EVENT(bs, BLKDBG_PWRITEV_RMW_AFTER_TAIL);
> + bdrv_debug_event(bs, BLKDBG_PWRITEV_RMW_AFTER_TAIL);
>
> memset(buf, 0, bytes);
> ret = bdrv_aligned_pwritev(bs, req, offset, align,
> @@ -1307,13 +1307,13 @@ static int coroutine_fn
> bdrv_co_do_pwritev(BlockDriverState *bs,
> };
> qemu_iovec_init_external(&head_qiov, &head_iov, 1);
>
> - BLKDBG_EVENT(bs, BLKDBG_PWRITEV_RMW_HEAD);
> + bdrv_debug_event(bs, BLKDBG_PWRITEV_RMW_HEAD);
> ret = bdrv_aligned_preadv(bs, &req, offset & ~(align - 1), align,
> align, &head_qiov, 0);
> if (ret < 0) {
> goto fail;
> }
> - BLKDBG_EVENT(bs, BLKDBG_PWRITEV_RMW_AFTER_HEAD);
> + bdrv_debug_event(bs, BLKDBG_PWRITEV_RMW_AFTER_HEAD);
>
> qemu_iovec_init(&local_qiov, qiov->niov + 2);
> qemu_iovec_add(&local_qiov, head_buf, offset & (align - 1));
> @@ -1341,13 +1341,13 @@ static int coroutine_fn
> bdrv_co_do_pwritev(BlockDriverState *bs,
> };
> qemu_iovec_init_external(&tail_qiov, &tail_iov, 1);
>
> - BLKDBG_EVENT(bs, BLKDBG_PWRITEV_RMW_TAIL);
> + bdrv_debug_event(bs, BLKDBG_PWRITEV_RMW_TAIL);
> ret = bdrv_aligned_preadv(bs, &req, (offset + bytes) & ~(align - 1),
> align,
> align, &tail_qiov, 0);
> if (ret < 0) {
> goto fail;
> }
> - BLKDBG_EVENT(bs, BLKDBG_PWRITEV_RMW_AFTER_TAIL);
> + bdrv_debug_event(bs, BLKDBG_PWRITEV_RMW_AFTER_TAIL);
>
> if (!use_local_qiov) {
> qemu_iovec_init(&local_qiov, qiov->niov + 1);
> @@ -1496,7 +1496,7 @@ static int64_t coroutine_fn
> bdrv_co_get_block_status(BlockDriverState *bs,
>
> if (ret & BDRV_BLOCK_RAW) {
> assert(ret & BDRV_BLOCK_OFFSET_VALID);
> - return bdrv_get_block_status(bs->file, ret >> BDRV_SECTOR_BITS,
> + return bdrv_get_block_status(bs->file->bs, ret >> BDRV_SECTOR_BITS,
> *pnum, pnum);
> }
>
> @@ -1519,7 +1519,7 @@ static int64_t coroutine_fn
> bdrv_co_get_block_status(BlockDriverState *bs,
> (ret & BDRV_BLOCK_OFFSET_VALID)) {
> int file_pnum;
>
> - ret2 = bdrv_co_get_block_status(bs->file, ret >> BDRV_SECTOR_BITS,
> + ret2 = bdrv_co_get_block_status(bs->file->bs, ret >>
> BDRV_SECTOR_BITS,
> *pnum, &file_pnum);
> if (ret2 >= 0) {
> /* Ignore errors. This is just providing extra information, it
> @@ -1723,7 +1723,7 @@ int bdrv_writev_vmstate(BlockDriverState *bs,
> QEMUIOVector *qiov, int64_t pos)
> } else if (drv->bdrv_save_vmstate) {
> return drv->bdrv_save_vmstate(bs, qiov, pos);
> } else if (bs->file) {
> - return bdrv_writev_vmstate(bs->file, qiov, pos);
> + return bdrv_writev_vmstate(bs->file->bs, qiov, pos);
> }
>
> return -ENOTSUP;
> @@ -1738,7 +1738,7 @@ int bdrv_load_vmstate(BlockDriverState *bs, uint8_t
> *buf,
> if (drv->bdrv_load_vmstate)
> return drv->bdrv_load_vmstate(bs, buf, pos, size);
> if (bs->file)
> - return bdrv_load_vmstate(bs->file, buf, pos, size);
> + return bdrv_load_vmstate(bs->file->bs, buf, pos, size);
> return -ENOTSUP;
> }
>
> @@ -2366,7 +2366,7 @@ int coroutine_fn bdrv_co_flush(BlockDriverState *bs)
> * in the case of cache=unsafe, so there are no useless flushes.
> */
> flush_parent:
> - return bdrv_co_flush(bs->file);
> + return bs->file ? bdrv_co_flush(bs->file->bs) : 0;
> }
>
> int bdrv_flush(BlockDriverState *bs)
> @@ -2594,7 +2594,7 @@ void bdrv_io_plug(BlockDriverState *bs)
> if (drv && drv->bdrv_io_plug) {
> drv->bdrv_io_plug(bs);
> } else if (bs->file) {
> - bdrv_io_plug(bs->file);
> + bdrv_io_plug(bs->file->bs);
> }
> }
>
> @@ -2604,7 +2604,7 @@ void bdrv_io_unplug(BlockDriverState *bs)
> if (drv && drv->bdrv_io_unplug) {
> drv->bdrv_io_unplug(bs);
> } else if (bs->file) {
> - bdrv_io_unplug(bs->file);
> + bdrv_io_unplug(bs->file->bs);
> }
> }
>
> @@ -2614,7 +2614,7 @@ void bdrv_flush_io_queue(BlockDriverState *bs)
> if (drv && drv->bdrv_flush_io_queue) {
> drv->bdrv_flush_io_queue(bs);
> } else if (bs->file) {
> - bdrv_flush_io_queue(bs->file);
> + bdrv_flush_io_queue(bs->file->bs);
> }
> bdrv_start_throttled_reqs(bs);
> }
> diff --git a/block/parallels.c b/block/parallels.c
> index 5cd6ec3..4f79293 100644
> --- a/block/parallels.c
> +++ b/block/parallels.c
> @@ -202,13 +202,13 @@ static int64_t allocate_clusters(BlockDriverState *bs,
> int64_t sector_num,
>
> to_allocate = (sector_num + *pnum + s->tracks - 1) / s->tracks - idx;
> space = to_allocate * s->tracks;
> - if (s->data_end + space > bdrv_getlength(bs->file) >> BDRV_SECTOR_BITS) {
> + if (s->data_end + space > bdrv_getlength(bs->file->bs) >>
> BDRV_SECTOR_BITS) {
> int ret;
> space += s->prealloc_size;
> if (s->prealloc_mode == PRL_PREALLOC_MODE_FALLOCATE) {
> - ret = bdrv_write_zeroes(bs->file, s->data_end, space, 0);
> + ret = bdrv_write_zeroes(bs->file->bs, s->data_end, space, 0);
> } else {
> - ret = bdrv_truncate(bs->file,
> + ret = bdrv_truncate(bs->file->bs,
> (s->data_end + space) << BDRV_SECTOR_BITS);
> }
> if (ret < 0) {
> @@ -244,7 +244,8 @@ static coroutine_fn int
> parallels_co_flush_to_os(BlockDriverState *bs)
> if (off + to_write > s->header_size) {
> to_write = s->header_size - off;
> }
> - ret = bdrv_pwrite(bs->file, off, (uint8_t *)s->header + off,
> to_write);
> + ret = bdrv_pwrite(bs->file->bs, off, (uint8_t *)s->header + off,
> + to_write);
> if (ret < 0) {
> qemu_co_mutex_unlock(&s->lock);
> return ret;
> @@ -303,7 +304,7 @@ static coroutine_fn int
> parallels_co_writev(BlockDriverState *bs,
> qemu_iovec_reset(&hd_qiov);
> qemu_iovec_concat(&hd_qiov, qiov, bytes_done, nbytes);
>
> - ret = bdrv_co_writev(bs->file, position, n, &hd_qiov);
> + ret = bdrv_co_writev(bs->file->bs, position, n, &hd_qiov);
> if (ret < 0) {
> break;
> }
> @@ -343,7 +344,7 @@ static coroutine_fn int
> parallels_co_readv(BlockDriverState *bs,
> qemu_iovec_reset(&hd_qiov);
> qemu_iovec_concat(&hd_qiov, qiov, bytes_done, nbytes);
>
> - ret = bdrv_co_readv(bs->file, position, n, &hd_qiov);
> + ret = bdrv_co_readv(bs->file->bs, position, n, &hd_qiov);
> if (ret < 0) {
> break;
> }
> @@ -369,7 +370,7 @@ static int parallels_check(BlockDriverState *bs,
> BdrvCheckResult *res,
> bool flush_bat = false;
> int cluster_size = s->tracks << BDRV_SECTOR_BITS;
>
> - size = bdrv_getlength(bs->file);
> + size = bdrv_getlength(bs->file->bs);
> if (size < 0) {
> res->check_errors++;
> return size;
> @@ -424,7 +425,7 @@ static int parallels_check(BlockDriverState *bs,
> BdrvCheckResult *res,
> }
>
> if (flush_bat) {
> - ret = bdrv_pwrite_sync(bs->file, 0, s->header, s->header_size);
> + ret = bdrv_pwrite_sync(bs->file->bs, 0, s->header, s->header_size);
> if (ret < 0) {
> res->check_errors++;
> return ret;
> @@ -440,7 +441,7 @@ static int parallels_check(BlockDriverState *bs,
> BdrvCheckResult *res,
> size - res->image_end_offset);
> res->leaks += count;
> if (fix & BDRV_FIX_LEAKS) {
> - ret = bdrv_truncate(bs->file, res->image_end_offset);
> + ret = bdrv_truncate(bs->file->bs, res->image_end_offset);
> if (ret < 0) {
> res->check_errors++;
> return ret;
> @@ -546,12 +547,13 @@ static int parallels_probe(const uint8_t *buf, int
> buf_size,
> static int parallels_update_header(BlockDriverState *bs)
> {
> BDRVParallelsState *s = bs->opaque;
> - unsigned size = MAX(bdrv_opt_mem_align(bs->file),
> sizeof(ParallelsHeader));
> + unsigned size = MAX(bdrv_opt_mem_align(bs->file->bs),
> + sizeof(ParallelsHeader));
>
> if (size > s->header_size) {
> size = s->header_size;
> }
> - return bdrv_pwrite_sync(bs->file, 0, s->header, size);
> + return bdrv_pwrite_sync(bs->file->bs, 0, s->header, size);
> }
>
> static int parallels_open(BlockDriverState *bs, QDict *options, int flags,
> @@ -564,7 +566,7 @@ static int parallels_open(BlockDriverState *bs, QDict
> *options, int flags,
> Error *local_err = NULL;
> char *buf;
>
> - ret = bdrv_pread(bs->file, 0, &ph, sizeof(ph));
> + ret = bdrv_pread(bs->file->bs, 0, &ph, sizeof(ph));
> if (ret < 0) {
> goto fail;
> }
> @@ -603,8 +605,8 @@ static int parallels_open(BlockDriverState *bs, QDict
> *options, int flags,
> }
>
> size = bat_entry_off(s->bat_size);
> - s->header_size = ROUND_UP(size, bdrv_opt_mem_align(bs->file));
> - s->header = qemu_try_blockalign(bs->file, s->header_size);
> + s->header_size = ROUND_UP(size, bdrv_opt_mem_align(bs->file->bs));
> + s->header = qemu_try_blockalign(bs->file->bs, s->header_size);
> if (s->header == NULL) {
> ret = -ENOMEM;
> goto fail;
> @@ -619,7 +621,7 @@ static int parallels_open(BlockDriverState *bs, QDict
> *options, int flags,
> s->header_size = size;
> }
>
> - ret = bdrv_pread(bs->file, 0, s->header, s->header_size);
> + ret = bdrv_pread(bs->file->bs, 0, s->header, s->header_size);
> if (ret < 0) {
> goto fail;
> }
> @@ -663,8 +665,8 @@ static int parallels_open(BlockDriverState *bs, QDict
> *options, int flags,
> if (local_err != NULL) {
> goto fail_options;
> }
> - if (!bdrv_has_zero_init(bs->file) ||
> - bdrv_truncate(bs->file, bdrv_getlength(bs->file)) != 0) {
> + if (!bdrv_has_zero_init(bs->file->bs) ||
> + bdrv_truncate(bs->file->bs, bdrv_getlength(bs->file->bs)) != 0) {
> s->prealloc_mode = PRL_PREALLOC_MODE_FALLOCATE;
> }
>
> @@ -707,7 +709,7 @@ static void parallels_close(BlockDriverState *bs)
> }
>
> if (bs->open_flags & BDRV_O_RDWR) {
> - bdrv_truncate(bs->file, s->data_end << BDRV_SECTOR_BITS);
> + bdrv_truncate(bs->file->bs, s->data_end << BDRV_SECTOR_BITS);
> }
>
> g_free(s->bat_dirty_bmap);
> diff --git a/block/qapi.c b/block/qapi.c
> index 2ce5097..0c4654e 100644
> --- a/block/qapi.c
> +++ b/block/qapi.c
> @@ -359,7 +359,7 @@ static BlockStats *bdrv_query_stats(const
> BlockDriverState *bs,
>
> if (bs->file) {
> s->has_parent = true;
> - s->parent = bdrv_query_stats(bs->file, query_backing);
> + s->parent = bdrv_query_stats(bs->file->bs, query_backing);
> }
>
> if (query_backing && bs->backing_hd) {
> diff --git a/block/qcow.c b/block/qcow.c
> index 6e35db1..4d20cd5 100644
> --- a/block/qcow.c
> +++ b/block/qcow.c
> @@ -100,7 +100,7 @@ static int qcow_open(BlockDriverState *bs, QDict
> *options, int flags,
> int ret;
> QCowHeader header;
>
> - ret = bdrv_pread(bs->file, 0, &header, sizeof(header));
> + ret = bdrv_pread(bs->file->bs, 0, &header, sizeof(header));
> if (ret < 0) {
> goto fail;
> }
> @@ -193,7 +193,7 @@ static int qcow_open(BlockDriverState *bs, QDict
> *options, int flags,
> goto fail;
> }
>
> - ret = bdrv_pread(bs->file, s->l1_table_offset, s->l1_table,
> + ret = bdrv_pread(bs->file->bs, s->l1_table_offset, s->l1_table,
> s->l1_size * sizeof(uint64_t));
> if (ret < 0) {
> goto fail;
> @@ -205,7 +205,7 @@ static int qcow_open(BlockDriverState *bs, QDict
> *options, int flags,
>
> /* alloc L2 cache (max. 64k * 16 * 8 = 8 MB) */
> s->l2_cache =
> - qemu_try_blockalign(bs->file,
> + qemu_try_blockalign(bs->file->bs,
> s->l2_size * L2_CACHE_SIZE * sizeof(uint64_t));
> if (s->l2_cache == NULL) {
> error_setg(errp, "Could not allocate L2 table cache");
> @@ -224,7 +224,7 @@ static int qcow_open(BlockDriverState *bs, QDict
> *options, int flags,
> ret = -EINVAL;
> goto fail;
> }
> - ret = bdrv_pread(bs->file, header.backing_file_offset,
> + ret = bdrv_pread(bs->file->bs, header.backing_file_offset,
> bs->backing_file, len);
> if (ret < 0) {
> goto fail;
> @@ -369,13 +369,13 @@ static uint64_t get_cluster_offset(BlockDriverState *bs,
> if (!allocate)
> return 0;
> /* allocate a new l2 entry */
> - l2_offset = bdrv_getlength(bs->file);
> + l2_offset = bdrv_getlength(bs->file->bs);
> /* round to cluster size */
> l2_offset = (l2_offset + s->cluster_size - 1) & ~(s->cluster_size -
> 1);
> /* update the L1 entry */
> s->l1_table[l1_index] = l2_offset;
> tmp = cpu_to_be64(l2_offset);
> - if (bdrv_pwrite_sync(bs->file,
> + if (bdrv_pwrite_sync(bs->file->bs,
> s->l1_table_offset + l1_index * sizeof(tmp),
> &tmp, sizeof(tmp)) < 0)
> return 0;
> @@ -405,11 +405,12 @@ static uint64_t get_cluster_offset(BlockDriverState *bs,
> l2_table = s->l2_cache + (min_index << s->l2_bits);
> if (new_l2_table) {
> memset(l2_table, 0, s->l2_size * sizeof(uint64_t));
> - if (bdrv_pwrite_sync(bs->file, l2_offset, l2_table,
> + if (bdrv_pwrite_sync(bs->file->bs, l2_offset, l2_table,
> s->l2_size * sizeof(uint64_t)) < 0)
> return 0;
> } else {
> - if (bdrv_pread(bs->file, l2_offset, l2_table, s->l2_size *
> sizeof(uint64_t)) !=
> + if (bdrv_pread(bs->file->bs, l2_offset, l2_table,
> + s->l2_size * sizeof(uint64_t)) !=
> s->l2_size * sizeof(uint64_t))
> return 0;
> }
> @@ -430,20 +431,21 @@ static uint64_t get_cluster_offset(BlockDriverState *bs,
> overwritten */
> if (decompress_cluster(bs, cluster_offset) < 0)
> return 0;
> - cluster_offset = bdrv_getlength(bs->file);
> + cluster_offset = bdrv_getlength(bs->file->bs);
> cluster_offset = (cluster_offset + s->cluster_size - 1) &
> ~(s->cluster_size - 1);
> /* write the cluster content */
> - if (bdrv_pwrite(bs->file, cluster_offset, s->cluster_cache,
> s->cluster_size) !=
> + if (bdrv_pwrite(bs->file->bs, cluster_offset, s->cluster_cache,
> + s->cluster_size) !=
> s->cluster_size)
> return -1;
> } else {
> - cluster_offset = bdrv_getlength(bs->file);
> + cluster_offset = bdrv_getlength(bs->file->bs);
> if (allocate == 1) {
> /* round to cluster size */
> cluster_offset = (cluster_offset + s->cluster_size - 1) &
> ~(s->cluster_size - 1);
> - bdrv_truncate(bs->file, cluster_offset + s->cluster_size);
> + bdrv_truncate(bs->file->bs, cluster_offset +
> s->cluster_size);
> /* if encrypted, we must initialize the cluster
> content which won't be written */
> if (bs->encrypted &&
> @@ -463,7 +465,7 @@ static uint64_t get_cluster_offset(BlockDriverState *bs,
> errno = EIO;
> return -1;
> }
> - if (bdrv_pwrite(bs->file, cluster_offset + i *
> 512,
> + if (bdrv_pwrite(bs->file->bs, cluster_offset + i
> * 512,
> s->cluster_data, 512) != 512)
> return -1;
> }
> @@ -477,7 +479,7 @@ static uint64_t get_cluster_offset(BlockDriverState *bs,
> /* update L2 table */
> tmp = cpu_to_be64(cluster_offset);
> l2_table[l2_index] = tmp;
> - if (bdrv_pwrite_sync(bs->file, l2_offset + l2_index * sizeof(tmp),
> + if (bdrv_pwrite_sync(bs->file->bs, l2_offset + l2_index *
> sizeof(tmp),
> &tmp, sizeof(tmp)) < 0)
> return 0;
> }
> @@ -546,7 +548,7 @@ static int decompress_cluster(BlockDriverState *bs,
> uint64_t cluster_offset)
> if (s->cluster_cache_offset != coffset) {
> csize = cluster_offset >> (63 - s->cluster_bits);
> csize &= (s->cluster_size - 1);
> - ret = bdrv_pread(bs->file, coffset, s->cluster_data, csize);
> + ret = bdrv_pread(bs->file->bs, coffset, s->cluster_data, csize);
> if (ret != csize)
> return -1;
> if (decompress_buffer(s->cluster_cache, s->cluster_size,
> @@ -625,7 +627,7 @@ static coroutine_fn int qcow_co_readv(BlockDriverState
> *bs, int64_t sector_num,
> hd_iov.iov_len = n * 512;
> qemu_iovec_init_external(&hd_qiov, &hd_iov, 1);
> qemu_co_mutex_unlock(&s->lock);
> - ret = bdrv_co_readv(bs->file,
> + ret = bdrv_co_readv(bs->file->bs,
> (cluster_offset >> 9) + index_in_cluster,
> n, &hd_qiov);
> qemu_co_mutex_lock(&s->lock);
> @@ -727,7 +729,7 @@ static coroutine_fn int qcow_co_writev(BlockDriverState
> *bs, int64_t sector_num,
> hd_iov.iov_len = n * 512;
> qemu_iovec_init_external(&hd_qiov, &hd_iov, 1);
> qemu_co_mutex_unlock(&s->lock);
> - ret = bdrv_co_writev(bs->file,
> + ret = bdrv_co_writev(bs->file->bs,
> (cluster_offset >> 9) + index_in_cluster,
> n, &hd_qiov);
> qemu_co_mutex_lock(&s->lock);
> @@ -879,10 +881,10 @@ static int qcow_make_empty(BlockDriverState *bs)
> int ret;
>
> memset(s->l1_table, 0, l1_length);
> - if (bdrv_pwrite_sync(bs->file, s->l1_table_offset, s->l1_table,
> + if (bdrv_pwrite_sync(bs->file->bs, s->l1_table_offset, s->l1_table,
> l1_length) < 0)
> return -1;
> - ret = bdrv_truncate(bs->file, s->l1_table_offset + l1_length);
> + ret = bdrv_truncate(bs->file->bs, s->l1_table_offset + l1_length);
> if (ret < 0)
> return ret;
>
> @@ -962,7 +964,7 @@ static int qcow_write_compressed(BlockDriverState *bs,
> int64_t sector_num,
> }
>
> cluster_offset &= s->cluster_offset_mask;
> - ret = bdrv_pwrite(bs->file, cluster_offset, out_buf, out_len);
> + ret = bdrv_pwrite(bs->file->bs, cluster_offset, out_buf, out_len);
> if (ret < 0) {
> goto fail;
> }
> diff --git a/block/qcow2-cache.c b/block/qcow2-cache.c
> index 7b14c5c..86dd7f2 100644
> --- a/block/qcow2-cache.c
> +++ b/block/qcow2-cache.c
> @@ -127,7 +127,7 @@ Qcow2Cache *qcow2_cache_create(BlockDriverState *bs, int
> num_tables)
> c = g_new0(Qcow2Cache, 1);
> c->size = num_tables;
> c->entries = g_try_new0(Qcow2CachedTable, num_tables);
> - c->table_array = qemu_try_blockalign(bs->file,
> + c->table_array = qemu_try_blockalign(bs->file->bs,
> (size_t) num_tables *
> s->cluster_size);
>
> if (!c->entries || !c->table_array) {
> @@ -185,7 +185,7 @@ static int qcow2_cache_entry_flush(BlockDriverState *bs,
> Qcow2Cache *c, int i)
> if (c->depends) {
> ret = qcow2_cache_flush_dependency(bs, c);
> } else if (c->depends_on_flush) {
> - ret = bdrv_flush(bs->file);
> + ret = bdrv_flush(bs->file->bs);
> if (ret >= 0) {
> c->depends_on_flush = false;
> }
> @@ -216,7 +216,7 @@ static int qcow2_cache_entry_flush(BlockDriverState *bs,
> Qcow2Cache *c, int i)
> BLKDBG_EVENT(bs->file, BLKDBG_L2_UPDATE);
> }
>
> - ret = bdrv_pwrite(bs->file, c->entries[i].offset,
> + ret = bdrv_pwrite(bs->file->bs, c->entries[i].offset,
> qcow2_cache_get_table_addr(bs, c, i), s->cluster_size);
> if (ret < 0) {
> return ret;
> @@ -244,7 +244,7 @@ int qcow2_cache_flush(BlockDriverState *bs, Qcow2Cache *c)
> }
>
> if (result == 0) {
> - ret = bdrv_flush(bs->file);
> + ret = bdrv_flush(bs->file->bs);
> if (ret < 0) {
> result = ret;
> }
> @@ -356,7 +356,8 @@ static int qcow2_cache_do_get(BlockDriverState *bs,
> Qcow2Cache *c,
> BLKDBG_EVENT(bs->file, BLKDBG_L2_LOAD);
> }
>
> - ret = bdrv_pread(bs->file, offset, qcow2_cache_get_table_addr(bs, c,
> i),
> + ret = bdrv_pread(bs->file->bs, offset,
> + qcow2_cache_get_table_addr(bs, c, i),
> s->cluster_size);
> if (ret < 0) {
> return ret;
> diff --git a/block/qcow2-cluster.c b/block/qcow2-cluster.c
> index 6ede629..7844f8e 100644
> --- a/block/qcow2-cluster.c
> +++ b/block/qcow2-cluster.c
> @@ -72,7 +72,7 @@ int qcow2_grow_l1_table(BlockDriverState *bs, uint64_t
> min_size,
> #endif
>
> new_l1_size2 = sizeof(uint64_t) * new_l1_size;
> - new_l1_table = qemu_try_blockalign(bs->file,
> + new_l1_table = qemu_try_blockalign(bs->file->bs,
> align_offset(new_l1_size2, 512));
> if (new_l1_table == NULL) {
> return -ENOMEM;
> @@ -105,7 +105,8 @@ int qcow2_grow_l1_table(BlockDriverState *bs, uint64_t
> min_size,
> BLKDBG_EVENT(bs->file, BLKDBG_L1_GROW_WRITE_TABLE);
> for(i = 0; i < s->l1_size; i++)
> new_l1_table[i] = cpu_to_be64(new_l1_table[i]);
> - ret = bdrv_pwrite_sync(bs->file, new_l1_table_offset, new_l1_table,
> new_l1_size2);
> + ret = bdrv_pwrite_sync(bs->file->bs, new_l1_table_offset,
> + new_l1_table, new_l1_size2);
> if (ret < 0)
> goto fail;
> for(i = 0; i < s->l1_size; i++)
> @@ -115,7 +116,8 @@ int qcow2_grow_l1_table(BlockDriverState *bs, uint64_t
> min_size,
> BLKDBG_EVENT(bs->file, BLKDBG_L1_GROW_ACTIVATE_TABLE);
> cpu_to_be32w((uint32_t*)data, new_l1_size);
> stq_be_p(data + 4, new_l1_table_offset);
> - ret = bdrv_pwrite_sync(bs->file, offsetof(QCowHeader, l1_size),
> data,sizeof(data));
> + ret = bdrv_pwrite_sync(bs->file->bs, offsetof(QCowHeader, l1_size),
> + data, sizeof(data));
> if (ret < 0) {
> goto fail;
> }
> @@ -182,8 +184,9 @@ int qcow2_write_l1_entry(BlockDriverState *bs, int
> l1_index)
> }
>
> BLKDBG_EVENT(bs->file, BLKDBG_L1_UPDATE);
> - ret = bdrv_pwrite_sync(bs->file, s->l1_table_offset + 8 * l1_start_index,
> - buf, sizeof(buf));
> + ret = bdrv_pwrite_sync(bs->file->bs,
> + s->l1_table_offset + 8 * l1_start_index,
> + buf, sizeof(buf));
> if (ret < 0) {
> return ret;
> }
> @@ -440,7 +443,8 @@ static int coroutine_fn copy_sectors(BlockDriverState *bs,
> }
>
> BLKDBG_EVENT(bs->file, BLKDBG_COW_WRITE);
> - ret = bdrv_co_writev(bs->file, (cluster_offset >> 9) + n_start, n,
> &qiov);
> + ret = bdrv_co_writev(bs->file->bs, (cluster_offset >> 9) + n_start, n,
> + &qiov);
> if (ret < 0) {
> goto out;
> }
> @@ -817,7 +821,7 @@ int qcow2_alloc_cluster_link_l2(BlockDriverState *bs,
> QCowL2Meta *m)
>
> /*
> * If this was a COW, we need to decrease the refcount of the old
> cluster.
> - * Also flush bs->file to get the right order for L2 and refcount update.
> + * Also flush bs->file->bs to get the right order for L2 and refcount
> update.
> *
> * Don't discard clusters that reach a refcount of 0 (e.g. compressed
> * clusters), the next write will reuse them anyway.
> @@ -1412,7 +1416,8 @@ int qcow2_decompress_cluster(BlockDriverState *bs,
> uint64_t cluster_offset)
> sector_offset = coffset & 511;
> csize = nb_csectors * 512 - sector_offset;
> BLKDBG_EVENT(bs->file, BLKDBG_READ_COMPRESSED);
> - ret = bdrv_read(bs->file, coffset >> 9, s->cluster_data,
> nb_csectors);
> + ret = bdrv_read(bs->file->bs, coffset >> 9, s->cluster_data,
> + nb_csectors);
> if (ret < 0) {
> return ret;
> }
> @@ -1645,7 +1650,7 @@ static int expand_zero_clusters_in_l1(BlockDriverState
> *bs, uint64_t *l1_table,
> if (!is_active_l1) {
> /* inactive L2 tables require a buffer to be stored in when loading
> * them from disk */
> - l2_table = qemu_try_blockalign(bs->file, s->cluster_size);
> + l2_table = qemu_try_blockalign(bs->file->bs, s->cluster_size);
> if (l2_table == NULL) {
> return -ENOMEM;
> }
> @@ -1679,8 +1684,8 @@ static int expand_zero_clusters_in_l1(BlockDriverState
> *bs, uint64_t *l1_table,
> (void **)&l2_table);
> } else {
> /* load inactive L2 tables from disk */
> - ret = bdrv_read(bs->file, l2_offset / BDRV_SECTOR_SIZE,
> - (void *)l2_table, s->cluster_sectors);
> + ret = bdrv_read(bs->file->bs, l2_offset / BDRV_SECTOR_SIZE,
> + (void *)l2_table, s->cluster_sectors);
> }
> if (ret < 0) {
> goto fail;
> @@ -1754,7 +1759,7 @@ static int expand_zero_clusters_in_l1(BlockDriverState
> *bs, uint64_t *l1_table,
> goto fail;
> }
>
> - ret = bdrv_write_zeroes(bs->file, offset / BDRV_SECTOR_SIZE,
> + ret = bdrv_write_zeroes(bs->file->bs, offset / BDRV_SECTOR_SIZE,
> s->cluster_sectors, 0);
> if (ret < 0) {
> if (!preallocated) {
> @@ -1787,8 +1792,8 @@ static int expand_zero_clusters_in_l1(BlockDriverState
> *bs, uint64_t *l1_table,
> goto fail;
> }
>
> - ret = bdrv_write(bs->file, l2_offset / BDRV_SECTOR_SIZE,
> - (void *)l2_table, s->cluster_sectors);
> + ret = bdrv_write(bs->file->bs, l2_offset / BDRV_SECTOR_SIZE,
> + (void *)l2_table, s->cluster_sectors);
> if (ret < 0) {
> goto fail;
> }
> @@ -1861,8 +1866,9 @@ int qcow2_expand_zero_clusters(BlockDriverState *bs,
>
> l1_table = g_realloc(l1_table, l1_sectors * BDRV_SECTOR_SIZE);
>
> - ret = bdrv_read(bs->file, s->snapshots[i].l1_table_offset /
> - BDRV_SECTOR_SIZE, (void *)l1_table, l1_sectors);
> + ret = bdrv_read(bs->file->bs,
> + s->snapshots[i].l1_table_offset / BDRV_SECTOR_SIZE,
> + (void *)l1_table, l1_sectors);
> if (ret < 0) {
> goto fail;
> }
> diff --git a/block/qcow2-refcount.c b/block/qcow2-refcount.c
> index 2110839..4b81c8d 100644
> --- a/block/qcow2-refcount.c
> +++ b/block/qcow2-refcount.c
> @@ -101,7 +101,7 @@ int qcow2_refcount_init(BlockDriverState *bs)
> goto fail;
> }
> BLKDBG_EVENT(bs->file, BLKDBG_REFTABLE_LOAD);
> - ret = bdrv_pread(bs->file, s->refcount_table_offset,
> + ret = bdrv_pread(bs->file->bs, s->refcount_table_offset,
> s->refcount_table, refcount_table_size2);
> if (ret < 0) {
> goto fail;
> @@ -431,7 +431,7 @@ static int alloc_refcount_block(BlockDriverState *bs,
> if (refcount_table_index < s->refcount_table_size) {
> uint64_t data64 = cpu_to_be64(new_block);
> BLKDBG_EVENT(bs->file, BLKDBG_REFBLOCK_ALLOC_HOOKUP);
> - ret = bdrv_pwrite_sync(bs->file,
> + ret = bdrv_pwrite_sync(bs->file->bs,
> s->refcount_table_offset + refcount_table_index *
> sizeof(uint64_t),
> &data64, sizeof(data64));
> if (ret < 0) {
> @@ -535,7 +535,7 @@ static int alloc_refcount_block(BlockDriverState *bs,
>
> /* Write refcount blocks to disk */
> BLKDBG_EVENT(bs->file, BLKDBG_REFBLOCK_ALLOC_WRITE_BLOCKS);
> - ret = bdrv_pwrite_sync(bs->file, meta_offset, new_blocks,
> + ret = bdrv_pwrite_sync(bs->file->bs, meta_offset, new_blocks,
> blocks_clusters * s->cluster_size);
> g_free(new_blocks);
> new_blocks = NULL;
> @@ -549,7 +549,7 @@ static int alloc_refcount_block(BlockDriverState *bs,
> }
>
> BLKDBG_EVENT(bs->file, BLKDBG_REFBLOCK_ALLOC_WRITE_TABLE);
> - ret = bdrv_pwrite_sync(bs->file, table_offset, new_table,
> + ret = bdrv_pwrite_sync(bs->file->bs, table_offset, new_table,
> table_size * sizeof(uint64_t));
> if (ret < 0) {
> goto fail_table;
> @@ -564,8 +564,9 @@ static int alloc_refcount_block(BlockDriverState *bs,
> cpu_to_be64w((uint64_t*)data, table_offset);
> cpu_to_be32w((uint32_t*)(data + 8), table_clusters);
> BLKDBG_EVENT(bs->file, BLKDBG_REFBLOCK_ALLOC_SWITCH_TABLE);
> - ret = bdrv_pwrite_sync(bs->file, offsetof(QCowHeader,
> refcount_table_offset),
> - data, sizeof(data));
> + ret = bdrv_pwrite_sync(bs->file->bs,
> + offsetof(QCowHeader, refcount_table_offset),
> + data, sizeof(data));
> if (ret < 0) {
> goto fail_table;
> }
> @@ -613,7 +614,7 @@ void qcow2_process_discards(BlockDriverState *bs, int ret)
>
> /* Discard is optional, ignore the return value */
> if (ret >= 0) {
> - bdrv_discard(bs->file,
> + bdrv_discard(bs->file->bs,
> d->offset >> BDRV_SECTOR_BITS,
> d->bytes >> BDRV_SECTOR_BITS);
> }
> @@ -1068,7 +1069,7 @@ int qcow2_update_snapshot_refcount(BlockDriverState *bs,
> }
> l1_allocated = true;
>
> - ret = bdrv_pread(bs->file, l1_table_offset, l1_table, l1_size2);
> + ret = bdrv_pread(bs->file->bs, l1_table_offset, l1_table, l1_size2);
> if (ret < 0) {
> goto fail;
> }
> @@ -1221,7 +1222,8 @@ fail:
> cpu_to_be64s(&l1_table[i]);
> }
>
> - ret = bdrv_pwrite_sync(bs->file, l1_table_offset, l1_table,
> l1_size2);
> + ret = bdrv_pwrite_sync(bs->file->bs, l1_table_offset,
> + l1_table, l1_size2);
>
> for (i = 0; i < l1_size; i++) {
> be64_to_cpus(&l1_table[i]);
> @@ -1376,7 +1378,7 @@ static int check_refcounts_l2(BlockDriverState *bs,
> BdrvCheckResult *res,
> l2_size = s->l2_size * sizeof(uint64_t);
> l2_table = g_malloc(l2_size);
>
> - ret = bdrv_pread(bs->file, l2_offset, l2_table, l2_size);
> + ret = bdrv_pread(bs->file->bs, l2_offset, l2_table, l2_size);
> if (ret < 0) {
> fprintf(stderr, "ERROR: I/O error in check_refcounts_l2\n");
> res->check_errors++;
> @@ -1508,7 +1510,7 @@ static int check_refcounts_l1(BlockDriverState *bs,
> res->check_errors++;
> goto fail;
> }
> - ret = bdrv_pread(bs->file, l1_table_offset, l1_table, l1_size2);
> + ret = bdrv_pread(bs->file->bs, l1_table_offset, l1_table, l1_size2);
> if (ret < 0) {
> fprintf(stderr, "ERROR: I/O error in check_refcounts_l1\n");
> res->check_errors++;
> @@ -1606,7 +1608,7 @@ static int check_oflag_copied(BlockDriverState *bs,
> BdrvCheckResult *res,
> }
> }
>
> - ret = bdrv_pread(bs->file, l2_offset, l2_table,
> + ret = bdrv_pread(bs->file->bs, l2_offset, l2_table,
> s->l2_size * sizeof(uint64_t));
> if (ret < 0) {
> fprintf(stderr, "ERROR: Could not read L2 table: %s\n",
> @@ -1658,7 +1660,8 @@ static int check_oflag_copied(BlockDriverState *bs,
> BdrvCheckResult *res,
> goto fail;
> }
>
> - ret = bdrv_pwrite(bs->file, l2_offset, l2_table,
> s->cluster_size);
> + ret = bdrv_pwrite(bs->file->bs, l2_offset, l2_table,
> + s->cluster_size);
> if (ret < 0) {
> fprintf(stderr, "ERROR: Could not write L2 table: %s\n",
> strerror(-ret));
> @@ -1713,11 +1716,11 @@ static int check_refblocks(BlockDriverState *bs,
> BdrvCheckResult *res,
> goto resize_fail;
> }
>
> - ret = bdrv_truncate(bs->file, offset + s->cluster_size);
> + ret = bdrv_truncate(bs->file->bs, offset + s->cluster_size);
> if (ret < 0) {
> goto resize_fail;
> }
> - size = bdrv_getlength(bs->file);
> + size = bdrv_getlength(bs->file->bs);
> if (size < 0) {
> ret = size;
> goto resize_fail;
> @@ -2091,7 +2094,7 @@ write_refblocks:
> on_disk_refblock = (void *)((char *) *refcount_table +
> refblock_index * s->cluster_size);
>
> - ret = bdrv_write(bs->file, refblock_offset / BDRV_SECTOR_SIZE,
> + ret = bdrv_write(bs->file->bs, refblock_offset / BDRV_SECTOR_SIZE,
> on_disk_refblock, s->cluster_sectors);
> if (ret < 0) {
> fprintf(stderr, "ERROR writing refblock: %s\n", strerror(-ret));
> @@ -2140,7 +2143,7 @@ write_refblocks:
> }
>
> assert(reftable_size < INT_MAX / sizeof(uint64_t));
> - ret = bdrv_pwrite(bs->file, reftable_offset, on_disk_reftable,
> + ret = bdrv_pwrite(bs->file->bs, reftable_offset, on_disk_reftable,
> reftable_size * sizeof(uint64_t));
> if (ret < 0) {
> fprintf(stderr, "ERROR writing reftable: %s\n", strerror(-ret));
> @@ -2152,8 +2155,8 @@ write_refblocks:
> reftable_offset);
> cpu_to_be32w(&reftable_offset_and_clusters.reftable_clusters,
> size_to_clusters(s, reftable_size * sizeof(uint64_t)));
> - ret = bdrv_pwrite_sync(bs->file, offsetof(QCowHeader,
> - refcount_table_offset),
> + ret = bdrv_pwrite_sync(bs->file->bs, offsetof(QCowHeader,
> + refcount_table_offset),
> &reftable_offset_and_clusters,
> sizeof(reftable_offset_and_clusters));
> if (ret < 0) {
> @@ -2191,7 +2194,7 @@ int qcow2_check_refcounts(BlockDriverState *bs,
> BdrvCheckResult *res,
> bool rebuild = false;
> int ret;
>
> - size = bdrv_getlength(bs->file);
> + size = bdrv_getlength(bs->file->bs);
> if (size < 0) {
> res->check_errors++;
> return size;
> @@ -2400,7 +2403,7 @@ int qcow2_check_metadata_overlap(BlockDriverState *bs,
> int ign, int64_t offset,
> return -ENOMEM;
> }
>
> - ret = bdrv_pread(bs->file, l1_ofs, l1, l1_sz2);
> + ret = bdrv_pread(bs->file->bs, l1_ofs, l1, l1_sz2);
> if (ret < 0) {
> g_free(l1);
> return ret;
> diff --git a/block/qcow2-snapshot.c b/block/qcow2-snapshot.c
> index 92f4dfc..def7201 100644
> --- a/block/qcow2-snapshot.c
> +++ b/block/qcow2-snapshot.c
> @@ -64,7 +64,7 @@ int qcow2_read_snapshots(BlockDriverState *bs)
> for(i = 0; i < s->nb_snapshots; i++) {
> /* Read statically sized part of the snapshot header */
> offset = align_offset(offset, 8);
> - ret = bdrv_pread(bs->file, offset, &h, sizeof(h));
> + ret = bdrv_pread(bs->file->bs, offset, &h, sizeof(h));
> if (ret < 0) {
> goto fail;
> }
> @@ -83,7 +83,7 @@ int qcow2_read_snapshots(BlockDriverState *bs)
> name_size = be16_to_cpu(h.name_size);
>
> /* Read extra data */
> - ret = bdrv_pread(bs->file, offset, &extra,
> + ret = bdrv_pread(bs->file->bs, offset, &extra,
> MIN(sizeof(extra), extra_data_size));
> if (ret < 0) {
> goto fail;
> @@ -102,7 +102,7 @@ int qcow2_read_snapshots(BlockDriverState *bs)
>
> /* Read snapshot ID */
> sn->id_str = g_malloc(id_str_size + 1);
> - ret = bdrv_pread(bs->file, offset, sn->id_str, id_str_size);
> + ret = bdrv_pread(bs->file->bs, offset, sn->id_str, id_str_size);
> if (ret < 0) {
> goto fail;
> }
> @@ -111,7 +111,7 @@ int qcow2_read_snapshots(BlockDriverState *bs)
>
> /* Read snapshot name */
> sn->name = g_malloc(name_size + 1);
> - ret = bdrv_pread(bs->file, offset, sn->name, name_size);
> + ret = bdrv_pread(bs->file->bs, offset, sn->name, name_size);
> if (ret < 0) {
> goto fail;
> }
> @@ -214,25 +214,25 @@ static int qcow2_write_snapshots(BlockDriverState *bs)
> h.name_size = cpu_to_be16(name_size);
> offset = align_offset(offset, 8);
>
> - ret = bdrv_pwrite(bs->file, offset, &h, sizeof(h));
> + ret = bdrv_pwrite(bs->file->bs, offset, &h, sizeof(h));
> if (ret < 0) {
> goto fail;
> }
> offset += sizeof(h);
>
> - ret = bdrv_pwrite(bs->file, offset, &extra, sizeof(extra));
> + ret = bdrv_pwrite(bs->file->bs, offset, &extra, sizeof(extra));
> if (ret < 0) {
> goto fail;
> }
> offset += sizeof(extra);
>
> - ret = bdrv_pwrite(bs->file, offset, sn->id_str, id_str_size);
> + ret = bdrv_pwrite(bs->file->bs, offset, sn->id_str, id_str_size);
> if (ret < 0) {
> goto fail;
> }
> offset += id_str_size;
>
> - ret = bdrv_pwrite(bs->file, offset, sn->name, name_size);
> + ret = bdrv_pwrite(bs->file->bs, offset, sn->name, name_size);
> if (ret < 0) {
> goto fail;
> }
> @@ -254,7 +254,7 @@ static int qcow2_write_snapshots(BlockDriverState *bs)
> header_data.nb_snapshots = cpu_to_be32(s->nb_snapshots);
> header_data.snapshots_offset = cpu_to_be64(snapshots_offset);
>
> - ret = bdrv_pwrite_sync(bs->file, offsetof(QCowHeader, nb_snapshots),
> + ret = bdrv_pwrite_sync(bs->file->bs, offsetof(QCowHeader, nb_snapshots),
> &header_data, sizeof(header_data));
> if (ret < 0) {
> goto fail;
> @@ -396,7 +396,7 @@ int qcow2_snapshot_create(BlockDriverState *bs,
> QEMUSnapshotInfo *sn_info)
> goto fail;
> }
>
> - ret = bdrv_pwrite(bs->file, sn->l1_table_offset, l1_table,
> + ret = bdrv_pwrite(bs->file->bs, sn->l1_table_offset, l1_table,
> s->l1_size * sizeof(uint64_t));
> if (ret < 0) {
> goto fail;
> @@ -509,7 +509,8 @@ int qcow2_snapshot_goto(BlockDriverState *bs, const char
> *snapshot_id)
> goto fail;
> }
>
> - ret = bdrv_pread(bs->file, sn->l1_table_offset, sn_l1_table,
> sn_l1_bytes);
> + ret = bdrv_pread(bs->file->bs, sn->l1_table_offset,
> + sn_l1_table, sn_l1_bytes);
> if (ret < 0) {
> goto fail;
> }
> @@ -526,7 +527,7 @@ int qcow2_snapshot_goto(BlockDriverState *bs, const char
> *snapshot_id)
> goto fail;
> }
>
> - ret = bdrv_pwrite_sync(bs->file, s->l1_table_offset, sn_l1_table,
> + ret = bdrv_pwrite_sync(bs->file->bs, s->l1_table_offset, sn_l1_table,
> cur_l1_bytes);
> if (ret < 0) {
> goto fail;
> @@ -706,13 +707,14 @@ int qcow2_snapshot_load_tmp(BlockDriverState *bs,
> return -EFBIG;
> }
> new_l1_bytes = sn->l1_size * sizeof(uint64_t);
> - new_l1_table = qemu_try_blockalign(bs->file,
> + new_l1_table = qemu_try_blockalign(bs->file->bs,
> align_offset(new_l1_bytes, 512));
> if (new_l1_table == NULL) {
> return -ENOMEM;
> }
>
> - ret = bdrv_pread(bs->file, sn->l1_table_offset, new_l1_table,
> new_l1_bytes);
> + ret = bdrv_pread(bs->file->bs, sn->l1_table_offset,
> + new_l1_table, new_l1_bytes);
> if (ret < 0) {
> error_setg(errp, "Failed to read l1 table for snapshot");
> qemu_vfree(new_l1_table);
> diff --git a/block/qcow2.c b/block/qcow2.c
> index 56ad808..38b2797 100644
> --- a/block/qcow2.c
> +++ b/block/qcow2.c
> @@ -104,7 +104,7 @@ static int qcow2_read_extensions(BlockDriverState *bs,
> uint64_t start_offset,
> printf("attempting to read extended header in offset %lu\n", offset);
> #endif
>
> - ret = bdrv_pread(bs->file, offset, &ext, sizeof(ext));
> + ret = bdrv_pread(bs->file->bs, offset, &ext, sizeof(ext));
> if (ret < 0) {
> error_setg_errno(errp, -ret, "qcow2_read_extension: ERROR: "
> "pread fail from offset %" PRIu64, offset);
> @@ -132,7 +132,7 @@ static int qcow2_read_extensions(BlockDriverState *bs,
> uint64_t start_offset,
> sizeof(bs->backing_format));
> return 2;
> }
> - ret = bdrv_pread(bs->file, offset, bs->backing_format, ext.len);
> + ret = bdrv_pread(bs->file->bs, offset, bs->backing_format,
> ext.len);
> if (ret < 0) {
> error_setg_errno(errp, -ret, "ERROR: ext_backing_format: "
> "Could not read format name");
> @@ -148,7 +148,7 @@ static int qcow2_read_extensions(BlockDriverState *bs,
> uint64_t start_offset,
> case QCOW2_EXT_MAGIC_FEATURE_TABLE:
> if (p_feature_table != NULL) {
> void* feature_table = g_malloc0(ext.len + 2 *
> sizeof(Qcow2Feature));
> - ret = bdrv_pread(bs->file, offset , feature_table, ext.len);
> + ret = bdrv_pread(bs->file->bs, offset , feature_table,
> ext.len);
> if (ret < 0) {
> error_setg_errno(errp, -ret, "ERROR: ext_feature_table: "
> "Could not read table");
> @@ -169,7 +169,7 @@ static int qcow2_read_extensions(BlockDriverState *bs,
> uint64_t start_offset,
> uext->len = ext.len;
> QLIST_INSERT_HEAD(&s->unknown_header_ext, uext, next);
>
> - ret = bdrv_pread(bs->file, offset , uext->data, uext->len);
> + ret = bdrv_pread(bs->file->bs, offset , uext->data,
> uext->len);
> if (ret < 0) {
> error_setg_errno(errp, -ret, "ERROR: unknown extension: "
> "Could not read data");
> @@ -260,12 +260,12 @@ int qcow2_mark_dirty(BlockDriverState *bs)
> }
>
> val = cpu_to_be64(s->incompatible_features | QCOW2_INCOMPAT_DIRTY);
> - ret = bdrv_pwrite(bs->file, offsetof(QCowHeader, incompatible_features),
> + ret = bdrv_pwrite(bs->file->bs, offsetof(QCowHeader,
> incompatible_features),
> &val, sizeof(val));
> if (ret < 0) {
> return ret;
> }
> - ret = bdrv_flush(bs->file);
> + ret = bdrv_flush(bs->file->bs);
> if (ret < 0) {
> return ret;
> }
> @@ -828,7 +828,7 @@ static int qcow2_open(BlockDriverState *bs, QDict
> *options, int flags,
> uint64_t ext_end;
> uint64_t l1_vm_state_index;
>
> - ret = bdrv_pread(bs->file, 0, &header, sizeof(header));
> + ret = bdrv_pread(bs->file->bs, 0, &header, sizeof(header));
> if (ret < 0) {
> error_setg_errno(errp, -ret, "Could not read qcow2 header");
> goto fail;
> @@ -903,7 +903,7 @@ static int qcow2_open(BlockDriverState *bs, QDict
> *options, int flags,
> if (header.header_length > sizeof(header)) {
> s->unknown_header_fields_size = header.header_length -
> sizeof(header);
> s->unknown_header_fields = g_malloc(s->unknown_header_fields_size);
> - ret = bdrv_pread(bs->file, sizeof(header), s->unknown_header_fields,
> + ret = bdrv_pread(bs->file->bs, sizeof(header),
> s->unknown_header_fields,
> s->unknown_header_fields_size);
> if (ret < 0) {
> error_setg_errno(errp, -ret, "Could not read unknown qcow2
> header "
> @@ -1056,14 +1056,14 @@ static int qcow2_open(BlockDriverState *bs, QDict
> *options, int flags,
>
>
> if (s->l1_size > 0) {
> - s->l1_table = qemu_try_blockalign(bs->file,
> + s->l1_table = qemu_try_blockalign(bs->file->bs,
> align_offset(s->l1_size * sizeof(uint64_t), 512));
> if (s->l1_table == NULL) {
> error_setg(errp, "Could not allocate L1 table");
> ret = -ENOMEM;
> goto fail;
> }
> - ret = bdrv_pread(bs->file, s->l1_table_offset, s->l1_table,
> + ret = bdrv_pread(bs->file->bs, s->l1_table_offset, s->l1_table,
> s->l1_size * sizeof(uint64_t));
> if (ret < 0) {
> error_setg_errno(errp, -ret, "Could not read L1 table");
> @@ -1082,7 +1082,7 @@ static int qcow2_open(BlockDriverState *bs, QDict
> *options, int flags,
>
> s->cluster_cache = g_malloc(s->cluster_size);
> /* one more sector for decompressed data alignment */
> - s->cluster_data = qemu_try_blockalign(bs->file, QCOW_MAX_CRYPT_CLUSTERS
> + s->cluster_data = qemu_try_blockalign(bs->file->bs,
> QCOW_MAX_CRYPT_CLUSTERS
> * s->cluster_size + 512);
> if (s->cluster_data == NULL) {
> error_setg(errp, "Could not allocate temporary cluster buffer");
> @@ -1119,7 +1119,7 @@ static int qcow2_open(BlockDriverState *bs, QDict
> *options, int flags,
> ret = -EINVAL;
> goto fail;
> }
> - ret = bdrv_pread(bs->file, header.backing_file_offset,
> + ret = bdrv_pread(bs->file->bs, header.backing_file_offset,
> bs->backing_file, len);
> if (ret < 0) {
> error_setg_errno(errp, -ret, "Could not read backing file name");
> @@ -1429,8 +1429,9 @@ static coroutine_fn int qcow2_co_readv(BlockDriverState
> *bs, int64_t sector_num,
> */
> if (!cluster_data) {
> cluster_data =
> - qemu_try_blockalign(bs->file, QCOW_MAX_CRYPT_CLUSTERS
> - * s->cluster_size);
> + qemu_try_blockalign(bs->file->bs,
> + QCOW_MAX_CRYPT_CLUSTERS
> + * s->cluster_size);
> if (cluster_data == NULL) {
> ret = -ENOMEM;
> goto fail;
> @@ -1446,7 +1447,7 @@ static coroutine_fn int qcow2_co_readv(BlockDriverState
> *bs, int64_t sector_num,
>
> BLKDBG_EVENT(bs->file, BLKDBG_READ_AIO);
> qemu_co_mutex_unlock(&s->lock);
> - ret = bdrv_co_readv(bs->file,
> + ret = bdrv_co_readv(bs->file->bs,
> (cluster_offset >> 9) + index_in_cluster,
> cur_nr_sectors, &hd_qiov);
> qemu_co_mutex_lock(&s->lock);
> @@ -1543,7 +1544,7 @@ static coroutine_fn int
> qcow2_co_writev(BlockDriverState *bs,
> Error *err = NULL;
> assert(s->cipher);
> if (!cluster_data) {
> - cluster_data = qemu_try_blockalign(bs->file,
> + cluster_data = qemu_try_blockalign(bs->file->bs,
> QCOW_MAX_CRYPT_CLUSTERS
> * s->cluster_size);
> if (cluster_data == NULL) {
> @@ -1580,7 +1581,7 @@ static coroutine_fn int
> qcow2_co_writev(BlockDriverState *bs,
> BLKDBG_EVENT(bs->file, BLKDBG_WRITE_AIO);
> trace_qcow2_writev_data(qemu_coroutine_self(),
> (cluster_offset >> 9) + index_in_cluster);
> - ret = bdrv_co_writev(bs->file,
> + ret = bdrv_co_writev(bs->file->bs,
> (cluster_offset >> 9) + index_in_cluster,
> cur_nr_sectors, &hd_qiov);
> qemu_co_mutex_lock(&s->lock);
> @@ -1703,7 +1704,7 @@ static void qcow2_invalidate_cache(BlockDriverState
> *bs, Error **errp)
>
> qcow2_close(bs);
>
> - bdrv_invalidate_cache(bs->file, &local_err);
> + bdrv_invalidate_cache(bs->file->bs, &local_err);
> if (local_err) {
> error_propagate(errp, local_err);
> return;
> @@ -1911,7 +1912,7 @@ int qcow2_update_header(BlockDriverState *bs)
> }
>
> /* Write the new header */
> - ret = bdrv_pwrite(bs->file, 0, header, s->cluster_size);
> + ret = bdrv_pwrite(bs->file->bs, 0, header, s->cluster_size);
> if (ret < 0) {
> goto fail;
> }
> @@ -1991,7 +1992,8 @@ static int preallocate(BlockDriverState *bs)
> if (host_offset != 0) {
> uint8_t buf[BDRV_SECTOR_SIZE];
> memset(buf, 0, BDRV_SECTOR_SIZE);
> - ret = bdrv_write(bs->file, (host_offset >> BDRV_SECTOR_BITS) + num -
> 1,
> + ret = bdrv_write(bs->file->bs,
> + (host_offset >> BDRV_SECTOR_BITS) + num - 1,
> buf, 1);
> if (ret < 0) {
> return ret;
> @@ -2403,7 +2405,7 @@ static int qcow2_truncate(BlockDriverState *bs, int64_t
> offset)
>
> /* write updated header.size */
> offset = cpu_to_be64(offset);
> - ret = bdrv_pwrite_sync(bs->file, offsetof(QCowHeader, size),
> + ret = bdrv_pwrite_sync(bs->file->bs, offsetof(QCowHeader, size),
> &offset, sizeof(uint64_t));
> if (ret < 0) {
> return ret;
> @@ -2427,8 +2429,8 @@ static int qcow2_write_compressed(BlockDriverState *bs,
> int64_t sector_num,
> if (nb_sectors == 0) {
> /* align end of file to a sector boundary to ease reading with
> sector based I/Os */
> - cluster_offset = bdrv_getlength(bs->file);
> - return bdrv_truncate(bs->file, cluster_offset);
> + cluster_offset = bdrv_getlength(bs->file->bs);
> + return bdrv_truncate(bs->file->bs, cluster_offset);
> }
>
> if (nb_sectors != s->cluster_sectors) {
> @@ -2495,7 +2497,7 @@ static int qcow2_write_compressed(BlockDriverState *bs,
> int64_t sector_num,
> }
>
> BLKDBG_EVENT(bs->file, BLKDBG_WRITE_COMPRESSED);
> - ret = bdrv_pwrite(bs->file, cluster_offset, out_buf, out_len);
> + ret = bdrv_pwrite(bs->file->bs, cluster_offset, out_buf, out_len);
> if (ret < 0) {
> goto fail;
> }
> @@ -2544,7 +2546,7 @@ static int make_completely_empty(BlockDriverState *bs)
> /* After this call, neither the in-memory nor the on-disk refcount
> * information accurately describe the actual references */
>
> - ret = bdrv_write_zeroes(bs->file, s->l1_table_offset / BDRV_SECTOR_SIZE,
> + ret = bdrv_write_zeroes(bs->file->bs, s->l1_table_offset /
> BDRV_SECTOR_SIZE,
> l1_clusters * s->cluster_sectors, 0);
> if (ret < 0) {
> goto fail_broken_refcounts;
> @@ -2558,7 +2560,7 @@ static int make_completely_empty(BlockDriverState *bs)
> * overwrite parts of the existing refcount and L1 table, which is not
> * an issue because the dirty flag is set, complete data loss is in fact
> * desired and partial data loss is consequently fine as well */
> - ret = bdrv_write_zeroes(bs->file, s->cluster_size / BDRV_SECTOR_SIZE,
> + ret = bdrv_write_zeroes(bs->file->bs, s->cluster_size / BDRV_SECTOR_SIZE,
> (2 + l1_clusters) * s->cluster_size /
> BDRV_SECTOR_SIZE, 0);
> /* This call (even if it failed overall) may have overwritten on-disk
> @@ -2578,7 +2580,7 @@ static int make_completely_empty(BlockDriverState *bs)
> cpu_to_be64w(&l1_ofs_rt_ofs_cls.l1_offset, 3 * s->cluster_size);
> cpu_to_be64w(&l1_ofs_rt_ofs_cls.reftable_offset, s->cluster_size);
> cpu_to_be32w(&l1_ofs_rt_ofs_cls.reftable_clusters, 1);
> - ret = bdrv_pwrite_sync(bs->file, offsetof(QCowHeader, l1_table_offset),
> + ret = bdrv_pwrite_sync(bs->file->bs, offsetof(QCowHeader,
> l1_table_offset),
> &l1_ofs_rt_ofs_cls, sizeof(l1_ofs_rt_ofs_cls));
> if (ret < 0) {
> goto fail_broken_refcounts;
> @@ -2609,7 +2611,7 @@ static int make_completely_empty(BlockDriverState *bs)
>
> /* Enter the first refblock into the reftable */
> rt_entry = cpu_to_be64(2 * s->cluster_size);
> - ret = bdrv_pwrite_sync(bs->file, s->cluster_size,
> + ret = bdrv_pwrite_sync(bs->file->bs, s->cluster_size,
> &rt_entry, sizeof(rt_entry));
> if (ret < 0) {
> goto fail_broken_refcounts;
> @@ -2634,7 +2636,7 @@ static int make_completely_empty(BlockDriverState *bs)
> goto fail;
> }
>
> - ret = bdrv_truncate(bs->file, (3 + l1_clusters) * s->cluster_size);
> + ret = bdrv_truncate(bs->file->bs, (3 + l1_clusters) * s->cluster_size);
> if (ret < 0) {
> goto fail;
> }
> @@ -2769,7 +2771,7 @@ static void dump_refcounts(BlockDriverState *bs)
> int64_t nb_clusters, k, k1, size;
> int refcount;
>
> - size = bdrv_getlength(bs->file);
> + size = bdrv_getlength(bs->file->bs);
> nb_clusters = size_to_clusters(s, size);
> for(k = 0; k < nb_clusters;) {
> k1 = k;
> diff --git a/block/qed-table.c b/block/qed-table.c
> index 513aa87..f4219b8 100644
> --- a/block/qed-table.c
> +++ b/block/qed-table.c
> @@ -63,7 +63,7 @@ static void qed_read_table(BDRVQEDState *s, uint64_t
> offset, QEDTable *table,
> read_table_cb->iov.iov_len = s->header.cluster_size *
> s->header.table_size,
>
> qemu_iovec_init_external(qiov, &read_table_cb->iov, 1);
> - bdrv_aio_readv(s->bs->file, offset / BDRV_SECTOR_SIZE, qiov,
> + bdrv_aio_readv(s->bs->file->bs, offset / BDRV_SECTOR_SIZE, qiov,
> qiov->size / BDRV_SECTOR_SIZE,
> qed_read_table_cb, read_table_cb);
> }
> @@ -152,7 +152,7 @@ static void qed_write_table(BDRVQEDState *s, uint64_t
> offset, QEDTable *table,
> /* Adjust for offset into table */
> offset += start * sizeof(uint64_t);
>
> - bdrv_aio_writev(s->bs->file, offset / BDRV_SECTOR_SIZE,
> + bdrv_aio_writev(s->bs->file->bs, offset / BDRV_SECTOR_SIZE,
> &write_table_cb->qiov,
> write_table_cb->qiov.size / BDRV_SECTOR_SIZE,
> qed_write_table_cb, write_table_cb);
> diff --git a/block/qed.c b/block/qed.c
> index a7ff1d9..d953f8c 100644
> --- a/block/qed.c
> +++ b/block/qed.c
> @@ -82,7 +82,7 @@ int qed_write_header_sync(BDRVQEDState *s)
> int ret;
>
> qed_header_cpu_to_le(&s->header, &le);
> - ret = bdrv_pwrite(s->bs->file, 0, &le, sizeof(le));
> + ret = bdrv_pwrite(s->bs->file->bs, 0, &le, sizeof(le));
> if (ret != sizeof(le)) {
> return ret;
> }
> @@ -119,7 +119,7 @@ static void qed_write_header_read_cb(void *opaque, int
> ret)
> /* Update header */
> qed_header_cpu_to_le(&s->header, (QEDHeader *)write_header_cb->buf);
>
> - bdrv_aio_writev(s->bs->file, 0, &write_header_cb->qiov,
> + bdrv_aio_writev(s->bs->file->bs, 0, &write_header_cb->qiov,
> write_header_cb->nsectors, qed_write_header_cb,
> write_header_cb);
> }
> @@ -152,7 +152,7 @@ static void qed_write_header(BDRVQEDState *s,
> BlockCompletionFunc cb,
> write_header_cb->iov.iov_len = len;
> qemu_iovec_init_external(&write_header_cb->qiov, &write_header_cb->iov,
> 1);
>
> - bdrv_aio_readv(s->bs->file, 0, &write_header_cb->qiov, nsectors,
> + bdrv_aio_readv(s->bs->file->bs, 0, &write_header_cb->qiov, nsectors,
> qed_write_header_read_cb, write_header_cb);
> }
>
> @@ -392,7 +392,7 @@ static int bdrv_qed_open(BlockDriverState *bs, QDict
> *options, int flags,
> s->bs = bs;
> QSIMPLEQ_INIT(&s->allocating_write_reqs);
>
> - ret = bdrv_pread(bs->file, 0, &le_header, sizeof(le_header));
> + ret = bdrv_pread(bs->file->bs, 0, &le_header, sizeof(le_header));
> if (ret < 0) {
> return ret;
> }
> @@ -416,7 +416,7 @@ static int bdrv_qed_open(BlockDriverState *bs, QDict
> *options, int flags,
> }
>
> /* Round down file size to the last cluster */
> - file_size = bdrv_getlength(bs->file);
> + file_size = bdrv_getlength(bs->file->bs);
> if (file_size < 0) {
> return file_size;
> }
> @@ -452,7 +452,7 @@ static int bdrv_qed_open(BlockDriverState *bs, QDict
> *options, int flags,
> return -EINVAL;
> }
>
> - ret = qed_read_string(bs->file, s->header.backing_filename_offset,
> + ret = qed_read_string(bs->file->bs,
> s->header.backing_filename_offset,
> s->header.backing_filename_size,
> bs->backing_file,
> sizeof(bs->backing_file));
> if (ret < 0) {
> @@ -471,7 +471,7 @@ static int bdrv_qed_open(BlockDriverState *bs, QDict
> *options, int flags,
> * feature is no longer valid.
> */
> if ((s->header.autoclear_features & ~QED_AUTOCLEAR_FEATURE_MASK) != 0 &&
> - !bdrv_is_read_only(bs->file) && !(flags & BDRV_O_INCOMING)) {
> + !bdrv_is_read_only(bs->file->bs) && !(flags & BDRV_O_INCOMING)) {
> s->header.autoclear_features &= QED_AUTOCLEAR_FEATURE_MASK;
>
> ret = qed_write_header_sync(s);
> @@ -480,7 +480,7 @@ static int bdrv_qed_open(BlockDriverState *bs, QDict
> *options, int flags,
> }
>
> /* From here on only known autoclear feature bits are valid */
> - bdrv_flush(bs->file);
> + bdrv_flush(bs->file->bs);
> }
>
> s->l1_table = qed_alloc_table(s);
> @@ -498,7 +498,7 @@ static int bdrv_qed_open(BlockDriverState *bs, QDict
> *options, int flags,
> * potentially inconsistent images to be opened read-only. This can
> * aid data recovery from an otherwise inconsistent image.
> */
> - if (!bdrv_is_read_only(bs->file) &&
> + if (!bdrv_is_read_only(bs->file->bs) &&
> !(flags & BDRV_O_INCOMING)) {
> BdrvCheckResult result = {0};
>
> @@ -541,7 +541,7 @@ static void bdrv_qed_close(BlockDriverState *bs)
> bdrv_qed_detach_aio_context(bs);
>
> /* Ensure writes reach stable storage */
> - bdrv_flush(bs->file);
> + bdrv_flush(bs->file->bs);
>
> /* Clean shutdown, no check required on next open */
> if (s->header.features & QED_F_NEED_CHECK) {
> @@ -839,7 +839,7 @@ static void qed_copy_from_backing_file_write(void
> *opaque, int ret)
> }
>
> BLKDBG_EVENT(s->bs->file, BLKDBG_COW_WRITE);
> - bdrv_aio_writev(s->bs->file, copy_cb->offset / BDRV_SECTOR_SIZE,
> + bdrv_aio_writev(s->bs->file->bs, copy_cb->offset / BDRV_SECTOR_SIZE,
> ©_cb->qiov, copy_cb->qiov.size / BDRV_SECTOR_SIZE,
> qed_copy_from_backing_file_cb, copy_cb);
> }
> @@ -1055,7 +1055,7 @@ static void qed_aio_write_flush_before_l2_update(void
> *opaque, int ret)
> QEDAIOCB *acb = opaque;
> BDRVQEDState *s = acb_to_s(acb);
>
> - if (!bdrv_aio_flush(s->bs->file, qed_aio_write_l2_update_cb, opaque)) {
> + if (!bdrv_aio_flush(s->bs->file->bs, qed_aio_write_l2_update_cb,
> opaque)) {
> qed_aio_complete(acb, -EIO);
> }
> }
> @@ -1089,7 +1089,7 @@ static void qed_aio_write_main(void *opaque, int ret)
> }
>
> BLKDBG_EVENT(s->bs->file, BLKDBG_WRITE_AIO);
> - bdrv_aio_writev(s->bs->file, offset / BDRV_SECTOR_SIZE,
> + bdrv_aio_writev(s->bs->file->bs, offset / BDRV_SECTOR_SIZE,
> &acb->cur_qiov, acb->cur_qiov.size / BDRV_SECTOR_SIZE,
> next_fn, acb);
> }
> @@ -1321,7 +1321,7 @@ static void qed_aio_read_data(void *opaque, int ret,
> }
>
> BLKDBG_EVENT(bs->file, BLKDBG_READ_AIO);
> - bdrv_aio_readv(bs->file, offset / BDRV_SECTOR_SIZE,
> + bdrv_aio_readv(bs->file->bs, offset / BDRV_SECTOR_SIZE,
> &acb->cur_qiov, acb->cur_qiov.size / BDRV_SECTOR_SIZE,
> qed_aio_next_io, acb);
> return;
> @@ -1580,7 +1580,7 @@ static int
> bdrv_qed_change_backing_file(BlockDriverState *bs,
> }
>
> /* Write new header */
> - ret = bdrv_pwrite_sync(bs->file, 0, buffer, buffer_len);
> + ret = bdrv_pwrite_sync(bs->file->bs, 0, buffer, buffer_len);
> g_free(buffer);
> if (ret == 0) {
> memcpy(&s->header, &new_header, sizeof(new_header));
> @@ -1596,7 +1596,7 @@ static void bdrv_qed_invalidate_cache(BlockDriverState
> *bs, Error **errp)
>
> bdrv_qed_close(bs);
>
> - bdrv_invalidate_cache(bs->file, &local_err);
> + bdrv_invalidate_cache(bs->file->bs, &local_err);
> if (local_err) {
> error_propagate(errp, local_err);
> return;
> diff --git a/block/raw_bsd.c b/block/raw_bsd.c
> index e3d2d04..63ee911 100644
> --- a/block/raw_bsd.c
> +++ b/block/raw_bsd.c
> @@ -52,7 +52,7 @@ static int coroutine_fn raw_co_readv(BlockDriverState *bs,
> int64_t sector_num,
> int nb_sectors, QEMUIOVector *qiov)
> {
> BLKDBG_EVENT(bs->file, BLKDBG_READ_AIO);
> - return bdrv_co_readv(bs->file, sector_num, nb_sectors, qiov);
> + return bdrv_co_readv(bs->file->bs, sector_num, nb_sectors, qiov);
> }
>
> static int coroutine_fn raw_co_writev(BlockDriverState *bs, int64_t
> sector_num,
> @@ -75,7 +75,7 @@ static int coroutine_fn raw_co_writev(BlockDriverState *bs,
> int64_t sector_num,
> return 0;
> }
>
> - buf = qemu_try_blockalign(bs->file, 512);
> + buf = qemu_try_blockalign(bs->file->bs, 512);
> if (!buf) {
> ret = -ENOMEM;
> goto fail;
> @@ -102,7 +102,7 @@ static int coroutine_fn raw_co_writev(BlockDriverState
> *bs, int64_t sector_num,
> }
>
> BLKDBG_EVENT(bs->file, BLKDBG_WRITE_AIO);
> - ret = bdrv_co_writev(bs->file, sector_num, nb_sectors, qiov);
> + ret = bdrv_co_writev(bs->file->bs, sector_num, nb_sectors, qiov);
>
> fail:
> if (qiov == &local_qiov) {
> @@ -125,58 +125,58 @@ static int coroutine_fn
> raw_co_write_zeroes(BlockDriverState *bs,
> int64_t sector_num, int
> nb_sectors,
> BdrvRequestFlags flags)
> {
> - return bdrv_co_write_zeroes(bs->file, sector_num, nb_sectors, flags);
> + return bdrv_co_write_zeroes(bs->file->bs, sector_num, nb_sectors, flags);
> }
>
> static int coroutine_fn raw_co_discard(BlockDriverState *bs,
> int64_t sector_num, int nb_sectors)
> {
> - return bdrv_co_discard(bs->file, sector_num, nb_sectors);
> + return bdrv_co_discard(bs->file->bs, sector_num, nb_sectors);
> }
>
> static int64_t raw_getlength(BlockDriverState *bs)
> {
> - return bdrv_getlength(bs->file);
> + return bdrv_getlength(bs->file->bs);
> }
>
> static int raw_get_info(BlockDriverState *bs, BlockDriverInfo *bdi)
> {
> - return bdrv_get_info(bs->file, bdi);
> + return bdrv_get_info(bs->file->bs, bdi);
> }
>
> static void raw_refresh_limits(BlockDriverState *bs, Error **errp)
> {
> - bs->bl = bs->file->bl;
> + bs->bl = bs->file->bs->bl;
> }
>
> static int raw_truncate(BlockDriverState *bs, int64_t offset)
> {
> - return bdrv_truncate(bs->file, offset);
> + return bdrv_truncate(bs->file->bs, offset);
> }
>
> static int raw_is_inserted(BlockDriverState *bs)
> {
> - return bdrv_is_inserted(bs->file);
> + return bdrv_is_inserted(bs->file->bs);
> }
>
> static int raw_media_changed(BlockDriverState *bs)
> {
> - return bdrv_media_changed(bs->file);
> + return bdrv_media_changed(bs->file->bs);
> }
>
> static void raw_eject(BlockDriverState *bs, bool eject_flag)
> {
> - bdrv_eject(bs->file, eject_flag);
> + bdrv_eject(bs->file->bs, eject_flag);
> }
>
> static void raw_lock_medium(BlockDriverState *bs, bool locked)
> {
> - bdrv_lock_medium(bs->file, locked);
> + bdrv_lock_medium(bs->file->bs, locked);
> }
>
> static int raw_ioctl(BlockDriverState *bs, unsigned long int req, void *buf)
> {
> - return bdrv_ioctl(bs->file, req, buf);
> + return bdrv_ioctl(bs->file->bs, req, buf);
> }
>
> static BlockAIOCB *raw_aio_ioctl(BlockDriverState *bs,
> @@ -184,12 +184,12 @@ static BlockAIOCB *raw_aio_ioctl(BlockDriverState *bs,
> BlockCompletionFunc *cb,
> void *opaque)
> {
> - return bdrv_aio_ioctl(bs->file, req, buf, cb, opaque);
> + return bdrv_aio_ioctl(bs->file->bs, req, buf, cb, opaque);
> }
>
> static int raw_has_zero_init(BlockDriverState *bs)
> {
> - return bdrv_has_zero_init(bs->file);
> + return bdrv_has_zero_init(bs->file->bs);
> }
>
> static int raw_create(const char *filename, QemuOpts *opts, Error **errp)
> @@ -207,7 +207,7 @@ static int raw_create(const char *filename, QemuOpts
> *opts, Error **errp)
> static int raw_open(BlockDriverState *bs, QDict *options, int flags,
> Error **errp)
> {
> - bs->sg = bs->file->sg;
> + bs->sg = bs->file->bs->sg;
>
> if (bs->probed && !bdrv_is_read_only(bs)) {
> fprintf(stderr,
> @@ -217,7 +217,7 @@ static int raw_open(BlockDriverState *bs, QDict *options,
> int flags,
> "raw images, write operations on block 0 will be
> restricted.\n"
> " Specify the 'raw' format explicitly to remove the "
> "restrictions.\n",
> - bs->file->filename);
> + bs->file->bs->filename);
> }
>
> return 0;
> @@ -237,12 +237,12 @@ static int raw_probe(const uint8_t *buf, int buf_size,
> const char *filename)
>
> static int raw_probe_blocksizes(BlockDriverState *bs, BlockSizes *bsz)
> {
> - return bdrv_probe_blocksizes(bs->file, bsz);
> + return bdrv_probe_blocksizes(bs->file->bs, bsz);
> }
>
> static int raw_probe_geometry(BlockDriverState *bs, HDGeometry *geo)
> {
> - return bdrv_probe_geometry(bs->file, geo);
> + return bdrv_probe_geometry(bs->file->bs, geo);
> }
>
> BlockDriver bdrv_raw = {
> diff --git a/block/snapshot.c b/block/snapshot.c
> index 49e143e..89500f2 100644
> --- a/block/snapshot.c
> +++ b/block/snapshot.c
> @@ -149,7 +149,7 @@ int bdrv_can_snapshot(BlockDriverState *bs)
>
> if (!drv->bdrv_snapshot_create) {
> if (bs->file != NULL) {
> - return bdrv_can_snapshot(bs->file);
> + return bdrv_can_snapshot(bs->file->bs);
> }
> return 0;
> }
> @@ -168,7 +168,7 @@ int bdrv_snapshot_create(BlockDriverState *bs,
> return drv->bdrv_snapshot_create(bs, sn_info);
> }
> if (bs->file) {
> - return bdrv_snapshot_create(bs->file, sn_info);
> + return bdrv_snapshot_create(bs->file->bs, sn_info);
> }
> return -ENOTSUP;
> }
> @@ -188,10 +188,10 @@ int bdrv_snapshot_goto(BlockDriverState *bs,
>
> if (bs->file) {
> drv->bdrv_close(bs);
> - ret = bdrv_snapshot_goto(bs->file, snapshot_id);
> + ret = bdrv_snapshot_goto(bs->file->bs, snapshot_id);
> open_ret = drv->bdrv_open(bs, NULL, bs->open_flags, NULL);
> if (open_ret < 0) {
> - bdrv_unref(bs->file);
> + bdrv_unref(bs->file->bs);
> bs->drv = NULL;
> return open_ret;
> }
> @@ -245,7 +245,7 @@ int bdrv_snapshot_delete(BlockDriverState *bs,
> return drv->bdrv_snapshot_delete(bs, snapshot_id, name, errp);
> }
> if (bs->file) {
> - return bdrv_snapshot_delete(bs->file, snapshot_id, name, errp);
> + return bdrv_snapshot_delete(bs->file->bs, snapshot_id, name, errp);
> }
> error_setg(errp, "Block format '%s' used by device '%s' "
> "does not support internal snapshot deletion",
> @@ -283,7 +283,7 @@ int bdrv_snapshot_list(BlockDriverState *bs,
> return drv->bdrv_snapshot_list(bs, psn_info);
> }
> if (bs->file) {
> - return bdrv_snapshot_list(bs->file, psn_info);
> + return bdrv_snapshot_list(bs->file->bs, psn_info);
> }
> return -ENOTSUP;
> }
> diff --git a/block/vdi.c b/block/vdi.c
> index 062a654..17626d4 100644
> --- a/block/vdi.c
> +++ b/block/vdi.c
> @@ -399,7 +399,7 @@ static int vdi_open(BlockDriverState *bs, QDict *options,
> int flags,
>
> logout("\n");
>
> - ret = bdrv_read(bs->file, 0, (uint8_t *)&header, 1);
> + ret = bdrv_read(bs->file->bs, 0, (uint8_t *)&header, 1);
> if (ret < 0) {
> goto fail;
> }
> @@ -490,13 +490,14 @@ static int vdi_open(BlockDriverState *bs, QDict
> *options, int flags,
>
> bmap_size = header.blocks_in_image * sizeof(uint32_t);
> bmap_size = DIV_ROUND_UP(bmap_size, SECTOR_SIZE);
> - s->bmap = qemu_try_blockalign(bs->file, bmap_size * SECTOR_SIZE);
> + s->bmap = qemu_try_blockalign(bs->file->bs, bmap_size * SECTOR_SIZE);
> if (s->bmap == NULL) {
> ret = -ENOMEM;
> goto fail;
> }
>
> - ret = bdrv_read(bs->file, s->bmap_sector, (uint8_t *)s->bmap, bmap_size);
> + ret = bdrv_read(bs->file->bs, s->bmap_sector, (uint8_t *)s->bmap,
> + bmap_size);
> if (ret < 0) {
> goto fail_free_bmap;
> }
> @@ -585,7 +586,7 @@ static int vdi_co_read(BlockDriverState *bs,
> uint64_t offset = s->header.offset_data / SECTOR_SIZE +
> (uint64_t)bmap_entry * s->block_sectors +
> sector_in_block;
> - ret = bdrv_read(bs->file, offset, buf, n_sectors);
> + ret = bdrv_read(bs->file->bs, offset, buf, n_sectors);
> }
> logout("%u sectors read\n", n_sectors);
>
> @@ -653,7 +654,7 @@ static int vdi_co_write(BlockDriverState *bs,
> * acquire the lock and thus the padded cluster is written before
> * the other coroutines can write to the affected area. */
> qemu_co_mutex_lock(&s->write_lock);
> - ret = bdrv_write(bs->file, offset, block, s->block_sectors);
> + ret = bdrv_write(bs->file->bs, offset, block, s->block_sectors);
> qemu_co_mutex_unlock(&s->write_lock);
> } else {
> uint64_t offset = s->header.offset_data / SECTOR_SIZE +
> @@ -669,7 +670,7 @@ static int vdi_co_write(BlockDriverState *bs,
> * that that write operation has returned (there may be other
> writes
> * in flight, but they do not concern this very operation). */
> qemu_co_mutex_unlock(&s->write_lock);
> - ret = bdrv_write(bs->file, offset, buf, n_sectors);
> + ret = bdrv_write(bs->file->bs, offset, buf, n_sectors);
> }
>
> nb_sectors -= n_sectors;
> @@ -694,7 +695,7 @@ static int vdi_co_write(BlockDriverState *bs,
> assert(VDI_IS_ALLOCATED(bmap_first));
> *header = s->header;
> vdi_header_to_le(header);
> - ret = bdrv_write(bs->file, 0, block, 1);
> + ret = bdrv_write(bs->file->bs, 0, block, 1);
> g_free(block);
> block = NULL;
>
> @@ -712,7 +713,7 @@ static int vdi_co_write(BlockDriverState *bs,
> base = ((uint8_t *)&s->bmap[0]) + bmap_first * SECTOR_SIZE;
> logout("will write %u block map sectors starting from entry %u\n",
> n_sectors, bmap_first);
> - ret = bdrv_write(bs->file, offset, base, n_sectors);
> + ret = bdrv_write(bs->file->bs, offset, base, n_sectors);
> }
>
> return ret;
> diff --git a/block/vhdx-log.c b/block/vhdx-log.c
> index 47fec63..47ae4b1 100644
> --- a/block/vhdx-log.c
> +++ b/block/vhdx-log.c
> @@ -81,7 +81,7 @@ static int vhdx_log_peek_hdr(BlockDriverState *bs,
> VHDXLogEntries *log,
>
> offset = log->offset + read;
>
> - ret = bdrv_pread(bs->file, offset, hdr, sizeof(VHDXLogEntryHeader));
> + ret = bdrv_pread(bs->file->bs, offset, hdr, sizeof(VHDXLogEntryHeader));
> if (ret < 0) {
> goto exit;
> }
> @@ -141,7 +141,7 @@ static int vhdx_log_read_sectors(BlockDriverState *bs,
> VHDXLogEntries *log,
> }
> offset = log->offset + read;
>
> - ret = bdrv_pread(bs->file, offset, buffer, VHDX_LOG_SECTOR_SIZE);
> + ret = bdrv_pread(bs->file->bs, offset, buffer, VHDX_LOG_SECTOR_SIZE);
> if (ret < 0) {
> goto exit;
> }
> @@ -191,7 +191,8 @@ static int vhdx_log_write_sectors(BlockDriverState *bs,
> VHDXLogEntries *log,
> /* full */
> break;
> }
> - ret = bdrv_pwrite(bs->file, offset, buffer_tmp,
> VHDX_LOG_SECTOR_SIZE);
> + ret = bdrv_pwrite(bs->file->bs, offset, buffer_tmp,
> + VHDX_LOG_SECTOR_SIZE);
> if (ret < 0) {
> goto exit;
> }
> @@ -353,7 +354,7 @@ static int vhdx_log_read_desc(BlockDriverState *bs,
> BDRVVHDXState *s,
> }
>
> desc_sectors = vhdx_compute_desc_sectors(hdr.descriptor_count);
> - desc_entries = qemu_try_blockalign(bs->file,
> + desc_entries = qemu_try_blockalign(bs->file->bs,
> desc_sectors * VHDX_LOG_SECTOR_SIZE);
> if (desc_entries == NULL) {
> ret = -ENOMEM;
> @@ -462,7 +463,7 @@ static int vhdx_log_flush_desc(BlockDriverState *bs,
> VHDXLogDescriptor *desc,
>
> /* count is only > 1 if we are writing zeroes */
> for (i = 0; i < count; i++) {
> - ret = bdrv_pwrite_sync(bs->file, file_offset, buffer,
> + ret = bdrv_pwrite_sync(bs->file->bs, file_offset, buffer,
> VHDX_LOG_SECTOR_SIZE);
> if (ret < 0) {
> goto exit;
> @@ -509,7 +510,7 @@ static int vhdx_log_flush(BlockDriverState *bs,
> BDRVVHDXState *s,
> /* if the log shows a FlushedFileOffset larger than our current file
> * size, then that means the file has been truncated / corrupted, and
> * we must refused to open it / use it */
> - if (hdr_tmp.flushed_file_offset > bdrv_getlength(bs->file)) {
> + if (hdr_tmp.flushed_file_offset > bdrv_getlength(bs->file->bs)) {
> ret = -EINVAL;
> goto exit;
> }
> @@ -539,12 +540,12 @@ static int vhdx_log_flush(BlockDriverState *bs,
> BDRVVHDXState *s,
> goto exit;
> }
> }
> - if (bdrv_getlength(bs->file) < desc_entries->hdr.last_file_offset) {
> + if (bdrv_getlength(bs->file->bs) <
> desc_entries->hdr.last_file_offset) {
> new_file_size = desc_entries->hdr.last_file_offset;
> if (new_file_size % (1024*1024)) {
> /* round up to nearest 1MB boundary */
> new_file_size = ((new_file_size >> 20) + 1) << 20;
> - bdrv_truncate(bs->file, new_file_size);
> + bdrv_truncate(bs->file->bs, new_file_size);
> }
> }
> qemu_vfree(desc_entries);
> @@ -908,8 +909,8 @@ static int vhdx_log_write(BlockDriverState *bs,
> BDRVVHDXState *s,
> .sequence_number = s->log.sequence,
> .descriptor_count = sectors,
> .reserved = 0,
> - .flushed_file_offset = bdrv_getlength(bs->file),
> - .last_file_offset = bdrv_getlength(bs->file),
> + .flushed_file_offset = bdrv_getlength(bs->file->bs),
> + .last_file_offset = bdrv_getlength(bs->file->bs),
> };
>
> new_hdr.log_guid = header->log_guid;
> @@ -940,7 +941,7 @@ static int vhdx_log_write(BlockDriverState *bs,
> BDRVVHDXState *s,
>
> if (i == 0 && leading_length) {
> /* partial sector at the front of the buffer */
> - ret = bdrv_pread(bs->file, file_offset, merged_sector,
> + ret = bdrv_pread(bs->file->bs, file_offset, merged_sector,
> VHDX_LOG_SECTOR_SIZE);
> if (ret < 0) {
> goto exit;
> @@ -950,7 +951,7 @@ static int vhdx_log_write(BlockDriverState *bs,
> BDRVVHDXState *s,
> sector_write = merged_sector;
> } else if (i == sectors - 1 && trailing_length) {
> /* partial sector at the end of the buffer */
> - ret = bdrv_pread(bs->file,
> + ret = bdrv_pread(bs->file->bs,
> file_offset,
> merged_sector + trailing_length,
> VHDX_LOG_SECTOR_SIZE - trailing_length);
> diff --git a/block/vhdx.c b/block/vhdx.c
> index d3bb1bd..2fe9a5e 100644
> --- a/block/vhdx.c
> +++ b/block/vhdx.c
> @@ -375,7 +375,7 @@ static int vhdx_update_header(BlockDriverState *bs,
> BDRVVHDXState *s,
> inactive_header->log_guid = *log_guid;
> }
>
> - ret = vhdx_write_header(bs->file, inactive_header, header_offset, true);
> + ret = vhdx_write_header(bs->file->bs, inactive_header, header_offset,
> true);
> if (ret < 0) {
> goto exit;
> }
> @@ -427,7 +427,8 @@ static void vhdx_parse_header(BlockDriverState *bs,
> BDRVVHDXState *s,
> /* We have to read the whole VHDX_HEADER_SIZE instead of
> * sizeof(VHDXHeader), because the checksum is over the whole
> * region */
> - ret = bdrv_pread(bs->file, VHDX_HEADER1_OFFSET, buffer,
> VHDX_HEADER_SIZE);
> + ret = bdrv_pread(bs->file->bs, VHDX_HEADER1_OFFSET, buffer,
> + VHDX_HEADER_SIZE);
> if (ret < 0) {
> goto fail;
> }
> @@ -443,7 +444,8 @@ static void vhdx_parse_header(BlockDriverState *bs,
> BDRVVHDXState *s,
> }
> }
>
> - ret = bdrv_pread(bs->file, VHDX_HEADER2_OFFSET, buffer,
> VHDX_HEADER_SIZE);
> + ret = bdrv_pread(bs->file->bs, VHDX_HEADER2_OFFSET, buffer,
> + VHDX_HEADER_SIZE);
> if (ret < 0) {
> goto fail;
> }
> @@ -516,7 +518,7 @@ static int vhdx_open_region_tables(BlockDriverState *bs,
> BDRVVHDXState *s)
> * whole block */
> buffer = qemu_blockalign(bs, VHDX_HEADER_BLOCK_SIZE);
>
> - ret = bdrv_pread(bs->file, VHDX_REGION_TABLE_OFFSET, buffer,
> + ret = bdrv_pread(bs->file->bs, VHDX_REGION_TABLE_OFFSET, buffer,
> VHDX_HEADER_BLOCK_SIZE);
> if (ret < 0) {
> goto fail;
> @@ -629,7 +631,7 @@ static int vhdx_parse_metadata(BlockDriverState *bs,
> BDRVVHDXState *s)
>
> buffer = qemu_blockalign(bs, VHDX_METADATA_TABLE_MAX_SIZE);
>
> - ret = bdrv_pread(bs->file, s->metadata_rt.file_offset, buffer,
> + ret = bdrv_pread(bs->file->bs, s->metadata_rt.file_offset, buffer,
> VHDX_METADATA_TABLE_MAX_SIZE);
> if (ret < 0) {
> goto exit;
> @@ -732,7 +734,7 @@ static int vhdx_parse_metadata(BlockDriverState *bs,
> BDRVVHDXState *s)
> goto exit;
> }
>
> - ret = bdrv_pread(bs->file,
> + ret = bdrv_pread(bs->file->bs,
> s->metadata_entries.file_parameters_entry.offset
> + s->metadata_rt.file_offset,
> &s->params,
> @@ -767,7 +769,7 @@ static int vhdx_parse_metadata(BlockDriverState *bs,
> BDRVVHDXState *s)
> /* determine virtual disk size, logical sector size,
> * and phys sector size */
>
> - ret = bdrv_pread(bs->file,
> + ret = bdrv_pread(bs->file->bs,
> s->metadata_entries.virtual_disk_size_entry.offset
> + s->metadata_rt.file_offset,
> &s->virtual_disk_size,
> @@ -775,7 +777,7 @@ static int vhdx_parse_metadata(BlockDriverState *bs,
> BDRVVHDXState *s)
> if (ret < 0) {
> goto exit;
> }
> - ret = bdrv_pread(bs->file,
> + ret = bdrv_pread(bs->file->bs,
> s->metadata_entries.logical_sector_size_entry.offset
> + s->metadata_rt.file_offset,
> &s->logical_sector_size,
> @@ -783,7 +785,7 @@ static int vhdx_parse_metadata(BlockDriverState *bs,
> BDRVVHDXState *s)
> if (ret < 0) {
> goto exit;
> }
> - ret = bdrv_pread(bs->file,
> + ret = bdrv_pread(bs->file->bs,
> s->metadata_entries.phys_sector_size_entry.offset
> + s->metadata_rt.file_offset,
> &s->physical_sector_size,
> @@ -906,7 +908,7 @@ static int vhdx_open(BlockDriverState *bs, QDict
> *options, int flags,
> QLIST_INIT(&s->regions);
>
> /* validate the file signature */
> - ret = bdrv_pread(bs->file, 0, &signature, sizeof(uint64_t));
> + ret = bdrv_pread(bs->file->bs, 0, &signature, sizeof(uint64_t));
> if (ret < 0) {
> goto fail;
> }
> @@ -959,13 +961,13 @@ static int vhdx_open(BlockDriverState *bs, QDict
> *options, int flags,
> }
>
> /* s->bat is freed in vhdx_close() */
> - s->bat = qemu_try_blockalign(bs->file, s->bat_rt.length);
> + s->bat = qemu_try_blockalign(bs->file->bs, s->bat_rt.length);
> if (s->bat == NULL) {
> ret = -ENOMEM;
> goto fail;
> }
>
> - ret = bdrv_pread(bs->file, s->bat_offset, s->bat, s->bat_rt.length);
> + ret = bdrv_pread(bs->file->bs, s->bat_offset, s->bat, s->bat_rt.length);
> if (ret < 0) {
> goto fail;
> }
> @@ -1118,7 +1120,7 @@ static coroutine_fn int vhdx_co_readv(BlockDriverState
> *bs, int64_t sector_num,
> break;
> case PAYLOAD_BLOCK_FULLY_PRESENT:
> qemu_co_mutex_unlock(&s->lock);
> - ret = bdrv_co_readv(bs->file,
> + ret = bdrv_co_readv(bs->file->bs,
> sinfo.file_offset >> BDRV_SECTOR_BITS,
> sinfo.sectors_avail, &hd_qiov);
> qemu_co_mutex_lock(&s->lock);
> @@ -1156,12 +1158,12 @@ exit:
> static int vhdx_allocate_block(BlockDriverState *bs, BDRVVHDXState *s,
> uint64_t *new_offset)
> {
> - *new_offset = bdrv_getlength(bs->file);
> + *new_offset = bdrv_getlength(bs->file->bs);
>
> /* per the spec, the address for a block is in units of 1MB */
> *new_offset = ROUND_UP(*new_offset, 1024 * 1024);
>
> - return bdrv_truncate(bs->file, *new_offset + s->block_size);
> + return bdrv_truncate(bs->file->bs, *new_offset + s->block_size);
> }
>
> /*
> @@ -1260,7 +1262,7 @@ static coroutine_fn int vhdx_co_writev(BlockDriverState
> *bs, int64_t sector_num,
> /* Queue another write of zero buffers if the underlying file
> * does not zero-fill on file extension */
>
> - if (bdrv_has_zero_init(bs->file) == 0) {
> + if (bdrv_has_zero_init(bs->file->bs) == 0) {
> use_zero_buffers = true;
>
> /* zero fill the front, if any */
> @@ -1327,7 +1329,7 @@ static coroutine_fn int vhdx_co_writev(BlockDriverState
> *bs, int64_t sector_num,
> }
> /* block exists, so we can just overwrite it */
> qemu_co_mutex_unlock(&s->lock);
> - ret = bdrv_co_writev(bs->file,
> + ret = bdrv_co_writev(bs->file->bs,
> sinfo.file_offset >> BDRV_SECTOR_BITS,
> sectors_to_write, &hd_qiov);
> qemu_co_mutex_lock(&s->lock);
> diff --git a/block/vmdk.c b/block/vmdk.c
> index 9702132..9f7e7db 100644
> --- a/block/vmdk.c
> +++ b/block/vmdk.c
> @@ -221,7 +221,7 @@ static void vmdk_free_extents(BlockDriverState *bs)
> g_free(e->l2_cache);
> g_free(e->l1_backup_table);
> g_free(e->type);
> - if (e->file != bs->file_child) {
> + if (e->file != bs->file) {
> bdrv_unref_child(bs, e->file);
> }
> }
> @@ -248,7 +248,7 @@ static uint32_t vmdk_read_cid(BlockDriverState *bs, int
> parent)
> BDRVVmdkState *s = bs->opaque;
> int ret;
>
> - ret = bdrv_pread(bs->file, s->desc_offset, desc, DESC_SIZE);
> + ret = bdrv_pread(bs->file->bs, s->desc_offset, desc, DESC_SIZE);
> if (ret < 0) {
> return 0;
> }
> @@ -278,7 +278,7 @@ static int vmdk_write_cid(BlockDriverState *bs, uint32_t
> cid)
> BDRVVmdkState *s = bs->opaque;
> int ret;
>
> - ret = bdrv_pread(bs->file, s->desc_offset, desc, DESC_SIZE);
> + ret = bdrv_pread(bs->file->bs, s->desc_offset, desc, DESC_SIZE);
> if (ret < 0) {
> return ret;
> }
> @@ -297,7 +297,7 @@ static int vmdk_write_cid(BlockDriverState *bs, uint32_t
> cid)
> pstrcat(desc, sizeof(desc), tmp_desc);
> }
>
> - ret = bdrv_pwrite_sync(bs->file, s->desc_offset, desc, DESC_SIZE);
> + ret = bdrv_pwrite_sync(bs->file->bs, s->desc_offset, desc, DESC_SIZE);
> if (ret < 0) {
> return ret;
> }
> @@ -340,7 +340,7 @@ static int vmdk_parent_open(BlockDriverState *bs)
> int ret;
>
> desc[DESC_SIZE] = '\0';
> - ret = bdrv_pread(bs->file, s->desc_offset, desc, DESC_SIZE);
> + ret = bdrv_pread(bs->file->bs, s->desc_offset, desc, DESC_SIZE);
> if (ret < 0) {
> return ret;
> }
> @@ -621,7 +621,7 @@ static int vmdk_open_vmdk4(BlockDriverState *bs,
> } QEMU_PACKED footer;
>
> ret = bdrv_pread(file->bs,
> - bs->file->total_sectors * 512 - 1536,
> + bs->file->bs->total_sectors * 512 - 1536,
> &footer, sizeof(footer));
> if (ret < 0) {
> error_setg_errno(errp, -ret, "Failed to read footer");
> @@ -819,7 +819,7 @@ static int vmdk_parse_extents(const char *desc,
> BlockDriverState *bs,
> !desc_file_path[0])
> {
> error_setg(errp, "Cannot use relative extent paths with VMDK "
> - "descriptor file '%s'", bs->file->filename);
> + "descriptor file '%s'", bs->file->bs->filename);
> return -EINVAL;
> }
>
> @@ -905,7 +905,8 @@ static int vmdk_open_desc_file(BlockDriverState *bs, int
> flags, char *buf,
> }
> s->create_type = g_strdup(ct);
> s->desc_offset = 0;
> - ret = vmdk_parse_extents(buf, bs, bs->file->exact_filename, options,
> errp);
> + ret = vmdk_parse_extents(buf, bs, bs->file->bs->exact_filename, options,
> + errp);
> exit:
> return ret;
> }
> @@ -918,7 +919,7 @@ static int vmdk_open(BlockDriverState *bs, QDict
> *options, int flags,
> BDRVVmdkState *s = bs->opaque;
> uint32_t magic;
>
> - buf = vmdk_read_desc(bs->file, 0, errp);
> + buf = vmdk_read_desc(bs->file->bs, 0, errp);
> if (!buf) {
> return -EINVAL;
> }
> @@ -927,7 +928,7 @@ static int vmdk_open(BlockDriverState *bs, QDict
> *options, int flags,
> switch (magic) {
> case VMDK3_MAGIC:
> case VMDK4_MAGIC:
> - ret = vmdk_open_sparse(bs, bs->file_child, flags, buf, options,
> + ret = vmdk_open_sparse(bs, bs->file, flags, buf, options,
> errp);
> s->desc_offset = 0x200;
> break;
> @@ -1275,7 +1276,7 @@ static int64_t coroutine_fn
> vmdk_co_get_block_status(BlockDriverState *bs,
> break;
> case VMDK_OK:
> ret = BDRV_BLOCK_DATA;
> - if (extent->file == bs->file_child && !extent->compressed) {
> + if (extent->file == bs->file && !extent->compressed) {
> ret |= BDRV_BLOCK_OFFSET_VALID | offset;
> }
>
> @@ -2051,12 +2052,12 @@ static int64_t
> vmdk_get_allocated_file_size(BlockDriverState *bs)
> int64_t r;
> BDRVVmdkState *s = bs->opaque;
>
> - ret = bdrv_get_allocated_file_size(bs->file);
> + ret = bdrv_get_allocated_file_size(bs->file->bs);
> if (ret < 0) {
> return ret;
> }
> for (i = 0; i < s->num_extents; i++) {
> - if (s->extents[i].file == bs->file_child) {
> + if (s->extents[i].file == bs->file) {
> continue;
> }
> r = bdrv_get_allocated_file_size(s->extents[i].file->bs);
> diff --git a/block/vpc.c b/block/vpc.c
> index 2b3b518..299d373 100644
> --- a/block/vpc.c
> +++ b/block/vpc.c
> @@ -172,14 +172,14 @@ static int vpc_open(BlockDriverState *bs, QDict
> *options, int flags,
> int disk_type = VHD_DYNAMIC;
> int ret;
>
> - ret = bdrv_pread(bs->file, 0, s->footer_buf, HEADER_SIZE);
> + ret = bdrv_pread(bs->file->bs, 0, s->footer_buf, HEADER_SIZE);
> if (ret < 0) {
> goto fail;
> }
>
> footer = (VHDFooter *) s->footer_buf;
> if (strncmp(footer->creator, "conectix", 8)) {
> - int64_t offset = bdrv_getlength(bs->file);
> + int64_t offset = bdrv_getlength(bs->file->bs);
> if (offset < 0) {
> ret = offset;
> goto fail;
> @@ -189,7 +189,7 @@ static int vpc_open(BlockDriverState *bs, QDict *options,
> int flags,
> }
>
> /* If a fixed disk, the footer is found only at the end of the file
> */
> - ret = bdrv_pread(bs->file, offset-HEADER_SIZE, s->footer_buf,
> + ret = bdrv_pread(bs->file->bs, offset-HEADER_SIZE, s->footer_buf,
> HEADER_SIZE);
> if (ret < 0) {
> goto fail;
> @@ -232,7 +232,7 @@ static int vpc_open(BlockDriverState *bs, QDict *options,
> int flags,
> }
>
> if (disk_type == VHD_DYNAMIC) {
> - ret = bdrv_pread(bs->file, be64_to_cpu(footer->data_offset), buf,
> + ret = bdrv_pread(bs->file->bs, be64_to_cpu(footer->data_offset), buf,
> HEADER_SIZE);
> if (ret < 0) {
> goto fail;
> @@ -280,7 +280,7 @@ static int vpc_open(BlockDriverState *bs, QDict *options,
> int flags,
>
> pagetable_size = (uint64_t) s->max_table_entries * 4;
>
> - s->pagetable = qemu_try_blockalign(bs->file, pagetable_size);
> + s->pagetable = qemu_try_blockalign(bs->file->bs, pagetable_size);
> if (s->pagetable == NULL) {
> ret = -ENOMEM;
> goto fail;
> @@ -288,7 +288,8 @@ static int vpc_open(BlockDriverState *bs, QDict *options,
> int flags,
>
> s->bat_offset = be64_to_cpu(dyndisk_header->table_offset);
>
> - ret = bdrv_pread(bs->file, s->bat_offset, s->pagetable,
> pagetable_size);
> + ret = bdrv_pread(bs->file->bs, s->bat_offset, s->pagetable,
> + pagetable_size);
> if (ret < 0) {
> goto fail;
> }
> @@ -308,7 +309,7 @@ static int vpc_open(BlockDriverState *bs, QDict *options,
> int flags,
> }
> }
>
> - if (s->free_data_block_offset > bdrv_getlength(bs->file)) {
> + if (s->free_data_block_offset > bdrv_getlength(bs->file->bs)) {
> error_setg(errp, "block-vpc: free_data_block_offset points after
> "
> "the end of file. The image has been
> truncated.");
> ret = -EINVAL;
> @@ -383,7 +384,7 @@ static inline int64_t get_sector_offset(BlockDriverState
> *bs,
>
> s->last_bitmap_offset = bitmap_offset;
> memset(bitmap, 0xff, s->bitmap_size);
> - bdrv_pwrite_sync(bs->file, bitmap_offset, bitmap, s->bitmap_size);
> + bdrv_pwrite_sync(bs->file->bs, bitmap_offset, bitmap,
> s->bitmap_size);
> }
>
> return block_offset;
> @@ -401,7 +402,7 @@ static int rewrite_footer(BlockDriverState* bs)
> BDRVVPCState *s = bs->opaque;
> int64_t offset = s->free_data_block_offset;
>
> - ret = bdrv_pwrite_sync(bs->file, offset, s->footer_buf, HEADER_SIZE);
> + ret = bdrv_pwrite_sync(bs->file->bs, offset, s->footer_buf, HEADER_SIZE);
> if (ret < 0)
> return ret;
>
> @@ -436,7 +437,7 @@ static int64_t alloc_block(BlockDriverState* bs, int64_t
> sector_num)
>
> // Initialize the block's bitmap
> memset(bitmap, 0xff, s->bitmap_size);
> - ret = bdrv_pwrite_sync(bs->file, s->free_data_block_offset, bitmap,
> + ret = bdrv_pwrite_sync(bs->file->bs, s->free_data_block_offset, bitmap,
> s->bitmap_size);
> if (ret < 0) {
> return ret;
> @@ -451,7 +452,7 @@ static int64_t alloc_block(BlockDriverState* bs, int64_t
> sector_num)
> // Write BAT entry to disk
> bat_offset = s->bat_offset + (4 * index);
> bat_value = cpu_to_be32(s->pagetable[index]);
> - ret = bdrv_pwrite_sync(bs->file, bat_offset, &bat_value, 4);
> + ret = bdrv_pwrite_sync(bs->file->bs, bat_offset, &bat_value, 4);
> if (ret < 0)
> goto fail;
>
> @@ -485,7 +486,7 @@ static int vpc_read(BlockDriverState *bs, int64_t
> sector_num,
> VHDFooter *footer = (VHDFooter *) s->footer_buf;
>
> if (be32_to_cpu(footer->type) == VHD_FIXED) {
> - return bdrv_read(bs->file, sector_num, buf, nb_sectors);
> + return bdrv_read(bs->file->bs, sector_num, buf, nb_sectors);
> }
> while (nb_sectors > 0) {
> offset = get_sector_offset(bs, sector_num, 0);
> @@ -499,7 +500,7 @@ static int vpc_read(BlockDriverState *bs, int64_t
> sector_num,
> if (offset == -1) {
> memset(buf, 0, sectors * BDRV_SECTOR_SIZE);
> } else {
> - ret = bdrv_pread(bs->file, offset, buf,
> + ret = bdrv_pread(bs->file->bs, offset, buf,
> sectors * BDRV_SECTOR_SIZE);
> if (ret != sectors * BDRV_SECTOR_SIZE) {
> return -1;
> @@ -534,7 +535,7 @@ static int vpc_write(BlockDriverState *bs, int64_t
> sector_num,
> VHDFooter *footer = (VHDFooter *) s->footer_buf;
>
> if (be32_to_cpu(footer->type) == VHD_FIXED) {
> - return bdrv_write(bs->file, sector_num, buf, nb_sectors);
> + return bdrv_write(bs->file->bs, sector_num, buf, nb_sectors);
> }
> while (nb_sectors > 0) {
> offset = get_sector_offset(bs, sector_num, 1);
> @@ -551,7 +552,8 @@ static int vpc_write(BlockDriverState *bs, int64_t
> sector_num,
> return -1;
> }
>
> - ret = bdrv_pwrite(bs->file, offset, buf, sectors * BDRV_SECTOR_SIZE);
> + ret = bdrv_pwrite(bs->file->bs, offset, buf,
> + sectors * BDRV_SECTOR_SIZE);
> if (ret != sectors * BDRV_SECTOR_SIZE) {
> return -1;
> }
> @@ -878,7 +880,7 @@ static int vpc_has_zero_init(BlockDriverState *bs)
> VHDFooter *footer = (VHDFooter *) s->footer_buf;
>
> if (be32_to_cpu(footer->type) == VHD_FIXED) {
> - return bdrv_has_zero_init(bs->file);
> + return bdrv_has_zero_init(bs->file->bs);
> } else {
> return 1;
> }
> diff --git a/include/block/block.h b/include/block/block.h
> index 2dd6630..7ebb35d 100644
> --- a/include/block/block.h
> +++ b/include/block/block.h
> @@ -585,7 +585,13 @@ typedef enum {
> BLKDBG_EVENT_MAX,
> } BlkDebugEvent;
>
> -#define BLKDBG_EVENT(bs, evt) bdrv_debug_event(bs, evt)
> +#define BLKDBG_EVENT(child, evt) \
> + do { \
> + if (child) { \
> + bdrv_debug_event(child->bs, evt); \
> + } \
> + } while(0)
> +
I'm a bit surprised you changed the argument type to BdrvChild, since adding a
BLKDBG_EVENT_CHILD is more natural to me. But that probably doesn't hurt much:
Reviewed-by: Fam Zheng <address@hidden>
- [Qemu-block] [PATCH v2 00/16] block: Get rid of bdrv_swap(), Kevin Wolf, 2015/10/08
- [Qemu-block] [PATCH v2 04/16] quorum: Convert to BdrvChild, Kevin Wolf, 2015/10/08
- [Qemu-block] [PATCH v2 06/16] block: Remove bdrv_open_image(), Kevin Wolf, 2015/10/08
- [Qemu-block] [PATCH v2 16/16] block: Remove bdrv_swap(), Kevin Wolf, 2015/10/08
- [Qemu-block] [PATCH v2 15/16] block: Add and use bdrv_replace_in_backing_chain(), Kevin Wolf, 2015/10/08
- [Qemu-block] [PATCH v2 05/16] block: Convert bs->file to BdrvChild, Kevin Wolf, 2015/10/08
- [Qemu-block] [PATCH v2 11/16] block-backend: Add blk_set_bs(), Kevin Wolf, 2015/10/08
- [Qemu-block] [PATCH v2 14/16] blockjob: Store device name at job creation, Kevin Wolf, 2015/10/08
- [Qemu-block] [PATCH v2 03/16] blkverify: Convert s->test_file to BdrvChild, Kevin Wolf, 2015/10/08
- [Qemu-block] [PATCH v2 09/16] block: Split bdrv_move_feature_fields(), Kevin Wolf, 2015/10/08