aboutsummaryrefslogtreecommitdiffstats
path: root/fs/btrfs
diff options
context:
space:
mode:
authorChris Mason <chris.mason@oracle.com>2012-01-31 20:27:41 -0500
committerChris Mason <chris.mason@oracle.com>2012-01-31 20:27:41 -0500
commitd98456fcafa6f3fd1985f9b7429aaa3531c6bfa0 (patch)
tree7cf01b839f68a612b8d612e7eb6315955f792d73 /fs/btrfs
parent9998eb703490589c3e8f1bf09b15203156776edb (diff)
Btrfs: don't reserve data with extents locked in btrfs_fallocate
btrfs_fallocate tries to allocate space only if ranges in the file don't already exist. But the enospc checks it does are not allowed with extents locked. Signed-off-by: Chris Mason <chris.mason@oracle.com>
Diffstat (limited to 'fs/btrfs')
-rw-r--r--fs/btrfs/file.c25
1 files changed, 10 insertions, 15 deletions
diff --git a/fs/btrfs/file.c b/fs/btrfs/file.c
index 0f61e11a2998..0621a3a7d5d1 100644
--- a/fs/btrfs/file.c
+++ b/fs/btrfs/file.c
@@ -1604,6 +1604,14 @@ static long btrfs_fallocate(struct file *file, int mode,
1604 return -EOPNOTSUPP; 1604 return -EOPNOTSUPP;
1605 1605
1606 /* 1606 /*
1607 * Make sure we have enough space before we do the
1608 * allocation.
1609 */
1610 ret = btrfs_check_data_free_space(inode, len);
1611 if (ret)
1612 return ret;
1613
1614 /*
1607 * wait for ordered IO before we have any locks. We'll loop again 1615 * wait for ordered IO before we have any locks. We'll loop again
1608 * below with the locks held. 1616 * below with the locks held.
1609 */ 1617 */
@@ -1666,27 +1674,12 @@ static long btrfs_fallocate(struct file *file, int mode,
1666 if (em->block_start == EXTENT_MAP_HOLE || 1674 if (em->block_start == EXTENT_MAP_HOLE ||
1667 (cur_offset >= inode->i_size && 1675 (cur_offset >= inode->i_size &&
1668 !test_bit(EXTENT_FLAG_PREALLOC, &em->flags))) { 1676 !test_bit(EXTENT_FLAG_PREALLOC, &em->flags))) {
1669
1670 /*
1671 * Make sure we have enough space before we do the
1672 * allocation.
1673 */
1674 ret = btrfs_check_data_free_space(inode, last_byte -
1675 cur_offset);
1676 if (ret) {
1677 free_extent_map(em);
1678 break;
1679 }
1680
1681 ret = btrfs_prealloc_file_range(inode, mode, cur_offset, 1677 ret = btrfs_prealloc_file_range(inode, mode, cur_offset,
1682 last_byte - cur_offset, 1678 last_byte - cur_offset,
1683 1 << inode->i_blkbits, 1679 1 << inode->i_blkbits,
1684 offset + len, 1680 offset + len,
1685 &alloc_hint); 1681 &alloc_hint);
1686 1682
1687 /* Let go of our reservation. */
1688 btrfs_free_reserved_data_space(inode, last_byte -
1689 cur_offset);
1690 if (ret < 0) { 1683 if (ret < 0) {
1691 free_extent_map(em); 1684 free_extent_map(em);
1692 break; 1685 break;
@@ -1714,6 +1707,8 @@ static long btrfs_fallocate(struct file *file, int mode,
1714 &cached_state, GFP_NOFS); 1707 &cached_state, GFP_NOFS);
1715out: 1708out:
1716 mutex_unlock(&inode->i_mutex); 1709 mutex_unlock(&inode->i_mutex);
1710 /* Let go of our reservation. */
1711 btrfs_free_reserved_data_space(inode, len);
1717 return ret; 1712 return ret;
1718} 1713}
1719 1714