26ba25
From a4029c787c5741de154d9d1ede7e70c7deaba416 Mon Sep 17 00:00:00 2001
26ba25
From: Max Reitz <mreitz@redhat.com>
26ba25
Date: Wed, 16 May 2018 12:00:18 +0200
26ba25
Subject: [PATCH 003/268] qemu-img: Check post-truncation size
26ba25
26ba25
RH-Author: Max Reitz <mreitz@redhat.com>
26ba25
Message-id: <20180516120018.27565-2-mreitz@redhat.com>
26ba25
Patchwork-id: 80281
26ba25
O-Subject: [RHV-7.6 qemu-kvm-rhev PATCH 1/1] qemu-img: Check post-truncation size
26ba25
Bugzilla: 1523065
26ba25
RH-Acked-by: John Snow <jsnow@redhat.com>
26ba25
RH-Acked-by: Stefan Hajnoczi <stefanha@redhat.com>
26ba25
RH-Acked-by: Kevin Wolf <kwolf@redhat.com>
26ba25
26ba25
Some block drivers (iscsi and file-posix when dealing with device files)
26ba25
do not actually support truncation, even though they provide a
26ba25
.bdrv_truncate() method and will happily return success when providing a
26ba25
new size that does not exceed the current size.  This is because these
26ba25
drivers expect the user to resize the image outside of qemu and then
26ba25
provide qemu with that information through the block_resize command
26ba25
(compare cb1b83e740384b4e0d950f3d7c81c02b8ce86c2e).
26ba25
26ba25
Of course, anyone using qemu-img resize will find that behavior useless.
26ba25
So we should check the actual size of the image after the supposedly
26ba25
successful truncation took place, emit an error if nothing changed and
26ba25
emit a warning if the target size was not met.
26ba25
26ba25
Signed-off-by: Max Reitz <mreitz@redhat.com>
26ba25
Message-id: 20180421163957.29872-1-mreitz@redhat.com
26ba25
Reviewed-by: Stefan Hajnoczi <stefanha@redhat.com>
26ba25
Signed-off-by: Max Reitz <mreitz@redhat.com>
26ba25
(cherry picked from commit 5279b30392da7a3248b320c75f20c61e3a95863c)
26ba25
Signed-off-by: Max Reitz <mreitz@redhat.com>
26ba25
Signed-off-by: Miroslav Rezanina <mrezanin@redhat.com>
26ba25
---
26ba25
 qemu-img.c | 39 +++++++++++++++++++++++++++++++++++----
26ba25
 1 file changed, 35 insertions(+), 4 deletions(-)
26ba25
26ba25
diff --git a/qemu-img.c b/qemu-img.c
26ba25
index 855fa52..8320887 100644
26ba25
--- a/qemu-img.c
26ba25
+++ b/qemu-img.c
26ba25
@@ -3381,7 +3381,7 @@ static int img_resize(int argc, char **argv)
26ba25
     Error *err = NULL;
26ba25
     int c, ret, relative;
26ba25
     const char *filename, *fmt, *size;
26ba25
-    int64_t n, total_size, current_size;
26ba25
+    int64_t n, total_size, current_size, new_size;
26ba25
     bool quiet = false;
26ba25
     BlockBackend *blk = NULL;
26ba25
     PreallocMode prealloc = PREALLOC_MODE_OFF;
26ba25
@@ -3557,11 +3557,42 @@ static int img_resize(int argc, char **argv)
26ba25
     }
26ba25
 
26ba25
     ret = blk_truncate(blk, total_size, prealloc, &err;;
26ba25
-    if (!ret) {
26ba25
-        qprintf(quiet, "Image resized.\n");
26ba25
-    } else {
26ba25
+    if (ret < 0) {
26ba25
         error_report_err(err);
26ba25
+        goto out;
26ba25
+    }
26ba25
+
26ba25
+    new_size = blk_getlength(blk);
26ba25
+    if (new_size < 0) {
26ba25
+        error_report("Failed to verify truncated image length: %s",
26ba25
+                     strerror(-new_size));
26ba25
+        ret = -1;
26ba25
+        goto out;
26ba25
     }
26ba25
+
26ba25
+    /* Some block drivers implement a truncation method, but only so
26ba25
+     * the user can cause qemu to refresh the image's size from disk.
26ba25
+     * The idea is that the user resizes the image outside of qemu and
26ba25
+     * then invokes block_resize to inform qemu about it.
26ba25
+     * (This includes iscsi and file-posix for device files.)
26ba25
+     * Of course, that is not the behavior someone invoking
26ba25
+     * qemu-img resize would find useful, so we catch that behavior
26ba25
+     * here and tell the user. */
26ba25
+    if (new_size != total_size && new_size == current_size) {
26ba25
+        error_report("Image was not resized; resizing may not be supported "
26ba25
+                     "for this image");
26ba25
+        ret = -1;
26ba25
+        goto out;
26ba25
+    }
26ba25
+
26ba25
+    if (new_size != total_size) {
26ba25
+        warn_report("Image should have been resized to %" PRIi64
26ba25
+                    " bytes, but was resized to %" PRIi64 " bytes",
26ba25
+                    total_size, new_size);
26ba25
+    }
26ba25
+
26ba25
+    qprintf(quiet, "Image resized.\n");
26ba25
+
26ba25
 out:
26ba25
     blk_unref(blk);
26ba25
     if (ret) {
26ba25
-- 
26ba25
1.8.3.1
26ba25