summaryrefslogtreecommitdiffstats
path: root/sys/fs
diff options
context:
space:
mode:
authorrmacklem <rmacklem@FreeBSD.org>2010-08-28 23:50:09 +0000
committerrmacklem <rmacklem@FreeBSD.org>2010-08-28 23:50:09 +0000
commitac834a438a850ee0cf4faee191d21404e0048009 (patch)
treebc028f5e213ce3881c6300a4d79624282fabccaf /sys/fs
parent3b7498e206fce85bc672ab8f554bcd6ebc0b398f (diff)
downloadFreeBSD-src-ac834a438a850ee0cf4faee191d21404e0048009.zip
FreeBSD-src-ac834a438a850ee0cf4faee191d21404e0048009.tar.gz
Add acquisition of a reference count on nfsv4root_lock to the
nfsd_recalldelegation() function, since this function is called by nfsd threads when they are handling NFSv2 or NFSv3 RPCs, where no reference count would have been acquired. MFC after: 2 weeks
Diffstat (limited to 'sys/fs')
-rw-r--r--sys/fs/nfsserver/nfs_nfsdstate.c13
1 files changed, 12 insertions, 1 deletions
diff --git a/sys/fs/nfsserver/nfs_nfsdstate.c b/sys/fs/nfsserver/nfs_nfsdstate.c
index c96907e..a531751 100644
--- a/sys/fs/nfsserver/nfs_nfsdstate.c
+++ b/sys/fs/nfsserver/nfs_nfsdstate.c
@@ -4563,6 +4563,14 @@ nfsd_recalldelegation(vnode_t vp, NFSPROC_T *p)
return;
/*
+ * First, get a reference on the nfsv4rootfs_lock so that an
+ * exclusive lock cannot be acquired by another thread.
+ */
+ NFSLOCKV4ROOTMUTEX();
+ nfsv4_getref(&nfsv4rootfs_lock, NULL, NFSV4ROOTLOCKMUTEXPTR);
+ NFSUNLOCKV4ROOTMUTEX();
+
+ /*
* Now, call nfsrv_checkremove() in a loop while it returns
* NFSERR_DELAY. Return upon any other error or when timed out.
*/
@@ -4576,11 +4584,14 @@ nfsd_recalldelegation(vnode_t vp, NFSPROC_T *p)
NFS_REMOVETIMEO &&
((u_int32_t)mytime.tv_sec - starttime) <
100000)
- return;
+ break;
/* Sleep for a short period of time */
(void) nfs_catnap(PZERO, 0, "nfsremove");
}
} while (error == NFSERR_DELAY);
+ NFSLOCKV4ROOTMUTEX();
+ nfsv4_relref(&nfsv4rootfs_lock);
+ NFSUNLOCKV4ROOTMUTEX();
}
APPLESTATIC void
OpenPOWER on IntegriCloud