diff options
| -rw-r--r-- | block/blk-barrier.c | 31 | ||||
| -rw-r--r-- | block/ioctl.c | 49 | ||||
| -rw-r--r-- | fs/btrfs/extent-tree.c | 3 | ||||
| -rw-r--r-- | fs/gfs2/rgrp.c | 6 | ||||
| -rw-r--r-- | include/linux/blkdev.h | 9 | ||||
| -rw-r--r-- | mm/swapfile.c | 6 |
6 files changed, 37 insertions, 67 deletions
diff --git a/block/blk-barrier.c b/block/blk-barrier.c index 30022b4e2f63..6593ab39cfe9 100644 --- a/block/blk-barrier.c +++ b/block/blk-barrier.c | |||
| @@ -348,6 +348,9 @@ static void blkdev_discard_end_io(struct bio *bio, int err) | |||
| 348 | clear_bit(BIO_UPTODATE, &bio->bi_flags); | 348 | clear_bit(BIO_UPTODATE, &bio->bi_flags); |
| 349 | } | 349 | } |
| 350 | 350 | ||
| 351 | if (bio->bi_private) | ||
| 352 | complete(bio->bi_private); | ||
| 353 | |||
| 351 | bio_put(bio); | 354 | bio_put(bio); |
| 352 | } | 355 | } |
| 353 | 356 | ||
| @@ -357,21 +360,20 @@ static void blkdev_discard_end_io(struct bio *bio, int err) | |||
| 357 | * @sector: start sector | 360 | * @sector: start sector |
| 358 | * @nr_sects: number of sectors to discard | 361 | * @nr_sects: number of sectors to discard |
| 359 | * @gfp_mask: memory allocation flags (for bio_alloc) | 362 | * @gfp_mask: memory allocation flags (for bio_alloc) |
| 363 | * @flags: DISCARD_FL_* flags to control behaviour | ||
| 360 | * | 364 | * |
| 361 | * Description: | 365 | * Description: |
| 362 | * Issue a discard request for the sectors in question. Does not wait. | 366 | * Issue a discard request for the sectors in question. |
| 363 | */ | 367 | */ |
| 364 | int blkdev_issue_discard(struct block_device *bdev, | 368 | int blkdev_issue_discard(struct block_device *bdev, sector_t sector, |
| 365 | sector_t sector, sector_t nr_sects, gfp_t gfp_mask) | 369 | sector_t nr_sects, gfp_t gfp_mask, int flags) |
| 366 | { | 370 | { |
| 367 | struct request_queue *q; | 371 | DECLARE_COMPLETION_ONSTACK(wait); |
| 368 | struct bio *bio; | 372 | struct request_queue *q = bdev_get_queue(bdev); |
| 373 | int type = flags & DISCARD_FL_BARRIER ? | ||
| 374 | DISCARD_BARRIER : DISCARD_NOBARRIER; | ||
| 369 | int ret = 0; | 375 | int ret = 0; |
| 370 | 376 | ||
| 371 | if (bdev->bd_disk == NULL) | ||
| 372 | return -ENXIO; | ||
| 373 | |||
| 374 | q = bdev_get_queue(bdev); | ||
| 375 | if (!q) | 377 | if (!q) |
| 376 | return -ENXIO; | 378 | return -ENXIO; |
| 377 | 379 | ||
| @@ -379,12 +381,14 @@ int blkdev_issue_discard(struct block_device *bdev, | |||
| 379 | return -EOPNOTSUPP; | 381 | return -EOPNOTSUPP; |
| 380 | 382 | ||
| 381 | while (nr_sects && !ret) { | 383 | while (nr_sects && !ret) { |
| 382 | bio = bio_alloc(gfp_mask, 0); | 384 | struct bio *bio = bio_alloc(gfp_mask, 0); |
| 383 | if (!bio) | 385 | if (!bio) |
| 384 | return -ENOMEM; | 386 | return -ENOMEM; |
| 385 | 387 | ||
| 386 | bio->bi_end_io = blkdev_discard_end_io; | 388 | bio->bi_end_io = blkdev_discard_end_io; |
| 387 | bio->bi_bdev = bdev; | 389 | bio->bi_bdev = bdev; |
| 390 | if (flags & DISCARD_FL_WAIT) | ||
| 391 | bio->bi_private = &wait; | ||
| 388 | 392 | ||
| 389 | bio->bi_sector = sector; | 393 | bio->bi_sector = sector; |
| 390 | 394 | ||
| @@ -396,10 +400,13 @@ int blkdev_issue_discard(struct block_device *bdev, | |||
| 396 | bio->bi_size = nr_sects << 9; | 400 | bio->bi_size = nr_sects << 9; |
| 397 | nr_sects = 0; | 401 | nr_sects = 0; |
| 398 | } | 402 | } |
| 403 | |||
| 399 | bio_get(bio); | 404 | bio_get(bio); |
| 400 | submit_bio(DISCARD_BARRIER, bio); | 405 | submit_bio(type, bio); |
| 406 | |||
| 407 | if (flags & DISCARD_FL_WAIT) | ||
| 408 | wait_for_completion(&wait); | ||
| 401 | 409 | ||
| 402 | /* Check if it failed immediately */ | ||
| 403 | if (bio_flagged(bio, BIO_EOPNOTSUPP)) | 410 | if (bio_flagged(bio, BIO_EOPNOTSUPP)) |
| 404 | ret = -EOPNOTSUPP; | 411 | ret = -EOPNOTSUPP; |
| 405 | else if (!bio_flagged(bio, BIO_UPTODATE)) | 412 | else if (!bio_flagged(bio, BIO_UPTODATE)) |
diff --git a/block/ioctl.c b/block/ioctl.c index 500e4c73cc52..d3e6b5827a34 100644 --- a/block/ioctl.c +++ b/block/ioctl.c | |||
| @@ -112,22 +112,9 @@ static int blkdev_reread_part(struct block_device *bdev) | |||
| 112 | return res; | 112 | return res; |
| 113 | } | 113 | } |
| 114 | 114 | ||
| 115 | static void blk_ioc_discard_endio(struct bio *bio, int err) | ||
| 116 | { | ||
| 117 | if (err) { | ||
| 118 | if (err == -EOPNOTSUPP) | ||
| 119 | set_bit(BIO_EOPNOTSUPP, &bio->bi_flags); | ||
| 120 | clear_bit(BIO_UPTODATE, &bio->bi_flags); | ||
| 121 | } | ||
| 122 | complete(bio->bi_private); | ||
| 123 | } | ||
| 124 | |||
| 125 | static int blk_ioctl_discard(struct block_device *bdev, uint64_t start, | 115 | static int blk_ioctl_discard(struct block_device *bdev, uint64_t start, |
| 126 | uint64_t len) | 116 | uint64_t len) |
| 127 | { | 117 | { |
| 128 | struct request_queue *q = bdev_get_queue(bdev); | ||
| 129 | int ret = 0; | ||
| 130 | |||
| 131 | if (start & 511) | 118 | if (start & 511) |
| 132 | return -EINVAL; | 119 | return -EINVAL; |
| 133 | if (len & 511) | 120 | if (len & 511) |
| @@ -137,40 +124,8 @@ static int blk_ioctl_discard(struct block_device *bdev, uint64_t start, | |||
| 137 | 124 | ||
| 138 | if (start + len > (bdev->bd_inode->i_size >> 9)) | 125 | if (start + len > (bdev->bd_inode->i_size >> 9)) |
| 139 | return -EINVAL; | 126 | return -EINVAL; |
| 140 | 127 | return blkdev_issue_discard(bdev, start, len, GFP_KERNEL, | |
| 141 | if (!q->prepare_discard_fn) | 128 | DISCARD_FL_WAIT); |
| 142 | return -EOPNOTSUPP; | ||
| 143 | |||
| 144 | while (len && !ret) { | ||
| 145 | DECLARE_COMPLETION_ONSTACK(wait); | ||
| 146 | struct bio *bio; | ||
| 147 | |||
| 148 | bio = bio_alloc(GFP_KERNEL, 0); | ||
| 149 | |||
| 150 | bio->bi_end_io = blk_ioc_discard_endio; | ||
| 151 | bio->bi_bdev = bdev; | ||
| 152 | bio->bi_private = &wait; | ||
| 153 | bio->bi_sector = start; | ||
| 154 | |||
| 155 | if (len > queue_max_hw_sectors(q)) { | ||
| 156 | bio->bi_size = queue_max_hw_sectors(q) << 9; | ||
| 157 | len -= queue_max_hw_sectors(q); | ||
| 158 | start += queue_max_hw_sectors(q); | ||
| 159 | } else { | ||
| 160 | bio->bi_size = len << 9; | ||
| 161 | len = 0; | ||
| 162 | } | ||
| 163 | submit_bio(DISCARD_NOBARRIER, bio); | ||
| 164 | |||
| 165 | wait_for_completion(&wait); | ||
| 166 | |||
| 167 | if (bio_flagged(bio, BIO_EOPNOTSUPP)) | ||
| 168 | ret = -EOPNOTSUPP; | ||
| 169 | else if (!bio_flagged(bio, BIO_UPTODATE)) | ||
| 170 | ret = -EIO; | ||
| 171 | bio_put(bio); | ||
| 172 | } | ||
| 173 | return ret; | ||
| 174 | } | 129 | } |
| 175 | 130 | ||
| 176 | static int put_ushort(unsigned long arg, unsigned short val) | 131 | static int put_ushort(unsigned long arg, unsigned short val) |
diff --git a/fs/btrfs/extent-tree.c b/fs/btrfs/extent-tree.c index 72a2b9c28e9f..535f85ba104f 100644 --- a/fs/btrfs/extent-tree.c +++ b/fs/btrfs/extent-tree.c | |||
| @@ -1511,7 +1511,8 @@ static int remove_extent_backref(struct btrfs_trans_handle *trans, | |||
| 1511 | static void btrfs_issue_discard(struct block_device *bdev, | 1511 | static void btrfs_issue_discard(struct block_device *bdev, |
| 1512 | u64 start, u64 len) | 1512 | u64 start, u64 len) |
| 1513 | { | 1513 | { |
| 1514 | blkdev_issue_discard(bdev, start >> 9, len >> 9, GFP_KERNEL); | 1514 | blkdev_issue_discard(bdev, start >> 9, len >> 9, GFP_KERNEL, |
| 1515 | DISCARD_FL_BARRIER); | ||
| 1515 | } | 1516 | } |
| 1516 | #endif | 1517 | #endif |
| 1517 | 1518 | ||
diff --git a/fs/gfs2/rgrp.c b/fs/gfs2/rgrp.c index fba795798d3a..fbc43241f2ef 100644 --- a/fs/gfs2/rgrp.c +++ b/fs/gfs2/rgrp.c | |||
| @@ -857,7 +857,8 @@ static void gfs2_rgrp_send_discards(struct gfs2_sbd *sdp, u64 offset, | |||
| 857 | goto start_new_extent; | 857 | goto start_new_extent; |
| 858 | if ((start + nr_sects) != blk) { | 858 | if ((start + nr_sects) != blk) { |
| 859 | rv = blkdev_issue_discard(bdev, start, | 859 | rv = blkdev_issue_discard(bdev, start, |
| 860 | nr_sects, GFP_NOFS); | 860 | nr_sects, GFP_NOFS, |
| 861 | DISCARD_FL_BARRIER); | ||
| 861 | if (rv) | 862 | if (rv) |
| 862 | goto fail; | 863 | goto fail; |
| 863 | nr_sects = 0; | 864 | nr_sects = 0; |
| @@ -871,7 +872,8 @@ start_new_extent: | |||
| 871 | } | 872 | } |
| 872 | } | 873 | } |
| 873 | if (nr_sects) { | 874 | if (nr_sects) { |
| 874 | rv = blkdev_issue_discard(bdev, start, nr_sects, GFP_NOFS); | 875 | rv = blkdev_issue_discard(bdev, start, nr_sects, GFP_NOFS, |
| 876 | DISCARD_FL_BARRIER); | ||
| 875 | if (rv) | 877 | if (rv) |
| 876 | goto fail; | 878 | goto fail; |
| 877 | } | 879 | } |
diff --git a/include/linux/blkdev.h b/include/linux/blkdev.h index 86c2bdff3b89..e23a86cae5ac 100644 --- a/include/linux/blkdev.h +++ b/include/linux/blkdev.h | |||
| @@ -998,15 +998,18 @@ static inline struct request *blk_map_queue_find_tag(struct blk_queue_tag *bqt, | |||
| 998 | } | 998 | } |
| 999 | 999 | ||
| 1000 | extern int blkdev_issue_flush(struct block_device *, sector_t *); | 1000 | extern int blkdev_issue_flush(struct block_device *, sector_t *); |
| 1001 | extern int blkdev_issue_discard(struct block_device *, | 1001 | #define DISCARD_FL_WAIT 0x01 /* wait for completion */ |
| 1002 | sector_t sector, sector_t nr_sects, gfp_t); | 1002 | #define DISCARD_FL_BARRIER 0x02 /* issue DISCARD_BARRIER request */ |
| 1003 | extern int blkdev_issue_discard(struct block_device *, sector_t sector, | ||
| 1004 | sector_t nr_sects, gfp_t, int flags); | ||
| 1003 | 1005 | ||
| 1004 | static inline int sb_issue_discard(struct super_block *sb, | 1006 | static inline int sb_issue_discard(struct super_block *sb, |
| 1005 | sector_t block, sector_t nr_blocks) | 1007 | sector_t block, sector_t nr_blocks) |
| 1006 | { | 1008 | { |
| 1007 | block <<= (sb->s_blocksize_bits - 9); | 1009 | block <<= (sb->s_blocksize_bits - 9); |
| 1008 | nr_blocks <<= (sb->s_blocksize_bits - 9); | 1010 | nr_blocks <<= (sb->s_blocksize_bits - 9); |
| 1009 | return blkdev_issue_discard(sb->s_bdev, block, nr_blocks, GFP_KERNEL); | 1011 | return blkdev_issue_discard(sb->s_bdev, block, nr_blocks, GFP_KERNEL, |
| 1012 | DISCARD_FL_BARRIER); | ||
| 1010 | } | 1013 | } |
| 1011 | 1014 | ||
| 1012 | extern int blk_verify_command(unsigned char *cmd, fmode_t has_write_perm); | 1015 | extern int blk_verify_command(unsigned char *cmd, fmode_t has_write_perm); |
diff --git a/mm/swapfile.c b/mm/swapfile.c index 8ffdc0d23c53..74f1102e8749 100644 --- a/mm/swapfile.c +++ b/mm/swapfile.c | |||
| @@ -161,7 +161,8 @@ static int discard_swap(struct swap_info_struct *si) | |||
| 161 | } | 161 | } |
| 162 | 162 | ||
| 163 | err = blkdev_issue_discard(si->bdev, start_block, | 163 | err = blkdev_issue_discard(si->bdev, start_block, |
| 164 | nr_blocks, GFP_KERNEL); | 164 | nr_blocks, GFP_KERNEL, |
| 165 | DISCARD_FL_BARRIER); | ||
| 165 | if (err) | 166 | if (err) |
| 166 | break; | 167 | break; |
| 167 | 168 | ||
| @@ -200,7 +201,8 @@ static void discard_swap_cluster(struct swap_info_struct *si, | |||
| 200 | start_block <<= PAGE_SHIFT - 9; | 201 | start_block <<= PAGE_SHIFT - 9; |
| 201 | nr_blocks <<= PAGE_SHIFT - 9; | 202 | nr_blocks <<= PAGE_SHIFT - 9; |
| 202 | if (blkdev_issue_discard(si->bdev, start_block, | 203 | if (blkdev_issue_discard(si->bdev, start_block, |
| 203 | nr_blocks, GFP_NOIO)) | 204 | nr_blocks, GFP_NOIO, |
| 205 | DISCARD_FL_BARRIER)) | ||
| 204 | break; | 206 | break; |
| 205 | } | 207 | } |
| 206 | 208 | ||
