Blame SOURCES/kvm-memory-cleanup-side-effects-of-memory_region_init_fo.patch

ae23c9
From cfa472eb467f8e5139a2a7d30aa85c9affbccd4f Mon Sep 17 00:00:00 2001
ae23c9
From: Igor Mammedov <imammedo@redhat.com>
ae23c9
Date: Fri, 5 Oct 2018 12:59:47 +0100
ae23c9
Subject: [PATCH 6/6] memory: cleanup side effects of memory_region_init_foo()
ae23c9
 on failure
ae23c9
ae23c9
RH-Author: Igor Mammedov <imammedo@redhat.com>
ae23c9
Message-id: <1538744387-84898-1-git-send-email-imammedo@redhat.com>
ae23c9
Patchwork-id: 82391
ae23c9
O-Subject: [RHEL-8 qemu-kvm PATCH] memory: cleanup side effects of memory_region_init_foo() on failure
ae23c9
Bugzilla: 1600365
ae23c9
RH-Acked-by: Paolo Bonzini <pbonzini@redhat.com>
ae23c9
RH-Acked-by: Dr. David Alan Gilbert <dgilbert@redhat.com>
ae23c9
RH-Acked-by: Pankaj Gupta <pagupta@redhat.com>
ae23c9
RH-Acked-by: Laszlo Ersek <lersek@redhat.com>
ae23c9
ae23c9
Bugzilla: https://bugzilla.redhat.com/show_bug.cgi?id=1600365
ae23c9
Brew: https://brewweb.engineering.redhat.com/brew/taskinfo?taskID=18658506
ae23c9
ae23c9
if MemoryRegion intialization fails it's left in semi-initialized state,
ae23c9
where it's size is not 0 and attached as child to owner object.
ae23c9
And this leds to crash in following use-case:
ae23c9
    (monitor) object_add memory-backend-file,id=mem1,size=99999G,mem-path=/tmp/foo,discard-data=yes
ae23c9
    memory.c:2083: memory_region_get_ram_ptr: Assertion `mr->ram_block' failed
ae23c9
    Aborted (core dumped)
ae23c9
it happens due to assumption that memory region is intialized when
ae23c9
   memory_region_size() != 0
ae23c9
and therefore it's ok to access it in
ae23c9
   file_backend_unparent()
ae23c9
      if (memory_region_size() != 0)
ae23c9
          memory_region_get_ram_ptr()
ae23c9
ae23c9
which happens when object_add fails and unparents failed backend making
ae23c9
file_backend_unparent() access invalid memory region.
ae23c9
ae23c9
Fix it by making sure that memory_region_init_foo() APIs cleanup externally
ae23c9
visible side effects on failure (like set size to 0 and unparenting object)
ae23c9
ae23c9
Signed-off-by: Igor Mammedov <imammedo@redhat.com>
ae23c9
Message-Id: <1536064777-42312-1-git-send-email-imammedo@redhat.com>
ae23c9
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
ae23c9
(cherry picked from commit 1cd3d492624da399d66c4c3e6a5eabb8f96bb0a2)
ae23c9
Signed-off-by: Igor Mammedov <imammedo@redhat.com>
ae23c9
Signed-off-by: Danilo C. L. de Paula <ddepaula@redhat.com>
ae23c9
ae23c9
Conflicts:
ae23c9
	memory.c
ae23c9
          due missing (cbfc01710 "memory, exec: switch file ram allocation functions to 'flags' parameters")
ae23c9
          not related to the patch signature mismatch of
ae23c9
           qemu_ram_alloc_from_file()/qemu_ram_alloc_from_fd()
ae23c9
---
ae23c9
 memory.c | 48 ++++++++++++++++++++++++++++++++++++++++++------
ae23c9
 1 file changed, 42 insertions(+), 6 deletions(-)
ae23c9
ae23c9
diff --git a/memory.c b/memory.c
ae23c9
index e70b64b..1a99b9c 100644
ae23c9
--- a/memory.c
ae23c9
+++ b/memory.c
ae23c9
@@ -1519,12 +1519,18 @@ void memory_region_init_ram_shared_nomigrate(MemoryRegion *mr,
ae23c9
                                              bool share,
ae23c9
                                              Error **errp)
ae23c9
 {
ae23c9
+    Error *err = NULL;
ae23c9
     memory_region_init(mr, owner, name, size);
ae23c9
     mr->ram = true;
ae23c9
     mr->terminates = true;
ae23c9
     mr->destructor = memory_region_destructor_ram;
ae23c9
-    mr->ram_block = qemu_ram_alloc(size, share, mr, errp);
ae23c9
+    mr->ram_block = qemu_ram_alloc(size, share, mr, &err;;
ae23c9
     mr->dirty_log_mask = tcg_enabled() ? (1 << DIRTY_MEMORY_CODE) : 0;
ae23c9
+    if (err) {
ae23c9
+        mr->size = int128_zero();
ae23c9
+        object_unparent(OBJECT(mr));
ae23c9
+        error_propagate(errp, err);
ae23c9
+    }
ae23c9
 }
ae23c9
 
ae23c9
 void memory_region_init_resizeable_ram(MemoryRegion *mr,
ae23c9
@@ -1537,13 +1543,19 @@ void memory_region_init_resizeable_ram(MemoryRegion *mr,
ae23c9
                                                        void *host),
ae23c9
                                        Error **errp)
ae23c9
 {
ae23c9
+    Error *err = NULL;
ae23c9
     memory_region_init(mr, owner, name, size);
ae23c9
     mr->ram = true;
ae23c9
     mr->terminates = true;
ae23c9
     mr->destructor = memory_region_destructor_ram;
ae23c9
     mr->ram_block = qemu_ram_alloc_resizeable(size, max_size, resized,
ae23c9
-                                              mr, errp);
ae23c9
+                                              mr, &err;;
ae23c9
     mr->dirty_log_mask = tcg_enabled() ? (1 << DIRTY_MEMORY_CODE) : 0;
ae23c9
+    if (err) {
ae23c9
+        mr->size = int128_zero();
ae23c9
+        object_unparent(OBJECT(mr));
ae23c9
+        error_propagate(errp, err);
ae23c9
+    }
ae23c9
 }
ae23c9
 
ae23c9
 #ifdef __linux__
ae23c9
@@ -1556,13 +1568,19 @@ void memory_region_init_ram_from_file(MemoryRegion *mr,
ae23c9
                                       const char *path,
ae23c9
                                       Error **errp)
ae23c9
 {
ae23c9
+    Error *err = NULL;
ae23c9
     memory_region_init(mr, owner, name, size);
ae23c9
     mr->ram = true;
ae23c9
     mr->terminates = true;
ae23c9
     mr->destructor = memory_region_destructor_ram;
ae23c9
     mr->align = align;
ae23c9
-    mr->ram_block = qemu_ram_alloc_from_file(size, mr, share, path, errp);
ae23c9
+    mr->ram_block = qemu_ram_alloc_from_file(size, mr, share, path, &err;;
ae23c9
     mr->dirty_log_mask = tcg_enabled() ? (1 << DIRTY_MEMORY_CODE) : 0;
ae23c9
+    if (err) {
ae23c9
+        mr->size = int128_zero();
ae23c9
+        object_unparent(OBJECT(mr));
ae23c9
+        error_propagate(errp, err);
ae23c9
+    }
ae23c9
 }
ae23c9
 
ae23c9
 void memory_region_init_ram_from_fd(MemoryRegion *mr,
ae23c9
@@ -1573,12 +1591,18 @@ void memory_region_init_ram_from_fd(MemoryRegion *mr,
ae23c9
                                     int fd,
ae23c9
                                     Error **errp)
ae23c9
 {
ae23c9
+    Error *err = NULL;
ae23c9
     memory_region_init(mr, owner, name, size);
ae23c9
     mr->ram = true;
ae23c9
     mr->terminates = true;
ae23c9
     mr->destructor = memory_region_destructor_ram;
ae23c9
-    mr->ram_block = qemu_ram_alloc_from_fd(size, mr, share, fd, errp);
ae23c9
+    mr->ram_block = qemu_ram_alloc_from_fd(size, mr, share, fd, &err;;
ae23c9
     mr->dirty_log_mask = tcg_enabled() ? (1 << DIRTY_MEMORY_CODE) : 0;
ae23c9
+    if (err) {
ae23c9
+        mr->size = int128_zero();
ae23c9
+        object_unparent(OBJECT(mr));
ae23c9
+        error_propagate(errp, err);
ae23c9
+    }
ae23c9
 }
ae23c9
 #endif
ae23c9
 
ae23c9
@@ -1629,13 +1653,19 @@ void memory_region_init_rom_nomigrate(MemoryRegion *mr,
ae23c9
                                       uint64_t size,
ae23c9
                                       Error **errp)
ae23c9
 {
ae23c9
+    Error *err = NULL;
ae23c9
     memory_region_init(mr, owner, name, size);
ae23c9
     mr->ram = true;
ae23c9
     mr->readonly = true;
ae23c9
     mr->terminates = true;
ae23c9
     mr->destructor = memory_region_destructor_ram;
ae23c9
-    mr->ram_block = qemu_ram_alloc(size, false, mr, errp);
ae23c9
+    mr->ram_block = qemu_ram_alloc(size, false, mr, &err;;
ae23c9
     mr->dirty_log_mask = tcg_enabled() ? (1 << DIRTY_MEMORY_CODE) : 0;
ae23c9
+    if (err) {
ae23c9
+        mr->size = int128_zero();
ae23c9
+        object_unparent(OBJECT(mr));
ae23c9
+        error_propagate(errp, err);
ae23c9
+    }
ae23c9
 }
ae23c9
 
ae23c9
 void memory_region_init_rom_device_nomigrate(MemoryRegion *mr,
ae23c9
@@ -1646,6 +1676,7 @@ void memory_region_init_rom_device_nomigrate(MemoryRegion *mr,
ae23c9
                                              uint64_t size,
ae23c9
                                              Error **errp)
ae23c9
 {
ae23c9
+    Error *err = NULL;
ae23c9
     assert(ops);
ae23c9
     memory_region_init(mr, owner, name, size);
ae23c9
     mr->ops = ops;
ae23c9
@@ -1653,7 +1684,12 @@ void memory_region_init_rom_device_nomigrate(MemoryRegion *mr,
ae23c9
     mr->terminates = true;
ae23c9
     mr->rom_device = true;
ae23c9
     mr->destructor = memory_region_destructor_ram;
ae23c9
-    mr->ram_block = qemu_ram_alloc(size, false,  mr, errp);
ae23c9
+    mr->ram_block = qemu_ram_alloc(size, false,  mr, &err;;
ae23c9
+    if (err) {
ae23c9
+        mr->size = int128_zero();
ae23c9
+        object_unparent(OBJECT(mr));
ae23c9
+        error_propagate(errp, err);
ae23c9
+    }
ae23c9
 }
ae23c9
 
ae23c9
 void memory_region_init_iommu(void *_iommu_mr,
ae23c9
-- 
ae23c9
1.8.3.1
ae23c9