ovl: check if upperdir fs supports O_TMPFILE
authorAmir Goldstein <amir73il@gmail.com>
Tue, 17 Jan 2017 04:34:53 +0000 (06:34 +0200)
committerMiklos Szeredi <mszeredi@redhat.com>
Tue, 7 Feb 2017 14:47:14 +0000 (15:47 +0100)
This is needed for choosing between concurrent copyup
using O_TMPFILE and legacy copyup using workdir+rename.

Signed-off-by: Amir Goldstein <amir73il@gmail.com>
Signed-off-by: Miklos Szeredi <mszeredi@redhat.com>
fs/overlayfs/overlayfs.h
fs/overlayfs/ovl_entry.h
fs/overlayfs/super.c

index 8af450b0e57a2d826f289d87a1ace6db2d8e7d73..3822a909ce1f41e0e75ec07663220f5d82fc75fb 100644 (file)
@@ -127,6 +127,15 @@ static inline int ovl_do_whiteout(struct inode *dir, struct dentry *dentry)
        return err;
 }
 
+static inline struct dentry *ovl_do_tmpfile(struct dentry *dentry, umode_t mode)
+{
+       struct dentry *ret = vfs_tmpfile(dentry, mode, 0);
+       int err = IS_ERR(ret) ? PTR_ERR(ret) : 0;
+
+       pr_debug("tmpfile(%pd2, 0%o) = %i\n", dentry, mode, err);
+       return ret;
+}
+
 static inline struct inode *ovl_inode_real(struct inode *inode, bool *is_upper)
 {
        unsigned long x = (unsigned long) READ_ONCE(inode->i_private);
index d14bca1850d95a310e8637d61eeec9efcd9924ee..65f240001aa698cadcfa08ce8aa6b8db6c2e6a47 100644 (file)
@@ -27,6 +27,7 @@ struct ovl_fs {
        struct ovl_config config;
        /* creds of process who forced instantiation of super block */
        const struct cred *creator_cred;
+       bool tmpfile;
 };
 
 /* private information held for every overlayfs dentry */
index 20f48abbb82fd3972cd58d7eecf09a2b37a70579..ff05065b510f7fcb8bcc7251f825ee0cacac5ab0 100644 (file)
@@ -825,6 +825,8 @@ static int ovl_fill_super(struct super_block *sb, void *data, int silent)
                 * creation of workdir in previous step.
                 */
                if (ufs->workdir) {
+                       struct dentry *temp;
+
                        err = ovl_check_d_type_supported(&workpath);
                        if (err < 0)
                                goto out_put_workdir;
@@ -836,6 +838,14 @@ static int ovl_fill_super(struct super_block *sb, void *data, int silent)
                         */
                        if (!err)
                                pr_warn("overlayfs: upper fs needs to support d_type.\n");
+
+                       /* Check if upper/work fs supports O_TMPFILE */
+                       temp = ovl_do_tmpfile(ufs->workdir, S_IFREG | 0);
+                       ufs->tmpfile = !IS_ERR(temp);
+                       if (ufs->tmpfile)
+                               dput(temp);
+                       else
+                               pr_warn("overlayfs: upper fs does not support tmpfile.\n");
                }
        }