diff options
author | Josef Bacik <josef@redhat.com> | 2011-08-18 10:36:39 -0400 |
---|---|---|
committer | Chris Mason <chris.mason@oracle.com> | 2011-08-18 10:36:39 -0400 |
commit | f1e490a7ebe41e06324abbbcd86005b0af02a375 (patch) | |
tree | c8ffa501d66399656377b7b4c633aaadd1604467 /fs | |
parent | f81c9cdc567cd3160ff9e64868d9a1a7ee226480 (diff) |
Btrfs: set i_size properly when fallocating and we already
xfstests exposed a problem with preallocate when it fallocates a range that
already has an extent. We don't set the new i_size properly because we see that
we already have an extent. This isn't right and we should update i_size if the
space already exists. With this patch we now pass xfstests 075. Thanks,
Signed-off-by: Josef Bacik <josef@redhat.com>
Signed-off-by: Chris Mason <chris.mason@oracle.com>
Diffstat (limited to 'fs')
-rw-r--r-- | fs/btrfs/file.c | 14 |
1 files changed, 14 insertions, 0 deletions
diff --git a/fs/btrfs/file.c b/fs/btrfs/file.c index 0705d15542c6..15e5a1cd8764 100644 --- a/fs/btrfs/file.c +++ b/fs/btrfs/file.c | |||
@@ -1631,11 +1631,15 @@ static long btrfs_fallocate(struct file *file, int mode, | |||
1631 | 1631 | ||
1632 | cur_offset = alloc_start; | 1632 | cur_offset = alloc_start; |
1633 | while (1) { | 1633 | while (1) { |
1634 | u64 actual_end; | ||
1635 | |||
1634 | em = btrfs_get_extent(inode, NULL, 0, cur_offset, | 1636 | em = btrfs_get_extent(inode, NULL, 0, cur_offset, |
1635 | alloc_end - cur_offset, 0); | 1637 | alloc_end - cur_offset, 0); |
1636 | BUG_ON(IS_ERR_OR_NULL(em)); | 1638 | BUG_ON(IS_ERR_OR_NULL(em)); |
1637 | last_byte = min(extent_map_end(em), alloc_end); | 1639 | last_byte = min(extent_map_end(em), alloc_end); |
1640 | actual_end = min_t(u64, extent_map_end(em), offset + len); | ||
1638 | last_byte = (last_byte + mask) & ~mask; | 1641 | last_byte = (last_byte + mask) & ~mask; |
1642 | |||
1639 | if (em->block_start == EXTENT_MAP_HOLE || | 1643 | if (em->block_start == EXTENT_MAP_HOLE || |
1640 | (cur_offset >= inode->i_size && | 1644 | (cur_offset >= inode->i_size && |
1641 | !test_bit(EXTENT_FLAG_PREALLOC, &em->flags))) { | 1645 | !test_bit(EXTENT_FLAG_PREALLOC, &em->flags))) { |
@@ -1648,6 +1652,16 @@ static long btrfs_fallocate(struct file *file, int mode, | |||
1648 | free_extent_map(em); | 1652 | free_extent_map(em); |
1649 | break; | 1653 | break; |
1650 | } | 1654 | } |
1655 | } else if (actual_end > inode->i_size && | ||
1656 | !(mode & FALLOC_FL_KEEP_SIZE)) { | ||
1657 | /* | ||
1658 | * We didn't need to allocate any more space, but we | ||
1659 | * still extended the size of the file so we need to | ||
1660 | * update i_size. | ||
1661 | */ | ||
1662 | inode->i_ctime = CURRENT_TIME; | ||
1663 | i_size_write(inode, actual_end); | ||
1664 | btrfs_ordered_update_i_size(inode, actual_end, NULL); | ||
1651 | } | 1665 | } |
1652 | free_extent_map(em); | 1666 | free_extent_map(em); |
1653 | 1667 | ||