Commit b10cdcdc authored by Amir Goldstein's avatar Amir Goldstein Committed by Miklos Szeredi

ovl: untangle copy up call chain

In an attempt to dedup ~100 LOC, we ended up creating a tangled call chain,
whose branches merge and diverge in several points according to the
immutable c->tmpfile copy up mode.

This call chain was hard to analyse for locking correctness because the
locking requirements for the c->tmpfile flow were very different from the
locking requirements for the !c->tmpfile flow (i.e. directory vs.  regulare
file copy up).

Split the copy up helpers of the c->tmpfile flow from those of the
!c->tmpfile (i.e. workdir) flow and remove the c->tmpfile mode from copy up
context.
Suggested-by: default avatarAl Viro <viro@zeniv.linux.org.uk>
Signed-off-by: default avatarAmir Goldstein <amir73il@gmail.com>
Signed-off-by: default avatarMiklos Szeredi <mszeredi@redhat.com>
parent 007ea448
...@@ -395,7 +395,6 @@ struct ovl_copy_up_ctx { ...@@ -395,7 +395,6 @@ struct ovl_copy_up_ctx {
struct dentry *destdir; struct dentry *destdir;
struct qstr destname; struct qstr destname;
struct dentry *workdir; struct dentry *workdir;
bool tmpfile;
bool origin; bool origin;
bool indexed; bool indexed;
bool metacopy; bool metacopy;
...@@ -440,30 +439,58 @@ static int ovl_link_up(struct ovl_copy_up_ctx *c) ...@@ -440,30 +439,58 @@ static int ovl_link_up(struct ovl_copy_up_ctx *c)
return err; return err;
} }
static int ovl_install_temp(struct ovl_copy_up_ctx *c, struct dentry *temp, static int ovl_copy_up_inode(struct ovl_copy_up_ctx *c, struct dentry *temp)
struct dentry **newdentry)
{ {
int err; int err;
struct dentry *upper;
struct inode *udir = d_inode(c->destdir);
upper = lookup_one_len(c->destname.name, c->destdir, c->destname.len); err = ovl_copy_xattr(c->lowerpath.dentry, temp);
if (IS_ERR(upper)) if (err)
return PTR_ERR(upper); return err;
if (c->tmpfile) /*
err = ovl_do_link(temp, udir, upper); * Store identifier of lower inode in upper inode xattr to
else * allow lookup of the copy up origin inode.
err = ovl_do_rename(d_inode(c->workdir), temp, udir, upper, 0); *
* Don't set origin when we are breaking the association with a lower
* hard link.
*/
if (c->origin) {
err = ovl_set_origin(c->dentry, c->lowerpath.dentry, temp);
if (err)
return err;
}
if (S_ISREG(c->stat.mode) && !c->metacopy) {
struct path upperpath, datapath;
ovl_path_upper(c->dentry, &upperpath);
BUG_ON(upperpath.dentry != NULL);
upperpath.dentry = temp;
ovl_path_lowerdata(c->dentry, &datapath);
err = ovl_copy_up_data(&datapath, &upperpath, c->stat.size);
if (err)
return err;
}
if (c->metacopy) {
err = ovl_check_setxattr(c->dentry, temp, OVL_XATTR_METACOPY,
NULL, 0, -EOPNOTSUPP);
if (err)
return err;
}
inode_lock(temp->d_inode);
if (c->metacopy)
err = ovl_set_size(temp, &c->stat);
if (!err) if (!err)
*newdentry = dget(c->tmpfile ? upper : temp); err = ovl_set_attr(temp, &c->stat);
dput(upper); inode_unlock(temp->d_inode);
return err; return err;
} }
static struct dentry *ovl_get_tmpfile(struct ovl_copy_up_ctx *c) static struct dentry *ovl_get_workdir_temp(struct ovl_copy_up_ctx *c)
{ {
int err; int err;
struct dentry *temp; struct dentry *temp;
...@@ -483,9 +510,6 @@ static struct dentry *ovl_get_tmpfile(struct ovl_copy_up_ctx *c) ...@@ -483,9 +510,6 @@ static struct dentry *ovl_get_tmpfile(struct ovl_copy_up_ctx *c)
if (new_creds) if (new_creds)
old_creds = override_creds(new_creds); old_creds = override_creds(new_creds);
if (c->tmpfile)
temp = ovl_do_tmpfile(c->workdir, c->stat.mode);
else
temp = ovl_create_temp(c->workdir, &cattr); temp = ovl_create_temp(c->workdir, &cattr);
if (new_creds) { if (new_creds) {
...@@ -496,61 +520,136 @@ static struct dentry *ovl_get_tmpfile(struct ovl_copy_up_ctx *c) ...@@ -496,61 +520,136 @@ static struct dentry *ovl_get_tmpfile(struct ovl_copy_up_ctx *c)
return temp; return temp;
} }
static int ovl_copy_up_inode(struct ovl_copy_up_ctx *c, struct dentry *temp) /*
* Move temp file from workdir into place on upper dir.
* Used when copying up directories and when upper fs doesn't support O_TMPFILE.
*
* Caller must hold ovl_lock_rename_workdir().
*/
static int ovl_rename_temp(struct ovl_copy_up_ctx *c, struct dentry *temp,
struct dentry **newdentry)
{ {
int err; int err;
struct dentry *upper;
struct inode *udir = d_inode(c->destdir);
upper = lookup_one_len(c->destname.name, c->destdir, c->destname.len);
if (IS_ERR(upper))
return PTR_ERR(upper);
err = ovl_do_rename(d_inode(c->workdir), temp, udir, upper, 0);
if (!err)
*newdentry = dget(temp);
dput(upper);
err = ovl_copy_xattr(c->lowerpath.dentry, temp);
if (err)
return err; return err;
}
/* /*
* Store identifier of lower inode in upper inode xattr to * Copyup using workdir to prepare temp file. Used when copying up directories,
* allow lookup of the copy up origin inode. * special files or when upper fs doesn't support O_TMPFILE.
*
* Don't set origin when we are breaking the association with a lower
* hard link.
*/ */
if (c->origin) { static int ovl_copy_up_workdir(struct ovl_copy_up_ctx *c)
err = ovl_set_origin(c->dentry, c->lowerpath.dentry, temp); {
struct inode *inode;
struct dentry *newdentry = NULL;
struct dentry *temp;
int err;
err = ovl_lock_rename_workdir(c->workdir, c->destdir);
if (err) if (err)
return err; return err;
}
if (S_ISREG(c->stat.mode) && !c->metacopy) { temp = ovl_get_workdir_temp(c);
struct path upperpath, datapath; err = PTR_ERR(temp);
if (IS_ERR(temp))
goto unlock;
ovl_path_upper(c->dentry, &upperpath); err = ovl_copy_up_inode(c, temp);
BUG_ON(upperpath.dentry != NULL); if (err)
upperpath.dentry = temp; goto cleanup;
ovl_path_lowerdata(c->dentry, &datapath); if (S_ISDIR(c->stat.mode) && c->indexed) {
err = ovl_copy_up_data(&datapath, &upperpath, c->stat.size); err = ovl_create_index(c->dentry, c->lowerpath.dentry, temp);
if (err) if (err)
return err; goto cleanup;
} }
if (c->metacopy) { err = ovl_rename_temp(c, temp, &newdentry);
err = ovl_check_setxattr(c->dentry, temp, OVL_XATTR_METACOPY,
NULL, 0, -EOPNOTSUPP);
if (err) if (err)
goto cleanup;
if (!c->metacopy)
ovl_set_upperdata(d_inode(c->dentry));
inode = d_inode(c->dentry);
ovl_inode_update(inode, newdentry);
if (S_ISDIR(inode->i_mode))
ovl_set_flag(OVL_WHITEOUTS, inode);
out_dput:
dput(temp);
unlock:
unlock_rename(c->workdir, c->destdir);
return err; return err;
cleanup:
ovl_cleanup(d_inode(c->workdir), temp);
goto out_dput;
}
static struct dentry *ovl_get_tmpfile(struct ovl_copy_up_ctx *c)
{
int err;
struct dentry *temp;
const struct cred *old_creds = NULL;
struct cred *new_creds = NULL;
err = security_inode_copy_up(c->dentry, &new_creds);
if (err < 0)
return ERR_PTR(err);
if (new_creds)
old_creds = override_creds(new_creds);
temp = ovl_do_tmpfile(c->workdir, c->stat.mode);
if (new_creds) {
revert_creds(old_creds);
put_cred(new_creds);
} }
inode_lock(temp->d_inode); return temp;
if (c->metacopy) }
err = ovl_set_size(temp, &c->stat);
/* Link O_TMPFILE into place on upper dir */
static int ovl_link_tmpfile(struct ovl_copy_up_ctx *c, struct dentry *temp,
struct dentry **newdentry)
{
int err;
struct dentry *upper;
struct inode *udir = d_inode(c->destdir);
inode_lock_nested(udir, I_MUTEX_PARENT);
upper = lookup_one_len(c->destname.name, c->destdir, c->destname.len);
err = PTR_ERR(upper);
if (IS_ERR(upper))
goto unlock;
err = ovl_do_link(temp, udir, upper);
if (!err) if (!err)
err = ovl_set_attr(temp, &c->stat); *newdentry = dget(upper);
inode_unlock(temp->d_inode); dput(upper);
unlock:
inode_unlock(udir);
return err; return err;
} }
static int ovl_copy_up_locked(struct ovl_copy_up_ctx *c) /* Copyup using O_TMPFILE which does not require cross dir locking */
static int ovl_copy_up_tmpfile(struct ovl_copy_up_ctx *c)
{ {
struct inode *udir = c->destdir->d_inode;
struct inode *inode;
struct dentry *newdentry = NULL; struct dentry *newdentry = NULL;
struct dentry *temp; struct dentry *temp;
int err; int err;
...@@ -563,35 +662,17 @@ static int ovl_copy_up_locked(struct ovl_copy_up_ctx *c) ...@@ -563,35 +662,17 @@ static int ovl_copy_up_locked(struct ovl_copy_up_ctx *c)
if (err) if (err)
goto out; goto out;
if (S_ISDIR(c->stat.mode) && c->indexed) { err = ovl_link_tmpfile(c, temp, &newdentry);
err = ovl_create_index(c->dentry, c->lowerpath.dentry, temp);
if (err)
goto out;
}
if (c->tmpfile) {
inode_lock_nested(udir, I_MUTEX_PARENT);
err = ovl_install_temp(c, temp, &newdentry);
inode_unlock(udir);
} else {
err = ovl_install_temp(c, temp, &newdentry);
}
if (err) if (err)
goto out; goto out;
if (!c->metacopy) if (!c->metacopy)
ovl_set_upperdata(d_inode(c->dentry)); ovl_set_upperdata(d_inode(c->dentry));
inode = d_inode(c->dentry); ovl_inode_update(d_inode(c->dentry), newdentry);
ovl_inode_update(inode, newdentry);
if (S_ISDIR(inode->i_mode))
ovl_set_flag(OVL_WHITEOUTS, inode);
out: out:
if (err && !c->tmpfile)
ovl_cleanup(d_inode(c->workdir), temp);
dput(temp); dput(temp);
return err; return err;
} }
/* /*
...@@ -645,18 +726,10 @@ static int ovl_do_copy_up(struct ovl_copy_up_ctx *c) ...@@ -645,18 +726,10 @@ static int ovl_do_copy_up(struct ovl_copy_up_ctx *c)
} }
/* Should we copyup with O_TMPFILE or with workdir? */ /* Should we copyup with O_TMPFILE or with workdir? */
if (S_ISREG(c->stat.mode) && ofs->tmpfile) { if (S_ISREG(c->stat.mode) && ofs->tmpfile)
c->tmpfile = true; err = ovl_copy_up_tmpfile(c);
err = ovl_copy_up_locked(c); else
} else { err = ovl_copy_up_workdir(c);
err = ovl_lock_rename_workdir(c->workdir, c->destdir);
if (!err) {
err = ovl_copy_up_locked(c);
unlock_rename(c->workdir, c->destdir);
}
}
if (err) if (err)
goto out; goto out;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment