mirror of
https://sourceware.org/git/glibc.git
synced 2024-12-26 04:31:03 +00:00
afe4de7d28
Add: int pthread_cond_clockwait (pthread_cond_t *cond, pthread_mutex_t *mutex, clockid_t clockid, const struct timespec *abstime) which behaves just like pthread_cond_timedwait except it always measures abstime against the supplied clockid. Currently supports CLOCK_REALTIME and CLOCK_MONOTONIC and returns EINVAL if any other clock is specified. Includes feedback from many others. This function was originally proposed[1] as pthread_cond_timedwaitonclock_np, but The Austin Group preferred the new name. * nptl/Makefile: Add tst-cond26 and tst-cond27 * nptl/Versions (GLIBC_2.30): Add pthread_cond_clockwait * sysdeps/nptl/pthread.h: Likewise * nptl/forward.c: Add __pthread_cond_clockwait * nptl/forward.c: Likewise * nptl/pthreadP.h: Likewise * sysdeps/nptl/pthread-functions.h: Likewise * nptl/pthread_cond_wait.c (__pthread_cond_wait_common): Add clockid parameter and comment describing why we don't need to check its value. Use that value when calling futex_abstimed_wait_cancelable rather than reading the clock from the flags. (__pthread_cond_wait): Pass unused clockid parameter. (__pthread_cond_timedwait): Read clock from flags and pass it to __pthread_cond_wait_common. (__pthread_cond_clockwait): Add new function with weak alias from pthread_cond_clockwait. * sysdeps/mach/hurd/i386/libpthread.abilist (GLIBC_2.30): * Likewise. * sysdeps/unix/sysv/linux/aarch64/libpthread.abilist * (GLIBC_2.30): Likewise. * sysdeps/unix/sysv/linux/alpha/libpthread.abilist (GLIBC_2.30): * Likewise. * sysdeps/unix/sysv/linux/arm/libpthread.abilist (GLIBC_2.30): * Likewise. * sysdeps/unix/sysv/linux/csky/libpthread.abilist (GLIBC_2.30): * Likewise. * sysdeps/unix/sysv/linux/hppa/libpthread.abilist (GLIBC_2.30): * Likewise. * sysdeps/unix/sysv/linux/i386/libpthread.abilist (GLIBC_2.30): * Likewise. * sysdeps/unix/sysv/linux/ia64/libpthread.abilist (GLIBC_2.30): * Likewise. * sysdeps/unix/sysv/linux/m68k/coldfire/libpthread.abilist (GLIBC_2.30): Likewise. * sysdeps/unix/sysv/linux/m68k/m680x0/libpthread.abilist (GLIBC_2.30): Likewise. * sysdeps/unix/sysv/linux/microblaze/libpthread.abilist (GLIBC_2.30): Likewise. * sysdeps/unix/sysv/linux/mips/mips32/libpthread.abilist (GLIBC_2.30): Likewise. * sysdeps/unix/sysv/linux/mips/mips64/libpthread.abilist (GLIBC_2.30): Likewise. * sysdeps/unix/sysv/linux/nios2/libpthread.abilist (GLIBC_2.30): * Likewise. * sysdeps/unix/sysv/linux/powerpc/powerpc32/libpthread.abilist (GLIBC_2.30): Likewise. * sysdeps/unix/sysv/linux/powerpc/powerpc64/be/libpthread.abilist (GLIBC_2.30): Likewise. * sysdeps/unix/sysv/linux/powerpc/powerpc64/le/libpthread.abilist (GLIBC_2.30): Likewise. * sysdeps/unix/sysv/linux/riscv/rv64/libpthread.abilist (GLIBC_2.30): Likewise. * sysdeps/unix/sysv/linux/s390/s390-32/libpthread.abilist (GLIBC_2.30): Likewise. * sysdeps/unix/sysv/linux/s390/s390-64/libpthread.abilist (GLIBC_2.30): Likewise. * sysdeps/unix/sysv/linux/sh/libpthread.abilist (GLIBC_2.30): * Likewise. * sysdeps/unix/sysv/linux/sparc/sparc32/libpthread.abilist (GLIBC_2.30): Likewise. * sysdeps/unix/sysv/linux/sparc/sparc64/libpthread.abilist (GLIBC_2.30): Likewise. * sysdeps/unix/sysv/linux/x86_64/64/libpthread.abilist (GLIBC_2.30): Likewise. * sysdeps/unix/sysv/linux/x86_64/x32/libpthread.abilist (GLIBC_2.30): Likewise. * nptl/tst-cond11.c (run_test): Support testing pthread_cond_clockwait too by using a special magic CLOCK_USE_ATTR_CLOCK value to determine whether to call pthread_cond_timedwait or pthread_cond_clockwait. (do_test): Pass CLOCK_USE_ATTR_CLOCK for existing tests, and add new tests using all combinations of CLOCK_MONOTONIC and CLOCK_REALTIME. * ntpl/tst-cond26.c: New test for passing unsupported and * invalid clocks to pthread_cond_clockwait. * nptl/tst-cond27.c: Add test similar to tst-cond5.c, but using struct timespec and pthread_cond_clockwait. * manual/threads.texi: Document pthread_cond_clockwait. The * comment was provided by Carlos O'Donell. [1] https://sourceware.org/ml/libc-alpha/2015-07/msg00193.html Reviewed-by: Adhemerval Zanella <adhemerval.zanella@linaro.org>
212 lines
7.7 KiB
C
212 lines
7.7 KiB
C
/* Copyright (C) 2002-2019 Free Software Foundation, Inc.
|
|
This file is part of the GNU C Library.
|
|
Contributed by Ulrich Drepper <drepper@redhat.com>, 2002.
|
|
|
|
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/>. */
|
|
|
|
#include <dlfcn.h>
|
|
#include <pthreadP.h>
|
|
#include <signal.h>
|
|
#include <stdlib.h>
|
|
|
|
#include <shlib-compat.h>
|
|
#include <atomic.h>
|
|
#include <safe-fatal.h>
|
|
|
|
|
|
/* Pointers to the libc functions. */
|
|
struct pthread_functions __libc_pthread_functions attribute_hidden;
|
|
int __libc_pthread_functions_init attribute_hidden;
|
|
|
|
|
|
#define FORWARD2(name, rettype, decl, params, defaction) \
|
|
rettype \
|
|
name decl \
|
|
{ \
|
|
if (!__libc_pthread_functions_init) \
|
|
defaction; \
|
|
\
|
|
return PTHFCT_CALL (ptr_##name, params); \
|
|
}
|
|
|
|
/* Same as FORWARD2, only without return. */
|
|
#define FORWARD_NORETURN(name, rettype, decl, params, defaction) \
|
|
rettype \
|
|
name decl \
|
|
{ \
|
|
if (!__libc_pthread_functions_init) \
|
|
defaction; \
|
|
\
|
|
PTHFCT_CALL (ptr_##name, params); \
|
|
}
|
|
|
|
#define FORWARD(name, decl, params, defretval) \
|
|
FORWARD2 (name, int, decl, params, return defretval)
|
|
|
|
|
|
FORWARD (pthread_attr_destroy, (pthread_attr_t *attr), (attr), 0)
|
|
|
|
#if SHLIB_COMPAT(libc, GLIBC_2_0, GLIBC_2_1)
|
|
FORWARD (__pthread_attr_init_2_0, (pthread_attr_t *attr), (attr), 0)
|
|
compat_symbol (libc, __pthread_attr_init_2_0, pthread_attr_init, GLIBC_2_0);
|
|
#endif
|
|
|
|
FORWARD (__pthread_attr_init_2_1, (pthread_attr_t *attr), (attr), 0)
|
|
versioned_symbol (libc, __pthread_attr_init_2_1, pthread_attr_init, GLIBC_2_1);
|
|
|
|
FORWARD (pthread_attr_getdetachstate,
|
|
(const pthread_attr_t *attr, int *detachstate), (attr, detachstate),
|
|
0)
|
|
FORWARD (pthread_attr_setdetachstate, (pthread_attr_t *attr, int detachstate),
|
|
(attr, detachstate), 0)
|
|
|
|
FORWARD (pthread_attr_getinheritsched,
|
|
(const pthread_attr_t *attr, int *inherit), (attr, inherit), 0)
|
|
FORWARD (pthread_attr_setinheritsched, (pthread_attr_t *attr, int inherit),
|
|
(attr, inherit), 0)
|
|
|
|
FORWARD (pthread_attr_getschedparam,
|
|
(const pthread_attr_t *attr, struct sched_param *param),
|
|
(attr, param), 0)
|
|
FORWARD (pthread_attr_setschedparam,
|
|
(pthread_attr_t *attr, const struct sched_param *param),
|
|
(attr, param), 0)
|
|
|
|
FORWARD (pthread_attr_getschedpolicy,
|
|
(const pthread_attr_t *attr, int *policy), (attr, policy), 0)
|
|
FORWARD (pthread_attr_setschedpolicy, (pthread_attr_t *attr, int policy),
|
|
(attr, policy), 0)
|
|
|
|
FORWARD (pthread_attr_getscope,
|
|
(const pthread_attr_t *attr, int *scope), (attr, scope), 0)
|
|
FORWARD (pthread_attr_setscope, (pthread_attr_t *attr, int scope),
|
|
(attr, scope), 0)
|
|
|
|
|
|
FORWARD (pthread_condattr_destroy, (pthread_condattr_t *attr), (attr), 0)
|
|
FORWARD (pthread_condattr_init, (pthread_condattr_t *attr), (attr), 0)
|
|
|
|
#if SHLIB_COMPAT(libc, GLIBC_2_0, GLIBC_2_3_2)
|
|
FORWARD2 (__pthread_cond_broadcast_2_0, int attribute_compat_text_section,
|
|
(pthread_cond_2_0_t *cond), (cond), return 0)
|
|
compat_symbol (libc, __pthread_cond_broadcast_2_0, pthread_cond_broadcast,
|
|
GLIBC_2_0);
|
|
#endif
|
|
FORWARD (__pthread_cond_broadcast, (pthread_cond_t *cond), (cond), 0)
|
|
versioned_symbol (libc, __pthread_cond_broadcast, pthread_cond_broadcast,
|
|
GLIBC_2_3_2);
|
|
|
|
#if SHLIB_COMPAT(libc, GLIBC_2_0, GLIBC_2_3_2)
|
|
FORWARD2 (__pthread_cond_destroy_2_0, int attribute_compat_text_section,
|
|
(pthread_cond_2_0_t *cond), (cond), return 0)
|
|
compat_symbol (libc, __pthread_cond_destroy_2_0, pthread_cond_destroy,
|
|
GLIBC_2_0);
|
|
#endif
|
|
FORWARD (__pthread_cond_destroy, (pthread_cond_t *cond), (cond), 0)
|
|
versioned_symbol (libc, __pthread_cond_destroy, pthread_cond_destroy,
|
|
GLIBC_2_3_2);
|
|
|
|
#if SHLIB_COMPAT(libc, GLIBC_2_0, GLIBC_2_3_2)
|
|
FORWARD2 (__pthread_cond_init_2_0, int attribute_compat_text_section,
|
|
(pthread_cond_2_0_t *cond, const pthread_condattr_t *cond_attr),
|
|
(cond, cond_attr), return 0)
|
|
compat_symbol (libc, __pthread_cond_init_2_0, pthread_cond_init, GLIBC_2_0);
|
|
#endif
|
|
FORWARD (__pthread_cond_init,
|
|
(pthread_cond_t *cond, const pthread_condattr_t *cond_attr),
|
|
(cond, cond_attr), 0)
|
|
versioned_symbol (libc, __pthread_cond_init, pthread_cond_init, GLIBC_2_3_2);
|
|
|
|
#if SHLIB_COMPAT(libc, GLIBC_2_0, GLIBC_2_3_2)
|
|
FORWARD2 (__pthread_cond_signal_2_0, int attribute_compat_text_section,
|
|
(pthread_cond_2_0_t *cond), (cond), return 0)
|
|
compat_symbol (libc, __pthread_cond_signal_2_0, pthread_cond_signal,
|
|
GLIBC_2_0);
|
|
#endif
|
|
FORWARD (__pthread_cond_signal, (pthread_cond_t *cond), (cond), 0)
|
|
versioned_symbol (libc, __pthread_cond_signal, pthread_cond_signal,
|
|
GLIBC_2_3_2);
|
|
|
|
#if SHLIB_COMPAT(libc, GLIBC_2_0, GLIBC_2_3_2)
|
|
FORWARD2 (__pthread_cond_wait_2_0, int attribute_compat_text_section,
|
|
(pthread_cond_2_0_t *cond, pthread_mutex_t *mutex), (cond, mutex),
|
|
return 0)
|
|
compat_symbol (libc, __pthread_cond_wait_2_0, pthread_cond_wait,
|
|
GLIBC_2_0);
|
|
#endif
|
|
FORWARD (__pthread_cond_wait, (pthread_cond_t *cond, pthread_mutex_t *mutex),
|
|
(cond, mutex), 0)
|
|
versioned_symbol (libc, __pthread_cond_wait, pthread_cond_wait,
|
|
GLIBC_2_3_2);
|
|
|
|
#if SHLIB_COMPAT(libc, GLIBC_2_0, GLIBC_2_3_2)
|
|
FORWARD2 (__pthread_cond_timedwait_2_0, int attribute_compat_text_section,
|
|
(pthread_cond_2_0_t *cond, pthread_mutex_t *mutex,
|
|
const struct timespec *abstime), (cond, mutex, abstime),
|
|
return 0)
|
|
compat_symbol (libc, __pthread_cond_timedwait_2_0, pthread_cond_timedwait,
|
|
GLIBC_2_0);
|
|
#endif
|
|
FORWARD (__pthread_cond_timedwait,
|
|
(pthread_cond_t *cond, pthread_mutex_t *mutex,
|
|
const struct timespec *abstime), (cond, mutex, abstime), 0)
|
|
versioned_symbol (libc, __pthread_cond_timedwait, pthread_cond_timedwait,
|
|
GLIBC_2_3_2);
|
|
FORWARD (__pthread_cond_clockwait,
|
|
(pthread_cond_t *cond, pthread_mutex_t *mutex, clockid_t clockid,
|
|
const struct timespec *abstime), (cond, mutex, clockid, abstime),
|
|
0)
|
|
weak_alias (__pthread_cond_clockwait, pthread_cond_clockwait);
|
|
|
|
|
|
FORWARD (pthread_equal, (pthread_t thread1, pthread_t thread2),
|
|
(thread1, thread2), 1)
|
|
|
|
|
|
FORWARD_NORETURN (__pthread_exit, void, (void *retval), (retval),
|
|
exit (EXIT_SUCCESS))
|
|
strong_alias (__pthread_exit, pthread_exit);
|
|
|
|
|
|
FORWARD (pthread_getschedparam,
|
|
(pthread_t target_thread, int *policy, struct sched_param *param),
|
|
(target_thread, policy, param), 0)
|
|
FORWARD (pthread_setschedparam,
|
|
(pthread_t target_thread, int policy,
|
|
const struct sched_param *param), (target_thread, policy, param), 0)
|
|
|
|
|
|
FORWARD (pthread_mutex_destroy, (pthread_mutex_t *mutex), (mutex), 0)
|
|
|
|
FORWARD (pthread_mutex_init,
|
|
(pthread_mutex_t *mutex, const pthread_mutexattr_t *mutexattr),
|
|
(mutex, mutexattr), 0)
|
|
|
|
FORWARD (pthread_mutex_lock, (pthread_mutex_t *mutex), (mutex), 0)
|
|
|
|
FORWARD (pthread_mutex_unlock, (pthread_mutex_t *mutex), (mutex), 0)
|
|
|
|
FORWARD (__pthread_setcancelstate, (int state, int *oldstate),
|
|
(state, oldstate), 0)
|
|
strong_alias (__pthread_setcancelstate, pthread_setcancelstate)
|
|
|
|
FORWARD (pthread_setcanceltype, (int type, int *oldtype), (type, oldtype), 0)
|
|
|
|
FORWARD_NORETURN (__pthread_unwind,
|
|
void attribute_hidden __attribute ((noreturn))
|
|
__cleanup_fct_attribute attribute_compat_text_section,
|
|
(__pthread_unwind_buf_t *buf), (buf),
|
|
__safe_fatal ())
|