linux-nfs.vger.kernel.org archive mirror
 help / color / mirror / Atom feed
* [PATCH v2] nfs: optimise readdir cache page invalidation
@ 2020-01-23  1:45 Dai Ngo
  2020-01-23 18:38 ` Trond Myklebust
  0 siblings, 1 reply; 2+ messages in thread
From: Dai Ngo @ 2020-01-23  1:45 UTC (permalink / raw)
  To: anna.schumaker, trond.myklebust, linux-nfs

When the directory is large and it's being modified by one client
while another client is doing the 'ls -l' on the same directory then
the cache page invalidation from nfs_force_use_readdirplus causes
the reading client to keep restarting READDIRPLUS from cookie 0
which causes the 'ls -l' to take a very long time to complete,
possibly never completing.

Currently when nfs_force_use_readdirplus is called to switch from
READDIR to READDIRPLUS, it invalidates all the cached pages of the
directory. This cache page invalidation causes the next nfs_readdir
to re-read the directory content from cookie 0.

This patch is to optimise the cache invalidation in
nfs_force_use_readdirplus by only truncating the cached pages from
last page index accessed to the end the file. It also marks the
inode to delay invalidating all the cached page of the directory
until the next initial nfs_readdir of the next 'ls' instance.

Signed-off-by: Dai Ngo <dai.ngo@oracle.com>

---
 fs/nfs/dir.c           | 13 ++++++++++++-
 include/linux/nfs_fs.h |  3 +++
 2 files changed, 15 insertions(+), 1 deletion(-)

diff --git a/fs/nfs/dir.c b/fs/nfs/dir.c
index e180033e35cf..16e3208c27d6 100644
--- a/fs/nfs/dir.c
+++ b/fs/nfs/dir.c
@@ -437,7 +437,9 @@ void nfs_force_use_readdirplus(struct inode *dir)
 	if (nfs_server_capable(dir, NFS_CAP_READDIRPLUS) &&
 	    !list_empty(&nfsi->open_files)) {
 		set_bit(NFS_INO_ADVISE_RDPLUS, &nfsi->flags);
-		invalidate_mapping_pages(dir->i_mapping, 0, -1);
+		nfs_zap_mapping(dir, dir->i_mapping);
+		invalidate_mapping_pages(dir->i_mapping,
+			nfsi->page_index + 1, -1);
 	}
 }
 
@@ -709,6 +711,9 @@ struct page *get_cache_page(nfs_readdir_descriptor_t *desc)
 int find_cache_page(nfs_readdir_descriptor_t *desc)
 {
 	int res;
+	struct inode *inode;
+	struct nfs_inode *nfsi;
+	struct dentry *dentry;
 
 	desc->page = get_cache_page(desc);
 	if (IS_ERR(desc->page))
@@ -717,6 +722,12 @@ int find_cache_page(nfs_readdir_descriptor_t *desc)
 	res = nfs_readdir_search_array(desc);
 	if (res != 0)
 		cache_page_release(desc);
+
+	dentry = file_dentry(desc->file);
+	inode = d_inode(dentry);
+	nfsi = NFS_I(inode);
+	nfsi->page_index = desc->page_index;
+
 	return res;
 }
 
diff --git a/include/linux/nfs_fs.h b/include/linux/nfs_fs.h
index c06b1fd130f3..a5f8f03ecd59 100644
--- a/include/linux/nfs_fs.h
+++ b/include/linux/nfs_fs.h
@@ -168,6 +168,9 @@ struct nfs_inode {
 	struct rw_semaphore	rmdir_sem;
 	struct mutex		commit_mutex;
 
+	/* track last access to cached pages */
+	unsigned long		page_index;
+
 #if IS_ENABLED(CONFIG_NFS_V4)
 	struct nfs4_cached_acl	*nfs4_acl;
         /* NFSv4 state */
-- 
1.8.3.1


^ permalink raw reply related	[flat|nested] 2+ messages in thread

* Re: [PATCH v2] nfs: optimise readdir cache page invalidation
  2020-01-23  1:45 [PATCH v2] nfs: optimise readdir cache page invalidation Dai Ngo
@ 2020-01-23 18:38 ` Trond Myklebust
  0 siblings, 0 replies; 2+ messages in thread
From: Trond Myklebust @ 2020-01-23 18:38 UTC (permalink / raw)
  To: dai.ngo, linux-nfs, anna.schumaker

On Wed, 2020-01-22 at 20:45 -0500, Dai Ngo wrote:
> When the directory is large and it's being modified by one client
> while another client is doing the 'ls -l' on the same directory then
> the cache page invalidation from nfs_force_use_readdirplus causes
> the reading client to keep restarting READDIRPLUS from cookie 0
> which causes the 'ls -l' to take a very long time to complete,
> possibly never completing.
> 
> Currently when nfs_force_use_readdirplus is called to switch from
> READDIR to READDIRPLUS, it invalidates all the cached pages of the
> directory. This cache page invalidation causes the next nfs_readdir
> to re-read the directory content from cookie 0.
> 
> This patch is to optimise the cache invalidation in
> nfs_force_use_readdirplus by only truncating the cached pages from
> last page index accessed to the end the file. It also marks the
> inode to delay invalidating all the cached page of the directory
> until the next initial nfs_readdir of the next 'ls' instance.
> 
> Signed-off-by: Dai Ngo <dai.ngo@oracle.com>

Reviewed-by: Trond Myklebust <trond.myklebust@hammerspace.com>

> 
> ---
>  fs/nfs/dir.c           | 13 ++++++++++++-
>  include/linux/nfs_fs.h |  3 +++
>  2 files changed, 15 insertions(+), 1 deletion(-)
> 
> diff --git a/fs/nfs/dir.c b/fs/nfs/dir.c
> index e180033e35cf..16e3208c27d6 100644
> --- a/fs/nfs/dir.c
> +++ b/fs/nfs/dir.c
> @@ -437,7 +437,9 @@ void nfs_force_use_readdirplus(struct inode *dir)
>  	if (nfs_server_capable(dir, NFS_CAP_READDIRPLUS) &&
>  	    !list_empty(&nfsi->open_files)) {
>  		set_bit(NFS_INO_ADVISE_RDPLUS, &nfsi->flags);
> -		invalidate_mapping_pages(dir->i_mapping, 0, -1);
> +		nfs_zap_mapping(dir, dir->i_mapping);
> +		invalidate_mapping_pages(dir->i_mapping,
> +			nfsi->page_index + 1, -1);
>  	}
>  }
>  
> @@ -709,6 +711,9 @@ struct page
> *get_cache_page(nfs_readdir_descriptor_t *desc)
>  int find_cache_page(nfs_readdir_descriptor_t *desc)
>  {
>  	int res;
> +	struct inode *inode;
> +	struct nfs_inode *nfsi;
> +	struct dentry *dentry;
>  
>  	desc->page = get_cache_page(desc);
>  	if (IS_ERR(desc->page))
> @@ -717,6 +722,12 @@ int find_cache_page(nfs_readdir_descriptor_t
> *desc)
>  	res = nfs_readdir_search_array(desc);
>  	if (res != 0)
>  		cache_page_release(desc);
> +
> +	dentry = file_dentry(desc->file);
> +	inode = d_inode(dentry);
> +	nfsi = NFS_I(inode);
> +	nfsi->page_index = desc->page_index;
> +
>  	return res;
>  }
>  
> diff --git a/include/linux/nfs_fs.h b/include/linux/nfs_fs.h
> index c06b1fd130f3..a5f8f03ecd59 100644
> --- a/include/linux/nfs_fs.h
> +++ b/include/linux/nfs_fs.h
> @@ -168,6 +168,9 @@ struct nfs_inode {
>  	struct rw_semaphore	rmdir_sem;
>  	struct mutex		commit_mutex;
>  
> +	/* track last access to cached pages */
> +	unsigned long		page_index;
> +
>  #if IS_ENABLED(CONFIG_NFS_V4)
>  	struct nfs4_cached_acl	*nfs4_acl;
>          /* NFSv4 state */
-- 
Trond Myklebust
Linux NFS client maintainer, Hammerspace
trond.myklebust@hammerspace.com



^ permalink raw reply	[flat|nested] 2+ messages in thread

end of thread, other threads:[~2020-01-23 18:39 UTC | newest]

Thread overview: 2+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2020-01-23  1:45 [PATCH v2] nfs: optimise readdir cache page invalidation Dai Ngo
2020-01-23 18:38 ` Trond Myklebust

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).