diff options
author | Al Viro <viro@zeniv.linux.org.uk> | 2010-10-29 03:30:42 -0400 |
---|---|---|
committer | Al Viro <viro@zeniv.linux.org.uk> | 2010-10-29 04:14:56 -0400 |
commit | d893f1bc2a9f0f7dcb4b433452c59f9bedac0d7d (patch) | |
tree | b3cf84a271ccb19529d83a544b6024bbb23a7801 /fs/namei.c | |
parent | a4118ee1d80b527c385cadd14db79559efb8a493 (diff) |
fix open/umount race
nameidata_to_filp() drops nd->path or transfers it to opened
file. In the former case it's a Bad Idea(tm) to do mnt_drop_write()
on nd->path.mnt, since we might race with umount and vfsmount in
question might be gone already.
Fix: don't drop it, then... IOW, have nameidata_to_filp() grab nd->path
in case it transfers it to file and do path_drop() in callers. After
they are through with accessing nd->path...
Reported-by: Miklos Szeredi <miklos@szeredi.hu>
Signed-off-by: Al Viro <viro@zeniv.linux.org.uk>
Diffstat (limited to 'fs/namei.c')
-rw-r--r-- | fs/namei.c | 2 |
1 files changed, 2 insertions, 0 deletions
diff --git a/fs/namei.c b/fs/namei.c index f7dbc06857ab..5362af9b7372 100644 --- a/fs/namei.c +++ b/fs/namei.c | |||
@@ -1574,6 +1574,7 @@ static struct file *finish_open(struct nameidata *nd, | |||
1574 | */ | 1574 | */ |
1575 | if (will_truncate) | 1575 | if (will_truncate) |
1576 | mnt_drop_write(nd->path.mnt); | 1576 | mnt_drop_write(nd->path.mnt); |
1577 | path_put(&nd->path); | ||
1577 | return filp; | 1578 | return filp; |
1578 | 1579 | ||
1579 | exit: | 1580 | exit: |
@@ -1675,6 +1676,7 @@ static struct file *do_last(struct nameidata *nd, struct path *path, | |||
1675 | } | 1676 | } |
1676 | filp = nameidata_to_filp(nd); | 1677 | filp = nameidata_to_filp(nd); |
1677 | mnt_drop_write(nd->path.mnt); | 1678 | mnt_drop_write(nd->path.mnt); |
1679 | path_put(&nd->path); | ||
1678 | if (!IS_ERR(filp)) { | 1680 | if (!IS_ERR(filp)) { |
1679 | error = ima_file_check(filp, acc_mode); | 1681 | error = ima_file_check(filp, acc_mode); |
1680 | if (error) { | 1682 | if (error) { |