Commit 79f687a3 authored by Trond Myklebust's avatar Trond Myklebust

NFS: Fix a performance regression in readdir

Ben Coddington reports that commit 311324ad, by adding the function
nfs_dir_mapping_need_revalidate() that checks page cache validity on
each call to nfs_readdir() causes a performance regression when
the directory is being modified.

If the directory is changing while we're iterating through the directory,
POSIX does not require us to invalidate the page cache unless the user
calls rewinddir(). However, we still do want to ensure that we use
readdirplus in order to avoid a load of stat() calls when the user
is doing an 'ls -l' workload.

The fix should be to invalidate the page cache immediately when we're
setting the NFS_INO_ADVISE_RDPLUS bit.
Reported-by: default avatarBenjamin Coddington <bcodding@redhat.com>
Fixes: 311324ad ("NFS: Be more aggressive in using readdirplus...")
Reviewed-by: default avatarBenjamin Coddington <bcodding@redhat.com>
Tested-by: default avatarBenjamin Coddington <bcodding@redhat.com>
Signed-off-by: default avatarTrond Myklebust <trond.myklebust@primarydata.com>
parent f36ab161
...@@ -477,7 +477,7 @@ void nfs_force_use_readdirplus(struct inode *dir) ...@@ -477,7 +477,7 @@ void nfs_force_use_readdirplus(struct inode *dir)
{ {
if (!list_empty(&NFS_I(dir)->open_files)) { if (!list_empty(&NFS_I(dir)->open_files)) {
nfs_advise_use_readdirplus(dir); nfs_advise_use_readdirplus(dir);
nfs_zap_mapping(dir, dir->i_mapping); invalidate_mapping_pages(dir->i_mapping, 0, -1);
} }
} }
...@@ -886,17 +886,6 @@ int uncached_readdir(nfs_readdir_descriptor_t *desc) ...@@ -886,17 +886,6 @@ int uncached_readdir(nfs_readdir_descriptor_t *desc)
goto out; goto out;
} }
static bool nfs_dir_mapping_need_revalidate(struct inode *dir)
{
struct nfs_inode *nfsi = NFS_I(dir);
if (nfs_attribute_cache_expired(dir))
return true;
if (nfsi->cache_validity & NFS_INO_INVALID_DATA)
return true;
return false;
}
/* The file offset position represents the dirent entry number. A /* The file offset position represents the dirent entry number. A
last cookie cache takes care of the common case of reading the last cookie cache takes care of the common case of reading the
whole directory. whole directory.
...@@ -928,7 +917,7 @@ static int nfs_readdir(struct file *file, struct dir_context *ctx) ...@@ -928,7 +917,7 @@ static int nfs_readdir(struct file *file, struct dir_context *ctx)
desc->decode = NFS_PROTO(inode)->decode_dirent; desc->decode = NFS_PROTO(inode)->decode_dirent;
desc->plus = nfs_use_readdirplus(inode, ctx) ? 1 : 0; desc->plus = nfs_use_readdirplus(inode, ctx) ? 1 : 0;
if (ctx->pos == 0 || nfs_dir_mapping_need_revalidate(inode)) if (ctx->pos == 0 || nfs_attribute_cache_expired(inode))
res = nfs_revalidate_mapping(inode, file->f_mapping); res = nfs_revalidate_mapping(inode, file->f_mapping);
if (res < 0) if (res < 0)
goto out; goto out;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment