thread_unix.odin 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187
  1. // +build linux, darwin, freebsd, openbsd, netbsd, haiku
  2. // +private
  3. package thread
  4. import "core:sync"
  5. import "core:sys/unix"
  6. import "core:time"
  7. _IS_SUPPORTED :: true
  8. CAS :: sync.atomic_compare_exchange_strong
  9. // NOTE(tetra): Aligned here because of core/unix/pthread_linux.odin/pthread_t.
  10. // Also see core/sys/darwin/mach_darwin.odin/semaphore_t.
  11. Thread_Os_Specific :: struct #align(16) {
  12. unix_thread: unix.pthread_t, // NOTE: very large on Darwin, small on Linux.
  13. cond: sync.Cond,
  14. mutex: sync.Mutex,
  15. }
  16. //
  17. // Creates a thread which will run the given procedure.
  18. // It then waits for `start` to be called.
  19. //
  20. _create :: proc(procedure: Thread_Proc, priority: Thread_Priority) -> ^Thread {
  21. __unix_thread_entry_proc :: proc "c" (t: rawptr) -> rawptr {
  22. t := (^Thread)(t)
  23. // We need to give the thread a moment to start up before we enable cancellation.
  24. can_set_thread_cancel_state := unix.pthread_setcancelstate(unix.PTHREAD_CANCEL_ENABLE, nil) == 0
  25. sync.lock(&t.mutex)
  26. t.id = sync.current_thread_id()
  27. for (.Started not_in sync.atomic_load(&t.flags)) {
  28. // HACK: use a timeout so in the event that the condition is signalled at THIS comment's exact point
  29. // (after checking flags, before starting the wait) it gets itself out of that deadlock after a ms.
  30. sync.wait_with_timeout(&t.cond, &t.mutex, time.Millisecond)
  31. }
  32. if .Joined in sync.atomic_load(&t.flags) {
  33. return nil
  34. }
  35. // Enable thread's cancelability.
  36. if can_set_thread_cancel_state {
  37. unix.pthread_setcanceltype (unix.PTHREAD_CANCEL_ASYNCHRONOUS, nil)
  38. unix.pthread_setcancelstate(unix.PTHREAD_CANCEL_ENABLE, nil)
  39. }
  40. {
  41. init_context := t.init_context
  42. // NOTE(tetra, 2023-05-31): Must do this AFTER thread.start() is called, so that the user can set the init_context, etc!
  43. // Here on Unix, we start the OS thread in a running state, and so we manually have it wait on a condition
  44. // variable above. We must perform that waiting BEFORE we select the context!
  45. context = _select_context_for_thread(init_context)
  46. defer _maybe_destroy_default_temp_allocator(init_context)
  47. t.procedure(t)
  48. }
  49. sync.atomic_or(&t.flags, { .Done })
  50. sync.unlock(&t.mutex)
  51. if .Self_Cleanup in sync.atomic_load(&t.flags) {
  52. t.unix_thread = {}
  53. // NOTE(ftphikari): It doesn't matter which context 'free' received, right?
  54. context = {}
  55. free(t, t.creation_allocator)
  56. }
  57. return nil
  58. }
  59. attrs: unix.pthread_attr_t
  60. if unix.pthread_attr_init(&attrs) != 0 {
  61. return nil // NOTE(tetra, 2019-11-01): POSIX OOM.
  62. }
  63. defer unix.pthread_attr_destroy(&attrs)
  64. // NOTE(tetra, 2019-11-01): These only fail if their argument is invalid.
  65. res: i32
  66. res = unix.pthread_attr_setdetachstate(&attrs, unix.PTHREAD_CREATE_JOINABLE)
  67. assert(res == 0)
  68. when ODIN_OS != .Haiku && ODIN_OS != .NetBSD {
  69. res = unix.pthread_attr_setinheritsched(&attrs, unix.PTHREAD_EXPLICIT_SCHED)
  70. assert(res == 0)
  71. }
  72. thread := new(Thread)
  73. if thread == nil {
  74. return nil
  75. }
  76. thread.creation_allocator = context.allocator
  77. // Set thread priority.
  78. policy: i32
  79. when ODIN_OS != .Haiku && ODIN_OS != .NetBSD {
  80. res = unix.pthread_attr_getschedpolicy(&attrs, &policy)
  81. assert(res == 0)
  82. }
  83. params: unix.sched_param
  84. res = unix.pthread_attr_getschedparam(&attrs, &params)
  85. assert(res == 0)
  86. low := unix.sched_get_priority_min(policy)
  87. high := unix.sched_get_priority_max(policy)
  88. switch priority {
  89. case .Normal: // Okay
  90. case .Low: params.sched_priority = low + 1
  91. case .High: params.sched_priority = high
  92. }
  93. res = unix.pthread_attr_setschedparam(&attrs, &params)
  94. assert(res == 0)
  95. thread.procedure = procedure
  96. if unix.pthread_create(&thread.unix_thread, &attrs, __unix_thread_entry_proc, thread) != 0 {
  97. free(thread, thread.creation_allocator)
  98. return nil
  99. }
  100. return thread
  101. }
  102. _start :: proc(t: ^Thread) {
  103. sync.atomic_or(&t.flags, { .Started })
  104. sync.signal(&t.cond)
  105. }
  106. _is_done :: proc(t: ^Thread) -> bool {
  107. return .Done in sync.atomic_load(&t.flags)
  108. }
  109. _join :: proc(t: ^Thread) {
  110. // sync.guard(&t.mutex)
  111. if unix.pthread_equal(unix.pthread_self(), t.unix_thread) {
  112. return
  113. }
  114. // Preserve other flags besides `.Joined`, like `.Started`.
  115. unjoined := sync.atomic_load(&t.flags) - {.Joined}
  116. joined := unjoined + {.Joined}
  117. // Try to set `t.flags` from unjoined to joined. If it returns joined,
  118. // it means the previous value had that flag set and we can return.
  119. if res, ok := CAS(&t.flags, unjoined, joined); res == joined && !ok {
  120. return
  121. }
  122. // Prevent non-started threads from blocking main thread with initial wait
  123. // condition.
  124. if .Started not_in unjoined {
  125. _start(t)
  126. }
  127. unix.pthread_join(t.unix_thread, nil)
  128. }
  129. _join_multiple :: proc(threads: ..^Thread) {
  130. for t in threads {
  131. _join(t)
  132. }
  133. }
  134. _destroy :: proc(t: ^Thread) {
  135. _join(t)
  136. t.unix_thread = {}
  137. free(t, t.creation_allocator)
  138. }
  139. _terminate :: proc(t: ^Thread, exit_code: int) {
  140. // NOTE(Feoramund): For thread cancellation to succeed on BSDs and
  141. // possibly Darwin systems, the thread must call one of the pthread
  142. // cancelation points at some point after this.
  143. //
  144. // The most obvious one of these is `pthread_cancel`, but there is an
  145. // entire list of functions that act as cancelation points available in the
  146. // pthreads manual page.
  147. //
  148. // This is in contrast to behavior I have seen on Linux where the thread is
  149. // just terminated.
  150. unix.pthread_cancel(t.unix_thread)
  151. }
  152. _yield :: proc() {
  153. unix.sched_yield()
  154. }