{ This file is part of the Free Pascal run time library. Copyright (c) 2022 by Michael Van Canneyt, member of the Free Pascal development team. wasm threading support implementation See the file COPYING.FPC, included in this distribution, for details about the copyright. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. **********************************************************************} {$mode objfpc} {$modeswitch advancedrecords} {$DEFINE DEBUG_MT} unit wasmthreads; interface Procedure SetWasmThreadManager; implementation Uses WebAssembly, wasiapi; {***************************************************************************** System unit import *****************************************************************************} procedure fpc_threaderror; [external name 'FPC_THREADERROR']; Type TTimeLockResult = (tlrOK,tlrTimeout,tlrError); TFPWasmMutex = record _lock : Longint; _owner : Pointer; function TryLock : Boolean; function Lock : Boolean; function TimedLock(aTimeOut : Longint) : TTimeLockResult; function Unlock : Boolean; end; TFPWasmEvent = record _mutex : TFPWasmMutex; _isset : Boolean; end; PFPWasmThread = ^TFPWasmThread; TFPWasmThread = record ThreadID : Integer; Next : PFPWasmThread; Previous : PFPWasmThread; end; Var MainThread : TFPWasmThread; threadvarblocksize : dword = 0; TLSInitialized : Integer = 0; {$IFDEF DEBUG_MT} Type TSmallString = string[100]; Procedure SetTLSMemory(aValue : Pointer); begin fpc_wasm32_init_tls(aValue); end; Function GetTLSMemory : Pointer; begin Result:=fpc_wasm32_tls_base; end; Procedure RawWrite(var S : TSmallString); begin // ToDo end; {$ENDIF DEBUG_MT} procedure WasmInitThreadvar(var offset : dword;size : dword); begin threadvarblocksize:=align(threadvarblocksize, fpc_wasm32_tls_align); offset:=threadvarblocksize; inc(threadvarblocksize,size); end; procedure WasmAllocateThreadVars; var tlsMemBlock : pointer; tlsBlockSize : Integer; begin tlsBlockSize:=fpc_wasm32_tls_size; if threadvarblocksize<>tlsBlocksize then Writeln('Warning : block sizes differ: ',tlsBlocksize,'<>',threadvarblocksize,'(calculated) !'); // DataIndex:=Pointer(Fpmmap(nil,threadvarblocksize,3,MAP_PRIVATE+MAP_ANONYMOUS,-1,0)); FillChar(DataIndex^,threadvarblocksize,0); // pthread_setspecific(tlskey,dataindex); end; procedure WasmThreadCleanup(p: pointer); cdecl; {$ifdef DEBUG_MT} var s: TSmallString; // not an ansistring {$endif DEBUG_MT} begin {$ifdef DEBUG_MT} s := 'finishing externally started thread'#10; RawWrite(s); {$endif DEBUG_MT} { Restore tlskey value as it may already have been set to null, in which case a) DoneThread can't release the memory b) accesses to threadvars from DoneThread or anything it calls would allocate new threadvar memory } { clean up } DoneThread; pthread_setspecific(CleanupKey,nil); end; procedure HookThread; { Set up externally created thread } begin WasmAllocateThreadVars; InitThread(1000000000); pthread_setspecific(CleanupKey,getTlsMemory); end; function WasmRelocateThreadvar(offset : dword) : pointer; var P : Pointer; begin P:=GetTLSMemory; if (P=Nil) then begin HookThread; P:=GetTLSMemory; end; WasmRelocateThreadvar:=P+Offset; end; procedure WasmReleaseThreadVars; begin Fpmunmap(pointer(pthread_getspecific(tlskey)),threadvarblocksize); end; function WasmThreadMain(param : pointer) : pointer; var {$ifdef DEBUG_MT} s: TSmallString; // not an ansistring {$endif DEBUG_MT} begin {$ifdef DEBUG_MT} s := 'New thread started, initing threadvars'#10; RawWrite(s); {$endif DEBUG_MT} { Must be first, many system unit things depend on threadvars} WasmAllocateThreadVars; { Copy parameter to local data } {$ifdef DEBUG_MT} s := 'New thread started, initialising ...'#10; RawWrite(s); {$endif DEBUG_MT} ti:=pthreadinfo(param)^; { Initialize thread } InitThread(ti.stklen); dispose(pthreadinfo(param)); { Start thread function } {$ifdef DEBUG_MT} writeln('Jumping to thread function'); {$endif DEBUG_MT} WasmThreadMain:=pointer(ti.f(ti.p)); DoneThread; pthread_exit(WasmThreadMain); end; Procedure InitWasmTLS; begin if (InterLockedExchange(longint(TLSInitialized),1) = 0) then begin { We're still running in single thread mode, setup the TLS } pthread_key_create(@TLSKey,nil); InitThreadVars(@WasmRelocateThreadvar); { used to clean up threads that we did not create ourselves: a) the default value for a key (and hence also this one) in new threads is NULL, and if it's still like that when the thread terminates, nothing will happen b) if it's non-NULL, the destructor routine will be called when the thread terminates -> we will set it to 1 if the threadvar relocation routine is called from a thread we did not create, so that we can clean up everything at the end } pthread_key_create(@CleanupKey,@WasmthreadCleanup); end end; function WasmBeginThread(sa : Pointer;stacksize : PtrUInt; ThreadFunction : tthreadfunc;p : pointer; creationFlags : dword; var ThreadId : TThreadId) : TThreadID; var ti : pthreadinfo; thread_attr : pthread_attr_t; {$ifdef DEBUG_MT} S : TSmallString; {$endif DEBUG_MT} begin {$ifdef DEBUG_MT} S:='Creating new thread'; RawWrite(S); {$endif DEBUG_MT} { Initialize multithreading if not done } if not TLSInitialized then InitWasmTLS; if not IsMultiThread then begin { We're still running in single thread mode, lazy initialize thread support } LazyInitThreading; IsMultiThread:=true; end; { the only way to pass data to the newly created thread in a MT safe way, is to use the heap } new(ti); ti^.f:=ThreadFunction; ti^.p:=p; ti^.stklen:=stacksize; { call pthread_create } {$ifdef DEBUG_MT} S:='Starting new thread'; RawWrite(S); {$endif DEBUG_MT} pthread_attr_init(@thread_attr); {$if not defined(HAIKU)and not defined(BEOS) and not defined(ANDROID)} {$if defined (solaris) or defined (netbsd) } pthread_attr_setinheritsched(@thread_attr, PTHREAD_INHERIT_SCHED); {$else not solaris} pthread_attr_setinheritsched(@thread_attr, PTHREAD_EXPLICIT_SCHED); {$endif not solaris} {$ifend} // will fail under linux -- apparently unimplemented pthread_attr_setscope(@thread_attr, PTHREAD_SCOPE_PROCESS); // don't create detached, we need to be able to join (waitfor) on // the newly created thread! //pthread_attr_setdetachstate(@thread_attr, PTHREAD_CREATE_DETACHED); // set the stack size if (pthread_attr_setstacksize(@thread_attr, stacksize)<>0) or // and create the thread (pthread_create(ppthread_t(@threadid), @thread_attr, @ThreadMain,ti) <> 0) then begin dispose(ti); threadid := TThreadID(0); end; CBeginThread:=threadid; pthread_attr_destroy(@thread_attr); {$ifdef DEBUG_MT} Str(ptrint(CBeginThread),S); S:= 'BeginThread returning '+S; RawWrite(S); {$endif DEBUG_MT} end; procedure WasmEndThread(ExitCode : DWord); begin DoneThread; pthread_detach(pthread_t(pthread_self())); pthread_exit(pointer(ptrint(ExitCode))); end; function WasmSuspendThread (threadHandle : TThreadID) : dword; // Not supported begin result:=dword(-1); end; function WasmResumeThread (threadHandle : TThreadID) : dword; // Not supported begin result:=dword(-1); end; procedure WasmThreadSwitch; {give time to other threads} begin // Not supported end; function WasmKillThread (threadHandle : TThreadID) : dword; begin pthread_detach(pthread_t(threadHandle)); WasmKillThread := pthread_cancel(pthread_t(threadHandle)); end; function WasmCloseThread (threadHandle : TThreadID) : dword; begin result:=0; end; function WasmWaitForThreadTerminate (threadHandle : TThreadID; TimeoutMs : longint) : dword; {0=no timeout} var LResultP: Pointer; begin pthread_join(pthread_t(threadHandle), @LResultP); WasmWaitForThreadTerminate := dword(LResultP); end; function WasmThreadSetPriority (threadHandle : TThreadID; Prio: longint): boolean; {-15..+15, 0=normal} begin result:=false; end; function WasmThreadGetPriority (threadHandle : TThreadID): Integer; begin result:=0; end; function CGetCurrentThreadId : TThreadID; begin CGetCurrentThreadId := TThreadID (pthread_self()); end; procedure CSetThreadDebugNameA(threadHandle: TThreadID; const ThreadName: AnsiString); {$if defined(Linux) or defined(Android)} var CuttedName: AnsiString; {$endif} begin {$if defined(Linux) or defined(Android)} if ThreadName = '' then Exit; {$ifdef dynpthreads} if Assigned(pthread_setname_np) then {$endif dynpthreads} begin // length restricted to 16 characters including terminating null byte CuttedName:=Copy(ThreadName, 1, 15); if threadHandle=TThreadID(-1) then begin pthread_setname_np(pthread_self(), @CuttedName[1]); end else begin pthread_setname_np(pthread_t(threadHandle), @CuttedName[1]); end; end; {$elseif defined(Darwin) or defined(iphonesim)} {$ifdef dynpthreads} if Assigned(pthread_setname_np) then {$endif dynpthreads} begin // only allowed to set from within the thread if threadHandle=TThreadID(-1) then pthread_setname_np(@ThreadName[1]); end; {$else} {$Warning SetThreadDebugName needs to be implemented} {$endif} end; procedure CSetThreadDebugNameU(threadHandle: TThreadID; const ThreadName: UnicodeString); begin {$if defined(Linux) or defined(Android)} {$ifdef dynpthreads} if Assigned(pthread_setname_np) then {$endif dynpthreads} begin CSetThreadDebugNameA(threadHandle, AnsiString(ThreadName)); end; {$elseif defined(Darwin) or defined(iphonesim)} {$ifdef dynpthreads} if Assigned(pthread_setname_np) then {$endif dynpthreads} begin CSetThreadDebugNameA(threadHandle, AnsiString(ThreadName)); end; {$else} {$Warning SetThreadDebugName needs to be implemented} {$endif} end; {***************************************************************************** Delphi/Win32 compatibility *****************************************************************************} procedure CInitCriticalSection(var CS); var MAttr : pthread_mutexattr_t; res: longint; begin res:=pthread_mutexattr_init(@MAttr); if res=0 then begin res:=pthread_mutexattr_settype(@MAttr,longint(_PTHREAD_MUTEX_RECURSIVE)); if res=0 then res := pthread_mutex_init(@CS,@MAttr) else { No recursive mutex support :/ } fpc_threaderror end else res:= pthread_mutex_init(@CS,NIL); pthread_mutexattr_destroy(@MAttr); if res <> 0 then fpc_threaderror; end; procedure CEnterCriticalSection(var CS); begin if pthread_mutex_lock(@CS) <> 0 then fpc_threaderror end; function CTryEnterCriticalSection(var CS):longint; begin if pthread_mutex_Trylock(@CS)=0 then result:=1 // succes else result:=0; // failure end; procedure CLeaveCriticalSection(var CS); begin if pthread_mutex_unlock(@CS) <> 0 then fpc_threaderror end; procedure CDoneCriticalSection(var CS); begin { unlock as long as unlocking works to unlock it if it is recursive some Delphi code might call this function with a locked mutex } while pthread_mutex_unlock(@CS)=0 do ; if pthread_mutex_destroy(@CS) <> 0 then fpc_threaderror; end; {***************************************************************************** Semaphore routines *****************************************************************************} type TPthreadCondition = pthread_cond_t; TPthreadMutex = pthread_mutex_t; Tbasiceventstate=record FCondVar: TPthreadCondition; {$if defined(Linux) and not defined(Android)} FAttr: pthread_condattr_t; FClockID: longint; {$ifend} FEventSection: TPthreadMutex; FWaiters: longint; FIsSet, FManualReset, FDestroying : Boolean; end; plocaleventstate = ^tbasiceventstate; // peventstate=pointer; Const wrSignaled = 0; wrTimeout = 1; wrAbandoned= 2; wrError = 3; function IntBasicEventCreate(EventAttributes : Pointer; AManualReset,InitialState : Boolean;const Name : ansistring):pEventState; var MAttr : pthread_mutexattr_t; res : cint; {$if defined(Linux) and not defined(Android)} timespec: ttimespec; {$ifend} begin new(plocaleventstate(result)); plocaleventstate(result)^.FManualReset:=AManualReset; plocaleventstate(result)^.FWaiters:=0; plocaleventstate(result)^.FDestroying:=False; plocaleventstate(result)^.FIsSet:=InitialState; {$if defined(Linux) and not defined(Android)} res := pthread_condattr_init(@plocaleventstate(result)^.FAttr); if (res <> 0) then begin FreeMem(result); fpc_threaderror; end; if clock_gettime(CLOCK_MONOTONIC_RAW, @timespec) = 0 then begin res := pthread_condattr_setclock(@plocaleventstate(result)^.FAttr, CLOCK_MONOTONIC_RAW); end else begin res := -1; // No support for CLOCK_MONOTONIC_RAW end; if (res = 0) then begin plocaleventstate(result)^.FClockID := CLOCK_MONOTONIC_RAW; end else begin res := pthread_condattr_setclock(@plocaleventstate(result)^.FAttr, CLOCK_MONOTONIC); if (res = 0) then begin plocaleventstate(result)^.FClockID := CLOCK_MONOTONIC; end else begin pthread_condattr_destroy(@plocaleventstate(result)^.FAttr); FreeMem(result); fpc_threaderror; end; end; res := pthread_cond_init(@plocaleventstate(result)^.FCondVar, @plocaleventstate(result)^.FAttr); if (res <> 0) then begin pthread_condattr_destroy(@plocaleventstate(result)^.FAttr); FreeMem(result); fpc_threaderror; end; {$else} res := pthread_cond_init(@plocaleventstate(result)^.FCondVar, nil); if (res <> 0) then begin FreeMem(result); fpc_threaderror; end; {$ifend} res:=pthread_mutexattr_init(@MAttr); if res=0 then begin res:=pthread_mutexattr_settype(@MAttr,longint(_PTHREAD_MUTEX_RECURSIVE)); if Res=0 then Res:=pthread_mutex_init(@plocaleventstate(result)^.feventsection,@MAttr) else res:=pthread_mutex_init(@plocaleventstate(result)^.feventsection,nil); end else res:=pthread_mutex_init(@plocaleventstate(result)^.feventsection,nil); pthread_mutexattr_destroy(@MAttr); if res <> 0 then begin pthread_cond_destroy(@plocaleventstate(result)^.FCondVar); {$if defined(Linux) and not defined(Android)} pthread_condattr_destroy(@plocaleventstate(result)^.FAttr); {$ifend} FreeMem(result); fpc_threaderror; end; end; procedure Intbasiceventdestroy(state:peventstate); begin { safely mark that we are destroying this event } pthread_mutex_lock(@plocaleventstate(state)^.feventsection); plocaleventstate(state)^.FDestroying:=true; { send a signal to all threads that are waiting } pthread_cond_broadcast(@plocaleventstate(state)^.FCondVar); pthread_mutex_unlock(@plocaleventstate(state)^.feventsection); { now wait until they've finished their business } while (plocaleventstate(state)^.FWaiters <> 0) do cThreadSwitch; { and clean up } pthread_cond_destroy(@plocaleventstate(state)^.Fcondvar); {$if defined(Linux) and not defined(Android)} pthread_condattr_destroy(@plocaleventstate(state)^.FAttr); {$ifend} pthread_mutex_destroy(@plocaleventstate(state)^.FEventSection); dispose(plocaleventstate(state)); end; procedure IntbasiceventResetEvent(state:peventstate); begin pthread_mutex_lock(@plocaleventstate(state)^.feventsection); plocaleventstate(state)^.fisset:=false; pthread_mutex_unlock(@plocaleventstate(state)^.feventsection); end; procedure IntbasiceventSetEvent(state:peventstate); begin pthread_mutex_lock(@plocaleventstate(state)^.feventsection); plocaleventstate(state)^.Fisset:=true; if not(plocaleventstate(state)^.FManualReset) then pthread_cond_signal(@plocaleventstate(state)^.Fcondvar) else pthread_cond_broadcast(@plocaleventstate(state)^.Fcondvar); pthread_mutex_unlock(@plocaleventstate(state)^.feventsection); end; function IntbasiceventWaitFor(Timeout : Cardinal;state:peventstate) : longint; var timespec: ttimespec; errres: cint; isset: boolean; tnow : timeval; begin { safely check whether we are being destroyed, if so immediately return. } { otherwise (under the same mutex) increase the number of waiters } pthread_mutex_lock(@plocaleventstate(state)^.feventsection); if (plocaleventstate(state)^.FDestroying) then begin pthread_mutex_unlock(@plocaleventstate(state)^.feventsection); result := wrAbandoned; exit; end; { not a regular inc() because it may happen simulatneously with the } { interlockeddecrement() at the end } interlockedincrement(plocaleventstate(state)^.FWaiters); //Wait without timeout using pthread_cond_wait if Timeout = $FFFFFFFF then begin while (not plocaleventstate(state)^.FIsSet) and (not plocaleventstate(state)^.FDestroying) do pthread_cond_wait(@plocaleventstate(state)^.Fcondvar, @plocaleventstate(state)^.feventsection); end else begin //Wait with timeout using pthread_cond_timedwait {$if defined(Linux) and not defined(Android)} if clock_gettime(plocaleventstate(state)^.FClockID, @timespec) <> 0 then begin Result := Ord(wrError); Exit; end; timespec.tv_sec := timespec.tv_sec + (clong(timeout) div 1000); timespec.tv_nsec := ((clong(timeout) mod 1000) * 1000000) + (timespec.tv_nsec); {$else} // TODO: FIX-ME: Also use monotonic clock for other *nix targets fpgettimeofday(@tnow, nil); timespec.tv_sec := tnow.tv_sec + (clong(timeout) div 1000); timespec.tv_nsec := ((clong(timeout) mod 1000) * 1000000) + (tnow.tv_usec * 1000); {$ifend} if timespec.tv_nsec >= 1000000000 then begin inc(timespec.tv_sec); dec(timespec.tv_nsec, 1000000000); end; errres := 0; while (not plocaleventstate(state)^.FDestroying) and (not plocaleventstate(state)^.FIsSet) and (errres<>ESysETIMEDOUT) do errres := pthread_cond_timedwait(@plocaleventstate(state)^.Fcondvar, @plocaleventstate(state)^.feventsection, @timespec); end; isset := plocaleventstate(state)^.FIsSet; { if ManualReset=false, reset the event immediately. } if (plocaleventstate(state)^.FManualReset=false) then plocaleventstate(state)^.FIsSet := false; //check the results... if plocaleventstate(state)^.FDestroying then Result := wrAbandoned else if IsSet then Result := wrSignaled else begin if errres=ESysETIMEDOUT then Result := wrTimeout else Result := wrError; end; pthread_mutex_unlock(@plocaleventstate(state)^.feventsection); { don't put this above the previous pthread_mutex_unlock, because } { otherwise we can get errors in case an object is destroyed between } { end of the wait/sleep loop and the signalling above. } { The pthread_mutex_unlock above takes care of the memory barrier } interlockeddecrement(plocaleventstate(state)^.FWaiters); end; function intRTLEventCreate: PRTLEvent; var p:pintrtlevent; begin new(p); if not assigned(p) then fpc_threaderror; if pthread_cond_init(@p^.condvar, nil)<>0 then begin dispose(p); fpc_threaderror; end; if pthread_mutex_init(@p^.mutex, nil)<>0 then begin pthread_cond_destroy(@p^.condvar); dispose(p); fpc_threaderror; end; p^.isset:=false; result:=PRTLEVENT(p); end; procedure intRTLEventDestroy(AEvent: PRTLEvent); var p:pintrtlevent; begin p:=pintrtlevent(aevent); pthread_cond_destroy(@p^.condvar); pthread_mutex_destroy(@p^.mutex); dispose(p); end; procedure intRTLEventSetEvent(AEvent: PRTLEvent); var p:pintrtlevent; begin p:=pintrtlevent(aevent); pthread_mutex_lock(@p^.mutex); p^.isset:=true; pthread_cond_signal(@p^.condvar); pthread_mutex_unlock(@p^.mutex); end; procedure intRTLEventResetEvent(AEvent: PRTLEvent); var p:pintrtlevent; begin p:=pintrtlevent(aevent); pthread_mutex_lock(@p^.mutex); p^.isset:=false; pthread_mutex_unlock(@p^.mutex); end; procedure intRTLEventWaitFor(AEvent: PRTLEvent); var p:pintrtlevent; begin p:=pintrtlevent(aevent); pthread_mutex_lock(@p^.mutex); while not p^.isset do pthread_cond_wait(@p^.condvar, @p^.mutex); p^.isset:=false; pthread_mutex_unlock(@p^.mutex); end; procedure intRTLEventWaitForTimeout(AEvent: PRTLEvent;timeout : longint); var p : pintrtlevent; errres : cint; timespec : ttimespec; tnow : timeval; begin p:=pintrtlevent(aevent); fpgettimeofday(@tnow,nil); timespec.tv_sec:=tnow.tv_sec+(timeout div 1000); timespec.tv_nsec:=(timeout mod 1000)*1000000 + tnow.tv_usec*1000; if timespec.tv_nsec >= 1000000000 then begin inc(timespec.tv_sec); dec(timespec.tv_nsec, 1000000000); end; errres:=0; pthread_mutex_lock(@p^.mutex); while (not p^.isset) and (errres <> ESysETIMEDOUT) do begin errres:=pthread_cond_timedwait(@p^.condvar, @p^.mutex, @timespec); end; p^.isset:=false; pthread_mutex_unlock(@p^.mutex); end; type threadmethod = procedure of object; Function CInitThreads : Boolean; begin {$ifdef DEBUG_MT} Writeln('Entering InitThreads.'); {$endif} {$ifndef dynpthreads} Result:=True; {$else} Result:=LoadPthreads; {$endif} ThreadID := TThreadID (pthread_self()); {$ifdef DEBUG_MT} Writeln('InitThreads : ',Result); {$endif DEBUG_MT} // We assume that if you set the thread manager, the application is multithreading. InitCTLS; end; Function CDoneThreads : Boolean; begin {$ifndef dynpthreads} Result:=True; {$else} Result:=UnloadPthreads; {$endif} end; Var CThreadManager : TThreadManager; Procedure SetCThreadManager; begin With CThreadManager do begin InitManager :=@WasmInitThreads; DoneManager :=@WasmDoneThreads; BeginThread :=@WasmBeginThread; EndThread :=@WasmEndThread; SuspendThread :=@WasmSuspendThread; ResumeThread :=@WasmResumeThread; KillThread :=@WasmKillThread; ThreadSwitch :=@WasmThreadSwitch; CloseThread :=@WasmCloseThread; WaitForThreadTerminate :=@WasmWaitForThreadTerminate; ThreadSetPriority :=@WasmThreadSetPriority; ThreadGetPriority :=@WasmThreadGetPriority; GetCurrentThreadId :=@WasmGetCurrentThreadId; SetThreadDebugNameA :=@WasmSetThreadDebugNameA; SetThreadDebugNameU :=@WasmSetThreadDebugNameU; InitCriticalSection :=@WasmInitCriticalSection; DoneCriticalSection :=@WasmDoneCriticalSection; EnterCriticalSection :=@WasmEnterCriticalSection; TryEnterCriticalSection:=@WasmTryEnterCriticalSection; LeaveCriticalSection :=@WasmLeaveCriticalSection; InitThreadVar :=@WasmInitThreadVar; RelocateThreadVar :=@WasmRelocateThreadVar; AllocateThreadVars :=@WasmAllocateThreadVars; ReleaseThreadVars :=@WasmReleaseThreadVars; BasicEventCreate :=@intBasicEventCreate; BasicEventDestroy :=@intBasicEventDestroy; BasicEventResetEvent :=@intBasicEventResetEvent; BasicEventSetEvent :=@intBasicEventSetEvent; BasiceventWaitFor :=@intBasiceventWaitFor; rtlEventCreate :=@intrtlEventCreate; rtlEventDestroy :=@intrtlEventDestroy; rtlEventSetEvent :=@intrtlEventSetEvent; rtlEventResetEvent :=@intrtlEventResetEvent; rtleventWaitForTimeout :=@intrtleventWaitForTimeout; rtleventWaitFor :=@intrtleventWaitFor; end; SetThreadManager(CThreadManager); end; initialization if ThreadingAlreadyUsed then begin writeln('Threading has been used before cthreads was initialized.'); writeln('Make wasmthreads one of the first units in your uses clause.'); runerror(211); end; SetWasmThreadManager; finalization end.