threading-windows.c 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223
  1. /*
  2. * Copyright (c) 2014 Hugh Bailey <[email protected]>
  3. *
  4. * Permission to use, copy, modify, and distribute this software for any
  5. * purpose with or without fee is hereby granted, provided that the above
  6. * copyright notice and this permission notice appear in all copies.
  7. *
  8. * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
  9. * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
  10. * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
  11. * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
  12. * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
  13. * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
  14. * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  15. */
  16. #include "bmem.h"
  17. #include "threading.h"
  18. #define WIN32_LEAN_AND_MEAN
  19. #include <windows.h>
  20. #ifdef __MINGW32__
  21. #include <excpt.h>
  22. #ifndef TRYLEVEL_NONE
  23. #ifndef __MINGW64__
  24. #define NO_SEH_MINGW
  25. #endif
  26. #ifndef __try
  27. #define __try
  28. #endif
  29. #ifndef __except
  30. #define __except(x) if (0)
  31. #endif
  32. #endif
  33. #endif
  34. struct os_event_data {
  35. HANDLE handle;
  36. };
  37. struct os_sem_data {
  38. HANDLE handle;
  39. };
  40. int os_event_init(os_event_t **event, enum os_event_type type)
  41. {
  42. HANDLE handle;
  43. struct os_event_data *data;
  44. handle = CreateEvent(NULL, (type == OS_EVENT_TYPE_MANUAL), FALSE, NULL);
  45. if (!handle)
  46. return -1;
  47. data = bmalloc(sizeof(struct os_event_data));
  48. data->handle = handle;
  49. *event = data;
  50. return 0;
  51. }
  52. void os_event_destroy(os_event_t *event)
  53. {
  54. if (event) {
  55. CloseHandle(event->handle);
  56. bfree(event);
  57. }
  58. }
  59. int os_event_wait(os_event_t *event)
  60. {
  61. DWORD code;
  62. if (!event)
  63. return EINVAL;
  64. code = WaitForSingleObject(event->handle, INFINITE);
  65. if (code != WAIT_OBJECT_0)
  66. return EINVAL;
  67. return 0;
  68. }
  69. int os_event_timedwait(os_event_t *event, unsigned long milliseconds)
  70. {
  71. DWORD code;
  72. if (!event)
  73. return EINVAL;
  74. code = WaitForSingleObject(event->handle, milliseconds);
  75. if (code == WAIT_TIMEOUT)
  76. return ETIMEDOUT;
  77. else if (code != WAIT_OBJECT_0)
  78. return EINVAL;
  79. return 0;
  80. }
  81. int os_event_try(os_event_t *event)
  82. {
  83. DWORD code;
  84. if (!event)
  85. return EINVAL;
  86. code = WaitForSingleObject(event->handle, 0);
  87. if (code == WAIT_TIMEOUT)
  88. return EAGAIN;
  89. else if (code != WAIT_OBJECT_0)
  90. return EINVAL;
  91. return 0;
  92. }
  93. int os_event_signal(os_event_t *event)
  94. {
  95. if (!event)
  96. return EINVAL;
  97. if (!SetEvent(event->handle))
  98. return EINVAL;
  99. return 0;
  100. }
  101. void os_event_reset(os_event_t *event)
  102. {
  103. if (!event)
  104. return;
  105. ResetEvent(event->handle);
  106. }
  107. int os_sem_init(os_sem_t **sem, int value)
  108. {
  109. HANDLE handle = CreateSemaphore(NULL, (LONG)value, 0x7FFFFFFF, NULL);
  110. if (!handle)
  111. return -1;
  112. *sem = bzalloc(sizeof(struct os_sem_data));
  113. (*sem)->handle = handle;
  114. return 0;
  115. }
  116. void os_sem_destroy(os_sem_t *sem)
  117. {
  118. if (sem) {
  119. CloseHandle(sem->handle);
  120. bfree(sem);
  121. }
  122. }
  123. int os_sem_post(os_sem_t *sem)
  124. {
  125. if (!sem) return -1;
  126. return ReleaseSemaphore(sem->handle, 1, NULL) ? 0 : -1;
  127. }
  128. int os_sem_wait(os_sem_t *sem)
  129. {
  130. DWORD ret;
  131. if (!sem) return -1;
  132. ret = WaitForSingleObject(sem->handle, INFINITE);
  133. return (ret == WAIT_OBJECT_0) ? 0 : -1;
  134. }
  135. long os_atomic_inc_long(volatile long *val)
  136. {
  137. return InterlockedIncrement(val);
  138. }
  139. long os_atomic_dec_long(volatile long *val)
  140. {
  141. return InterlockedDecrement(val);
  142. }
  143. bool os_atomic_compare_swap_long(volatile long *val, long old_val, long new_val)
  144. {
  145. return InterlockedCompareExchange(val, new_val, old_val) == old_val;
  146. }
  147. #define VC_EXCEPTION 0x406D1388
  148. #pragma pack(push,8)
  149. struct vs_threadname_info {
  150. DWORD type; /* 0x1000 */
  151. const char *name;
  152. DWORD thread_id;
  153. DWORD flags;
  154. };
  155. #pragma pack(pop)
  156. #define THREADNAME_INFO_SIZE \
  157. (sizeof(struct vs_threadname_info) / sizeof(ULONG_PTR))
  158. void os_set_thread_name(const char *name)
  159. {
  160. #ifdef __MINGW32__
  161. UNUSED_PARAMETER(name);
  162. #else
  163. struct vs_threadname_info info;
  164. info.type = 0x1000;
  165. info.name = name;
  166. info.thread_id = GetCurrentThreadId();
  167. info.flags = 0;
  168. #ifdef NO_SEH_MINGW
  169. __try1(EXCEPTION_EXECUTE_HANDLER) {
  170. #else
  171. __try {
  172. #endif
  173. RaiseException(VC_EXCEPTION, 0, THREADNAME_INFO_SIZE,
  174. (ULONG_PTR*)&info);
  175. #ifdef NO_SEH_MINGW
  176. } __except1 {
  177. #else
  178. } __except(EXCEPTION_EXECUTE_HANDLER) {
  179. #endif
  180. }
  181. #endif
  182. }