diff options
author | Christoph Hellwig <hch@infradead.org> | 2012-06-18 10:47:03 -0400 |
---|---|---|
committer | Al Viro <viro@zeniv.linux.org.uk> | 2012-07-14 08:35:35 -0400 |
commit | 408ef013cc9e2f94a14f7ccbbe52ddfb18437a99 (patch) | |
tree | 3168e06e8090210a59af788ca97cd40d87b5282e | |
parent | ac481d6ca4081bdd348cbd84963d1ece843a3407 (diff) |
fs: move path_put on failure out of ->follow_link
Currently the non-nd_set_link based versions of ->follow_link are expected
to do a path_put(&nd->path) on failure. This calling convention is unexpected,
undocumented and doesn't match what the nd_set_link-based instances do.
Move the path_put out of the only non-nd_set_link based ->follow_link
instance into the caller.
Signed-off-by: Christoph Hellwig <hch@lst.de>
Signed-off-by: Al Viro <viro@zeniv.linux.org.uk>
-rw-r--r-- | fs/namei.c | 3 | ||||
-rw-r--r-- | fs/proc/base.c | 12 |
2 files changed, 9 insertions, 6 deletions
diff --git a/fs/namei.c b/fs/namei.c index 6b29a51bef5d..a9b94c62c303 100644 --- a/fs/namei.c +++ b/fs/namei.c | |||
@@ -624,7 +624,7 @@ follow_link(struct path *link, struct nameidata *nd, void **p) | |||
624 | *p = dentry->d_inode->i_op->follow_link(dentry, nd); | 624 | *p = dentry->d_inode->i_op->follow_link(dentry, nd); |
625 | error = PTR_ERR(*p); | 625 | error = PTR_ERR(*p); |
626 | if (IS_ERR(*p)) | 626 | if (IS_ERR(*p)) |
627 | goto out_put_link; | 627 | goto out_put_nd_path; |
628 | 628 | ||
629 | error = 0; | 629 | error = 0; |
630 | s = nd_get_link(nd); | 630 | s = nd_get_link(nd); |
@@ -646,7 +646,6 @@ follow_link(struct path *link, struct nameidata *nd, void **p) | |||
646 | 646 | ||
647 | out_put_nd_path: | 647 | out_put_nd_path: |
648 | path_put(&nd->path); | 648 | path_put(&nd->path); |
649 | out_put_link: | ||
650 | path_put(link); | 649 | path_put(link); |
651 | return error; | 650 | return error; |
652 | } | 651 | } |
diff --git a/fs/proc/base.c b/fs/proc/base.c index 8eaa5ea1c613..3bd5ac1ff018 100644 --- a/fs/proc/base.c +++ b/fs/proc/base.c | |||
@@ -1427,16 +1427,20 @@ static int proc_exe_link(struct dentry *dentry, struct path *exe_path) | |||
1427 | static void *proc_pid_follow_link(struct dentry *dentry, struct nameidata *nd) | 1427 | static void *proc_pid_follow_link(struct dentry *dentry, struct nameidata *nd) |
1428 | { | 1428 | { |
1429 | struct inode *inode = dentry->d_inode; | 1429 | struct inode *inode = dentry->d_inode; |
1430 | struct path path; | ||
1430 | int error = -EACCES; | 1431 | int error = -EACCES; |
1431 | 1432 | ||
1432 | /* We don't need a base pointer in the /proc filesystem */ | ||
1433 | path_put(&nd->path); | ||
1434 | |||
1435 | /* Are we allowed to snoop on the tasks file descriptors? */ | 1433 | /* Are we allowed to snoop on the tasks file descriptors? */ |
1436 | if (!proc_fd_access_allowed(inode)) | 1434 | if (!proc_fd_access_allowed(inode)) |
1437 | goto out; | 1435 | goto out; |
1438 | 1436 | ||
1439 | error = PROC_I(inode)->op.proc_get_link(dentry, &nd->path); | 1437 | error = PROC_I(inode)->op.proc_get_link(dentry, &path); |
1438 | if (error) | ||
1439 | goto out; | ||
1440 | |||
1441 | path_put(&nd->path); | ||
1442 | nd->path = path; | ||
1443 | return NULL; | ||
1440 | out: | 1444 | out: |
1441 | return ERR_PTR(error); | 1445 | return ERR_PTR(error); |
1442 | } | 1446 | } |