summaryrefslogtreecommitdiff
path: root/fs/overlayfs/dir.c
diff options
context:
space:
mode:
authorMiklos Szeredi <mszeredi@redhat.com>2016-12-16 11:02:57 +0100
committerMiklos Szeredi <mszeredi@redhat.com>2016-12-16 11:02:57 +0100
commit5cf5b477f0ca33f56a30c7ec00e61a6204da2efb (patch)
treee418f95e7847541a12dc96411d6929f89849ce53 /fs/overlayfs/dir.c
parentc5bef3a72b9d8a2040d5e9f4bde03db7c86bbfce (diff)
ovl: opaque cleanup
oe->opaque is set for a) whiteouts b) directories having the "trusted.overlay.opaque" xattr Case b can be simplified, since setting the xattr always implies setting oe->opaque. Also once set, the opaque flag is never cleared. Don't need to set opaque flag for non-directories. Signed-off-by: Miklos Szeredi <mszeredi@redhat.com>
Diffstat (limited to 'fs/overlayfs/dir.c')
-rw-r--r--fs/overlayfs/dir.c43
1 files changed, 21 insertions, 22 deletions
diff --git a/fs/overlayfs/dir.c b/fs/overlayfs/dir.c
index b84d61b353cd..76e39aaaa038 100644
--- a/fs/overlayfs/dir.c
+++ b/fs/overlayfs/dir.c
@@ -128,9 +128,15 @@ int ovl_create_real(struct inode *dir, struct dentry *newdentry,
return err;
}
-static int ovl_set_opaque(struct dentry *upperdentry)
+static int ovl_set_opaque(struct dentry *dentry, struct dentry *upperdentry)
{
- return ovl_do_setxattr(upperdentry, OVL_XATTR_OPAQUE, "y", 1, 0);
+ int err;
+
+ err = ovl_do_setxattr(upperdentry, OVL_XATTR_OPAQUE, "y", 1, 0);
+ if (!err)
+ ovl_dentry_set_opaque(dentry);
+
+ return err;
}
static int ovl_dir_getattr(struct vfsmount *mnt, struct dentry *dentry,
@@ -274,7 +280,7 @@ static struct dentry *ovl_clear_empty(struct dentry *dentry,
if (err)
goto out_cleanup;
- err = ovl_set_opaque(opaquedir);
+ err = ovl_set_opaque(dentry, opaquedir);
if (err)
goto out_cleanup;
@@ -435,7 +441,7 @@ static int ovl_create_over_whiteout(struct dentry *dentry, struct inode *inode,
}
if (!hardlink && S_ISDIR(stat->mode)) {
- err = ovl_set_opaque(newdentry);
+ err = ovl_set_opaque(dentry, newdentry);
if (err)
goto out_cleanup;
@@ -996,29 +1002,22 @@ static int ovl_rename(struct inode *olddir, struct dentry *old,
if (WARN_ON(olddentry->d_inode == newdentry->d_inode))
goto out_dput;
+ err = 0;
if (is_dir) {
- if (ovl_type_merge_or_lower(old)) {
+ if (ovl_type_merge_or_lower(old))
err = ovl_set_redirect(old, samedir);
- if (err)
- goto out_dput;
- } else if (!old_opaque && ovl_lower_positive(new)) {
- err = ovl_set_opaque(olddentry);
- if (err)
- goto out_dput;
- ovl_dentry_set_opaque(old, true);
- }
+ else if (!old_opaque && ovl_lower_positive(new))
+ err = ovl_set_opaque(old, olddentry);
+ if (err)
+ goto out_dput;
}
if (!overwrite && new_is_dir) {
- if (ovl_type_merge_or_lower(new)) {
+ if (ovl_type_merge_or_lower(new))
err = ovl_set_redirect(new, samedir);
- if (err)
- goto out_dput;
- } else if (!new_opaque && ovl_lower_positive(old)) {
- err = ovl_set_opaque(newdentry);
- if (err)
- goto out_dput;
- ovl_dentry_set_opaque(new, true);
- }
+ else if (!new_opaque && ovl_lower_positive(old))
+ err = ovl_set_opaque(new, newdentry);
+ if (err)
+ goto out_dput;
}
err = ovl_do_rename(old_upperdir->d_inode, olddentry,