aboutsummaryrefslogtreecommitdiff
path: root/core/thread
diff options
context:
space:
mode:
authorFeoramund <161657516+Feoramund@users.noreply.github.com>2024-09-08 18:05:34 -0400
committerFeoramund <161657516+Feoramund@users.noreply.github.com>2024-09-09 16:19:14 -0400
commit4d14b4257e7570216826c5cbcee94aa51116e3b3 (patch)
tree8cfb7a4d27a913463b2580c3259ba8ba48ee6b21 /core/thread
parentcbd4d5e765646ef07c4133ab65e06652a87a1916 (diff)
Convert POSIX `Thread` to use semaphore instead
One less value to store, and it should be less of a hack too. Semaphores will not wait around if they have the go-ahead; they depend on an internal value being non-zero, instead of whatever was loaded when they started waiting, which is the case with a `Cond`.
Diffstat (limited to 'core/thread')
-rw-r--r--core/thread/thread_unix.odin16
1 files changed, 4 insertions, 12 deletions
diff --git a/core/thread/thread_unix.odin b/core/thread/thread_unix.odin
index d165560ac..3d3b419b0 100644
--- a/core/thread/thread_unix.odin
+++ b/core/thread/thread_unix.odin
@@ -5,7 +5,6 @@ package thread
import "base:runtime"
import "core:sync"
import "core:sys/unix"
-import "core:time"
_IS_SUPPORTED :: true
@@ -13,8 +12,7 @@ _IS_SUPPORTED :: true
// Also see core/sys/darwin/mach_darwin.odin/semaphore_t.
Thread_Os_Specific :: struct #align(16) {
unix_thread: unix.pthread_t, // NOTE: very large on Darwin, small on Linux.
- cond: sync.Cond,
- mutex: sync.Mutex,
+ start_ok: sync.Sema,
}
//
// Creates a thread which will run the given procedure.
@@ -27,14 +25,10 @@ _create :: proc(procedure: Thread_Proc, priority: Thread_Priority) -> ^Thread {
// We need to give the thread a moment to start up before we enable cancellation.
can_set_thread_cancel_state := unix.pthread_setcancelstate(unix.PTHREAD_CANCEL_ENABLE, nil) == 0
- sync.lock(&t.mutex)
-
t.id = sync.current_thread_id()
- for (.Started not_in sync.atomic_load(&t.flags)) {
- // HACK: use a timeout so in the event that the condition is signalled at THIS comment's exact point
- // (after checking flags, before starting the wait) it gets itself out of that deadlock after a ms.
- sync.wait_with_timeout(&t.cond, &t.mutex, time.Millisecond)
+ if .Started not_in sync.atomic_load(&t.flags) {
+ sync.wait(&t.start_ok)
}
if .Joined in sync.atomic_load(&t.flags) {
@@ -64,8 +58,6 @@ _create :: proc(procedure: Thread_Proc, priority: Thread_Priority) -> ^Thread {
sync.atomic_or(&t.flags, { .Done })
- sync.unlock(&t.mutex)
-
if .Self_Cleanup in sync.atomic_load(&t.flags) {
res := unix.pthread_detach(t.unix_thread)
assert_contextless(res == 0)
@@ -130,7 +122,7 @@ _create :: proc(procedure: Thread_Proc, priority: Thread_Priority) -> ^Thread {
_start :: proc(t: ^Thread) {
sync.atomic_or(&t.flags, { .Started })
- sync.signal(&t.cond)
+ sync.post(&t.start_ok)
}
_is_done :: proc(t: ^Thread) -> bool {