136 lines
2.2 KiB
C
136 lines
2.2 KiB
C
/*
|
|
* The UCW Library -- Threading Helpers
|
|
*
|
|
* (c) 2006--2010 Martin Mares <mj@ucw.cz>
|
|
*
|
|
* This software may be freely distributed and used according to the terms
|
|
* of the GNU Lesser General Public License.
|
|
*/
|
|
|
|
#include <ucw/lib.h>
|
|
#include <ucw/threads.h>
|
|
|
|
#ifdef CONFIG_UCW_THREADS
|
|
|
|
#include <pthread.h>
|
|
|
|
#ifdef CONFIG_LINUX
|
|
#include <sys/types.h>
|
|
#include <sys/syscall.h>
|
|
#include <unistd.h>
|
|
#ifdef __NR_gettid
|
|
static pid_t
|
|
ucw_gettid(void)
|
|
{
|
|
return syscall(__NR_gettid);
|
|
}
|
|
#define CONFIG_USE_GETTID
|
|
#endif
|
|
#endif
|
|
|
|
/*** Library lock ***/
|
|
|
|
static pthread_mutex_t ucwlib_master_mutex;
|
|
|
|
void
|
|
ucwlib_lock(void)
|
|
{
|
|
pthread_mutex_lock(&ucwlib_master_mutex);
|
|
}
|
|
|
|
void
|
|
ucwlib_unlock(void)
|
|
{
|
|
pthread_mutex_unlock(&ucwlib_master_mutex);
|
|
}
|
|
|
|
/*** Thread identifiers ***/
|
|
|
|
static int
|
|
ucwlib_tid(void)
|
|
{
|
|
static int tid_counter;
|
|
int tid;
|
|
|
|
#ifdef CONFIG_USE_GETTID
|
|
tid = ucw_gettid();
|
|
if (tid > 0)
|
|
return tid;
|
|
/* The syscall might be unimplemented */
|
|
#endif
|
|
|
|
ucwlib_lock();
|
|
tid = ++tid_counter;
|
|
ucwlib_unlock();
|
|
return tid;
|
|
}
|
|
|
|
/*** Thread context ***/
|
|
|
|
static void CONSTRUCTOR_WITH_PRIORITY(10000)
|
|
ucwlib_threads_init_master(void)
|
|
{
|
|
pthread_mutex_init(&ucwlib_master_mutex, NULL);
|
|
}
|
|
|
|
#ifdef CONFIG_UCW_TLS
|
|
|
|
__thread struct ucwlib_context ucwlib_context;
|
|
|
|
int
|
|
ucwlib_thread_id(struct ucwlib_context *c)
|
|
{
|
|
if (!c->_thread_id)
|
|
c->_thread_id = ucwlib_tid();
|
|
return c->_thread_id;
|
|
}
|
|
|
|
#else
|
|
|
|
static pthread_key_t ucwlib_context_key;
|
|
|
|
static void
|
|
ucwlib_free_thread_context(void *p)
|
|
{
|
|
xfree(p);
|
|
}
|
|
|
|
static void CONSTRUCTOR_WITH_PRIORITY(10000)
|
|
ucwlib_threads_init(void)
|
|
{
|
|
if (pthread_key_create(&ucwlib_context_key, ucwlib_free_thread_context) < 0)
|
|
die("Cannot create pthread_key: %m");
|
|
}
|
|
|
|
struct ucwlib_context *
|
|
ucwlib_thread_context(void)
|
|
{
|
|
struct ucwlib_context *c = pthread_getspecific(ucwlib_context_key);
|
|
if (!c)
|
|
{
|
|
c = xmalloc_zero(sizeof(*c));
|
|
c->_thread_id = ucwlib_tid();
|
|
pthread_setspecific(ucwlib_context_key, c);
|
|
}
|
|
return c;
|
|
}
|
|
|
|
#endif /* CONFIG_UCW_TLS */
|
|
|
|
#else /* !CONFIG_UCW_THREADS */
|
|
|
|
struct ucwlib_context ucwlib_default_context;
|
|
|
|
#endif
|
|
|
|
#ifdef TEST
|
|
|
|
int main(void)
|
|
{
|
|
ucwlib_lock();
|
|
ucwlib_unlock();
|
|
msg(L_INFO, "tid=%d", ucwlib_thread_id(ucwlib_thread_context()));
|
|
return 0;
|
|
}
|
|
|
|
#endif
|