Commit df14c32b authored by William Lee Irwin III's avatar William Lee Irwin III Committed by Linus Torvalds

[PATCH] invalidate_inodes speedup

With Kirill Korotaev <kk@sw.ru>

invalidate_inodes() can take a large amount of time searching the inode
lists for inodes which belong to this fs.  Add a separate list for this
search.
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@osdl.org>
parent 1d590388
...@@ -225,6 +225,7 @@ static void hugetlbfs_delete_inode(struct inode *inode) ...@@ -225,6 +225,7 @@ static void hugetlbfs_delete_inode(struct inode *inode)
hlist_del_init(&inode->i_hash); hlist_del_init(&inode->i_hash);
list_del_init(&inode->i_list); list_del_init(&inode->i_list);
list_del_init(&inode->i_sb_list);
inode->i_state |= I_FREEING; inode->i_state |= I_FREEING;
inodes_stat.nr_inodes--; inodes_stat.nr_inodes--;
spin_unlock(&inode_lock); spin_unlock(&inode_lock);
...@@ -267,6 +268,7 @@ static void hugetlbfs_forget_inode(struct inode *inode) ...@@ -267,6 +268,7 @@ static void hugetlbfs_forget_inode(struct inode *inode)
hlist_del_init(&inode->i_hash); hlist_del_init(&inode->i_hash);
out_truncate: out_truncate:
list_del_init(&inode->i_list); list_del_init(&inode->i_list);
list_del_init(&inode->i_sb_list);
inode->i_state |= I_FREEING; inode->i_state |= I_FREEING;
inodes_stat.nr_inodes--; inodes_stat.nr_inodes--;
spin_unlock(&inode_lock); spin_unlock(&inode_lock);
......
...@@ -296,7 +296,7 @@ static void dispose_list(struct list_head *head) ...@@ -296,7 +296,7 @@ static void dispose_list(struct list_head *head)
/* /*
* Invalidate all inodes for a device. * Invalidate all inodes for a device.
*/ */
static int invalidate_list(struct list_head *head, struct super_block * sb, struct list_head * dispose) static int invalidate_list(struct list_head *head, struct list_head *dispose)
{ {
struct list_head *next; struct list_head *next;
int busy = 0, count = 0; int busy = 0, count = 0;
...@@ -309,12 +309,11 @@ static int invalidate_list(struct list_head *head, struct super_block * sb, stru ...@@ -309,12 +309,11 @@ static int invalidate_list(struct list_head *head, struct super_block * sb, stru
next = next->next; next = next->next;
if (tmp == head) if (tmp == head)
break; break;
inode = list_entry(tmp, struct inode, i_list); inode = list_entry(tmp, struct inode, i_sb_list);
if (inode->i_sb != sb)
continue;
invalidate_inode_buffers(inode); invalidate_inode_buffers(inode);
if (!atomic_read(&inode->i_count)) { if (!atomic_read(&inode->i_count)) {
hlist_del_init(&inode->i_hash); hlist_del_init(&inode->i_hash);
list_del(&inode->i_sb_list);
list_move(&inode->i_list, dispose); list_move(&inode->i_list, dispose);
inode->i_state |= I_FREEING; inode->i_state |= I_FREEING;
count++; count++;
...@@ -350,10 +349,7 @@ int invalidate_inodes(struct super_block * sb) ...@@ -350,10 +349,7 @@ int invalidate_inodes(struct super_block * sb)
down(&iprune_sem); down(&iprune_sem);
spin_lock(&inode_lock); spin_lock(&inode_lock);
busy = invalidate_list(&inode_in_use, sb, &throw_away); busy = invalidate_list(&sb->s_inodes, &throw_away);
busy |= invalidate_list(&inode_unused, sb, &throw_away);
busy |= invalidate_list(&sb->s_dirty, sb, &throw_away);
busy |= invalidate_list(&sb->s_io, sb, &throw_away);
spin_unlock(&inode_lock); spin_unlock(&inode_lock);
dispose_list(&throw_away); dispose_list(&throw_away);
...@@ -453,6 +449,7 @@ static void prune_icache(int nr_to_scan) ...@@ -453,6 +449,7 @@ static void prune_icache(int nr_to_scan)
continue; continue;
} }
hlist_del_init(&inode->i_hash); hlist_del_init(&inode->i_hash);
list_del_init(&inode->i_sb_list);
list_move(&inode->i_list, &freeable); list_move(&inode->i_list, &freeable);
inode->i_state |= I_FREEING; inode->i_state |= I_FREEING;
nr_pruned++; nr_pruned++;
...@@ -564,6 +561,7 @@ struct inode *new_inode(struct super_block *sb) ...@@ -564,6 +561,7 @@ struct inode *new_inode(struct super_block *sb)
spin_lock(&inode_lock); spin_lock(&inode_lock);
inodes_stat.nr_inodes++; inodes_stat.nr_inodes++;
list_add(&inode->i_list, &inode_in_use); list_add(&inode->i_list, &inode_in_use);
list_add(&inode->i_sb_list, &sb->s_inodes);
inode->i_ino = ++last_ino; inode->i_ino = ++last_ino;
inode->i_state = 0; inode->i_state = 0;
spin_unlock(&inode_lock); spin_unlock(&inode_lock);
...@@ -612,6 +610,7 @@ static struct inode * get_new_inode(struct super_block *sb, struct hlist_head *h ...@@ -612,6 +610,7 @@ static struct inode * get_new_inode(struct super_block *sb, struct hlist_head *h
inodes_stat.nr_inodes++; inodes_stat.nr_inodes++;
list_add(&inode->i_list, &inode_in_use); list_add(&inode->i_list, &inode_in_use);
list_add(&inode->i_sb_list, &sb->s_inodes);
hlist_add_head(&inode->i_hash, head); hlist_add_head(&inode->i_hash, head);
inode->i_state = I_LOCK|I_NEW; inode->i_state = I_LOCK|I_NEW;
spin_unlock(&inode_lock); spin_unlock(&inode_lock);
...@@ -660,6 +659,7 @@ static struct inode * get_new_inode_fast(struct super_block *sb, struct hlist_he ...@@ -660,6 +659,7 @@ static struct inode * get_new_inode_fast(struct super_block *sb, struct hlist_he
inode->i_ino = ino; inode->i_ino = ino;
inodes_stat.nr_inodes++; inodes_stat.nr_inodes++;
list_add(&inode->i_list, &inode_in_use); list_add(&inode->i_list, &inode_in_use);
list_add(&inode->i_sb_list, &sb->s_inodes);
hlist_add_head(&inode->i_hash, head); hlist_add_head(&inode->i_hash, head);
inode->i_state = I_LOCK|I_NEW; inode->i_state = I_LOCK|I_NEW;
spin_unlock(&inode_lock); spin_unlock(&inode_lock);
...@@ -996,6 +996,7 @@ void generic_delete_inode(struct inode *inode) ...@@ -996,6 +996,7 @@ void generic_delete_inode(struct inode *inode)
struct super_operations *op = inode->i_sb->s_op; struct super_operations *op = inode->i_sb->s_op;
list_del_init(&inode->i_list); list_del_init(&inode->i_list);
list_del_init(&inode->i_sb_list);
inode->i_state|=I_FREEING; inode->i_state|=I_FREEING;
inodes_stat.nr_inodes--; inodes_stat.nr_inodes--;
spin_unlock(&inode_lock); spin_unlock(&inode_lock);
...@@ -1041,6 +1042,7 @@ static void generic_forget_inode(struct inode *inode) ...@@ -1041,6 +1042,7 @@ static void generic_forget_inode(struct inode *inode)
hlist_del_init(&inode->i_hash); hlist_del_init(&inode->i_hash);
} }
list_del_init(&inode->i_list); list_del_init(&inode->i_list);
list_del_init(&inode->i_sb_list);
inode->i_state|=I_FREEING; inode->i_state|=I_FREEING;
inodes_stat.nr_inodes--; inodes_stat.nr_inodes--;
spin_unlock(&inode_lock); spin_unlock(&inode_lock);
...@@ -1215,36 +1217,23 @@ EXPORT_SYMBOL(inode_needs_sync); ...@@ -1215,36 +1217,23 @@ EXPORT_SYMBOL(inode_needs_sync);
/* Function back in dquot.c */ /* Function back in dquot.c */
int remove_inode_dquot_ref(struct inode *, int, struct list_head *); int remove_inode_dquot_ref(struct inode *, int, struct list_head *);
void remove_dquot_ref(struct super_block *sb, int type, struct list_head *tofree_head) void remove_dquot_ref(struct super_block *sb, int type,
struct list_head *tofree_head)
{ {
struct inode *inode; struct inode *inode;
struct list_head *act_head;
if (!sb->dq_op) if (!sb->dq_op)
return; /* nothing to do */ return; /* nothing to do */
spin_lock(&inode_lock); /* This lock is for inodes code */ spin_lock(&inode_lock); /* This lock is for inodes code */
/* We hold dqptr_sem so we are safe against the quota code */ /*
list_for_each(act_head, &inode_in_use) { * We don't have to lock against quota code - test IS_QUOTAINIT is
inode = list_entry(act_head, struct inode, i_list); * just for speedup...
if (inode->i_sb == sb && !IS_NOQUOTA(inode)) */
remove_inode_dquot_ref(inode, type, tofree_head); list_for_each_entry(inode, &sb->s_inodes, i_sb_list)
}
list_for_each(act_head, &inode_unused) {
inode = list_entry(act_head, struct inode, i_list);
if (inode->i_sb == sb && !IS_NOQUOTA(inode))
remove_inode_dquot_ref(inode, type, tofree_head);
}
list_for_each(act_head, &sb->s_dirty) {
inode = list_entry(act_head, struct inode, i_list);
if (!IS_NOQUOTA(inode))
remove_inode_dquot_ref(inode, type, tofree_head);
}
list_for_each(act_head, &sb->s_io) {
inode = list_entry(act_head, struct inode, i_list);
if (!IS_NOQUOTA(inode)) if (!IS_NOQUOTA(inode))
remove_inode_dquot_ref(inode, type, tofree_head); remove_inode_dquot_ref(inode, type, tofree_head);
}
spin_unlock(&inode_lock); spin_unlock(&inode_lock);
} }
......
...@@ -70,6 +70,7 @@ static struct super_block *alloc_super(void) ...@@ -70,6 +70,7 @@ static struct super_block *alloc_super(void)
INIT_LIST_HEAD(&s->s_files); INIT_LIST_HEAD(&s->s_files);
INIT_LIST_HEAD(&s->s_instances); INIT_LIST_HEAD(&s->s_instances);
INIT_HLIST_HEAD(&s->s_anon); INIT_HLIST_HEAD(&s->s_anon);
INIT_LIST_HEAD(&s->s_inodes);
init_rwsem(&s->s_umount); init_rwsem(&s->s_umount);
sema_init(&s->s_lock, 1); sema_init(&s->s_lock, 1);
down_write(&s->s_umount); down_write(&s->s_umount);
......
...@@ -427,6 +427,7 @@ static inline int mapping_writably_mapped(struct address_space *mapping) ...@@ -427,6 +427,7 @@ static inline int mapping_writably_mapped(struct address_space *mapping)
struct inode { struct inode {
struct hlist_node i_hash; struct hlist_node i_hash;
struct list_head i_list; struct list_head i_list;
struct list_head i_sb_list;
struct list_head i_dentry; struct list_head i_dentry;
unsigned long i_ino; unsigned long i_ino;
atomic_t i_count; atomic_t i_count;
...@@ -775,6 +776,7 @@ struct super_block { ...@@ -775,6 +776,7 @@ struct super_block {
void *s_security; void *s_security;
struct xattr_handler **s_xattr; struct xattr_handler **s_xattr;
struct list_head s_inodes; /* all inodes */
struct list_head s_dirty; /* dirty inodes */ struct list_head s_dirty; /* dirty inodes */
struct list_head s_io; /* parked for writeback */ struct list_head s_io; /* parked for writeback */
struct hlist_head s_anon; /* anonymous dentries for (nfs) exporting */ struct hlist_head s_anon; /* anonymous dentries for (nfs) exporting */
......
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