aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--fs/dcache.c27
1 files changed, 15 insertions, 12 deletions
diff --git a/fs/dcache.c b/fs/dcache.c
index a8f89765d60..19570637469 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); 585relock:
586again:
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)) {
593relock2:
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);