summaryrefslogtreecommitdiffstats
path: root/sys/kern
diff options
context:
space:
mode:
authorpjd <pjd@FreeBSD.org>2007-09-21 10:16:56 +0000
committerpjd <pjd@FreeBSD.org>2007-09-21 10:16:56 +0000
commit763f7449dc1d3190bc67d34d3c15c7d9702810a1 (patch)
treea5b1aba6df8555ac8e6d2f48e313aecb9a042293 /sys/kern
parent427fb1f9be8e582a99ba5f06375c89296481d692 (diff)
downloadFreeBSD-src-763f7449dc1d3190bc67d34d3c15c7d9702810a1.zip
FreeBSD-src-763f7449dc1d3190bc67d34d3c15c7d9702810a1.tar.gz
Fix some locking cases where we ask for exclusively locked vnode, but we get
shared locked vnode in instead when vfs.lookup_shared is set to 1. Discussed with: kib, kris Tested by: kris Approved by: re (kensmith)
Diffstat (limited to 'sys/kern')
-rw-r--r--sys/kern/vfs_cache.c21
-rw-r--r--sys/kern/vfs_lookup.c8
2 files changed, 25 insertions, 4 deletions
diff --git a/sys/kern/vfs_cache.c b/sys/kern/vfs_cache.c
index 43428b3..7c1e428 100644
--- a/sys/kern/vfs_cache.c
+++ b/sys/kern/vfs_cache.c
@@ -314,6 +314,7 @@ cache_lookup(dvp, vpp, cnp)
struct componentname *cnp;
{
struct namecache *ncp;
+ struct thread *td;
u_int32_t hash;
int error, ltype;
@@ -321,6 +322,7 @@ cache_lookup(dvp, vpp, cnp)
cnp->cn_flags &= ~MAKEENTRY;
return (0);
}
+ td = cnp->cn_thread;
retry:
CACHE_LOCK();
numcalls++;
@@ -419,18 +421,29 @@ success:
if (dvp == *vpp) { /* lookup on "." */
VREF(*vpp);
CACHE_UNLOCK();
+ /*
+ * When we lookup "." we still can be asked to lock it
+ * differently...
+ */
+ ltype = cnp->cn_lkflags & (LK_SHARED | LK_EXCLUSIVE);
+ if (ltype == VOP_ISLOCKED(*vpp, td))
+ return (-1);
+ else if (ltype == LK_EXCLUSIVE)
+ vn_lock(*vpp, LK_UPGRADE | LK_RETRY, td);
return (-1);
}
ltype = 0; /* silence gcc warning */
if (cnp->cn_flags & ISDOTDOT) {
- ltype = VOP_ISLOCKED(dvp, cnp->cn_thread);
- VOP_UNLOCK(dvp, 0, cnp->cn_thread);
+ ltype = VOP_ISLOCKED(dvp, td);
+ VOP_UNLOCK(dvp, 0, td);
}
VI_LOCK(*vpp);
CACHE_UNLOCK();
- error = vget(*vpp, cnp->cn_lkflags | LK_INTERLOCK, cnp->cn_thread);
+ error = vget(*vpp, cnp->cn_lkflags | LK_INTERLOCK, td);
if (cnp->cn_flags & ISDOTDOT)
- vn_lock(dvp, ltype | LK_RETRY, cnp->cn_thread);
+ vn_lock(dvp, ltype | LK_RETRY, td);
+ if ((cnp->cn_flags & ISLASTCN) && (cnp->cn_lkflags & LK_EXCLUSIVE))
+ ASSERT_VOP_ELOCKED(*vpp, "cache_lookup");
if (error) {
*vpp = NULL;
goto retry;
diff --git a/sys/kern/vfs_lookup.c b/sys/kern/vfs_lookup.c
index a291a25..6349c36 100644
--- a/sys/kern/vfs_lookup.c
+++ b/sys/kern/vfs_lookup.c
@@ -772,6 +772,14 @@ nextname:
if ((cnp->cn_flags & LOCKLEAF) == 0)
VOP_UNLOCK(dp, 0, td);
success:
+ /*
+ * Because of lookup_shared we may have the vnode shared locked, but
+ * the caller may want it to be exclusively locked.
+ */
+ if ((cnp->cn_flags & (ISLASTCN | LOCKSHARED | LOCKLEAF)) ==
+ (ISLASTCN | LOCKLEAF) && VOP_ISLOCKED(dp, td) != LK_EXCLUSIVE) {
+ vn_lock(dp, LK_UPGRADE | LK_RETRY, td);
+ }
if (vfslocked && dvfslocked)
VFS_UNLOCK_GIANT(dvfslocked); /* Only need one */
if (vfslocked || dvfslocked)
OpenPOWER on IntegriCloud