76daa3
From 02e6c183afd2203aaf1910a0150e54cf1c8efa3e Mon Sep 17 00:00:00 2001
76daa3
From: Kevin Wolf <kwolf@redhat.com>
76daa3
Date: Fri, 26 May 2017 15:28:54 +0200
76daa3
Subject: [PATCH 5/6] qemu-img: wait for convert coroutines to complete
76daa3
76daa3
RH-Author: Kevin Wolf <kwolf@redhat.com>
76daa3
Message-id: <1495812534-25595-2-git-send-email-kwolf@redhat.com>
76daa3
Patchwork-id: 75425
76daa3
O-Subject: [RHEL-7.4 qemu-kvm-rhev PATCH] qemu-img: wait for convert coroutines to complete
76daa3
Bugzilla: 1451849
76daa3
RH-Acked-by: Max Reitz <mreitz@redhat.com>
76daa3
RH-Acked-by: Jeffrey Cody <jcody@redhat.com>
76daa3
RH-Acked-by: Paolo Bonzini <pbonzini@redhat.com>
76daa3
76daa3
From: Anton Nefedov <anton.nefedov@virtuozzo.com>
76daa3
76daa3
On error path (like i/o error in one of the coroutines), it's required to
76daa3
  - wait for coroutines completion before cleaning the common structures
76daa3
  - reenter dependent coroutines so they ever finish
76daa3
76daa3
Introduced in 2d9187bc65.
76daa3
76daa3
Cc: qemu-stable@nongnu.org
76daa3
Signed-off-by: Anton Nefedov <anton.nefedov@virtuozzo.com>
76daa3
Reviewed-by: Peter Lieven <pl@kamp.de>
76daa3
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
76daa3
(cherry picked from commit b91127edd0ff96f27f1e58e47f4e9f9d6a0fed02)
76daa3
Signed-off-by: Miroslav Rezanina <mrezanin@redhat.com>
76daa3
---
76daa3
 qemu-img.c | 26 +++++++++++---------------
76daa3
 1 file changed, 11 insertions(+), 15 deletions(-)
76daa3
76daa3
diff --git a/qemu-img.c b/qemu-img.c
76daa3
index 37c2894..53d0bf1 100644
76daa3
--- a/qemu-img.c
76daa3
+++ b/qemu-img.c
76daa3
@@ -1761,13 +1761,13 @@ static void coroutine_fn convert_co_do_copy(void *opaque)
76daa3
         qemu_co_mutex_lock(&s->lock);
76daa3
         if (s->ret != -EINPROGRESS || s->sector_num >= s->total_sectors) {
76daa3
             qemu_co_mutex_unlock(&s->lock);
76daa3
-            goto out;
76daa3
+            break;
76daa3
         }
76daa3
         n = convert_iteration_sectors(s, s->sector_num);
76daa3
         if (n < 0) {
76daa3
             qemu_co_mutex_unlock(&s->lock);
76daa3
             s->ret = n;
76daa3
-            goto out;
76daa3
+            break;
76daa3
         }
76daa3
         /* save current sector and allocation status to local variables */
76daa3
         sector_num = s->sector_num;
76daa3
@@ -1792,7 +1792,6 @@ static void coroutine_fn convert_co_do_copy(void *opaque)
76daa3
                 error_report("error while reading sector %" PRId64
76daa3
                              ": %s", sector_num, strerror(-ret));
76daa3
                 s->ret = ret;
76daa3
-                goto out;
76daa3
             }
76daa3
         } else if (!s->min_sparse && status == BLK_ZERO) {
76daa3
             status = BLK_DATA;
76daa3
@@ -1801,22 +1800,20 @@ static void coroutine_fn convert_co_do_copy(void *opaque)
76daa3
 
76daa3
         if (s->wr_in_order) {
76daa3
             /* keep writes in order */
76daa3
-            while (s->wr_offs != sector_num) {
76daa3
-                if (s->ret != -EINPROGRESS) {
76daa3
-                    goto out;
76daa3
-                }
76daa3
+            while (s->wr_offs != sector_num && s->ret == -EINPROGRESS) {
76daa3
                 s->wait_sector_num[index] = sector_num;
76daa3
                 qemu_coroutine_yield();
76daa3
             }
76daa3
             s->wait_sector_num[index] = -1;
76daa3
         }
76daa3
 
76daa3
-        ret = convert_co_write(s, sector_num, n, buf, status);
76daa3
-        if (ret < 0) {
76daa3
-            error_report("error while writing sector %" PRId64
76daa3
-                         ": %s", sector_num, strerror(-ret));
76daa3
-            s->ret = ret;
76daa3
-            goto out;
76daa3
+        if (s->ret == -EINPROGRESS) {
76daa3
+            ret = convert_co_write(s, sector_num, n, buf, status);
76daa3
+            if (ret < 0) {
76daa3
+                error_report("error while writing sector %" PRId64
76daa3
+                             ": %s", sector_num, strerror(-ret));
76daa3
+                s->ret = ret;
76daa3
+            }
76daa3
         }
76daa3
 
76daa3
         if (s->wr_in_order) {
76daa3
@@ -1837,7 +1834,6 @@ static void coroutine_fn convert_co_do_copy(void *opaque)
76daa3
         }
76daa3
     }
76daa3
 
76daa3
-out:
76daa3
     qemu_vfree(buf);
76daa3
     s->co[index] = NULL;
76daa3
     s->running_coroutines--;
76daa3
@@ -1899,7 +1895,7 @@ static int convert_do_copy(ImgConvertState *s)
76daa3
         qemu_coroutine_enter(s->co[i]);
76daa3
     }
76daa3
 
76daa3
-    while (s->ret == -EINPROGRESS) {
76daa3
+    while (s->running_coroutines) {
76daa3
         main_loop_wait(false);
76daa3
     }
76daa3
 
76daa3
-- 
76daa3
1.8.3.1
76daa3