ovl: take mnt_want_write() for removing impure xattr
authorAmir Goldstein <amir73il@gmail.com>
Wed, 3 Jan 2018 16:54:42 +0000 (18:54 +0200)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Fri, 16 Feb 2018 19:23:10 +0000 (20:23 +0100)
commit a5a927a7c82e28ea76599dee4019c41e372c911f upstream.

The optimization in ovl_cache_get_impure() that tries to remove an
unneeded "impure" xattr needs to take mnt_want_write() on upper fs.

Fixes: 4edb83bb1041 ("ovl: constant d_ino for non-merge dirs")
Signed-off-by: Amir Goldstein <amir73il@gmail.com>
Signed-off-by: Miklos Szeredi <mszeredi@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
fs/overlayfs/readdir.c

index f1e9b24c994516c8c8fb5659c779837fec04c3c5..7fa7d68baa6d197d81f868ab795d5316f236f006 100644 (file)
@@ -575,8 +575,15 @@ static struct ovl_dir_cache *ovl_cache_get_impure(struct path *path)
                return ERR_PTR(res);
        }
        if (list_empty(&cache->entries)) {
-               /* Good oportunity to get rid of an unnecessary "impure" flag */
-               ovl_do_removexattr(ovl_dentry_upper(dentry), OVL_XATTR_IMPURE);
+               /*
+                * A good opportunity to get rid of an unneeded "impure" flag.
+                * Removing the "impure" xattr is best effort.
+                */
+               if (!ovl_want_write(dentry)) {
+                       ovl_do_removexattr(ovl_dentry_upper(dentry),
+                                          OVL_XATTR_IMPURE);
+                       ovl_drop_write(dentry);
+               }
                ovl_clear_flag(OVL_IMPURE, d_inode(dentry));
                kfree(cache);
                return NULL;