untangling do_lookup() - merge failure exits in !dentry case
authorAl Viro <viro@zeniv.linux.org.uk>
Fri, 30 Mar 2012 18:37:42 +0000 (14:37 -0400)
committerAl Viro <viro@zeniv.linux.org.uk>
Sat, 31 Mar 2012 20:03:16 +0000 (16:03 -0400)
Signed-off-by: Al Viro <viro@zeniv.linux.org.uk>
fs/namei.c

index 5414438abff0445bc9daac4e9d041ae870e75041..a0f9a0294ff2ec8d7a764d4abee47966f8111ae7 100644 (file)
@@ -1176,35 +1176,28 @@ retry:
                dentry = d_lookup(parent, name);
                if (dentry && d_need_lookup(dentry)) {
                        dentry = d_inode_lookup(parent, dentry, nd);
-                       if (IS_ERR(dentry)) {
-                               mutex_unlock(&dir->i_mutex);
-                               return PTR_ERR(dentry);
-                       }
-               } else if (dentry && (dentry->d_flags & DCACHE_OP_REVALIDATE)) {
+                       goto l;
+               }
+               if (dentry && (dentry->d_flags & DCACHE_OP_REVALIDATE)) {
                        status = d_revalidate(dentry, nd);
                        if (unlikely(status <= 0)) {
                                if (status < 0) {
-                                       mutex_unlock(&dir->i_mutex);
                                        dput(dentry);
-                                       return status;
+                                       dentry = ERR_PTR(status);
+                                       goto l;
                                }
                                if (!d_invalidate(dentry)) {
                                        dput(dentry);
                                        dentry = d_alloc_and_lookup(parent, name, nd);
-                                       if (IS_ERR(dentry)) {
-                                               mutex_unlock(&dir->i_mutex);
-                                               return PTR_ERR(dentry);
-                                       }
                                }
                        }
                } else if (!dentry) {
                        dentry = d_alloc_and_lookup(parent, name, nd);
-                       if (IS_ERR(dentry)) {
-                               mutex_unlock(&dir->i_mutex);
-                               return PTR_ERR(dentry);
-                       }
                }
+       l:
                mutex_unlock(&dir->i_mutex);
+               if (IS_ERR(dentry))
+                       return PTR_ERR(dentry);
                goto done;
        }
        if (unlikely(dentry->d_flags & DCACHE_OP_REVALIDATE) && need_reval)