Implement shared/exclusive locks.
Reviewed by: bmilekic, jake, jhb
This commit is contained in:
parent
7a00c25671
commit
6281b30a73
@ -26,7 +26,7 @@ MAN9= CONDSPLASSERT.9 DELAY.9 KASSERT.9 MD5.9 SPLASSERT.9 \
|
||||
panic.9 pfil.9 physio.9 posix4.9 psignal.9 \
|
||||
random.9 resettodr.9 rtalloc.9 rtentry.9 runqueue.9 \
|
||||
scheduler.9 sleep.9 sleepqueue.9 spl.9 store.9 style.9 suser.9 swi.9 \
|
||||
time.9 timeout.9 tvtohz.9 \
|
||||
sx.9 time.9 timeout.9 tvtohz.9 \
|
||||
uio.9 \
|
||||
vcount.9 vget.9 vnode.9 vput.9 vref.9 vrele.9 vslock.9
|
||||
|
||||
@ -131,6 +131,12 @@ MLINKS+=spl.9 splnet.9 spl.9 splsoftclock.9 spl.9 splsofttty.9
|
||||
MLINKS+=spl.9 splstatclock.9 spl.9 spltty.9 spl.9 splvm.9 spl.9 splx.9
|
||||
MLINKS+=store.9 subyte.9 store.9 suswintr.9 store.9 susword.9 store.9 suword.9
|
||||
MLINKS+=swi.9 sched_swi.9 swi.9 sinthand_add.9
|
||||
MLINKS+=sx.9 sx_init.9
|
||||
MLINKS+=sx.9 sx_destroy.9
|
||||
MLINKS+=sx.9 sx_slock.9
|
||||
MLINKS+=sx.9 sx_xlock.9
|
||||
MLINKS+=sx.9 sx_sunlock.9
|
||||
MLINKS+=sx.9 sx_xunlock.9
|
||||
MLINKS+=time.9 boottime.9 time.9 mono_time.9 time.9 runtime.9
|
||||
MLINKS+=timeout.9 untimeout.9
|
||||
MLINKS+=timeout.9 callout_handle_init.9 timeout.9 callout_init.9
|
||||
|
85
share/man/man9/sx.9
Normal file
85
share/man/man9/sx.9
Normal file
@ -0,0 +1,85 @@
|
||||
.\"
|
||||
.\" Copyright (C) 2001 Jason Evans <jasone@freebsd.org>. All rights reserved.
|
||||
.\"
|
||||
.\" Redistribution and use in source and binary forms, with or without
|
||||
.\" modification, are permitted provided that the following conditions
|
||||
.\" are met:
|
||||
.\" 1. Redistributions of source code must retain the above copyright
|
||||
.\" notice(s), this list of conditions and the following disclaimer as
|
||||
.\" the first lines of this file unmodified other than the possible
|
||||
.\" addition of one or more copyright notices.
|
||||
.\" 2. Redistributions in binary form must reproduce the above copyright
|
||||
.\" notice(s), this list of conditions and the following disclaimer in the
|
||||
.\" documentation and/or other materials provided with the distribution.
|
||||
.\"
|
||||
.\" THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDER(S) ``AS IS'' AND ANY
|
||||
.\" EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||
.\" WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||
.\" DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER(S) BE LIABLE FOR ANY
|
||||
.\" DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||
.\" (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||
.\" SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||||
.\" CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
||||
.\" LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
||||
.\" OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH
|
||||
.\" DAMAGE.
|
||||
.\"
|
||||
.\" $FreeBSD$
|
||||
.\"
|
||||
.Dd February 26, 2001
|
||||
.Dt SX 9
|
||||
.Os
|
||||
.Sh NAME
|
||||
.Nm sx ,
|
||||
.Nm sx_init ,
|
||||
.Nm sx_destroy ,
|
||||
.Nm sx_slock ,
|
||||
.Nm sx_xlock ,
|
||||
.Nm sx_sunlock ,
|
||||
.Nm sx_xunlock
|
||||
.Nd kernel shared/exclusive lock.
|
||||
.Sh SYNOPSIS
|
||||
.Fd #include <sys/sx.h>
|
||||
.Ft void
|
||||
.Fn sx_init "struct sx *sx" "const char *description"
|
||||
.Ft void
|
||||
.Fn sx_destroy "struct sx *sx"
|
||||
.Ft void
|
||||
.Fn sx_slock "struct sx *sx"
|
||||
.Ft void
|
||||
.Fn sx_xlock "struct sx *sx"
|
||||
.Ft void
|
||||
.Fn sx_sunlock "struct sx *sx"
|
||||
.Ft void
|
||||
.Fn sx_xunlock "struct sx *sx"
|
||||
.Sh DESCRIPTION
|
||||
Shared/exclusive locks are used to protect data that are read far more often
|
||||
than they are written.
|
||||
Mutexes are inherently more efficient than shared/exclusive locks, so
|
||||
shared/exclusive locks should be used prudently.
|
||||
.Pp
|
||||
Shared/exclusive locks are created with
|
||||
.Fn sx_init ,
|
||||
where
|
||||
.Fa sx
|
||||
is a pointer to space for a
|
||||
.Vt struct sx ,
|
||||
and
|
||||
.Fa description
|
||||
is a pointer to a null-terminated character string that describes the
|
||||
shared/exclusive lock.
|
||||
Shared/exclusive locks are destroyed with
|
||||
.Fn sx_destroy .
|
||||
Threads acquire and release a shared lock by calling
|
||||
.Fn sx_slock
|
||||
and
|
||||
.Fn sx_sunlock .
|
||||
Threads acquire and release an exclusive lock by calling
|
||||
.Fn sx_xlock
|
||||
and
|
||||
.Fn sx_xunlock .
|
||||
.Pp
|
||||
A thread may not own a shared lock and an exclusive lock simultaneously;
|
||||
attempting to do so will result in deadlock.
|
||||
.Sh SEE ALSO
|
||||
.Xr mutex 9
|
@ -681,6 +681,7 @@ kern/kern_shutdown.c standard
|
||||
kern/kern_sig.c standard
|
||||
kern/kern_subr.c standard
|
||||
kern/kern_switch.c standard
|
||||
kern/kern_sx.c standard
|
||||
kern/kern_synch.c standard
|
||||
kern/kern_syscalls.c standard
|
||||
kern/kern_sysctl.c standard
|
||||
|
160
sys/kern/kern_sx.c
Normal file
160
sys/kern/kern_sx.c
Normal file
@ -0,0 +1,160 @@
|
||||
/*
|
||||
* Copyright (C) 2001 Jason Evans <jasone@freebsd.org>. All rights reserved.
|
||||
*
|
||||
* Redistribution and use in source and binary forms, with or without
|
||||
* modification, are permitted provided that the following conditions
|
||||
* are met:
|
||||
* 1. Redistributions of source code must retain the above copyright
|
||||
* notice(s), this list of conditions and the following disclaimer as
|
||||
* the first lines of this file unmodified other than the possible
|
||||
* addition of one or more copyright notices.
|
||||
* 2. Redistributions in binary form must reproduce the above copyright
|
||||
* notice(s), this list of conditions and the following disclaimer in the
|
||||
* documentation and/or other materials provided with the distribution.
|
||||
*
|
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDER(S) ``AS IS'' AND ANY
|
||||
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER(S) BE LIABLE FOR ANY
|
||||
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||||
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
||||
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
||||
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH
|
||||
* DAMAGE.
|
||||
*
|
||||
* $FreeBSD$
|
||||
*/
|
||||
|
||||
/*
|
||||
* Shared/exclusive locks. This implementation assures deterministic lock
|
||||
* granting behavior, so that slocks and xlocks are interleaved.
|
||||
*
|
||||
* Priority propagation will not generally raise the priority of lock holders,
|
||||
* so should not be relied upon in combination with sx locks.
|
||||
*
|
||||
* The witness code can not detect lock cycles.
|
||||
*
|
||||
* slock --> xlock (deadlock)
|
||||
* slock --> slock (slock recursion, not fatal)
|
||||
* xlock --> xlock (deadlock)
|
||||
* xlock --> slock (deadlock)
|
||||
*/
|
||||
|
||||
#include <sys/param.h>
|
||||
#include <sys/systm.h>
|
||||
#include <sys/ktr.h>
|
||||
#include <sys/condvar.h>
|
||||
#include <sys/mutex.h>
|
||||
#include <sys/sx.h>
|
||||
|
||||
void
|
||||
sx_init(struct sx *sx, const char *description)
|
||||
{
|
||||
|
||||
mtx_init(&sx->sx_lock, description, MTX_DEF);
|
||||
sx->sx_cnt = 0;
|
||||
cv_init(&sx->sx_shrd_cv, description);
|
||||
sx->sx_shrd_wcnt = 0;
|
||||
cv_init(&sx->sx_excl_cv, description);
|
||||
sx->sx_excl_wcnt = 0;
|
||||
}
|
||||
|
||||
void
|
||||
sx_destroy(struct sx *sx)
|
||||
{
|
||||
|
||||
KASSERT((sx->sx_cnt == 0 && sx->sx_shrd_wcnt == 0 && sx->sx_excl_wcnt ==
|
||||
0), ("%s: holders or waiters\n", __FUNCTION__));
|
||||
|
||||
mtx_destroy(&sx->sx_lock);
|
||||
cv_destroy(&sx->sx_shrd_cv);
|
||||
cv_destroy(&sx->sx_excl_cv);
|
||||
}
|
||||
|
||||
void
|
||||
sx_slock(struct sx *sx)
|
||||
{
|
||||
|
||||
mtx_lock(&sx->sx_lock);
|
||||
|
||||
/*
|
||||
* Loop in case we lose the race for lock acquisition.
|
||||
*/
|
||||
while (sx->sx_cnt < 0) {
|
||||
sx->sx_shrd_wcnt++;
|
||||
cv_wait(&sx->sx_shrd_cv, &sx->sx_lock);
|
||||
sx->sx_shrd_wcnt--;
|
||||
}
|
||||
|
||||
/* Acquire a shared lock. */
|
||||
sx->sx_cnt++;
|
||||
|
||||
mtx_unlock(&sx->sx_lock);
|
||||
}
|
||||
|
||||
void
|
||||
sx_xlock(struct sx *sx)
|
||||
{
|
||||
|
||||
mtx_lock(&sx->sx_lock);
|
||||
|
||||
/* Loop in case we lose the race for lock acquisition. */
|
||||
while (sx->sx_cnt != 0) {
|
||||
sx->sx_excl_wcnt++;
|
||||
cv_wait(&sx->sx_excl_cv, &sx->sx_lock);
|
||||
sx->sx_excl_wcnt--;
|
||||
}
|
||||
|
||||
/* Acquire an exclusive lock. */
|
||||
sx->sx_cnt--;
|
||||
|
||||
mtx_unlock(&sx->sx_lock);
|
||||
}
|
||||
|
||||
void
|
||||
sx_sunlock(struct sx *sx)
|
||||
{
|
||||
|
||||
mtx_lock(&sx->sx_lock);
|
||||
KASSERT((sx->sx_cnt > 0), ("%s: lacking slock\n", __FUNCTION__));
|
||||
|
||||
/* Release. */
|
||||
sx->sx_cnt--;
|
||||
|
||||
/*
|
||||
* If we just released the last shared lock, wake any waiters up, giving
|
||||
* exclusive lockers precedence. In order to make sure that exclusive
|
||||
* lockers won't be blocked forever, don't wake shared lock waiters if
|
||||
* there are exclusive lock waiters.
|
||||
*/
|
||||
if (sx->sx_excl_wcnt > 0) {
|
||||
if (sx->sx_cnt == 0)
|
||||
cv_signal(&sx->sx_excl_cv);
|
||||
} else if (sx->sx_shrd_wcnt > 0)
|
||||
cv_broadcast(&sx->sx_shrd_cv);
|
||||
|
||||
mtx_unlock(&sx->sx_lock);
|
||||
}
|
||||
|
||||
void
|
||||
sx_xunlock(struct sx *sx)
|
||||
{
|
||||
|
||||
mtx_lock(&sx->sx_lock);
|
||||
KASSERT((sx->sx_cnt == -1), ("%s: lacking xlock\n", __FUNCTION__));
|
||||
|
||||
/* Release. */
|
||||
sx->sx_cnt++;
|
||||
|
||||
/*
|
||||
* Wake up waiters if there are any. Give precedence to slock waiters.
|
||||
*/
|
||||
if (sx->sx_shrd_wcnt > 0)
|
||||
cv_broadcast(&sx->sx_shrd_cv);
|
||||
else if (sx->sx_excl_wcnt > 0)
|
||||
cv_signal(&sx->sx_excl_cv);
|
||||
|
||||
mtx_unlock(&sx->sx_lock);
|
||||
}
|
82
sys/sys/sx.h
Normal file
82
sys/sys/sx.h
Normal file
@ -0,0 +1,82 @@
|
||||
/*
|
||||
* Copyright (C) 2001 Jason Evans <jasone@freebsd.org>. All rights reserved.
|
||||
*
|
||||
* Redistribution and use in source and binary forms, with or without
|
||||
* modification, are permitted provided that the following conditions
|
||||
* are met:
|
||||
* 1. Redistributions of source code must retain the above copyright
|
||||
* notice(s), this list of conditions and the following disclaimer as
|
||||
* the first lines of this file unmodified other than the possible
|
||||
* addition of one or more copyright notices.
|
||||
* 2. Redistributions in binary form must reproduce the above copyright
|
||||
* notice(s), this list of conditions and the following disclaimer in the
|
||||
* documentation and/or other materials provided with the distribution.
|
||||
*
|
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDER(S) ``AS IS'' AND ANY
|
||||
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER(S) BE LIABLE FOR ANY
|
||||
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||||
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
||||
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
||||
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH
|
||||
* DAMAGE.
|
||||
*
|
||||
* $FreeBSD$
|
||||
*/
|
||||
|
||||
#ifndef _SYS_SX_H_
|
||||
#define _SYS_SX_H_
|
||||
|
||||
#ifndef LOCORE
|
||||
#include <sys/mutex.h>
|
||||
#include <sys/condvar.h>
|
||||
|
||||
struct sx {
|
||||
struct mtx sx_lock; /* General protection lock and xlock. */
|
||||
int sx_cnt; /* -1: xlock, > 0: slock count. */
|
||||
struct cv sx_shrd_cv; /* slock waiters. */
|
||||
int sx_shrd_wcnt; /* Number of slock waiters. */
|
||||
struct cv sx_excl_cv; /* xlock waiters. */
|
||||
int sx_excl_wcnt; /* Number of xlock waiters. */
|
||||
};
|
||||
|
||||
#ifdef _KERNEL
|
||||
void sx_init(struct sx *sx, const char *description);
|
||||
void sx_destroy(struct sx *sx);
|
||||
void sx_slock(struct sx *sx);
|
||||
void sx_xlock(struct sx *sx);
|
||||
void sx_sunlock(struct sx *sx);
|
||||
void sx_xunlock(struct sx *sx);
|
||||
|
||||
#ifdef INVARIANTS
|
||||
/*
|
||||
* SX_ASSERT_SLOCKED() can only detect that at least *some* thread owns an
|
||||
* slock, but it cannot guarantee that *this* thread owns an slock.
|
||||
*/
|
||||
#define SX_ASSERT_SLOCKED(sx) do { \
|
||||
mtx_lock(&(sx)->sx_lock); \
|
||||
KASSERT(((sx)->sx_cnt > 0), ("%s: lacking slock\n", \
|
||||
__FUNCTION__)); \
|
||||
mtx_unlock(&(sx)->sx_lock); \
|
||||
} while (0)
|
||||
/*
|
||||
* SX_ASSERT_XLOCKED() can only detect that at least *some* thread owns an
|
||||
* xlock, but it cannot guarantee that *this* thread owns an xlock.
|
||||
*/
|
||||
#define SX_ASSERT_XLOCKED(sx) do { \
|
||||
mtx_lock(&(sx)->sx_lock); \
|
||||
KASSERT(((sx)->sx_cnt == -1), ("%s: lacking xlock\n", \
|
||||
__FUNCTION__)); \
|
||||
mtx_unlock(&(sx)->sx_lock); \
|
||||
} while (0)
|
||||
#else /* INVARIANTS */
|
||||
#define SX_ASSERT_SLOCKED(sx)
|
||||
#define SX_ASSERT_XLOCKER(sx)
|
||||
#endif /* INVARIANTS */
|
||||
|
||||
#endif /* _KERNEL */
|
||||
#endif /* !LOCORE */
|
||||
#endif /* _SYS_SX_H_ */
|
Loading…
x
Reference in New Issue
Block a user