simplify lookup_open()/atomic_open() - do the temporary mnt_want_write() early

The write ref to vfsmount taken in lookup_open()/atomic_open() is going to
be dropped; we take the one to stay in dentry_open().  Just grab the temporary
in caller if it looks like we are going to need it (create/truncate/writable open)
and pass (by value) "has it succeeded" flag.  Instead of doing mnt_want_write()
inside, check that flag and treat "false" as "mnt_want_write() has just failed".
mnt_want_write() is cheap and the things get considerably simpler and more robust
that way - we get it and drop it in the same function, to start with, rather
than passing a "has something in the guts of really scary functions taken it"
back to caller.

Signed-off-by: Al Viro <viro@zeniv.linux.org.uk>
This commit is contained in:
Al Viro 2012-07-31 00:53:35 +04:00
parent f8310c5920
commit 64894cf843

View File

@ -2395,7 +2395,7 @@ static int may_o_create(struct path *dir, struct dentry *dentry, umode_t mode)
static int atomic_open(struct nameidata *nd, struct dentry *dentry, static int atomic_open(struct nameidata *nd, struct dentry *dentry,
struct path *path, struct file *file, struct path *path, struct file *file,
const struct open_flags *op, const struct open_flags *op,
bool *want_write, bool need_lookup, bool got_write, bool need_lookup,
int *opened) int *opened)
{ {
struct inode *dir = nd->path.dentry->d_inode; struct inode *dir = nd->path.dentry->d_inode;
@ -2432,12 +2432,9 @@ static int atomic_open(struct nameidata *nd, struct dentry *dentry,
* Another problem is returing the "right" error value (e.g. for an * Another problem is returing the "right" error value (e.g. for an
* O_EXCL open we want to return EEXIST not EROFS). * O_EXCL open we want to return EEXIST not EROFS).
*/ */
if ((open_flag & (O_CREAT | O_TRUNC)) || if (((open_flag & (O_CREAT | O_TRUNC)) ||
(open_flag & O_ACCMODE) != O_RDONLY) { (open_flag & O_ACCMODE) != O_RDONLY) && unlikely(!got_write)) {
error = mnt_want_write(nd->path.mnt); if (!(open_flag & O_CREAT)) {
if (!error) {
*want_write = true;
} else if (!(open_flag & O_CREAT)) {
/* /*
* No O_CREATE -> atomicity not a requirement -> fall * No O_CREATE -> atomicity not a requirement -> fall
* back to lookup + open * back to lookup + open
@ -2445,11 +2442,11 @@ static int atomic_open(struct nameidata *nd, struct dentry *dentry,
goto no_open; goto no_open;
} else if (open_flag & (O_EXCL | O_TRUNC)) { } else if (open_flag & (O_EXCL | O_TRUNC)) {
/* Fall back and fail with the right error */ /* Fall back and fail with the right error */
create_error = error; create_error = -EROFS;
goto no_open; goto no_open;
} else { } else {
/* No side effects, safe to clear O_CREAT */ /* No side effects, safe to clear O_CREAT */
create_error = error; create_error = -EROFS;
open_flag &= ~O_CREAT; open_flag &= ~O_CREAT;
} }
} }
@ -2556,7 +2553,7 @@ static int atomic_open(struct nameidata *nd, struct dentry *dentry,
static int lookup_open(struct nameidata *nd, struct path *path, static int lookup_open(struct nameidata *nd, struct path *path,
struct file *file, struct file *file,
const struct open_flags *op, const struct open_flags *op,
bool *want_write, int *opened) bool got_write, int *opened)
{ {
struct dentry *dir = nd->path.dentry; struct dentry *dir = nd->path.dentry;
struct inode *dir_inode = dir->d_inode; struct inode *dir_inode = dir->d_inode;
@ -2574,7 +2571,7 @@ static int lookup_open(struct nameidata *nd, struct path *path,
goto out_no_open; goto out_no_open;
if ((nd->flags & LOOKUP_OPEN) && dir_inode->i_op->atomic_open) { if ((nd->flags & LOOKUP_OPEN) && dir_inode->i_op->atomic_open) {
return atomic_open(nd, dentry, path, file, op, want_write, return atomic_open(nd, dentry, path, file, op, got_write,
need_lookup, opened); need_lookup, opened);
} }
@ -2598,10 +2595,10 @@ static int lookup_open(struct nameidata *nd, struct path *path,
* a permanent write count is taken through * a permanent write count is taken through
* the 'struct file' in finish_open(). * the 'struct file' in finish_open().
*/ */
error = mnt_want_write(nd->path.mnt); if (!got_write) {
if (error) error = -EROFS;
goto out_dput; goto out_dput;
*want_write = true; }
*opened |= FILE_CREATED; *opened |= FILE_CREATED;
error = security_path_mknod(&nd->path, dentry, mode, 0); error = security_path_mknod(&nd->path, dentry, mode, 0);
if (error) if (error)
@ -2631,7 +2628,7 @@ static int do_last(struct nameidata *nd, struct path *path,
struct dentry *dir = nd->path.dentry; struct dentry *dir = nd->path.dentry;
int open_flag = op->open_flag; int open_flag = op->open_flag;
bool will_truncate = (open_flag & O_TRUNC) != 0; bool will_truncate = (open_flag & O_TRUNC) != 0;
bool want_write = false; bool got_write = false;
int acc_mode = op->acc_mode; int acc_mode = op->acc_mode;
struct inode *inode; struct inode *inode;
bool symlink_ok = false; bool symlink_ok = false;
@ -2700,8 +2697,18 @@ static int do_last(struct nameidata *nd, struct path *path,
} }
retry_lookup: retry_lookup:
if (op->open_flag & (O_CREAT | O_TRUNC | O_WRONLY | O_RDWR)) {
error = mnt_want_write(nd->path.mnt);
if (!error)
got_write = true;
/*
* do _not_ fail yet - we might not need that or fail with
* a different error; let lookup_open() decide; we'll be
* dropping this one anyway.
*/
}
mutex_lock(&dir->d_inode->i_mutex); mutex_lock(&dir->d_inode->i_mutex);
error = lookup_open(nd, path, file, op, &want_write, opened); error = lookup_open(nd, path, file, op, got_write, opened);
mutex_unlock(&dir->d_inode->i_mutex); mutex_unlock(&dir->d_inode->i_mutex);
if (error <= 0) { if (error <= 0) {
@ -2736,9 +2743,9 @@ static int do_last(struct nameidata *nd, struct path *path,
* possible mount and symlink following (this might be optimized away if * possible mount and symlink following (this might be optimized away if
* necessary...) * necessary...)
*/ */
if (want_write) { if (got_write) {
mnt_drop_write(nd->path.mnt); mnt_drop_write(nd->path.mnt);
want_write = false; got_write = false;
} }
error = -EEXIST; error = -EEXIST;
@ -2803,7 +2810,7 @@ static int do_last(struct nameidata *nd, struct path *path,
error = mnt_want_write(nd->path.mnt); error = mnt_want_write(nd->path.mnt);
if (error) if (error)
goto out; goto out;
want_write = true; got_write = true;
} }
finish_open_created: finish_open_created:
error = may_open(&nd->path, acc_mode, open_flag); error = may_open(&nd->path, acc_mode, open_flag);
@ -2830,7 +2837,7 @@ static int do_last(struct nameidata *nd, struct path *path,
goto exit_fput; goto exit_fput;
} }
out: out:
if (want_write) if (got_write)
mnt_drop_write(nd->path.mnt); mnt_drop_write(nd->path.mnt);
path_put(&save_parent); path_put(&save_parent);
terminate_walk(nd); terminate_walk(nd);
@ -2854,9 +2861,9 @@ static int do_last(struct nameidata *nd, struct path *path,
nd->inode = dir->d_inode; nd->inode = dir->d_inode;
save_parent.mnt = NULL; save_parent.mnt = NULL;
save_parent.dentry = NULL; save_parent.dentry = NULL;
if (want_write) { if (got_write) {
mnt_drop_write(nd->path.mnt); mnt_drop_write(nd->path.mnt);
want_write = false; got_write = false;
} }
retried = true; retried = true;
goto retry_lookup; goto retry_lookup;