diff options
author | Dan Williams <dan.j.williams@intel.com> | 2017-01-25 19:54:45 -0500 |
---|---|---|
committer | Dan Williams <dan.j.williams@intel.com> | 2017-04-19 18:14:36 -0400 |
commit | 1647b9b959c7b1b3c20e8efa2c40529e7dce756a (patch) | |
tree | ead50daa12cc606be8f24d6cc9fb18cb4a18bcfe /drivers/block | |
parent | 60fcd55cc236dbb3d6587f48120f00f59cb08540 (diff) |
brd: add dax_operations support
Setup a dax_inode to have the same lifetime as the brd block device and
add a ->direct_access() method that is equivalent to
brd_direct_access(). Once fs/dax.c has been converted to use
dax_operations the old brd_direct_access() will be removed.
Signed-off-by: Dan Williams <dan.j.williams@intel.com>
Diffstat (limited to 'drivers/block')
-rw-r--r-- | drivers/block/Kconfig | 1 | ||||
-rw-r--r-- | drivers/block/brd.c | 65 |
2 files changed, 55 insertions, 11 deletions
diff --git a/drivers/block/Kconfig b/drivers/block/Kconfig index f744de7a0f9b..e66956fc2c88 100644 --- a/drivers/block/Kconfig +++ b/drivers/block/Kconfig | |||
@@ -339,6 +339,7 @@ config BLK_DEV_SX8 | |||
339 | 339 | ||
340 | config BLK_DEV_RAM | 340 | config BLK_DEV_RAM |
341 | tristate "RAM block device support" | 341 | tristate "RAM block device support" |
342 | select DAX if BLK_DEV_RAM_DAX | ||
342 | ---help--- | 343 | ---help--- |
343 | Saying Y here will allow you to use a portion of your RAM memory as | 344 | Saying Y here will allow you to use a portion of your RAM memory as |
344 | a block device, so that you can make file systems on it, read and | 345 | a block device, so that you can make file systems on it, read and |
diff --git a/drivers/block/brd.c b/drivers/block/brd.c index 3adc32a3153b..60f3193c9ce2 100644 --- a/drivers/block/brd.c +++ b/drivers/block/brd.c | |||
@@ -21,6 +21,7 @@ | |||
21 | #include <linux/slab.h> | 21 | #include <linux/slab.h> |
22 | #ifdef CONFIG_BLK_DEV_RAM_DAX | 22 | #ifdef CONFIG_BLK_DEV_RAM_DAX |
23 | #include <linux/pfn_t.h> | 23 | #include <linux/pfn_t.h> |
24 | #include <linux/dax.h> | ||
24 | #endif | 25 | #endif |
25 | 26 | ||
26 | #include <linux/uaccess.h> | 27 | #include <linux/uaccess.h> |
@@ -41,6 +42,9 @@ struct brd_device { | |||
41 | 42 | ||
42 | struct request_queue *brd_queue; | 43 | struct request_queue *brd_queue; |
43 | struct gendisk *brd_disk; | 44 | struct gendisk *brd_disk; |
45 | #ifdef CONFIG_BLK_DEV_RAM_DAX | ||
46 | struct dax_device *dax_dev; | ||
47 | #endif | ||
44 | struct list_head brd_list; | 48 | struct list_head brd_list; |
45 | 49 | ||
46 | /* | 50 | /* |
@@ -375,30 +379,53 @@ static int brd_rw_page(struct block_device *bdev, sector_t sector, | |||
375 | } | 379 | } |
376 | 380 | ||
377 | #ifdef CONFIG_BLK_DEV_RAM_DAX | 381 | #ifdef CONFIG_BLK_DEV_RAM_DAX |
378 | static long brd_direct_access(struct block_device *bdev, sector_t sector, | 382 | static long __brd_direct_access(struct brd_device *brd, pgoff_t pgoff, |
379 | void **kaddr, pfn_t *pfn, long size) | 383 | long nr_pages, void **kaddr, pfn_t *pfn) |
380 | { | 384 | { |
381 | struct brd_device *brd = bdev->bd_disk->private_data; | ||
382 | struct page *page; | 385 | struct page *page; |
383 | 386 | ||
384 | if (!brd) | 387 | if (!brd) |
385 | return -ENODEV; | 388 | return -ENODEV; |
386 | page = brd_insert_page(brd, sector); | 389 | page = brd_insert_page(brd, PFN_PHYS(pgoff) / 512); |
387 | if (!page) | 390 | if (!page) |
388 | return -ENOSPC; | 391 | return -ENOSPC; |
389 | *kaddr = page_address(page); | 392 | *kaddr = page_address(page); |
390 | *pfn = page_to_pfn_t(page); | 393 | *pfn = page_to_pfn_t(page); |
391 | 394 | ||
392 | return PAGE_SIZE; | 395 | return 1; |
396 | } | ||
397 | |||
398 | static long brd_blk_direct_access(struct block_device *bdev, sector_t sector, | ||
399 | void **kaddr, pfn_t *pfn, long size) | ||
400 | { | ||
401 | struct brd_device *brd = bdev->bd_disk->private_data; | ||
402 | long nr_pages = __brd_direct_access(brd, PHYS_PFN(sector * 512), | ||
403 | PHYS_PFN(size), kaddr, pfn); | ||
404 | |||
405 | if (nr_pages < 0) | ||
406 | return nr_pages; | ||
407 | return nr_pages * PAGE_SIZE; | ||
408 | } | ||
409 | |||
410 | static long brd_dax_direct_access(struct dax_device *dax_dev, | ||
411 | pgoff_t pgoff, long nr_pages, void **kaddr, pfn_t *pfn) | ||
412 | { | ||
413 | struct brd_device *brd = dax_get_private(dax_dev); | ||
414 | |||
415 | return __brd_direct_access(brd, pgoff, nr_pages, kaddr, pfn); | ||
393 | } | 416 | } |
417 | |||
418 | static const struct dax_operations brd_dax_ops = { | ||
419 | .direct_access = brd_dax_direct_access, | ||
420 | }; | ||
394 | #else | 421 | #else |
395 | #define brd_direct_access NULL | 422 | #define brd_blk_direct_access NULL |
396 | #endif | 423 | #endif |
397 | 424 | ||
398 | static const struct block_device_operations brd_fops = { | 425 | static const struct block_device_operations brd_fops = { |
399 | .owner = THIS_MODULE, | 426 | .owner = THIS_MODULE, |
400 | .rw_page = brd_rw_page, | 427 | .rw_page = brd_rw_page, |
401 | .direct_access = brd_direct_access, | 428 | .direct_access = brd_blk_direct_access, |
402 | }; | 429 | }; |
403 | 430 | ||
404 | /* | 431 | /* |
@@ -441,7 +468,9 @@ static struct brd_device *brd_alloc(int i) | |||
441 | { | 468 | { |
442 | struct brd_device *brd; | 469 | struct brd_device *brd; |
443 | struct gendisk *disk; | 470 | struct gendisk *disk; |
444 | 471 | #ifdef CONFIG_BLK_DEV_RAM_DAX | |
472 | struct dax_device *dax_dev; | ||
473 | #endif | ||
445 | brd = kzalloc(sizeof(*brd), GFP_KERNEL); | 474 | brd = kzalloc(sizeof(*brd), GFP_KERNEL); |
446 | if (!brd) | 475 | if (!brd) |
447 | goto out; | 476 | goto out; |
@@ -469,9 +498,6 @@ static struct brd_device *brd_alloc(int i) | |||
469 | blk_queue_max_discard_sectors(brd->brd_queue, UINT_MAX); | 498 | blk_queue_max_discard_sectors(brd->brd_queue, UINT_MAX); |
470 | brd->brd_queue->limits.discard_zeroes_data = 1; | 499 | brd->brd_queue->limits.discard_zeroes_data = 1; |
471 | queue_flag_set_unlocked(QUEUE_FLAG_DISCARD, brd->brd_queue); | 500 | queue_flag_set_unlocked(QUEUE_FLAG_DISCARD, brd->brd_queue); |
472 | #ifdef CONFIG_BLK_DEV_RAM_DAX | ||
473 | queue_flag_set_unlocked(QUEUE_FLAG_DAX, brd->brd_queue); | ||
474 | #endif | ||
475 | disk = brd->brd_disk = alloc_disk(max_part); | 501 | disk = brd->brd_disk = alloc_disk(max_part); |
476 | if (!disk) | 502 | if (!disk) |
477 | goto out_free_queue; | 503 | goto out_free_queue; |
@@ -484,8 +510,21 @@ static struct brd_device *brd_alloc(int i) | |||
484 | sprintf(disk->disk_name, "ram%d", i); | 510 | sprintf(disk->disk_name, "ram%d", i); |
485 | set_capacity(disk, rd_size * 2); | 511 | set_capacity(disk, rd_size * 2); |
486 | 512 | ||
513 | #ifdef CONFIG_BLK_DEV_RAM_DAX | ||
514 | queue_flag_set_unlocked(QUEUE_FLAG_DAX, brd->brd_queue); | ||
515 | dax_dev = alloc_dax(brd, disk->disk_name, &brd_dax_ops); | ||
516 | if (!dax_dev) | ||
517 | goto out_free_inode; | ||
518 | #endif | ||
519 | |||
520 | |||
487 | return brd; | 521 | return brd; |
488 | 522 | ||
523 | #ifdef CONFIG_BLK_DEV_RAM_DAX | ||
524 | out_free_inode: | ||
525 | kill_dax(dax_dev); | ||
526 | put_dax(dax_dev); | ||
527 | #endif | ||
489 | out_free_queue: | 528 | out_free_queue: |
490 | blk_cleanup_queue(brd->brd_queue); | 529 | blk_cleanup_queue(brd->brd_queue); |
491 | out_free_dev: | 530 | out_free_dev: |
@@ -525,6 +564,10 @@ out: | |||
525 | static void brd_del_one(struct brd_device *brd) | 564 | static void brd_del_one(struct brd_device *brd) |
526 | { | 565 | { |
527 | list_del(&brd->brd_list); | 566 | list_del(&brd->brd_list); |
567 | #ifdef CONFIG_BLK_DEV_RAM_DAX | ||
568 | kill_dax(brd->dax_dev); | ||
569 | put_dax(brd->dax_dev); | ||
570 | #endif | ||
528 | del_gendisk(brd->brd_disk); | 571 | del_gendisk(brd->brd_disk); |
529 | brd_free(brd); | 572 | brd_free(brd); |
530 | } | 573 | } |