Add initial implementation of threads/mutexes/semaphores/cvs/spinlocks

Added initial implementation (or wrapper) of the threading library.
It's rather basic, but should work for most of the tasks.

Unfortunately, spinlock implementation relies on GCC/Clang compiler
built-ins (or in-worst-case-scenario on Win32 - InterlockExchange).
In the future, I should revisit this code and fix/reimplement some stuff
(or add support for Windows XP).
This commit is contained in:
2025-03-02 23:18:23 +03:00
parent 2ca6a3e316
commit d403d41f2c
21 changed files with 1492 additions and 9 deletions

View File

@@ -0,0 +1,98 @@
#include "Thread.h"
#include <BH/Thread.h>
#include <limits.h>
#include <stdlib.h>
struct BH_ThreadContext
{
BH_ThreadCallback callback;
void *data;
};
static void *BH_ThreadRun(void *context)
{
BH_ThreadCallback callback;
void *data, *result;
callback = ((struct BH_ThreadContext *)context)->callback;
data = ((struct BH_ThreadContext *)context)->data;
free(context);
callback(data);
BH_TssCleanup();
pthread_exit(0);
}
static int BH_ThreadInit(BH_Thread *thread,
size_t stack,
BH_ThreadCallback callback,
void *data)
{
struct BH_ThreadContext *context;
pthread_attr_t attributes;
int result;
context = malloc(sizeof(*context));
if (!context)
return BH_ERROR;
context->callback = callback;
context->data = data;
/* Create thread with specified stack size */
pthread_attr_init(&attributes);
if (!stack)
result = pthread_create(&thread->handle, NULL, BH_ThreadRun, context);
else
{
if (stack < PTHREAD_STACK_MIN)
stack = PTHREAD_STACK_MIN;
pthread_attr_setstacksize(&attributes, stack);
result = pthread_create(&thread->handle, &attributes, BH_ThreadRun, context);
}
pthread_attr_destroy(&attributes);
return result;
}
BH_Thread *BH_ThreadNew(size_t stack,
BH_ThreadCallback callback,
void *data)
{
BH_Thread *thread;
thread = malloc(sizeof(BH_Thread));
if (thread && BH_ThreadInit(thread, stack, callback, data))
{
free(thread);
return NULL;
}
return thread;
}
int BH_ThreadJoin(BH_Thread *thread)
{
if (pthread_join(thread->handle, NULL))
return BH_ERROR;
free(thread);
return BH_OK;
}
int BH_ThreadDetach(BH_Thread *thread)
{
if (pthread_detach(thread->handle))
return BH_ERROR;
free(thread);
return BH_OK;
}