diff options
author | Chris Mason <chris.mason@oracle.com> | 2008-02-08 13:49:28 -0500 |
---|---|---|
committer | Chris Mason <chris.mason@oracle.com> | 2008-09-25 11:04:00 -0400 |
commit | 9069218d448ea547dbad5f1cbd537e88d6519d66 (patch) | |
tree | fafb142a36805f26678bb4e7a0ebe50d8c7f859a /fs/btrfs/inode.c | |
parent | a86c12c73d982c545a37a8ecdd66528ab260b770 (diff) |
Btrfs: Fix i_blocks accounting
Now that delayed allocation accounting works, i_blocks accounting is changed
to only modify i_blocks when extents inserted or removed.
The fillattr call is changed to include the delayed allocation byte count
in the i_blocks result.
Signed-off-by: Chris Mason <chris.mason@oracle.com>
Diffstat (limited to 'fs/btrfs/inode.c')
-rw-r--r-- | fs/btrfs/inode.c | 43 |
1 files changed, 29 insertions, 14 deletions
diff --git a/fs/btrfs/inode.c b/fs/btrfs/inode.c index 44fc94fee2c4..913ab128eee1 100644 --- a/fs/btrfs/inode.c +++ b/fs/btrfs/inode.c | |||
@@ -134,6 +134,7 @@ static int cow_file_range(struct inode *inode, u64 start, u64 end) | |||
134 | ret = btrfs_insert_file_extent(trans, root, inode->i_ino, | 134 | ret = btrfs_insert_file_extent(trans, root, inode->i_ino, |
135 | start, ins.objectid, ins.offset, | 135 | start, ins.objectid, ins.offset, |
136 | ins.offset); | 136 | ins.offset); |
137 | inode->i_blocks += ins.offset >> 9; | ||
137 | btrfs_check_file(root, inode); | 138 | btrfs_check_file(root, inode); |
138 | num_bytes -= cur_alloc_size; | 139 | num_bytes -= cur_alloc_size; |
139 | alloc_hint = ins.objectid + ins.offset; | 140 | alloc_hint = ins.objectid + ins.offset; |
@@ -142,6 +143,7 @@ static int cow_file_range(struct inode *inode, u64 start, u64 end) | |||
142 | btrfs_drop_extent_cache(inode, orig_start, | 143 | btrfs_drop_extent_cache(inode, orig_start, |
143 | orig_start + orig_num_bytes - 1); | 144 | orig_start + orig_num_bytes - 1); |
144 | btrfs_add_ordered_inode(inode); | 145 | btrfs_add_ordered_inode(inode); |
146 | btrfs_update_inode(trans, root, inode); | ||
145 | out: | 147 | out: |
146 | btrfs_end_transaction(trans, root); | 148 | btrfs_end_transaction(trans, root); |
147 | return ret; | 149 | return ret; |
@@ -265,6 +267,7 @@ int btrfs_set_bit_hook(struct inode *inode, u64 start, u64 end, | |||
265 | if (!(old & EXTENT_DELALLOC) && (bits & EXTENT_DELALLOC)) { | 267 | if (!(old & EXTENT_DELALLOC) && (bits & EXTENT_DELALLOC)) { |
266 | struct btrfs_root *root = BTRFS_I(inode)->root; | 268 | struct btrfs_root *root = BTRFS_I(inode)->root; |
267 | spin_lock(&root->fs_info->delalloc_lock); | 269 | spin_lock(&root->fs_info->delalloc_lock); |
270 | BTRFS_I(inode)->delalloc_bytes += end - start + 1; | ||
268 | root->fs_info->delalloc_bytes += end - start + 1; | 271 | root->fs_info->delalloc_bytes += end - start + 1; |
269 | spin_unlock(&root->fs_info->delalloc_lock); | 272 | spin_unlock(&root->fs_info->delalloc_lock); |
270 | } | 273 | } |
@@ -281,8 +284,10 @@ int btrfs_clear_bit_hook(struct inode *inode, u64 start, u64 end, | |||
281 | printk("warning: delalloc account %Lu %Lu\n", | 284 | printk("warning: delalloc account %Lu %Lu\n", |
282 | end - start + 1, root->fs_info->delalloc_bytes); | 285 | end - start + 1, root->fs_info->delalloc_bytes); |
283 | root->fs_info->delalloc_bytes = 0; | 286 | root->fs_info->delalloc_bytes = 0; |
287 | BTRFS_I(inode)->delalloc_bytes = 0; | ||
284 | } else { | 288 | } else { |
285 | root->fs_info->delalloc_bytes -= end - start + 1; | 289 | root->fs_info->delalloc_bytes -= end - start + 1; |
290 | BTRFS_I(inode)->delalloc_bytes -= end - start + 1; | ||
286 | } | 291 | } |
287 | spin_unlock(&root->fs_info->delalloc_lock); | 292 | spin_unlock(&root->fs_info->delalloc_lock); |
288 | } | 293 | } |
@@ -833,32 +838,37 @@ search_again: | |||
833 | btrfs_set_file_extent_num_bytes(leaf, fi, | 838 | btrfs_set_file_extent_num_bytes(leaf, fi, |
834 | extent_num_bytes); | 839 | extent_num_bytes); |
835 | num_dec = (orig_num_bytes - | 840 | num_dec = (orig_num_bytes - |
836 | extent_num_bytes) >> 9; | 841 | extent_num_bytes); |
837 | if (extent_start != 0) { | 842 | if (extent_start != 0) |
838 | inode->i_blocks -= num_dec; | 843 | dec_i_blocks(inode, num_dec); |
839 | } | ||
840 | btrfs_mark_buffer_dirty(leaf); | 844 | btrfs_mark_buffer_dirty(leaf); |
841 | } else { | 845 | } else { |
842 | extent_num_bytes = | 846 | extent_num_bytes = |
843 | btrfs_file_extent_disk_num_bytes(leaf, | 847 | btrfs_file_extent_disk_num_bytes(leaf, |
844 | fi); | 848 | fi); |
845 | /* FIXME blocksize != 4096 */ | 849 | /* FIXME blocksize != 4096 */ |
846 | num_dec = btrfs_file_extent_num_bytes(leaf, | 850 | num_dec = btrfs_file_extent_num_bytes(leaf, fi); |
847 | fi) >> 9; | ||
848 | if (extent_start != 0) { | 851 | if (extent_start != 0) { |
849 | found_extent = 1; | 852 | found_extent = 1; |
850 | inode->i_blocks -= num_dec; | 853 | dec_i_blocks(inode, num_dec); |
851 | } | 854 | } |
852 | root_gen = btrfs_header_generation(leaf); | 855 | root_gen = btrfs_header_generation(leaf); |
853 | root_owner = btrfs_header_owner(leaf); | 856 | root_owner = btrfs_header_owner(leaf); |
854 | } | 857 | } |
855 | } else if (extent_type == BTRFS_FILE_EXTENT_INLINE && | 858 | } else if (extent_type == BTRFS_FILE_EXTENT_INLINE) { |
856 | !del_item) { | 859 | if (!del_item) { |
857 | u32 newsize = inode->i_size - found_key.offset; | 860 | u32 newsize = inode->i_size - found_key.offset; |
858 | newsize = btrfs_file_extent_calc_inline_size(newsize); | 861 | dec_i_blocks(inode, item_end + 1 - |
859 | ret = btrfs_truncate_item(trans, root, path, | 862 | found_key.offset - newsize); |
860 | newsize, 1); | 863 | newsize = |
861 | BUG_ON(ret); | 864 | btrfs_file_extent_calc_inline_size(newsize); |
865 | ret = btrfs_truncate_item(trans, root, path, | ||
866 | newsize, 1); | ||
867 | BUG_ON(ret); | ||
868 | } else { | ||
869 | dec_i_blocks(inode, item_end + 1 - | ||
870 | found_key.offset); | ||
871 | } | ||
862 | } | 872 | } |
863 | delete: | 873 | delete: |
864 | if (del_item) { | 874 | if (del_item) { |
@@ -1222,6 +1232,7 @@ static int btrfs_init_locked_inode(struct inode *inode, void *p) | |||
1222 | struct btrfs_iget_args *args = p; | 1232 | struct btrfs_iget_args *args = p; |
1223 | inode->i_ino = args->ino; | 1233 | inode->i_ino = args->ino; |
1224 | BTRFS_I(inode)->root = args->root; | 1234 | BTRFS_I(inode)->root = args->root; |
1235 | BTRFS_I(inode)->delalloc_bytes = 0; | ||
1225 | extent_map_tree_init(&BTRFS_I(inode)->extent_tree, GFP_NOFS); | 1236 | extent_map_tree_init(&BTRFS_I(inode)->extent_tree, GFP_NOFS); |
1226 | extent_io_tree_init(&BTRFS_I(inode)->io_tree, | 1237 | extent_io_tree_init(&BTRFS_I(inode)->io_tree, |
1227 | inode->i_mapping, GFP_NOFS); | 1238 | inode->i_mapping, GFP_NOFS); |
@@ -1528,6 +1539,7 @@ static struct inode *btrfs_new_inode(struct btrfs_trans_handle *trans, | |||
1528 | extent_map_tree_init(&BTRFS_I(inode)->extent_tree, GFP_NOFS); | 1539 | extent_map_tree_init(&BTRFS_I(inode)->extent_tree, GFP_NOFS); |
1529 | extent_io_tree_init(&BTRFS_I(inode)->io_tree, | 1540 | extent_io_tree_init(&BTRFS_I(inode)->io_tree, |
1530 | inode->i_mapping, GFP_NOFS); | 1541 | inode->i_mapping, GFP_NOFS); |
1542 | BTRFS_I(inode)->delalloc_bytes = 0; | ||
1531 | BTRFS_I(inode)->root = root; | 1543 | BTRFS_I(inode)->root = root; |
1532 | 1544 | ||
1533 | if (mode & S_IFDIR) | 1545 | if (mode & S_IFDIR) |
@@ -1746,6 +1758,7 @@ static int btrfs_create(struct inode *dir, struct dentry *dentry, | |||
1746 | extent_map_tree_init(&BTRFS_I(inode)->extent_tree, GFP_NOFS); | 1758 | extent_map_tree_init(&BTRFS_I(inode)->extent_tree, GFP_NOFS); |
1747 | extent_io_tree_init(&BTRFS_I(inode)->io_tree, | 1759 | extent_io_tree_init(&BTRFS_I(inode)->io_tree, |
1748 | inode->i_mapping, GFP_NOFS); | 1760 | inode->i_mapping, GFP_NOFS); |
1761 | BTRFS_I(inode)->delalloc_bytes = 0; | ||
1749 | BTRFS_I(inode)->io_tree.ops = &btrfs_extent_io_ops; | 1762 | BTRFS_I(inode)->io_tree.ops = &btrfs_extent_io_ops; |
1750 | } | 1763 | } |
1751 | dir->i_sb->s_dirt = 1; | 1764 | dir->i_sb->s_dirt = 1; |
@@ -2797,6 +2810,7 @@ static int btrfs_getattr(struct vfsmount *mnt, | |||
2797 | struct inode *inode = dentry->d_inode; | 2810 | struct inode *inode = dentry->d_inode; |
2798 | generic_fillattr(inode, stat); | 2811 | generic_fillattr(inode, stat); |
2799 | stat->blksize = PAGE_CACHE_SIZE; | 2812 | stat->blksize = PAGE_CACHE_SIZE; |
2813 | stat->blocks = inode->i_blocks + (BTRFS_I(inode)->delalloc_bytes >> 9); | ||
2800 | return 0; | 2814 | return 0; |
2801 | } | 2815 | } |
2802 | 2816 | ||
@@ -2912,6 +2926,7 @@ static int btrfs_symlink(struct inode *dir, struct dentry *dentry, | |||
2912 | extent_map_tree_init(&BTRFS_I(inode)->extent_tree, GFP_NOFS); | 2926 | extent_map_tree_init(&BTRFS_I(inode)->extent_tree, GFP_NOFS); |
2913 | extent_io_tree_init(&BTRFS_I(inode)->io_tree, | 2927 | extent_io_tree_init(&BTRFS_I(inode)->io_tree, |
2914 | inode->i_mapping, GFP_NOFS); | 2928 | inode->i_mapping, GFP_NOFS); |
2929 | BTRFS_I(inode)->delalloc_bytes = 0; | ||
2915 | BTRFS_I(inode)->io_tree.ops = &btrfs_extent_io_ops; | 2930 | BTRFS_I(inode)->io_tree.ops = &btrfs_extent_io_ops; |
2916 | } | 2931 | } |
2917 | dir->i_sb->s_dirt = 1; | 2932 | dir->i_sb->s_dirt = 1; |