mirror of
https://github.com/bkaradzic/bx.git
synced 2026-02-18 04:53:06 +01:00
274 lines
5.6 KiB
C++
274 lines
5.6 KiB
C++
/*
|
|
* Copyright 2010-2015 Branimir Karadzic. All rights reserved.
|
|
* License: http://www.opensource.org/licenses/BSD-2-Clause
|
|
*/
|
|
|
|
#ifndef BX_THREAD_H_HEADER_GUARD
|
|
#define BX_THREAD_H_HEADER_GUARD
|
|
|
|
#if BX_PLATFORM_POSIX
|
|
# include <pthread.h>
|
|
#elif BX_PLATFORM_WINRT
|
|
using namespace Platform;
|
|
using namespace Windows::Foundation;
|
|
using namespace Windows::System::Threading;
|
|
#endif
|
|
|
|
#include "sem.h"
|
|
|
|
#if BX_CONFIG_SUPPORTS_THREADING
|
|
|
|
namespace bx
|
|
{
|
|
typedef int32_t (*ThreadFn)(void* _userData);
|
|
|
|
class Thread
|
|
{
|
|
BX_CLASS(Thread
|
|
, NO_COPY
|
|
, NO_ASSIGNMENT
|
|
);
|
|
|
|
public:
|
|
Thread()
|
|
#if BX_PLATFORM_WINDOWS|BX_PLATFORM_XBOX360|BX_PLATFORM_WINRT
|
|
: m_handle(INVALID_HANDLE_VALUE)
|
|
#elif BX_PLATFORM_POSIX
|
|
: m_handle(0)
|
|
#endif // BX_PLATFORM_
|
|
, m_fn(NULL)
|
|
, m_userData(NULL)
|
|
, m_stackSize(0)
|
|
, m_exitCode(0 /*EXIT_SUCCESS*/)
|
|
, m_running(false)
|
|
{
|
|
}
|
|
|
|
virtual ~Thread()
|
|
{
|
|
if (m_running)
|
|
{
|
|
shutdown();
|
|
}
|
|
}
|
|
|
|
void init(ThreadFn _fn, void* _userData = NULL, uint32_t _stackSize = 0)
|
|
{
|
|
BX_CHECK(!m_running, "Already running!");
|
|
|
|
m_fn = _fn;
|
|
m_userData = _userData;
|
|
m_stackSize = _stackSize;
|
|
m_running = true;
|
|
|
|
#if BX_PLATFORM_WINDOWS|BX_PLATFORM_XBOX360
|
|
m_handle = CreateThread(NULL
|
|
, m_stackSize
|
|
, threadFunc
|
|
, this
|
|
, 0
|
|
, NULL
|
|
);
|
|
#elif BX_PLATFORM_WINRT
|
|
m_handle = CreateEventEx(nullptr, nullptr, CREATE_EVENT_MANUAL_RESET, EVENT_ALL_ACCESS);
|
|
auto workItemHandler = ref new WorkItemHandler([=](IAsyncAction^)
|
|
{
|
|
m_exitCode = threadFunc(this);
|
|
SetEvent(m_handle);
|
|
}, CallbackContext::Any);
|
|
|
|
ThreadPool::RunAsync(workItemHandler, WorkItemPriority::Normal, WorkItemOptions::TimeSliced);
|
|
#elif BX_PLATFORM_POSIX
|
|
int result;
|
|
BX_UNUSED(result);
|
|
|
|
pthread_attr_t attr;
|
|
result = pthread_attr_init(&attr);
|
|
BX_CHECK(0 == result, "pthread_attr_init failed! %d", result);
|
|
|
|
if (0 != m_stackSize)
|
|
{
|
|
result = pthread_attr_setstacksize(&attr, m_stackSize);
|
|
BX_CHECK(0 == result, "pthread_attr_setstacksize failed! %d", result);
|
|
}
|
|
|
|
// sched_param sched;
|
|
// sched.sched_priority = 0;
|
|
// result = pthread_attr_setschedparam(&attr, &sched);
|
|
// BX_CHECK(0 == result, "pthread_attr_setschedparam failed! %d", result);
|
|
|
|
result = pthread_create(&m_handle, &attr, &threadFunc, this);
|
|
BX_CHECK(0 == result, "pthread_attr_setschedparam failed! %d", result);
|
|
#endif // BX_PLATFORM_
|
|
|
|
m_sem.wait();
|
|
}
|
|
|
|
void shutdown()
|
|
{
|
|
BX_CHECK(m_running, "Not running!");
|
|
#if BX_PLATFORM_WINDOWS|BX_PLATFORM_XBOX360
|
|
WaitForSingleObject(m_handle, INFINITE);
|
|
GetExitCodeThread(m_handle, (DWORD*)&m_exitCode);
|
|
CloseHandle(m_handle);
|
|
m_handle = INVALID_HANDLE_VALUE;
|
|
#elif BX_PLATFORM_WINRT
|
|
WaitForSingleObjectEx(m_handle, INFINITE, FALSE);
|
|
CloseHandle(m_handle);
|
|
m_handle = INVALID_HANDLE_VALUE;
|
|
#elif BX_PLATFORM_POSIX
|
|
union
|
|
{
|
|
void* ptr;
|
|
int32_t i;
|
|
} cast;
|
|
pthread_join(m_handle, &cast.ptr);
|
|
m_exitCode = cast.i;
|
|
m_handle = 0;
|
|
#endif // BX_PLATFORM_
|
|
m_running = false;
|
|
}
|
|
|
|
bool isRunning() const
|
|
{
|
|
return m_running;
|
|
}
|
|
|
|
int32_t getExitCode() const
|
|
{
|
|
return m_exitCode;
|
|
}
|
|
|
|
void setThreadName(const char* name)
|
|
{
|
|
if (name != NULL)
|
|
{
|
|
strncpy(m_name, name, BX_CONFIG_THREAD_NAME_LENGTH);
|
|
}
|
|
else
|
|
{
|
|
memset(m_name, 0, BX_CONFIG_THREAD_NAME_LENGTH);
|
|
}
|
|
|
|
#if BX_PLATFORM_WINDOWS
|
|
#elif BX_PLATFORM_OSX|BX_PLATFORM_IOS
|
|
strncpy(m_name, name, BX_CONFIG_THREAD_NAME_LENGTH);
|
|
pthread_setname_np(m_name);
|
|
#elif BX_PLATFORM_POSIX
|
|
strncpy(m_name, name, BX_CONFIG_THREAD_NAME_LENGTH);
|
|
pthread_setname_np(m_handle, m_name);
|
|
#endif // BX_PLATFORM_
|
|
}
|
|
|
|
private:
|
|
int32_t entry()
|
|
{
|
|
m_sem.post();
|
|
return m_fn(m_userData);
|
|
}
|
|
|
|
#if BX_PLATFORM_WINDOWS|BX_PLATFORM_XBOX360|BX_PLATFORM_WINRT
|
|
static DWORD WINAPI threadFunc(LPVOID _arg)
|
|
{
|
|
Thread* thread = (Thread*)_arg;
|
|
int32_t result = thread->entry();
|
|
return result;
|
|
}
|
|
#else
|
|
static void* threadFunc(void* _arg)
|
|
{
|
|
Thread* thread = (Thread*)_arg;
|
|
union
|
|
{
|
|
void* ptr;
|
|
int32_t i;
|
|
} cast;
|
|
cast.i = thread->entry();
|
|
return cast.ptr;
|
|
}
|
|
#endif // BX_PLATFORM_
|
|
|
|
#if BX_PLATFORM_WINDOWS|BX_PLATFORM_XBOX360|BX_PLATFORM_WINRT
|
|
HANDLE m_handle;
|
|
#elif BX_PLATFORM_POSIX
|
|
pthread_t m_handle;
|
|
#endif // BX_PLATFORM_
|
|
|
|
ThreadFn m_fn;
|
|
void* m_userData;
|
|
Semaphore m_sem;
|
|
uint32_t m_stackSize;
|
|
int32_t m_exitCode;
|
|
bool m_running;
|
|
char m_name[BX_CONFIG_THREAD_NAME_LENGTH];
|
|
};
|
|
|
|
#if BX_PLATFORM_WINDOWS
|
|
class TlsData
|
|
{
|
|
public:
|
|
TlsData()
|
|
{
|
|
m_id = TlsAlloc();
|
|
BX_CHECK(TLS_OUT_OF_INDEXES != m_id, "Failed to allocated TLS index (err: 0x%08x).", GetLastError() );
|
|
}
|
|
|
|
~TlsData()
|
|
{
|
|
BOOL result = TlsFree(m_id);
|
|
BX_CHECK(0 != result, "Failed to free TLS index (err: 0x%08x).", GetLastError() ); BX_UNUSED(result);
|
|
}
|
|
|
|
void* get() const
|
|
{
|
|
return TlsGetValue(m_id);
|
|
}
|
|
|
|
void set(void* _ptr)
|
|
{
|
|
TlsSetValue(m_id, _ptr);
|
|
}
|
|
|
|
private:
|
|
uint32_t m_id;
|
|
};
|
|
|
|
#elif !(BX_PLATFORM_WINRT)
|
|
|
|
class TlsData
|
|
{
|
|
public:
|
|
TlsData()
|
|
{
|
|
int result = pthread_key_create(&m_id, NULL);
|
|
BX_CHECK(0 == result, "pthread_key_create failed %d.", result); BX_UNUSED(result);
|
|
}
|
|
|
|
~TlsData()
|
|
{
|
|
int result = pthread_key_delete(m_id);
|
|
BX_CHECK(0 == result, "pthread_key_delete failed %d.", result); BX_UNUSED(result);
|
|
}
|
|
|
|
void* get() const
|
|
{
|
|
return pthread_getspecific(m_id);
|
|
}
|
|
|
|
void set(void* _ptr)
|
|
{
|
|
int result = pthread_setspecific(m_id, _ptr);
|
|
BX_CHECK(0 == result, "pthread_setspecific failed %d.", result); BX_UNUSED(result);
|
|
}
|
|
|
|
private:
|
|
pthread_key_t m_id;
|
|
};
|
|
#endif // BX_PLATFORM_WINDOWS
|
|
|
|
} // namespace bx
|
|
|
|
#endif // BX_CONFIG_SUPPORTS_THREADING
|
|
|
|
#endif // BX_THREAD_H_HEADER_GUARD
|