From 8482d48c151b04fa784ccc61142b3d75ec6185f4 Mon Sep 17 00:00:00 2001 From: Quinton Miller Date: Thu, 23 Dec 2021 23:01:30 +0800 Subject: [PATCH 1/7] Implement multithreading primitives on Windows --- spec/std/thread/mutex_spec.cr | 9 +- spec/win32_std_spec.cr | 6 +- src/crystal/system/thread.cr | 2 +- .../system/thread_condition_variable.cr | 29 +++++++ src/crystal/system/thread_mutex.cr | 2 +- .../system/unix/pthread_condition_variable.cr | 2 +- src/crystal/system/win32/process.cr | 2 +- src/crystal/system/win32/thread.cr | 86 +++++++++++++++---- .../system/win32/thread_condition_variable.cr | 41 +++++++++ src/crystal/system/win32/thread_mutex.cr | 57 +++++++++++- src/gc/boehm.cr | 11 ++- .../c/processthreadsapi.cr | 2 + src/lib_c/x86_64-windows-msvc/c/synchapi.cr | 28 ++++++ src/lib_c/x86_64-windows-msvc/c/winbase.cr | 5 ++ src/lib_c/x86_64-windows-msvc/c/winsock2.cr | 6 +- 15 files changed, 251 insertions(+), 37 deletions(-) create mode 100644 src/crystal/system/thread_condition_variable.cr create mode 100644 src/crystal/system/win32/thread_condition_variable.cr diff --git a/spec/std/thread/mutex_spec.cr b/spec/std/thread/mutex_spec.cr index 037c735b38e9..d028ffdd8f8a 100644 --- a/spec/std/thread/mutex_spec.cr +++ b/spec/std/thread/mutex_spec.cr @@ -13,11 +13,11 @@ describe Thread::Mutex do a = 0 mutex = Thread::Mutex.new - threads = 10.times.map do + threads = Array.new(10) do Thread.new do mutex.synchronize { a += 1 } end - end.to_a + end threads.each(&.join) a.should eq(10) @@ -27,15 +27,16 @@ describe Thread::Mutex do mutex = Thread::Mutex.new mutex.try_lock.should be_true mutex.try_lock.should be_false - expect_raises(RuntimeError, "pthread_mutex_lock: ") { mutex.lock } + expect_raises(RuntimeError) { mutex.lock } mutex.unlock + Thread.new { mutex.synchronize { } }.join end it "won't unlock from another thread" do mutex = Thread::Mutex.new mutex.lock - expect_raises(RuntimeError, "pthread_mutex_unlock: ") do + expect_raises(RuntimeError) do Thread.new { mutex.unlock }.join end diff --git a/spec/win32_std_spec.cr b/spec/win32_std_spec.cr index 69744ac1a865..6057f3d8df05 100644 --- a/spec/win32_std_spec.cr +++ b/spec/win32_std_spec.cr @@ -219,9 +219,9 @@ require "./std/symbol_spec.cr" # require "./std/system/user_spec.cr" (failed codegen) require "./std/system_error_spec.cr" require "./std/system_spec.cr" -# require "./std/thread/condition_variable_spec.cr" (failed codegen) -# require "./std/thread/mutex_spec.cr" (failed codegen) -# require "./std/thread_spec.cr" (failed codegen) +require "./std/thread/condition_variable_spec.cr" +require "./std/thread/mutex_spec.cr" +require "./std/thread_spec.cr" require "./std/time/custom_formats_spec.cr" require "./std/time/format_spec.cr" require "./std/time/location_spec.cr" diff --git a/src/crystal/system/thread.cr b/src/crystal/system/thread.cr index ffaf51555782..35485459e5e6 100644 --- a/src/crystal/system/thread.cr +++ b/src/crystal/system/thread.cr @@ -27,10 +27,10 @@ class Thread end require "./thread_linked_list" +require "./thread_condition_variable" {% if flag?(:unix) %} require "./unix/pthread" - require "./unix/pthread_condition_variable" {% elsif flag?(:win32) %} require "./win32/thread" {% else %} diff --git a/src/crystal/system/thread_condition_variable.cr b/src/crystal/system/thread_condition_variable.cr new file mode 100644 index 000000000000..fa6fc69546b2 --- /dev/null +++ b/src/crystal/system/thread_condition_variable.cr @@ -0,0 +1,29 @@ +class Thread + class ConditionVariable + # Creates a new condition variable. + # def initialize + + # Unblocks one thread that is waiting on `self`. + # def signal : Nil + + # Unblocks all threads that are waiting on `self`. + # def broadcast : Nil + + # Causes the calling thread to wait on `self` and unlock the given *mutex* + # atomically. + # def wait(mutex : Thread::Mutex) : Nil + + # Causes the calling thread to wait on `self` and unlock the given *mutex* + # atomically within the given *time* span. Yields to the given block if a + # timeout occurs. + # def wait(mutex : Thread::Mutex, time : Time::Span, & : ->) + end +end + +{% if flag?(:unix) %} + require "./unix/pthread_condition_variable" +{% elsif flag?(:win32) %} + require "./win32/thread_condition_variable" +{% else %} + {% raise "thread condition variable not supported" %} +{% end %} diff --git a/src/crystal/system/thread_mutex.cr b/src/crystal/system/thread_mutex.cr index ced2563cefbc..c2b05c61cafa 100644 --- a/src/crystal/system/thread_mutex.cr +++ b/src/crystal/system/thread_mutex.cr @@ -22,5 +22,5 @@ end {% elsif flag?(:win32) %} require "./win32/thread_mutex" {% else %} - {% raise "thread not supported" %} + {% raise "thread mutex not supported" %} {% end %} diff --git a/src/crystal/system/unix/pthread_condition_variable.cr b/src/crystal/system/unix/pthread_condition_variable.cr index 225aad1c7105..a09811c79281 100644 --- a/src/crystal/system/unix/pthread_condition_variable.cr +++ b/src/crystal/system/unix/pthread_condition_variable.cr @@ -33,7 +33,7 @@ class Thread raise RuntimeError.from_os_error("pthread_cond_wait", Errno.new(ret)) unless ret == 0 end - def wait(mutex : Thread::Mutex, time : Time::Span) + def wait(mutex : Thread::Mutex, time : Time::Span, & : ->) ret = {% if flag?(:darwin) %} ts = uninitialized LibC::Timespec diff --git a/src/crystal/system/win32/process.cr b/src/crystal/system/win32/process.cr index 00167b230fdb..05b5140d0338 100644 --- a/src/crystal/system/win32/process.cr +++ b/src/crystal/system/win32/process.cr @@ -20,7 +20,7 @@ struct Crystal::System::Process end def wait - if LibC.WaitForSingleObject(@process_handle, LibC::INFINITE) != 0 + if LibC.WaitForSingleObject(@process_handle, LibC::INFINITE) != LibC::WAIT_OBJECT_0 raise RuntimeError.from_winerror("WaitForSingleObject") end diff --git a/src/crystal/system/win32/thread.cr b/src/crystal/system/win32/thread.cr index 1af2ee9660f2..77047c7bb1c8 100644 --- a/src/crystal/system/win32/thread.cr +++ b/src/crystal/system/win32/thread.cr @@ -1,10 +1,11 @@ require "c/processthreadsapi" +require "c/synchapi" -# TODO: Implement for multithreading. class Thread # all thread objects, so the GC can see them (it doesn't scan thread locals) - @@threads = Thread::LinkedList(Thread).new + protected class_getter(threads) { Thread::LinkedList(Thread).new } + @th : LibC::HANDLE @exception : Exception? @detached = Atomic(UInt8).new(0) @main_fiber : Fiber? @@ -16,42 +17,88 @@ class Thread property previous : Thread? def self.unsafe_each - @@threads.unsafe_each { |thread| yield thread } + threads.unsafe_each { |thread| yield thread } end + # Starts a new system thread. + def initialize(&@func : ->) + @th = uninitialized LibC::HANDLE + + @th = LibGC.beginthreadex(nil, 0, ->(data : Void*) { + (data.as(Thread)).start + LibC::UInt.zero + }, self.as(Void*), 0, nil).as(LibC::HANDLE) + + if @th.null? + raise RuntimeError.from_errno("_beginthreadex") + end + end + + # Used once to initialize the thread object representing the main thread of + # the process (that already exists). def initialize + # `GetCurrentThread` returns a _constant_ and is only meaningful as an + # argument to Win32 APIs; to uniquely identify it we must duplicate the handle + @th = uninitialized LibC::HANDLE + cur_proc = LibC.GetCurrentProcess + LibC.DuplicateHandle(cur_proc, LibC.GetCurrentThread, cur_proc, pointerof(@th), 0, true, LibC::DUPLICATE_SAME_ACCESS) + + @func = ->{} @main_fiber = Fiber.new(stack_address, self) - @@threads.push(self) + + Thread.threads.push(self) end - @@current : Thread? = nil + private def detach + if @detached.compare_and_set(0, 1).last + yield + end + end - # Associates the Thread object to the running system thread. - protected def self.current=(@@current : Thread) : Thread + # Suspends the current thread until this thread terminates. + def join : Nil + detach do + if LibC.WaitForSingleObject(@th, LibC::INFINITE) != LibC::WAIT_OBJECT_0 + @exception ||= RuntimeError.from_winerror("WaitForSingleObject") + end + if LibC.CloseHandle(@th) == 0 + @exception ||= RuntimeError.from_winerror("CloseHandle") + end + end + + if exception = @exception + raise exception + end end + @[ThreadLocal] + @@current : Thread? + # Returns the Thread object associated to the running system thread. def self.current : Thread - @@current || raise "BUG: Thread.current returned NULL" + @@current ||= new end - # Create the thread object for the current thread (aka the main thread of the - # process). - # - # TODO: consider moving to `kernel.cr` or `crystal/main.cr` - self.current = new + # Associates the Thread object to the running system thread. + protected def self.current=(@@current : Thread) : Thread + end + + def self.yield : Nil + LibC.SwitchToThread + end # Returns the Fiber representing the thread's main stack. - def main_fiber + def main_fiber : Fiber @main_fiber.not_nil! end # :nodoc: - def scheduler + def scheduler : Crystal::Scheduler @scheduler ||= Crystal::Scheduler.new(main_fiber) end protected def start + Thread.threads.push(self) Thread.current = self @main_fiber = fiber = Fiber.new(stack_address, self) @@ -60,9 +107,9 @@ class Thread rescue ex @exception = ex ensure - @@threads.delete(self) + Thread.threads.delete(self) Fiber.inactive(fiber) - detach_self + detach { LibC.CloseHandle(@th) } end end @@ -71,4 +118,9 @@ class Thread Pointer(Void).new(low_limit) end + + # :nodoc: + def to_unsafe + @th + end end diff --git a/src/crystal/system/win32/thread_condition_variable.cr b/src/crystal/system/win32/thread_condition_variable.cr new file mode 100644 index 000000000000..423de9dc57f3 --- /dev/null +++ b/src/crystal/system/win32/thread_condition_variable.cr @@ -0,0 +1,41 @@ +require "c/synchapi" + +# :nodoc: +class Thread + # :nodoc: + class ConditionVariable + def initialize + @cond = uninitialized LibC::CONDITION_VARIABLE + LibC.InitializeConditionVariable(self) + end + + def signal : Nil + LibC.WakeConditionVariable(self) + end + + def broadcast : Nil + LibC.WakeAllConditionVariable(self) + end + + def wait(mutex : Thread::Mutex) : Nil + ret = LibC.SleepConditionVariableCS(self, mutex, LibC::INFINITE) + raise RuntimeError.from_winerror("SleepConditionVariableCS") if ret == 0 + end + + def wait(mutex : Thread::Mutex, time : Time::Span, & : ->) + ret = LibC.SleepConditionVariableCS(self, mutex, time.total_milliseconds) + return if ret != 0 + + error = WinError.value + if error == WinError::ERROR_TIMEOUT + yield + else + raise RuntimeError.from_os_error("SleepConditionVariableCS", error) + end + end + + def to_unsafe + pointerof(@cond) + end + end +end diff --git a/src/crystal/system/win32/thread_mutex.cr b/src/crystal/system/win32/thread_mutex.cr index be681c31398a..afd4cb1fbdcb 100644 --- a/src/crystal/system/win32/thread_mutex.cr +++ b/src/crystal/system/win32/thread_mutex.cr @@ -1,8 +1,61 @@ -# TODO: Implement +require "c/synchapi" + +# :nodoc: class Thread + # :nodoc: + # for Win32 condition variable interop we must use either a critical section + # or a slim reader/writer lock, not a Win32 mutex + # also note critical sections are reentrant; to match the behaviour in + # `../unix/pthread_mutex.cr` we must do extra housekeeping ourselves class Mutex + def initialize + @cs = uninitialized LibC::CRITICAL_SECTION + LibC.InitializeCriticalSectionAndSpinCount(self, 1000) + end + + def lock : Nil + LibC.EnterCriticalSection(self) + if @cs.recursionCount > 1 + LibC.LeaveCriticalSection(self) + raise RuntimeError.new "Attempt to lock a mutex recursively (deadlock)" + end + end + + def try_lock : Bool + if LibC.TryEnterCriticalSection(self) != 0 + if @cs.recursionCount > 1 + LibC.LeaveCriticalSection(self) + false + else + true + end + else + false + end + end + + def unlock : Nil + # `owningThread` is declared as `LibC::HANDLE` for historical reasons, so + # the following comparison is correct + unless @cs.owningThread == LibC::HANDLE.new(LibC.GetCurrentThreadId) + raise RuntimeError.new "Attempt to unlock a mutex locked by another thread" + end + LibC.LeaveCriticalSection(self) + end + def synchronize - yield + lock + yield self + ensure + unlock + end + + def finalize + LibC.DeleteCriticalSection(self) + end + + def to_unsafe + pointerof(@cs) end end end diff --git a/src/gc/boehm.cr b/src/gc/boehm.cr index 36782e3ddad6..2bf9a4c75ce2 100644 --- a/src/gc/boehm.cr +++ b/src/gc/boehm.cr @@ -88,8 +88,15 @@ lib LibGC fun size = GC_size(addr : Void*) : LibC::SizeT - {% unless flag?(:win32) %} - # Boehm GC requires to use GC_pthread_create and GC_pthread_join instead of pthread_create and pthread_join + # Boehm GC requires to use its own thread manipulation routines instead of pthread's or Win32's + {% if flag?(:win32) %} + {% if flag?(:preview_dll) %} + {% raise "TODO: implement GC_CreateThread" %} + {% else %} + fun beginthreadex = GC_beginthreadex(security : Void*, stack_size : LibC::UInt, start_address : Void* -> LibC::UInt, + arglist : Void*, initflag : LibC::UInt, thrdaddr : LibC::UInt*) : Void* + {% end %} + {% else %} fun pthread_create = GC_pthread_create(thread : LibC::PthreadT*, attr : LibC::PthreadAttrT*, start : Void* -> Void*, arg : Void*) : LibC::Int fun pthread_join = GC_pthread_join(thread : LibC::PthreadT, value : Void**) : LibC::Int fun pthread_detach = GC_pthread_detach(thread : LibC::PthreadT) : LibC::Int diff --git a/src/lib_c/x86_64-windows-msvc/c/processthreadsapi.cr b/src/lib_c/x86_64-windows-msvc/c/processthreadsapi.cr index ca76d7ca36cf..8d527996a36c 100644 --- a/src/lib_c/x86_64-windows-msvc/c/processthreadsapi.cr +++ b/src/lib_c/x86_64-windows-msvc/c/processthreadsapi.cr @@ -33,6 +33,7 @@ lib LibC end fun GetCurrentThread : HANDLE + fun GetCurrentThreadId : DWORD fun GetCurrentThreadStackLimits(lowLimit : ULONG_PTR*, highLimit : ULONG_PTR*) : Void fun GetCurrentProcess : HANDLE fun GetCurrentProcessId : DWORD @@ -45,6 +46,7 @@ lib LibC lpStartupInfo : STARTUPINFOW*, lpProcessInformation : PROCESS_INFORMATION*) : BOOL fun GetProcessTimes(hProcess : HANDLE, lpCreationTime : FILETIME*, lpExitTime : FILETIME*, lpKernelTime : FILETIME*, lpUserTime : FILETIME*) : BOOL + fun SwitchToThread : BOOL PROCESS_QUERY_INFORMATION = 0x0400 end diff --git a/src/lib_c/x86_64-windows-msvc/c/synchapi.cr b/src/lib_c/x86_64-windows-msvc/c/synchapi.cr index 23804d0f3aaf..e101b7f6284b 100644 --- a/src/lib_c/x86_64-windows-msvc/c/synchapi.cr +++ b/src/lib_c/x86_64-windows-msvc/c/synchapi.cr @@ -1,7 +1,35 @@ require "c/basetsd" require "c/int_safe" +require "c/winbase" +require "c/wtypesbase" lib LibC + # the meanings of these fields are documented not in the Win32 API docs but in + # https://docs.microsoft.com/en-us/windows-hardware/drivers/debugger/displaying-a-critical-section + struct CRITICAL_SECTION + debugInfo : Void* # PRTL_CRITICAL_SECTION_DEBUG + lockCount : LONG + recursionCount : LONG + owningThread : HANDLE + lockSemaphore : HANDLE + spinCount : UInt64 + end + + struct CONDITION_VARIABLE + ptr : Void* + end + + fun InitializeCriticalSectionAndSpinCount(lpCriticalSection : CRITICAL_SECTION*, dwSpinCount : DWORD) : BOOL + fun DeleteCriticalSection(lpCriticalSection : CRITICAL_SECTION*) + fun EnterCriticalSection(lpCriticalSection : CRITICAL_SECTION*) + fun TryEnterCriticalSection(lpCriticalSection : CRITICAL_SECTION*) : BOOL + fun LeaveCriticalSection(lpCriticalSection : CRITICAL_SECTION*) + + fun InitializeConditionVariable(conditionVariable : CONDITION_VARIABLE*) + fun SleepConditionVariableCS(conditionVariable : CONDITION_VARIABLE*, criticalSection : CRITICAL_SECTION*, dwMilliseconds : DWORD) : BOOL + fun WakeConditionVariable(conditionVariable : CONDITION_VARIABLE*) + fun WakeAllConditionVariable(conditionVariable : CONDITION_VARIABLE*) + fun Sleep(dwMilliseconds : DWORD) fun WaitForSingleObject(hHandle : HANDLE, dwMilliseconds : DWORD) : DWORD end diff --git a/src/lib_c/x86_64-windows-msvc/c/winbase.cr b/src/lib_c/x86_64-windows-msvc/c/winbase.cr index a5d2ccbe72ae..5b245479df25 100644 --- a/src/lib_c/x86_64-windows-msvc/c/winbase.cr +++ b/src/lib_c/x86_64-windows-msvc/c/winbase.cr @@ -31,6 +31,11 @@ lib LibC INFINITE = 0xFFFFFFFF + WAIT_OBJECT_0 = 0x00000000_u32 + WAIT_IO_COMPLETION = 0x000000C0_u32 + WAIT_TIMEOUT = 0x00000102_u32 + WAIT_FAILED = 0xFFFFFFFF_u32 + STARTF_USESTDHANDLES = 0x00000100 MOVEFILE_REPLACE_EXISTING = 0x1_u32 diff --git a/src/lib_c/x86_64-windows-msvc/c/winsock2.cr b/src/lib_c/x86_64-windows-msvc/c/winsock2.cr index d35a426597c5..5b12edb5b41f 100644 --- a/src/lib_c/x86_64-windows-msvc/c/winsock2.cr +++ b/src/lib_c/x86_64-windows-msvc/c/winsock2.cr @@ -1,6 +1,7 @@ require "./ws2def" require "./basetsd" require "./guiddef" +require "./winbase" @[Link("WS2_32")] lib LibC @@ -65,13 +66,8 @@ lib LibC WSA_INVALID_EVENT = Pointer(WSAEVENT).null WSA_MAXIMUM_WAIT_EVENTS = MAXIMUM_WAIT_OBJECTS WSA_WAIT_FAILED = WAIT_FAILED - STATUS_WAIT_0 = 0_i64 - WAIT_OBJECT_0 = ((STATUS_WAIT_0) + 0) WSA_WAIT_EVENT_0 = WAIT_OBJECT_0 - STATUS_USER_APC = 0xc0 - WAIT_IO_COMPLETION = STATUS_USER_APC WSA_WAIT_IO_COMPLETION = WAIT_IO_COMPLETION - WAIT_TIMEOUT = 258_i64 WSA_WAIT_TIMEOUT = WAIT_TIMEOUT WSA_INFINITE = INFINITE From 1e8e80fc336e5bf5f19d04ce4d733ff1c604b995 Mon Sep 17 00:00:00 2001 From: Quinton Miller Date: Fri, 24 Dec 2021 03:09:38 +0800 Subject: [PATCH 2/7] support `-Dgc_none` --- src/crystal/system/win32/thread.cr | 11 +++++++---- src/gc/boehm.cr | 9 ++++++++- src/gc/none.cr | 15 +++++++++++++-- src/lib_c/x86_64-windows-msvc/c/process.cr | 5 +++++ 4 files changed, 33 insertions(+), 7 deletions(-) create mode 100644 src/lib_c/x86_64-windows-msvc/c/process.cr diff --git a/src/crystal/system/win32/thread.cr b/src/crystal/system/win32/thread.cr index 77047c7bb1c8..194a9374e053 100644 --- a/src/crystal/system/win32/thread.cr +++ b/src/crystal/system/win32/thread.cr @@ -24,10 +24,13 @@ class Thread def initialize(&@func : ->) @th = uninitialized LibC::HANDLE - @th = LibGC.beginthreadex(nil, 0, ->(data : Void*) { - (data.as(Thread)).start - LibC::UInt.zero - }, self.as(Void*), 0, nil).as(LibC::HANDLE) + @th = GC.beginthreadex( + security: Pointer(Void).null, + stack_size: LibC::UInt.zero, + start_address: ->(data : Void*) { data.as(Thread).start; LibC::UInt.zero }, + arglist: self.as(Void*), + initflag: LibC::UInt.zero, + thrdaddr: Pointer(LibC::UInt).null) if @th.null? raise RuntimeError.from_errno("_beginthreadex") diff --git a/src/gc/boehm.cr b/src/gc/boehm.cr index 2bf9a4c75ce2..3845a85cba8f 100644 --- a/src/gc/boehm.cr +++ b/src/gc/boehm.cr @@ -227,7 +227,14 @@ module GC reclaimed_bytes_before_gc: stats.reclaimed_bytes_before_gc) end - {% unless flag?(:win32) %} + {% if flag?(:win32) %} + {% unless flag?(:preview_dll) %} + # :nodoc: + def self.beginthreadex(security : Void*, stack_size : LibC::UInt, start_address : Void* -> LibC::UInt, arglist : Void*, initflag : LibC::UInt, thrdaddr : LibC::UInt*) : LibC::HANDLE + LibGC.beginthreadex(security, stack_size, start_address, arglist, initflag, thrdaddr).as(LibC::HANDLE) + end + {% end %} + {% else %} # :nodoc: def self.pthread_create(thread : LibC::PthreadT*, attr : LibC::PthreadAttrT*, start : Void* -> Void*, arg : Void*) LibGC.pthread_create(thread, attr, start, arg) diff --git a/src/gc/none.cr b/src/gc/none.cr index 24ed4cb597b1..afbc346f842c 100644 --- a/src/gc/none.cr +++ b/src/gc/none.cr @@ -1,4 +1,6 @@ -{% unless flag?(:win32) %} +{% if flag?(:win32) %} + require "c/process" +{% else %} @[Link("pthread")] lib LibC end @@ -66,7 +68,16 @@ module GC reclaimed_bytes_before_gc: zero) end - {% unless flag?(:win32) %} + {% if flag?(:win32) %} + {% if flag?(:preview_dll) %} + {% raise "TODO: implement CreateThread" %} + {% else %} + # :nodoc: + def self.beginthreadex(security : Void*, stack_size : LibC::UInt, start_address : Void* -> LibC::UInt, arglist : Void*, initflag : LibC::UInt, thrdaddr : LibC::UInt*) : LibC::HANDLE + LibC._beginthreadex(security, stack_size, start_address, arglist, initflag, thrdaddr).as(LibC::HANDLE) + end + {% end %} + {% else %} # :nodoc: def self.pthread_create(thread : LibC::PthreadT*, attr : LibC::PthreadAttrT*, start : Void* -> Void*, arg : Void*) LibC.pthread_create(thread, attr, start, arg) diff --git a/src/lib_c/x86_64-windows-msvc/c/process.cr b/src/lib_c/x86_64-windows-msvc/c/process.cr new file mode 100644 index 000000000000..e567068fb1e7 --- /dev/null +++ b/src/lib_c/x86_64-windows-msvc/c/process.cr @@ -0,0 +1,5 @@ +require "lib_c" + +lib LibC + fun _beginthreadex(security : Void*, stack_size : UInt, start_address : Void* -> UInt, arglist : Void*, initflag : UInt, thrdaddr : UInt*) : Void* +end From 3ea07d45070710ee60b1e58db00cd8b997332201 Mon Sep 17 00:00:00 2001 From: Quinton Miller Date: Fri, 24 Dec 2021 03:15:32 +0800 Subject: [PATCH 3/7] `GC_beginthreadex` is fine in all cases --- src/gc/boehm.cr | 20 ++++++++------------ src/gc/none.cr | 14 ++++++-------- 2 files changed, 14 insertions(+), 20 deletions(-) diff --git a/src/gc/boehm.cr b/src/gc/boehm.cr index 3845a85cba8f..52882ff3c4ed 100644 --- a/src/gc/boehm.cr +++ b/src/gc/boehm.cr @@ -90,12 +90,8 @@ lib LibGC # Boehm GC requires to use its own thread manipulation routines instead of pthread's or Win32's {% if flag?(:win32) %} - {% if flag?(:preview_dll) %} - {% raise "TODO: implement GC_CreateThread" %} - {% else %} - fun beginthreadex = GC_beginthreadex(security : Void*, stack_size : LibC::UInt, start_address : Void* -> LibC::UInt, - arglist : Void*, initflag : LibC::UInt, thrdaddr : LibC::UInt*) : Void* - {% end %} + fun beginthreadex = GC_beginthreadex(security : Void*, stack_size : LibC::UInt, start_address : Void* -> LibC::UInt, + arglist : Void*, initflag : LibC::UInt, thrdaddr : LibC::UInt*) : Void* {% else %} fun pthread_create = GC_pthread_create(thread : LibC::PthreadT*, attr : LibC::PthreadAttrT*, start : Void* -> Void*, arg : Void*) : LibC::Int fun pthread_join = GC_pthread_join(thread : LibC::PthreadT, value : Void**) : LibC::Int @@ -228,12 +224,12 @@ module GC end {% if flag?(:win32) %} - {% unless flag?(:preview_dll) %} - # :nodoc: - def self.beginthreadex(security : Void*, stack_size : LibC::UInt, start_address : Void* -> LibC::UInt, arglist : Void*, initflag : LibC::UInt, thrdaddr : LibC::UInt*) : LibC::HANDLE - LibGC.beginthreadex(security, stack_size, start_address, arglist, initflag, thrdaddr).as(LibC::HANDLE) - end - {% end %} + # :nodoc: + def self.beginthreadex(security : Void*, stack_size : LibC::UInt, start_address : Void* -> LibC::UInt, arglist : Void*, initflag : LibC::UInt, thrdaddr : LibC::UInt*) : LibC::HANDLE + ret = LibGC.beginthreadex(security, stack_size, start_address, arglist, initflag, thrdaddr) + raise RuntimeError.from_errno("GC_beginthreadex") if ret.null? + ret.as(LibC::HANDLE) + end {% else %} # :nodoc: def self.pthread_create(thread : LibC::PthreadT*, attr : LibC::PthreadAttrT*, start : Void* -> Void*, arg : Void*) diff --git a/src/gc/none.cr b/src/gc/none.cr index afbc346f842c..16ecf129f16d 100644 --- a/src/gc/none.cr +++ b/src/gc/none.cr @@ -69,14 +69,12 @@ module GC end {% if flag?(:win32) %} - {% if flag?(:preview_dll) %} - {% raise "TODO: implement CreateThread" %} - {% else %} - # :nodoc: - def self.beginthreadex(security : Void*, stack_size : LibC::UInt, start_address : Void* -> LibC::UInt, arglist : Void*, initflag : LibC::UInt, thrdaddr : LibC::UInt*) : LibC::HANDLE - LibC._beginthreadex(security, stack_size, start_address, arglist, initflag, thrdaddr).as(LibC::HANDLE) - end - {% end %} + # :nodoc: + def self.beginthreadex(security : Void*, stack_size : LibC::UInt, start_address : Void* -> LibC::UInt, arglist : Void*, initflag : LibC::UInt, thrdaddr : LibC::UInt*) : LibC::HANDLE + ret = LibC._beginthreadex(security, stack_size, start_address, arglist, initflag, thrdaddr) + raise RuntimeError.from_errno("_beginthreadex") if ret.null? + ret.as(LibC::HANDLE) + end {% else %} # :nodoc: def self.pthread_create(thread : LibC::PthreadT*, attr : LibC::PthreadAttrT*, start : Void* -> Void*, arg : Void*) From 83ead7b58a3b9abad46c667780c2a0de67634fd0 Mon Sep 17 00:00:00 2001 From: Quinton Miller Date: Fri, 24 Dec 2021 04:06:36 +0800 Subject: [PATCH 4/7] format --- src/gc/boehm.cr | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/gc/boehm.cr b/src/gc/boehm.cr index 52882ff3c4ed..edf9463d258b 100644 --- a/src/gc/boehm.cr +++ b/src/gc/boehm.cr @@ -91,7 +91,7 @@ lib LibGC # Boehm GC requires to use its own thread manipulation routines instead of pthread's or Win32's {% if flag?(:win32) %} fun beginthreadex = GC_beginthreadex(security : Void*, stack_size : LibC::UInt, start_address : Void* -> LibC::UInt, - arglist : Void*, initflag : LibC::UInt, thrdaddr : LibC::UInt*) : Void* + arglist : Void*, initflag : LibC::UInt, thrdaddr : LibC::UInt*) : Void* {% else %} fun pthread_create = GC_pthread_create(thread : LibC::PthreadT*, attr : LibC::PthreadAttrT*, start : Void* -> Void*, arg : Void*) : LibC::Int fun pthread_join = GC_pthread_join(thread : LibC::PthreadT, value : Void**) : LibC::Int From 7a675994ab8bebef8c412ac85169c99722631844 Mon Sep 17 00:00:00 2001 From: Quinton Miller Date: Wed, 1 Jun 2022 16:24:18 +0800 Subject: [PATCH 5/7] fix wasi ci --- src/crystal/system/thread_condition_variable.cr | 4 +++- src/crystal/system/wasi/thread.cr | 16 ---------------- .../system/wasi/thread_condition_variable.cr | 16 ++++++++++++++++ 3 files changed, 19 insertions(+), 17 deletions(-) create mode 100644 src/crystal/system/wasi/thread_condition_variable.cr diff --git a/src/crystal/system/thread_condition_variable.cr b/src/crystal/system/thread_condition_variable.cr index fa6fc69546b2..ea5923601aec 100644 --- a/src/crystal/system/thread_condition_variable.cr +++ b/src/crystal/system/thread_condition_variable.cr @@ -20,7 +20,9 @@ class Thread end end -{% if flag?(:unix) %} +{% if flag?(:wasi) %} + require "./wasi/thread_condition_variable" +{% elsif flag?(:unix) %} require "./unix/pthread_condition_variable" {% elsif flag?(:win32) %} require "./win32/thread_condition_variable" diff --git a/src/crystal/system/wasi/thread.cr b/src/crystal/system/wasi/thread.cr index b10439852f55..805c7fbb77a6 100644 --- a/src/crystal/system/wasi/thread.cr +++ b/src/crystal/system/wasi/thread.cr @@ -54,20 +54,4 @@ class Thread # TODO: Implement Pointer(Void).null end - - # :nodoc: - # TODO: Implement - class ConditionVariable - def signal : Nil - end - - def broadcast : Nil - end - - def wait(mutex : Thread::Mutex) : Nil - end - - def wait(mutex : Thread::Mutex, time : Time::Span, &) - end - end end diff --git a/src/crystal/system/wasi/thread_condition_variable.cr b/src/crystal/system/wasi/thread_condition_variable.cr new file mode 100644 index 000000000000..eb205333acdd --- /dev/null +++ b/src/crystal/system/wasi/thread_condition_variable.cr @@ -0,0 +1,16 @@ +# TODO: Implement +class Thread + class ConditionVariable + def signal : Nil + end + + def broadcast : Nil + end + + def wait(mutex : Thread::Mutex) : Nil + end + + def wait(mutex : Thread::Mutex, time : Time::Span, &) + end + end +end From 9091e8360e1d1ca375f567a3b3afe73c618e687a Mon Sep 17 00:00:00 2001 From: Quinton Miller Date: Thu, 9 Jun 2022 13:47:59 +0800 Subject: [PATCH 6/7] Update src/crystal/system/win32/thread.cr MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: Johannes Müller --- src/crystal/system/win32/thread.cr | 3 --- 1 file changed, 3 deletions(-) diff --git a/src/crystal/system/win32/thread.cr b/src/crystal/system/win32/thread.cr index 194a9374e053..1b6562c9a029 100644 --- a/src/crystal/system/win32/thread.cr +++ b/src/crystal/system/win32/thread.cr @@ -32,9 +32,6 @@ class Thread initflag: LibC::UInt.zero, thrdaddr: Pointer(LibC::UInt).null) - if @th.null? - raise RuntimeError.from_errno("_beginthreadex") - end end # Used once to initialize the thread object representing the main thread of From f701feeb10fbeb2c887bc4bdcd38265a2f978c9a Mon Sep 17 00:00:00 2001 From: Quinton Miller Date: Thu, 7 Jul 2022 03:41:48 +0800 Subject: [PATCH 7/7] crystal tool format --- src/crystal/system/win32/thread.cr | 1 - 1 file changed, 1 deletion(-) diff --git a/src/crystal/system/win32/thread.cr b/src/crystal/system/win32/thread.cr index 1b6562c9a029..5e7ae49b30fc 100644 --- a/src/crystal/system/win32/thread.cr +++ b/src/crystal/system/win32/thread.cr @@ -31,7 +31,6 @@ class Thread arglist: self.as(Void*), initflag: LibC::UInt.zero, thrdaddr: Pointer(LibC::UInt).null) - end # Used once to initialize the thread object representing the main thread of