threading-windows.c 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195
  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. int os_event_init(os_event_t **event, enum os_event_type type)
  35. {
  36. HANDLE handle;
  37. handle = CreateEvent(NULL, (type == OS_EVENT_TYPE_MANUAL), FALSE, NULL);
  38. if (!handle)
  39. return -1;
  40. *event = (os_event_t *)handle;
  41. return 0;
  42. }
  43. void os_event_destroy(os_event_t *event)
  44. {
  45. if (event)
  46. CloseHandle((HANDLE)event);
  47. }
  48. int os_event_wait(os_event_t *event)
  49. {
  50. DWORD code;
  51. if (!event)
  52. return EINVAL;
  53. code = WaitForSingleObject((HANDLE)event, INFINITE);
  54. if (code != WAIT_OBJECT_0)
  55. return EINVAL;
  56. return 0;
  57. }
  58. int os_event_timedwait(os_event_t *event, unsigned long milliseconds)
  59. {
  60. DWORD code;
  61. if (!event)
  62. return EINVAL;
  63. code = WaitForSingleObject((HANDLE)event, milliseconds);
  64. if (code == WAIT_TIMEOUT)
  65. return ETIMEDOUT;
  66. else if (code != WAIT_OBJECT_0)
  67. return EINVAL;
  68. return 0;
  69. }
  70. int os_event_try(os_event_t *event)
  71. {
  72. DWORD code;
  73. if (!event)
  74. return EINVAL;
  75. code = WaitForSingleObject((HANDLE)event, 0);
  76. if (code == WAIT_TIMEOUT)
  77. return EAGAIN;
  78. else if (code != WAIT_OBJECT_0)
  79. return EINVAL;
  80. return 0;
  81. }
  82. int os_event_signal(os_event_t *event)
  83. {
  84. if (!event)
  85. return EINVAL;
  86. if (!SetEvent((HANDLE)event))
  87. return EINVAL;
  88. return 0;
  89. }
  90. void os_event_reset(os_event_t *event)
  91. {
  92. if (!event)
  93. return;
  94. ResetEvent((HANDLE)event);
  95. }
  96. int os_sem_init(os_sem_t **sem, int value)
  97. {
  98. HANDLE handle = CreateSemaphore(NULL, (LONG)value, 0x7FFFFFFF, NULL);
  99. if (!handle)
  100. return -1;
  101. *sem = (os_sem_t *)handle;
  102. return 0;
  103. }
  104. void os_sem_destroy(os_sem_t *sem)
  105. {
  106. if (sem)
  107. CloseHandle((HANDLE)sem);
  108. }
  109. int os_sem_post(os_sem_t *sem)
  110. {
  111. if (!sem)
  112. return -1;
  113. return ReleaseSemaphore((HANDLE)sem, 1, NULL) ? 0 : -1;
  114. }
  115. int os_sem_wait(os_sem_t *sem)
  116. {
  117. DWORD ret;
  118. if (!sem)
  119. return -1;
  120. ret = WaitForSingleObject((HANDLE)sem, INFINITE);
  121. return (ret == WAIT_OBJECT_0) ? 0 : -1;
  122. }
  123. #define VC_EXCEPTION 0x406D1388
  124. #pragma pack(push, 8)
  125. struct vs_threadname_info {
  126. DWORD type; /* 0x1000 */
  127. const char *name;
  128. DWORD thread_id;
  129. DWORD flags;
  130. };
  131. #pragma pack(pop)
  132. #define THREADNAME_INFO_SIZE \
  133. (sizeof(struct vs_threadname_info) / sizeof(ULONG_PTR))
  134. void os_set_thread_name(const char *name)
  135. {
  136. #ifdef __MINGW32__
  137. UNUSED_PARAMETER(name);
  138. #else
  139. struct vs_threadname_info info;
  140. info.type = 0x1000;
  141. info.name = name;
  142. info.thread_id = GetCurrentThreadId();
  143. info.flags = 0;
  144. #ifdef NO_SEH_MINGW
  145. __try1(EXCEPTION_EXECUTE_HANDLER)
  146. {
  147. #else
  148. __try {
  149. #endif
  150. RaiseException(VC_EXCEPTION, 0, THREADNAME_INFO_SIZE,
  151. (ULONG_PTR *)&info);
  152. #ifdef NO_SEH_MINGW
  153. }
  154. __except1{
  155. #else
  156. } __except (EXCEPTION_EXECUTE_HANDLER) {
  157. #endif
  158. }
  159. #endif
  160. }