|
@@ -0,0 +1,193 @@
|
|
|
+package sync2
|
|
|
+
|
|
|
+import "core:runtime"
|
|
|
+import "core:sync"
|
|
|
+import "core:mem"
|
|
|
+import "intrinsics"
|
|
|
+
|
|
|
+_ :: intrinsics;
|
|
|
+
|
|
|
+Thread_Proc :: #type proc(^Thread);
|
|
|
+
|
|
|
+MAX_USER_ARGUMENTS :: 8;
|
|
|
+
|
|
|
+Thread :: struct {
|
|
|
+ using specific: Thread_Os_Specific,
|
|
|
+ procedure: Thread_Proc,
|
|
|
+ data: rawptr,
|
|
|
+ user_index: int,
|
|
|
+ user_args: [MAX_USER_ARGUMENTS]rawptr,
|
|
|
+
|
|
|
+ init_context: Maybe(runtime.Context),
|
|
|
+
|
|
|
+
|
|
|
+ creation_allocator: mem.Allocator,
|
|
|
+}
|
|
|
+
|
|
|
+#assert(size_of(Thread{}.user_index) == size_of(uintptr));
|
|
|
+
|
|
|
+Thread_Priority :: enum {
|
|
|
+ Normal,
|
|
|
+ Low,
|
|
|
+ High,
|
|
|
+}
|
|
|
+
|
|
|
+thread_create :: proc(procedure: Thread_Proc, priority := Thread_Priority.Normal) -> ^Thread {
|
|
|
+ return _thread_create(procedure, priority);
|
|
|
+}
|
|
|
+thread_destroy :: proc(thread: ^Thread) {
|
|
|
+ _thread_destroy(thread);
|
|
|
+}
|
|
|
+
|
|
|
+thread_start :: proc(thread: ^Thread) {
|
|
|
+ _thread_start(thread);
|
|
|
+}
|
|
|
+
|
|
|
+thread_is_done :: proc(thread: ^Thread) -> bool {
|
|
|
+ return _thread_is_done(thread);
|
|
|
+}
|
|
|
+
|
|
|
+
|
|
|
+thread_join :: proc(thread: ^Thread) {
|
|
|
+ _thread_join(thread);
|
|
|
+}
|
|
|
+
|
|
|
+
|
|
|
+thread_join_mulitple :: proc(threads: ..^Thread) {
|
|
|
+ _thread_join_multiple(..threads);
|
|
|
+}
|
|
|
+
|
|
|
+thread_terminate :: proc(thread: ^Thread, exit_code: int) {
|
|
|
+ _thread_terminate(thread, exit_code);
|
|
|
+}
|
|
|
+
|
|
|
+thread_yield :: proc() {
|
|
|
+ _thread_yield();
|
|
|
+}
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+thread_run :: proc(fn: proc(), init_context: Maybe(runtime.Context) = nil, priority := Thread_Priority.Normal) {
|
|
|
+ thread_proc :: proc(t: ^Thread) {
|
|
|
+ fn := cast(proc())t.data;
|
|
|
+ fn();
|
|
|
+ thread_destroy(t);
|
|
|
+ }
|
|
|
+ t := thread_create(thread_proc, priority);
|
|
|
+ t.data = rawptr(fn);
|
|
|
+ t.init_context = init_context;
|
|
|
+ thread_start(t);
|
|
|
+}
|
|
|
+
|
|
|
+thread_run_with_data :: proc(data: rawptr, fn: proc(data: rawptr), init_context: Maybe(runtime.Context) = nil, priority := Thread_Priority.Normal) {
|
|
|
+ thread_proc :: proc(t: ^Thread) {
|
|
|
+ fn := cast(proc(rawptr))t.data;
|
|
|
+ assert(t.user_index >= 1);
|
|
|
+ data := t.user_args[0];
|
|
|
+ fn(data);
|
|
|
+ thread_destroy(t);
|
|
|
+ }
|
|
|
+ t := thread_create(thread_proc, priority);
|
|
|
+ t.data = rawptr(fn);
|
|
|
+ t.user_index = 1;
|
|
|
+ t.user_args = data;
|
|
|
+ t.init_context = init_context;
|
|
|
+ thread_start(t);
|
|
|
+}
|
|
|
+
|
|
|
+thread_run_with_poly_data :: proc(data: $T, fn: proc(data: T), init_context: Maybe(runtime.Context) = nil, priority := Thread_Priority.Normal)
|
|
|
+ where size_of(T) <= size_of(rawptr) {
|
|
|
+ thread_proc :: proc(t: ^Thread) {
|
|
|
+ fn := cast(proc(T))t.data;
|
|
|
+ assert(t.user_index >= 1);
|
|
|
+ data := (^T)(&t.user_args[0])^;
|
|
|
+ fn(data);
|
|
|
+ thread_destroy(t);
|
|
|
+ }
|
|
|
+ t := thread_create(thread_proc, priority);
|
|
|
+ t.data = rawptr(fn);
|
|
|
+ t.user_index = 1;
|
|
|
+ data := data;
|
|
|
+ mem.copy(&t.user_args[0], &data, size_of(data));
|
|
|
+ t.init_context = init_context;
|
|
|
+ thread_start(t);
|
|
|
+}
|
|
|
+
|
|
|
+thread_run_with_poly_data2 :: proc(arg1: $T1, arg2: $T2, fn: proc(T1, T2), init_context: Maybe(runtime.Context) = nil, priority := Thread_Priority.Normal)
|
|
|
+ where size_of(T1) <= size_of(rawptr),
|
|
|
+ size_of(T2) <= size_of(rawptr) {
|
|
|
+ thread_proc :: proc(t: ^Thread) {
|
|
|
+ fn := cast(proc(T1, T2))t.data;
|
|
|
+ assert(t.user_index >= 2);
|
|
|
+ arg1 := (^T1)(&t.user_args[0])^;
|
|
|
+ arg2 := (^T2)(&t.user_args[1])^;
|
|
|
+ fn(arg1, arg2);
|
|
|
+ thread_destroy(t);
|
|
|
+ }
|
|
|
+ t := thread_create(thread_proc, priority);
|
|
|
+ t.data = rawptr(fn);
|
|
|
+ t.user_index = 2;
|
|
|
+ arg1, arg2 := arg1, arg2;
|
|
|
+ mem.copy(&t.user_args[0], &arg1, size_of(arg1));
|
|
|
+ mem.copy(&t.user_args[1], &arg2, size_of(arg2));
|
|
|
+ t.init_context = init_context;
|
|
|
+ thread_start(t);
|
|
|
+}
|
|
|
+
|
|
|
+thread_run_with_poly_data3 :: proc(arg1: $T1, arg2: $T2, arg3: $T3, fn: proc(arg1: T1, arg2: T2, arg3: T3), init_context: Maybe(runtime.Context) = nil, priority := Thread_Priority.Normal)
|
|
|
+ where size_of(T1) <= size_of(rawptr),
|
|
|
+ size_of(T2) <= size_of(rawptr),
|
|
|
+ size_of(T3) <= size_of(rawptr) {
|
|
|
+ thread_proc :: proc(t: ^Thread) {
|
|
|
+ fn := cast(proc(T1, T2, T3))t.data;
|
|
|
+ assert(t.user_index >= 3);
|
|
|
+ arg1 := (^T1)(&t.user_args[0])^;
|
|
|
+ arg2 := (^T2)(&t.user_args[1])^;
|
|
|
+ arg3 := (^T3)(&t.user_args[2])^;
|
|
|
+ fn(arg1, arg2, arg3);
|
|
|
+ thread_destroy(t);
|
|
|
+ }
|
|
|
+ t := thread_create(thread_proc, priority);
|
|
|
+ t.data = rawptr(fn);
|
|
|
+ t.user_index = 3;
|
|
|
+ arg1, arg2, arg3 := arg1, arg2, arg3;
|
|
|
+ mem.copy(&t.user_args[0], &arg1, size_of(arg1));
|
|
|
+ mem.copy(&t.user_args[1], &arg2, size_of(arg2));
|
|
|
+ mem.copy(&t.user_args[2], &arg3, size_of(arg3));
|
|
|
+ t.init_context = init_context;
|
|
|
+ thread_start(t);
|
|
|
+}
|
|
|
+thread_run_with_poly_data4 :: proc(arg1: $T1, arg2: $T2, arg3: $T3, arg4: $T4, fn: proc(arg1: T1, arg2: T2, arg3: T3, arg4: T4), init_context: Maybe(runtime.Context) = nil, priority := Thread_Priority.Normal)
|
|
|
+ where size_of(T1) <= size_of(rawptr),
|
|
|
+ size_of(T2) <= size_of(rawptr),
|
|
|
+ size_of(T3) <= size_of(rawptr) {
|
|
|
+ thread_proc :: proc(t: ^Thread) {
|
|
|
+ fn := cast(proc(T1, T2, T3, T4))t.data;
|
|
|
+ assert(t.user_index >= 4);
|
|
|
+ arg1 := (^T1)(&t.user_args[0])^;
|
|
|
+ arg2 := (^T2)(&t.user_args[1])^;
|
|
|
+ arg3 := (^T3)(&t.user_args[2])^;
|
|
|
+ arg4 := (^T4)(&t.user_args[3])^;
|
|
|
+ fn(arg1, arg2, arg3, arg4);
|
|
|
+ thread_destroy(t);
|
|
|
+ }
|
|
|
+ t := thread_create(thread_proc, priority);
|
|
|
+ t.data = rawptr(fn);
|
|
|
+ t.user_index = 4;
|
|
|
+ arg1, arg2, arg3, arg4 := arg1, arg2, arg3, arg4;
|
|
|
+ mem.copy(&t.user_args[0], &arg1, size_of(arg1));
|
|
|
+ mem.copy(&t.user_args[1], &arg2, size_of(arg2));
|
|
|
+ mem.copy(&t.user_args[2], &arg3, size_of(arg3));
|
|
|
+ mem.copy(&t.user_args[3], &arg4, size_of(arg4));
|
|
|
+ t.init_context = init_context;
|
|
|
+ thread_start(t);
|
|
|
+}
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+thread_create_and_start :: proc(fn: Thread_Proc, init_context: Maybe(runtime.Context) = nil, priority := Thread_Priority.Normal) -> ^Thread {
|
|
|
+ t := thread_create(fn, priority);
|
|
|
+ t.init_context = init_context;
|
|
|
+ thread_start(t);
|
|
|
+ return t;
|
|
|
+}
|