Blame SOURCES/kvm-block-Convert-.bdrv_truncate-callback-to-coroutine_f.patch

357786
From c557973bf44fa50d5ef2334a2179546698a9bbf0 Mon Sep 17 00:00:00 2001
357786
From: Kevin Wolf <kwolf@redhat.com>
357786
Date: Thu, 12 Jul 2018 14:42:54 +0200
357786
Subject: [PATCH 35/89] block: Convert .bdrv_truncate callback to coroutine_fn
357786
357786
RH-Author: Kevin Wolf <kwolf@redhat.com>
357786
Message-id: <20180712144258.17303-3-kwolf@redhat.com>
357786
Patchwork-id: 81327
357786
O-Subject: [RHV-7.6 qemu-kvm-rhev PATCH 2/6] block: Convert .bdrv_truncate callback to coroutine_fn
357786
Bugzilla: 1595173
357786
RH-Acked-by: Max Reitz <mreitz@redhat.com>
357786
RH-Acked-by: Stefan Hajnoczi <stefanha@redhat.com>
357786
RH-Acked-by: John Snow <jsnow@redhat.com>
357786
357786
bdrv_truncate() is an operation that can block (even for a quite long
357786
time, depending on the PreallocMode) in I/O paths that shouldn't block.
357786
Convert it to a coroutine_fn so that we have the infrastructure for
357786
drivers to make their .bdrv_co_truncate implementation asynchronous.
357786
357786
This change could potentially introduce new race conditions because
357786
bdrv_truncate() isn't necessarily executed atomically any more. Whether
357786
this is a problem needs to be evaluated for each block driver that
357786
supports truncate:
357786
357786
* file-posix/win32, gluster, iscsi, nfs, rbd, ssh, sheepdog: The
357786
  protocol drivers are trivially safe because they don't actually yield
357786
  yet, so there is no change in behaviour.
357786
357786
* copy-on-read, crypto, raw-format: Essentially just filter drivers that
357786
  pass the request to a child node, no problem.
357786
357786
* qcow2: The implementation modifies metadata, so it needs to hold
357786
  s->lock to be safe with concurrent I/O requests. In order to avoid
357786
  double locking, this requires pulling the locking out into
357786
  preallocate_co() and using qcow2_write_caches() instead of
357786
  bdrv_flush().
357786
357786
* qed: Does a single header update, this is fine without locking.
357786
357786
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
357786
Reviewed-by: Stefan Hajnoczi <stefanha@redhat.com>
357786
(cherry picked from commit 061ca8a368165fae300748c17971824a089f521f)
357786
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
357786
Signed-off-by: Miroslav Rezanina <mrezanin@redhat.com>
357786
---
357786
 block.c                   | 63 +++++++++++++++++++++++++++++++++++-----
357786
 block/copy-on-read.c      |  8 ++---
357786
 block/crypto.c            |  9 +++---
357786
 block/file-posix.c        | 12 ++++----
357786
 block/file-win32.c        |  6 ++--
357786
 block/gluster.c           | 14 +++++----
357786
 block/iscsi.c             |  8 ++---
357786
 block/nfs.c               |  7 +++--
357786
 block/qcow2.c             | 74 ++++++++++++++++++++++++++++-------------------
357786
 block/qed.c               |  8 +++--
357786
 block/raw-format.c        |  8 ++---
357786
 block/rbd.c               |  8 +++--
357786
 block/sheepdog.c          | 12 ++++----
357786
 block/ssh.c               |  6 ++--
357786
 include/block/block.h     |  4 +++
357786
 include/block/block_int.h |  4 +--
357786
 16 files changed, 162 insertions(+), 89 deletions(-)
357786
357786
diff --git a/block.c b/block.c
357786
index 0516284..0af08ca 100644
357786
--- a/block.c
357786
+++ b/block.c
357786
@@ -3738,8 +3738,8 @@ exit:
357786
 /**
357786
  * Truncate file to 'offset' bytes (needed only for file protocols)
357786
  */
357786
-int bdrv_truncate(BdrvChild *child, int64_t offset, PreallocMode prealloc,
357786
-                  Error **errp)
357786
+int coroutine_fn bdrv_co_truncate(BdrvChild *child, int64_t offset,
357786
+                                  PreallocMode prealloc, Error **errp)
357786
 {
357786
     BlockDriverState *bs = child->bs;
357786
     BlockDriver *drv = bs->drv;
357786
@@ -3757,23 +3757,28 @@ int bdrv_truncate(BdrvChild *child, int64_t offset, PreallocMode prealloc,
357786
         return -EINVAL;
357786
     }
357786
 
357786
-    if (!drv->bdrv_truncate) {
357786
+    bdrv_inc_in_flight(bs);
357786
+
357786
+    if (!drv->bdrv_co_truncate) {
357786
         if (bs->file && drv->is_filter) {
357786
-            return bdrv_truncate(bs->file, offset, prealloc, errp);
357786
+            ret = bdrv_co_truncate(bs->file, offset, prealloc, errp);
357786
+            goto out;
357786
         }
357786
         error_setg(errp, "Image format driver does not support resize");
357786
-        return -ENOTSUP;
357786
+        ret = -ENOTSUP;
357786
+        goto out;
357786
     }
357786
     if (bs->read_only) {
357786
         error_setg(errp, "Image is read-only");
357786
-        return -EACCES;
357786
+        ret = -EACCES;
357786
+        goto out;
357786
     }
357786
 
357786
     assert(!(bs->open_flags & BDRV_O_INACTIVE));
357786
 
357786
-    ret = drv->bdrv_truncate(bs, offset, prealloc, errp);
357786
+    ret = drv->bdrv_co_truncate(bs, offset, prealloc, errp);
357786
     if (ret < 0) {
357786
-        return ret;
357786
+        goto out;
357786
     }
357786
     ret = refresh_total_sectors(bs, offset >> BDRV_SECTOR_BITS);
357786
     if (ret < 0) {
357786
@@ -3784,9 +3789,51 @@ int bdrv_truncate(BdrvChild *child, int64_t offset, PreallocMode prealloc,
357786
     bdrv_dirty_bitmap_truncate(bs, offset);
357786
     bdrv_parent_cb_resize(bs);
357786
     atomic_inc(&bs->write_gen);
357786
+
357786
+out:
357786
+    bdrv_dec_in_flight(bs);
357786
     return ret;
357786
 }
357786
 
357786
+typedef struct TruncateCo {
357786
+    BdrvChild *child;
357786
+    int64_t offset;
357786
+    PreallocMode prealloc;
357786
+    Error **errp;
357786
+    int ret;
357786
+} TruncateCo;
357786
+
357786
+static void coroutine_fn bdrv_truncate_co_entry(void *opaque)
357786
+{
357786
+    TruncateCo *tco = opaque;
357786
+    tco->ret = bdrv_co_truncate(tco->child, tco->offset, tco->prealloc,
357786
+                                tco->errp);
357786
+}
357786
+
357786
+int bdrv_truncate(BdrvChild *child, int64_t offset, PreallocMode prealloc,
357786
+                  Error **errp)
357786
+{
357786
+    Coroutine *co;
357786
+    TruncateCo tco = {
357786
+        .child      = child,
357786
+        .offset     = offset,
357786
+        .prealloc   = prealloc,
357786
+        .errp       = errp,
357786
+        .ret        = NOT_DONE,
357786
+    };
357786
+
357786
+    if (qemu_in_coroutine()) {
357786
+        /* Fast-path if already in coroutine context */
357786
+        bdrv_truncate_co_entry(&tco;;
357786
+    } else {
357786
+        co = qemu_coroutine_create(bdrv_truncate_co_entry, &tco;;
357786
+        qemu_coroutine_enter(co);
357786
+        BDRV_POLL_WHILE(child->bs, tco.ret == NOT_DONE);
357786
+    }
357786
+
357786
+    return tco.ret;
357786
+}
357786
+
357786
 /**
357786
  * Length of a allocated file in bytes. Sparse files are counted by actual
357786
  * allocated space. Return < 0 if error or unknown.
357786
diff --git a/block/copy-on-read.c b/block/copy-on-read.c
357786
index 6a97208..1dcdaee 100644
357786
--- a/block/copy-on-read.c
357786
+++ b/block/copy-on-read.c
357786
@@ -80,10 +80,10 @@ static int64_t cor_getlength(BlockDriverState *bs)
357786
 }
357786
 
357786
 
357786
-static int cor_truncate(BlockDriverState *bs, int64_t offset,
357786
-                        PreallocMode prealloc, Error **errp)
357786
+static int coroutine_fn cor_co_truncate(BlockDriverState *bs, int64_t offset,
357786
+                                        PreallocMode prealloc, Error **errp)
357786
 {
357786
-    return bdrv_truncate(bs->file, offset, prealloc, errp);
357786
+    return bdrv_co_truncate(bs->file, offset, prealloc, errp);
357786
 }
357786
 
357786
 
357786
@@ -147,7 +147,7 @@ BlockDriver bdrv_copy_on_read = {
357786
     .bdrv_child_perm                    = cor_child_perm,
357786
 
357786
     .bdrv_getlength                     = cor_getlength,
357786
-    .bdrv_truncate                      = cor_truncate,
357786
+    .bdrv_co_truncate                   = cor_co_truncate,
357786
 
357786
     .bdrv_co_preadv                     = cor_co_preadv,
357786
     .bdrv_co_pwritev                    = cor_co_pwritev,
357786
diff --git a/block/crypto.c b/block/crypto.c
357786
index f5151f4..02f04f3 100644
357786
--- a/block/crypto.c
357786
+++ b/block/crypto.c
357786
@@ -357,8 +357,9 @@ static int block_crypto_co_create_generic(BlockDriverState *bs,
357786
     return ret;
357786
 }
357786
 
357786
-static int block_crypto_truncate(BlockDriverState *bs, int64_t offset,
357786
-                                 PreallocMode prealloc, Error **errp)
357786
+static int coroutine_fn
357786
+block_crypto_co_truncate(BlockDriverState *bs, int64_t offset,
357786
+                         PreallocMode prealloc, Error **errp)
357786
 {
357786
     BlockCrypto *crypto = bs->opaque;
357786
     uint64_t payload_offset =
357786
@@ -371,7 +372,7 @@ static int block_crypto_truncate(BlockDriverState *bs, int64_t offset,
357786
 
357786
     offset += payload_offset;
357786
 
357786
-    return bdrv_truncate(bs->file, offset, prealloc, errp);
357786
+    return bdrv_co_truncate(bs->file, offset, prealloc, errp);
357786
 }
357786
 
357786
 static void block_crypto_close(BlockDriverState *bs)
357786
@@ -700,7 +701,7 @@ BlockDriver bdrv_crypto_luks = {
357786
     .bdrv_child_perm    = bdrv_format_default_perms,
357786
     .bdrv_co_create     = block_crypto_co_create_luks,
357786
     .bdrv_co_create_opts = block_crypto_co_create_opts_luks,
357786
-    .bdrv_truncate      = block_crypto_truncate,
357786
+    .bdrv_co_truncate   = block_crypto_co_truncate,
357786
     .create_opts        = &block_crypto_create_opts_luks,
357786
 
357786
     .bdrv_reopen_prepare = block_crypto_reopen_prepare,
357786
diff --git a/block/file-posix.c b/block/file-posix.c
357786
index cbf7c11..f8488ec 100644
357786
--- a/block/file-posix.c
357786
+++ b/block/file-posix.c
357786
@@ -1832,8 +1832,8 @@ out:
357786
     return result;
357786
 }
357786
 
357786
-static int raw_truncate(BlockDriverState *bs, int64_t offset,
357786
-                        PreallocMode prealloc, Error **errp)
357786
+static int coroutine_fn raw_co_truncate(BlockDriverState *bs, int64_t offset,
357786
+                                        PreallocMode prealloc, Error **errp)
357786
 {
357786
     BDRVRawState *s = bs->opaque;
357786
     struct stat st;
357786
@@ -2474,7 +2474,7 @@ BlockDriver bdrv_file = {
357786
     .bdrv_io_plug = raw_aio_plug,
357786
     .bdrv_io_unplug = raw_aio_unplug,
357786
 
357786
-    .bdrv_truncate = raw_truncate,
357786
+    .bdrv_co_truncate = raw_co_truncate,
357786
     .bdrv_getlength = raw_getlength,
357786
     .bdrv_get_info = raw_get_info,
357786
     .bdrv_get_allocated_file_size
357786
@@ -2953,7 +2953,7 @@ static BlockDriver bdrv_host_device = {
357786
     .bdrv_io_plug = raw_aio_plug,
357786
     .bdrv_io_unplug = raw_aio_unplug,
357786
 
357786
-    .bdrv_truncate      = raw_truncate,
357786
+    .bdrv_co_truncate       = raw_co_truncate,
357786
     .bdrv_getlength	= raw_getlength,
357786
     .bdrv_get_info = raw_get_info,
357786
     .bdrv_get_allocated_file_size
357786
@@ -3074,7 +3074,7 @@ static BlockDriver bdrv_host_cdrom = {
357786
     .bdrv_io_plug = raw_aio_plug,
357786
     .bdrv_io_unplug = raw_aio_unplug,
357786
 
357786
-    .bdrv_truncate      = raw_truncate,
357786
+    .bdrv_co_truncate    = raw_co_truncate,
357786
     .bdrv_getlength      = raw_getlength,
357786
     .has_variable_length = true,
357786
     .bdrv_get_allocated_file_size
357786
@@ -3204,7 +3204,7 @@ static BlockDriver bdrv_host_cdrom = {
357786
     .bdrv_io_plug = raw_aio_plug,
357786
     .bdrv_io_unplug = raw_aio_unplug,
357786
 
357786
-    .bdrv_truncate      = raw_truncate,
357786
+    .bdrv_co_truncate    = raw_co_truncate,
357786
     .bdrv_getlength      = raw_getlength,
357786
     .has_variable_length = true,
357786
     .bdrv_get_allocated_file_size
357786
diff --git a/block/file-win32.c b/block/file-win32.c
357786
index 2e2f746..6573338 100644
357786
--- a/block/file-win32.c
357786
+++ b/block/file-win32.c
357786
@@ -463,8 +463,8 @@ static void raw_close(BlockDriverState *bs)
357786
     }
357786
 }
357786
 
357786
-static int raw_truncate(BlockDriverState *bs, int64_t offset,
357786
-                        PreallocMode prealloc, Error **errp)
357786
+static int coroutine_fn raw_co_truncate(BlockDriverState *bs, int64_t offset,
357786
+                                        PreallocMode prealloc, Error **errp)
357786
 {
357786
     BDRVRawState *s = bs->opaque;
357786
     LONG low, high;
357786
@@ -636,7 +636,7 @@ BlockDriver bdrv_file = {
357786
     .bdrv_aio_writev    = raw_aio_writev,
357786
     .bdrv_aio_flush     = raw_aio_flush,
357786
 
357786
-    .bdrv_truncate	= raw_truncate,
357786
+    .bdrv_co_truncate   = raw_co_truncate,
357786
     .bdrv_getlength	= raw_getlength,
357786
     .bdrv_get_allocated_file_size
357786
                         = raw_get_allocated_file_size,
357786
diff --git a/block/gluster.c b/block/gluster.c
357786
index 418bb73..cecfe09 100644
357786
--- a/block/gluster.c
357786
+++ b/block/gluster.c
357786
@@ -1177,8 +1177,10 @@ static coroutine_fn int qemu_gluster_co_rw(BlockDriverState *bs,
357786
     return acb.ret;
357786
 }
357786
 
357786
-static int qemu_gluster_truncate(BlockDriverState *bs, int64_t offset,
357786
-                                 PreallocMode prealloc, Error **errp)
357786
+static coroutine_fn int qemu_gluster_co_truncate(BlockDriverState *bs,
357786
+                                                 int64_t offset,
357786
+                                                 PreallocMode prealloc,
357786
+                                                 Error **errp)
357786
 {
357786
     BDRVGlusterState *s = bs->opaque;
357786
     return qemu_gluster_do_truncate(s->fd, offset, prealloc, errp);
357786
@@ -1497,7 +1499,7 @@ static BlockDriver bdrv_gluster = {
357786
     .bdrv_co_create_opts          = qemu_gluster_co_create_opts,
357786
     .bdrv_getlength               = qemu_gluster_getlength,
357786
     .bdrv_get_allocated_file_size = qemu_gluster_allocated_file_size,
357786
-    .bdrv_truncate                = qemu_gluster_truncate,
357786
+    .bdrv_co_truncate             = qemu_gluster_co_truncate,
357786
     .bdrv_co_readv                = qemu_gluster_co_readv,
357786
     .bdrv_co_writev               = qemu_gluster_co_writev,
357786
     .bdrv_co_flush_to_disk        = qemu_gluster_co_flush_to_disk,
357786
@@ -1526,7 +1528,7 @@ static BlockDriver bdrv_gluster_tcp = {
357786
     .bdrv_co_create_opts          = qemu_gluster_co_create_opts,
357786
     .bdrv_getlength               = qemu_gluster_getlength,
357786
     .bdrv_get_allocated_file_size = qemu_gluster_allocated_file_size,
357786
-    .bdrv_truncate                = qemu_gluster_truncate,
357786
+    .bdrv_co_truncate             = qemu_gluster_co_truncate,
357786
     .bdrv_co_readv                = qemu_gluster_co_readv,
357786
     .bdrv_co_writev               = qemu_gluster_co_writev,
357786
     .bdrv_co_flush_to_disk        = qemu_gluster_co_flush_to_disk,
357786
@@ -1555,7 +1557,7 @@ static BlockDriver bdrv_gluster_unix = {
357786
     .bdrv_co_create_opts          = qemu_gluster_co_create_opts,
357786
     .bdrv_getlength               = qemu_gluster_getlength,
357786
     .bdrv_get_allocated_file_size = qemu_gluster_allocated_file_size,
357786
-    .bdrv_truncate                = qemu_gluster_truncate,
357786
+    .bdrv_co_truncate             = qemu_gluster_co_truncate,
357786
     .bdrv_co_readv                = qemu_gluster_co_readv,
357786
     .bdrv_co_writev               = qemu_gluster_co_writev,
357786
     .bdrv_co_flush_to_disk        = qemu_gluster_co_flush_to_disk,
357786
@@ -1590,7 +1592,7 @@ static BlockDriver bdrv_gluster_rdma = {
357786
     .bdrv_co_create_opts          = qemu_gluster_co_create_opts,
357786
     .bdrv_getlength               = qemu_gluster_getlength,
357786
     .bdrv_get_allocated_file_size = qemu_gluster_allocated_file_size,
357786
-    .bdrv_truncate                = qemu_gluster_truncate,
357786
+    .bdrv_co_truncate             = qemu_gluster_co_truncate,
357786
     .bdrv_co_readv                = qemu_gluster_co_readv,
357786
     .bdrv_co_writev               = qemu_gluster_co_writev,
357786
     .bdrv_co_flush_to_disk        = qemu_gluster_co_flush_to_disk,
357786
diff --git a/block/iscsi.c b/block/iscsi.c
357786
index 751884d..5047e83 100644
357786
--- a/block/iscsi.c
357786
+++ b/block/iscsi.c
357786
@@ -2085,8 +2085,8 @@ static void iscsi_reopen_commit(BDRVReopenState *reopen_state)
357786
     }
357786
 }
357786
 
357786
-static int iscsi_truncate(BlockDriverState *bs, int64_t offset,
357786
-                          PreallocMode prealloc, Error **errp)
357786
+static int coroutine_fn iscsi_co_truncate(BlockDriverState *bs, int64_t offset,
357786
+                                          PreallocMode prealloc, Error **errp)
357786
 {
357786
     IscsiLun *iscsilun = bs->opaque;
357786
     Error *local_err = NULL;
357786
@@ -2431,7 +2431,7 @@ static BlockDriver bdrv_iscsi = {
357786
 
357786
     .bdrv_getlength  = iscsi_getlength,
357786
     .bdrv_get_info   = iscsi_get_info,
357786
-    .bdrv_truncate   = iscsi_truncate,
357786
+    .bdrv_co_truncate    = iscsi_co_truncate,
357786
     .bdrv_refresh_limits = iscsi_refresh_limits,
357786
 
357786
     .bdrv_co_block_status  = iscsi_co_block_status,
357786
@@ -2468,7 +2468,7 @@ static BlockDriver bdrv_iser = {
357786
 
357786
     .bdrv_getlength  = iscsi_getlength,
357786
     .bdrv_get_info   = iscsi_get_info,
357786
-    .bdrv_truncate   = iscsi_truncate,
357786
+    .bdrv_co_truncate    = iscsi_co_truncate,
357786
     .bdrv_refresh_limits = iscsi_refresh_limits,
357786
 
357786
     .bdrv_co_block_status  = iscsi_co_block_status,
357786
diff --git a/block/nfs.c b/block/nfs.c
357786
index 743ca04..eab1a2c 100644
357786
--- a/block/nfs.c
357786
+++ b/block/nfs.c
357786
@@ -743,8 +743,9 @@ static int64_t nfs_get_allocated_file_size(BlockDriverState *bs)
357786
     return (task.ret < 0 ? task.ret : st.st_blocks * 512);
357786
 }
357786
 
357786
-static int nfs_file_truncate(BlockDriverState *bs, int64_t offset,
357786
-                             PreallocMode prealloc, Error **errp)
357786
+static int coroutine_fn
357786
+nfs_file_co_truncate(BlockDriverState *bs, int64_t offset,
357786
+                     PreallocMode prealloc, Error **errp)
357786
 {
357786
     NFSClient *client = bs->opaque;
357786
     int ret;
357786
@@ -873,7 +874,7 @@ static BlockDriver bdrv_nfs = {
357786
 
357786
     .bdrv_has_zero_init             = nfs_has_zero_init,
357786
     .bdrv_get_allocated_file_size   = nfs_get_allocated_file_size,
357786
-    .bdrv_truncate                  = nfs_file_truncate,
357786
+    .bdrv_co_truncate               = nfs_file_co_truncate,
357786
 
357786
     .bdrv_file_open                 = nfs_file_open,
357786
     .bdrv_close                     = nfs_file_close,
357786
diff --git a/block/qcow2.c b/block/qcow2.c
357786
index dbd448c..c5c6ae9 100644
357786
--- a/block/qcow2.c
357786
+++ b/block/qcow2.c
357786
@@ -2539,15 +2539,12 @@ static void coroutine_fn preallocate_co(void *opaque)
357786
     BlockDriverState *bs = params->bs;
357786
     uint64_t offset = params->offset;
357786
     uint64_t new_length = params->new_length;
357786
-    BDRVQcow2State *s = bs->opaque;
357786
     uint64_t bytes;
357786
     uint64_t host_offset = 0;
357786
     unsigned int cur_bytes;
357786
     int ret;
357786
     QCowL2Meta *meta;
357786
 
357786
-    qemu_co_mutex_lock(&s->lock);
357786
-
357786
     assert(offset <= new_length);
357786
     bytes = new_length - offset;
357786
 
357786
@@ -2600,7 +2597,6 @@ static void coroutine_fn preallocate_co(void *opaque)
357786
     ret = 0;
357786
 
357786
 done:
357786
-    qemu_co_mutex_unlock(&s->lock);
357786
     params->ret = ret;
357786
 }
357786
 
357786
@@ -3037,7 +3033,11 @@ qcow2_co_create(BlockdevCreateOptions *create_options, Error **errp)
357786
 
357786
     /* And if we're supposed to preallocate metadata, do that now */
357786
     if (qcow2_opts->preallocation != PREALLOC_MODE_OFF) {
357786
+        BDRVQcow2State *s = blk_bs(blk)->opaque;
357786
+        qemu_co_mutex_lock(&s->lock);
357786
         ret = preallocate(blk_bs(blk), 0, qcow2_opts->size);
357786
+        qemu_co_mutex_unlock(&s->lock);
357786
+
357786
         if (ret < 0) {
357786
             error_setg_errno(errp, -ret, "Could not preallocate metadata");
357786
             goto out;
357786
@@ -3434,8 +3434,8 @@ fail:
357786
     return ret;
357786
 }
357786
 
357786
-static int qcow2_truncate(BlockDriverState *bs, int64_t offset,
357786
-                          PreallocMode prealloc, Error **errp)
357786
+static int coroutine_fn qcow2_co_truncate(BlockDriverState *bs, int64_t offset,
357786
+                                          PreallocMode prealloc, Error **errp)
357786
 {
357786
     BDRVQcow2State *s = bs->opaque;
357786
     uint64_t old_length;
357786
@@ -3455,17 +3455,21 @@ static int qcow2_truncate(BlockDriverState *bs, int64_t offset,
357786
         return -EINVAL;
357786
     }
357786
 
357786
+    qemu_co_mutex_lock(&s->lock);
357786
+
357786
     /* cannot proceed if image has snapshots */
357786
     if (s->nb_snapshots) {
357786
         error_setg(errp, "Can't resize an image which has snapshots");
357786
-        return -ENOTSUP;
357786
+        ret = -ENOTSUP;
357786
+        goto fail;
357786
     }
357786
 
357786
     /* cannot proceed if image has bitmaps */
357786
     if (s->nb_bitmaps) {
357786
         /* TODO: resize bitmaps in the image */
357786
         error_setg(errp, "Can't resize an image which has bitmaps");
357786
-        return -ENOTSUP;
357786
+        ret = -ENOTSUP;
357786
+        goto fail;
357786
     }
357786
 
357786
     old_length = bs->total_sectors * 512;
357786
@@ -3476,7 +3480,8 @@ static int qcow2_truncate(BlockDriverState *bs, int64_t offset,
357786
         if (prealloc != PREALLOC_MODE_OFF) {
357786
             error_setg(errp,
357786
                        "Preallocation can't be used for shrinking an image");
357786
-            return -EINVAL;
357786
+            ret = -EINVAL;
357786
+            goto fail;
357786
         }
357786
 
357786
         ret = qcow2_cluster_discard(bs, ROUND_UP(offset, s->cluster_size),
357786
@@ -3485,40 +3490,42 @@ static int qcow2_truncate(BlockDriverState *bs, int64_t offset,
357786
                                     QCOW2_DISCARD_ALWAYS, true);
357786
         if (ret < 0) {
357786
             error_setg_errno(errp, -ret, "Failed to discard cropped clusters");
357786
-            return ret;
357786
+            goto fail;
357786
         }
357786
 
357786
         ret = qcow2_shrink_l1_table(bs, new_l1_size);
357786
         if (ret < 0) {
357786
             error_setg_errno(errp, -ret,
357786
                              "Failed to reduce the number of L2 tables");
357786
-            return ret;
357786
+            goto fail;
357786
         }
357786
 
357786
         ret = qcow2_shrink_reftable(bs);
357786
         if (ret < 0) {
357786
             error_setg_errno(errp, -ret,
357786
                              "Failed to discard unused refblocks");
357786
-            return ret;
357786
+            goto fail;
357786
         }
357786
 
357786
         old_file_size = bdrv_getlength(bs->file->bs);
357786
         if (old_file_size < 0) {
357786
             error_setg_errno(errp, -old_file_size,
357786
                              "Failed to inquire current file length");
357786
-            return old_file_size;
357786
+            ret = old_file_size;
357786
+            goto fail;
357786
         }
357786
         last_cluster = qcow2_get_last_cluster(bs, old_file_size);
357786
         if (last_cluster < 0) {
357786
             error_setg_errno(errp, -last_cluster,
357786
                              "Failed to find the last cluster");
357786
-            return last_cluster;
357786
+            ret = last_cluster;
357786
+            goto fail;
357786
         }
357786
         if ((last_cluster + 1) * s->cluster_size < old_file_size) {
357786
             Error *local_err = NULL;
357786
 
357786
-            bdrv_truncate(bs->file, (last_cluster + 1) * s->cluster_size,
357786
-                          PREALLOC_MODE_OFF, &local_err);
357786
+            bdrv_co_truncate(bs->file, (last_cluster + 1) * s->cluster_size,
357786
+                             PREALLOC_MODE_OFF, &local_err);
357786
             if (local_err) {
357786
                 warn_reportf_err(local_err,
357786
                                  "Failed to truncate the tail of the image: ");
357786
@@ -3528,7 +3535,7 @@ static int qcow2_truncate(BlockDriverState *bs, int64_t offset,
357786
         ret = qcow2_grow_l1_table(bs, new_l1_size, true);
357786
         if (ret < 0) {
357786
             error_setg_errno(errp, -ret, "Failed to grow the L1 table");
357786
-            return ret;
357786
+            goto fail;
357786
         }
357786
     }
357786
 
357786
@@ -3540,7 +3547,7 @@ static int qcow2_truncate(BlockDriverState *bs, int64_t offset,
357786
         ret = preallocate(bs, old_length, offset);
357786
         if (ret < 0) {
357786
             error_setg_errno(errp, -ret, "Preallocation failed");
357786
-            return ret;
357786
+            goto fail;
357786
         }
357786
         break;
357786
 
357786
@@ -3556,7 +3563,8 @@ static int qcow2_truncate(BlockDriverState *bs, int64_t offset,
357786
         if (old_file_size < 0) {
357786
             error_setg_errno(errp, -old_file_size,
357786
                              "Failed to inquire current file length");
357786
-            return old_file_size;
357786
+            ret = old_file_size;
357786
+            goto fail;
357786
         }
357786
         old_file_size = ROUND_UP(old_file_size, s->cluster_size);
357786
 
357786
@@ -3586,7 +3594,8 @@ static int qcow2_truncate(BlockDriverState *bs, int64_t offset,
357786
         if (allocation_start < 0) {
357786
             error_setg_errno(errp, -allocation_start,
357786
                              "Failed to resize refcount structures");
357786
-            return allocation_start;
357786
+            ret = allocation_start;
357786
+            goto fail;
357786
         }
357786
 
357786
         clusters_allocated = qcow2_alloc_clusters_at(bs, allocation_start,
357786
@@ -3594,7 +3603,8 @@ static int qcow2_truncate(BlockDriverState *bs, int64_t offset,
357786
         if (clusters_allocated < 0) {
357786
             error_setg_errno(errp, -clusters_allocated,
357786
                              "Failed to allocate data clusters");
357786
-            return clusters_allocated;
357786
+            ret = clusters_allocated;
357786
+            goto fail;
357786
         }
357786
 
357786
         assert(clusters_allocated == nb_new_data_clusters);
357786
@@ -3602,13 +3612,13 @@ static int qcow2_truncate(BlockDriverState *bs, int64_t offset,
357786
         /* Allocate the data area */
357786
         new_file_size = allocation_start +
357786
                         nb_new_data_clusters * s->cluster_size;
357786
-        ret = bdrv_truncate(bs->file, new_file_size, prealloc, errp);
357786
+        ret = bdrv_co_truncate(bs->file, new_file_size, prealloc, errp);
357786
         if (ret < 0) {
357786
             error_prepend(errp, "Failed to resize underlying file: ");
357786
             qcow2_free_clusters(bs, allocation_start,
357786
                                 nb_new_data_clusters * s->cluster_size,
357786
                                 QCOW2_DISCARD_OTHER);
357786
-            return ret;
357786
+            goto fail;
357786
         }
357786
 
357786
         /* Create the necessary L2 entries */
357786
@@ -3631,7 +3641,7 @@ static int qcow2_truncate(BlockDriverState *bs, int64_t offset,
357786
                 qcow2_free_clusters(bs, host_offset,
357786
                                     nb_new_data_clusters * s->cluster_size,
357786
                                     QCOW2_DISCARD_OTHER);
357786
-                return ret;
357786
+                goto fail;
357786
             }
357786
 
357786
             guest_offset += nb_clusters * s->cluster_size;
357786
@@ -3647,11 +3657,11 @@ static int qcow2_truncate(BlockDriverState *bs, int64_t offset,
357786
 
357786
     if (prealloc != PREALLOC_MODE_OFF) {
357786
         /* Flush metadata before actually changing the image size */
357786
-        ret = bdrv_flush(bs);
357786
+        ret = qcow2_write_caches(bs);
357786
         if (ret < 0) {
357786
             error_setg_errno(errp, -ret,
357786
                              "Failed to flush the preallocated area to disk");
357786
-            return ret;
357786
+            goto fail;
357786
         }
357786
     }
357786
 
357786
@@ -3661,11 +3671,14 @@ static int qcow2_truncate(BlockDriverState *bs, int64_t offset,
357786
                            &offset, sizeof(uint64_t));
357786
     if (ret < 0) {
357786
         error_setg_errno(errp, -ret, "Failed to update the image size");
357786
-        return ret;
357786
+        goto fail;
357786
     }
357786
 
357786
     s->l1_vm_state_index = new_l1_size;
357786
-    return 0;
357786
+    ret = 0;
357786
+fail:
357786
+    qemu_co_mutex_unlock(&s->lock);
357786
+    return ret;
357786
 }
357786
 
357786
 /* XXX: put compressed sectors first, then all the cluster aligned
357786
@@ -3689,7 +3702,8 @@ qcow2_co_pwritev_compressed(BlockDriverState *bs, uint64_t offset,
357786
         if (cluster_offset < 0) {
357786
             return cluster_offset;
357786
         }
357786
-        return bdrv_truncate(bs->file, cluster_offset, PREALLOC_MODE_OFF, NULL);
357786
+        return bdrv_co_truncate(bs->file, cluster_offset, PREALLOC_MODE_OFF,
357786
+                                NULL);
357786
     }
357786
 
357786
     if (offset_into_cluster(s, offset)) {
357786
@@ -4694,7 +4708,7 @@ BlockDriver bdrv_qcow2 = {
357786
     .bdrv_co_pdiscard       = qcow2_co_pdiscard,
357786
     .bdrv_co_copy_range_from = qcow2_co_copy_range_from,
357786
     .bdrv_co_copy_range_to  = qcow2_co_copy_range_to,
357786
-    .bdrv_truncate          = qcow2_truncate,
357786
+    .bdrv_co_truncate       = qcow2_co_truncate,
357786
     .bdrv_co_pwritev_compressed = qcow2_co_pwritev_compressed,
357786
     .bdrv_make_empty        = qcow2_make_empty,
357786
 
357786
diff --git a/block/qed.c b/block/qed.c
357786
index 4c5d7e8..34cccea 100644
357786
--- a/block/qed.c
357786
+++ b/block/qed.c
357786
@@ -1466,8 +1466,10 @@ static int coroutine_fn bdrv_qed_co_pwrite_zeroes(BlockDriverState *bs,
357786
                           QED_AIOCB_WRITE | QED_AIOCB_ZERO);
357786
 }
357786
 
357786
-static int bdrv_qed_truncate(BlockDriverState *bs, int64_t offset,
357786
-                             PreallocMode prealloc, Error **errp)
357786
+static int coroutine_fn bdrv_qed_co_truncate(BlockDriverState *bs,
357786
+                                             int64_t offset,
357786
+                                             PreallocMode prealloc,
357786
+                                             Error **errp)
357786
 {
357786
     BDRVQEDState *s = bs->opaque;
357786
     uint64_t old_image_size;
357786
@@ -1677,7 +1679,7 @@ static BlockDriver bdrv_qed = {
357786
     .bdrv_co_readv            = bdrv_qed_co_readv,
357786
     .bdrv_co_writev           = bdrv_qed_co_writev,
357786
     .bdrv_co_pwrite_zeroes    = bdrv_qed_co_pwrite_zeroes,
357786
-    .bdrv_truncate            = bdrv_qed_truncate,
357786
+    .bdrv_co_truncate         = bdrv_qed_co_truncate,
357786
     .bdrv_getlength           = bdrv_qed_getlength,
357786
     .bdrv_get_info            = bdrv_qed_get_info,
357786
     .bdrv_refresh_limits      = bdrv_qed_refresh_limits,
357786
diff --git a/block/raw-format.c b/block/raw-format.c
357786
index f2e468d..b78da56 100644
357786
--- a/block/raw-format.c
357786
+++ b/block/raw-format.c
357786
@@ -366,8 +366,8 @@ static void raw_refresh_limits(BlockDriverState *bs, Error **errp)
357786
     }
357786
 }
357786
 
357786
-static int raw_truncate(BlockDriverState *bs, int64_t offset,
357786
-                        PreallocMode prealloc, Error **errp)
357786
+static int coroutine_fn raw_co_truncate(BlockDriverState *bs, int64_t offset,
357786
+                                        PreallocMode prealloc, Error **errp)
357786
 {
357786
     BDRVRawState *s = bs->opaque;
357786
 
357786
@@ -383,7 +383,7 @@ static int raw_truncate(BlockDriverState *bs, int64_t offset,
357786
 
357786
     s->size = offset;
357786
     offset += s->offset;
357786
-    return bdrv_truncate(bs->file, offset, prealloc, errp);
357786
+    return bdrv_co_truncate(bs->file, offset, prealloc, errp);
357786
 }
357786
 
357786
 static void raw_eject(BlockDriverState *bs, bool eject_flag)
357786
@@ -545,7 +545,7 @@ BlockDriver bdrv_raw = {
357786
     .bdrv_co_block_status = &raw_co_block_status,
357786
     .bdrv_co_copy_range_from = &raw_co_copy_range_from,
357786
     .bdrv_co_copy_range_to  = &raw_co_copy_range_to,
357786
-    .bdrv_truncate        = &raw_truncate,
357786
+    .bdrv_co_truncate     = &raw_co_truncate,
357786
     .bdrv_getlength       = &raw_getlength,
357786
     .has_variable_length  = true,
357786
     .bdrv_measure         = &raw_measure,
357786
diff --git a/block/rbd.c b/block/rbd.c
357786
index 3242bcd..b93046b 100644
357786
--- a/block/rbd.c
357786
+++ b/block/rbd.c
357786
@@ -987,8 +987,10 @@ static int64_t qemu_rbd_getlength(BlockDriverState *bs)
357786
     return info.size;
357786
 }
357786
 
357786
-static int qemu_rbd_truncate(BlockDriverState *bs, int64_t offset,
357786
-                             PreallocMode prealloc, Error **errp)
357786
+static int coroutine_fn qemu_rbd_co_truncate(BlockDriverState *bs,
357786
+                                             int64_t offset,
357786
+                                             PreallocMode prealloc,
357786
+                                             Error **errp)
357786
 {
357786
     BDRVRBDState *s = bs->opaque;
357786
     int r;
357786
@@ -1180,7 +1182,7 @@ static BlockDriver bdrv_rbd = {
357786
     .bdrv_get_info          = qemu_rbd_getinfo,
357786
     .create_opts            = &qemu_rbd_create_opts,
357786
     .bdrv_getlength         = qemu_rbd_getlength,
357786
-    .bdrv_truncate          = qemu_rbd_truncate,
357786
+    .bdrv_co_truncate       = qemu_rbd_co_truncate,
357786
     .protocol_name          = "rbd",
357786
 
357786
     .bdrv_aio_readv         = qemu_rbd_aio_readv,
357786
diff --git a/block/sheepdog.c b/block/sheepdog.c
357786
index 933880c..14165d3 100644
357786
--- a/block/sheepdog.c
357786
+++ b/block/sheepdog.c
357786
@@ -2294,8 +2294,8 @@ static int64_t sd_getlength(BlockDriverState *bs)
357786
     return s->inode.vdi_size;
357786
 }
357786
 
357786
-static int sd_truncate(BlockDriverState *bs, int64_t offset,
357786
-                       PreallocMode prealloc, Error **errp)
357786
+static int coroutine_fn sd_co_truncate(BlockDriverState *bs, int64_t offset,
357786
+                                       PreallocMode prealloc, Error **errp)
357786
 {
357786
     BDRVSheepdogState *s = bs->opaque;
357786
     int ret, fd;
357786
@@ -2609,7 +2609,7 @@ static coroutine_fn int sd_co_writev(BlockDriverState *bs, int64_t sector_num,
357786
     BDRVSheepdogState *s = bs->opaque;
357786
 
357786
     if (offset > s->inode.vdi_size) {
357786
-        ret = sd_truncate(bs, offset, PREALLOC_MODE_OFF, NULL);
357786
+        ret = sd_co_truncate(bs, offset, PREALLOC_MODE_OFF, NULL);
357786
         if (ret < 0) {
357786
             return ret;
357786
         }
357786
@@ -3229,7 +3229,7 @@ static BlockDriver bdrv_sheepdog = {
357786
     .bdrv_has_zero_init           = bdrv_has_zero_init_1,
357786
     .bdrv_getlength               = sd_getlength,
357786
     .bdrv_get_allocated_file_size = sd_get_allocated_file_size,
357786
-    .bdrv_truncate                = sd_truncate,
357786
+    .bdrv_co_truncate             = sd_co_truncate,
357786
 
357786
     .bdrv_co_readv                = sd_co_readv,
357786
     .bdrv_co_writev               = sd_co_writev,
357786
@@ -3266,7 +3266,7 @@ static BlockDriver bdrv_sheepdog_tcp = {
357786
     .bdrv_has_zero_init           = bdrv_has_zero_init_1,
357786
     .bdrv_getlength               = sd_getlength,
357786
     .bdrv_get_allocated_file_size = sd_get_allocated_file_size,
357786
-    .bdrv_truncate                = sd_truncate,
357786
+    .bdrv_co_truncate             = sd_co_truncate,
357786
 
357786
     .bdrv_co_readv                = sd_co_readv,
357786
     .bdrv_co_writev               = sd_co_writev,
357786
@@ -3303,7 +3303,7 @@ static BlockDriver bdrv_sheepdog_unix = {
357786
     .bdrv_has_zero_init           = bdrv_has_zero_init_1,
357786
     .bdrv_getlength               = sd_getlength,
357786
     .bdrv_get_allocated_file_size = sd_get_allocated_file_size,
357786
-    .bdrv_truncate                = sd_truncate,
357786
+    .bdrv_co_truncate             = sd_co_truncate,
357786
 
357786
     .bdrv_co_readv                = sd_co_readv,
357786
     .bdrv_co_writev               = sd_co_writev,
357786
diff --git a/block/ssh.c b/block/ssh.c
357786
index aab6996..6a55d82 100644
357786
--- a/block/ssh.c
357786
+++ b/block/ssh.c
357786
@@ -1241,8 +1241,8 @@ static int64_t ssh_getlength(BlockDriverState *bs)
357786
     return length;
357786
 }
357786
 
357786
-static int ssh_truncate(BlockDriverState *bs, int64_t offset,
357786
-                        PreallocMode prealloc, Error **errp)
357786
+static int coroutine_fn ssh_co_truncate(BlockDriverState *bs, int64_t offset,
357786
+                                        PreallocMode prealloc, Error **errp)
357786
 {
357786
     BDRVSSHState *s = bs->opaque;
357786
 
357786
@@ -1277,7 +1277,7 @@ static BlockDriver bdrv_ssh = {
357786
     .bdrv_co_readv                = ssh_co_readv,
357786
     .bdrv_co_writev               = ssh_co_writev,
357786
     .bdrv_getlength               = ssh_getlength,
357786
-    .bdrv_truncate                = ssh_truncate,
357786
+    .bdrv_co_truncate             = ssh_co_truncate,
357786
     .bdrv_co_flush_to_disk        = ssh_co_flush,
357786
     .create_opts                  = &ssh_create_opts,
357786
 };
357786
diff --git a/include/block/block.h b/include/block/block.h
357786
index e677080..c3cfb40 100644
357786
--- a/include/block/block.h
357786
+++ b/include/block/block.h
357786
@@ -300,8 +300,12 @@ int coroutine_fn bdrv_co_pwrite_zeroes(BdrvChild *child, int64_t offset,
357786
 BlockDriverState *bdrv_find_backing_image(BlockDriverState *bs,
357786
     const char *backing_file);
357786
 void bdrv_refresh_filename(BlockDriverState *bs);
357786
+
357786
+int coroutine_fn bdrv_co_truncate(BdrvChild *child, int64_t offset,
357786
+                                  PreallocMode prealloc, Error **errp);
357786
 int bdrv_truncate(BdrvChild *child, int64_t offset, PreallocMode prealloc,
357786
                   Error **errp);
357786
+
357786
 int64_t bdrv_nb_sectors(BlockDriverState *bs);
357786
 int64_t bdrv_getlength(BlockDriverState *bs);
357786
 int64_t bdrv_get_allocated_file_size(BlockDriverState *bs);
357786
diff --git a/include/block/block_int.h b/include/block/block_int.h
357786
index 3da86a7..46b2f87 100644
357786
--- a/include/block/block_int.h
357786
+++ b/include/block/block_int.h
357786
@@ -291,8 +291,8 @@ struct BlockDriver {
357786
      * bdrv_parse_filename.
357786
      */
357786
     const char *protocol_name;
357786
-    int (*bdrv_truncate)(BlockDriverState *bs, int64_t offset,
357786
-                         PreallocMode prealloc, Error **errp);
357786
+    int coroutine_fn (*bdrv_co_truncate)(BlockDriverState *bs, int64_t offset,
357786
+                                         PreallocMode prealloc, Error **errp);
357786
 
357786
     int64_t (*bdrv_getlength)(BlockDriverState *bs);
357786
     bool has_variable_length;
357786
-- 
357786
1.8.3.1
357786