Commit 26176e7c authored by Miao Xie's avatar Miao Xie Committed by Chris Mason

Btrfs: restructure btrfs_run_defrag_inodes()

This patch restructure btrfs_run_defrag_inodes() and make the code of the auto
defragment more readable.
Signed-off-by: default avatarMiao Xie <miaox@cn.fujitsu.com>
Signed-off-by: default avatarChris Mason <chris.mason@fusionio.com>
parent 8ddc4734
...@@ -3510,6 +3510,7 @@ void btrfs_auto_defrag_exit(void); ...@@ -3510,6 +3510,7 @@ void btrfs_auto_defrag_exit(void);
int btrfs_add_inode_defrag(struct btrfs_trans_handle *trans, int btrfs_add_inode_defrag(struct btrfs_trans_handle *trans,
struct inode *inode); struct inode *inode);
int btrfs_run_defrag_inodes(struct btrfs_fs_info *fs_info); int btrfs_run_defrag_inodes(struct btrfs_fs_info *fs_info);
void btrfs_cleanup_defrag_inodes(struct btrfs_fs_info *fs_info);
int btrfs_sync_file(struct file *file, loff_t start, loff_t end, int datasync); int btrfs_sync_file(struct file *file, loff_t start, loff_t end, int datasync);
void btrfs_drop_extent_cache(struct inode *inode, u64 start, u64 end, void btrfs_drop_extent_cache(struct inode *inode, u64 start, u64 end,
int skip_pinned); int skip_pinned);
......
...@@ -3329,7 +3329,7 @@ int close_ctree(struct btrfs_root *root) ...@@ -3329,7 +3329,7 @@ int close_ctree(struct btrfs_root *root)
(atomic_read(&fs_info->defrag_running) == 0)); (atomic_read(&fs_info->defrag_running) == 0));
/* clear out the rbtree of defraggable inodes */ /* clear out the rbtree of defraggable inodes */
btrfs_run_defrag_inodes(fs_info); btrfs_cleanup_defrag_inodes(fs_info);
if (!(fs_info->sb->s_flags & MS_RDONLY)) { if (!(fs_info->sb->s_flags & MS_RDONLY)) {
ret = btrfs_commit_super(root); ret = btrfs_commit_super(root);
......
...@@ -216,11 +216,11 @@ void btrfs_requeue_inode_defrag(struct inode *inode, ...@@ -216,11 +216,11 @@ void btrfs_requeue_inode_defrag(struct inode *inode,
} }
/* /*
* must be called with the defrag_inodes lock held * pick the defragable inode that we want, if it doesn't exist, we will get
* the next one.
*/ */
struct inode_defrag *btrfs_find_defrag_inode(struct btrfs_fs_info *info, static struct inode_defrag *
u64 root, u64 ino, btrfs_pick_defrag_inode(struct btrfs_fs_info *fs_info, u64 root, u64 ino)
struct rb_node **next)
{ {
struct inode_defrag *entry = NULL; struct inode_defrag *entry = NULL;
struct inode_defrag tmp; struct inode_defrag tmp;
...@@ -231,7 +231,8 @@ struct inode_defrag *btrfs_find_defrag_inode(struct btrfs_fs_info *info, ...@@ -231,7 +231,8 @@ struct inode_defrag *btrfs_find_defrag_inode(struct btrfs_fs_info *info,
tmp.ino = ino; tmp.ino = ino;
tmp.root = root; tmp.root = root;
p = info->defrag_inodes.rb_node; spin_lock(&fs_info->defrag_inodes_lock);
p = fs_info->defrag_inodes.rb_node;
while (p) { while (p) {
parent = p; parent = p;
entry = rb_entry(parent, struct inode_defrag, rb_node); entry = rb_entry(parent, struct inode_defrag, rb_node);
...@@ -242,104 +243,91 @@ struct inode_defrag *btrfs_find_defrag_inode(struct btrfs_fs_info *info, ...@@ -242,104 +243,91 @@ struct inode_defrag *btrfs_find_defrag_inode(struct btrfs_fs_info *info,
else if (ret > 0) else if (ret > 0)
p = parent->rb_right; p = parent->rb_right;
else else
return entry; goto out;
} }
if (next) { if (parent && __compare_inode_defrag(&tmp, entry) > 0) {
while (parent && __compare_inode_defrag(&tmp, entry) > 0) {
parent = rb_next(parent); parent = rb_next(parent);
if (parent)
entry = rb_entry(parent, struct inode_defrag, rb_node); entry = rb_entry(parent, struct inode_defrag, rb_node);
else
entry = NULL;
} }
*next = parent; out:
} if (entry)
return NULL; rb_erase(parent, &fs_info->defrag_inodes);
spin_unlock(&fs_info->defrag_inodes_lock);
return entry;
} }
/* void btrfs_cleanup_defrag_inodes(struct btrfs_fs_info *fs_info)
* run through the list of inodes in the FS that need
* defragging
*/
int btrfs_run_defrag_inodes(struct btrfs_fs_info *fs_info)
{ {
struct inode_defrag *defrag; struct inode_defrag *defrag;
struct btrfs_root *inode_root; struct rb_node *node;
struct inode *inode;
struct rb_node *n;
struct btrfs_key key;
struct btrfs_ioctl_defrag_range_args range;
u64 first_ino = 0;
u64 root_objectid = 0;
int num_defrag;
int defrag_batch = 1024;
memset(&range, 0, sizeof(range));
range.len = (u64)-1;
atomic_inc(&fs_info->defrag_running);
spin_lock(&fs_info->defrag_inodes_lock); spin_lock(&fs_info->defrag_inodes_lock);
while(1) { node = rb_first(&fs_info->defrag_inodes);
n = NULL; while (node) {
rb_erase(node, &fs_info->defrag_inodes);
defrag = rb_entry(node, struct inode_defrag, rb_node);
kmem_cache_free(btrfs_inode_defrag_cachep, defrag);
/* find an inode to defrag */ if (need_resched()) {
defrag = btrfs_find_defrag_inode(fs_info, root_objectid, spin_unlock(&fs_info->defrag_inodes_lock);
first_ino, &n); cond_resched();
if (!defrag) { spin_lock(&fs_info->defrag_inodes_lock);
if (n) {
defrag = rb_entry(n, struct inode_defrag,
rb_node);
} else if (root_objectid || first_ino) {
root_objectid = 0;
first_ino = 0;
continue;
} else {
break;
}
} }
/* remove it from the rbtree */ node = rb_first(&fs_info->defrag_inodes);
first_ino = defrag->ino + 1; }
root_objectid = defrag->root; spin_unlock(&fs_info->defrag_inodes_lock);
rb_erase(&defrag->rb_node, &fs_info->defrag_inodes); }
if (btrfs_fs_closing(fs_info)) #define BTRFS_DEFRAG_BATCH 1024
goto next_free;
spin_unlock(&fs_info->defrag_inodes_lock); static int __btrfs_run_defrag_inode(struct btrfs_fs_info *fs_info,
struct inode_defrag *defrag)
{
struct btrfs_root *inode_root;
struct inode *inode;
struct btrfs_key key;
struct btrfs_ioctl_defrag_range_args range;
int num_defrag;
/* get the inode */ /* get the inode */
key.objectid = defrag->root; key.objectid = defrag->root;
btrfs_set_key_type(&key, BTRFS_ROOT_ITEM_KEY); btrfs_set_key_type(&key, BTRFS_ROOT_ITEM_KEY);
key.offset = (u64)-1; key.offset = (u64)-1;
inode_root = btrfs_read_fs_root_no_name(fs_info, &key); inode_root = btrfs_read_fs_root_no_name(fs_info, &key);
if (IS_ERR(inode_root)) if (IS_ERR(inode_root)) {
goto next; kmem_cache_free(btrfs_inode_defrag_cachep, defrag);
return PTR_ERR(inode_root);
}
key.objectid = defrag->ino; key.objectid = defrag->ino;
btrfs_set_key_type(&key, BTRFS_INODE_ITEM_KEY); btrfs_set_key_type(&key, BTRFS_INODE_ITEM_KEY);
key.offset = 0; key.offset = 0;
inode = btrfs_iget(fs_info->sb, &key, inode_root, NULL); inode = btrfs_iget(fs_info->sb, &key, inode_root, NULL);
if (IS_ERR(inode)) if (IS_ERR(inode)) {
goto next; kmem_cache_free(btrfs_inode_defrag_cachep, defrag);
return PTR_ERR(inode);
}
/* do a chunk of defrag */ /* do a chunk of defrag */
clear_bit(BTRFS_INODE_IN_DEFRAG, &BTRFS_I(inode)->runtime_flags); clear_bit(BTRFS_INODE_IN_DEFRAG, &BTRFS_I(inode)->runtime_flags);
memset(&range, 0, sizeof(range));
range.len = (u64)-1;
range.start = defrag->last_offset; range.start = defrag->last_offset;
num_defrag = btrfs_defrag_file(inode, NULL, &range, defrag->transid, num_defrag = btrfs_defrag_file(inode, NULL, &range, defrag->transid,
defrag_batch); BTRFS_DEFRAG_BATCH);
/* /*
* if we filled the whole defrag batch, there * if we filled the whole defrag batch, there
* must be more work to do. Queue this defrag * must be more work to do. Queue this defrag
* again * again
*/ */
if (num_defrag == defrag_batch) { if (num_defrag == BTRFS_DEFRAG_BATCH) {
defrag->last_offset = range.start; defrag->last_offset = range.start;
btrfs_requeue_inode_defrag(inode, defrag); btrfs_requeue_inode_defrag(inode, defrag);
/*
* we don't want to kfree defrag, we added it back to
* the rbtree
*/
defrag = NULL;
} else if (defrag->last_offset && !defrag->cycled) { } else if (defrag->last_offset && !defrag->cycled) {
/* /*
* we didn't fill our defrag batch, but * we didn't fill our defrag batch, but
...@@ -349,18 +337,47 @@ int btrfs_run_defrag_inodes(struct btrfs_fs_info *fs_info) ...@@ -349,18 +337,47 @@ int btrfs_run_defrag_inodes(struct btrfs_fs_info *fs_info)
defrag->last_offset = 0; defrag->last_offset = 0;
defrag->cycled = 1; defrag->cycled = 1;
btrfs_requeue_inode_defrag(inode, defrag); btrfs_requeue_inode_defrag(inode, defrag);
defrag = NULL; } else {
kmem_cache_free(btrfs_inode_defrag_cachep, defrag);
} }
iput(inode); iput(inode);
next: return 0;
spin_lock(&fs_info->defrag_inodes_lock); }
next_free:
if (defrag) /*
kmem_cache_free(btrfs_inode_defrag_cachep, defrag); * run through the list of inodes in the FS that need
* defragging
*/
int btrfs_run_defrag_inodes(struct btrfs_fs_info *fs_info)
{
struct inode_defrag *defrag;
u64 first_ino = 0;
u64 root_objectid = 0;
atomic_inc(&fs_info->defrag_running);
while(1) {
if (!__need_auto_defrag(fs_info->tree_root))
break;
/* find an inode to defrag */
defrag = btrfs_pick_defrag_inode(fs_info, root_objectid,
first_ino);
if (!defrag) {
if (root_objectid || first_ino) {
root_objectid = 0;
first_ino = 0;
continue;
} else {
break;
} }
spin_unlock(&fs_info->defrag_inodes_lock); }
first_ino = defrag->ino + 1;
root_objectid = defrag->root;
__btrfs_run_defrag_inode(fs_info, defrag);
}
atomic_dec(&fs_info->defrag_running); atomic_dec(&fs_info->defrag_running);
/* /*
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment