aboutsummaryrefslogtreecommitdiffstats
path: root/fs/namei.c
diff options
context:
space:
mode:
authorLinus Torvalds <torvalds@linux-foundation.org>2011-01-14 12:08:29 -0500
committerLinus Torvalds <torvalds@linux-foundation.org>2011-01-14 12:08:29 -0500
commitacda4721ae876dedab3fef04bbd8020bfa67ff0a (patch)
treeec48d554fe48b3915912e2ae62bc962ade0553bd /fs/namei.c
parent822e5215f9eef86c1dd56d5696bf55a212b0e3f0 (diff)
parent32385c7cf60a78375b63afc4f02001df84dfd1a0 (diff)
Merge branch 'vfs-scale-working' of git://git.kernel.org/pub/scm/linux/kernel/git/npiggin/linux-npiggin
* 'vfs-scale-working' of git://git.kernel.org/pub/scm/linux/kernel/git/npiggin/linux-npiggin: kernel: fix hlist_bl again cgroups: Fix a lockdep warning at cgroup removal fs: namei fix ->put_link on wrong inode in do_filp_open
Diffstat (limited to 'fs/namei.c')
-rw-r--r--fs/namei.c37
1 files changed, 19 insertions, 18 deletions
diff --git a/fs/namei.c b/fs/namei.c
index 86643302079..8df7a78ace5 100644
--- a/fs/namei.c
+++ b/fs/namei.c
@@ -779,7 +779,8 @@ static void path_put_conditional(struct path *path, struct nameidata *nd)
779 mntput(path->mnt); 779 mntput(path->mnt);
780} 780}
781 781
782static inline void path_to_nameidata(struct path *path, struct nameidata *nd) 782static inline void path_to_nameidata(const struct path *path,
783 struct nameidata *nd)
783{ 784{
784 if (!(nd->flags & LOOKUP_RCU)) { 785 if (!(nd->flags & LOOKUP_RCU)) {
785 dput(nd->path.dentry); 786 dput(nd->path.dentry);
@@ -791,20 +792,20 @@ static inline void path_to_nameidata(struct path *path, struct nameidata *nd)
791} 792}
792 793
793static __always_inline int 794static __always_inline int
794__do_follow_link(struct path *path, struct nameidata *nd, void **p) 795__do_follow_link(const struct path *link, struct nameidata *nd, void **p)
795{ 796{
796 int error; 797 int error;
797 struct dentry *dentry = path->dentry; 798 struct dentry *dentry = link->dentry;
798 799
799 touch_atime(path->mnt, dentry); 800 touch_atime(link->mnt, dentry);
800 nd_set_link(nd, NULL); 801 nd_set_link(nd, NULL);
801 802
802 if (path->mnt != nd->path.mnt) { 803 if (link->mnt != nd->path.mnt) {
803 path_to_nameidata(path, nd); 804 path_to_nameidata(link, nd);
804 nd->inode = nd->path.dentry->d_inode; 805 nd->inode = nd->path.dentry->d_inode;
805 dget(dentry); 806 dget(dentry);
806 } 807 }
807 mntget(path->mnt); 808 mntget(link->mnt);
808 809
809 nd->last_type = LAST_BIND; 810 nd->last_type = LAST_BIND;
810 *p = dentry->d_inode->i_op->follow_link(dentry, nd); 811 *p = dentry->d_inode->i_op->follow_link(dentry, nd);
@@ -2348,11 +2349,12 @@ reval:
2348 nd.flags = flags; 2349 nd.flags = flags;
2349 filp = do_last(&nd, &path, open_flag, acc_mode, mode, pathname); 2350 filp = do_last(&nd, &path, open_flag, acc_mode, mode, pathname);
2350 while (unlikely(!filp)) { /* trailing symlink */ 2351 while (unlikely(!filp)) { /* trailing symlink */
2351 struct path holder; 2352 struct path link = path;
2353 struct inode *linki = link.dentry->d_inode;
2352 void *cookie; 2354 void *cookie;
2353 error = -ELOOP; 2355 error = -ELOOP;
2354 /* S_ISDIR part is a temporary automount kludge */ 2356 /* S_ISDIR part is a temporary automount kludge */
2355 if (!(nd.flags & LOOKUP_FOLLOW) && !S_ISDIR(nd.inode->i_mode)) 2357 if (!(nd.flags & LOOKUP_FOLLOW) && !S_ISDIR(linki->i_mode))
2356 goto exit_dput; 2358 goto exit_dput;
2357 if (count++ == 32) 2359 if (count++ == 32)
2358 goto exit_dput; 2360 goto exit_dput;
@@ -2368,23 +2370,22 @@ reval:
2368 * just set LAST_BIND. 2370 * just set LAST_BIND.
2369 */ 2371 */
2370 nd.flags |= LOOKUP_PARENT; 2372 nd.flags |= LOOKUP_PARENT;
2371 error = security_inode_follow_link(path.dentry, &nd); 2373 error = security_inode_follow_link(link.dentry, &nd);
2372 if (error) 2374 if (error)
2373 goto exit_dput; 2375 goto exit_dput;
2374 error = __do_follow_link(&path, &nd, &cookie); 2376 error = __do_follow_link(&link, &nd, &cookie);
2375 if (unlikely(error)) { 2377 if (unlikely(error)) {
2376 if (!IS_ERR(cookie) && nd.inode->i_op->put_link) 2378 if (!IS_ERR(cookie) && linki->i_op->put_link)
2377 nd.inode->i_op->put_link(path.dentry, &nd, cookie); 2379 linki->i_op->put_link(link.dentry, &nd, cookie);
2378 /* nd.path had been dropped */ 2380 /* nd.path had been dropped */
2379 nd.path = path; 2381 nd.path = link;
2380 goto out_path; 2382 goto out_path;
2381 } 2383 }
2382 holder = path;
2383 nd.flags &= ~LOOKUP_PARENT; 2384 nd.flags &= ~LOOKUP_PARENT;
2384 filp = do_last(&nd, &path, open_flag, acc_mode, mode, pathname); 2385 filp = do_last(&nd, &path, open_flag, acc_mode, mode, pathname);
2385 if (nd.inode->i_op->put_link) 2386 if (linki->i_op->put_link)
2386 nd.inode->i_op->put_link(holder.dentry, &nd, cookie); 2387 linki->i_op->put_link(link.dentry, &nd, cookie);
2387 path_put(&holder); 2388 path_put(&link);
2388 } 2389 }
2389out: 2390out:
2390 if (nd.root.mnt) 2391 if (nd.root.mnt)