Merge branch 'master' of https://github.com/bscheinman/ck
commit
b36e35ec76
@ -0,0 +1,61 @@
|
||||
.\"
|
||||
.\" Copyright 2013 Brendon Scheinman.
|
||||
.\" 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, this list of conditions and the following disclaimer.
|
||||
.\" 2. Redistributions in binary form must reproduce the above copyright
|
||||
.\" notice, 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 REGENTS AND CONTRIBUTORS ``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 REGENTS OR CONTRIBUTORS 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.
|
||||
.\"
|
||||
.\"
|
||||
.Dd February 24, 2013.
|
||||
.Dt CK_RWCOHORT_INIT 3
|
||||
.Sh NAME
|
||||
.Nm CK_RWCOHORT_INIT
|
||||
.Nd initialize instance of a cohort-based reader-writer lock type
|
||||
.Sh LIBRARY
|
||||
Concurrency Kit (libck, \-lck)
|
||||
.Sh SYNOPSIS
|
||||
.In ck_rwcohort.h
|
||||
.Fn CK_RWCOHORT_NEUTRAL_INIT "COHORT_NAME cohort_name" "LOCK *lock"
|
||||
.Fn CK_RWCOHORT_RP_INIT "COHORT_NAME cohort_name" "LOCK *lock" "unsigned int wait_limit"
|
||||
.Fn CK_RWCOHORT_WP_INIT "COHORT_NAME cohort_name" "LOCK *lock" "unsigned int wait_limit"
|
||||
.Sh DESCRIPTION
|
||||
This macro initializes the lock instance pointed to by the
|
||||
.Fa lock
|
||||
argument. Until a lock instance is initialized using the CK_RWCOHORT_INIT macro, any operations
|
||||
involving it will have undefined behavior. Note that the
|
||||
.Fa wait_limit
|
||||
argument should only be used with reader-preference or writer-preference locks. For neutral
|
||||
locks, this argument should be excluded.
|
||||
If you are unsure of a value to use for the
|
||||
.Fa wait_limit
|
||||
argument, you should use CK_RWCOHORT_STRATEGY_DEFAULT_LOCAL_WAIT_LIMIT.
|
||||
.Sh SEE ALSO
|
||||
.Xr ck_rwcohort 3 ,
|
||||
.Xr CK_RWCOHORT_PROTOTYPE 3 ,
|
||||
.Xr CK_RWCOHORT_TRYLOCK_PROTOTYPE 3 ,
|
||||
.Xr CK_RWCOHORT_INSTANCE 3 ,
|
||||
.Xr CK_RWCOHORT_INITIALIZER 3 ,
|
||||
.Xr CK_RWCOHORT_LOCK 3 ,
|
||||
.Xr CK_RWCOHORT_UNLOCK 3 ,
|
||||
.Xr CK_RWCOHORT_LOCKED 3 ,
|
||||
.Xr CK_RWCOHORT_TRYLOCK 3 ,
|
||||
.Pp
|
||||
Additional information available at http://concurrencykit.org/
|
@ -0,0 +1,64 @@
|
||||
.\"
|
||||
.\" Copyright 2013 Brendon Scheinman.
|
||||
.\" 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, this list of conditions and the following disclaimer.
|
||||
.\" 2. Redistributions in binary form must reproduce the above copyright
|
||||
.\" notice, 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 REGENTS AND CONTRIBUTORS ``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 REGENTS OR CONTRIBUTORS 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.
|
||||
.\"
|
||||
.\"
|
||||
.Dd February 24, 2013.
|
||||
.Dt CK_RWCOHORT_INSTANCE 3
|
||||
.Sh NAME
|
||||
.Nm CK_RWCOHORT_INSTANCE
|
||||
.Nd declare an instance of a cohort-based reader-writer lock type
|
||||
.Sh LIBRARY
|
||||
Concurrency Kit (libck, \-lck)
|
||||
.Sh SYNOPSIS
|
||||
.In ck_cohort.h
|
||||
.Fn CK_RWCOHORT_NEUTRAL_INSTANCE "COHORT_NAME cohort_name"
|
||||
.Fn CK_RWCOHORT_RP_INSTANCE "COHORT_NAME cohort_name"
|
||||
.Fn CK_RWCOHORT_WP_INSTANCE "COHORT_NAME cohort_name"
|
||||
.Sh DESCRIPTION
|
||||
The user must use this macro to declare instances of lock types that they have
|
||||
defined using the
|
||||
.Xr CK_RWCOHORT_PROTOTYPE 3
|
||||
macro. The cohort_name must be the same as the one used in the prototype macro.
|
||||
For instance, if CK_RWCOHORT_PROTOTYPE was called with the name "foo", the
|
||||
CK_RWCOHORT_INSTANCE macro should be called as
|
||||
.br
|
||||
CK_RWCOHORT_INSTANCE(foo) cohort;
|
||||
.Pp
|
||||
This macro should also be used when allocating memory for cohorts. For instance,
|
||||
to allocate a block of 4 cohorts:
|
||||
.br
|
||||
CK_RWCOHORT_WP_INSTANCE(foo) *cohorts = malloc(4 * sizeof(CK_RWCOHORT_WP_INSTANCE(foo)));
|
||||
.Sh SEE ALSO
|
||||
.Xr ck_rwcohort 3 ,
|
||||
.Xr CK_RWCOHORT_PROTOTYPE 3 ,
|
||||
.Xr CK_RWCOHORT_TRYLOCK_PROTOTYPE 3 ,
|
||||
.Xr CK_RWCOHORT_INSTANCE 3 ,
|
||||
.Xr CK_RWCOHORT_INITIALIZER 3 ,
|
||||
.Xr CK_RWCOHORT_LOCK 3 ,
|
||||
.Xr CK_RWCOHORT_UNLOCK 3 ,
|
||||
.Xr CK_RWCOHORT_LOCKED 3 ,
|
||||
.Xr CK_RWCOHORT_TRYLOCK 3 ,
|
||||
.Pp
|
||||
Additional information available at http://concurrencykit.org/
|
@ -0,0 +1,65 @@
|
||||
.\"
|
||||
.\" Copyright 2013 Brendon Scheinman.
|
||||
.\" 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, this list of conditions and the following disclaimer.
|
||||
.\" 2. Redistributions in binary form must reproduce the above copyright
|
||||
.\" notice, 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 REGENTS AND CONTRIBUTORS ``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 REGENTS OR CONTRIBUTORS 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.
|
||||
.\"
|
||||
.\"
|
||||
.Dd February 24, 2013.
|
||||
.Dt CK_RWCOHORT_PROTOTYPE 3
|
||||
.Sh NAME
|
||||
.Nm CK_RWCOHORT_PROTOTYPE
|
||||
.Nd define reader-writer cohort-based lock using the specified cohort type
|
||||
.Sh LIBRARY
|
||||
Concurrency Kit (libck, \-lck)
|
||||
.Sh SYNOPSIS
|
||||
.In ck_rwcohort.h
|
||||
.Fn CK_RWCOHORT_NEUTRAL_PROTOTYPE "COHORT_NAME cohort_name"
|
||||
.Fn CK_RWCOHORT_RP_PROTOTYPE "COHORT_NAME cohort_name"
|
||||
.Fn CK_RWCOHORT_WP_PROTOTYPE "COHORT_NAME cohort_name"
|
||||
.Sh DESCRIPTION
|
||||
The ck_rwcohort.h header file does not define any cohort types. Instead, the user must use
|
||||
the CK_RWCOHORT_PROTOTYPE macro to define any types they want to use.
|
||||
This macro takes a single argument which corresponds to the type of the cohort lock that
|
||||
the reader-writer lock should use. A cohort type must have already been defined with that name
|
||||
using the
|
||||
.Xr CK_COHORT_PROTOTYPE 3
|
||||
or
|
||||
.Xr CK_COHORT_TRYLOCK_PROTOTYPE 3
|
||||
macros.
|
||||
.Pp
|
||||
Instances of the defined lock type can be declared as:
|
||||
.br
|
||||
CK_RWCOHORT_INSTANCE(cohort_name) lock;
|
||||
.Sh SEE ALSO
|
||||
.Xr ck_rwcohort 3 ,
|
||||
.Xr CK_COHORT_PROTOTYPE 3 ,
|
||||
.Xr CK_COHORT_TRYLOCK_PROTOTYPE 3 ,
|
||||
.Xr CK_RWCOHORT_INSTANCE 3 ,
|
||||
.Xr CK_RWCOHORT_INITIALIZER 3 ,
|
||||
.Xr CK_RWCOHORT_INIT 3 ,
|
||||
.Xr CK_RWCOHORT_READ_LOCK 3 ,
|
||||
.Xr CK_RWCOHORT_READ_UNLOCK 3 ,
|
||||
.Xr CK_RWCOHORT_WRITE_LOCK 3 ,
|
||||
.Xr CK_RWCOHORT_WRITE_UNLOCK 3 ,
|
||||
.Pp
|
||||
Additional information available at http://concurrencykit.org/
|
@ -0,0 +1,66 @@
|
||||
.\"
|
||||
.\" Copyright 2013 Brendon Scheinman.
|
||||
.\" 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, this list of conditions and the following disclaimer.
|
||||
.\" 2. Redistributions in binary form must reproduce the above copyright
|
||||
.\" notice, 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 REGENTS AND CONTRIBUTORS ``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 REGENTS OR CONTRIBUTORS 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.
|
||||
.\"
|
||||
.\"
|
||||
.Dd February 24, 2013.
|
||||
.Dt CK_RWCOHORT_READ_LOCK 3
|
||||
.Sh NAME
|
||||
.Nm CK_RWCOHORT_READ_LOCK
|
||||
.Nd acquire read-only permission for cohort-based reader-writer lock
|
||||
.Sh LIBRARY
|
||||
Concurrency Kit (libck, \-lck)
|
||||
.Sh SYNOPSIS
|
||||
.In ck_cohort.h
|
||||
.Fn CK_RWCOHORT_NEUTRAL_READ_LOCK "COHORT_NAME cohort_name" "LOCK *lock" "COHORT *cohort"\
|
||||
"void *global_context" "void *local_context"
|
||||
.Fn CK_RWCOHORT_RP_READ_LOCK "COHORT_NAME cohort_name" "LOCK *lock" "COHORT *cohort"\
|
||||
"void *global_context" "void *local_context"
|
||||
.Fn CK_RWCOHORT_WP_READ_LOCK "COHORT_NAME cohort_name" "LOCK *lock" "COHORT *cohort"\
|
||||
"void *global_context" "void *local_context"
|
||||
.Sh DESCRIPTION
|
||||
This call will acquire read-only permission from
|
||||
.Fa lock .
|
||||
The call will block until this permission has been acquired.
|
||||
.Fa cohort
|
||||
must point to a cohort whose global lock is the same as all other cohorts used with
|
||||
.Fa lock .
|
||||
The
|
||||
.Fa global_context
|
||||
and
|
||||
.Fa local_context
|
||||
arguments will be passed along as the context arguments to any calls to
|
||||
.Fa cohort .
|
||||
.
|
||||
.Sh SEE ALSO
|
||||
.Xr ck_cohort 3 ,
|
||||
.Xr CK_RWCOHORT_PROTOTYPE 3 ,
|
||||
.Xr CK_RWCOHORT_INSTANCE 3 ,
|
||||
.Xr CK_RWCOHORT_INITIALIZER 3 ,
|
||||
.Xr CK_RWCOHORT_INIT 3 ,
|
||||
.Xr CK_RWCOHORT_READ_UNLOCK 3 ,
|
||||
.Xr CK_RWCOHORT_WRITE_LOCK 3 ,
|
||||
.Xr CK_RWCOHORT_WRITE_UNLOCK 3 ,
|
||||
.Pp
|
||||
Additional information available at http://concurrencykit.org/
|
@ -0,0 +1,65 @@
|
||||
.\"
|
||||
.\" Copyright 2013 Brendon Scheinman.
|
||||
.\" 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, this list of conditions and the following disclaimer.
|
||||
.\" 2. Redistributions in binary form must reproduce the above copyright
|
||||
.\" notice, 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 REGENTS AND CONTRIBUTORS ``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 REGENTS OR CONTRIBUTORS 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.
|
||||
.\"
|
||||
.\"
|
||||
.Dd February 24, 2013.
|
||||
.Dt CK_RWCOHORT_READ_UNLOCK 3
|
||||
.Sh NAME
|
||||
.Nm CK_RWCOHORT_READ_UNLOCK
|
||||
.Nd relinquish read-only access to cohort-based reader-writer lock
|
||||
.Sh LIBRARY
|
||||
Concurrency Kit (libck, \-lck)
|
||||
.Sh SYNOPSIS
|
||||
.In ck_cohort.h
|
||||
.Fn CK_RWCOHORT_NEUTRAL_READ_UNLOCK "COHORT_NAME cohort_name" "LOCK *lock" "COHORT *cohort"\
|
||||
"void *global_context" "void *local_context"
|
||||
.Fn CK_RWCOHORT_RP_READ_UNLOCK "COHORT_NAME cohort_name" "LOCK *lock" "COHORT *cohort"\
|
||||
"void *global_context" "void *local_context"
|
||||
.Fn CK_RWCOHORT_WP_READ_UNLOCK "COHORT_NAME cohort_name" "LOCK *lock" "COHORT *cohort"\
|
||||
"void *global_context" "void *local_context"
|
||||
.Sh DESCRIPTION
|
||||
This call will relinquish read-only permission to
|
||||
.Fa lock .
|
||||
.Fa cohort
|
||||
must point to a cohort whose global lock is the same as all other cohorts used with
|
||||
.Fa lock .
|
||||
The
|
||||
.Fa global_context
|
||||
and
|
||||
.Fa local_context
|
||||
arguments will be passed along as the context arguments to any calls to
|
||||
.Fa cohort .
|
||||
.
|
||||
.Sh SEE ALSO
|
||||
.Xr ck_cohort 3 ,
|
||||
.Xr CK_RWCOHORT_PROTOTYPE 3 ,
|
||||
.Xr CK_RWCOHORT_INSTANCE 3 ,
|
||||
.Xr CK_RWCOHORT_INITIALIZER 3 ,
|
||||
.Xr CK_RWCOHORT_INIT 3 ,
|
||||
.Xr CK_RWCOHORT_READ_LOCK 3 ,
|
||||
.Xr CK_RWCOHORT_WRITE_LOCK 3 ,
|
||||
.Xr CK_RWCOHORT_WRITE_UNLOCK 3 ,
|
||||
.Pp
|
||||
Additional information available at http://concurrencykit.org/
|
@ -0,0 +1,66 @@
|
||||
.\"
|
||||
.\" Copyright 2013 Brendon Scheinman.
|
||||
.\" 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, this list of conditions and the following disclaimer.
|
||||
.\" 2. Redistributions in binary form must reproduce the above copyright
|
||||
.\" notice, 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 REGENTS AND CONTRIBUTORS ``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 REGENTS OR CONTRIBUTORS 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.
|
||||
.\"
|
||||
.\"
|
||||
.Dd February 24, 2013.
|
||||
.Dt CK_RWCOHORT_READ_LOCK 3
|
||||
.Sh NAME
|
||||
.Nm CK_RWCOHORT_READ_LOCK
|
||||
.Nd acquite write access for a cohort-based reader-writer lock
|
||||
.Sh LIBRARY
|
||||
Concurrency Kit (libck, \-lck)
|
||||
.Sh SYNOPSIS
|
||||
.In ck_cohort.h
|
||||
.Fn CK_RWCOHORT_NEUTRAL_WRITE_LOCK "COHORT_NAME cohort_name" "LOCK *lock" "COHORT *cohort"\
|
||||
"void *global_context" "void *local_context"
|
||||
.Fn CK_RWCOHORT_RP_WRITE_LOCK "COHORT_NAME cohort_name" "LOCK *lock" "COHORT *cohort"\
|
||||
"void *global_context" "void *local_context"
|
||||
.Fn CK_RWCOHORT_WP_WRITE_LOCK "COHORT_NAME cohort_name" "LOCK *lock" "COHORT *cohort"\
|
||||
"void *global_context" "void *local_context"
|
||||
.Sh DESCRIPTION
|
||||
This call will acquire write permission for
|
||||
.Fa lock .
|
||||
The call will block until this permission has been acquired.
|
||||
.Fa cohort
|
||||
must point to a cohort whose global lock is the same as all other cohorts used with
|
||||
.Fa lock .
|
||||
The
|
||||
.Fa global_context
|
||||
and
|
||||
.Fa local_context
|
||||
arguments will be passed along as the context arguments to any calls to
|
||||
.Fa cohort .
|
||||
.
|
||||
.Sh SEE ALSO
|
||||
.Xr ck_cohort 3 ,
|
||||
.Xr CK_RWCOHORT_PROTOTYPE 3 ,
|
||||
.Xr CK_RWCOHORT_INSTANCE 3 ,
|
||||
.Xr CK_RWCOHORT_INITIALIZER 3 ,
|
||||
.Xr CK_RWCOHORT_INIT 3 ,
|
||||
.Xr CK_RWCOHORT_READ_LOCK 3 ,
|
||||
.Xr CK_RWCOHORT_READ_UNLOCK 3 ,
|
||||
.Xr CK_RWCOHORT_WRITE_UNLOCK 3 ,
|
||||
.Pp
|
||||
Additional information available at http://concurrencykit.org/
|
@ -0,0 +1,65 @@
|
||||
.\"
|
||||
.\" Copyright 2013 Brendon Scheinman.
|
||||
.\" 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, this list of conditions and the following disclaimer.
|
||||
.\" 2. Redistributions in binary form must reproduce the above copyright
|
||||
.\" notice, 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 REGENTS AND CONTRIBUTORS ``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 REGENTS OR CONTRIBUTORS 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.
|
||||
.\"
|
||||
.\"
|
||||
.Dd February 24, 2013.
|
||||
.Dt CK_RWCOHORT_READ_LOCK 3
|
||||
.Sh NAME
|
||||
.Nm CK_RWCOHORT_READ_LOCK
|
||||
.Nd relinquish write access for cohort-based reader-writer lock
|
||||
.Sh LIBRARY
|
||||
Concurrency Kit (libck, \-lck)
|
||||
.Sh SYNOPSIS
|
||||
.In ck_cohort.h
|
||||
.Fn CK_RWCOHORT_NEUTRAL_WRITE_UNLOCK "COHORT_NAME cohort_name" "LOCK *lock" "COHORT *cohort"\
|
||||
"void *global_context" "void *local_context"
|
||||
.Fn CK_RWCOHORT_RP_WRITE_UNLOCK "COHORT_NAME cohort_name" "LOCK *lock" "COHORT *cohort"\
|
||||
"void *global_context" "void *local_context"
|
||||
.Fn CK_RWCOHORT_WP_WRITE_UNLOCK "COHORT_NAME cohort_name" "LOCK *lock" "COHORT *cohort"\
|
||||
"void *global_context" "void *local_context"
|
||||
.Sh DESCRIPTION
|
||||
This call will relinquish write permission for
|
||||
.Fa lock .
|
||||
.Fa cohort
|
||||
must point to a cohort whose global lock is the same as all other cohorts used with
|
||||
.Fa lock .
|
||||
The
|
||||
.Fa global_context
|
||||
and
|
||||
.Fa local_context
|
||||
arguments will be passed along as the context arguments to any calls to
|
||||
.Fa cohort .
|
||||
.
|
||||
.Sh SEE ALSO
|
||||
.Xr ck_cohort 3 ,
|
||||
.Xr CK_RWCOHORT_PROTOTYPE 3 ,
|
||||
.Xr CK_RWCOHORT_INSTANCE 3 ,
|
||||
.Xr CK_RWCOHORT_INITIALIZER 3 ,
|
||||
.Xr CK_RWCOHORT_INIT 3 ,
|
||||
.Xr CK_RWCOHORT_READ_LOCK 3 ,
|
||||
.Xr CK_RWCOHORT_READ_UNLOCK 3 ,
|
||||
.Xr CK_RWCOHORT_WRITE_LOCK 3 ,
|
||||
.Pp
|
||||
Additional information available at http://concurrencykit.org/
|
@ -0,0 +1,204 @@
|
||||
.\"
|
||||
.\" Copyright 2013 Brendon Scheinman.
|
||||
.\" 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, this list of conditions and the following disclaimer.
|
||||
.\" 2. Redistributions in binary form must reproduce the above copyright
|
||||
.\" notice, 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 REGENTS AND CONTRIBUTORS ``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 REGENTS OR CONTRIBUTORS 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.
|
||||
.\"
|
||||
.\"
|
||||
.Dd April 23, 2013.
|
||||
.Dt ck_rwcohort 3
|
||||
.Sh NAME
|
||||
.Nm ck_rwcohort
|
||||
.Nd generalized interface for reader-writer locks using cohort locks
|
||||
.Sh LIBRARY
|
||||
Concurrency Kit (libck, \-lck)
|
||||
.Sh SYNOPSIS
|
||||
.In ck_rwcohort.h
|
||||
In each of the following macros, "STRATEGY" should be replaced with either "NEUTRAL", "RP", or "WP"
|
||||
depending on which locking strategy the user prefers. RP and WP represent reader preference and
|
||||
writer preference, respectively, while NEUTRAL represents a strategy neutral to reads vs. writes.
|
||||
.Fn CK_RWCOHORT_STRATEGY_PROTOTYPE "COHORT_NAME cohort_name"
|
||||
.Fn CK_RWCOHORT_STRATEGY_NAME "COHORT_NAME cohort_name"
|
||||
.Fn CK_RWCOHORT_STRATEGY_INSTANCE "COHORT_NAME cohort_name"
|
||||
.Fn CK_RWCOHORT_STRATEGY_INIT "COHORT_NAME cohort_name" "RWCOHORT lock" "unsigned int wait_limit"
|
||||
Note: the wait_limit argument should be omitted for locks using the neutral strategy
|
||||
.Fn CK_RWCOHORT_STRATEGY_READ_LOCK "COHORT_NAME cohort_name" "RWCOHORT lock" "COHORT cohort" \
|
||||
"void *global_context" "void *local_context"
|
||||
.Fn CK_RWCOHORT_STRATEGY_READ_UNLOCK "COHORT_NAME cohort_name" "RWCOHORT lock" "COHORT cohort" \
|
||||
"void *global_context" "void *local_context"
|
||||
.Fn CK_RWCOHORT_STRATEGY_WRITE_LOCK "COHORT_NAME cohort_name" "RWCOHORT lock" "COHORT cohort" \
|
||||
"void *global_context" "void *local_context"
|
||||
.Fn CK_RWCOHORT_STRATEGY_WRITE_UNLOCK "COHORT_NAME cohort_name" "RWCOHORT lock" "COHORT cohort" \
|
||||
"void *global_context" "void *local_context"
|
||||
|
||||
Arguments of type RWCOHORT must be pointers to structs defined using the
|
||||
.Xr CK_RWCOHORT_STRATEGY_PROTOTYPE 3
|
||||
macro with the same strategy and cohort name as the current call.
|
||||
|
||||
Arguments of type COHORT must be pointers to structs defined using the
|
||||
.Xr CK_COHORT_PROTOTYPE 3
|
||||
macro.
|
||||
|
||||
.Sh DESCRIPTION
|
||||
ck_rwcohort.h provides an interface for defining reader-writer locks
|
||||
that use cohort locks internally to increase performance on NUMA
|
||||
architectures. See
|
||||
.Xr ck_cohort 3
|
||||
for more information about cohort locks.
|
||||
.Pp
|
||||
Before using a reader-writer cohort lock, the user must define a cohort type using
|
||||
either the
|
||||
.Xr CK_COHORT_PROTOTYPE 3
|
||||
or the
|
||||
.Xr CK_COHORT_TRYLOCK_PROTOTYPE 3
|
||||
macros, and define a reader-writer lock type using the
|
||||
.Xr CK_RWCOHORT_PROTOTYPE 3
|
||||
macro.
|
||||
.Pp
|
||||
.Sh EXAMPLE
|
||||
.Bd -literal -offset indent
|
||||
#include <stdlib.h>
|
||||
#include <pthread.h>
|
||||
|
||||
#include <ck_pr.h>
|
||||
#include <ck_cohort.h>
|
||||
#include <ck_rwcohort.h>
|
||||
#include <ck_spinlock.h>
|
||||
|
||||
/* Create cohort methods with signatures that match the required signature */
|
||||
|
||||
static void
|
||||
ck_spinlock_lock_with_context(ck_spinlock_t *lock, void *context)
|
||||
{
|
||||
(void)context;
|
||||
ck_spinlock_lock(lock);
|
||||
return;
|
||||
}
|
||||
|
||||
static void
|
||||
ck_spinlock_unlock_with_context(ck_spinlock_t *lock, void *context)
|
||||
{
|
||||
(void)context;
|
||||
ck_spinlock_unlock(lock);
|
||||
return;
|
||||
}
|
||||
|
||||
static bool
|
||||
ck_spinlock_locked_with_context(ck_spinlock_t *lock, void *context)
|
||||
{
|
||||
(void)context;
|
||||
return ck_spinlock_locked(lock);
|
||||
}
|
||||
|
||||
/*
|
||||
* define a cohort type named "test_cohort" that will use
|
||||
* the above methods for both its global and local locks
|
||||
*/
|
||||
CK_COHORT_PROTOTYPE(test_cohort,
|
||||
ck_spinlock_lock_with_context, ck_spinlock_unlock_with_context, ck_spinlock_locked_with_context,
|
||||
ck_spinlock_lock_with_context, ck_spinlock_unlock_with_context, ck_spinlock_locked_with_context)
|
||||
|
||||
/* define a reader-writer type using the same cohort type */
|
||||
CK_RWCOHORT_WP_PROTOTYPE(test_cohort)
|
||||
|
||||
static ck_spinlock_t global_lock = CK_SPINLOCK_INITIALIZER;
|
||||
static CK_COHORT_INSTANCE(test_cohort) *cohorts;
|
||||
static CK_RWCOHORT_WP_INSTANCE(test_cohort) rw_cohort = CK_RWCOHORT_WP_INITIALIZER;
|
||||
static unsigned int ready;
|
||||
|
||||
static void *
|
||||
function(void *context)
|
||||
{
|
||||
CK_COHORT_INSTANCE(test_cohort) *cohort = context;
|
||||
|
||||
while (ck_pr_load_uint(&ready) == 0);
|
||||
|
||||
while (ck_pr_load_uint(&ready) > 0) {
|
||||
/*
|
||||
* acquire the cohort lock before performing critical section.
|
||||
* note that we pass NULL for both the global and local context
|
||||
* arguments because neither the lock nor unlock functions
|
||||
* will use them.
|
||||
*/
|
||||
CK_COHORT_LOCK(test_cohort, cohort, NULL, NULL);
|
||||
|
||||
/* perform critical section */
|
||||
|
||||
/* relinquish cohort lock */
|
||||
CK_COHORT_UNLOCK(test_cohort, cohort, NULL, NULL);
|
||||
}
|
||||
|
||||
return NULL;
|
||||
}
|
||||
|
||||
int
|
||||
main(void)
|
||||
{
|
||||
unsigned int nthr = 4;
|
||||
unsigned int n_cohorts = 2;
|
||||
unsigned int i;
|
||||
|
||||
/* allocate 2 cohorts of the defined type */
|
||||
CK_COHORT_INSTANCE(test_cohort) *cohorts =
|
||||
calloc(n_cohorts, sizeof(CK_COHORT_INSTANCE(test_cohort)));
|
||||
|
||||
/* create local locks to use with each cohort */
|
||||
ck_spinlock_t *local_locks =
|
||||
calloc(n_cohorts, sizeof(ck_spinlock_t));
|
||||
|
||||
pthread_t *threads =
|
||||
calloc(nthr, sizeof(pthread_t));
|
||||
|
||||
/* initialize each of the cohorts before using them */
|
||||
for (i = 0 ; i < n_cohorts ; ++i) {
|
||||
CK_COHORT_INIT(test_cohort, cohorts + i, &global_lock, local_locks + i,
|
||||
CK_COHORT_DEFAULT_LOCAL_PASS_LIMIT);
|
||||
}
|
||||
|
||||
/* start each thread and assign cohorts equally */
|
||||
for (i = 0 ; i < nthr ; ++i) {
|
||||
pthread_create(threads + i, NULL, function, cohorts + (i % n_cohorts));
|
||||
}
|
||||
|
||||
ck_pr_store_uint(&ready, 1);
|
||||
sleep(10);
|
||||
ck_pr_store_uint(&ready, 0);
|
||||
|
||||
for (i = 0 ; i < nthr ; ++i) {
|
||||
pthread_join(threads[i], NULL);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
.Ed
|
||||
.Sh SEE ALSO
|
||||
.Xr CK_COHORT_PROTOTYPE 3 ,
|
||||
.Xr CK_COHORT_TRYLOCK_PROTOTYPE 3 ,
|
||||
.Xr CK_COHORT_INSTANCE 3 ,
|
||||
.Xr CK_COHORT_INITIALIZER 3 ,
|
||||
.Xr CK_COHORT_INIT 3 ,
|
||||
.Xr CK_COHORT_LOCK 3 ,
|
||||
.Xr CK_COHORT_UNLOCK 3 ,
|
||||
.Xr CK_COHORT_LOCKED 3 ,
|
||||
.Xr CK_COHORT_TRYLOCK 3 ,
|
||||
.Pp
|
||||
Additional information available at http://concurrencykit.org/
|
@ -0,0 +1,292 @@
|
||||
/*
|
||||
* Copyright 2013 Samy Al Bahra.
|
||||
* Copyright 2013 Brendon Scheinman.
|
||||
* 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, this list of conditions and the following disclaimer.
|
||||
* 2. Redistributions in binary form must reproduce the above copyright
|
||||
* notice, 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 AUTHOR AND CONTRIBUTORS ``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 AUTHOR OR CONTRIBUTORS 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.
|
||||
*/
|
||||
|
||||
#ifndef _CK_RWCOHORT_H
|
||||
#define _CK_RWCOHORT_H
|
||||
|
||||
/*
|
||||
* This is an implementation of NUMA-aware reader-writer locks as described in:
|
||||
* Calciu, I.; Dice, D.; Lev, Y.; Luchangco, V.; Marathe, V.; and Shavit, N. 2013.
|
||||
* NUMA-Aware Reader-Writer Locks
|
||||
*/
|
||||
|
||||
#include <ck_cc.h>
|
||||
#include <ck_pr.h>
|
||||
#include <stddef.h>
|
||||
#include <ck_cohort.h>
|
||||
|
||||
#define CK_RWCOHORT_WP_NAME(N) ck_rwcohort_wp_##N
|
||||
#define CK_RWCOHORT_WP_INSTANCE(N) struct CK_RWCOHORT_WP_NAME(N)
|
||||
#define CK_RWCOHORT_WP_INIT(N, RW, WL) ck_rwcohort_wp_##N##_init(RW, WL)
|
||||
#define CK_RWCOHORT_WP_READ_LOCK(N, RW, C, GC, LC) ck_rwcohort_wp_##N##_read_lock(RW, C, GC, LC)
|
||||
#define CK_RWCOHORT_WP_READ_UNLOCK(N, RW, C, GC, LC) ck_rwcohort_wp_##N##_read_unlock(RW)
|
||||
#define CK_RWCOHORT_WP_WRITE_LOCK(N, RW, C, GC, LC) ck_rwcohort_wp_##N##_write_lock(RW, C, GC, LC)
|
||||
#define CK_RWCOHORT_WP_WRITE_UNLOCK(N, RW, C, GC, LC) ck_rwcohort_wp_##N##_write_unlock(RW, C, GC, LC)
|
||||
#define CK_RWCOHORT_WP_DEFAULT_WAIT_LIMIT 1000
|
||||
|
||||
#define CK_RWCOHORT_WP_PROTOTYPE(N) \
|
||||
CK_RWCOHORT_WP_INSTANCE(N) { \
|
||||
unsigned int read_counter; \
|
||||
unsigned int write_barrier; \
|
||||
unsigned int wait_limit; \
|
||||
}; \
|
||||
CK_CC_INLINE static void \
|
||||
ck_rwcohort_wp_##N##_init(CK_RWCOHORT_WP_INSTANCE(N) *rw_cohort, \
|
||||
unsigned int wait_limit) \
|
||||
{ \
|
||||
rw_cohort->read_counter = 0; \
|
||||
rw_cohort->write_barrier = 0; \
|
||||
rw_cohort->wait_limit = wait_limit; \
|
||||
ck_pr_barrier(); \
|
||||
return; \
|
||||
} \
|
||||
CK_CC_INLINE static void \
|
||||
ck_rwcohort_wp_##N##_write_lock(CK_RWCOHORT_WP_INSTANCE(N) *rw_cohort, \
|
||||
CK_COHORT_INSTANCE(N) *cohort, void *global_context, \
|
||||
void *local_context) \
|
||||
{ \
|
||||
while (ck_pr_load_uint(&rw_cohort->write_barrier) > 0) { \
|
||||
ck_pr_stall(); \
|
||||
} \
|
||||
\
|
||||
CK_COHORT_LOCK(N, cohort, global_context, local_context); \
|
||||
\
|
||||
while (ck_pr_load_uint(&rw_cohort->read_counter) > 0) { \
|
||||
ck_pr_stall(); \
|
||||
} \
|
||||
\
|
||||
return; \
|
||||
} \
|
||||
CK_CC_INLINE static void \
|
||||
ck_rwcohort_wp_##N##_write_unlock(CK_RWCOHORT_WP_INSTANCE(N) *rw_cohort, \
|
||||
CK_COHORT_INSTANCE(N) *cohort, void *global_context, \
|
||||
void *local_context) \
|
||||
{ \
|
||||
(void)rw_cohort; \
|
||||
CK_COHORT_UNLOCK(N, cohort, global_context, local_context); \
|
||||
return; \
|
||||
} \
|
||||
CK_CC_INLINE static void \
|
||||
ck_rwcohort_wp_##N##_read_lock(CK_RWCOHORT_WP_INSTANCE(N) *rw_cohort, \
|
||||
CK_COHORT_INSTANCE(N) *cohort, void *global_context, \
|
||||
void *local_context) \
|
||||
{ \
|
||||
unsigned int wait_count = 0; \
|
||||
bool raised = false; \
|
||||
\
|
||||
for (;;) { \
|
||||
ck_pr_inc_uint(&rw_cohort->read_counter); \
|
||||
if (CK_COHORT_LOCKED(N, cohort, global_context, local_context) == false) {\
|
||||
break; \
|
||||
} \
|
||||
\
|
||||
ck_pr_dec_uint(&rw_cohort->read_counter); \
|
||||
while (CK_COHORT_LOCKED(N, cohort, global_context, local_context) == true) {\
|
||||
ck_pr_stall(); \
|
||||
if (++wait_count > rw_cohort->wait_limit && raised == false) { \
|
||||
ck_pr_inc_uint(&rw_cohort->write_barrier); \
|
||||
raised = true; \
|
||||
} \
|
||||
} \
|
||||
} \
|
||||
\
|
||||
if (raised == true) { \
|
||||
ck_pr_dec_uint(&rw_cohort->write_barrier); \
|
||||
} \
|
||||
\
|
||||
return; \
|
||||
} \
|
||||
CK_CC_INLINE static void \
|
||||
ck_rwcohort_wp_##N##_read_unlock(CK_RWCOHORT_WP_INSTANCE(N) *cohort) \
|
||||
{ \
|
||||
ck_pr_dec_uint(&cohort->read_counter); \
|
||||
return; \
|
||||
}
|
||||
|
||||
#define CK_RWCOHORT_WP_INITIALIZER { \
|
||||
.read_counter = 0, \
|
||||
.write_barrier = 0, \
|
||||
.wait_limit = 0 \
|
||||
}
|
||||
|
||||
|
||||
#define CK_RWCOHORT_RP_NAME(N) ck_rwcohort_rp_##N
|
||||
#define CK_RWCOHORT_RP_INSTANCE(N) struct CK_RWCOHORT_RP_NAME(N)
|
||||
#define CK_RWCOHORT_RP_INIT(N, RW, WL) ck_rwcohort_rp_##N##_init(RW, WL)
|
||||
#define CK_RWCOHORT_RP_READ_LOCK(N, RW, C, GC, LC) ck_rwcohort_rp_##N##_read_lock(RW, C, GC, LC)
|
||||
#define CK_RWCOHORT_RP_READ_UNLOCK(N, RW, C, GC, LC) ck_rwcohort_rp_##N##_read_unlock(RW)
|
||||
#define CK_RWCOHORT_RP_WRITE_LOCK(N, RW, C, GC, LC) ck_rwcohort_rp_##N##_write_lock(RW, C, GC, LC)
|
||||
#define CK_RWCOHORT_RP_WRITE_UNLOCK(N, RW, C, GC, LC) ck_rwcohort_rp_##N##_write_unlock(RW, C, GC, LC)
|
||||
#define CK_RWCOHORT_RP_DEFAULT_WAIT_LIMIT 1000
|
||||
|
||||
#define CK_RWCOHORT_RP_PROTOTYPE(N) \
|
||||
CK_RWCOHORT_RP_INSTANCE(N) { \
|
||||
unsigned int read_counter; \
|
||||
unsigned int read_barrier; \
|
||||
unsigned int wait_limit; \
|
||||
}; \
|
||||
CK_CC_INLINE static void \
|
||||
ck_rwcohort_rp_##N##_init(CK_RWCOHORT_RP_INSTANCE(N) *rw_cohort, \
|
||||
unsigned int wait_limit) \
|
||||
{ \
|
||||
rw_cohort->read_counter = 0; \
|
||||
rw_cohort->read_barrier = 0; \
|
||||
rw_cohort->wait_limit = wait_limit; \
|
||||
ck_pr_barrier(); \
|
||||
return; \
|
||||
} \
|
||||
CK_CC_INLINE static void \
|
||||
ck_rwcohort_rp_##N##_write_lock(CK_RWCOHORT_RP_INSTANCE(N) *rw_cohort, \
|
||||
CK_COHORT_INSTANCE(N) *cohort, void *global_context, \
|
||||
void *local_context) \
|
||||
{ \
|
||||
unsigned int wait_count = 0; \
|
||||
bool raised = false; \
|
||||
\
|
||||
for (;;) { \
|
||||
CK_COHORT_LOCK(N, cohort, global_context, local_context); \
|
||||
if (ck_pr_load_uint(&rw_cohort->read_counter) == 0) { \
|
||||
break; \
|
||||
} \
|
||||
\
|
||||
CK_COHORT_UNLOCK(N, cohort, global_context, local_context); \
|
||||
while (ck_pr_load_uint(&rw_cohort->read_counter) > 0) { \
|
||||
ck_pr_stall(); \
|
||||
if (++wait_count > rw_cohort->wait_limit && raised == false) { \
|
||||
ck_pr_inc_uint(&rw_cohort->read_barrier); \
|
||||
raised = true; \
|
||||
} \
|
||||
} \
|
||||
} \
|
||||
\
|
||||
if (raised == true) { \
|
||||
ck_pr_dec_uint(&rw_cohort->read_barrier); \
|
||||
} \
|
||||
\
|
||||
return; \
|
||||
} \
|
||||
CK_CC_INLINE static void \
|
||||
ck_rwcohort_rp_##N##_write_unlock(CK_RWCOHORT_RP_INSTANCE(N) *rw_cohort, \
|
||||
CK_COHORT_INSTANCE(N) *cohort, void *global_context, void *local_context) \
|
||||
{ \
|
||||
(void)rw_cohort; \
|
||||
CK_COHORT_UNLOCK(N, cohort, global_context, local_context); \
|
||||
return; \
|
||||
} \
|
||||
CK_CC_INLINE static void \
|
||||
ck_rwcohort_rp_##N##_read_lock(CK_RWCOHORT_RP_INSTANCE(N) *rw_cohort, \
|
||||
CK_COHORT_INSTANCE(N) *cohort, void *global_context, \
|
||||
void *local_context) \
|
||||
{ \
|
||||
while (ck_pr_load_uint(&rw_cohort->read_barrier) > 0) { \
|
||||
ck_pr_stall(); \
|
||||
} \
|
||||
ck_pr_inc_uint(&rw_cohort->read_counter); \
|
||||
\
|
||||
while (CK_COHORT_LOCKED(N, cohort, global_context, local_context) == true) { \
|
||||
ck_pr_stall(); \
|
||||
} \
|
||||
\
|
||||
return; \
|
||||
} \
|
||||
CK_CC_INLINE static void \
|
||||
ck_rwcohort_rp_##N##_read_unlock(CK_RWCOHORT_RP_INSTANCE(N) *cohort) \
|
||||
{ \
|
||||
ck_pr_dec_uint(&cohort->read_counter); \
|
||||
return; \
|
||||
}
|
||||
|
||||
#define CK_RWCOHORT_RP_INITIALIZER { \
|
||||
.read_counter = 0, \
|
||||
.read_barrier = 0, \
|
||||
.wait_limit = 0 \
|
||||
}
|
||||
|
||||
|
||||
#define CK_RWCOHORT_NEUTRAL_NAME(N) ck_rwcohort_neutral_##N
|
||||
#define CK_RWCOHORT_NEUTRAL_INSTANCE(N) struct CK_RWCOHORT_NEUTRAL_NAME(N)
|
||||
#define CK_RWCOHORT_NEUTRAL_INIT(N, RW) ck_rwcohort_neutral_##N##_init(RW)
|
||||
#define CK_RWCOHORT_NEUTRAL_READ_LOCK(N, RW, C, GC, LC) ck_rwcohort_neutral_##N##_read_lock(RW, C, GC, LC)
|
||||
#define CK_RWCOHORT_NEUTRAL_READ_UNLOCK(N, RW, C, GC, LC) ck_rwcohort_neutral_##N##_read_unlock(RW)
|
||||
#define CK_RWCOHORT_NEUTRAL_WRITE_LOCK(N, RW, C, GC, LC) ck_rwcohort_neutral_##N##_write_lock(RW, C, GC, LC)
|
||||
#define CK_RWCOHORT_NEUTRAL_WRITE_UNLOCK(N, RW, C, GC, LC) ck_rwcohort_neutral_##N##_write_unlock(RW, C, GC, LC)
|
||||
#define CK_RWCOHORT_NEUTRAL_DEFAULT_WAIT_LIMIT 1000
|
||||
|
||||
#define CK_RWCOHORT_NEUTRAL_PROTOTYPE(N) \
|
||||
CK_RWCOHORT_NEUTRAL_INSTANCE(N) { \
|
||||
unsigned int read_counter; \
|
||||
}; \
|
||||
CK_CC_INLINE static void \
|
||||
ck_rwcohort_neutral_##N##_init(CK_RWCOHORT_NEUTRAL_INSTANCE(N) *rw_cohort) \
|
||||
{ \
|
||||
rw_cohort->read_counter = 0; \
|
||||
ck_pr_barrier(); \
|
||||
return; \
|
||||
} \
|
||||
CK_CC_INLINE static void \
|
||||
ck_rwcohort_neutral_##N##_write_lock(CK_RWCOHORT_NEUTRAL_INSTANCE(N) *rw_cohort, \
|
||||
CK_COHORT_INSTANCE(N) *cohort, void *global_context, \
|
||||
void *local_context) \
|
||||
{ \
|
||||
CK_COHORT_LOCK(N, cohort, global_context, local_context); \
|
||||
while (ck_pr_load_uint(&rw_cohort->read_counter) > 0) { \
|
||||
ck_pr_stall(); \
|
||||
} \
|
||||
return; \
|
||||
} \
|
||||
CK_CC_INLINE static void \
|
||||
ck_rwcohort_neutral_##N##_write_unlock(CK_RWCOHORT_NEUTRAL_INSTANCE(N) *rw_cohort, \
|
||||
CK_COHORT_INSTANCE(N) *cohort, void *global_context, void *local_context) \
|
||||
{ \
|
||||
(void)rw_cohort; \
|
||||
CK_COHORT_UNLOCK(N, cohort, global_context, local_context); \
|
||||
return; \
|
||||
} \
|
||||
CK_CC_INLINE static void \
|
||||
ck_rwcohort_neutral_##N##_read_lock(CK_RWCOHORT_NEUTRAL_INSTANCE(N) *rw_cohort, \
|
||||
CK_COHORT_INSTANCE(N) *cohort, void *global_context, \
|
||||
void *local_context) \
|
||||
{ \
|
||||
CK_COHORT_LOCK(N, cohort, global_context, local_context); \
|
||||
ck_pr_inc_uint(&rw_cohort->read_counter); \
|
||||
CK_COHORT_UNLOCK(N, cohort, global_context, local_context); \
|
||||
\
|
||||
return; \
|
||||
} \
|
||||
CK_CC_INLINE static void \
|
||||
ck_rwcohort_neutral_##N##_read_unlock(CK_RWCOHORT_NEUTRAL_INSTANCE(N) *cohort) \
|
||||
{ \
|
||||
ck_pr_dec_uint(&cohort->read_counter); \
|
||||
return; \
|
||||
}
|
||||
|
||||
#define CK_RWCOHORT_NEUTRAL_INITIALIZER { \
|
||||
.read_counter = 0, \
|
||||
}
|
||||
|
||||
|
||||
#endif /* _CK_RWCOHORT_H */
|
@ -0,0 +1,32 @@
|
||||
.PHONY: clean distribution
|
||||
|
||||
OBJECTS=latency throughput
|
||||
OBJECTS=ck_neutral.THROUGHPUT ck_neutral.LATENCY \
|
||||
ck_rp.THROUGHPUT ck_rp.LATENCY \
|
||||
ck_wp.THROUGHPUT ck_wp.LATENCY
|
||||
|
||||
all: $(OBJECTS)
|
||||
|
||||
ck_neutral.THROUGHPUT: ck_neutral.c
|
||||
$(CC) -DTHROUGHPUT $(CFLAGS) -o ck_neutral.THROUGHPUT ck_neutral.c
|
||||
|
||||
ck_neutral.LATENCY: ck_neutral.c
|
||||
$(CC) -DLATENCY $(CFLAGS) -o ck_neutral.LATENCY ck_neutral.c
|
||||
|
||||
ck_rp.THROUGHPUT: ck_rp.c
|
||||
$(CC) -DTHROUGHPUT $(CFLAGS) -o ck_rp.THROUGHPUT ck_rp.c
|
||||
|
||||
ck_rp.LATENCY: ck_rp.c
|
||||
$(CC) -DLATENCY $(CFLAGS) -o ck_rp.LATENCY ck_rp.c
|
||||
|
||||
ck_wp.THROUGHPUT: ck_wp.c
|
||||
$(CC) -DTHROUGHPUT $(CFLAGS) -o ck_wp.THROUGHPUT ck_wp.c
|
||||
|
||||
ck_wp.LATENCY: ck_wp.c
|
||||
$(CC) -DLATENCY $(CFLAGS) -o ck_wp.LATENCY ck_wp.c
|
||||
|
||||
clean:
|
||||
rm -rf *.dSYM *~ *.o $(OBJECTS)
|
||||
|
||||
include ../../../build/regressions.build
|
||||
CFLAGS+=$(PTHREAD_CFLAGS) -D_GNU_SOURCE
|
@ -0,0 +1,7 @@
|
||||
#include "../ck_neutral.h"
|
||||
|
||||
#ifdef THROUGHPUT
|
||||
#include "throughput.h"
|
||||
#elif defined(LATENCY)
|
||||
#include "latency.h"
|
||||
#endif
|
@ -0,0 +1,7 @@
|
||||
#include "../ck_rp.h"
|
||||
|
||||
#ifdef THROUGHPUT
|
||||
#include "throughput.h"
|
||||
#elif defined(LATENCY)
|
||||
#include "latency.h"
|
||||
#endif
|
@ -0,0 +1,7 @@
|
||||
#include "../ck_wp.h"
|
||||
|
||||
#ifdef THROUGHPUT
|
||||
#include "throughput.h"
|
||||
#elif defined(LATENCY)
|
||||
#include "latency.h"
|
||||
#endif
|
@ -0,0 +1,106 @@
|
||||
/*
|
||||
* Copyright 2013 Samy Al Bahra.
|
||||
* Copyright 2013 Brendon Scheinman.
|
||||
* 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, this list of conditions and the following disclaimer.
|
||||
* 2. Redistributions in binary form must reproduce the above copyright
|
||||
* notice, 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 AUTHOR AND CONTRIBUTORS ``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 AUTHOR OR CONTRIBUTORS 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.
|
||||
*/
|
||||
|
||||
#include <ck_rwcohort.h>
|
||||
#include <ck_spinlock.h>
|
||||
#include <inttypes.h>
|
||||
#include <stdio.h>
|
||||
|
||||
#include "../../common.h"
|
||||
|
||||
#ifndef STEPS
|
||||
#define STEPS 1000000
|
||||
#endif
|
||||
|
||||
static void
|
||||
ck_spinlock_fas_lock_with_context(ck_spinlock_fas_t *lock, void *context)
|
||||
{
|
||||
(void)context;
|
||||
ck_spinlock_fas_lock(lock);
|
||||
}
|
||||
|
||||
static void
|
||||
ck_spinlock_fas_unlock_with_context(ck_spinlock_fas_t *lock, void *context)
|
||||
{
|
||||
(void)context;
|
||||
ck_spinlock_fas_unlock(lock);
|
||||
}
|
||||
|
||||
static bool
|
||||
ck_spinlock_fas_locked_with_context(ck_spinlock_fas_t *lock, void *context)
|
||||
{
|
||||
(void)context;
|
||||
return ck_spinlock_fas_locked(lock);
|
||||
}
|
||||
|
||||
CK_COHORT_PROTOTYPE(fas_fas,
|
||||
ck_spinlock_fas_lock_with_context, ck_spinlock_fas_unlock_with_context, ck_spinlock_fas_locked_with_context,
|
||||
ck_spinlock_fas_lock_with_context, ck_spinlock_fas_unlock_with_context, ck_spinlock_fas_locked_with_context)
|
||||
LOCK_PROTOTYPE(fas_fas)
|
||||
|
||||
int
|
||||
main(void)
|
||||
{
|
||||
uint64_t s_b, e_b, i;
|
||||
ck_spinlock_fas_t global_lock = CK_SPINLOCK_FAS_INITIALIZER;
|
||||
ck_spinlock_fas_t local_lock = CK_SPINLOCK_FAS_INITIALIZER;
|
||||
CK_COHORT_INSTANCE(fas_fas) cohort = CK_COHORT_INITIALIZER;
|
||||
LOCK_INSTANCE(fas_fas) rw_cohort = LOCK_INITIALIZER;
|
||||
|
||||
CK_COHORT_INIT(fas_fas, &cohort, &global_lock, &local_lock,
|
||||
CK_COHORT_DEFAULT_LOCAL_PASS_LIMIT);
|
||||
LOCK_INIT(fas_fas, &rw_cohort, CK_RWCOHORT_WP_DEFAULT_WAIT_LIMIT);
|
||||
|
||||
for (i = 0; i < STEPS; i++) {
|
||||
WRITE_LOCK(fas_fas, &rw_cohort, &cohort, NULL, NULL);
|
||||
WRITE_UNLOCK(fas_fas, &rw_cohort, &cohort, NULL, NULL);
|
||||
}
|
||||
|
||||
s_b = rdtsc();
|
||||
for (i = 0; i < STEPS; i++) {
|
||||
WRITE_LOCK(fas_fas, &rw_cohort, &cohort, NULL, NULL);
|
||||
WRITE_UNLOCK(fas_fas, &rw_cohort, &cohort, NULL, NULL);
|
||||
}
|
||||
e_b = rdtsc();
|
||||
printf("WRITE: rwlock %15" PRIu64 "\n", (e_b - s_b) / STEPS);
|
||||
|
||||
for (i = 0; i < STEPS; i++) {
|
||||
READ_LOCK(fas_fas, &rw_cohort, &cohort, NULL, NULL);
|
||||
READ_UNLOCK(fas_fas, &rw_cohort, &cohort, NULL, NULL);
|
||||
}
|
||||
|
||||
s_b = rdtsc();
|
||||
for (i = 0; i < STEPS; i++) {
|
||||
READ_LOCK(fas_fas, &rw_cohort, &cohort, NULL, NULL);
|
||||
READ_UNLOCK(fas_fas, &rw_cohort, &cohort, NULL, NULL);
|
||||
}
|
||||
e_b = rdtsc();
|
||||
printf("READ: rwlock %15" PRIu64 "\n", (e_b - s_b) / STEPS);
|
||||
|
||||
return (0);
|
||||
}
|
||||
|
@ -0,0 +1,245 @@
|
||||
/*
|
||||
* Copyright 2013 Samy Al Bahra.
|
||||
* Copyright 2013 Brendon Scheinman.
|
||||
* 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, this list of conditions and the following disclaimer.
|
||||
* 2. Redistributions in binary form must reproduce the above copyright
|
||||
* notice, 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 AUTHOR AND CONTRIBUTORS ``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 AUTHOR OR CONTRIBUTORS 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.
|
||||
*/
|
||||
|
||||
#include <ck_cohort.h>
|
||||
#include <ck_rwcohort.h>
|
||||
#include <ck_spinlock.h>
|
||||
#include <inttypes.h>
|
||||
#include <pthread.h>
|
||||
#include <stdio.h>
|
||||
#include <stdint.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <unistd.h>
|
||||
|
||||
#include "../../common.h"
|
||||
|
||||
#define max(x, y) (((x) > (y)) ? (x) : (y))
|
||||
|
||||
#ifndef STEPS
|
||||
#define STEPS 1000000
|
||||
#endif
|
||||
|
||||
static unsigned int barrier;
|
||||
static unsigned int flag CK_CC_CACHELINE;
|
||||
static struct affinity affinity;
|
||||
static unsigned int nthr;
|
||||
|
||||
static void
|
||||
ck_spinlock_fas_lock_with_context(ck_spinlock_fas_t *lock, void *context)
|
||||
{
|
||||
|
||||
(void)context;
|
||||
ck_spinlock_fas_lock(lock);
|
||||
return;
|
||||
}
|
||||
|
||||
static void
|
||||
ck_spinlock_fas_unlock_with_context(ck_spinlock_fas_t *lock, void *context)
|
||||
{
|
||||
|
||||
(void)context;
|
||||
ck_spinlock_fas_unlock(lock);
|
||||
return;
|
||||
}
|
||||
|
||||
static bool
|
||||
ck_spinlock_fas_locked_with_context(ck_spinlock_fas_t *lock, void *context)
|
||||
{
|
||||
|
||||
(void)context;
|
||||
return ck_spinlock_fas_locked(lock);
|
||||
}
|
||||
|
||||
CK_COHORT_PROTOTYPE(fas_fas,
|
||||
ck_spinlock_fas_lock_with_context, ck_spinlock_fas_unlock_with_context, ck_spinlock_fas_locked_with_context,
|
||||
ck_spinlock_fas_lock_with_context, ck_spinlock_fas_unlock_with_context, ck_spinlock_fas_locked_with_context)
|
||||
LOCK_PROTOTYPE(fas_fas)
|
||||
|
||||
struct cohort_record {
|
||||
CK_COHORT_INSTANCE(fas_fas) cohort;
|
||||
} CK_CC_CACHELINE;
|
||||
static struct cohort_record *cohorts;
|
||||
|
||||
static ck_spinlock_t global_lock = CK_SPINLOCK_INITIALIZER;
|
||||
static LOCK_INSTANCE(fas_fas) rw_cohort = LOCK_INITIALIZER;
|
||||
static unsigned int n_cohorts;
|
||||
|
||||
struct block {
|
||||
unsigned int tid;
|
||||
};
|
||||
|
||||
static void *
|
||||
thread_rwlock(void *pun)
|
||||
{
|
||||
uint64_t s_b, e_b, a, i;
|
||||
uint64_t *value = pun;
|
||||
CK_COHORT_INSTANCE(fas_fas) *cohort;
|
||||
unsigned int core;
|
||||
|
||||
if (aff_iterate_core(&affinity, &core) != 0) {
|
||||
perror("ERROR: Could not affine thread");
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
|
||||
cohort = &((cohorts + (core / (int)(affinity.delta)) % n_cohorts)->cohort);
|
||||
|
||||
ck_pr_inc_uint(&barrier);
|
||||
while (ck_pr_load_uint(&barrier) != nthr)
|
||||
ck_pr_stall();
|
||||
|
||||
for (i = 1, a = 0;; i++) {
|
||||
s_b = rdtsc();
|
||||
READ_LOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_UNLOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_LOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_UNLOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_LOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_UNLOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_LOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_UNLOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_LOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_UNLOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_LOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_UNLOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_LOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_UNLOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_LOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_UNLOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_LOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_UNLOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_LOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_UNLOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_LOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_UNLOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_LOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_UNLOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_LOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_UNLOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_LOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_UNLOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_LOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_UNLOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_LOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
READ_UNLOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
e_b = rdtsc();
|
||||
|
||||
a += (e_b - s_b) >> 4;
|
||||
|
||||
if (ck_pr_load_uint(&flag) == 1)
|
||||
break;
|
||||
}
|
||||
|
||||
ck_pr_inc_uint(&barrier);
|
||||
while (ck_pr_load_uint(&barrier) != nthr * 2)
|
||||
ck_pr_stall();
|
||||
|
||||
*value = (a / i);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
int
|
||||
main(int argc, char *argv[])
|
||||
{
|
||||
unsigned int i;
|
||||
pthread_t *threads;
|
||||
uint64_t *latency;
|
||||
struct block *context;
|
||||
ck_spinlock_fas_t *local_lock;
|
||||
|
||||
if (argc != 4) {
|
||||
ck_error("Usage: throughput <number of cohorts> <threads per cohort> <affinity delta>\n");
|
||||
}
|
||||
|
||||
n_cohorts = atoi(argv[1]);
|
||||
if (n_cohorts <= 0) {
|
||||
ck_error("ERROR: Number of cohorts must be greater than 0\n");
|
||||
}
|
||||
|
||||
nthr = n_cohorts * atoi(argv[2]);
|
||||
if (nthr <= 0) {
|
||||
ck_error("ERROR: Number of threads must be greater than 0\n");
|
||||
}
|
||||
|
||||
threads = malloc(sizeof(pthread_t) * nthr);
|
||||
if (threads == NULL) {
|
||||
ck_error("ERROR: Could not allocate thread structures\n");
|
||||
}
|
||||
|
||||
cohorts = malloc(sizeof(struct cohort_record) * n_cohorts);
|
||||
if (cohorts == NULL) {
|
||||
ck_error("ERROR: Could not allocate cohort structures\n");
|
||||
}
|
||||
|
||||
context = malloc(sizeof(struct block) * nthr);
|
||||
if (context == NULL) {
|
||||
ck_error("ERROR: Could not allocate thread contexts\n");
|
||||
}
|
||||
|
||||
affinity.delta = atoi(argv[3]);
|
||||
affinity.request = 0;
|
||||
|
||||
latency = malloc(sizeof(*latency) * nthr);
|
||||
if (latency == NULL) {
|
||||
ck_error("ERROR: Could not create latency buffer\n");
|
||||
}
|
||||
memset(latency, 0, sizeof(*latency) * nthr);
|
||||
|
||||
fprintf(stderr, "Creating cohorts...");
|
||||
for (i = 0 ; i < n_cohorts ; i++) {
|
||||
local_lock = malloc(max(CK_MD_CACHELINE, sizeof(ck_spinlock_fas_t)));
|
||||
if (local_lock == NULL) {
|
||||
ck_error("ERROR: Could not allocate local lock\n");
|
||||
}
|
||||
CK_COHORT_INIT(fas_fas, &((cohorts + i)->cohort), &global_lock, local_lock,
|
||||
CK_COHORT_DEFAULT_LOCAL_PASS_LIMIT);
|
||||
local_lock = NULL;
|
||||
}
|
||||
fprintf(stderr, "done\n");
|
||||
|
||||
fprintf(stderr, "Creating threads (rwlock)...");
|
||||
for (i = 0; i < nthr; i++) {
|
||||
if (pthread_create(&threads[i], NULL, thread_rwlock, latency + i) != 0) {
|
||||
ck_error("ERROR: Could not create thread %d\n", i);
|
||||
}
|
||||
}
|
||||
fprintf(stderr, "done\n");
|
||||
|
||||
common_sleep(10);
|
||||
ck_pr_store_uint(&flag, 1);
|
||||
|
||||
fprintf(stderr, "Waiting for threads to finish acquisition regression...");
|
||||
for (i = 0; i < nthr; i++)
|
||||
pthread_join(threads[i], NULL);
|
||||
fprintf(stderr, "done\n\n");
|
||||
|
||||
for (i = 1; i <= nthr; i++)
|
||||
printf("%10u %20" PRIu64 "\n", i, latency[i - 1]);
|
||||
|
||||
return (0);
|
||||
}
|
||||
|
@ -0,0 +1,8 @@
|
||||
#define LOCK_PROTOTYPE CK_RWCOHORT_NEUTRAL_PROTOTYPE
|
||||
#define LOCK_INSTANCE CK_RWCOHORT_NEUTRAL_INSTANCE
|
||||
#define LOCK_INITIALIZER CK_RWCOHORT_NEUTRAL_INITIALIZER
|
||||
#define LOCK_INIT(N, C, W) CK_RWCOHORT_NEUTRAL_INIT(N, C)
|
||||
#define READ_LOCK CK_RWCOHORT_NEUTRAL_READ_LOCK
|
||||
#define WRITE_LOCK CK_RWCOHORT_NEUTRAL_WRITE_LOCK
|
||||
#define READ_UNLOCK CK_RWCOHORT_NEUTRAL_READ_UNLOCK
|
||||
#define WRITE_UNLOCK CK_RWCOHORT_NEUTRAL_WRITE_UNLOCK
|
@ -0,0 +1,8 @@
|
||||
#define LOCK_PROTOTYPE CK_RWCOHORT_RP_PROTOTYPE
|
||||
#define LOCK_INSTANCE CK_RWCOHORT_RP_INSTANCE
|
||||
#define LOCK_INITIALIZER CK_RWCOHORT_RP_INITIALIZER
|
||||
#define LOCK_INIT CK_RWCOHORT_RP_INIT
|
||||
#define READ_LOCK CK_RWCOHORT_RP_READ_LOCK
|
||||
#define READ_UNLOCK CK_RWCOHORT_RP_READ_UNLOCK
|
||||
#define WRITE_LOCK CK_RWCOHORT_RP_WRITE_LOCK
|
||||
#define WRITE_UNLOCK CK_RWCOHORT_RP_WRITE_UNLOCK
|
@ -0,0 +1,8 @@
|
||||
#define LOCK_PROTOTYPE CK_RWCOHORT_WP_PROTOTYPE
|
||||
#define LOCK_INSTANCE CK_RWCOHORT_WP_INSTANCE
|
||||
#define LOCK_INITIALIZER CK_RWCOHORT_WP_INITIALIZER
|
||||
#define LOCK_INIT CK_RWCOHORT_WP_INIT
|
||||
#define READ_LOCK CK_RWCOHORT_WP_READ_LOCK
|
||||
#define WRITE_LOCK CK_RWCOHORT_WP_WRITE_LOCK
|
||||
#define READ_UNLOCK CK_RWCOHORT_WP_READ_UNLOCK
|
||||
#define WRITE_UNLOCK CK_RWCOHORT_WP_WRITE_UNLOCK
|
@ -0,0 +1,25 @@
|
||||
.PHONY: check clean distribution
|
||||
|
||||
OBJECTS=ck_neutral ck_rp ck_wp
|
||||
|
||||
all: $(OBJECTS)
|
||||
|
||||
ck_neutral: ck_neutral.c ../../../include/ck_rwcohort.h
|
||||
$(CC) $(CFLAGS) -o ck_neutral ck_neutral.c -g
|
||||
|
||||
ck_rp: ck_rp.c ../../../include/ck_rwcohort.h
|
||||
$(CC) $(CFLAGS) -o ck_rp ck_rp.c -g
|
||||
|
||||
ck_wp: ck_wp.c ../../../include/ck_rwcohort.h
|
||||
$(CC) $(CFLAGS) -o ck_wp ck_wp.c -g
|
||||
|
||||
check: all
|
||||
./ck_neutral `expr $(CORES) / 2` 2 1
|
||||
./ck_rp `expr $(CORES) / 2` 2 1
|
||||
./ck_wp `expr $(CORES) / 2` 2 1
|
||||
|
||||
clean:
|
||||
rm -rf *.dSYM *~ *.o $(OBJECTS)
|
||||
|
||||
include ../../../build/regressions.build
|
||||
CFLAGS+=$(PTHREAD_CFLAGS) -D_GNU_SOURCE
|
@ -0,0 +1,2 @@
|
||||
#include "../ck_neutral.h"
|
||||
#include "validate.h"
|
@ -0,0 +1,2 @@
|
||||
#include "../ck_rp.h"
|
||||
#include "validate.h"
|
@ -0,0 +1,2 @@
|
||||
#include "../ck_wp.h"
|
||||
#include "validate.h"
|
@ -0,0 +1,209 @@
|
||||
/*
|
||||
* Copyright 2013 Samy Al Bahra.
|
||||
* Copything 2013 Brendon Scheinman.
|
||||
* 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, this list of conditions and the following disclaimer.
|
||||
* 2. Redistributions in binary form must reproduce the above copyright
|
||||
* notice, 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 AUTHOR AND CONTRIBUTORS ``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 AUTHOR OR CONTRIBUTORS 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.
|
||||
*/
|
||||
|
||||
#include <errno.h>
|
||||
#include <inttypes.h>
|
||||
#include <pthread.h>
|
||||
#include <math.h>
|
||||
#include <stdint.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <strings.h>
|
||||
#include <unistd.h>
|
||||
#include <sys/time.h>
|
||||
|
||||
#include <ck_pr.h>
|
||||
#include <ck_rwcohort.h>
|
||||
#include <ck_spinlock.h>
|
||||
|
||||
#include "../../common.h"
|
||||
|
||||
#ifndef ITERATE
|
||||
#define ITERATE 1000000
|
||||
#endif
|
||||
|
||||
|
||||
static struct affinity a;
|
||||
static unsigned int locked;
|
||||
static int nthr;
|
||||
static ck_spinlock_fas_t global_fas_lock = CK_SPINLOCK_FAS_INITIALIZER;
|
||||
|
||||
static void
|
||||
ck_spinlock_fas_lock_with_context(ck_spinlock_fas_t *lock, void *context)
|
||||
{
|
||||
(void)context;
|
||||
ck_spinlock_fas_lock(lock);
|
||||
}
|
||||
|
||||
static void
|
||||
ck_spinlock_fas_unlock_with_context(ck_spinlock_fas_t *lock, void *context)
|
||||
{
|
||||
(void)context;
|
||||
ck_spinlock_fas_unlock(lock);
|
||||
}
|
||||
|
||||
static bool
|
||||
ck_spinlock_fas_locked_with_context(ck_spinlock_fas_t *lock, void *context)
|
||||
{
|
||||
(void)context;
|
||||
return ck_spinlock_fas_locked(lock);
|
||||
}
|
||||
|
||||
CK_COHORT_PROTOTYPE(fas_fas,
|
||||
ck_spinlock_fas_lock_with_context, ck_spinlock_fas_unlock_with_context, ck_spinlock_fas_locked_with_context,
|
||||
ck_spinlock_fas_lock_with_context, ck_spinlock_fas_unlock_with_context, ck_spinlock_fas_locked_with_context)
|
||||
LOCK_PROTOTYPE(fas_fas)
|
||||
|
||||
static CK_COHORT_INSTANCE(fas_fas) *cohorts;
|
||||
static LOCK_INSTANCE(fas_fas) rw_cohort = LOCK_INITIALIZER;
|
||||
static int n_cohorts;
|
||||
|
||||
static void *
|
||||
thread(void *null CK_CC_UNUSED)
|
||||
{
|
||||
int i = ITERATE;
|
||||
unsigned int l;
|
||||
unsigned int core;
|
||||
CK_COHORT_INSTANCE(fas_fas) *cohort;
|
||||
|
||||
if (aff_iterate_core(&a, &core)) {
|
||||
perror("ERROR: Could not affine thread");
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
|
||||
cohort = cohorts + (core / (int)(a.delta)) % n_cohorts;
|
||||
|
||||
while (i--) {
|
||||
WRITE_LOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
{
|
||||
l = ck_pr_load_uint(&locked);
|
||||
if (l != 0) {
|
||||
ck_error("ERROR [WR:%d]: %u != 0\n", __LINE__, l);
|
||||
}
|
||||
|
||||
ck_pr_inc_uint(&locked);
|
||||
ck_pr_inc_uint(&locked);
|
||||
ck_pr_inc_uint(&locked);
|
||||
ck_pr_inc_uint(&locked);
|
||||
ck_pr_inc_uint(&locked);
|
||||
ck_pr_inc_uint(&locked);
|
||||
ck_pr_inc_uint(&locked);
|
||||
ck_pr_inc_uint(&locked);
|
||||
|
||||
l = ck_pr_load_uint(&locked);
|
||||
if (l != 8) {
|
||||
ck_error("ERROR [WR:%d]: %u != 2\n", __LINE__, l);
|
||||
}
|
||||
|
||||
ck_pr_dec_uint(&locked);
|
||||
ck_pr_dec_uint(&locked);
|
||||
ck_pr_dec_uint(&locked);
|
||||
ck_pr_dec_uint(&locked);
|
||||
ck_pr_dec_uint(&locked);
|
||||
ck_pr_dec_uint(&locked);
|
||||
ck_pr_dec_uint(&locked);
|
||||
ck_pr_dec_uint(&locked);
|
||||
|
||||
l = ck_pr_load_uint(&locked);
|
||||
if (l != 0) {
|
||||
ck_error("ERROR [WR:%d]: %u != 0\n", __LINE__, l);
|
||||
}
|
||||
}
|
||||
WRITE_UNLOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
|
||||
READ_LOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
{
|
||||
l = ck_pr_load_uint(&locked);
|
||||
if (l != 0) {
|
||||
ck_error("ERROR [RD:%d]: %u != 0\n", __LINE__, l);
|
||||
}
|
||||
}
|
||||
READ_UNLOCK(fas_fas, &rw_cohort, cohort, NULL, NULL);
|
||||
}
|
||||
|
||||
return (NULL);
|
||||
}
|
||||
|
||||
int
|
||||
main(int argc, char *argv[])
|
||||
{
|
||||
pthread_t *threads;
|
||||
int threads_per_cohort;
|
||||
ck_spinlock_fas_t *local_lock;
|
||||
int i;
|
||||
|
||||
if (argc != 4) {
|
||||
ck_error("Usage: validate <number of cohorts> <threads per cohort> <affinity delta>\n");
|
||||
}
|
||||
|
||||
n_cohorts = atoi(argv[1]);
|
||||
if (n_cohorts <= 0) {
|
||||
ck_error("ERROR: Number of cohorts must be greater than 0\n");
|
||||
}
|
||||
|
||||
threads_per_cohort = atoi(argv[2]);
|
||||
if (threads_per_cohort <= 0) {
|
||||
ck_error("ERROR: Threads per cohort must be greater than 0\n");
|
||||
}
|
||||
|
||||
nthr = n_cohorts * threads_per_cohort;
|
||||
|
||||
threads = malloc(sizeof(pthread_t) * nthr);
|
||||
if (threads == NULL) {
|
||||
ck_error("ERROR: Could not allocate thread structures\n");
|
||||
}
|
||||
|
||||
a.delta = atoi(argv[3]);
|
||||
|
||||
fprintf(stderr, "Creating cohorts...");
|
||||
cohorts = malloc(sizeof(CK_COHORT_INSTANCE(fas_fas)) * n_cohorts);
|
||||
if (cohorts == NULL) {
|
||||
ck_error("ERROR: Could not allocate base cohort structures\n");
|
||||
}
|
||||
for (i = 0 ; i < n_cohorts ; i++) {
|
||||
local_lock = malloc(sizeof(ck_spinlock_fas_t));
|
||||
CK_COHORT_INIT(fas_fas, cohorts + i, &global_fas_lock, local_lock,
|
||||
CK_COHORT_DEFAULT_LOCAL_PASS_LIMIT);
|
||||
}
|
||||
fprintf(stderr, "done\n");
|
||||
|
||||
fprintf(stderr, "Creating threads...");
|
||||
for (i = 0; i < nthr; i++) {
|
||||
if (pthread_create(&threads[i], NULL, thread, NULL)) {
|
||||
ck_error("ERROR: Could not create thread %d\n", i);
|
||||
}
|
||||
}
|
||||
fprintf(stderr, "done\n");
|
||||
|
||||
fprintf(stderr, "Waiting for threads to finish correctness regression...");
|
||||
for (i = 0; i < nthr; i++)
|
||||
pthread_join(threads[i], NULL);
|
||||
fprintf(stderr, "done (passed)\n");
|
||||
|
||||
return (0);
|
||||
}
|
||||
|
Loading…
Reference in new issue