threadlocalstorage.nim 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  1. when defined(windows):
  2. type
  3. SysThread* = Handle
  4. WinThreadProc = proc (x: pointer): int32 {.stdcall.}
  5. proc createThread(lpThreadAttributes: pointer, dwStackSize: int32,
  6. lpStartAddress: WinThreadProc,
  7. lpParameter: pointer,
  8. dwCreationFlags: int32,
  9. lpThreadId: var int32): SysThread {.
  10. stdcall, dynlib: "kernel32", importc: "CreateThread".}
  11. proc winSuspendThread(hThread: SysThread): int32 {.
  12. stdcall, dynlib: "kernel32", importc: "SuspendThread".}
  13. proc winResumeThread(hThread: SysThread): int32 {.
  14. stdcall, dynlib: "kernel32", importc: "ResumeThread".}
  15. proc waitForMultipleObjects(nCount: int32,
  16. lpHandles: ptr SysThread,
  17. bWaitAll: int32,
  18. dwMilliseconds: int32): int32 {.
  19. stdcall, dynlib: "kernel32", importc: "WaitForMultipleObjects".}
  20. proc terminateThread(hThread: SysThread, dwExitCode: int32): int32 {.
  21. stdcall, dynlib: "kernel32", importc: "TerminateThread".}
  22. proc getCurrentThreadId(): int32 {.
  23. stdcall, dynlib: "kernel32", importc: "GetCurrentThreadId".}
  24. type
  25. ThreadVarSlot = distinct int32
  26. when true:
  27. proc threadVarAlloc(): ThreadVarSlot {.
  28. importc: "TlsAlloc", stdcall, header: "<windows.h>".}
  29. proc threadVarSetValue(dwTlsIndex: ThreadVarSlot, lpTlsValue: pointer) {.
  30. importc: "TlsSetValue", stdcall, header: "<windows.h>".}
  31. proc tlsGetValue(dwTlsIndex: ThreadVarSlot): pointer {.
  32. importc: "TlsGetValue", stdcall, header: "<windows.h>".}
  33. proc getLastError(): uint32 {.
  34. importc: "GetLastError", stdcall, header: "<windows.h>".}
  35. proc setLastError(x: uint32) {.
  36. importc: "SetLastError", stdcall, header: "<windows.h>".}
  37. proc threadVarGetValue(dwTlsIndex: ThreadVarSlot): pointer =
  38. let realLastError = getLastError()
  39. result = tlsGetValue(dwTlsIndex)
  40. setLastError(realLastError)
  41. else:
  42. proc threadVarAlloc(): ThreadVarSlot {.
  43. importc: "TlsAlloc", stdcall, dynlib: "kernel32".}
  44. proc threadVarSetValue(dwTlsIndex: ThreadVarSlot, lpTlsValue: pointer) {.
  45. importc: "TlsSetValue", stdcall, dynlib: "kernel32".}
  46. proc threadVarGetValue(dwTlsIndex: ThreadVarSlot): pointer {.
  47. importc: "TlsGetValue", stdcall, dynlib: "kernel32".}
  48. proc setThreadAffinityMask(hThread: SysThread, dwThreadAffinityMask: uint) {.
  49. importc: "SetThreadAffinityMask", stdcall, header: "<windows.h>".}
  50. elif defined(genode):
  51. import genode/env
  52. const
  53. GenodeHeader = "genode_cpp/threads.h"
  54. type
  55. SysThread* {.importcpp: "Nim::SysThread",
  56. header: GenodeHeader, final, pure.} = object
  57. GenodeThreadProc = proc (x: pointer) {.noconv.}
  58. ThreadVarSlot = int
  59. proc initThread(s: var SysThread,
  60. env: GenodeEnv,
  61. stackSize: culonglong,
  62. entry: GenodeThreadProc,
  63. arg: pointer,
  64. affinity: cuint) {.
  65. importcpp: "#.initThread(@)".}
  66. proc threadVarAlloc(): ThreadVarSlot = 0
  67. proc offMainThread(): bool {.
  68. importcpp: "Nim::SysThread::offMainThread",
  69. header: GenodeHeader.}
  70. proc threadVarSetValue(value: pointer) {.
  71. importcpp: "Nim::SysThread::threadVarSetValue(@)",
  72. header: GenodeHeader.}
  73. proc threadVarGetValue(): pointer {.
  74. importcpp: "Nim::SysThread::threadVarGetValue()",
  75. header: GenodeHeader.}
  76. var mainTls: pointer
  77. proc threadVarSetValue(s: ThreadVarSlot, value: pointer) {.inline.} =
  78. if offMainThread():
  79. threadVarSetValue(value);
  80. else:
  81. mainTls = value
  82. proc threadVarGetValue(s: ThreadVarSlot): pointer {.inline.} =
  83. if offMainThread():
  84. threadVarGetValue();
  85. else:
  86. mainTls
  87. else:
  88. when not (defined(macosx) or defined(haiku)):
  89. {.passL: "-pthread".}
  90. when not defined(haiku):
  91. {.passC: "-pthread".}
  92. const
  93. schedh = "#define _GNU_SOURCE\n#include <sched.h>"
  94. pthreadh = "#define _GNU_SOURCE\n#include <pthread.h>"
  95. when not declared(Time):
  96. when defined(linux):
  97. type Time = clong
  98. else:
  99. type Time = int
  100. when (defined(linux) or defined(nintendoswitch)) and defined(amd64):
  101. type
  102. SysThread* {.importc: "pthread_t",
  103. header: "<sys/types.h>" .} = distinct culong
  104. Pthread_attr {.importc: "pthread_attr_t",
  105. header: "<sys/types.h>".} = object
  106. abi: array[56 div sizeof(clong), clong]
  107. ThreadVarSlot {.importc: "pthread_key_t",
  108. header: "<sys/types.h>".} = distinct cuint
  109. else:
  110. type
  111. SysThread* {.importc: "pthread_t", header: "<sys/types.h>".} = object
  112. Pthread_attr {.importc: "pthread_attr_t",
  113. header: "<sys/types.h>".} = object
  114. ThreadVarSlot {.importc: "pthread_key_t",
  115. header: "<sys/types.h>".} = object
  116. type
  117. Timespec {.importc: "struct timespec", header: "<time.h>".} = object
  118. tv_sec: Time
  119. tv_nsec: clong
  120. proc pthread_attr_init(a1: var PthreadAttr) {.
  121. importc, header: pthreadh.}
  122. proc pthread_attr_setstacksize(a1: var PthreadAttr, a2: int) {.
  123. importc, header: pthreadh.}
  124. proc pthread_create(a1: var SysThread, a2: var PthreadAttr,
  125. a3: proc (x: pointer): pointer {.noconv.},
  126. a4: pointer): cint {.importc: "pthread_create",
  127. header: pthreadh.}
  128. proc pthread_join(a1: SysThread, a2: ptr pointer): cint {.
  129. importc, header: pthreadh.}
  130. proc pthread_cancel(a1: SysThread): cint {.
  131. importc: "pthread_cancel", header: pthreadh.}
  132. proc pthread_getspecific(a1: ThreadVarSlot): pointer {.
  133. importc: "pthread_getspecific", header: pthreadh.}
  134. proc pthread_key_create(a1: ptr ThreadVarSlot,
  135. destruct: proc (x: pointer) {.noconv.}): int32 {.
  136. importc: "pthread_key_create", header: pthreadh.}
  137. proc pthread_key_delete(a1: ThreadVarSlot): int32 {.
  138. importc: "pthread_key_delete", header: pthreadh.}
  139. proc pthread_setspecific(a1: ThreadVarSlot, a2: pointer): int32 {.
  140. importc: "pthread_setspecific", header: pthreadh.}
  141. proc threadVarAlloc(): ThreadVarSlot {.inline.} =
  142. discard pthread_key_create(addr(result), nil)
  143. proc threadVarSetValue(s: ThreadVarSlot, value: pointer) {.inline.} =
  144. discard pthread_setspecific(s, value)
  145. proc threadVarGetValue(s: ThreadVarSlot): pointer {.inline.} =
  146. result = pthread_getspecific(s)
  147. type CpuSet {.importc: "cpu_set_t", header: schedh.} = object
  148. when defined(linux) and defined(amd64):
  149. abi: array[1024 div (8 * sizeof(culong)), culong]
  150. proc cpusetZero(s: var CpuSet) {.importc: "CPU_ZERO", header: schedh.}
  151. proc cpusetIncl(cpu: cint; s: var CpuSet) {.
  152. importc: "CPU_SET", header: schedh.}
  153. proc setAffinity(thread: SysThread; setsize: csize; s: var CpuSet) {.
  154. importc: "pthread_setaffinity_np", header: pthreadh.}
  155. const
  156. emulatedThreadVars = compileOption("tlsEmulation")
  157. when emulatedThreadVars:
  158. # the compiler generates this proc for us, so that we can get the size of
  159. # the thread local var block; we use this only for sanity checking though
  160. proc nimThreadVarsSize(): int {.noconv, importc: "NimThreadVarsSize".}
  161. # we preallocate a fixed size for thread local storage, so that no heap
  162. # allocations are needed. Currently less than 16K are used on a 64bit machine.
  163. # We use ``float`` for proper alignment:
  164. const nimTlsSize {.intdefine.} = 16000
  165. type
  166. ThreadLocalStorage = array[0..(nimTlsSize div sizeof(float)), float]
  167. PGcThread = ptr GcThread
  168. GcThread {.pure, inheritable.} = object
  169. when emulatedThreadVars:
  170. tls: ThreadLocalStorage
  171. else:
  172. nil
  173. when hasSharedHeap:
  174. next, prev: PGcThread
  175. stackBottom, stackTop: pointer
  176. stackSize: int
  177. else:
  178. nil
  179. when emulatedThreadVars:
  180. var globalsSlot: ThreadVarSlot
  181. when not defined(useNimRtl):
  182. var mainThread: GcThread
  183. proc GetThreadLocalVars(): pointer {.compilerRtl, inl.} =
  184. result = addr(cast[PGcThread](threadVarGetValue(globalsSlot)).tls)
  185. proc initThreadVarsEmulation() {.compilerProc, inline.} =
  186. when not defined(useNimRtl):
  187. globalsSlot = threadVarAlloc()
  188. when declared(mainThread):
  189. threadVarSetValue(globalsSlot, addr(mainThread))
  190. when not defined(useNimRtl):
  191. when emulatedThreadVars:
  192. if nimThreadVarsSize() > sizeof(ThreadLocalStorage):
  193. c_fprintf(cstderr, """too large thread local storage size requested,
  194. use -d:\"nimTlsSize=X\" to setup even more or stop using unittest.nim""")
  195. quit 1