diff options
author | Nick Piggin <npiggin@kernel.dk> | 2011-01-07 01:49:45 -0500 |
---|---|---|
committer | Nick Piggin <npiggin@kernel.dk> | 2011-01-07 01:50:25 -0500 |
commit | 89e6054836a7b1e7500cd70a14b5579e752c9250 (patch) | |
tree | b74d2ecd1ac2a571d289e5525a0dc067ec1f35eb /fs/dcache.c | |
parent | a734eb458ab2bd11479a27dd54f48e1b26a55845 (diff) |
fs: dcache reduce prune_one_dentry locking
prune_one_dentry can avoid quite a bit of locking in the common case where
ancestors have an elevated refcount. Alternatively, we could have gone the
other way and made fewer trylocks in the case where d_count goes to zero, but
is probably less common.
Signed-off-by: Nick Piggin <npiggin@kernel.dk>
Diffstat (limited to 'fs/dcache.c')
-rw-r--r-- | fs/dcache.c | 27 |
1 files changed, 15 insertions, 12 deletions
diff --git a/fs/dcache.c b/fs/dcache.c index a8f89765d602..195706374697 100644 --- a/fs/dcache.c +++ b/fs/dcache.c | |||
@@ -582,26 +582,29 @@ static void prune_one_dentry(struct dentry *dentry, struct dentry *parent) | |||
582 | * Prune ancestors. | 582 | * Prune ancestors. |
583 | */ | 583 | */ |
584 | while (dentry) { | 584 | while (dentry) { |
585 | spin_lock(&dcache_inode_lock); | 585 | relock: |
586 | again: | ||
587 | spin_lock(&dentry->d_lock); | 586 | spin_lock(&dentry->d_lock); |
587 | if (dentry->d_count > 1) { | ||
588 | dentry->d_count--; | ||
589 | spin_unlock(&dentry->d_lock); | ||
590 | return; | ||
591 | } | ||
592 | if (!spin_trylock(&dcache_inode_lock)) { | ||
593 | relock2: | ||
594 | spin_unlock(&dentry->d_lock); | ||
595 | cpu_relax(); | ||
596 | goto relock; | ||
597 | } | ||
598 | |||
588 | if (IS_ROOT(dentry)) | 599 | if (IS_ROOT(dentry)) |
589 | parent = NULL; | 600 | parent = NULL; |
590 | else | 601 | else |
591 | parent = dentry->d_parent; | 602 | parent = dentry->d_parent; |
592 | if (parent && !spin_trylock(&parent->d_lock)) { | 603 | if (parent && !spin_trylock(&parent->d_lock)) { |
593 | spin_unlock(&dentry->d_lock); | ||
594 | goto again; | ||
595 | } | ||
596 | dentry->d_count--; | ||
597 | if (dentry->d_count) { | ||
598 | if (parent) | ||
599 | spin_unlock(&parent->d_lock); | ||
600 | spin_unlock(&dentry->d_lock); | ||
601 | spin_unlock(&dcache_inode_lock); | 604 | spin_unlock(&dcache_inode_lock); |
602 | return; | 605 | goto relock2; |
603 | } | 606 | } |
604 | 607 | dentry->d_count--; | |
605 | dentry_lru_del(dentry); | 608 | dentry_lru_del(dentry); |
606 | __d_drop(dentry); | 609 | __d_drop(dentry); |
607 | dentry = d_kill(dentry, parent); | 610 | dentry = d_kill(dentry, parent); |