zram: correct offset usage in zram_bio_discard
authorWeijie Yang <weijie.yang@samsung.com>
Wed, 4 Jun 2014 23:11:06 +0000 (16:11 -0700)
committerAlex Shi <alex.shi@linaro.org>
Mon, 11 May 2015 13:03:18 +0000 (21:03 +0800)
We want to skip the physical block(PAGE_SIZE) which is partially covered
by the discard bio, so we check the remaining size and subtract it if
there is a need to goto the next physical block.

The current offset usage in zram_bio_discard is incorrect, it will cause
its upper filesystem breakdown.  Consider the following scenario:

On some architecture or config, PAGE_SIZE is 64K for example, filesystem
is set up on zram disk without PAGE_SIZE aligned, a discard bio leads to a
offset = 4K and size=72K, normally, it should not really discard any
physical block as it partially cover two physical blocks.  However, with
the current offset usage, it will discard the second physical block and
free its memory, which will cause filesystem breakdown.

This patch corrects the offset usage in zram_bio_discard.

Signed-off-by: Weijie Yang <weijie.yang@samsung.com>
Cc: Minchan Kim <minchan@kernel.org>
Cc: Nitin Gupta <ngupta@vflare.org>
Acked-by: Joonsoo Kim <iamjoonsoo.kim@lge.com>
Cc: Sergey Senozhatsky <sergey.senozhatsky@gmail.com>
Cc: Bob Liu <bob.liu@oracle.com>
Cc: <stable@vger.kernel.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
(cherry picked from commit 38515c73398a4c58059ecf1087e844561b58ee0f)
Signed-off-by: Alex Shi <alex.shi@linaro.org>
drivers/block/zram/zram_drv.c

index 19cf51ad48ef7f0d53ff14039413555ec16a50af..efb6ff2a373506ee68981bf4172634d09f57e0bc 100644 (file)
@@ -571,10 +571,10 @@ static void zram_bio_discard(struct zram *zram, u32 index,
         * skipping this logical block is appropriate here.
         */
        if (offset) {
-               if (n < offset)
+               if (n <= (PAGE_SIZE - offset))
                        return;
 
-               n -= offset;
+               n -= (PAGE_SIZE - offset);
                index++;
        }