Remove zpl_revalidate
This patch removes the need for zpl_revalidate altogether. There were 3 main reasons why we used d_revalidate: 1. periodic automounted snapshots umount deferral 2. negative dentries created before snapshot rollback 3. stale inodes referenced by dentry cache after snapshot rollback Periodic snapshots deferral solution introduces zfs_exit_fs function, which is called as a part of ZFS_EXIT(zfsvfs_t) macro. Negative dentries and stale inodes are solved by flushing the dcache for the particular dataset on zfs_resume_fs call. This patch also removes now unused HAVE_S_D_OP configure test. Reviewed-by: Aleksa Sarai <cyphar@cyphar.com> Reviewed-by: Brian Behlendorf <behlendorf1@llnl.gov> Signed-off-by: Pavel Snajdr <snajpa@snajpa.net> Closes #8774 Closes #9549
This commit is contained in:
parent
f15d6a5457
commit
5a6ac4cffc
|
@ -150,29 +150,6 @@ AC_DEFUN([ZFS_AC_KERNEL_CONST_DENTRY_OPERATIONS], [
|
||||||
])
|
])
|
||||||
])
|
])
|
||||||
|
|
||||||
dnl #
|
|
||||||
dnl # 2.6.38 API change
|
|
||||||
dnl # Added sb->s_d_op default dentry_operations member
|
|
||||||
dnl #
|
|
||||||
AC_DEFUN([ZFS_AC_KERNEL_SRC_S_D_OP], [
|
|
||||||
ZFS_LINUX_TEST_SRC([super_block_s_d_op], [
|
|
||||||
#include <linux/fs.h>
|
|
||||||
],[
|
|
||||||
struct super_block sb __attribute__ ((unused));
|
|
||||||
sb.s_d_op = NULL;
|
|
||||||
])
|
|
||||||
])
|
|
||||||
|
|
||||||
AC_DEFUN([ZFS_AC_KERNEL_S_D_OP], [
|
|
||||||
AC_MSG_CHECKING([whether super_block has s_d_op])
|
|
||||||
ZFS_LINUX_TEST_RESULT([super_block_s_d_op], [
|
|
||||||
AC_MSG_RESULT(yes)
|
|
||||||
AC_DEFINE(HAVE_S_D_OP, 1, [struct super_block has s_d_op])
|
|
||||||
], [
|
|
||||||
AC_MSG_RESULT(no)
|
|
||||||
])
|
|
||||||
])
|
|
||||||
|
|
||||||
AC_DEFUN([ZFS_AC_KERNEL_SRC_DENTRY], [
|
AC_DEFUN([ZFS_AC_KERNEL_SRC_DENTRY], [
|
||||||
ZFS_AC_KERNEL_SRC_D_MAKE_ROOT
|
ZFS_AC_KERNEL_SRC_D_MAKE_ROOT
|
||||||
ZFS_AC_KERNEL_SRC_D_OBTAIN_ALIAS
|
ZFS_AC_KERNEL_SRC_D_OBTAIN_ALIAS
|
||||||
|
@ -180,7 +157,6 @@ AC_DEFUN([ZFS_AC_KERNEL_SRC_DENTRY], [
|
||||||
ZFS_AC_KERNEL_SRC_D_SET_D_OP
|
ZFS_AC_KERNEL_SRC_D_SET_D_OP
|
||||||
ZFS_AC_KERNEL_SRC_D_REVALIDATE_NAMEIDATA
|
ZFS_AC_KERNEL_SRC_D_REVALIDATE_NAMEIDATA
|
||||||
ZFS_AC_KERNEL_SRC_CONST_DENTRY_OPERATIONS
|
ZFS_AC_KERNEL_SRC_CONST_DENTRY_OPERATIONS
|
||||||
ZFS_AC_KERNEL_SRC_S_D_OP
|
|
||||||
])
|
])
|
||||||
|
|
||||||
AC_DEFUN([ZFS_AC_KERNEL_DENTRY], [
|
AC_DEFUN([ZFS_AC_KERNEL_DENTRY], [
|
||||||
|
@ -190,5 +166,4 @@ AC_DEFUN([ZFS_AC_KERNEL_DENTRY], [
|
||||||
ZFS_AC_KERNEL_D_SET_D_OP
|
ZFS_AC_KERNEL_D_SET_D_OP
|
||||||
ZFS_AC_KERNEL_D_REVALIDATE_NAMEIDATA
|
ZFS_AC_KERNEL_D_REVALIDATE_NAMEIDATA
|
||||||
ZFS_AC_KERNEL_CONST_DENTRY_OPERATIONS
|
ZFS_AC_KERNEL_CONST_DENTRY_OPERATIONS
|
||||||
ZFS_AC_KERNEL_S_D_OP
|
|
||||||
])
|
])
|
||||||
|
|
|
@ -197,6 +197,7 @@ typedef struct zfid_long {
|
||||||
extern int zfs_suspend_fs(zfsvfs_t *zfsvfs);
|
extern int zfs_suspend_fs(zfsvfs_t *zfsvfs);
|
||||||
extern int zfs_resume_fs(zfsvfs_t *zfsvfs, struct dsl_dataset *ds);
|
extern int zfs_resume_fs(zfsvfs_t *zfsvfs, struct dsl_dataset *ds);
|
||||||
extern int zfs_end_fs(zfsvfs_t *zfsvfs, struct dsl_dataset *ds);
|
extern int zfs_end_fs(zfsvfs_t *zfsvfs, struct dsl_dataset *ds);
|
||||||
|
extern void zfs_exit_fs(zfsvfs_t *zfsvfs);
|
||||||
extern int zfs_userspace_one(zfsvfs_t *zfsvfs, zfs_userquota_prop_t type,
|
extern int zfs_userspace_one(zfsvfs_t *zfsvfs, zfs_userquota_prop_t type,
|
||||||
const char *domain, uint64_t rid, uint64_t *valuep);
|
const char *domain, uint64_t rid, uint64_t *valuep);
|
||||||
extern int zfs_userspace_many(zfsvfs_t *zfsvfs, zfs_userquota_prop_t type,
|
extern int zfs_userspace_many(zfsvfs_t *zfsvfs, zfs_userquota_prop_t type,
|
||||||
|
|
|
@ -84,6 +84,7 @@ do { \
|
||||||
/* Must be called before exiting the operation. */
|
/* Must be called before exiting the operation. */
|
||||||
#define ZFS_EXIT(zfsvfs) \
|
#define ZFS_EXIT(zfsvfs) \
|
||||||
do { \
|
do { \
|
||||||
|
zfs_exit_fs(zfsvfs); \
|
||||||
rrm_exit(&(zfsvfs)->z_teardown_lock, FTAG); \
|
rrm_exit(&(zfsvfs)->z_teardown_lock, FTAG); \
|
||||||
} while (0)
|
} while (0)
|
||||||
#define ZPL_EXIT(zfsvfs) ZFS_EXIT(zfsvfs)
|
#define ZPL_EXIT(zfsvfs) ZFS_EXIT(zfsvfs)
|
||||||
|
|
|
@ -45,7 +45,6 @@ extern const struct inode_operations zpl_inode_operations;
|
||||||
extern const struct inode_operations zpl_dir_inode_operations;
|
extern const struct inode_operations zpl_dir_inode_operations;
|
||||||
extern const struct inode_operations zpl_symlink_inode_operations;
|
extern const struct inode_operations zpl_symlink_inode_operations;
|
||||||
extern const struct inode_operations zpl_special_inode_operations;
|
extern const struct inode_operations zpl_special_inode_operations;
|
||||||
extern dentry_operations_t zpl_dentry_operations;
|
|
||||||
|
|
||||||
/* zpl_file.c */
|
/* zpl_file.c */
|
||||||
extern ssize_t zpl_read_common(struct inode *ip, const char *buf,
|
extern ssize_t zpl_read_common(struct inode *ip, const char *buf,
|
||||||
|
|
|
@ -1931,9 +1931,6 @@ zfs_domount(struct super_block *sb, zfs_mnt_t *zm, int silent)
|
||||||
sb->s_op = &zpl_super_operations;
|
sb->s_op = &zpl_super_operations;
|
||||||
sb->s_xattr = zpl_xattr_handlers;
|
sb->s_xattr = zpl_xattr_handlers;
|
||||||
sb->s_export_op = &zpl_export_operations;
|
sb->s_export_op = &zpl_export_operations;
|
||||||
#ifdef HAVE_S_D_OP
|
|
||||||
sb->s_d_op = &zpl_dentry_operations;
|
|
||||||
#endif /* HAVE_S_D_OP */
|
|
||||||
|
|
||||||
/* Set features for file system. */
|
/* Set features for file system. */
|
||||||
zfs_set_fuid_feature(zfsvfs);
|
zfs_set_fuid_feature(zfsvfs);
|
||||||
|
@ -2303,6 +2300,16 @@ zfs_resume_fs(zfsvfs_t *zfsvfs, dsl_dataset_t *ds)
|
||||||
zfs_unlinked_drain(zfsvfs);
|
zfs_unlinked_drain(zfsvfs);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Most of the time zfs_suspend_fs is used for changing the contents
|
||||||
|
* of the underlying dataset. ZFS rollback and receive operations
|
||||||
|
* might create files for which negative dentries are present in
|
||||||
|
* the cache. Since walking the dcache would require a lot of GPL-only
|
||||||
|
* code duplication, it's much easier on these rather rare occasions
|
||||||
|
* just to flush the whole dcache for the given dataset/filesystem.
|
||||||
|
*/
|
||||||
|
shrink_dcache_sb(zfsvfs->z_sb);
|
||||||
|
|
||||||
bail:
|
bail:
|
||||||
if (err != 0)
|
if (err != 0)
|
||||||
zfsvfs->z_unmounted = B_TRUE;
|
zfsvfs->z_unmounted = B_TRUE;
|
||||||
|
@ -2353,6 +2360,26 @@ zfs_end_fs(zfsvfs_t *zfsvfs, dsl_dataset_t *ds)
|
||||||
return (0);
|
return (0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Automounted snapshots rely on periodic revalidation
|
||||||
|
* to defer snapshots from being automatically unmounted.
|
||||||
|
*/
|
||||||
|
|
||||||
|
inline void
|
||||||
|
zfs_exit_fs(zfsvfs_t *zfsvfs)
|
||||||
|
{
|
||||||
|
if (!zfsvfs->z_issnap)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (time_after(jiffies, zfsvfs->z_snap_defer_time +
|
||||||
|
MAX(zfs_expire_snapshot * HZ / 2, HZ))) {
|
||||||
|
zfsvfs->z_snap_defer_time = jiffies;
|
||||||
|
zfsctl_snapshot_unmount_delay(zfsvfs->z_os->os_spa,
|
||||||
|
dmu_objset_id(zfsvfs->z_os),
|
||||||
|
zfs_expire_snapshot);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
zfs_set_version(zfsvfs_t *zfsvfs, uint64_t newvers)
|
zfs_set_version(zfsvfs_t *zfsvfs, uint64_t newvers)
|
||||||
{
|
{
|
||||||
|
|
|
@ -70,9 +70,6 @@ zpl_lookup(struct inode *dir, struct dentry *dentry, unsigned int flags)
|
||||||
|
|
||||||
spin_lock(&dentry->d_lock);
|
spin_lock(&dentry->d_lock);
|
||||||
dentry->d_time = jiffies;
|
dentry->d_time = jiffies;
|
||||||
#ifndef HAVE_S_D_OP
|
|
||||||
d_set_d_op(dentry, &zpl_dentry_operations);
|
|
||||||
#endif /* HAVE_S_D_OP */
|
|
||||||
spin_unlock(&dentry->d_lock);
|
spin_unlock(&dentry->d_lock);
|
||||||
|
|
||||||
if (error) {
|
if (error) {
|
||||||
|
@ -655,59 +652,6 @@ zpl_fallocate(struct inode *ip, int mode, loff_t offset, loff_t len)
|
||||||
}
|
}
|
||||||
#endif /* HAVE_INODE_FALLOCATE */
|
#endif /* HAVE_INODE_FALLOCATE */
|
||||||
|
|
||||||
static int
|
|
||||||
#ifdef HAVE_D_REVALIDATE_NAMEIDATA
|
|
||||||
zpl_revalidate(struct dentry *dentry, struct nameidata *nd)
|
|
||||||
{
|
|
||||||
unsigned int flags = (nd ? nd->flags : 0);
|
|
||||||
#else
|
|
||||||
zpl_revalidate(struct dentry *dentry, unsigned int flags)
|
|
||||||
{
|
|
||||||
#endif /* HAVE_D_REVALIDATE_NAMEIDATA */
|
|
||||||
/* CSTYLED */
|
|
||||||
zfsvfs_t *zfsvfs = dentry->d_sb->s_fs_info;
|
|
||||||
int error;
|
|
||||||
|
|
||||||
if (flags & LOOKUP_RCU)
|
|
||||||
return (-ECHILD);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Automounted snapshots rely on periodic dentry revalidation
|
|
||||||
* to defer snapshots from being automatically unmounted.
|
|
||||||
*/
|
|
||||||
if (zfsvfs->z_issnap) {
|
|
||||||
if (time_after(jiffies, zfsvfs->z_snap_defer_time +
|
|
||||||
MAX(zfs_expire_snapshot * HZ / 2, HZ))) {
|
|
||||||
zfsvfs->z_snap_defer_time = jiffies;
|
|
||||||
zfsctl_snapshot_unmount_delay(zfsvfs->z_os->os_spa,
|
|
||||||
dmu_objset_id(zfsvfs->z_os), zfs_expire_snapshot);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* After a rollback negative dentries created before the rollback
|
|
||||||
* time must be invalidated. Otherwise they can obscure files which
|
|
||||||
* are only present in the rolled back dataset.
|
|
||||||
*/
|
|
||||||
if (dentry->d_inode == NULL) {
|
|
||||||
spin_lock(&dentry->d_lock);
|
|
||||||
error = time_before(dentry->d_time, zfsvfs->z_rollback_time);
|
|
||||||
spin_unlock(&dentry->d_lock);
|
|
||||||
|
|
||||||
if (error)
|
|
||||||
return (0);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* The dentry may reference a stale inode if a mounted file system
|
|
||||||
* was rolled back to a point in time where the object didn't exist.
|
|
||||||
*/
|
|
||||||
if (dentry->d_inode && ITOZ(dentry->d_inode)->z_is_stale)
|
|
||||||
return (0);
|
|
||||||
|
|
||||||
return (1);
|
|
||||||
}
|
|
||||||
|
|
||||||
const struct inode_operations zpl_inode_operations = {
|
const struct inode_operations zpl_inode_operations = {
|
||||||
.setattr = zpl_setattr,
|
.setattr = zpl_setattr,
|
||||||
.getattr = zpl_getattr,
|
.getattr = zpl_getattr,
|
||||||
|
@ -820,7 +764,3 @@ const struct inode_operations zpl_special_inode_operations = {
|
||||||
#endif /* HAVE_GET_ACL | HAVE_CHECK_ACL | HAVE_PERMISSION */
|
#endif /* HAVE_GET_ACL | HAVE_CHECK_ACL | HAVE_PERMISSION */
|
||||||
#endif /* CONFIG_FS_POSIX_ACL */
|
#endif /* CONFIG_FS_POSIX_ACL */
|
||||||
};
|
};
|
||||||
|
|
||||||
dentry_operations_t zpl_dentry_operations = {
|
|
||||||
.d_revalidate = zpl_revalidate,
|
|
||||||
};
|
|
||||||
|
|
Loading…
Reference in New Issue