mirror of
https://sourceware.org/git/glibc.git
synced 2024-11-30 21:23:52 +08:00
cc25c8b4c1
This replaces the pthread rwlock with a new implementation that uses a more scalable algorithm (primarily through not using a critical section anymore to make state changes). The fast path for rdlock acquisition and release is now basically a single atomic read-modify write or CAS and a few branches. See nptl/pthread_rwlock_common.c for details. * nptl/DESIGN-rwlock.txt: Remove. * nptl/lowlevelrwlock.sym: Remove. * nptl/Makefile: Add new tests. * nptl/pthread_rwlock_common.c: New file. Contains the new rwlock. * nptl/pthreadP.h (PTHREAD_RWLOCK_PREFER_READER_P): Remove. (PTHREAD_RWLOCK_WRPHASE, PTHREAD_RWLOCK_WRLOCKED, PTHREAD_RWLOCK_RWAITING, PTHREAD_RWLOCK_READER_SHIFT, PTHREAD_RWLOCK_READER_OVERFLOW, PTHREAD_RWLOCK_WRHANDOVER, PTHREAD_RWLOCK_FUTEX_USED): New. * nptl/pthread_rwlock_init.c (__pthread_rwlock_init): Adapt to new implementation. * nptl/pthread_rwlock_rdlock.c (__pthread_rwlock_rdlock_slow): Remove. (__pthread_rwlock_rdlock): Adapt. * nptl/pthread_rwlock_timedrdlock.c (pthread_rwlock_timedrdlock): Adapt. * nptl/pthread_rwlock_timedwrlock.c (pthread_rwlock_timedwrlock): Adapt. * nptl/pthread_rwlock_trywrlock.c (pthread_rwlock_trywrlock): Adapt. * nptl/pthread_rwlock_tryrdlock.c (pthread_rwlock_tryrdlock): Adapt. * nptl/pthread_rwlock_unlock.c (pthread_rwlock_unlock): Adapt. * nptl/pthread_rwlock_wrlock.c (__pthread_rwlock_wrlock_slow): Remove. (__pthread_rwlock_wrlock): Adapt. * nptl/tst-rwlock10.c: Adapt. * nptl/tst-rwlock11.c: Adapt. * nptl/tst-rwlock17.c: New file. * nptl/tst-rwlock18.c: New file. * nptl/tst-rwlock19.c: New file. * nptl/tst-rwlock2b.c: New file. * nptl/tst-rwlock8.c: Adapt. * nptl/tst-rwlock9.c: Adapt. * sysdeps/aarch64/nptl/bits/pthreadtypes.h (pthread_rwlock_t): Adapt. * sysdeps/arm/nptl/bits/pthreadtypes.h (pthread_rwlock_t): Adapt. * sysdeps/hppa/nptl/bits/pthreadtypes.h (pthread_rwlock_t): Adapt. * sysdeps/ia64/nptl/bits/pthreadtypes.h (pthread_rwlock_t): Adapt. * sysdeps/m68k/nptl/bits/pthreadtypes.h (pthread_rwlock_t): Adapt. * sysdeps/microblaze/nptl/bits/pthreadtypes.h (pthread_rwlock_t): Adapt. * sysdeps/mips/nptl/bits/pthreadtypes.h (pthread_rwlock_t): Adapt. * sysdeps/nios2/nptl/bits/pthreadtypes.h (pthread_rwlock_t): Adapt. * sysdeps/s390/nptl/bits/pthreadtypes.h (pthread_rwlock_t): Adapt. * sysdeps/sh/nptl/bits/pthreadtypes.h (pthread_rwlock_t): Adapt. * sysdeps/sparc/nptl/bits/pthreadtypes.h (pthread_rwlock_t): Adapt. * sysdeps/tile/nptl/bits/pthreadtypes.h (pthread_rwlock_t): Adapt. * sysdeps/unix/sysv/linux/alpha/bits/pthreadtypes.h (pthread_rwlock_t): Adapt. * sysdeps/unix/sysv/linux/powerpc/bits/pthreadtypes.h (pthread_rwlock_t): Adapt. * sysdeps/x86/bits/pthreadtypes.h (pthread_rwlock_t): Adapt. * nptl/nptl-printers.py (): Adapt. * nptl/nptl_lock_constants.pysym: Adapt. * nptl/test-rwlock-printers.py: Adapt. * nptl/test-rwlockattr-printers.c: Adapt. * nptl/test-rwlockattr-printers.py: Adapt.
101 lines
3.2 KiB
C
101 lines
3.2 KiB
C
/* Helper program for testing the pthread_rwlock_t and pthread_rwlockattr_t
|
|
pretty printers.
|
|
|
|
Copyright (C) 2016-2017 Free Software Foundation, Inc.
|
|
This file is part of the GNU C Library.
|
|
|
|
The GNU C Library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Lesser General Public
|
|
License as published by the Free Software Foundation; either
|
|
version 2.1 of the License, or (at your option) any later version.
|
|
|
|
The GNU C Library is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
Lesser General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public
|
|
License along with the GNU C Library; if not, see
|
|
<http://www.gnu.org/licenses/>. */
|
|
|
|
/* Keep the calls to the pthread_* functions on separate lines to make it easy
|
|
to advance through the program using the gdb 'next' command. */
|
|
|
|
#include <pthread.h>
|
|
|
|
#define PASS 0
|
|
#define FAIL 1
|
|
|
|
/* Need these so we don't have lines longer than 79 chars. */
|
|
#define SET_KIND(attr, kind) pthread_rwlockattr_setkind_np (attr, kind)
|
|
#define SET_SHARED(attr, shared) pthread_rwlockattr_setpshared (attr, shared)
|
|
|
|
static int rwlock_reinit (pthread_rwlock_t *rwlock,
|
|
const pthread_rwlockattr_t *attr);
|
|
static int test_setkind_np (pthread_rwlock_t *rwlock,
|
|
pthread_rwlockattr_t *attr);
|
|
static int test_setpshared (pthread_rwlock_t *rwlock,
|
|
pthread_rwlockattr_t *attr);
|
|
|
|
int
|
|
main (void)
|
|
{
|
|
pthread_rwlock_t rwlock;
|
|
pthread_rwlockattr_t attr;
|
|
int result = FAIL;
|
|
|
|
if (pthread_rwlockattr_init (&attr) == 0
|
|
&& pthread_rwlock_init (&rwlock, NULL) == 0
|
|
&& test_setkind_np (&rwlock, &attr) == PASS
|
|
&& test_setpshared (&rwlock, &attr) == PASS)
|
|
result = PASS;
|
|
/* Else, one of the pthread_rwlock* functions failed. */
|
|
|
|
return result;
|
|
}
|
|
|
|
/* Destroys RWLOCK and re-initializes it using ATTR. */
|
|
static int
|
|
rwlock_reinit (pthread_rwlock_t *rwlock, const pthread_rwlockattr_t *attr)
|
|
{
|
|
int result = FAIL;
|
|
|
|
if (pthread_rwlock_destroy (rwlock) == 0
|
|
&& pthread_rwlock_init (rwlock, attr) == 0)
|
|
result = PASS;
|
|
|
|
return result;
|
|
}
|
|
|
|
/* Tests setting whether the rwlock prefers readers or writers. */
|
|
static int
|
|
test_setkind_np (pthread_rwlock_t *rwlock, pthread_rwlockattr_t *attr)
|
|
{
|
|
int result = FAIL;
|
|
|
|
if (SET_KIND (attr, PTHREAD_RWLOCK_PREFER_READER_NP) == 0 /* Set kind. */
|
|
&& rwlock_reinit (rwlock, attr) == PASS
|
|
&& SET_KIND (attr, PTHREAD_RWLOCK_PREFER_WRITER_NP) == 0
|
|
&& rwlock_reinit (rwlock, attr) == PASS
|
|
&& SET_KIND (attr, PTHREAD_RWLOCK_PREFER_WRITER_NONRECURSIVE_NP) == 0
|
|
&& rwlock_reinit (rwlock, attr) == PASS)
|
|
result = PASS;
|
|
|
|
return result;
|
|
}
|
|
|
|
/* Tests setting whether the rwlock can be shared between processes. */
|
|
static int
|
|
test_setpshared (pthread_rwlock_t *rwlock, pthread_rwlockattr_t *attr)
|
|
{
|
|
int result = FAIL;
|
|
|
|
if (SET_SHARED (attr, PTHREAD_PROCESS_SHARED) == 0 /* Set shared. */
|
|
&& rwlock_reinit (rwlock, attr) == PASS
|
|
&& SET_SHARED (attr, PTHREAD_PROCESS_PRIVATE) == 0
|
|
&& rwlock_reinit (rwlock, attr) == PASS)
|
|
result = PASS;
|
|
|
|
return result;
|
|
}
|