* Fix typos in rwlock stubs. * Add pthread_XXX counterparts to the _pthread_XXX stubs which libraries like libX11 can use to ensure thread-safety without requiring the use of a thread library. Submitted by: Terry Lambert (pthread_cond_broadcast) Reviewed by: deischen
454 lines
12 KiB
C
454 lines
12 KiB
C
/*
|
|
* Copyright (c) 2001 Daniel Eischen <deischen@FreeBSD.org>.
|
|
* 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 DANIEL EISCHEN 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 <sys/cdefs.h>
|
|
__FBSDID("$FreeBSD$");
|
|
|
|
#include <signal.h>
|
|
#include <pthread.h>
|
|
#include <pthread_np.h>
|
|
|
|
void *_pthread_getspecific(pthread_key_t key);
|
|
pthread_t _pthread_self(void);
|
|
|
|
/*
|
|
* Weak symbols: All libc internal usage of these functions should
|
|
* use the weak symbol versions (_pthread_XXX). If libpthread is
|
|
* linked, it will override these functions with (non-weak) routines.
|
|
* The _pthread_XXX functions are provided solely for internal libc
|
|
* usage to avoid unwanted cancellation points and to differentiate
|
|
* between application locks and libc locks (threads holding the
|
|
* latter can't be allowed to exit/terminate).
|
|
*
|
|
* We also provide weak pthread_XXX stubs which call their
|
|
* _pthread_XXX counterparts. These stubs may be used be other
|
|
* libraries for ensuring thread-safety without requiring the presence
|
|
* of a thread library.
|
|
*/
|
|
__weak_reference(_pthread_cond_init_stub, _pthread_cond_init);
|
|
__weak_reference(_pthread_cond_signal_stub, _pthread_cond_signal);
|
|
__weak_reference(_pthread_cond_broadcast_stub, _pthread_cond_broadcast);
|
|
__weak_reference(_pthread_cond_wait_stub, _pthread_cond_wait);
|
|
__weak_reference(_pthread_cond_destroy_stub, _pthread_cond_destroy);
|
|
__weak_reference(_pthread_getspecific_stub, _pthread_getspecific);
|
|
__weak_reference(_pthread_key_create_stub, _pthread_key_create);
|
|
__weak_reference(_pthread_key_delete_stub, _pthread_key_delete);
|
|
__weak_reference(_pthread_main_np_stub, _pthread_main_np);
|
|
__weak_reference(_pthread_mutex_destroy_stub, _pthread_mutex_destroy);
|
|
__weak_reference(_pthread_mutex_init_stub, _pthread_mutex_init);
|
|
__weak_reference(_pthread_mutex_lock_stub, _pthread_mutex_lock);
|
|
__weak_reference(_pthread_mutex_trylock_stub, _pthread_mutex_trylock);
|
|
__weak_reference(_pthread_mutex_unlock_stub, _pthread_mutex_unlock);
|
|
__weak_reference(_pthread_mutexattr_init_stub, _pthread_mutexattr_init);
|
|
__weak_reference(_pthread_mutexattr_destroy_stub, _pthread_mutexattr_destroy);
|
|
__weak_reference(_pthread_mutexattr_settype_stub, _pthread_mutexattr_settype);
|
|
__weak_reference(_pthread_once_stub, _pthread_once);
|
|
__weak_reference(_pthread_self_stub, _pthread_self);
|
|
__weak_reference(_pthread_rwlock_init_stub, _pthread_rwlock_init);
|
|
__weak_reference(_pthread_rwlock_destroy_stub, _pthread_rwlock_destroy);
|
|
__weak_reference(_pthread_rwlock_rdlock_stub, _pthread_rwlock_rdlock);
|
|
__weak_reference(_pthread_rwlock_tryrdlock_stub, _pthread_rwlock_tryrdlock);
|
|
__weak_reference(_pthread_rwlock_trywrlock_stub, _pthread_rwlock_trywrlock);
|
|
__weak_reference(_pthread_rwlock_unlock_stub, _pthread_rwlock_unlock);
|
|
__weak_reference(_pthread_rwlock_wrlock_stub, _pthread_rwlock_wrlock);
|
|
__weak_reference(_pthread_setspecific_stub, _pthread_setspecific);
|
|
__weak_reference(_pthread_sigmask_stub, _pthread_sigmask);
|
|
|
|
__weak_reference(pthread_cond_init_stub, pthread_cond_init);
|
|
__weak_reference(pthread_cond_signal_stub, pthread_cond_signal);
|
|
__weak_reference(pthread_cond_broadcast_stub, pthread_cond_broadcast);
|
|
__weak_reference(pthread_cond_wait_stub, pthread_cond_wait);
|
|
__weak_reference(pthread_cond_destroy_stub, pthread_cond_destroy);
|
|
__weak_reference(pthread_getspecific_stub, pthread_getspecific);
|
|
__weak_reference(pthread_key_create_stub, pthread_key_create);
|
|
__weak_reference(pthread_key_delete_stub, pthread_key_delete);
|
|
__weak_reference(pthread_main_np_stub, pthread_main_np);
|
|
__weak_reference(pthread_mutex_destroy_stub, pthread_mutex_destroy);
|
|
__weak_reference(pthread_mutex_init_stub, pthread_mutex_init);
|
|
__weak_reference(pthread_mutex_lock_stub, pthread_mutex_lock);
|
|
__weak_reference(pthread_mutex_trylock_stub, pthread_mutex_trylock);
|
|
__weak_reference(pthread_mutex_unlock_stub, pthread_mutex_unlock);
|
|
__weak_reference(pthread_mutexattr_init_stub, pthread_mutexattr_init);
|
|
__weak_reference(pthread_mutexattr_destroy_stub, pthread_mutexattr_destroy);
|
|
__weak_reference(pthread_mutexattr_settype_stub, pthread_mutexattr_settype);
|
|
__weak_reference(pthread_once_stub, pthread_once);
|
|
__weak_reference(pthread_self_stub, pthread_self);
|
|
__weak_reference(pthread_rwlock_init_stub, pthread_rwlock_init);
|
|
__weak_reference(pthread_rwlock_destroy_stub, pthread_rwlock_destroy);
|
|
__weak_reference(pthread_rwlock_rdlock_stub, pthread_rwlock_rdlock);
|
|
__weak_reference(pthread_rwlock_tryrdlock_stub, pthread_rwlock_tryrdlock);
|
|
__weak_reference(pthread_rwlock_trywrlock_stub, pthread_rwlock_trywrlock);
|
|
__weak_reference(pthread_rwlock_unlock_stub, pthread_rwlock_unlock);
|
|
__weak_reference(pthread_rwlock_wrlock_stub, pthread_rwlock_wrlock);
|
|
__weak_reference(pthread_setspecific_stub, pthread_setspecific);
|
|
__weak_reference(pthread_sigmask_stub, pthread_sigmask);
|
|
|
|
/* Define a null pthread structure just to satisfy _pthread_self. */
|
|
struct pthread {
|
|
};
|
|
|
|
static struct pthread main_thread;
|
|
|
|
static int
|
|
_pthread_cond_init_stub(pthread_cond_t *cond,
|
|
const pthread_condattr_t *cond_attr)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
_pthread_cond_signal_stub(pthread_cond_t *cond)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
_pthread_cond_broadcast_stub(pthread_cond_t *cond)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
_pthread_cond_wait_stub(pthread_cond_t *cond, pthread_mutex_t *mutex)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
_pthread_cond_destroy_stub(pthread_cond_t *cond)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
static void *
|
|
_pthread_getspecific_stub(pthread_key_t key)
|
|
{
|
|
return (NULL);
|
|
}
|
|
|
|
static int
|
|
_pthread_key_create_stub(pthread_key_t *key, void (*destructor) (void *))
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
_pthread_key_delete_stub(pthread_key_t key)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
_pthread_main_np_stub()
|
|
{
|
|
return (-1);
|
|
}
|
|
|
|
static int
|
|
_pthread_mutex_destroy_stub(pthread_mutex_t *mattr)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
_pthread_mutex_init_stub(pthread_mutex_t *mutex, const pthread_mutexattr_t *mattr)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
_pthread_mutex_lock_stub(pthread_mutex_t *mutex)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
_pthread_mutex_trylock_stub(pthread_mutex_t *mutex)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
_pthread_mutex_unlock_stub(pthread_mutex_t *mutex)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
_pthread_mutexattr_init_stub(pthread_mutexattr_t *mattr)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
_pthread_mutexattr_destroy_stub(pthread_mutexattr_t *mattr)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
_pthread_mutexattr_settype_stub(pthread_mutexattr_t *mattr, int type)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
_pthread_once_stub(pthread_once_t *once_control, void (*init_routine) (void))
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
_pthread_rwlock_init_stub(pthread_rwlock_t *rwlock,
|
|
const pthread_rwlockattr_t *attr)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
_pthread_rwlock_destroy_stub(pthread_rwlock_t *rwlock)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
_pthread_rwlock_rdlock_stub(pthread_rwlock_t *rwlock)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
_pthread_rwlock_tryrdlock_stub(pthread_rwlock_t *rwlock)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
_pthread_rwlock_trywrlock_stub(pthread_rwlock_t *rwlock)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
_pthread_rwlock_unlock_stub(pthread_rwlock_t *rwlock)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
_pthread_rwlock_wrlock_stub(pthread_rwlock_t *rwlock)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
static pthread_t
|
|
_pthread_self_stub(void)
|
|
{
|
|
return (&main_thread);
|
|
}
|
|
|
|
static int
|
|
_pthread_setspecific_stub(pthread_key_t key, const void *value)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
_pthread_sigmask_stub(int how, const sigset_t *set, sigset_t *oset)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
pthread_cond_init_stub(pthread_cond_t *cond,
|
|
const pthread_condattr_t *cond_attr)
|
|
{
|
|
return (_pthread_cond_init(cond, cond_attr));
|
|
}
|
|
|
|
static int
|
|
pthread_cond_signal_stub(pthread_cond_t *cond)
|
|
{
|
|
return (_pthread_cond_signal(cond));
|
|
}
|
|
|
|
static int
|
|
pthread_cond_broadcast_stub(pthread_cond_t *cond)
|
|
{
|
|
return (_pthread_cond_broadcast(cond));
|
|
}
|
|
|
|
static int
|
|
pthread_cond_wait_stub(pthread_cond_t *cond, pthread_mutex_t *mutex)
|
|
{
|
|
return (_pthread_cond_wait(cond, mutex));
|
|
}
|
|
|
|
static int
|
|
pthread_cond_destroy_stub(pthread_cond_t *cond)
|
|
{
|
|
return (_pthread_cond_destroy(cond));
|
|
}
|
|
|
|
static void *
|
|
pthread_getspecific_stub(pthread_key_t key)
|
|
{
|
|
return (_pthread_getspecific(key));
|
|
}
|
|
|
|
static int
|
|
pthread_key_create_stub(pthread_key_t *key, void (*destructor) (void *))
|
|
{
|
|
return (_pthread_key_create(key, destructor));
|
|
}
|
|
|
|
static int
|
|
pthread_key_delete_stub(pthread_key_t key)
|
|
{
|
|
return (_pthread_key_delete(key));
|
|
}
|
|
|
|
static int
|
|
pthread_main_np_stub()
|
|
{
|
|
return (_pthread_main_np());
|
|
}
|
|
|
|
static int
|
|
pthread_mutex_destroy_stub(pthread_mutex_t *mattr)
|
|
{
|
|
return (_pthread_mutex_destroy(mattr));
|
|
}
|
|
|
|
static int
|
|
pthread_mutex_init_stub(pthread_mutex_t *mutex, const pthread_mutexattr_t *mattr)
|
|
{
|
|
return (_pthread_mutex_init(mutex, mattr));
|
|
}
|
|
|
|
static int
|
|
pthread_mutex_lock_stub(pthread_mutex_t *mutex)
|
|
{
|
|
return (_pthread_mutex_lock(mutex));
|
|
}
|
|
|
|
static int
|
|
pthread_mutex_trylock_stub(pthread_mutex_t *mutex)
|
|
{
|
|
return (_pthread_mutex_trylock(mutex));
|
|
}
|
|
|
|
static int
|
|
pthread_mutex_unlock_stub(pthread_mutex_t *mutex)
|
|
{
|
|
return (_pthread_mutex_unlock(mutex));
|
|
}
|
|
|
|
static int
|
|
pthread_mutexattr_init_stub(pthread_mutexattr_t *mattr)
|
|
{
|
|
return (_pthread_mutexattr_init(mattr));
|
|
}
|
|
|
|
static int
|
|
pthread_mutexattr_destroy_stub(pthread_mutexattr_t *mattr)
|
|
{
|
|
return (_pthread_mutexattr_destroy(mattr));
|
|
}
|
|
|
|
static int
|
|
pthread_mutexattr_settype_stub(pthread_mutexattr_t *mattr, int type)
|
|
{
|
|
return (_pthread_mutexattr_settype(mattr, type));
|
|
}
|
|
|
|
static int
|
|
pthread_once_stub(pthread_once_t *once_control, void (*init_routine) (void))
|
|
{
|
|
return (_pthread_once(once_control, init_routine));
|
|
}
|
|
|
|
static int
|
|
pthread_rwlock_init_stub(pthread_rwlock_t *rwlock,
|
|
const pthread_rwlockattr_t *attr)
|
|
{
|
|
return (_pthread_rwlock_init(rwlock, attr));
|
|
}
|
|
|
|
static int
|
|
pthread_rwlock_destroy_stub(pthread_rwlock_t *rwlock)
|
|
{
|
|
return (_pthread_rwlock_destroy(rwlock));
|
|
}
|
|
|
|
static int
|
|
pthread_rwlock_rdlock_stub(pthread_rwlock_t *rwlock)
|
|
{
|
|
return (_pthread_rwlock_rdlock(rwlock));
|
|
}
|
|
|
|
static int
|
|
pthread_rwlock_tryrdlock_stub(pthread_rwlock_t *rwlock)
|
|
{
|
|
return (_pthread_rwlock_tryrdlock(rwlock));
|
|
}
|
|
|
|
static int
|
|
pthread_rwlock_trywrlock_stub(pthread_rwlock_t *rwlock)
|
|
{
|
|
return (_pthread_rwlock_trywrlock(rwlock));
|
|
}
|
|
|
|
static int
|
|
pthread_rwlock_unlock_stub(pthread_rwlock_t *rwlock)
|
|
{
|
|
return (_pthread_rwlock_unlock(rwlock));
|
|
}
|
|
|
|
static int
|
|
pthread_rwlock_wrlock_stub(pthread_rwlock_t *rwlock)
|
|
{
|
|
return (_pthread_rwlock_wrlock(rwlock));
|
|
}
|
|
|
|
static pthread_t
|
|
pthread_self_stub(void)
|
|
{
|
|
return (_pthread_self());
|
|
}
|
|
|
|
static int
|
|
pthread_setspecific_stub(pthread_key_t key, const void *value)
|
|
{
|
|
return (_pthread_setspecific(key, value));
|
|
}
|
|
|
|
static int
|
|
pthread_sigmask_stub(int how, const sigset_t *set, sigset_t *oset)
|
|
{
|
|
return (_pthread_sigmask(how, set, oset));
|
|
}
|