aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--fs/namei.c70
1 files changed, 33 insertions, 37 deletions
diff --git a/fs/namei.c b/fs/namei.c
index c2742b7dec59..b815a4d2e1d6 100644
--- a/fs/namei.c
+++ b/fs/namei.c
@@ -686,6 +686,35 @@ static __always_inline void follow_dotdot(struct nameidata *nd)
686} 686}
687 687
688/* 688/*
689 * Allocate a dentry with name and parent, and perform a parent
690 * directory ->lookup on it. Returns the new dentry, or ERR_PTR
691 * on error. parent->d_inode->i_mutex must be held. d_lookup must
692 * have verified that no child exists while under i_mutex.
693 */
694static struct dentry *d_alloc_and_lookup(struct dentry *parent,
695 struct qstr *name, struct nameidata *nd)
696{
697 struct inode *inode = parent->d_inode;
698 struct dentry *dentry;
699 struct dentry *old;
700
701 /* Don't create child dentry for a dead directory. */
702 if (unlikely(IS_DEADDIR(inode)))
703 return ERR_PTR(-ENOENT);
704
705 dentry = d_alloc(parent, name);
706 if (unlikely(!dentry))
707 return ERR_PTR(-ENOMEM);
708
709 old = inode->i_op->lookup(inode, dentry, nd);
710 if (unlikely(old)) {
711 dput(dentry);
712 dentry = old;
713 }
714 return dentry;
715}
716
717/*
689 * It's more convoluted than I'd like it to be, but... it's still fairly 718 * It's more convoluted than I'd like it to be, but... it's still fairly
690 * small and for now I'd prefer to have fast path as straight as possible. 719 * small and for now I'd prefer to have fast path as straight as possible.
691 * It _is_ time-critical. 720 * It _is_ time-critical.
@@ -738,30 +767,13 @@ need_lookup:
738 * so doing d_lookup() (with seqlock), instead of lockfree __d_lookup 767 * so doing d_lookup() (with seqlock), instead of lockfree __d_lookup
739 */ 768 */
740 dentry = d_lookup(parent, name); 769 dentry = d_lookup(parent, name);
741 if (!dentry) { 770 if (likely(!dentry)) {
742 struct dentry *new; 771 dentry = d_alloc_and_lookup(parent, name, nd);
743
744 /* Don't create child dentry for a dead directory. */
745 dentry = ERR_PTR(-ENOENT);
746 if (IS_DEADDIR(dir))
747 goto out_unlock;
748
749 new = d_alloc(parent, name);
750 dentry = ERR_PTR(-ENOMEM);
751 if (new) {
752 dentry = dir->i_op->lookup(dir, new, nd);
753 if (dentry)
754 dput(new);
755 else
756 dentry = new;
757 }
758out_unlock:
759 mutex_unlock(&dir->i_mutex); 772 mutex_unlock(&dir->i_mutex);
760 if (IS_ERR(dentry)) 773 if (IS_ERR(dentry))
761 goto fail; 774 goto fail;
762 goto done; 775 goto done;
763 } 776 }
764
765 /* 777 /*
766 * Uhhuh! Nasty case: the cache was re-populated while 778 * Uhhuh! Nasty case: the cache was re-populated while
767 * we waited on the semaphore. Need to revalidate. 779 * we waited on the semaphore. Need to revalidate.
@@ -1135,24 +1147,8 @@ static struct dentry *__lookup_hash(struct qstr *name,
1135 if (dentry && dentry->d_op && dentry->d_op->d_revalidate) 1147 if (dentry && dentry->d_op && dentry->d_op->d_revalidate)
1136 dentry = do_revalidate(dentry, nd); 1148 dentry = do_revalidate(dentry, nd);
1137 1149
1138 if (!dentry) { 1150 if (!dentry)
1139 struct dentry *new; 1151 dentry = d_alloc_and_lookup(base, name, nd);
1140
1141 /* Don't create child dentry for a dead directory. */
1142 dentry = ERR_PTR(-ENOENT);
1143 if (IS_DEADDIR(inode))
1144 goto out;
1145
1146 new = d_alloc(base, name);
1147 dentry = ERR_PTR(-ENOMEM);
1148 if (!new)
1149 goto out;
1150 dentry = inode->i_op->lookup(inode, new, nd);
1151 if (!dentry)
1152 dentry = new;
1153 else
1154 dput(new);
1155 }
1156out: 1152out:
1157 return dentry; 1153 return dentry;
1158} 1154}