threading-windows.c 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191
  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) return -1;
  112. return ReleaseSemaphore((HANDLE)sem, 1, NULL) ? 0 : -1;
  113. }
  114. int os_sem_wait(os_sem_t *sem)
  115. {
  116. DWORD ret;
  117. if (!sem) return -1;
  118. ret = WaitForSingleObject((HANDLE)sem, INFINITE);
  119. return (ret == WAIT_OBJECT_0) ? 0 : -1;
  120. }
  121. #define VC_EXCEPTION 0x406D1388
  122. #pragma pack(push,8)
  123. struct vs_threadname_info {
  124. DWORD type; /* 0x1000 */
  125. const char *name;
  126. DWORD thread_id;
  127. DWORD flags;
  128. };
  129. #pragma pack(pop)
  130. #define THREADNAME_INFO_SIZE \
  131. (sizeof(struct vs_threadname_info) / sizeof(ULONG_PTR))
  132. void os_set_thread_name(const char *name)
  133. {
  134. #ifdef __MINGW32__
  135. UNUSED_PARAMETER(name);
  136. #else
  137. struct vs_threadname_info info;
  138. info.type = 0x1000;
  139. info.name = name;
  140. info.thread_id = GetCurrentThreadId();
  141. info.flags = 0;
  142. #ifdef NO_SEH_MINGW
  143. __try1(EXCEPTION_EXECUTE_HANDLER) {
  144. #else
  145. __try {
  146. #endif
  147. RaiseException(VC_EXCEPTION, 0, THREADNAME_INFO_SIZE,
  148. (ULONG_PTR*)&info);
  149. #ifdef NO_SEH_MINGW
  150. } __except1 {
  151. #else
  152. } __except(EXCEPTION_EXECUTE_HANDLER) {
  153. #endif
  154. }
  155. #endif
  156. }