cvs commit: src/sys/nfsclient nfs_vnops.c

Bruce Evans bde at zeta.org.au
Sun Oct 15 00:09:45 PDT 2006


On Sat, 14 Oct 2006 mjacob at FreeBSD.org wrote:

> On Sat, 14 Oct 2006, mjacob at FreeBSD.org wrote:
>
>> So, inquiring minds want to know why defaults/rc.conf still sets this to 2 
>> then.
>
> *smack*- I meant in RE_6.

Seems to be just because no one got around to it in RELENG_6.

Recovering some context:

On a date too hard for me to recover, Kris wrote:

>> On Sun, Oct 15, 2006 at 12:37:28PM +1000, Bruce Evans wrote:
>> 
>> > ISTR a discussion of fixing this in rc.conf, but nothing seems to have been
>> > committed.  The setting is confusing in the kernel too:
>> 
>> ----------------------------
>> revision 1.285
>> date: 2006/05/24 00:06:14;  author: kris;  state: Exp;  lines: +1 -1
>> Increase the nfs access cache timeout from 2 to 60.  The latter is a
>> more appropriate value and is also the default set by the kernel.  I
>> could not find a justification of why rc.conf began overriding it back
>> in 1998.
>> 
>> This dramatically cuts NFS traffic on e.g. a busy system with NFS root.
>> 
>> Reviewed by:    mohans
>> MFC After:      2 weeks
>> ----------------------------

Thanks.  I must have been confused about which machine I was on when I
grepped for this.  In the FreeBSD cluster, it is on a -current machine
but not on a 6.1 machine.

My previous mail more or less explained why rc.conf began setting it to
2 in 1998:  It didn't exist before then, so it was initially set to a
conservative default of 2.  Only the mount options for the _attribute_
cache existed before then.  rc.conf and fstab never had any special
support for these, so I think rc.conf shouldn't have any special support
for the _access_ cache timeout (it now defaults to setting it to its
kernel default value).

I just noticed even sillier configuration bogusness for the default
attribute cache timeouts:
- in 1994, the default timeouts were ifdefed.  I think there was no
   other (easy) way to change the timeouts.
- in May 1998, mount_nfs started supporting setting the timeouts per-mount.
- in June 1998, 6 weeks after 1994 hack became unnecesary, the timeouts
   were turned into first class kernel options (put in an options header).

I just got around to looking at some nfs RFCs.  nfs4 has a lot to say
about the problem of too many RPCs.  nfs3 has less to say.

I think I now know how to fix the second largest source of extra RPCs
properly:

We used clear the _attribute_ cache (and the access cache as a side
effect?) at the _end_ of nfs_open(), but we are supposed to clear it
at the _start_ of every open().  We can't do the latter properly because
a fresh set of attributes are needed or at least preferred when
nfs_lookup() is called before nfs_open().  Clearing the attribute cache
at the end of nfs_open() seems to be just a hack which gives a good
chance of the clearing living until the next open().  It doesn't always
work.  Clearing the cache in nfs_close() is further from always working.
It fails if the file is re-open()ed before the first open() instance
is close()d, and wasn't done.

Now we clear the attribute cache at the start of nfs_open() and clear
it in nfs_close().  For simple open-close sequences, this gives 2
clearings and 2 refreshes.  First, nfs_lookup() refreshes; then
nfs_open() clears and refreshes; finally, nfs_close() clears.

I think the correct refreshing is:
- force a clear in nfs_lookup() only if we are sure that the lookup is for
   open().  Then refresh normally (if we just cleared or the cache was
   already clear).  Implement this using a namei() flag?
- force a clear in nfs_open() only if we aren't sure that nfs_lookup()
   didn't already do it.  This is for safety in cases like core dumps
   where namei() isn't called by open(2) and we forget to tell namei()
   that the lookup is for open().  Then refresh normally.  Implement this
   using a generation count or another flag?

Bruce


More information about the cvs-all mailing list