2018-07-25 18:20:44 +00:00
|
|
|
/* ISO C11 Standard: 7.26 - Thread support library <threads.h>.
|
2024-01-01 18:12:26 +00:00
|
|
|
Copyright (C) 2018-2024 Free Software Foundation, Inc.
|
2018-07-25 18:20:44 +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/>. */
|
2018-07-25 18:20:44 +00:00
|
|
|
|
|
|
|
#ifndef _THREADS_H
|
|
|
|
#define _THREADS_H 1
|
|
|
|
|
|
|
|
#include <features.h>
|
|
|
|
#include <time.h>
|
|
|
|
|
|
|
|
__BEGIN_DECLS
|
|
|
|
|
2020-01-13 21:15:46 +00:00
|
|
|
#include <bits/thread-shared-types.h>
|
2018-07-25 18:20:44 +00:00
|
|
|
#include <bits/types/struct_timespec.h>
|
|
|
|
|
Do not define static_assert or thread_local in headers for C2x
C2x makes static_assert and thread_local into keywords, removing the
definitions as macros in assert.h and threads.h. Thus, disable those
macros in those glibc headers for C2x.
The disabling is done based on a combination of language version and
__GNUC_PREREQ, *not* based on __GLIBC_USE (ISOC2X), on the principle
that users of the header (when requesting C11 or later APIs - not
assert.h for C99 and older API versions) should always have the names
static_assert or thread_local available after inclusion of the header,
whether as a keyword or as a macro. Thus, when using a compiler
without the keywords (whether an older compiler, possibly in C2x mode,
or _GNU_SOURCE with any compiler but in an older language mode, for
example) the macros should be defined, even when C2x APIs have been
requested. The __GNUC_PREREQ conditionals here may well need updating
with the versions of other compilers that gained support for these
keywords in C2x mode.
Tested for x86_64.
2022-09-07 18:39:28 +00:00
|
|
|
#if (!defined __STDC_VERSION__ \
|
|
|
|
|| __STDC_VERSION__ <= 201710L \
|
|
|
|
|| !__GNUC_PREREQ (13, 0)) && !defined __cplusplus
|
2018-07-25 18:20:44 +00:00
|
|
|
# define thread_local _Thread_local
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define TSS_DTOR_ITERATIONS 4
|
2020-01-13 21:15:46 +00:00
|
|
|
typedef __tss_t tss_t;
|
2018-07-25 18:20:44 +00:00
|
|
|
typedef void (*tss_dtor_t) (void*);
|
|
|
|
|
2020-01-13 21:15:46 +00:00
|
|
|
typedef __thrd_t thrd_t;
|
2018-07-25 18:20:44 +00:00
|
|
|
typedef int (*thrd_start_t) (void*);
|
|
|
|
|
|
|
|
/* Exit and error codes. */
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
thrd_success = 0,
|
|
|
|
thrd_busy = 1,
|
|
|
|
thrd_error = 2,
|
|
|
|
thrd_nomem = 3,
|
|
|
|
thrd_timedout = 4
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Mutex types. */
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
mtx_plain = 0,
|
|
|
|
mtx_recursive = 1,
|
|
|
|
mtx_timed = 2
|
|
|
|
};
|
|
|
|
|
2020-01-13 21:15:46 +00:00
|
|
|
typedef __once_flag once_flag;
|
|
|
|
#define ONCE_FLAG_INIT __ONCE_FLAG_INIT
|
2018-07-25 18:20:44 +00:00
|
|
|
|
|
|
|
typedef union
|
|
|
|
{
|
|
|
|
char __size[__SIZEOF_PTHREAD_MUTEX_T];
|
|
|
|
long int __align __LOCK_ALIGNMENT;
|
|
|
|
} mtx_t;
|
|
|
|
|
|
|
|
typedef union
|
|
|
|
{
|
|
|
|
char __size[__SIZEOF_PTHREAD_COND_T];
|
|
|
|
__extension__ long long int __align __LOCK_ALIGNMENT;
|
|
|
|
} cnd_t;
|
|
|
|
|
|
|
|
/* Threads functions. */
|
|
|
|
|
|
|
|
/* Create a new thread executing the function __FUNC. Arguments for __FUNC
|
2023-05-30 23:02:29 +00:00
|
|
|
are passed through __ARG. If successful, __THR is set to new thread
|
2018-07-25 18:20:44 +00:00
|
|
|
identifier. */
|
|
|
|
extern int thrd_create (thrd_t *__thr, thrd_start_t __func, void *__arg);
|
|
|
|
|
|
|
|
/* Check if __LHS and __RHS point to the same thread. */
|
|
|
|
extern int thrd_equal (thrd_t __lhs, thrd_t __rhs);
|
|
|
|
|
|
|
|
/* Return current thread identifier. */
|
|
|
|
extern thrd_t thrd_current (void);
|
|
|
|
|
|
|
|
/* Block current thread execution for at least the time pointed by
|
|
|
|
__TIME_POINT. The current thread may resume if receives a signal. In
|
|
|
|
that case, if __REMAINING is not NULL, the remaining time is stored in
|
|
|
|
the object pointed by it. */
|
2024-01-18 13:18:01 +00:00
|
|
|
#ifndef __USE_TIME64_REDIRECTS
|
2018-07-25 18:20:44 +00:00
|
|
|
extern int thrd_sleep (const struct timespec *__time_point,
|
|
|
|
struct timespec *__remaining);
|
y2038: Add support for 64-bit time on legacy ABIs
A new build flag, _TIME_BITS, enables the usage of the newer 64-bit
time symbols for legacy ABI (where 32-bit time_t is default). The 64
bit time support is only enabled if LFS (_FILE_OFFSET_BITS=64) is
also used.
Different than LFS support, the y2038 symbols are added only for the
required ABIs (armhf, csky, hppa, i386, m68k, microblaze, mips32,
mips64-n32, nios2, powerpc32, sparc32, s390-32, and sh). The ABIs with
64-bit time support are unchanged, both for symbol and types
redirection.
On Linux the full 64-bit time support requires a minimum of kernel
version v5.1. Otherwise, the 32-bit fallbacks are used and might
results in error with overflow return code (EOVERFLOW).
The i686-gnu does not yet support 64-bit time.
This patch exports following rediretions to support 64-bit time:
* libc:
adjtime
adjtimex
clock_adjtime
clock_getres
clock_gettime
clock_nanosleep
clock_settime
cnd_timedwait
ctime
ctime_r
difftime
fstat
fstatat
futimens
futimes
futimesat
getitimer
getrusage
gettimeofday
gmtime
gmtime_r
localtime
localtime_r
lstat_time
lutimes
mktime
msgctl
mtx_timedlock
nanosleep
nanosleep
ntp_gettime
ntp_gettimex
ppoll
pselec
pselect
pthread_clockjoin_np
pthread_cond_clockwait
pthread_cond_timedwait
pthread_mutex_clocklock
pthread_mutex_timedlock
pthread_rwlock_clockrdlock
pthread_rwlock_clockwrlock
pthread_rwlock_timedrdlock
pthread_rwlock_timedwrlock
pthread_timedjoin_np
recvmmsg
sched_rr_get_interval
select
sem_clockwait
semctl
semtimedop
sem_timedwait
setitimer
settimeofday
shmctl
sigtimedwait
stat
thrd_sleep
time
timegm
timerfd_gettime
timerfd_settime
timespec_get
utime
utimensat
utimes
utimes
wait3
wait4
* librt:
aio_suspend
mq_timedreceive
mq_timedsend
timer_gettime
timer_settime
* libanl:
gai_suspend
Reviewed-by: Lukasz Majewski <lukma@denx.de>
Reviewed-by: Carlos O'Donell <carlos@redhat.com>
Tested-by: Carlos O'Donell <carlos@redhat.com>
2021-02-17 16:35:19 +00:00
|
|
|
#else
|
|
|
|
# ifdef __REDIRECT
|
|
|
|
extern int __REDIRECT (thrd_sleep, (const struct timespec *__time_point,
|
|
|
|
struct timespec *__remaining),
|
|
|
|
__thrd_sleep64);
|
|
|
|
# else
|
|
|
|
# define thrd_sleep __thrd_sleep64
|
|
|
|
# endif
|
|
|
|
#endif
|
2018-07-25 18:20:44 +00:00
|
|
|
|
|
|
|
/* Terminate current thread execution, cleaning up any thread local
|
|
|
|
storage and freeing resources. Returns the value specified in __RES. */
|
|
|
|
extern void thrd_exit (int __res) __attribute__ ((__noreturn__));
|
|
|
|
|
|
|
|
/* Detach the thread identified by __THR from the current environment
|
|
|
|
(it does not allow join or wait for it). */
|
|
|
|
extern int thrd_detach (thrd_t __thr);
|
|
|
|
|
|
|
|
/* Block current thread until execution of __THR is complete. In case that
|
|
|
|
__RES is not NULL, will store the return value of __THR when exiting. */
|
|
|
|
extern int thrd_join (thrd_t __thr, int *__res);
|
|
|
|
|
|
|
|
/* Stop current thread execution and call the scheduler to decide which
|
|
|
|
thread should execute next. The current thread may be selected by the
|
|
|
|
scheduler to keep running. */
|
|
|
|
extern void thrd_yield (void);
|
|
|
|
|
|
|
|
#ifdef __USE_EXTERN_INLINES
|
|
|
|
/* Optimizations. */
|
|
|
|
__extern_inline int
|
|
|
|
thrd_equal (thrd_t __thread1, thrd_t __thread2)
|
|
|
|
{
|
|
|
|
return __thread1 == __thread2;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
/* Mutex functions. */
|
|
|
|
|
|
|
|
/* Creates a new mutex object with type __TYPE. If successful the new
|
|
|
|
object is pointed by __MUTEX. */
|
|
|
|
extern int mtx_init (mtx_t *__mutex, int __type);
|
|
|
|
|
|
|
|
/* Block the current thread until the mutex pointed to by __MUTEX is
|
|
|
|
unlocked. In that case current thread will not be blocked. */
|
|
|
|
extern int mtx_lock (mtx_t *__mutex);
|
|
|
|
|
|
|
|
/* Block the current thread until the mutex pointed by __MUTEX is unlocked
|
|
|
|
or time pointed by __TIME_POINT is reached. In case the mutex is unlock,
|
|
|
|
the current thread will not be blocked. */
|
2024-01-18 13:18:01 +00:00
|
|
|
#ifndef __USE_TIME64_REDIRECTS
|
2018-07-25 18:20:44 +00:00
|
|
|
extern int mtx_timedlock (mtx_t *__restrict __mutex,
|
|
|
|
const struct timespec *__restrict __time_point);
|
y2038: Add support for 64-bit time on legacy ABIs
A new build flag, _TIME_BITS, enables the usage of the newer 64-bit
time symbols for legacy ABI (where 32-bit time_t is default). The 64
bit time support is only enabled if LFS (_FILE_OFFSET_BITS=64) is
also used.
Different than LFS support, the y2038 symbols are added only for the
required ABIs (armhf, csky, hppa, i386, m68k, microblaze, mips32,
mips64-n32, nios2, powerpc32, sparc32, s390-32, and sh). The ABIs with
64-bit time support are unchanged, both for symbol and types
redirection.
On Linux the full 64-bit time support requires a minimum of kernel
version v5.1. Otherwise, the 32-bit fallbacks are used and might
results in error with overflow return code (EOVERFLOW).
The i686-gnu does not yet support 64-bit time.
This patch exports following rediretions to support 64-bit time:
* libc:
adjtime
adjtimex
clock_adjtime
clock_getres
clock_gettime
clock_nanosleep
clock_settime
cnd_timedwait
ctime
ctime_r
difftime
fstat
fstatat
futimens
futimes
futimesat
getitimer
getrusage
gettimeofday
gmtime
gmtime_r
localtime
localtime_r
lstat_time
lutimes
mktime
msgctl
mtx_timedlock
nanosleep
nanosleep
ntp_gettime
ntp_gettimex
ppoll
pselec
pselect
pthread_clockjoin_np
pthread_cond_clockwait
pthread_cond_timedwait
pthread_mutex_clocklock
pthread_mutex_timedlock
pthread_rwlock_clockrdlock
pthread_rwlock_clockwrlock
pthread_rwlock_timedrdlock
pthread_rwlock_timedwrlock
pthread_timedjoin_np
recvmmsg
sched_rr_get_interval
select
sem_clockwait
semctl
semtimedop
sem_timedwait
setitimer
settimeofday
shmctl
sigtimedwait
stat
thrd_sleep
time
timegm
timerfd_gettime
timerfd_settime
timespec_get
utime
utimensat
utimes
utimes
wait3
wait4
* librt:
aio_suspend
mq_timedreceive
mq_timedsend
timer_gettime
timer_settime
* libanl:
gai_suspend
Reviewed-by: Lukasz Majewski <lukma@denx.de>
Reviewed-by: Carlos O'Donell <carlos@redhat.com>
Tested-by: Carlos O'Donell <carlos@redhat.com>
2021-02-17 16:35:19 +00:00
|
|
|
#else
|
|
|
|
# ifdef __REDIRECT
|
|
|
|
extern int __REDIRECT (mtx_timedlock, (mtx_t *__restrict __mutex,
|
|
|
|
const struct timespec *__restrict
|
|
|
|
__time_point),
|
|
|
|
__mtx_timedlock64);
|
|
|
|
# else
|
|
|
|
# define mtx_timedlock __mtx_timedlock64
|
|
|
|
# endif
|
|
|
|
#endif
|
2018-07-25 18:20:44 +00:00
|
|
|
|
|
|
|
/* Try to lock the mutex pointed by __MUTEX without blocking. If the mutex
|
|
|
|
is free the current threads takes control of it, otherwise it returns
|
|
|
|
immediately. */
|
|
|
|
extern int mtx_trylock (mtx_t *__mutex);
|
|
|
|
|
|
|
|
/* Unlock the mutex pointed by __MUTEX. It may potentially awake other
|
|
|
|
threads waiting on this mutex. */
|
|
|
|
extern int mtx_unlock (mtx_t *__mutex);
|
|
|
|
|
|
|
|
/* Destroy the mutex object pointed by __MUTEX. */
|
|
|
|
extern void mtx_destroy (mtx_t *__mutex);
|
|
|
|
|
|
|
|
|
|
|
|
/* Call function __FUNC exactly once, even if invoked from several threads.
|
|
|
|
All calls must be made with the same __FLAGS object. */
|
|
|
|
extern void call_once (once_flag *__flag, void (*__func)(void));
|
|
|
|
|
|
|
|
|
|
|
|
/* Condition variable functions. */
|
|
|
|
|
|
|
|
/* Initialize new condition variable pointed by __COND. */
|
|
|
|
extern int cnd_init (cnd_t *__cond);
|
|
|
|
|
|
|
|
/* Unblock one thread that currently waits on condition variable pointed
|
|
|
|
by __COND. */
|
|
|
|
extern int cnd_signal (cnd_t *__cond);
|
|
|
|
|
|
|
|
/* Unblock all threads currently waiting on condition variable pointed by
|
|
|
|
__COND. */
|
|
|
|
extern int cnd_broadcast (cnd_t *__cond);
|
|
|
|
|
|
|
|
/* Block current thread on the condition variable pointed by __COND. */
|
|
|
|
extern int cnd_wait (cnd_t *__cond, mtx_t *__mutex);
|
|
|
|
|
|
|
|
/* Block current thread on the condition variable until condition variable
|
|
|
|
pointed by __COND is signaled or time pointed by __TIME_POINT is
|
|
|
|
reached. */
|
2024-01-18 13:18:01 +00:00
|
|
|
#ifndef __USE_TIME64_REDIRECTS
|
2018-07-25 18:20:44 +00:00
|
|
|
extern int cnd_timedwait (cnd_t *__restrict __cond,
|
|
|
|
mtx_t *__restrict __mutex,
|
|
|
|
const struct timespec *__restrict __time_point);
|
y2038: Add support for 64-bit time on legacy ABIs
A new build flag, _TIME_BITS, enables the usage of the newer 64-bit
time symbols for legacy ABI (where 32-bit time_t is default). The 64
bit time support is only enabled if LFS (_FILE_OFFSET_BITS=64) is
also used.
Different than LFS support, the y2038 symbols are added only for the
required ABIs (armhf, csky, hppa, i386, m68k, microblaze, mips32,
mips64-n32, nios2, powerpc32, sparc32, s390-32, and sh). The ABIs with
64-bit time support are unchanged, both for symbol and types
redirection.
On Linux the full 64-bit time support requires a minimum of kernel
version v5.1. Otherwise, the 32-bit fallbacks are used and might
results in error with overflow return code (EOVERFLOW).
The i686-gnu does not yet support 64-bit time.
This patch exports following rediretions to support 64-bit time:
* libc:
adjtime
adjtimex
clock_adjtime
clock_getres
clock_gettime
clock_nanosleep
clock_settime
cnd_timedwait
ctime
ctime_r
difftime
fstat
fstatat
futimens
futimes
futimesat
getitimer
getrusage
gettimeofday
gmtime
gmtime_r
localtime
localtime_r
lstat_time
lutimes
mktime
msgctl
mtx_timedlock
nanosleep
nanosleep
ntp_gettime
ntp_gettimex
ppoll
pselec
pselect
pthread_clockjoin_np
pthread_cond_clockwait
pthread_cond_timedwait
pthread_mutex_clocklock
pthread_mutex_timedlock
pthread_rwlock_clockrdlock
pthread_rwlock_clockwrlock
pthread_rwlock_timedrdlock
pthread_rwlock_timedwrlock
pthread_timedjoin_np
recvmmsg
sched_rr_get_interval
select
sem_clockwait
semctl
semtimedop
sem_timedwait
setitimer
settimeofday
shmctl
sigtimedwait
stat
thrd_sleep
time
timegm
timerfd_gettime
timerfd_settime
timespec_get
utime
utimensat
utimes
utimes
wait3
wait4
* librt:
aio_suspend
mq_timedreceive
mq_timedsend
timer_gettime
timer_settime
* libanl:
gai_suspend
Reviewed-by: Lukasz Majewski <lukma@denx.de>
Reviewed-by: Carlos O'Donell <carlos@redhat.com>
Tested-by: Carlos O'Donell <carlos@redhat.com>
2021-02-17 16:35:19 +00:00
|
|
|
#else
|
|
|
|
# ifdef __REDIRECT
|
|
|
|
extern int __REDIRECT (cnd_timedwait, (cnd_t *__restrict __cond,
|
|
|
|
mtx_t *__restrict __mutex,
|
|
|
|
const struct timespec *__restrict
|
|
|
|
__time_point),
|
|
|
|
__cnd_timedwait64);
|
|
|
|
# else
|
|
|
|
# define cnd_timedwait __cnd_timedwait64
|
|
|
|
# endif
|
|
|
|
#endif
|
2018-07-25 18:20:44 +00:00
|
|
|
|
|
|
|
/* Destroy condition variable pointed by __cond and free all of its
|
|
|
|
resources. */
|
|
|
|
extern void cnd_destroy (cnd_t *__COND);
|
|
|
|
|
|
|
|
|
|
|
|
/* Thread specific storage functions. */
|
|
|
|
|
|
|
|
/* Create new thread-specific storage key and stores it in the object pointed
|
|
|
|
by __TSS_ID. If __DESTRUCTOR is not NULL, the function will be called when
|
|
|
|
the thread terminates. */
|
|
|
|
extern int tss_create (tss_t *__tss_id, tss_dtor_t __destructor);
|
|
|
|
|
|
|
|
/* Return the value held in thread-specific storage for the current thread
|
|
|
|
identified by __TSS_ID. */
|
|
|
|
extern void *tss_get (tss_t __tss_id);
|
|
|
|
|
|
|
|
/* Sets the value of the thread-specific storage identified by __TSS_ID for
|
|
|
|
the current thread to __VAL. */
|
|
|
|
extern int tss_set (tss_t __tss_id, void *__val);
|
|
|
|
|
|
|
|
/* Destroys the thread-specific storage identified by __TSS_ID. The
|
|
|
|
destructor is not called until thrd_exit is called. */
|
|
|
|
extern void tss_delete (tss_t __tss_id);
|
|
|
|
|
|
|
|
__END_DECLS
|
|
|
|
|
|
|
|
#endif /* _THREADS_H */
|