From f08945a7d20b6469c2e4361ba0cc17bdc0df0bfe Mon Sep 17 00:00:00 2001 From: Attilio Rao Date: Tue, 26 Jun 2007 21:31:56 +0000 Subject: [PATCH] Introduce a new rwlocks initialization function: rw_init_flags. This is very similar to sx_init_flags: it initializes the rwlock using special flags passed as third argument (RW_DUPOK, RW_NOPROFILE, RW_NOWITNESS, RW_QUIET, RW_RECURSE). Among these, the most important new feature is probabilly that rwlocks can be acquired recursively now (for both shared and exclusive paths). Because of the recursion counter, the ABI is changed. Tested by: Timothy Redaelli Reviewed by: jhb Approved by: jeff (mentor) Approved by: re --- sys/kern/kern_rwlock.c | 94 ++++++++++++++++++++++++++++++++++++++---- sys/sys/_rwlock.h | 1 + sys/sys/rwlock.h | 20 +++++++-- 3 files changed, 103 insertions(+), 12 deletions(-) diff --git a/sys/kern/kern_rwlock.c b/sys/kern/kern_rwlock.c index 90a214264e70..539d67b4e2f8 100644 --- a/sys/kern/kern_rwlock.c +++ b/sys/kern/kern_rwlock.c @@ -48,6 +48,8 @@ __FBSDID("$FreeBSD$"); #include #include +CTASSERT((RW_RECURSE & LO_CLASSFLAGS) == RW_RECURSE); + #if defined(SMP) && !defined(NO_ADAPTIVE_RWLOCKS) #define ADAPTIVE_RWLOCKS #endif @@ -78,6 +80,17 @@ struct lock_class lock_class_rw = { ((rw)->rw_lock & RW_LOCK_READ ? NULL : \ (struct thread *)RW_OWNER((rw)->rw_lock)) +/* + * Returns if a write owner is recursed. Write ownership is not assured + * here and should be previously checked. + */ +#define rw_recursed(rw) ((rw)->rw_recurse != 0) + +/* + * Return true if curthread helds the lock. + */ +#define rw_wlocked(rw) (rw_wowner((rw)) == curthread) + /* * Return a pointer to the owning thread for this lock who should receive * any priority lent by threads that block on this lock. Currently this @@ -118,13 +131,27 @@ unlock_rw(struct lock_object *lock) } void -rw_init(struct rwlock *rw, const char *name) +rw_init_flags(struct rwlock *rw, const char *name, int opts) { + int flags; + + MPASS((opts & ~(RW_DUPOK | RW_NOPROFILE | RW_NOWITNESS | RW_QUIET | + RW_RECURSE)) == 0); + + flags = LO_UPGRADABLE | LO_RECURSABLE; + if (opts & RW_DUPOK) + flags |= LO_DUPOK; + if (opts & RW_NOPROFILE) + flags |= LO_NOPROFILE; + if (!(opts & RW_NOWITNESS)) + flags |= LO_WITNESS; + if (opts & RW_QUIET) + flags |= LO_QUIET; + flags |= opts & RW_RECURSE; rw->rw_lock = RW_UNLOCKED; - - lock_init(&rw->lock_object, &lock_class_rw, name, NULL, LO_WITNESS | - LO_RECURSABLE | LO_UPGRADABLE); + rw->rw_recurse = 0; + lock_init(&rw->lock_object, &lock_class_rw, name, NULL, flags); } void @@ -132,6 +159,7 @@ rw_destroy(struct rwlock *rw) { KASSERT(rw->rw_lock == RW_UNLOCKED, ("rw lock not unlocked")); + KASSERT(rw->rw_recurse == 0, ("rw lock still recursed")); rw->rw_lock = RW_DESTROYED; lock_destroy(&rw->lock_object); } @@ -164,7 +192,7 @@ _rw_wlock(struct rwlock *rw, const char *file, int line) WITNESS_CHECKORDER(&rw->lock_object, LOP_NEWORDER | LOP_EXCLUSIVE, file, line); __rw_wlock(rw, curthread, file, line); - LOCK_LOG_LOCK("WLOCK", &rw->lock_object, 0, 0, file, line); + LOCK_LOG_LOCK("WLOCK", &rw->lock_object, 0, rw->rw_recurse, file, line); WITNESS_LOCK(&rw->lock_object, LOP_EXCLUSIVE, file, line); curthread->td_locks++; } @@ -179,8 +207,10 @@ _rw_wunlock(struct rwlock *rw, const char *file, int line) _rw_assert(rw, RA_WLOCKED, file, line); curthread->td_locks--; WITNESS_UNLOCK(&rw->lock_object, LOP_EXCLUSIVE, file, line); - LOCK_LOG_LOCK("WUNLOCK", &rw->lock_object, 0, 0, file, line); - lock_profile_release_lock(&rw->lock_object); + LOCK_LOG_LOCK("WUNLOCK", &rw->lock_object, 0, rw->rw_recurse, file, + line); + if (!rw_recursed(rw)) + lock_profile_release_lock(&rw->lock_object); __rw_wunlock(rw, curthread, file, line); } @@ -466,6 +496,17 @@ _rw_wlock_hard(struct rwlock *rw, uintptr_t tid, const char *file, int line) #endif uintptr_t v; + if (rw_wlocked(rw)) { + KASSERT(rw->lock_object.lo_flags & RW_RECURSE, + ("%s: recursing but non-recursive rw %s @ %s:%d\n", + __func__, rw->lock_object.lo_name, file, line)); + rw->rw_recurse++; + atomic_set_ptr(&rw->rw_lock, RW_LOCK_RECURSED); + if (LOCK_LOG_TEST(&rw->lock_object, 0)) + CTR2(KTR_LOCK, "%s: %p recursing", __func__, rw); + return; + } + if (LOCK_LOG_TEST(&rw->lock_object, 0)) CTR5(KTR_LOCK, "%s: %s contested (lock=%p) at %s:%d", __func__, rw->lock_object.lo_name, (void *)rw->rw_lock, file, line); @@ -570,6 +611,14 @@ _rw_wunlock_hard(struct rwlock *rw, uintptr_t tid, const char *file, int line) uintptr_t v; int queue; + if (rw_wlocked(rw) && rw_recursed(rw)) { + if ((--rw->rw_recurse) == 0) + atomic_clear_ptr(&rw->rw_lock, RW_LOCK_RECURSED); + if (LOCK_LOG_TEST(&rw->lock_object, 0)) + CTR2(KTR_LOCK, "%s: %p unrecursing", __func__, rw); + return; + } + KASSERT(rw->rw_lock & (RW_LOCK_READ_WAITERS | RW_LOCK_WRITE_WAITERS), ("%s: neither of the waiter flags are set", __func__)); @@ -732,7 +781,11 @@ _rw_downgrade(struct rwlock *rw, const char *file, int line) KASSERT(rw->rw_lock != RW_DESTROYED, ("rw_downgrade() of destroyed rwlock @ %s:%d", file, line)); - _rw_assert(rw, RA_WLOCKED, file, line); + _rw_assert(rw, RA_WLOCKED | RA_NOTRECURSED, file, line); +#ifndef INVARIANTS + if (rw_recursed(rw)) + panic("downgrade of a recursed lock"); +#endif WITNESS_DOWNGRADE(&rw->lock_object, 0, file, line); @@ -809,7 +862,8 @@ _rw_assert(struct rwlock *rw, int what, const char *file, int line) return; switch (what) { case RA_LOCKED: - case RA_LOCKED | LA_NOTRECURSED: + case RA_LOCKED | RA_RECURSED: + case RA_LOCKED | RA_NOTRECURSED: case RA_RLOCKED: #ifdef WITNESS witness_assert(&rw->lock_object, what, file, line); @@ -825,12 +879,32 @@ _rw_assert(struct rwlock *rw, int what, const char *file, int line) panic("Lock %s not %slocked @ %s:%d\n", rw->lock_object.lo_name, (what == RA_RLOCKED) ? "read " : "", file, line); + + if (!(rw->rw_lock & RW_LOCK_READ)) { + if (rw_recursed(rw)) { + if (what & RA_NOTRECURSED) + panic("Lock %s recursed @ %s:%d\n", + rw->lock_object.lo_name, file, + line); + } else if (what & RA_RECURSED) + panic("Lock %s not recursed @ %s:%d\n", + rw->lock_object.lo_name, file, line); + } #endif break; case RA_WLOCKED: + case RA_WLOCKED | RA_RECURSED: + case RA_WLOCKED | RA_NOTRECURSED: if (rw_wowner(rw) != curthread) panic("Lock %s not exclusively locked @ %s:%d\n", rw->lock_object.lo_name, file, line); + if (rw_recursed(rw)) { + if (what & RA_NOTRECURSED) + panic("Lock %s recursed @ %s:%d\n", + rw->lock_object.lo_name, file, line); + } else if (what & RA_RECURSED) + panic("Lock %s not recursed @ %s:%d\n", + rw->lock_object.lo_name, file, line); break; case RA_UNLOCKED: #ifdef WITNESS @@ -874,6 +948,8 @@ db_show_rwlock(struct lock_object *lock) td = rw_wowner(rw); db_printf("WLOCK: %p (tid %d, pid %d, \"%s\")\n", td, td->td_tid, td->td_proc->p_pid, td->td_proc->p_comm); + if (rw_recursed(rw)) + db_printf(" recursed: %u\n", rw->rw_recurse); } db_printf(" waiters: "); switch (rw->rw_lock & (RW_LOCK_READ_WAITERS | RW_LOCK_WRITE_WAITERS)) { diff --git a/sys/sys/_rwlock.h b/sys/sys/_rwlock.h index c5adac0e4646..f65d3fa1e88f 100644 --- a/sys/sys/_rwlock.h +++ b/sys/sys/_rwlock.h @@ -38,6 +38,7 @@ struct rwlock { struct lock_object lock_object; volatile uintptr_t rw_lock; + volatile unsigned rw_recurse; }; #endif /* !_SYS__RWLOCK_H_ */ diff --git a/sys/sys/rwlock.h b/sys/sys/rwlock.h index 564d6fd863fd..2eb30bda18c3 100644 --- a/sys/sys/rwlock.h +++ b/sys/sys/rwlock.h @@ -65,11 +65,13 @@ #define RW_LOCK_READ 0x01 #define RW_LOCK_READ_WAITERS 0x02 #define RW_LOCK_WRITE_WAITERS 0x04 +#define RW_LOCK_RECURSED 0x08 #define RW_LOCK_FLAGMASK \ - (RW_LOCK_READ | RW_LOCK_READ_WAITERS | RW_LOCK_WRITE_WAITERS) + (RW_LOCK_READ | RW_LOCK_READ_WAITERS | RW_LOCK_WRITE_WAITERS | \ + RW_LOCK_RECURSED) #define RW_OWNER(x) ((x) & ~RW_LOCK_FLAGMASK) -#define RW_READERS_SHIFT 3 +#define RW_READERS_SHIFT 4 #define RW_READERS(x) (RW_OWNER((x)) >> RW_READERS_SHIFT) #define RW_READERS_LOCK(x) ((x) << RW_READERS_SHIFT | RW_LOCK_READ) #define RW_ONE_READER (1 << RW_READERS_SHIFT) @@ -124,7 +126,8 @@ * be used instead. */ -void rw_init(struct rwlock *rw, const char *name); +#define rw_init(rw, name) rw_init_flags((rw), (name), 0) +void rw_init_flags(struct rwlock *rw, const char *name, int opts); void rw_destroy(struct rwlock *rw); void rw_sysinit(void *arg); int rw_wowned(struct rwlock *rw); @@ -184,6 +187,15 @@ struct rw_args { SYSUNINIT(name##_rw_sysuninit, SI_SUB_LOCK, SI_ORDER_MIDDLE, \ rw_destroy, (rw)) +/* + * Options passed to rw_init_flags(). + */ +#define RW_DUPOK 0x01 +#define RW_NOPROFILE 0x02 +#define RW_NOWITNESS 0x04 +#define RW_QUIET 0x08 +#define RW_RECURSE 0x10 + /* * The INVARIANTS-enabled rw_assert() functionality. * @@ -196,6 +208,8 @@ struct rw_args { #define RA_RLOCKED LA_SLOCKED #define RA_WLOCKED LA_XLOCKED #define RA_UNLOCKED LA_UNLOCKED +#define RA_RECURSED LA_RECURSED +#define RA_NOTRECURSED LA_NOTRECURSED #endif #ifdef INVARIANTS