1
0

threading-windows.c 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216
  1. /*
  2. * Copyright (c) 2023 Lain 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. #include "util/platform.h"
  19. #define WIN32_LEAN_AND_MEAN
  20. #include <windows.h>
  21. #ifdef __MINGW32__
  22. #include <excpt.h>
  23. #ifndef TRYLEVEL_NONE
  24. #ifndef __MINGW64__
  25. #define NO_SEH_MINGW
  26. #endif
  27. #ifndef __try
  28. #define __try
  29. #endif
  30. #ifndef __except
  31. #define __except (x) if (0)
  32. #endif
  33. #endif
  34. #endif
  35. int os_event_init(os_event_t **event, enum os_event_type type)
  36. {
  37. HANDLE handle;
  38. handle = CreateEvent(NULL, (type == OS_EVENT_TYPE_MANUAL), FALSE, NULL);
  39. if (!handle)
  40. return -1;
  41. *event = (os_event_t *)handle;
  42. return 0;
  43. }
  44. void os_event_destroy(os_event_t *event)
  45. {
  46. if (event)
  47. CloseHandle((HANDLE)event);
  48. }
  49. int os_event_wait(os_event_t *event)
  50. {
  51. DWORD code;
  52. if (!event)
  53. return EINVAL;
  54. code = WaitForSingleObject((HANDLE)event, INFINITE);
  55. if (code != WAIT_OBJECT_0)
  56. return EINVAL;
  57. return 0;
  58. }
  59. int os_event_timedwait(os_event_t *event, unsigned long milliseconds)
  60. {
  61. DWORD code;
  62. if (!event)
  63. return EINVAL;
  64. code = WaitForSingleObject((HANDLE)event, milliseconds);
  65. if (code == WAIT_TIMEOUT)
  66. return ETIMEDOUT;
  67. else if (code != WAIT_OBJECT_0)
  68. return EINVAL;
  69. return 0;
  70. }
  71. int os_event_try(os_event_t *event)
  72. {
  73. DWORD code;
  74. if (!event)
  75. return EINVAL;
  76. code = WaitForSingleObject((HANDLE)event, 0);
  77. if (code == WAIT_TIMEOUT)
  78. return EAGAIN;
  79. else if (code != WAIT_OBJECT_0)
  80. return EINVAL;
  81. return 0;
  82. }
  83. int os_event_signal(os_event_t *event)
  84. {
  85. if (!event)
  86. return EINVAL;
  87. if (!SetEvent((HANDLE)event))
  88. return EINVAL;
  89. return 0;
  90. }
  91. void os_event_reset(os_event_t *event)
  92. {
  93. if (!event)
  94. return;
  95. ResetEvent((HANDLE)event);
  96. }
  97. int os_sem_init(os_sem_t **sem, int value)
  98. {
  99. HANDLE handle = CreateSemaphore(NULL, (LONG)value, 0x7FFFFFFF, NULL);
  100. if (!handle)
  101. return -1;
  102. *sem = (os_sem_t *)handle;
  103. return 0;
  104. }
  105. void os_sem_destroy(os_sem_t *sem)
  106. {
  107. if (sem)
  108. CloseHandle((HANDLE)sem);
  109. }
  110. int os_sem_post(os_sem_t *sem)
  111. {
  112. if (!sem)
  113. return -1;
  114. return ReleaseSemaphore((HANDLE)sem, 1, NULL) ? 0 : -1;
  115. }
  116. int os_sem_wait(os_sem_t *sem)
  117. {
  118. DWORD ret;
  119. if (!sem)
  120. return -1;
  121. ret = WaitForSingleObject((HANDLE)sem, INFINITE);
  122. return (ret == WAIT_OBJECT_0) ? 0 : -1;
  123. }
  124. #define VC_EXCEPTION 0x406D1388
  125. #pragma pack(push, 8)
  126. struct vs_threadname_info {
  127. DWORD type; /* 0x1000 */
  128. const char *name;
  129. DWORD thread_id;
  130. DWORD flags;
  131. };
  132. #pragma pack(pop)
  133. #define THREADNAME_INFO_SIZE \
  134. (sizeof(struct vs_threadname_info) / sizeof(ULONG_PTR))
  135. void os_set_thread_name(const char *name)
  136. {
  137. #ifdef __MINGW32__
  138. UNUSED_PARAMETER(name);
  139. #else
  140. struct vs_threadname_info info;
  141. info.type = 0x1000;
  142. info.name = name;
  143. info.thread_id = GetCurrentThreadId();
  144. info.flags = 0;
  145. #ifdef NO_SEH_MINGW
  146. __try1(EXCEPTION_EXECUTE_HANDLER)
  147. {
  148. #else
  149. __try {
  150. #endif
  151. RaiseException(VC_EXCEPTION, 0, THREADNAME_INFO_SIZE,
  152. (ULONG_PTR *)&info);
  153. #ifdef NO_SEH_MINGW
  154. }
  155. __except1{
  156. #else
  157. } __except (EXCEPTION_EXECUTE_HANDLER) {
  158. #endif
  159. }
  160. #endif
  161. const HMODULE hModule = LoadLibrary(L"KernelBase.dll");
  162. if (hModule) {
  163. typedef HRESULT(WINAPI * set_thread_description_t)(HANDLE,
  164. PCWSTR);
  165. const set_thread_description_t std =
  166. (set_thread_description_t)GetProcAddress(
  167. hModule, "SetThreadDescription");
  168. if (std) {
  169. wchar_t *wname;
  170. os_utf8_to_wcs_ptr(name, 0, &wname);
  171. std(GetCurrentThread(), wname);
  172. bfree(wname);
  173. }
  174. FreeLibrary(hModule);
  175. }
  176. }