Return-Path: linux-nfs-owner@vger.kernel.org Received: from mx1.redhat.com ([209.132.183.28]:17290 "EHLO mx1.redhat.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1753257Ab3FENwD (ORCPT ); Wed, 5 Jun 2013 09:52:03 -0400 Received: from int-mx10.intmail.prod.int.phx2.redhat.com (int-mx10.intmail.prod.int.phx2.redhat.com [10.5.11.23]) by mx1.redhat.com (8.14.4/8.14.4) with ESMTP id r55Dq3CZ032670 (version=TLSv1/SSLv3 cipher=DHE-RSA-AES256-SHA bits=256 verify=OK) for ; Wed, 5 Jun 2013 09:52:03 -0400 From: Scott Mayhew To: linux-nfs@vger.kernel.org Cc: jlayton@redhat.com Subject: [PATCH RFC 0/2] NFS: Improve readdir performance Date: Wed, 5 Jun 2013 09:51:59 -0400 Message-Id: <1370440321-56488-1-git-send-email-smayhew@redhat.com> Sender: linux-nfs-owner@vger.kernel.org List-ID: I was investigating some performance concerns from a customer and came across something interesting. Whenever the directory that we're reading is changed on the server, we start re-reading the directory from the beginning. On a LAN connection or in a directory with a small number of entries, the impact isn't too noticeable... but reading a directory with a large number of entries over a WAN connection gets pretty bad. For NFS v3, what happens is that after each on-the-wire READDIR we call nfs_refresh_inode() and from there we get to nfs_update_inode(), where we wind up setting NFS_INO_INVALID_DATA in the directory's cache_validity flags. Then on a subsequent call to nfs_readdir() we call nfs_revalidate_mapping(), and seeing that NFS_INO_INVALIDATE_DATA is set we call nfs_invalidate_mapping(), flushing all our cached data for the directory. So for each nfs_readdir() call, we wind up redoing all of the on-the-wire readdir operations just to get back where we were, and then we're able to get just one more operation's worth of entries on top of that. If the directory on the NFS server is constantly being modified then this winds up being a lot of extra READDIR ops. I had an idea that maybe we could call nfs_refresh_inode() only when we've reached the end of the directory. I talked to Jeff about it and he suggested that maybe we could only revalidate if we're at the beginning of the directory or if nfs_attribute_cache_expired() for the dir. The attached patches take that approach. For example, on a test environment of two VMs, I have a directory of 100,000 entries that takes 981 READDIR operations to read if no modifications being made to the directory at the same time. If I add a 35ms delay between the client and the server and start a script on the server that repeatedly creates and removes a file in the directory being listed I get the following results: [root@localhost ~]# mount -t nfs -o nfsvers=3,nordirplus server:/export /mnt [root@localhost ~]# time /bin/ls /mnt/bigdir >/dev/null real 29m52.594s user 0m0.376s sys 0m2.191s [root@localhost ~]# mountstats --rpc /mnt | grep -A3 READDIR READDIR: 49729 ops (99%) 0 retrans (0%) 0 major timeouts avg bytes sent per op: 144 avg bytes received per op: 4196 backlog wait: 0.003620 RTT: 35.889501 total execute time: 35.925858 (milliseconds) [root@localhost ~]# With the patched kernel, that same test yields these results: [root@localhost ~]# time /bin/ls /mnt/bigdir >/dev/null real 0m35.952s user 0m0.460s sys 0m0.100s [root@localhost ~]# mountstats --rpc /mnt | grep -A3 READDIR READDIR: 981 ops (98%) 0 retrans (0%) 0 major timeouts avg bytes sent per op: 144 avg bytes received per op: 4194 backlog wait: 0.004077 RTT: 35.887870 total execute time: 35.926606 (milliseconds) [root@localhost ~]# For NFS v4, the situation is slightly different. We don't get post-op attributes from each READDIR, so we're not calling nfs_refresh_inode()/nfs_update_inode() after every operation and therefore not updating read_cache_jiffies. If we don't manage to read through the whole directory before the directory attributes from the initial GETATTR expire, then we're going to wind up calling __nfs_revalidate_inode() from nfs_revalidate_mapping(), and the attributes that we get from that GETATTR are going ultimately to lead us into nfs_invalidate_mapping() anyway. If the attached patches make sense then maybe it would be worthwhile to add a GETATTR operation to the compound that gets sent for a READDIR so that read_cache_jiffies stays updated. Finally there's the question of the dentry cache. Any time we find that the parent directory changes we're still going to wind up doing an on-the-wire LOOKUP. I don't think there's anything that can be done about that, but at least these patches prevent us doing the same LOOKUPs multiple times in the course of reading through the directory. -Scott Scott Mayhew (2): NFS: Make nfs_attribute_cache_expired() non-static NFS: Make nfs_readdir revalidate less often fs/nfs/dir.c | 5 +++-- fs/nfs/inode.c | 2 +- include/linux/nfs_fs.h | 1 + 3 files changed, 5 insertions(+), 3 deletions(-) -- 1.7.11.7