blob: ec9f41684c2603764f363e7495e95603409bf14b [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
//
// Enable debug for asserts in this module regardless of what the project-wide
// setting is.
//
#undef NDEBUG
#include <qpid/dispatch/threading.h>
#include <qpid/dispatch/ctools.h>
#include <stdio.h>
#include <stdint.h>
#include <pthread.h>
#include <assert.h>
struct sys_mutex_t {
pthread_mutex_t mutex;
};
sys_mutex_t *sys_mutex(void)
{
sys_mutex_t *mutex = 0;
NEW_CACHE_ALIGNED(sys_mutex_t, mutex);
assert(mutex != 0);
pthread_mutex_init(&(mutex->mutex), 0);
return mutex;
}
void sys_mutex_free(sys_mutex_t *mutex)
{
pthread_mutex_destroy(&(mutex->mutex));
free(mutex);
}
void sys_mutex_lock(sys_mutex_t *mutex)
{
int result = pthread_mutex_lock(&(mutex->mutex));
assert(result == 0);
}
void sys_mutex_unlock(sys_mutex_t *mutex)
{
int result = pthread_mutex_unlock(&(mutex->mutex));
assert(result == 0);
}
struct sys_cond_t {
pthread_cond_t cond;
};
sys_cond_t *sys_cond(void)
{
sys_cond_t *cond = 0;
NEW_CACHE_ALIGNED(sys_cond_t, cond);
pthread_cond_init(&(cond->cond), 0);
return cond;
}
void sys_cond_free(sys_cond_t *cond)
{
pthread_cond_destroy(&(cond->cond));
free(cond);
}
void sys_cond_wait(sys_cond_t *cond, sys_mutex_t *held_mutex)
{
int result = pthread_cond_wait(&(cond->cond), &(held_mutex->mutex));
assert(result == 0);
}
void sys_cond_signal(sys_cond_t *cond)
{
int result = pthread_cond_signal(&(cond->cond));
assert(result == 0);
}
void sys_cond_signal_all(sys_cond_t *cond)
{
int result = pthread_cond_broadcast(&(cond->cond));
assert(result == 0);
}
struct sys_rwlock_t {
pthread_rwlock_t lock;
};
sys_rwlock_t *sys_rwlock(void)
{
sys_rwlock_t *lock = NEW(sys_rwlock_t);
pthread_rwlock_init(&(lock->lock), 0);
return lock;
}
void sys_rwlock_free(sys_rwlock_t *lock)
{
pthread_rwlock_destroy(&(lock->lock));
free(lock);
}
void sys_rwlock_wrlock(sys_rwlock_t *lock)
{
int result = pthread_rwlock_wrlock(&(lock->lock));
assert(result == 0);
}
void sys_rwlock_rdlock(sys_rwlock_t *lock)
{
int result = pthread_rwlock_rdlock(&(lock->lock));
assert(result == 0);
}
void sys_rwlock_unlock(sys_rwlock_t *lock)
{
int result = pthread_rwlock_unlock(&(lock->lock));
assert(result == 0);
}
struct sys_thread_t {
pthread_t thread;
void *(*f)(void *);
void *arg;
};
// initialize the per-thread _self to a non-zero value. This dummy value will
// be returned when sys_thread_self() is called from the process's main thread
// of execution (which is not a pthread). Using a non-zero value provides a
// way to distinguish a thread id from a zero (unset) value.
//
static sys_thread_t _main_thread_id;
static __thread sys_thread_t *_self = &_main_thread_id;
// bootstrap _self before calling thread's main function
//
static void *_thread_init(void *arg)
{
_self = (sys_thread_t*) arg;
return _self->f(_self->arg);
}
sys_thread_t *sys_thread(void *(*run_function) (void *), void *arg)
{
sys_thread_t *thread = NEW(sys_thread_t);
thread->f = run_function;
thread->arg = arg;
pthread_create(&(thread->thread), 0, _thread_init, (void*) thread);
return thread;
}
sys_thread_t *sys_thread_self()
{
return _self;
}
void sys_thread_free(sys_thread_t *thread)
{
assert(thread != &_main_thread_id);
free(thread);
}
void sys_thread_join(sys_thread_t *thread)
{
assert(thread != &_main_thread_id);
pthread_join(thread->thread, 0);
}