summaryrefslogtreecommitdiffstats
path: root/include/zix
diff options
context:
space:
mode:
authorDavid Robillard <d@drobilla.net>2022-08-18 16:57:17 -0400
committerDavid Robillard <d@drobilla.net>2022-08-18 16:57:17 -0400
commit0c001d5c7fd4c6c5d9ceeb55ba5fac1780b441af (patch)
tree7e42a1e6ac699fd7c59ad3dfcd4c7c5be2818c75 /include/zix
parentb9d58f74ea1d072e7a2d1c862dc7a1e0fe5fccb0 (diff)
downloadzix-0c001d5c7fd4c6c5d9ceeb55ba5fac1780b441af.tar.gz
zix-0c001d5c7fd4c6c5d9ceeb55ba5fac1780b441af.tar.bz2
zix-0c001d5c7fd4c6c5d9ceeb55ba5fac1780b441af.zip
Fix or remove non-portable features in thread API
Thread function return values are inconsistent between nearly every threading API out there. So, just ignore them entirely, and provide a typedef and sentinel value so user code can be portable.
Diffstat (limited to 'include/zix')
-rw-r--r--include/zix/thread.h57
1 files changed, 39 insertions, 18 deletions
diff --git a/include/zix/thread.h b/include/zix/thread.h
index c80add0..29dee44 100644
--- a/include/zix/thread.h
+++ b/include/zix/thread.h
@@ -26,47 +26,68 @@ extern "C" {
*/
#ifdef _WIN32
+# define ZIX_THREAD_RESULT 0
+# define ZIX_THREAD_FUNC __attribute__((stdcall))
+
typedef HANDLE ZixThread;
+typedef DWORD ZixThreadResult;
+
#else
+# define ZIX_THREAD_RESULT NULL
+# define ZIX_THREAD_FUNC
+
typedef pthread_t ZixThread;
+typedef void* ZixThreadResult;
#endif
/**
+ A thread function.
+
+ For portability reasons, the return type varies between platforms, and the
+ return value is ignored. Thread functions should always return
+ #ZIX_THREAD_RESULT. This allows thread functions to be used directly by the
+ system without any wrapper overhead.
+
+ "Returning" a result, and communicating with the parent thread in general,
+ can be done through the pointer argument.
+*/
+typedef ZIX_THREAD_FUNC
+ZixThreadResult (*ZixThreadFunc)(void*);
+
+/**
Initialize `thread` to a new thread.
The thread will immediately be launched, calling `function` with `arg`
as the only parameter.
*/
static inline ZixStatus
-zix_thread_create(ZixThread* thread,
- size_t stack_size,
- void* (*function)(void*),
- void* arg);
+zix_thread_create(ZixThread* thread,
+ size_t stack_size,
+ ZixThreadFunc function,
+ void* arg);
/// Join `thread` (block until `thread` exits)
static inline ZixStatus
-zix_thread_join(ZixThread thread, void** retval);
+zix_thread_join(ZixThread thread);
#ifdef _WIN32
static inline ZixStatus
-zix_thread_create(ZixThread* thread,
- size_t stack_size,
- void* (*function)(void*),
- void* arg)
+zix_thread_create(ZixThread* thread,
+ size_t stack_size,
+ ZixThreadFunc function,
+ void* arg)
{
- *thread = CreateThread(
- NULL, stack_size, (LPTHREAD_START_ROUTINE)function, arg, 0, NULL);
+ *thread = CreateThread(NULL, stack_size, function, arg, 0, NULL);
return *thread ? ZIX_STATUS_SUCCESS : ZIX_STATUS_ERROR;
}
static inline ZixStatus
-zix_thread_join(ZixThread thread, void** retval)
+zix_thread_join(ZixThread thread)
{
- (void)retval;
-
- return WaitForSingleObject(thread, INFINITE) ? ZIX_STATUS_SUCCESS
- : ZIX_STATUS_ERROR;
+ return (WaitForSingleObject(thread, INFINITE) == WAIT_OBJECT_0)
+ ? ZIX_STATUS_SUCCESS
+ : ZIX_STATUS_ERROR;
}
#else /* !defined(_WIN32) */
@@ -88,9 +109,9 @@ zix_thread_create(ZixThread* thread,
}
static inline ZixStatus
-zix_thread_join(ZixThread thread, void** retval)
+zix_thread_join(ZixThread thread)
{
- return pthread_join(thread, retval) ? ZIX_STATUS_ERROR : ZIX_STATUS_SUCCESS;
+ return pthread_join(thread, NULL) ? ZIX_STATUS_ERROR : ZIX_STATUS_SUCCESS;
}
#endif