aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorDan Williams <dan.j.williams@intel.com>2017-04-12 16:37:44 -0400
committerDan Williams <dan.j.williams@intel.com>2017-04-25 16:20:36 -0400
commit817bf40265459578abc36c6bd53e27775b5c7ec4 (patch)
tree7f77cc1bb97457d10d92df83a640d8d90023a653
parentf26c5719b2d7b00de69eb83eb1c1c831759fdc9b (diff)
dm: teach dm-targets to use a dax_device + dax_operations
Arrange for dm to lookup the dax services available from member devices. Update the dax-capable targets, linear and stripe, to route dax operations to the underlying device. Changes the target-internal ->direct_access() method to more closely align with the dax_operations ->direct_access() calling convention. Cc: Toshi Kani <toshi.kani@hpe.com> Reviewed-by: Mike Snitzer <snitzer@redhat.com> Signed-off-by: Dan Williams <dan.j.williams@intel.com>
-rw-r--r--drivers/md/dm-linear.c27
-rw-r--r--drivers/md/dm-snap.c6
-rw-r--r--drivers/md/dm-stripe.c29
-rw-r--r--drivers/md/dm-target.c6
-rw-r--r--drivers/md/dm.c16
-rw-r--r--include/linux/device-mapper.h7
6 files changed, 43 insertions, 48 deletions
diff --git a/drivers/md/dm-linear.c b/drivers/md/dm-linear.c
index 4788b0b989a9..c5a52f4dae81 100644
--- a/drivers/md/dm-linear.c
+++ b/drivers/md/dm-linear.c
@@ -9,6 +9,7 @@
9#include <linux/init.h> 9#include <linux/init.h>
10#include <linux/blkdev.h> 10#include <linux/blkdev.h>
11#include <linux/bio.h> 11#include <linux/bio.h>
12#include <linux/dax.h>
12#include <linux/slab.h> 13#include <linux/slab.h>
13#include <linux/device-mapper.h> 14#include <linux/device-mapper.h>
14 15
@@ -141,22 +142,20 @@ static int linear_iterate_devices(struct dm_target *ti,
141 return fn(ti, lc->dev, lc->start, ti->len, data); 142 return fn(ti, lc->dev, lc->start, ti->len, data);
142} 143}
143 144
144static long linear_direct_access(struct dm_target *ti, sector_t sector, 145static long linear_dax_direct_access(struct dm_target *ti, pgoff_t pgoff,
145 void **kaddr, pfn_t *pfn, long size) 146 long nr_pages, void **kaddr, pfn_t *pfn)
146{ 147{
148 long ret;
147 struct linear_c *lc = ti->private; 149 struct linear_c *lc = ti->private;
148 struct block_device *bdev = lc->dev->bdev; 150 struct block_device *bdev = lc->dev->bdev;
149 struct blk_dax_ctl dax = { 151 struct dax_device *dax_dev = lc->dev->dax_dev;
150 .sector = linear_map_sector(ti, sector), 152 sector_t dev_sector, sector = pgoff * PAGE_SECTORS;
151 .size = size, 153
152 }; 154 dev_sector = linear_map_sector(ti, sector);
153 long ret; 155 ret = bdev_dax_pgoff(bdev, dev_sector, nr_pages * PAGE_SIZE, &pgoff);
154 156 if (ret)
155 ret = bdev_direct_access(bdev, &dax); 157 return ret;
156 *kaddr = dax.addr; 158 return dax_direct_access(dax_dev, pgoff, nr_pages, kaddr, pfn);
157 *pfn = dax.pfn;
158
159 return ret;
160} 159}
161 160
162static struct target_type linear_target = { 161static struct target_type linear_target = {
@@ -169,7 +168,7 @@ static struct target_type linear_target = {
169 .status = linear_status, 168 .status = linear_status,
170 .prepare_ioctl = linear_prepare_ioctl, 169 .prepare_ioctl = linear_prepare_ioctl,
171 .iterate_devices = linear_iterate_devices, 170 .iterate_devices = linear_iterate_devices,
172 .direct_access = linear_direct_access, 171 .direct_access = linear_dax_direct_access,
173}; 172};
174 173
175int __init dm_linear_init(void) 174int __init dm_linear_init(void)
diff --git a/drivers/md/dm-snap.c b/drivers/md/dm-snap.c
index c65feeada864..e152d9817c81 100644
--- a/drivers/md/dm-snap.c
+++ b/drivers/md/dm-snap.c
@@ -2302,8 +2302,8 @@ static int origin_map(struct dm_target *ti, struct bio *bio)
2302 return do_origin(o->dev, bio); 2302 return do_origin(o->dev, bio);
2303} 2303}
2304 2304
2305static long origin_direct_access(struct dm_target *ti, sector_t sector, 2305static long origin_dax_direct_access(struct dm_target *ti, pgoff_t pgoff,
2306 void **kaddr, pfn_t *pfn, long size) 2306 long nr_pages, void **kaddr, pfn_t *pfn)
2307{ 2307{
2308 DMWARN("device does not support dax."); 2308 DMWARN("device does not support dax.");
2309 return -EIO; 2309 return -EIO;
@@ -2368,7 +2368,7 @@ static struct target_type origin_target = {
2368 .postsuspend = origin_postsuspend, 2368 .postsuspend = origin_postsuspend,
2369 .status = origin_status, 2369 .status = origin_status,
2370 .iterate_devices = origin_iterate_devices, 2370 .iterate_devices = origin_iterate_devices,
2371 .direct_access = origin_direct_access, 2371 .direct_access = origin_dax_direct_access,
2372}; 2372};
2373 2373
2374static struct target_type snapshot_target = { 2374static struct target_type snapshot_target = {
diff --git a/drivers/md/dm-stripe.c b/drivers/md/dm-stripe.c
index 28193a57bf47..cb4b1e9e16ab 100644
--- a/drivers/md/dm-stripe.c
+++ b/drivers/md/dm-stripe.c
@@ -11,6 +11,7 @@
11#include <linux/init.h> 11#include <linux/init.h>
12#include <linux/blkdev.h> 12#include <linux/blkdev.h>
13#include <linux/bio.h> 13#include <linux/bio.h>
14#include <linux/dax.h>
14#include <linux/slab.h> 15#include <linux/slab.h>
15#include <linux/log2.h> 16#include <linux/log2.h>
16 17
@@ -308,27 +309,25 @@ static int stripe_map(struct dm_target *ti, struct bio *bio)
308 return DM_MAPIO_REMAPPED; 309 return DM_MAPIO_REMAPPED;
309} 310}
310 311
311static long stripe_direct_access(struct dm_target *ti, sector_t sector, 312static long stripe_dax_direct_access(struct dm_target *ti, pgoff_t pgoff,
312 void **kaddr, pfn_t *pfn, long size) 313 long nr_pages, void **kaddr, pfn_t *pfn)
313{ 314{
315 sector_t dev_sector, sector = pgoff * PAGE_SECTORS;
314 struct stripe_c *sc = ti->private; 316 struct stripe_c *sc = ti->private;
315 uint32_t stripe; 317 struct dax_device *dax_dev;
316 struct block_device *bdev; 318 struct block_device *bdev;
317 struct blk_dax_ctl dax = { 319 uint32_t stripe;
318 .size = size,
319 };
320 long ret; 320 long ret;
321 321
322 stripe_map_sector(sc, sector, &stripe, &dax.sector); 322 stripe_map_sector(sc, sector, &stripe, &dev_sector);
323 323 dev_sector += sc->stripe[stripe].physical_start;
324 dax.sector += sc->stripe[stripe].physical_start; 324 dax_dev = sc->stripe[stripe].dev->dax_dev;
325 bdev = sc->stripe[stripe].dev->bdev; 325 bdev = sc->stripe[stripe].dev->bdev;
326 326
327 ret = bdev_direct_access(bdev, &dax); 327 ret = bdev_dax_pgoff(bdev, dev_sector, nr_pages * PAGE_SIZE, &pgoff);
328 *kaddr = dax.addr; 328 if (ret)
329 *pfn = dax.pfn; 329 return ret;
330 330 return dax_direct_access(dax_dev, pgoff, nr_pages, kaddr, pfn);
331 return ret;
332} 331}
333 332
334/* 333/*
@@ -448,7 +447,7 @@ static struct target_type stripe_target = {
448 .status = stripe_status, 447 .status = stripe_status,
449 .iterate_devices = stripe_iterate_devices, 448 .iterate_devices = stripe_iterate_devices,
450 .io_hints = stripe_io_hints, 449 .io_hints = stripe_io_hints,
451 .direct_access = stripe_direct_access, 450 .direct_access = stripe_dax_direct_access,
452}; 451};
453 452
454int __init dm_stripe_init(void) 453int __init dm_stripe_init(void)
diff --git a/drivers/md/dm-target.c b/drivers/md/dm-target.c
index 43d3445b121d..6a7968f93f3c 100644
--- a/drivers/md/dm-target.c
+++ b/drivers/md/dm-target.c
@@ -142,8 +142,8 @@ static void io_err_release_clone_rq(struct request *clone)
142{ 142{
143} 143}
144 144
145static long io_err_direct_access(struct dm_target *ti, sector_t sector, 145static long io_err_dax_direct_access(struct dm_target *ti, pgoff_t pgoff,
146 void **kaddr, pfn_t *pfn, long size) 146 long nr_pages, void **kaddr, pfn_t *pfn)
147{ 147{
148 return -EIO; 148 return -EIO;
149} 149}
@@ -157,7 +157,7 @@ static struct target_type error_target = {
157 .map = io_err_map, 157 .map = io_err_map,
158 .clone_and_map_rq = io_err_clone_and_map_rq, 158 .clone_and_map_rq = io_err_clone_and_map_rq,
159 .release_clone_rq = io_err_release_clone_rq, 159 .release_clone_rq = io_err_release_clone_rq,
160 .direct_access = io_err_direct_access, 160 .direct_access = io_err_dax_direct_access,
161}; 161};
162 162
163int __init dm_target_init(void) 163int __init dm_target_init(void)
diff --git a/drivers/md/dm.c b/drivers/md/dm.c
index bd56dfe43a99..ef4c6f8cad47 100644
--- a/drivers/md/dm.c
+++ b/drivers/md/dm.c
@@ -630,6 +630,7 @@ static int open_table_device(struct table_device *td, dev_t dev,
630 } 630 }
631 631
632 td->dm_dev.bdev = bdev; 632 td->dm_dev.bdev = bdev;
633 td->dm_dev.dax_dev = dax_get_by_host(bdev->bd_disk->disk_name);
633 return 0; 634 return 0;
634} 635}
635 636
@@ -643,7 +644,9 @@ static void close_table_device(struct table_device *td, struct mapped_device *md
643 644
644 bd_unlink_disk_holder(td->dm_dev.bdev, dm_disk(md)); 645 bd_unlink_disk_holder(td->dm_dev.bdev, dm_disk(md));
645 blkdev_put(td->dm_dev.bdev, td->dm_dev.mode | FMODE_EXCL); 646 blkdev_put(td->dm_dev.bdev, td->dm_dev.mode | FMODE_EXCL);
647 put_dax(td->dm_dev.dax_dev);
646 td->dm_dev.bdev = NULL; 648 td->dm_dev.bdev = NULL;
649 td->dm_dev.dax_dev = NULL;
647} 650}
648 651
649static struct table_device *find_table_device(struct list_head *l, dev_t dev, 652static struct table_device *find_table_device(struct list_head *l, dev_t dev,
@@ -945,16 +948,9 @@ static long dm_dax_direct_access(struct dax_device *dax_dev, pgoff_t pgoff,
945 if (len < 1) 948 if (len < 1)
946 goto out; 949 goto out;
947 nr_pages = min(len, nr_pages); 950 nr_pages = min(len, nr_pages);
948 if (ti->type->direct_access) { 951 if (ti->type->direct_access)
949 ret = ti->type->direct_access(ti, sector, kaddr, pfn, 952 ret = ti->type->direct_access(ti, pgoff, nr_pages, kaddr, pfn);
950 nr_pages * PAGE_SIZE); 953
951 /*
952 * FIXME: convert ti->type->direct_access to return
953 * nr_pages directly.
954 */
955 if (ret >= 0)
956 ret /= PAGE_SIZE;
957 }
958 out: 954 out:
959 dm_put_live_table(md, srcu_idx); 955 dm_put_live_table(md, srcu_idx);
960 956
diff --git a/include/linux/device-mapper.h b/include/linux/device-mapper.h
index bcba4d89089c..df830d167892 100644
--- a/include/linux/device-mapper.h
+++ b/include/linux/device-mapper.h
@@ -128,14 +128,15 @@ typedef int (*dm_busy_fn) (struct dm_target *ti);
128 * < 0 : error 128 * < 0 : error
129 * >= 0 : the number of bytes accessible at the address 129 * >= 0 : the number of bytes accessible at the address
130 */ 130 */
131typedef long (*dm_direct_access_fn) (struct dm_target *ti, sector_t sector, 131typedef long (*dm_dax_direct_access_fn) (struct dm_target *ti, pgoff_t pgoff,
132 void **kaddr, pfn_t *pfn, long size); 132 long nr_pages, void **kaddr, pfn_t *pfn);
133#define PAGE_SECTORS (PAGE_SIZE / 512) 133#define PAGE_SECTORS (PAGE_SIZE / 512)
134 134
135void dm_error(const char *message); 135void dm_error(const char *message);
136 136
137struct dm_dev { 137struct dm_dev {
138 struct block_device *bdev; 138 struct block_device *bdev;
139 struct dax_device *dax_dev;
139 fmode_t mode; 140 fmode_t mode;
140 char name[16]; 141 char name[16];
141}; 142};
@@ -177,7 +178,7 @@ struct target_type {
177 dm_busy_fn busy; 178 dm_busy_fn busy;
178 dm_iterate_devices_fn iterate_devices; 179 dm_iterate_devices_fn iterate_devices;
179 dm_io_hints_fn io_hints; 180 dm_io_hints_fn io_hints;
180 dm_direct_access_fn direct_access; 181 dm_dax_direct_access_fn direct_access;
181 182
182 /* For internal device-mapper use. */ 183 /* For internal device-mapper use. */
183 struct list_head list; 184 struct list_head list;