提交 4d89d377 编写于 作者: J Jeff Mahoney 提交者: Chris Mason

btrfs: btrfs_issue_discard ensure offset/length are aligned to sector boundaries

It's possible, though unexpected, to pass unaligned offsets and lengths
to btrfs_issue_discard.  We then shift the offset/length values to sector
units.  If an unaligned offset has been passed, it will result in the
entire sector being discarded, possibly losing data.  An unaligned
length is safe but we'll end up returning an inaccurate number of
discarded bytes.

This patch aligns the offset to the 512B boundary, adjusts the length,
and warns, since we shouldn't be discarding on an offset that isn't
aligned with our sector size.
Signed-off-by: NJeff Mahoney <jeffm@suse.com>
Reviewed-by: NFilipe Manana <fdmanana@suse.com>
Tested-by: NFilipe Manana <fdmanana@suse.com>
Signed-off-by: NChris Mason <clm@fb.com>
上级 d04c6b88
...@@ -1887,12 +1887,21 @@ static int btrfs_issue_discard(struct block_device *bdev, u64 start, u64 len, ...@@ -1887,12 +1887,21 @@ static int btrfs_issue_discard(struct block_device *bdev, u64 start, u64 len,
u64 *discarded_bytes) u64 *discarded_bytes)
{ {
int ret = 0; int ret = 0;
u64 aligned_start = ALIGN(start, 1 << 9);
*discarded_bytes = 0; if (WARN_ON(start != aligned_start)) {
ret = blkdev_issue_discard(bdev, start >> 9, len >> 9, GFP_NOFS, 0); len -= aligned_start - start;
if (!ret) len = round_down(len, 1 << 9);
*discarded_bytes = len; start = aligned_start;
}
*discarded_bytes = 0;
if (len) {
ret = blkdev_issue_discard(bdev, start >> 9, len >> 9,
GFP_NOFS, 0);
if (!ret)
*discarded_bytes = len;
}
return ret; return ret;
} }
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册