Skip to content

Commit 5842eac

Browse files
zhangyi089chrome-bot
authored andcommitted
UPSTREAM: ovl: fix rmdir problem on non-merge dir with origin xattr
An "origin && non-merge" upper dir may have leftover whiteouts that were created in past mount. overlayfs does no clear this dir when we delete it, which may lead to rmdir fail or temp file left in workdir. Simple reproducer: mkdir lower upper work merge mkdir -p lower/dir touch lower/dir/a mount -t overlay overlay -olowerdir=lower,upperdir=upper,\ workdir=work merge rm merge/dir/a umount merge rm -rf lower/* touch lower/dir (*) mount -t overlay overlay -olowerdir=lower,upperdir=upper,\ workdir=work merge rm -rf merge/dir Syslog dump: overlayfs: cleanup of 'work/thesofproject#7' failed (-39) (*): if we do not create the regular file, the result is different: rm: cannot remove "dir/": Directory not empty This patch adds a check for the case of non-merge dir that may contain whiteouts, and calls ovl_check_empty_dir() to check and clear whiteouts from upper dir when an empty dir is being deleted. [amir: split patch from ovl_check_empty_dir() cleanup rename ovl_is_origin() to ovl_may_have_whiteouts() check OVL_WHITEOUTS flag instead of checking origin xattr] Signed-off-by: zhangyi (F) <yi.zhang@huawei.com> Signed-off-by: Amir Goldstein <amir73il@gmail.com> Signed-off-by: Miklos Szeredi <mszeredi@redhat.com> (cherry picked from commit 07f6fff) Signed-off-by: Xuewei Zhang <xueweiz@google.com> BUG=b:112484793 TEST=xfstests overlay/031 Change-Id: Ibb67890881ab2724d81bd29cef9807e5ac97c8d4 Reviewed-on: https://chromium-review.googlesource.com/1194695 Commit-Ready: Xuewei Zhang <xueweiz@google.com> Tested-by: Xuewei Zhang <xueweiz@google.com> Reviewed-by: Guenter Roeck <groeck@chromium.org>
1 parent 865ebe4 commit 5842eac

File tree

1 file changed

+10
-3
lines changed

1 file changed

+10
-3
lines changed

fs/overlayfs/dir.c

Lines changed: 10 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -181,6 +181,11 @@ static bool ovl_type_origin(struct dentry *dentry)
181181
return OVL_TYPE_ORIGIN(ovl_path_type(dentry));
182182
}
183183

184+
static bool ovl_may_have_whiteouts(struct dentry *dentry)
185+
{
186+
return ovl_test_flag(OVL_WHITEOUTS, d_inode(dentry));
187+
}
188+
184189
static int ovl_create_upper(struct dentry *dentry, struct inode *inode,
185190
struct cattr *attr, struct dentry *hardlink)
186191
{
@@ -698,8 +703,9 @@ static int ovl_remove_upper(struct dentry *dentry, bool is_dir)
698703
struct dentry *opaquedir = NULL;
699704
int err;
700705

701-
/* Redirect dir can be !ovl_lower_positive && OVL_TYPE_MERGE */
702-
if (is_dir && ovl_dentry_get_redirect(dentry)) {
706+
/* Redirect/origin dir can be !ovl_lower_positive && not clean */
707+
if (is_dir && (ovl_dentry_get_redirect(dentry) ||
708+
ovl_may_have_whiteouts(dentry))) {
703709
opaquedir = ovl_check_empty_and_clear(dentry);
704710
err = PTR_ERR(opaquedir);
705711
if (IS_ERR(opaquedir))
@@ -946,7 +952,8 @@ static int ovl_rename(struct inode *olddir, struct dentry *old,
946952

947953
old_cred = ovl_override_creds(old->d_sb);
948954

949-
if (overwrite && new_is_dir && ovl_type_merge_or_lower(new)) {
955+
if (overwrite && new_is_dir && (ovl_type_merge_or_lower(new) ||
956+
ovl_may_have_whiteouts(new))) {
950957
opaquedir = ovl_check_empty_and_clear(new);
951958
err = PTR_ERR(opaquedir);
952959
if (IS_ERR(opaquedir)) {

0 commit comments

Comments
 (0)