From: Huang Ying <[email protected]>

The .rw_page in struct block_device_operations is used by the swap
subsystem to read/write the page contents from/into the corresponding
swap slot in the swap device.  To support the THP (Transparent Huge
Page) swap optimization, the .rw_page is enhanced to support to
read/write THP if possible.

Signed-off-by: "Huang, Ying" <[email protected]>
Cc: Johannes Weiner <[email protected]>
Cc: Minchan Kim <[email protected]>
Cc: Dan Williams <[email protected]>
Cc: Ross Zwisler <[email protected]>
Cc: Vishal L Verma <[email protected]>
Cc: Jens Axboe <[email protected]>
Cc: [email protected]
---
 drivers/block/brd.c           |  6 +++++-
 drivers/block/zram/zram_drv.c |  2 ++
 drivers/nvdimm/btt.c          |  4 +++-
 drivers/nvdimm/pmem.c         | 42 +++++++++++++++++++++++++++++++-----------
 4 files changed, 41 insertions(+), 13 deletions(-)

diff --git a/drivers/block/brd.c b/drivers/block/brd.c
index 57b574f2f66a..4240d2a9dcf9 100644
--- a/drivers/block/brd.c
+++ b/drivers/block/brd.c
@@ -324,7 +324,11 @@ static int brd_rw_page(struct block_device *bdev, sector_t 
sector,
                       struct page *page, bool is_write)
 {
        struct brd_device *brd = bdev->bd_disk->private_data;
-       int err = brd_do_bvec(brd, page, PAGE_SIZE, 0, is_write, sector);
+       int err;
+
+       if (PageTransHuge(page))
+               return -ENOTSUPP;
+       err = brd_do_bvec(brd, page, PAGE_SIZE, 0, is_write, sector);
        page_endio(page, is_write, err);
        return err;
 }
diff --git a/drivers/block/zram/zram_drv.c b/drivers/block/zram/zram_drv.c
index 5f2a862d8e31..09b11286c927 100644
--- a/drivers/block/zram/zram_drv.c
+++ b/drivers/block/zram/zram_drv.c
@@ -1049,6 +1049,8 @@ static int zram_rw_page(struct block_device *bdev, 
sector_t sector,
        struct zram *zram;
        struct bio_vec bv;
 
+       if (PageTransHuge(page))
+               return -ENOTSUPP;
        zram = bdev->bd_disk->private_data;
 
        if (!valid_io_request(zram, sector, PAGE_SIZE)) {
diff --git a/drivers/nvdimm/btt.c b/drivers/nvdimm/btt.c
index 983718b8fd9b..46d4a0bd2ae6 100644
--- a/drivers/nvdimm/btt.c
+++ b/drivers/nvdimm/btt.c
@@ -1248,8 +1248,10 @@ static int btt_rw_page(struct block_device *bdev, 
sector_t sector,
                struct page *page, bool is_write)
 {
        struct btt *btt = bdev->bd_disk->private_data;
+       unsigned int len;
 
-       btt_do_bvec(btt, NULL, page, PAGE_SIZE, 0, is_write, sector);
+       len = hpage_nr_pages(page) * PAGE_SIZE;
+       btt_do_bvec(btt, NULL, page, len, 0, is_write, sector);
        page_endio(page, is_write, 0);
        return 0;
 }
diff --git a/drivers/nvdimm/pmem.c b/drivers/nvdimm/pmem.c
index c544d466ea51..e644115d56a7 100644
--- a/drivers/nvdimm/pmem.c
+++ b/drivers/nvdimm/pmem.c
@@ -78,22 +78,40 @@ static int pmem_clear_poison(struct pmem_device *pmem, 
phys_addr_t offset,
 static void write_pmem(void *pmem_addr, struct page *page,
                unsigned int off, unsigned int len)
 {
-       void *mem = kmap_atomic(page);
-
-       memcpy_to_pmem(pmem_addr, mem + off, len);
-       kunmap_atomic(mem);
+       unsigned int chunk;
+       void *mem;
+
+       while (len) {
+               mem = kmap_atomic(page);
+               chunk = min_t(unsigned int, len, PAGE_SIZE);
+               memcpy_to_pmem(pmem_addr, mem + off, chunk);
+               kunmap_atomic(mem);
+               len -= chunk;
+               off = 0;
+               page++;
+               pmem_addr += PAGE_SIZE;
+       }
 }
 
 static int read_pmem(struct page *page, unsigned int off,
                void *pmem_addr, unsigned int len)
 {
+       unsigned int chunk;
        int rc;
-       void *mem = kmap_atomic(page);
-
-       rc = memcpy_mcsafe(mem + off, pmem_addr, len);
-       kunmap_atomic(mem);
-       if (rc)
-               return -EIO;
+       void *mem;
+
+       while (len) {
+               mem = kmap_atomic(page);
+               chunk = min_t(unsigned int, len, PAGE_SIZE);
+               rc = memcpy_mcsafe(mem + off, pmem_addr, chunk);
+               kunmap_atomic(mem);
+               if (rc)
+                       return -EIO;
+               len -= chunk;
+               off = 0;
+               page++;
+               pmem_addr += PAGE_SIZE;
+       }
        return 0;
 }
 
@@ -184,9 +202,11 @@ static int pmem_rw_page(struct block_device *bdev, 
sector_t sector,
                       struct page *page, bool is_write)
 {
        struct pmem_device *pmem = bdev->bd_queue->queuedata;
+       unsigned int len;
        int rc;
 
-       rc = pmem_do_bvec(pmem, page, PAGE_SIZE, 0, is_write, sector);
+       len = hpage_nr_pages(page) * PAGE_SIZE;
+       rc = pmem_do_bvec(pmem, page, len, 0, is_write, sector);
 
        /*
         * The ->rw_page interface is subtle and tricky.  The core
-- 
2.11.0

Reply via email to