svn commit: r220761 - head/sys/fs/nfsclient
Rick Macklem
rmacklem at uoguelph.ca
Mon Apr 18 21:58:25 UTC 2011
> On Sun, Apr 17, 2011 at 11:04:04PM +0000, Rick Macklem wrote:
> > Author: rmacklem
> > Date: Sun Apr 17 23:04:03 2011
> > New Revision: 220761
> > URL: http://svn.freebsd.org/changeset/base/220761
> >
> > Log:
> > Add checks for MNTK_UNMOUNTF at the beginning of three
> > functions, so that threads don't get stuck in them during
> > a forced dismount. nfs_sync/VFS_SYNC() needs this, since it is
> > called by dounmount() before VFS_UNMOUNT(). The nfscl_nget()
> > case makes sure that a thread doing an VOP_OPEN() or
> > VOP_ADVLOCK() call doesn't get blocked before attempting
> > the RPC. Attempting RPCs don't block, since they all
> > fail once a forced dismount is in progress.
> > The third one at the beginning of nfsrpc_close()
> > is done so threads don't get blocked while doing VOP_INACTIVE()
> > as the vnodes are cleared out.
> > With these three changes plus a change to the umount(1)
> > command so that it doesn't do "sync()" for the forced case
> > seem to make forced dismounts work for the experimental NFS
> > client.
> >
> > MFC after: 2 weeks
> >
> > Modified:
> > head/sys/fs/nfsclient/nfs_clrpcops.c
> > head/sys/fs/nfsclient/nfs_clstate.c
> > head/sys/fs/nfsclient/nfs_clvfsops.c
> >
> > Modified: head/sys/fs/nfsclient/nfs_clrpcops.c
> > ==============================================================================
> > --- head/sys/fs/nfsclient/nfs_clrpcops.c Sun Apr 17 22:31:36 2011
> > (r220760)
> > +++ head/sys/fs/nfsclient/nfs_clrpcops.c Sun Apr 17 23:04:03 2011
> > (r220761)
> > @@ -567,6 +567,11 @@ nfsrpc_close(vnode_t vp, int doclose, NF
> >
> > if (vnode_vtype(vp) != VREG)
> > return (0);
> > +
> > + /* For forced unmounts, just return. */
> > + if ((vp->v_mount->mnt_kern_flag & MNTK_UNMOUNTF) != 0)
> > + return (0);
> > +
> Is there anything that would prevent the MNTK_UNMOUNTF flag from being
> set immediately after the test returned success ?
>
> Usually, the tests for MNTK_UNMOUNTF are bugs. I coould see how
> terminating
> the RPCs would be useful for forced unmounts, but do not think that
> preventing entry into close would help.
>
I think you are correct, in that the one in nfsrpc_close() is harmless,
but not needed. (I suspect I added it to fix a crash and then didn't
realize that it's the one in nfscl_getcl() that matters.)
Basically, ncl_inactive() would call nfsrpc_close() which calls
nfscl_doclose() which calls nfscl_getcl(), so when nfscl_getcl()
returns failure, that should be sufficient.
As for the nfscl_getcl() one, I don't think it is currently quite
right either. The problem is a race with nfscl_umount(), which gets
called from nfs_unmount().
It goes like this:
- for a forced dismount, nfs_unmount():
- cancels all in-progress RPCs plus all new RPC attempts will fail
{newnfs_nmcancelreqs() does this }
- calls nfscl_umount()
--> nfscl_umount() gets the exclusive lock for open/lock state
modifications.
- igotlock = nfsv4_lock(&clp->nfsc_lock,...);
This will delay while any refcnt (shared lock) on nfsc_lock
is held.
- once this exclusive lock is acquired, it zaps the state
information (any RPC attempts simply fail, due to the
above)
- in nfscl_getcl(), it gets a refcnt (shared lock) on nfsc_lock, so
once that happens, the above will block until it is released.
--> My mistake was putting the test for MNTK_UNMOUNTF above that point
in nfscl_getcl(), so there is a race if it is set after the test,
but before acquiring the refcnt on nfsc_lock.
Thanks for pointing this out so I took another look. I'll either revert
the patch or add one that fixes the above. (ie. move the test in nfscl_getcl()
down and probably get rid of the one in nfsrpc_close(), since the test in
nfscl_getcl() should be sufficient.)
rick
More information about the svn-src-head
mailing list