Blame include/posix/threads.h

Packit b1f7ae
/*
Packit b1f7ae
 * Copyright (c) 2014-2017, Intel Corporation
Packit b1f7ae
 *
Packit b1f7ae
 * Redistribution and use in source and binary forms, with or without
Packit b1f7ae
 * modification, are permitted provided that the following conditions are met:
Packit b1f7ae
 *
Packit b1f7ae
 *  * Redistributions of source code must retain the above copyright notice,
Packit b1f7ae
 *    this list of conditions and the following disclaimer.
Packit b1f7ae
 *  * Redistributions in binary form must reproduce the above copyright notice,
Packit b1f7ae
 *    this list of conditions and the following disclaimer in the documentation
Packit b1f7ae
 *    and/or other materials provided with the distribution.
Packit b1f7ae
 *  * Neither the name of Intel Corporation nor the names of its contributors
Packit b1f7ae
 *    may be used to endorse or promote products derived from this software
Packit b1f7ae
 *    without specific prior written permission.
Packit b1f7ae
 *
Packit b1f7ae
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
Packit b1f7ae
 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
Packit b1f7ae
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
Packit b1f7ae
 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
Packit b1f7ae
 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
Packit b1f7ae
 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
Packit b1f7ae
 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
Packit b1f7ae
 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
Packit b1f7ae
 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
Packit b1f7ae
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
Packit b1f7ae
 * POSSIBILITY OF SUCH DAMAGE.
Packit b1f7ae
 *
Packit b1f7ae
 *
Packit b1f7ae
 * It looks like there is still no support for C11's threads.h.
Packit b1f7ae
 *
Packit b1f7ae
 * We implement the few features we actually need hoping that this file will
Packit b1f7ae
 * soon go away.
Packit b1f7ae
 */
Packit b1f7ae
Packit b1f7ae
#ifndef THREADS_H
Packit b1f7ae
#define THREADS_H
Packit b1f7ae
Packit b1f7ae
#include <pthread.h>
Packit b1f7ae
Packit b1f7ae
#ifndef PTHREAD_MUTEX_NORMAL
Packit b1f7ae
#  define PTHREAD_MUTEX_NORMAL PTHREAD_MUTEX_TIMED_NP
Packit b1f7ae
#endif
Packit b1f7ae
Packit b1f7ae
#include <stdint.h>
Packit b1f7ae
#include <stdlib.h>
Packit b1f7ae
Packit b1f7ae
enum {
Packit b1f7ae
	thrd_success	= 1,
Packit b1f7ae
	thrd_error
Packit b1f7ae
};
Packit b1f7ae
Packit b1f7ae
struct pt_thread {
Packit b1f7ae
	pthread_t thread;
Packit b1f7ae
};
Packit b1f7ae
typedef struct pt_thread thrd_t;
Packit b1f7ae
Packit b1f7ae
typedef int (*thrd_start_t)(void *);
Packit b1f7ae
Packit b1f7ae
Packit b1f7ae
struct thrd_args {
Packit b1f7ae
	thrd_start_t fun;
Packit b1f7ae
	void *arg;
Packit b1f7ae
};
Packit b1f7ae
Packit b1f7ae
static void *thrd_routine(void *arg)
Packit b1f7ae
{
Packit b1f7ae
	struct thrd_args *args;
Packit b1f7ae
	int result;
Packit b1f7ae
Packit b1f7ae
	args = arg;
Packit b1f7ae
	if (!args)
Packit b1f7ae
		return (void *) (intptr_t) -1;
Packit b1f7ae
Packit b1f7ae
	result = -1;
Packit b1f7ae
	if (args->fun)
Packit b1f7ae
		result = args->fun(args->arg);
Packit b1f7ae
Packit b1f7ae
	free(args);
Packit b1f7ae
Packit b1f7ae
	return (void *) (intptr_t) result;
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
static inline int thrd_create(thrd_t *thrd, thrd_start_t fun, void *arg)
Packit b1f7ae
{
Packit b1f7ae
	struct thrd_args *args;
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	if (!thrd || !fun)
Packit b1f7ae
		return thrd_error;
Packit b1f7ae
Packit b1f7ae
	args = malloc(sizeof(*args));
Packit b1f7ae
	if (!args)
Packit b1f7ae
		return thrd_error;
Packit b1f7ae
Packit b1f7ae
	args->fun = fun;
Packit b1f7ae
	args->arg = arg;
Packit b1f7ae
Packit b1f7ae
	errcode = pthread_create(&thrd->thread, NULL, thrd_routine, args);
Packit b1f7ae
	if (errcode) {
Packit b1f7ae
		free(args);
Packit b1f7ae
		return thrd_error;
Packit b1f7ae
	}
Packit b1f7ae
Packit b1f7ae
	return thrd_success;
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
static inline int thrd_join(thrd_t *thrd, int *res)
Packit b1f7ae
{
Packit b1f7ae
	void *result;
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	if (!thrd)
Packit b1f7ae
		return thrd_error;
Packit b1f7ae
Packit b1f7ae
	errcode = pthread_join(thrd->thread, &result);
Packit b1f7ae
	if (errcode)
Packit b1f7ae
		return thrd_error;
Packit b1f7ae
Packit b1f7ae
	if (res)
Packit b1f7ae
		*res = (int) (intptr_t) result;
Packit b1f7ae
Packit b1f7ae
	return thrd_success;
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
Packit b1f7ae
struct pt_mutex {
Packit b1f7ae
	pthread_mutex_t mutex;
Packit b1f7ae
};
Packit b1f7ae
typedef struct pt_mutex mtx_t;
Packit b1f7ae
Packit b1f7ae
enum {
Packit b1f7ae
	mtx_plain = PTHREAD_MUTEX_NORMAL
Packit b1f7ae
};
Packit b1f7ae
Packit b1f7ae
static inline int mtx_init(mtx_t *mtx, int type)
Packit b1f7ae
{
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	if (!mtx || type != mtx_plain)
Packit b1f7ae
		return thrd_error;
Packit b1f7ae
Packit b1f7ae
	errcode = pthread_mutex_init(&mtx->mutex, NULL);
Packit b1f7ae
	if (errcode)
Packit b1f7ae
		return thrd_error;
Packit b1f7ae
Packit b1f7ae
	return thrd_success;
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
static inline void mtx_destroy(mtx_t *mtx)
Packit b1f7ae
{
Packit b1f7ae
	if (mtx)
Packit b1f7ae
		(void) pthread_mutex_destroy(&mtx->mutex);
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
static inline int mtx_lock(mtx_t *mtx)
Packit b1f7ae
{
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	if (!mtx)
Packit b1f7ae
		return thrd_error;
Packit b1f7ae
Packit b1f7ae
	errcode = pthread_mutex_lock(&mtx->mutex);
Packit b1f7ae
	if (errcode)
Packit b1f7ae
		return thrd_error;
Packit b1f7ae
Packit b1f7ae
	return thrd_success;
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
static inline int mtx_unlock(mtx_t *mtx)
Packit b1f7ae
{
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	if (!mtx)
Packit b1f7ae
		return thrd_error;
Packit b1f7ae
Packit b1f7ae
	errcode = pthread_mutex_unlock(&mtx->mutex);
Packit b1f7ae
	if (errcode)
Packit b1f7ae
		return thrd_error;
Packit b1f7ae
Packit b1f7ae
	return thrd_success;
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
Packit b1f7ae
struct pt_cond {
Packit b1f7ae
	pthread_cond_t cond;
Packit b1f7ae
};
Packit b1f7ae
typedef struct pt_cond cnd_t;
Packit b1f7ae
Packit b1f7ae
static inline int cnd_init(cnd_t *cnd)
Packit b1f7ae
{
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	if (!cnd)
Packit b1f7ae
		return thrd_error;
Packit b1f7ae
Packit b1f7ae
	errcode = pthread_cond_init(&cnd->cond, NULL);
Packit b1f7ae
	if (errcode)
Packit b1f7ae
		return thrd_error;
Packit b1f7ae
Packit b1f7ae
	return thrd_success;
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
static inline int cnd_destroy(cnd_t *cnd)
Packit b1f7ae
{
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	if (!cnd)
Packit b1f7ae
		return thrd_error;
Packit b1f7ae
Packit b1f7ae
	errcode = pthread_cond_destroy(&cnd->cond);
Packit b1f7ae
	if (errcode)
Packit b1f7ae
		return thrd_error;
Packit b1f7ae
Packit b1f7ae
	return thrd_success;
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
static inline int cnd_signal(cnd_t *cnd)
Packit b1f7ae
{
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	if (!cnd)
Packit b1f7ae
		return thrd_error;
Packit b1f7ae
Packit b1f7ae
	errcode = pthread_cond_signal(&cnd->cond);
Packit b1f7ae
	if (errcode)
Packit b1f7ae
		return thrd_error;
Packit b1f7ae
Packit b1f7ae
	return thrd_success;
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
static inline int cnd_broadcast(cnd_t *cnd)
Packit b1f7ae
{
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	if (!cnd)
Packit b1f7ae
		return thrd_error;
Packit b1f7ae
Packit b1f7ae
	errcode = pthread_cond_broadcast(&cnd->cond);
Packit b1f7ae
	if (errcode)
Packit b1f7ae
		return thrd_error;
Packit b1f7ae
Packit b1f7ae
	return thrd_success;
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
static inline int cnd_wait(cnd_t *cnd, mtx_t *mtx)
Packit b1f7ae
{
Packit b1f7ae
	int errcode;
Packit b1f7ae
Packit b1f7ae
	if (!cnd || !mtx)
Packit b1f7ae
		return thrd_error;
Packit b1f7ae
Packit b1f7ae
	errcode = pthread_cond_wait(&cnd->cond, &mtx->mutex);
Packit b1f7ae
	if (errcode)
Packit b1f7ae
		return thrd_error;
Packit b1f7ae
Packit b1f7ae
	return thrd_success;
Packit b1f7ae
}
Packit b1f7ae
Packit b1f7ae
#endif /* THREADS_H */