xref: /freebsd/share/man/man9/rwlock.9 (revision fa9896e082a1046ff4fbc75fcba4d18d1f2efc19)
13182fc7bSGleb Smirnoff.\" Copyright (c) 2006 Gleb Smirnoff <glebius@FreeBSD.org>
23182fc7bSGleb Smirnoff.\" All rights reserved.
33182fc7bSGleb Smirnoff.\"
43182fc7bSGleb Smirnoff.\" Redistribution and use in source and binary forms, with or without
53182fc7bSGleb Smirnoff.\" modification, are permitted provided that the following conditions
63182fc7bSGleb Smirnoff.\" are met:
73182fc7bSGleb Smirnoff.\" 1. Redistributions of source code must retain the above copyright
83182fc7bSGleb Smirnoff.\"    notice, this list of conditions and the following disclaimer.
93182fc7bSGleb Smirnoff.\" 2. Redistributions in binary form must reproduce the above copyright
103182fc7bSGleb Smirnoff.\"    notice, this list of conditions and the following disclaimer in the
113182fc7bSGleb Smirnoff.\"    documentation and/or other materials provided with the distribution.
123182fc7bSGleb Smirnoff.\"
133182fc7bSGleb Smirnoff.\" THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
143182fc7bSGleb Smirnoff.\" ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
153182fc7bSGleb Smirnoff.\" IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
163182fc7bSGleb Smirnoff.\" ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
173182fc7bSGleb Smirnoff.\" FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
183182fc7bSGleb Smirnoff.\" DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
193182fc7bSGleb Smirnoff.\" OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
203182fc7bSGleb Smirnoff.\" HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
213182fc7bSGleb Smirnoff.\" LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
223182fc7bSGleb Smirnoff.\" OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
233182fc7bSGleb Smirnoff.\" SUCH DAMAGE.
243182fc7bSGleb Smirnoff.\"
25*755230ebSMark Johnston.Dd November 11, 2017
263182fc7bSGleb Smirnoff.Dt RWLOCK 9
273182fc7bSGleb Smirnoff.Os
283182fc7bSGleb Smirnoff.Sh NAME
290b79fedbSJohn Baldwin.Nm rwlock ,
303182fc7bSGleb Smirnoff.Nm rw_init ,
316442838dSAttilio Rao.Nm rw_init_flags ,
32b58aa2bdSJohn Baldwin.Nm rw_destroy ,
333182fc7bSGleb Smirnoff.Nm rw_rlock ,
343182fc7bSGleb Smirnoff.Nm rw_wlock ,
353182fc7bSGleb Smirnoff.Nm rw_runlock ,
363182fc7bSGleb Smirnoff.Nm rw_wunlock ,
37b12455f3SPawel Jakub Dawidek.Nm rw_unlock ,
38c709e029SAttilio Rao.Nm rw_try_rlock ,
397d3052bbSJohn Baldwin.Nm rw_try_upgrade ,
40c709e029SAttilio Rao.Nm rw_try_wlock ,
417d3052bbSJohn Baldwin.Nm rw_downgrade ,
42e7573e7aSJohn Baldwin.Nm rw_sleep ,
437d3052bbSJohn Baldwin.Nm rw_initialized ,
447d3052bbSJohn Baldwin.Nm rw_wowned ,
457d3052bbSJohn Baldwin.Nm rw_assert ,
46*755230ebSMark Johnston.Nm RW_SYSINIT ,
47*755230ebSMark Johnston.Nm RW_SYSINIT_FLAGS
480b79fedbSJohn Baldwin.Nd kernel reader/writer lock
493182fc7bSGleb Smirnoff.Sh SYNOPSIS
503182fc7bSGleb Smirnoff.In sys/param.h
513182fc7bSGleb Smirnoff.In sys/lock.h
523182fc7bSGleb Smirnoff.In sys/rwlock.h
533182fc7bSGleb Smirnoff.Ft void
540b79fedbSJohn Baldwin.Fn rw_init "struct rwlock *rw" "const char *name"
553182fc7bSGleb Smirnoff.Ft void
566442838dSAttilio Rao.Fn rw_init_flags "struct rwlock *rw" "const char *name" "int opts"
576442838dSAttilio Rao.Ft void
58b58aa2bdSJohn Baldwin.Fn rw_destroy "struct rwlock *rw"
59b58aa2bdSJohn Baldwin.Ft void
600b79fedbSJohn Baldwin.Fn rw_rlock "struct rwlock *rw"
613182fc7bSGleb Smirnoff.Ft void
620b79fedbSJohn Baldwin.Fn rw_wlock "struct rwlock *rw"
63c709e029SAttilio Rao.Ft int
64c709e029SAttilio Rao.Fn rw_try_rlock "struct rwlock *rw"
65c709e029SAttilio Rao.Ft int
66c709e029SAttilio Rao.Fn rw_try_wlock "struct rwlock *rw"
673182fc7bSGleb Smirnoff.Ft void
680b79fedbSJohn Baldwin.Fn rw_runlock "struct rwlock *rw"
693182fc7bSGleb Smirnoff.Ft void
700b79fedbSJohn Baldwin.Fn rw_wunlock "struct rwlock *rw"
71b12455f3SPawel Jakub Dawidek.Ft void
72b12455f3SPawel Jakub Dawidek.Fn rw_unlock "struct rwlock *rw"
733182fc7bSGleb Smirnoff.Ft int
7496cb5f99SJohn Baldwin.Fn rw_try_upgrade "struct rwlock *rw"
7596cb5f99SJohn Baldwin.Ft void
7696cb5f99SJohn Baldwin.Fn rw_downgrade "struct rwlock *rw"
77dd348217SRobert Watson.Ft int
78e7573e7aSJohn Baldwin.Fn rw_sleep "void *chan" "struct rwlock *rw" "int priority" "const char *wmesg" "int timo"
79e7573e7aSJohn Baldwin.Ft int
80d576deedSPawel Jakub Dawidek.Fn rw_initialized "const struct rwlock *rw"
817d3052bbSJohn Baldwin.Ft int
82d576deedSPawel Jakub Dawidek.Fn rw_wowned "const struct rwlock *rw"
833182fc7bSGleb Smirnoff.Pp
843182fc7bSGleb Smirnoff.Cd "options INVARIANTS"
853182fc7bSGleb Smirnoff.Cd "options INVARIANT_SUPPORT"
863182fc7bSGleb Smirnoff.Ft void
87d576deedSPawel Jakub Dawidek.Fn rw_assert "const struct rwlock *rw" "int what"
883182fc7bSGleb Smirnoff.In sys/kernel.h
890b79fedbSJohn Baldwin.Fn RW_SYSINIT "name" "struct rwlock *rw" "const char *desc"
90*755230ebSMark Johnston.Fn RW_SYSINIT_FLAGS "name" "struct rwlock *rw" "const char *desc" "int flags"
913182fc7bSGleb Smirnoff.Sh DESCRIPTION
920b79fedbSJohn BaldwinReader/writer locks allow shared access to protected data by multiple threads,
930b79fedbSJohn Baldwinor exclusive access by a single thread.
940b79fedbSJohn BaldwinThe threads with shared access are known as
953182fc7bSGleb Smirnoff.Em readers
960b79fedbSJohn Baldwinsince they only read the protected data.
970b79fedbSJohn BaldwinA thread with exclusive access is known as a
983182fc7bSGleb Smirnoff.Em writer
993182fc7bSGleb Smirnoffsince it can modify protected data.
1003182fc7bSGleb Smirnoff.Pp
101f1acac21SFlorent ThoumieAlthough reader/writer locks look very similar to
1023182fc7bSGleb Smirnoff.Xr sx 9
1033182fc7bSGleb Smirnofflocks, their usage pattern is different.
1040b79fedbSJohn BaldwinReader/writer locks can be treated as mutexes (see
1053182fc7bSGleb Smirnoff.Xr mutex 9 )
1063182fc7bSGleb Smirnoffwith shared/exclusive semantics.
1073182fc7bSGleb SmirnoffUnlike
1083182fc7bSGleb Smirnoff.Xr sx 9 ,
1093182fc7bSGleb Smirnoffan
1103182fc7bSGleb Smirnoff.Nm
1110b79fedbSJohn Baldwincan be locked while holding a non-spin mutex, and an
1123182fc7bSGleb Smirnoff.Nm
1133182fc7bSGleb Smirnoffcannot be held while sleeping.
1143182fc7bSGleb SmirnoffThe
1153182fc7bSGleb Smirnoff.Nm
1163182fc7bSGleb Smirnofflocks have priority propagation like mutexes, but priority
1173d2a7480SJohn Baldwincan be propagated only to writers.
1183d2a7480SJohn BaldwinThis limitation comes from the fact that readers
1193182fc7bSGleb Smirnoffare anonymous.
1203d2a7480SJohn BaldwinAnother important property is that readers can always recurse,
1216442838dSAttilio Raoand exclusive locks can be made recursive selectively.
1223182fc7bSGleb Smirnoff.Ss Macros and Functions
1233182fc7bSGleb Smirnoff.Bl -tag -width indent
1240b79fedbSJohn Baldwin.It Fn rw_init "struct rwlock *rw" "const char *name"
1253182fc7bSGleb SmirnoffInitialize structure located at
1260b79fedbSJohn Baldwin.Fa rw
1270b79fedbSJohn Baldwinas reader/writer lock, described by name
1283182fc7bSGleb Smirnoff.Fa name .
1293182fc7bSGleb SmirnoffThe description is used solely for debugging purposes.
1300b79fedbSJohn BaldwinThis function must be called before any other operations
1310b79fedbSJohn Baldwinon the lock.
1326442838dSAttilio Rao.It Fn rw_init_flags "struct rwlock *rw" "const char *name" "int opts"
1334faa4b08SChristian BruefferInitialize the rw lock just like the
1346442838dSAttilio Rao.Fn rw_init
1354faa4b08SChristian Bruefferfunction, but specifying a set of optional flags to alter the
1366442838dSAttilio Raobehaviour of
1376442838dSAttilio Rao.Fa rw ,
1386442838dSAttilio Raothrough the
1396442838dSAttilio Rao.Fa opts
1404faa4b08SChristian Bruefferargument.
1414faa4b08SChristian BruefferIt contains one or more of the following flags:
1426442838dSAttilio Rao.Bl -tag -width ".Dv RW_NOPROFILE"
1436442838dSAttilio Rao.It Dv RW_DUPOK
1446442838dSAttilio RaoWitness should not log messages about duplicate locks being acquired.
1456442838dSAttilio Rao.It Dv RW_NOPROFILE
1466442838dSAttilio RaoDo not profile this lock.
1476442838dSAttilio Rao.It Dv RW_NOWITNESS
1486442838dSAttilio RaoInstruct
1496442838dSAttilio Rao.Xr witness 4
1506442838dSAttilio Raoto ignore this lock.
1516442838dSAttilio Rao.It Dv RW_QUIET
1526442838dSAttilio RaoDo not log any operations for this lock via
1536442838dSAttilio Rao.Xr ktr 4 .
1546442838dSAttilio Rao.It Dv RW_RECURSE
1556442838dSAttilio RaoAllow threads to recursively acquire exclusive locks for
1566442838dSAttilio Rao.Fa rw .
157fd07ddcfSDmitry Chagin.It Dv RW_NEW
158fd07ddcfSDmitry ChaginIf the kernel has been compiled with
159fd07ddcfSDmitry Chagin.Cd "option INVARIANTS" ,
160fd07ddcfSDmitry Chagin.Fn rw_init_flags
161fd07ddcfSDmitry Chaginwill assert that the
162fd07ddcfSDmitry Chagin.Fa rw
163fd07ddcfSDmitry Chaginhas not been initialized multiple times without intervening calls to
164fd07ddcfSDmitry Chagin.Fn rw_destroy
165fd07ddcfSDmitry Chaginunless this option is specified.
16639887c5cSPawel Jakub Dawidek.El
1670b79fedbSJohn Baldwin.It Fn rw_rlock "struct rwlock *rw"
1680b79fedbSJohn BaldwinLock
1690b79fedbSJohn Baldwin.Fa rw
1700b79fedbSJohn Baldwinas a reader.
1713182fc7bSGleb SmirnoffIf any thread holds this lock exclusively, the current thread blocks,
1720b79fedbSJohn Baldwinand its priority is propagated to the exclusive holder.
1733182fc7bSGleb SmirnoffThe
1743182fc7bSGleb Smirnoff.Fn rw_rlock
1753182fc7bSGleb Smirnofffunction can be called when the thread has already acquired reader
1763182fc7bSGleb Smirnoffaccess on
1770b79fedbSJohn Baldwin.Fa rw .
1783182fc7bSGleb SmirnoffThis is called
1793182fc7bSGleb Smirnoff.Dq "recursing on a lock" .
1800b79fedbSJohn Baldwin.It Fn rw_wlock "struct rwlock *rw"
1810b79fedbSJohn BaldwinLock
1820b79fedbSJohn Baldwin.Fa rw
1830b79fedbSJohn Baldwinas a writer.
1843182fc7bSGleb SmirnoffIf there are any shared owners of the lock, the current thread blocks.
1853182fc7bSGleb SmirnoffThe
1863182fc7bSGleb Smirnoff.Fn rw_wlock
1876442838dSAttilio Raofunction can be called recursively only if
1886442838dSAttilio Rao.Fa rw
1896442838dSAttilio Raohas been initialized with the
1906442838dSAttilio Rao.Dv RW_RECURSE
1916442838dSAttilio Raooption enabled.
192c709e029SAttilio Rao.It Fn rw_try_rlock "struct rwlock *rw"
193c709e029SAttilio RaoTry to lock
194c709e029SAttilio Rao.Fa rw
195c709e029SAttilio Raoas a reader.
196c709e029SAttilio RaoThis function will return true if the operation succeeds, otherwise 0
197c709e029SAttilio Raowill be returned.
198c709e029SAttilio Rao.It Fn rw_try_wlock "struct rwlock *rw"
199c709e029SAttilio RaoTry to lock
200c709e029SAttilio Rao.Fa rw
201c709e029SAttilio Raoas a writer.
202c709e029SAttilio RaoThis function will return true if the operation succeeds, otherwise 0
203c709e029SAttilio Raowill be returned.
2040b79fedbSJohn Baldwin.It Fn rw_runlock "struct rwlock *rw"
2050b79fedbSJohn BaldwinThis function releases a shared lock previously acquired by
2063182fc7bSGleb Smirnoff.Fn rw_rlock .
2070b79fedbSJohn Baldwin.It Fn rw_wunlock "struct rwlock *rw"
2080b79fedbSJohn BaldwinThis function releases an exclusive lock previously acquired by
2093182fc7bSGleb Smirnoff.Fn rw_wlock .
210b12455f3SPawel Jakub Dawidek.It Fn rw_unlock "struct rwlock *rw"
211b12455f3SPawel Jakub DawidekThis function releases a shared lock previously acquired by
212b12455f3SPawel Jakub Dawidek.Fn rw_rlock
213b12455f3SPawel Jakub Dawidekor an exclusive lock previously acquired by
214b12455f3SPawel Jakub Dawidek.Fn rw_wlock .
21596cb5f99SJohn Baldwin.It Fn rw_try_upgrade "struct rwlock *rw"
21696cb5f99SJohn BaldwinAttempt to upgrade a single shared lock to an exclusive lock.
21796cb5f99SJohn BaldwinThe current thread must hold a shared lock of
21896cb5f99SJohn Baldwin.Fa rw .
21996cb5f99SJohn BaldwinThis will only succeed if the current thread holds the only shared lock on
22096cb5f99SJohn Baldwin.Fa rw ,
22196cb5f99SJohn Baldwinand it only holds a single shared lock.
22296cb5f99SJohn BaldwinIf the attempt succeeds
22396cb5f99SJohn Baldwin.Fn rw_try_upgrade
22496cb5f99SJohn Baldwinwill return a non-zero value,
22596cb5f99SJohn Baldwinand the current thread will hold an exclusive lock.
22696cb5f99SJohn BaldwinIf the attempt fails
22796cb5f99SJohn Baldwin.Fn rw_try_upgrade
22896cb5f99SJohn Baldwinwill return zero,
22996cb5f99SJohn Baldwinand the current thread will still hold a shared lock.
23096cb5f99SJohn Baldwin.It Fn rw_downgrade "struct rwlock *rw"
23196cb5f99SJohn BaldwinConvert an exclusive lock into a single shared lock.
23296cb5f99SJohn BaldwinThe current thread must hold an exclusive lock of
23396cb5f99SJohn Baldwin.Fa rw .
234e7573e7aSJohn Baldwin.It Fn rw_sleep "void *chan" "struct rwlock *rw" "int priority" "const char *wmesg" "int timo"
235e7573e7aSJohn BaldwinAtomically release
236e7573e7aSJohn Baldwin.Fa rw
237e7573e7aSJohn Baldwinwhile waiting for an event.
238e7573e7aSJohn BaldwinFor more details on the parameters to this function,
239e7573e7aSJohn Baldwinsee
240e7573e7aSJohn Baldwin.Xr sleep 9 .
241d576deedSPawel Jakub Dawidek.It Fn rw_initialized "const struct rwlock *rw"
2420b79fedbSJohn BaldwinThis function returns non-zero if
2430b79fedbSJohn Baldwin.Fa rw
2443182fc7bSGleb Smirnoffhas been initialized, and zero otherwise.
2450b79fedbSJohn Baldwin.It Fn rw_destroy "struct rwlock *rw"
2463182fc7bSGleb SmirnoffThis functions destroys a lock previously initialized with
2473182fc7bSGleb Smirnoff.Fn rw_init .
2483182fc7bSGleb SmirnoffThe
2490b79fedbSJohn Baldwin.Fa rw
2500b79fedbSJohn Baldwinlock must be unlocked.
251d576deedSPawel Jakub Dawidek.It Fn rw_wowned "const struct rwlock *rw"
252dd348217SRobert WatsonThis function returns a non-zero value if the current thread owns an
253dd348217SRobert Watsonexclusive lock on
254dd348217SRobert Watson.Fa rw .
255d576deedSPawel Jakub Dawidek.It Fn rw_assert "const struct rwlock *rw" "int what"
2563182fc7bSGleb SmirnoffThis function allows assertions specified in
2573182fc7bSGleb Smirnoff.Fa what
2583182fc7bSGleb Smirnoffto be made about
2590b79fedbSJohn Baldwin.Fa rw .
2603182fc7bSGleb SmirnoffIf the assertions are not true and the kernel is compiled
2613182fc7bSGleb Smirnoffwith
2623182fc7bSGleb Smirnoff.Cd "options INVARIANTS"
2633182fc7bSGleb Smirnoffand
2643182fc7bSGleb Smirnoff.Cd "options INVARIANT_SUPPORT" ,
2653182fc7bSGleb Smirnoffthe kernel will panic.
266a2feb19bSJohn BaldwinCurrently the following base assertions are supported:
2673182fc7bSGleb Smirnoff.Bl -tag -width ".Dv RA_UNLOCKED"
2683182fc7bSGleb Smirnoff.It Dv RA_LOCKED
2690b79fedbSJohn BaldwinAssert that current thread holds either a shared or exclusive lock
2700b79fedbSJohn Baldwinof
2710b79fedbSJohn Baldwin.Fa rw .
2723182fc7bSGleb Smirnoff.It Dv RA_RLOCKED
2730b79fedbSJohn BaldwinAssert that current thread holds a shared lock of
2740b79fedbSJohn Baldwin.Fa rw .
2753182fc7bSGleb Smirnoff.It Dv RA_WLOCKED
2760b79fedbSJohn BaldwinAssert that current thread holds an exclusive lock of
2770b79fedbSJohn Baldwin.Fa rw .
2783182fc7bSGleb Smirnoff.It Dv RA_UNLOCKED
2790b79fedbSJohn BaldwinAssert that current thread holds neither a shared nor exclusive lock of
2800b79fedbSJohn Baldwin.Fa rw .
2813182fc7bSGleb Smirnoff.El
282a2feb19bSJohn Baldwin.Pp
283a2feb19bSJohn BaldwinIn addition, one of the following optional flags may be specified with
284a2feb19bSJohn Baldwin.Dv RA_LOCKED ,
285a2feb19bSJohn Baldwin.Dv RA_RLOCKED ,
286a2feb19bSJohn Baldwinor
287a2feb19bSJohn Baldwin.Dv RA_WLOCKED :
288a2feb19bSJohn Baldwin.Bl -tag -width ".Dv RA_NOTRECURSED"
289a2feb19bSJohn Baldwin.It Dv RA_RECURSED
290a2feb19bSJohn BaldwinAssert that the current thread holds a recursive lock of
291a2feb19bSJohn Baldwin.Fa rw .
292a2feb19bSJohn Baldwin.It Dv RA_NOTRECURSED
293a2feb19bSJohn BaldwinAssert that the current thread does not hold a recursive lock of
294a2feb19bSJohn Baldwin.Fa rw .
295a2feb19bSJohn Baldwin.El
2963182fc7bSGleb Smirnoff.El
2973182fc7bSGleb Smirnoff.Sh SEE ALSO
298a280550aSJulian Elischer.Xr locking 9 ,
2993182fc7bSGleb Smirnoff.Xr mutex 9 ,
3003182fc7bSGleb Smirnoff.Xr panic 9 ,
3013182fc7bSGleb Smirnoff.Xr sema 9 ,
3023182fc7bSGleb Smirnoff.Xr sx 9
3033182fc7bSGleb Smirnoff.Sh HISTORY
3043182fc7bSGleb SmirnoffThese
3053182fc7bSGleb Smirnofffunctions appeared in
3063182fc7bSGleb Smirnoff.Fx 7.0 .
3073182fc7bSGleb Smirnoff.Sh AUTHORS
3083182fc7bSGleb Smirnoff.An -nosplit
3093182fc7bSGleb SmirnoffThe
3103182fc7bSGleb Smirnoff.Nm
3113182fc7bSGleb Smirnofffacility was written by
3123182fc7bSGleb Smirnoff.An "John Baldwin" .
3133182fc7bSGleb SmirnoffThis manual page was written by
3143182fc7bSGleb Smirnoff.An "Gleb Smirnoff" .
3150b79fedbSJohn Baldwin.Sh BUGS
316a23b48feSBryan DreweryA kernel without
3170b79fedbSJohn Baldwin.Dv WITNESS
318a23b48feSBryan Drewerycannot assert whether the current thread does or does not hold a read lock.
3190b79fedbSJohn Baldwin.Dv RA_LOCKED
3200b79fedbSJohn Baldwinand
3210b79fedbSJohn Baldwin.Dv RA_RLOCKED
322a23b48feSBryan Drewerycan only assert that
323a23b48feSBryan Drewery.Em any
324a23b48feSBryan Drewerythread holds a read lock.
325a23b48feSBryan DreweryThey cannot ensure that the current thread holds a read lock.
326a23b48feSBryan DreweryFurther,
327a23b48feSBryan Drewery.Dv RA_UNLOCKED
328a23b48feSBryan Drewerycan only assert that the current thread does not hold a write lock.
3290b79fedbSJohn Baldwin.Pp
3300b79fedbSJohn BaldwinReader/writer is a bit of an awkward name.
3310b79fedbSJohn BaldwinAn
3320b79fedbSJohn Baldwin.Nm
3330b79fedbSJohn Baldwincan also be called a
3340b79fedbSJohn Baldwin.Dq Robert Watson
3350b79fedbSJohn Baldwinlock if desired.
336