diff options
-rw-r--r-- | fs/dcache.c | 11 |
1 files changed, 9 insertions, 2 deletions
diff --git a/fs/dcache.c b/fs/dcache.c index 8c09db9bb2a4..a191eebf1d63 100644 --- a/fs/dcache.c +++ b/fs/dcache.c | |||
@@ -2653,6 +2653,9 @@ static void __d_materialise_dentry(struct dentry *dentry, struct dentry *anon) | |||
2653 | * DCACHE_DISCONNECTED), then d_move that in place of the given dentry | 2653 | * DCACHE_DISCONNECTED), then d_move that in place of the given dentry |
2654 | * and return it, else simply d_add the inode to the dentry and return NULL. | 2654 | * and return it, else simply d_add the inode to the dentry and return NULL. |
2655 | * | 2655 | * |
2656 | * If a non-IS_ROOT directory is found, the filesystem is corrupt, and | ||
2657 | * we should error out: directories can't have multiple aliases. | ||
2658 | * | ||
2656 | * This is needed in the lookup routine of any filesystem that is exportable | 2659 | * This is needed in the lookup routine of any filesystem that is exportable |
2657 | * (via knfsd) so that we can build dcache paths to directories effectively. | 2660 | * (via knfsd) so that we can build dcache paths to directories effectively. |
2658 | * | 2661 | * |
@@ -2673,9 +2676,13 @@ struct dentry *d_splice_alias(struct inode *inode, struct dentry *dentry) | |||
2673 | 2676 | ||
2674 | if (inode && S_ISDIR(inode->i_mode)) { | 2677 | if (inode && S_ISDIR(inode->i_mode)) { |
2675 | spin_lock(&inode->i_lock); | 2678 | spin_lock(&inode->i_lock); |
2676 | new = __d_find_alias(inode, 1); | 2679 | new = __d_find_any_alias(inode); |
2677 | if (new) { | 2680 | if (new) { |
2678 | BUG_ON(!(new->d_flags & DCACHE_DISCONNECTED)); | 2681 | if (!IS_ROOT(new) || !(new->d_flags & DCACHE_DISCONNECTED)) { |
2682 | spin_unlock(&inode->i_lock); | ||
2683 | dput(new); | ||
2684 | return ERR_PTR(-EIO); | ||
2685 | } | ||
2679 | write_seqlock(&rename_lock); | 2686 | write_seqlock(&rename_lock); |
2680 | __d_materialise_dentry(dentry, new); | 2687 | __d_materialise_dentry(dentry, new); |
2681 | write_sequnlock(&rename_lock); | 2688 | write_sequnlock(&rename_lock); |