diff options
Diffstat (limited to 'fs/btrfs/inode.c')
-rw-r--r-- | fs/btrfs/inode.c | 13 |
1 files changed, 9 insertions, 4 deletions
diff --git a/fs/btrfs/inode.c b/fs/btrfs/inode.c index f46c57276844..5606361b5f04 100644 --- a/fs/btrfs/inode.c +++ b/fs/btrfs/inode.c | |||
@@ -5800,7 +5800,7 @@ out_fail: | |||
5800 | } | 5800 | } |
5801 | 5801 | ||
5802 | static int prealloc_file_range(struct inode *inode, u64 start, u64 end, | 5802 | static int prealloc_file_range(struct inode *inode, u64 start, u64 end, |
5803 | u64 alloc_hint, int mode) | 5803 | u64 alloc_hint, int mode, loff_t actual_len) |
5804 | { | 5804 | { |
5805 | struct btrfs_trans_handle *trans; | 5805 | struct btrfs_trans_handle *trans; |
5806 | struct btrfs_root *root = BTRFS_I(inode)->root; | 5806 | struct btrfs_root *root = BTRFS_I(inode)->root; |
@@ -5809,6 +5809,7 @@ static int prealloc_file_range(struct inode *inode, u64 start, u64 end, | |||
5809 | u64 cur_offset = start; | 5809 | u64 cur_offset = start; |
5810 | u64 num_bytes = end - start; | 5810 | u64 num_bytes = end - start; |
5811 | int ret = 0; | 5811 | int ret = 0; |
5812 | u64 i_size; | ||
5812 | 5813 | ||
5813 | while (num_bytes > 0) { | 5814 | while (num_bytes > 0) { |
5814 | alloc_size = min(num_bytes, root->fs_info->max_extent); | 5815 | alloc_size = min(num_bytes, root->fs_info->max_extent); |
@@ -5847,8 +5848,12 @@ static int prealloc_file_range(struct inode *inode, u64 start, u64 end, | |||
5847 | BTRFS_I(inode)->flags |= BTRFS_INODE_PREALLOC; | 5848 | BTRFS_I(inode)->flags |= BTRFS_INODE_PREALLOC; |
5848 | if (!(mode & FALLOC_FL_KEEP_SIZE) && | 5849 | if (!(mode & FALLOC_FL_KEEP_SIZE) && |
5849 | cur_offset > inode->i_size) { | 5850 | cur_offset > inode->i_size) { |
5850 | i_size_write(inode, cur_offset); | 5851 | if (cur_offset > actual_len) |
5851 | btrfs_ordered_update_i_size(inode, cur_offset, NULL); | 5852 | i_size = actual_len; |
5853 | else | ||
5854 | i_size = cur_offset; | ||
5855 | i_size_write(inode, i_size); | ||
5856 | btrfs_ordered_update_i_size(inode, i_size, NULL); | ||
5852 | } | 5857 | } |
5853 | 5858 | ||
5854 | ret = btrfs_update_inode(trans, root, inode); | 5859 | ret = btrfs_update_inode(trans, root, inode); |
@@ -5941,7 +5946,7 @@ static long btrfs_fallocate(struct inode *inode, int mode, | |||
5941 | !test_bit(EXTENT_FLAG_PREALLOC, &em->flags))) { | 5946 | !test_bit(EXTENT_FLAG_PREALLOC, &em->flags))) { |
5942 | ret = prealloc_file_range(inode, | 5947 | ret = prealloc_file_range(inode, |
5943 | cur_offset, last_byte, | 5948 | cur_offset, last_byte, |
5944 | alloc_hint, mode); | 5949 | alloc_hint, mode, offset+len); |
5945 | if (ret < 0) { | 5950 | if (ret < 0) { |
5946 | free_extent_map(em); | 5951 | free_extent_map(em); |
5947 | break; | 5952 | break; |