Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/miscfs/genfs Restructure genfs_deadlock() and genfs_lock...
details: https://anonhg.NetBSD.org/src/rev/58c82e7b84c9
branches: trunk
changeset: 794333:58c82e7b84c9
user: hannken <hannken%NetBSD.org@localhost>
date: Wed Mar 12 09:38:51 2014 +0000
description:
Restructure genfs_deadlock() and genfs_lock() to always lock before
testing for dead node. Use ISSET() to test flags, add assertions.
Save the mount for fstrans_done() before genfs_unlock() unlocks the node.
diffstat:
sys/miscfs/genfs/genfs_vnops.c | 94 ++++++++++++++++++++++++-----------------
1 files changed, 55 insertions(+), 39 deletions(-)
diffs (155 lines):
diff -r 68a818a7427f -r 58c82e7b84c9 sys/miscfs/genfs/genfs_vnops.c
--- a/sys/miscfs/genfs/genfs_vnops.c Wed Mar 12 08:36:59 2014 +0000
+++ b/sys/miscfs/genfs/genfs_vnops.c Wed Mar 12 09:38:51 2014 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: genfs_vnops.c,v 1.190 2014/02/27 16:51:38 hannken Exp $ */
+/* $NetBSD: genfs_vnops.c,v 1.191 2014/03/12 09:38:51 hannken Exp $ */
/*-
* Copyright (c) 2008 The NetBSD Foundation, Inc.
@@ -57,7 +57,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: genfs_vnops.c,v 1.190 2014/02/27 16:51:38 hannken Exp $");
+__KERNEL_RCSID(0, "$NetBSD: genfs_vnops.c,v 1.191 2014/03/12 09:38:51 hannken Exp $");
#include <sys/param.h>
#include <sys/systm.h>
@@ -290,33 +290,42 @@
struct vnode *vp = ap->a_vp;
int flags = ap->a_flags;
krw_t op;
-
- if ((flags & LK_NOWAIT) != 0) {
- if (!mutex_tryenter(vp->v_interlock))
- return EBUSY;
- if ((vp->v_iflag & VI_XLOCK)) {
- mutex_exit(vp->v_interlock);
- return EBUSY;
- }
- }
+ int error;
- mutex_enter(vp->v_interlock);
- if ((vp->v_iflag & VI_XLOCK))
- vwait(vp, VI_XLOCK);
- mutex_exit(vp->v_interlock);
-
- if ((flags & LK_RETRY) == 0)
- return ENOENT;
-
- op = ((flags & LK_EXCLUSIVE) != 0 ? RW_WRITER : RW_READER);
- if ((flags & LK_NOWAIT) != 0) {
+ op = (ISSET(flags, LK_EXCLUSIVE) ? RW_WRITER : RW_READER);
+ if (ISSET(flags, LK_NOWAIT)) {
if (! rw_tryenter(&vp->v_lock, op))
return EBUSY;
- return 0;
+ if (mutex_tryenter(vp->v_interlock)) {
+ if (ISSET(vp->v_iflag, VI_XLOCK))
+ error = EBUSY;
+ else {
+ KASSERT(ISSET(vp->v_iflag, VI_CLEAN));
+ error = (ISSET(flags, LK_RETRY) ? 0 : ENOENT);
+ }
+ mutex_exit(vp->v_interlock);
+ } else
+ error = EBUSY;
+ if (error)
+ rw_exit(&vp->v_lock);
+ return error;
}
rw_enter(&vp->v_lock, op);
-
+ mutex_enter(vp->v_interlock);
+ if (ISSET(vp->v_iflag, VI_XLOCK)) {
+ rw_exit(&vp->v_lock);
+ vwait(vp, VI_XLOCK);
+ mutex_exit(vp->v_interlock);
+ rw_enter(&vp->v_lock, op);
+ mutex_enter(vp->v_interlock);
+ }
+ KASSERT(ISSET(vp->v_iflag, VI_CLEAN));
+ mutex_exit(vp->v_interlock);
+ if (! ISSET(flags, LK_RETRY)) {
+ rw_exit(&vp->v_lock);
+ return ENOENT;
+ }
return 0;
}
@@ -350,39 +359,45 @@
struct mount *mp = vp->v_mount;
int flags = ap->a_flags;
krw_t op;
+ int error;
- op = ((flags & LK_EXCLUSIVE) != 0 ? RW_WRITER : RW_READER);
- if ((flags & LK_NOWAIT) != 0) {
- if (!mutex_tryenter(vp->v_interlock))
- return EBUSY;
- if ((vp->v_iflag & (VI_XLOCK | VI_CLEAN)) != 0) {
- mutex_exit(vp->v_interlock);
- return EBUSY;
- }
- mutex_exit(vp->v_interlock);
+ op = (ISSET(flags, LK_EXCLUSIVE) ? RW_WRITER : RW_READER);
+ if (ISSET(flags, LK_NOWAIT)) {
if (fstrans_start_nowait(mp, FSTRANS_SHARED))
return EBUSY;
if (! rw_tryenter(&vp->v_lock, op)) {
fstrans_done(mp);
return EBUSY;
}
- return 0;
+ if (mutex_tryenter(vp->v_interlock)) {
+ if (ISSET(vp->v_iflag, VI_XLOCK))
+ error = EBUSY;
+ else if (ISSET(vp->v_iflag, VI_CLEAN))
+ error = ENOENT;
+ else
+ error = 0;
+ mutex_exit(vp->v_interlock);
+ } else
+ error = EBUSY;
+ if (error) {
+ rw_exit(&vp->v_lock);
+ fstrans_done(mp);
+ }
+ return error;
}
fstrans_start(mp, FSTRANS_SHARED);
rw_enter(&vp->v_lock, op);
-
mutex_enter(vp->v_interlock);
- if ((vp->v_iflag & (VI_XLOCK | VI_CLEAN)) != 0) {
+ if (ISSET(vp->v_iflag, VI_XLOCK) || ISSET(vp->v_iflag, VI_CLEAN)) {
rw_exit(&vp->v_lock);
fstrans_done(mp);
- if ((vp->v_iflag & VI_XLOCK))
- vwait(vp, VI_XLOCK);
+ vwait(vp, VI_XLOCK);
+ KASSERT(ISSET(vp->v_iflag, VI_CLEAN));
mutex_exit(vp->v_interlock);
return ENOENT;
}
mutex_exit(vp->v_interlock);
-
return 0;
}
@@ -396,9 +411,10 @@
struct vnode *a_vp;
} */ *ap = v;
struct vnode *vp = ap->a_vp;
+ struct mount *mp = vp->v_mount;
rw_exit(&vp->v_lock);
- fstrans_done(vp->v_mount);
+ fstrans_done(mp);
return 0;
}
Home |
Main Index |
Thread Index |
Old Index