diff options
Diffstat (limited to 'fs')
-rw-r--r-- | fs/buffer.c | 26 |
1 files changed, 12 insertions, 14 deletions
diff --git a/fs/buffer.c b/fs/buffer.c index 3588a80854b2..e442a26e80f7 100644 --- a/fs/buffer.c +++ b/fs/buffer.c | |||
@@ -2956,7 +2956,7 @@ static void end_bio_bh_io_sync(struct bio *bio, int err) | |||
2956 | 2956 | ||
2957 | /* | 2957 | /* |
2958 | * This allows us to do IO even on the odd last sectors | 2958 | * This allows us to do IO even on the odd last sectors |
2959 | * of a device, even if the bh block size is some multiple | 2959 | * of a device, even if the block size is some multiple |
2960 | * of the physical sector size. | 2960 | * of the physical sector size. |
2961 | * | 2961 | * |
2962 | * We'll just truncate the bio to the size of the device, | 2962 | * We'll just truncate the bio to the size of the device, |
@@ -2966,10 +2966,11 @@ static void end_bio_bh_io_sync(struct bio *bio, int err) | |||
2966 | * errors, this only handles the "we need to be able to | 2966 | * errors, this only handles the "we need to be able to |
2967 | * do IO at the final sector" case. | 2967 | * do IO at the final sector" case. |
2968 | */ | 2968 | */ |
2969 | static void guard_bh_eod(int rw, struct bio *bio, struct buffer_head *bh) | 2969 | static void guard_bio_eod(int rw, struct bio *bio) |
2970 | { | 2970 | { |
2971 | sector_t maxsector; | 2971 | sector_t maxsector; |
2972 | unsigned bytes; | 2972 | struct bio_vec *bvec = &bio->bi_io_vec[bio->bi_vcnt - 1]; |
2973 | unsigned truncated_bytes; | ||
2973 | 2974 | ||
2974 | maxsector = i_size_read(bio->bi_bdev->bd_inode) >> 9; | 2975 | maxsector = i_size_read(bio->bi_bdev->bd_inode) >> 9; |
2975 | if (!maxsector) | 2976 | if (!maxsector) |
@@ -2984,23 +2985,20 @@ static void guard_bh_eod(int rw, struct bio *bio, struct buffer_head *bh) | |||
2984 | return; | 2985 | return; |
2985 | 2986 | ||
2986 | maxsector -= bio->bi_iter.bi_sector; | 2987 | maxsector -= bio->bi_iter.bi_sector; |
2987 | bytes = bio->bi_iter.bi_size; | 2988 | if (likely((bio->bi_iter.bi_size >> 9) <= maxsector)) |
2988 | if (likely((bytes >> 9) <= maxsector)) | ||
2989 | return; | 2989 | return; |
2990 | 2990 | ||
2991 | /* Uhhuh. We've got a bh that straddles the device size! */ | 2991 | /* Uhhuh. We've got a bio that straddles the device size! */ |
2992 | bytes = maxsector << 9; | 2992 | truncated_bytes = bio->bi_iter.bi_size - (maxsector << 9); |
2993 | 2993 | ||
2994 | /* Truncate the bio.. */ | 2994 | /* Truncate the bio.. */ |
2995 | bio->bi_iter.bi_size = bytes; | 2995 | bio->bi_iter.bi_size -= truncated_bytes; |
2996 | bio->bi_io_vec[0].bv_len = bytes; | 2996 | bvec->bv_len -= truncated_bytes; |
2997 | 2997 | ||
2998 | /* ..and clear the end of the buffer for reads */ | 2998 | /* ..and clear the end of the buffer for reads */ |
2999 | if ((rw & RW_MASK) == READ) { | 2999 | if ((rw & RW_MASK) == READ) { |
3000 | void *kaddr = kmap_atomic(bh->b_page); | 3000 | zero_user(bvec->bv_page, bvec->bv_offset + bvec->bv_len, |
3001 | memset(kaddr + bh_offset(bh) + bytes, 0, bh->b_size - bytes); | 3001 | truncated_bytes); |
3002 | kunmap_atomic(kaddr); | ||
3003 | flush_dcache_page(bh->b_page); | ||
3004 | } | 3002 | } |
3005 | } | 3003 | } |
3006 | 3004 | ||
@@ -3041,7 +3039,7 @@ int _submit_bh(int rw, struct buffer_head *bh, unsigned long bio_flags) | |||
3041 | bio->bi_flags |= bio_flags; | 3039 | bio->bi_flags |= bio_flags; |
3042 | 3040 | ||
3043 | /* Take care of bh's that straddle the end of the device */ | 3041 | /* Take care of bh's that straddle the end of the device */ |
3044 | guard_bh_eod(rw, bio, bh); | 3042 | guard_bio_eod(rw, bio); |
3045 | 3043 | ||
3046 | if (buffer_meta(bh)) | 3044 | if (buffer_meta(bh)) |
3047 | rw |= REQ_META; | 3045 | rw |= REQ_META; |