mirror of
https://github.com/AuxXxilium/linux_dsm_epyc7002.git
synced 2024-11-24 01:30:55 +07:00
ovl: invalidate readdir cache on changes to dir with origin
[ Upstream commit 65cd913ec9d9d71529665924c81015b7ab7d9381 ]
The test in ovl_dentry_version_inc() was out-dated and did not include
the case where readdir cache is used on a non-merge dir that has origin
xattr, indicating that it may contain leftover whiteouts.
To make the code more robust, use the same helper ovl_dir_is_real()
to determine if readdir cache should be used and if readdir cache should
be invalidated.
Fixes: b79e05aaa1
("ovl: no direct iteration for dir with origin xattr")
Link: https://lore.kernel.org/linux-unionfs/CAOQ4uxht70nODhNHNwGFMSqDyOKLXOKrY0H6g849os4BQ7cokA@mail.gmail.com/
Cc: Chris Murphy <lists@colorremedies.com>
Signed-off-by: Amir Goldstein <amir73il@gmail.com>
Signed-off-by: Miklos Szeredi <mszeredi@redhat.com>
Signed-off-by: Sasha Levin <sashal@kernel.org>
This commit is contained in:
parent
082fa65bf6
commit
0f8528c78f
@ -308,9 +308,6 @@ int ovl_check_setxattr(struct dentry *dentry, struct dentry *upperdentry,
|
|||||||
enum ovl_xattr ox, const void *value, size_t size,
|
enum ovl_xattr ox, const void *value, size_t size,
|
||||||
int xerr);
|
int xerr);
|
||||||
int ovl_set_impure(struct dentry *dentry, struct dentry *upperdentry);
|
int ovl_set_impure(struct dentry *dentry, struct dentry *upperdentry);
|
||||||
void ovl_set_flag(unsigned long flag, struct inode *inode);
|
|
||||||
void ovl_clear_flag(unsigned long flag, struct inode *inode);
|
|
||||||
bool ovl_test_flag(unsigned long flag, struct inode *inode);
|
|
||||||
bool ovl_inuse_trylock(struct dentry *dentry);
|
bool ovl_inuse_trylock(struct dentry *dentry);
|
||||||
void ovl_inuse_unlock(struct dentry *dentry);
|
void ovl_inuse_unlock(struct dentry *dentry);
|
||||||
bool ovl_is_inuse(struct dentry *dentry);
|
bool ovl_is_inuse(struct dentry *dentry);
|
||||||
@ -324,6 +321,21 @@ char *ovl_get_redirect_xattr(struct ovl_fs *ofs, struct dentry *dentry,
|
|||||||
int padding);
|
int padding);
|
||||||
int ovl_sync_status(struct ovl_fs *ofs);
|
int ovl_sync_status(struct ovl_fs *ofs);
|
||||||
|
|
||||||
|
static inline void ovl_set_flag(unsigned long flag, struct inode *inode)
|
||||||
|
{
|
||||||
|
set_bit(flag, &OVL_I(inode)->flags);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void ovl_clear_flag(unsigned long flag, struct inode *inode)
|
||||||
|
{
|
||||||
|
clear_bit(flag, &OVL_I(inode)->flags);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline bool ovl_test_flag(unsigned long flag, struct inode *inode)
|
||||||
|
{
|
||||||
|
return test_bit(flag, &OVL_I(inode)->flags);
|
||||||
|
}
|
||||||
|
|
||||||
static inline bool ovl_is_impuredir(struct super_block *sb,
|
static inline bool ovl_is_impuredir(struct super_block *sb,
|
||||||
struct dentry *dentry)
|
struct dentry *dentry)
|
||||||
{
|
{
|
||||||
@ -427,6 +439,18 @@ int ovl_workdir_cleanup(struct inode *dir, struct vfsmount *mnt,
|
|||||||
struct dentry *dentry, int level);
|
struct dentry *dentry, int level);
|
||||||
int ovl_indexdir_cleanup(struct ovl_fs *ofs);
|
int ovl_indexdir_cleanup(struct ovl_fs *ofs);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Can we iterate real dir directly?
|
||||||
|
*
|
||||||
|
* Non-merge dir may contain whiteouts from a time it was a merge upper, before
|
||||||
|
* lower dir was removed under it and possibly before it was rotated from upper
|
||||||
|
* to lower layer.
|
||||||
|
*/
|
||||||
|
static inline bool ovl_dir_is_real(struct dentry *dir)
|
||||||
|
{
|
||||||
|
return !ovl_test_flag(OVL_WHITEOUTS, d_inode(dir));
|
||||||
|
}
|
||||||
|
|
||||||
/* inode.c */
|
/* inode.c */
|
||||||
int ovl_set_nlink_upper(struct dentry *dentry);
|
int ovl_set_nlink_upper(struct dentry *dentry);
|
||||||
int ovl_set_nlink_lower(struct dentry *dentry);
|
int ovl_set_nlink_lower(struct dentry *dentry);
|
||||||
|
@ -319,18 +319,6 @@ static inline int ovl_dir_read(struct path *realpath,
|
|||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* Can we iterate real dir directly?
|
|
||||||
*
|
|
||||||
* Non-merge dir may contain whiteouts from a time it was a merge upper, before
|
|
||||||
* lower dir was removed under it and possibly before it was rotated from upper
|
|
||||||
* to lower layer.
|
|
||||||
*/
|
|
||||||
static bool ovl_dir_is_real(struct dentry *dir)
|
|
||||||
{
|
|
||||||
return !ovl_test_flag(OVL_WHITEOUTS, d_inode(dir));
|
|
||||||
}
|
|
||||||
|
|
||||||
static void ovl_dir_reset(struct file *file)
|
static void ovl_dir_reset(struct file *file)
|
||||||
{
|
{
|
||||||
struct ovl_dir_file *od = file->private_data;
|
struct ovl_dir_file *od = file->private_data;
|
||||||
|
@ -419,18 +419,20 @@ void ovl_inode_update(struct inode *inode, struct dentry *upperdentry)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ovl_dentry_version_inc(struct dentry *dentry, bool impurity)
|
static void ovl_dir_version_inc(struct dentry *dentry, bool impurity)
|
||||||
{
|
{
|
||||||
struct inode *inode = d_inode(dentry);
|
struct inode *inode = d_inode(dentry);
|
||||||
|
|
||||||
WARN_ON(!inode_is_locked(inode));
|
WARN_ON(!inode_is_locked(inode));
|
||||||
|
WARN_ON(!d_is_dir(dentry));
|
||||||
/*
|
/*
|
||||||
* Version is used by readdir code to keep cache consistent. For merge
|
* Version is used by readdir code to keep cache consistent.
|
||||||
* dirs all changes need to be noted. For non-merge dirs, cache only
|
* For merge dirs (or dirs with origin) all changes need to be noted.
|
||||||
* contains impure (ones which have been copied up and have origins)
|
* For non-merge dirs, cache contains only impure entries (i.e. ones
|
||||||
* entries, so only need to note changes to impure entries.
|
* which have been copied up and have origins), so only need to note
|
||||||
|
* changes to impure entries.
|
||||||
*/
|
*/
|
||||||
if (OVL_TYPE_MERGE(ovl_path_type(dentry)) || impurity)
|
if (!ovl_dir_is_real(dentry) || impurity)
|
||||||
OVL_I(inode)->version++;
|
OVL_I(inode)->version++;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -439,7 +441,7 @@ void ovl_dir_modified(struct dentry *dentry, bool impurity)
|
|||||||
/* Copy mtime/ctime */
|
/* Copy mtime/ctime */
|
||||||
ovl_copyattr(d_inode(ovl_dentry_upper(dentry)), d_inode(dentry));
|
ovl_copyattr(d_inode(ovl_dentry_upper(dentry)), d_inode(dentry));
|
||||||
|
|
||||||
ovl_dentry_version_inc(dentry, impurity);
|
ovl_dir_version_inc(dentry, impurity);
|
||||||
}
|
}
|
||||||
|
|
||||||
u64 ovl_dentry_version_get(struct dentry *dentry)
|
u64 ovl_dentry_version_get(struct dentry *dentry)
|
||||||
@ -634,21 +636,6 @@ int ovl_set_impure(struct dentry *dentry, struct dentry *upperdentry)
|
|||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ovl_set_flag(unsigned long flag, struct inode *inode)
|
|
||||||
{
|
|
||||||
set_bit(flag, &OVL_I(inode)->flags);
|
|
||||||
}
|
|
||||||
|
|
||||||
void ovl_clear_flag(unsigned long flag, struct inode *inode)
|
|
||||||
{
|
|
||||||
clear_bit(flag, &OVL_I(inode)->flags);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool ovl_test_flag(unsigned long flag, struct inode *inode)
|
|
||||||
{
|
|
||||||
return test_bit(flag, &OVL_I(inode)->flags);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Caller must hold a reference to inode to prevent it from being freed while
|
* Caller must hold a reference to inode to prevent it from being freed while
|
||||||
* it is marked inuse.
|
* it is marked inuse.
|
||||||
|
Loading…
Reference in New Issue
Block a user