2014-02-26 22:43:31 -08:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2014 Hugh Bailey <obs.jim@gmail.com>
|
|
|
|
*
|
|
|
|
* Permission to use, copy, modify, and distribute this software for any
|
|
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
|
|
* copyright notice and this permission notice appear in all copies.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
|
|
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
|
|
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
|
|
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
|
|
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
|
|
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
|
|
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "bmem.h"
|
|
|
|
#include "threading.h"
|
|
|
|
|
|
|
|
#define WIN32_LEAN_AND_MEAN
|
|
|
|
#include <windows.h>
|
|
|
|
|
2015-02-08 09:23:02 -08:00
|
|
|
#ifdef __MINGW32__
|
|
|
|
#include <excpt.h>
|
|
|
|
#ifndef TRYLEVEL_NONE
|
|
|
|
#ifndef __MINGW64__
|
|
|
|
#define NO_SEH_MINGW
|
|
|
|
#endif
|
|
|
|
#ifndef __try
|
|
|
|
#define __try
|
|
|
|
#endif
|
|
|
|
#ifndef __except
|
|
|
|
#define __except(x) if (0)
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
2014-03-10 19:04:00 -07:00
|
|
|
struct os_event_data {
|
2014-02-26 22:43:31 -08:00
|
|
|
HANDLE handle;
|
|
|
|
};
|
|
|
|
|
2014-03-10 19:04:00 -07:00
|
|
|
struct os_sem_data {
|
|
|
|
HANDLE handle;
|
|
|
|
};
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
int os_event_init(os_event_t **event, enum os_event_type type)
|
2014-02-26 22:43:31 -08:00
|
|
|
{
|
|
|
|
HANDLE handle;
|
2014-03-10 19:04:00 -07:00
|
|
|
struct os_event_data *data;
|
2014-02-26 22:43:31 -08:00
|
|
|
|
2014-03-10 19:04:00 -07:00
|
|
|
handle = CreateEvent(NULL, (type == OS_EVENT_TYPE_MANUAL), FALSE, NULL);
|
2014-02-26 22:43:31 -08:00
|
|
|
if (!handle)
|
|
|
|
return -1;
|
|
|
|
|
2014-03-10 19:04:00 -07:00
|
|
|
data = bmalloc(sizeof(struct os_event_data));
|
2014-02-26 22:43:31 -08:00
|
|
|
data->handle = handle;
|
|
|
|
|
|
|
|
*event = data;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
void os_event_destroy(os_event_t *event)
|
2014-02-26 22:43:31 -08:00
|
|
|
{
|
|
|
|
if (event) {
|
|
|
|
CloseHandle(event->handle);
|
|
|
|
bfree(event);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
int os_event_wait(os_event_t *event)
|
2014-02-26 22:43:31 -08:00
|
|
|
{
|
|
|
|
DWORD code;
|
|
|
|
|
|
|
|
if (!event)
|
|
|
|
return EINVAL;
|
|
|
|
|
|
|
|
code = WaitForSingleObject(event->handle, INFINITE);
|
|
|
|
if (code != WAIT_OBJECT_0)
|
|
|
|
return EINVAL;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
int os_event_timedwait(os_event_t *event, unsigned long milliseconds)
|
2014-02-26 22:43:31 -08:00
|
|
|
{
|
|
|
|
DWORD code;
|
|
|
|
|
|
|
|
if (!event)
|
|
|
|
return EINVAL;
|
|
|
|
|
|
|
|
code = WaitForSingleObject(event->handle, milliseconds);
|
|
|
|
if (code == WAIT_TIMEOUT)
|
|
|
|
return ETIMEDOUT;
|
|
|
|
else if (code != WAIT_OBJECT_0)
|
|
|
|
return EINVAL;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
int os_event_try(os_event_t *event)
|
2014-02-26 22:43:31 -08:00
|
|
|
{
|
|
|
|
DWORD code;
|
|
|
|
|
|
|
|
if (!event)
|
|
|
|
return EINVAL;
|
|
|
|
|
|
|
|
code = WaitForSingleObject(event->handle, 0);
|
|
|
|
if (code == WAIT_TIMEOUT)
|
|
|
|
return EAGAIN;
|
|
|
|
else if (code != WAIT_OBJECT_0)
|
|
|
|
return EINVAL;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
int os_event_signal(os_event_t *event)
|
2014-02-26 22:43:31 -08:00
|
|
|
{
|
|
|
|
if (!event)
|
|
|
|
return EINVAL;
|
|
|
|
|
|
|
|
if (!SetEvent(event->handle))
|
|
|
|
return EINVAL;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
void os_event_reset(os_event_t *event)
|
2014-02-26 22:43:31 -08:00
|
|
|
{
|
|
|
|
if (!event)
|
|
|
|
return;
|
|
|
|
|
|
|
|
ResetEvent(event->handle);
|
|
|
|
}
|
2014-03-10 19:04:00 -07:00
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
int os_sem_init(os_sem_t **sem, int value)
|
2014-03-10 19:04:00 -07:00
|
|
|
{
|
|
|
|
HANDLE handle = CreateSemaphore(NULL, (LONG)value, 0x7FFFFFFF, NULL);
|
|
|
|
if (!handle)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
*sem = bzalloc(sizeof(struct os_sem_data));
|
|
|
|
(*sem)->handle = handle;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
void os_sem_destroy(os_sem_t *sem)
|
2014-03-10 19:04:00 -07:00
|
|
|
{
|
|
|
|
if (sem) {
|
|
|
|
CloseHandle(sem->handle);
|
|
|
|
bfree(sem);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
int os_sem_post(os_sem_t *sem)
|
2014-03-10 19:04:00 -07:00
|
|
|
{
|
|
|
|
if (!sem) return -1;
|
|
|
|
return ReleaseSemaphore(sem->handle, 1, NULL) ? 0 : -1;
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
int os_sem_wait(os_sem_t *sem)
|
2014-03-10 19:04:00 -07:00
|
|
|
{
|
|
|
|
DWORD ret;
|
|
|
|
|
|
|
|
if (!sem) return -1;
|
|
|
|
ret = WaitForSingleObject(sem->handle, INFINITE);
|
|
|
|
return (ret == WAIT_OBJECT_0) ? 0 : -1;
|
|
|
|
}
|
2014-03-16 18:26:46 -07:00
|
|
|
|
|
|
|
long os_atomic_inc_long(volatile long *val)
|
|
|
|
{
|
|
|
|
return InterlockedIncrement(val);
|
|
|
|
}
|
|
|
|
|
|
|
|
long os_atomic_dec_long(volatile long *val)
|
|
|
|
{
|
|
|
|
return InterlockedDecrement(val);
|
|
|
|
}
|
2015-01-02 05:35:04 -08:00
|
|
|
|
2015-05-03 11:40:33 -07:00
|
|
|
bool os_atomic_compare_swap_long(volatile long *val, long old_val, long new_val)
|
|
|
|
{
|
|
|
|
return InterlockedCompareExchange(val, new_val, old_val) == old_val;
|
|
|
|
}
|
|
|
|
|
2015-01-02 05:35:04 -08:00
|
|
|
#define VC_EXCEPTION 0x406D1388
|
|
|
|
|
|
|
|
#pragma pack(push,8)
|
|
|
|
struct vs_threadname_info {
|
|
|
|
DWORD type; /* 0x1000 */
|
|
|
|
const char *name;
|
|
|
|
DWORD thread_id;
|
|
|
|
DWORD flags;
|
|
|
|
};
|
|
|
|
#pragma pack(pop)
|
|
|
|
|
|
|
|
#define THREADNAME_INFO_SIZE \
|
|
|
|
(sizeof(struct vs_threadname_info) / sizeof(ULONG_PTR))
|
|
|
|
|
|
|
|
void os_set_thread_name(const char *name)
|
|
|
|
{
|
2015-02-06 16:40:26 -08:00
|
|
|
#ifdef __MINGW32__
|
|
|
|
UNUSED_PARAMETER(name);
|
|
|
|
#else
|
2015-01-02 05:35:04 -08:00
|
|
|
struct vs_threadname_info info;
|
|
|
|
info.type = 0x1000;
|
|
|
|
info.name = name;
|
|
|
|
info.thread_id = GetCurrentThreadId();
|
|
|
|
info.flags = 0;
|
|
|
|
|
2015-02-08 09:23:02 -08:00
|
|
|
#ifdef NO_SEH_MINGW
|
|
|
|
__try1(EXCEPTION_EXECUTE_HANDLER) {
|
|
|
|
#else
|
2015-01-02 05:35:04 -08:00
|
|
|
__try {
|
2015-02-08 09:23:02 -08:00
|
|
|
#endif
|
2015-01-02 05:35:04 -08:00
|
|
|
RaiseException(VC_EXCEPTION, 0, THREADNAME_INFO_SIZE,
|
|
|
|
(ULONG_PTR*)&info);
|
2015-02-08 09:23:02 -08:00
|
|
|
#ifdef NO_SEH_MINGW
|
|
|
|
} __except1 {
|
|
|
|
#else
|
2015-01-02 05:35:04 -08:00
|
|
|
} __except(EXCEPTION_EXECUTE_HANDLER) {
|
2015-02-08 09:23:02 -08:00
|
|
|
#endif
|
2015-01-02 05:35:04 -08:00
|
|
|
}
|
2015-02-06 16:40:26 -08:00
|
|
|
#endif
|
2015-01-02 05:35:04 -08:00
|
|
|
}
|