Blame SOURCES/kvm-virtio-blk-Treat-read-write-beyond-end-as-invalid.patch

958e1b
From 55767ddb818ef53eee50777bd276cbf7e7813bfb Mon Sep 17 00:00:00 2001
958e1b
From: Markus Armbruster <armbru@redhat.com>
958e1b
Date: Fri, 31 Oct 2014 16:29:54 +0100
958e1b
Subject: [PATCH 17/19] virtio-blk: Treat read/write beyond end as invalid
958e1b
958e1b
Message-id: <1414772996-17272-4-git-send-email-armbru@redhat.com>
958e1b
Patchwork-id: 62019
958e1b
O-Subject: [PATCH RHEL-7.1 qemu-kvm 3/5] virtio-blk: Treat read/write beyond end as invalid
958e1b
Bugzilla: 1085232
958e1b
RH-Acked-by: Paolo Bonzini <pbonzini@redhat.com>
958e1b
RH-Acked-by: Fam Zheng <famz@redhat.com>
958e1b
958e1b
The block layer fails such reads and writes just fine.  However, they
958e1b
then get treated like valid operations that fail: the error action
958e1b
gets executed.  Unwanted; reporting the error to the guest is the only
958e1b
sensible action.
958e1b
958e1b
Reject them before passing them to the block layer.  This bypasses the
958e1b
error action and I/O accounting.
958e1b
958e1b
Signed-off-by: Markus Armbruster <armbru@redhat.com>
958e1b
Reviewed-by: Fam Zheng <famz@redhat.com>
958e1b
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
958e1b
(cherry picked from commit 3c2daac0b98952a858277878cb11294256b39e43)
958e1b
Signed-off-by: Miroslav Rezanina <mrezanin@redhat.com>
958e1b
---
958e1b
 hw/block/virtio-blk.c | 7 +++++++
958e1b
 1 file changed, 7 insertions(+)
958e1b
958e1b
diff --git a/hw/block/virtio-blk.c b/hw/block/virtio-blk.c
958e1b
index ffb311c..4f6aa22 100644
958e1b
--- a/hw/block/virtio-blk.c
958e1b
+++ b/hw/block/virtio-blk.c
958e1b
@@ -284,12 +284,19 @@ static void virtio_blk_handle_flush(VirtIOBlockReq *req, MultiReqBuffer *mrb)
958e1b
 static bool virtio_blk_sect_range_ok(VirtIOBlock *dev,
958e1b
                                      uint64_t sector, size_t size)
958e1b
 {
958e1b
+    uint64_t nb_sectors = size >> BDRV_SECTOR_BITS;
958e1b
+    uint64_t total_sectors;
958e1b
+
958e1b
     if (sector & dev->sector_mask) {
958e1b
         return false;
958e1b
     }
958e1b
     if (size % dev->conf->logical_block_size) {
958e1b
         return false;
958e1b
     }
958e1b
+    bdrv_get_geometry(dev->bs, &total_sectors);
958e1b
+    if (sector > total_sectors || nb_sectors > total_sectors - sector) {
958e1b
+        return false;
958e1b
+    }
958e1b
     return true;
958e1b
 }
958e1b
 
958e1b
-- 
958e1b
1.8.3.1
958e1b