2012-11-10 08:51:26 +00:00
|
|
|
/* elision-conf.c: Lock elision tunable parameters.
|
2020-01-01 00:14:33 +00:00
|
|
|
Copyright (C) 2013-2020 Free Software Foundation, Inc.
|
2012-11-10 08:51:26 +00:00
|
|
|
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
|
Prefer https to http for gnu.org and fsf.org URLs
Also, change sources.redhat.com to sourceware.org.
This patch was automatically generated by running the following shell
script, which uses GNU sed, and which avoids modifying files imported
from upstream:
sed -ri '
s,(http|ftp)(://(.*\.)?(gnu|fsf|sourceware)\.org($|[^.]|\.[^a-z])),https\2,g
s,(http|ftp)(://(.*\.)?)sources\.redhat\.com($|[^.]|\.[^a-z]),https\2sourceware.org\4,g
' \
$(find $(git ls-files) -prune -type f \
! -name '*.po' \
! -name 'ChangeLog*' \
! -path COPYING ! -path COPYING.LIB \
! -path manual/fdl-1.3.texi ! -path manual/lgpl-2.1.texi \
! -path manual/texinfo.tex ! -path scripts/config.guess \
! -path scripts/config.sub ! -path scripts/install-sh \
! -path scripts/mkinstalldirs ! -path scripts/move-if-change \
! -path INSTALL ! -path locale/programs/charmap-kw.h \
! -path po/libc.pot ! -path sysdeps/gnu/errlist.c \
! '(' -name configure \
-execdir test -f configure.ac -o -f configure.in ';' ')' \
! '(' -name preconfigure \
-execdir test -f preconfigure.ac ';' ')' \
-print)
and then by running 'make dist-prepare' to regenerate files built
from the altered files, and then executing the following to cleanup:
chmod a+x sysdeps/unix/sysv/linux/riscv/configure
# Omit irrelevant whitespace and comment-only changes,
# perhaps from a slightly-different Autoconf version.
git checkout -f \
sysdeps/csky/configure \
sysdeps/hppa/configure \
sysdeps/riscv/configure \
sysdeps/unix/sysv/linux/csky/configure
# Omit changes that caused a pre-commit check to fail like this:
# remote: *** error: sysdeps/powerpc/powerpc64/ppc-mcount.S: trailing lines
git checkout -f \
sysdeps/powerpc/powerpc64/ppc-mcount.S \
sysdeps/unix/sysv/linux/s390/s390-64/syscall.S
# Omit change that caused a pre-commit check to fail like this:
# remote: *** error: sysdeps/sparc/sparc64/multiarch/memcpy-ultra3.S: last line does not end in newline
git checkout -f sysdeps/sparc/sparc64/multiarch/memcpy-ultra3.S
2019-09-07 05:40:42 +00:00
|
|
|
<https://www.gnu.org/licenses/>. */
|
2012-11-10 08:51:26 +00:00
|
|
|
|
2013-05-17 02:17:14 +00:00
|
|
|
#include "config.h"
|
2012-11-10 08:51:26 +00:00
|
|
|
#include <pthreadP.h>
|
|
|
|
#include <init-arch.h>
|
|
|
|
#include <elision-conf.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
2017-12-05 16:24:14 +00:00
|
|
|
#if HAVE_TUNABLES
|
|
|
|
# define TUNABLE_NAMESPACE elision
|
|
|
|
#endif
|
|
|
|
#include <elf/dl-tunables.h>
|
|
|
|
|
2012-11-10 08:51:26 +00:00
|
|
|
/* Reasonable initial tuning values, may be revised in the future.
|
|
|
|
This is a conservative initial value. */
|
|
|
|
|
|
|
|
struct elision_config __elision_aconf =
|
|
|
|
{
|
|
|
|
/* How often to not attempt to use elision if a transaction aborted
|
|
|
|
because the lock is already acquired. Expressed in number of lock
|
|
|
|
acquisition attempts. */
|
|
|
|
.skip_lock_busy = 3,
|
|
|
|
/* How often to not attempt to use elision if a transaction aborted due
|
2013-07-19 05:16:28 +00:00
|
|
|
to reasons other than other threads' memory accesses. Expressed in
|
2012-11-10 08:51:26 +00:00
|
|
|
number of lock acquisition attempts. */
|
|
|
|
.skip_lock_internal_abort = 3,
|
|
|
|
/* How often we retry using elision if there is chance for the transaction
|
|
|
|
to finish execution (e.g., it wasn't aborted due to the lock being
|
|
|
|
already acquired. */
|
|
|
|
.retry_try_xbegin = 3,
|
|
|
|
/* Same as SKIP_LOCK_INTERNAL_ABORT but for trylock. */
|
|
|
|
.skip_trylock_internal_abort = 3,
|
|
|
|
};
|
|
|
|
|
2013-07-19 05:16:28 +00:00
|
|
|
/* Force elision for all new locks. This is used to decide whether existing
|
2012-11-10 08:51:26 +00:00
|
|
|
DEFAULT locks should be automatically upgraded to elision in
|
2013-07-19 05:16:28 +00:00
|
|
|
pthread_mutex_lock(). Disabled for suid programs. Only used when elision
|
2012-11-10 08:51:26 +00:00
|
|
|
is available. */
|
|
|
|
|
2017-12-05 16:24:14 +00:00
|
|
|
int __pthread_force_elision attribute_hidden = 0;
|
|
|
|
|
|
|
|
#if HAVE_TUNABLES
|
2019-02-06 17:16:43 +00:00
|
|
|
static __always_inline void
|
2017-12-05 16:24:14 +00:00
|
|
|
do_set_elision_enable (int32_t elision_enable)
|
|
|
|
{
|
|
|
|
/* Enable elision if it's avaliable in hardware. It's not necessary to check
|
|
|
|
if __libc_enable_secure isn't enabled since elision_enable will be set
|
|
|
|
according to the default, which is disabled. */
|
|
|
|
if (elision_enable == 1)
|
2020-06-29 23:36:08 +00:00
|
|
|
__pthread_force_elision = CPU_FEATURE_USABLE (RTM) ? 1 : 0;
|
2017-12-05 16:24:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* The pthread->elision_enable tunable is 0 or 1 indicating that elision
|
|
|
|
should be disabled or enabled respectively. The feature will only be used
|
|
|
|
if it's supported by the hardware. */
|
2012-11-10 08:51:26 +00:00
|
|
|
|
2017-12-05 16:24:14 +00:00
|
|
|
void
|
|
|
|
TUNABLE_CALLBACK (set_elision_enable) (tunable_val_t *valp)
|
|
|
|
{
|
|
|
|
int32_t elision_enable = (int32_t) valp->numval;
|
|
|
|
do_set_elision_enable (elision_enable);
|
|
|
|
}
|
|
|
|
|
|
|
|
#define TUNABLE_CALLBACK_FNDECL(__name, __type) \
|
2019-02-06 17:16:43 +00:00
|
|
|
static __always_inline void \
|
2017-12-05 16:24:14 +00:00
|
|
|
do_set_elision_ ## __name (__type value) \
|
|
|
|
{ \
|
|
|
|
__elision_aconf.__name = value; \
|
|
|
|
} \
|
|
|
|
void \
|
|
|
|
TUNABLE_CALLBACK (set_elision_ ## __name) (tunable_val_t *valp) \
|
|
|
|
{ \
|
|
|
|
__type value = (__type) (valp)->numval; \
|
|
|
|
do_set_elision_ ## __name (value); \
|
|
|
|
}
|
|
|
|
|
|
|
|
TUNABLE_CALLBACK_FNDECL (skip_lock_busy, int32_t);
|
|
|
|
TUNABLE_CALLBACK_FNDECL (skip_lock_internal_abort, int32_t);
|
|
|
|
TUNABLE_CALLBACK_FNDECL (retry_try_xbegin, int32_t);
|
|
|
|
TUNABLE_CALLBACK_FNDECL (skip_trylock_internal_abort, int32_t);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Initialize elision. */
|
2012-11-10 08:51:26 +00:00
|
|
|
|
|
|
|
static void
|
|
|
|
elision_init (int argc __attribute__ ((unused)),
|
|
|
|
char **argv __attribute__ ((unused)),
|
|
|
|
char **environ)
|
|
|
|
{
|
2017-12-05 16:24:14 +00:00
|
|
|
#if HAVE_TUNABLES
|
|
|
|
/* Elision depends on tunables and must be explicitly turned on by setting
|
|
|
|
the appropriate tunable on a supported platform. */
|
|
|
|
|
|
|
|
TUNABLE_GET (enable, int32_t,
|
|
|
|
TUNABLE_CALLBACK (set_elision_enable));
|
|
|
|
TUNABLE_GET (skip_lock_busy, int32_t,
|
|
|
|
TUNABLE_CALLBACK (set_elision_skip_lock_busy));
|
|
|
|
TUNABLE_GET (skip_lock_internal_abort, int32_t,
|
|
|
|
TUNABLE_CALLBACK (set_elision_skip_lock_internal_abort));
|
|
|
|
TUNABLE_GET (tries, int32_t,
|
|
|
|
TUNABLE_CALLBACK (set_elision_retry_try_xbegin));
|
|
|
|
TUNABLE_GET (skip_trylock_internal_abort, int32_t,
|
|
|
|
TUNABLE_CALLBACK (set_elision_skip_trylock_internal_abort));
|
2013-05-17 02:17:14 +00:00
|
|
|
#endif
|
2017-12-05 16:24:14 +00:00
|
|
|
|
|
|
|
if (!__pthread_force_elision)
|
|
|
|
__elision_aconf.retry_try_xbegin = 0; /* Disable elision on rwlocks. */
|
2012-11-10 08:51:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef SHARED
|
|
|
|
# define INIT_SECTION ".init_array"
|
|
|
|
#else
|
|
|
|
# define INIT_SECTION ".preinit_array"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
void (*const __pthread_init_array []) (int, char **, char **)
|
|
|
|
__attribute__ ((section (INIT_SECTION), aligned (sizeof (void *)))) =
|
|
|
|
{
|
|
|
|
&elision_init
|
|
|
|
};
|