{ $Id$ This file is part of the Free Pascal run time library. Copyright (c) 2002 by Peter Vreman, member of the Free Pascal development team. Win32 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} unit systhrds; interface {$S-} type { the fields of this record are os dependent } { and they shouldn't be used in a program } { only the type TCriticalSection is important } PRTLCriticalSection = ^TRTLCriticalSection; TRTLCriticalSection = packed record DebugInfo : pointer; LockCount : longint; RecursionCount : longint; OwningThread : DWord; LockSemaphore : DWord; Reserved : DWord; end; { Include generic thread interface } {$i threadh.inc} implementation function SysGetCurrentThreadId : dword;forward; {***************************************************************************** Generic overloaded *****************************************************************************} { Include generic overloaded routines } {$i thread.inc} {***************************************************************************** Local WINApi imports *****************************************************************************} const { GlobalAlloc, GlobalFlags } GMEM_FIXED = 0; GMEM_ZEROINIT = 64; function TlsAlloc : DWord; stdcall;external 'kernel32' name 'TlsAlloc'; function TlsGetValue(dwTlsIndex : DWord) : pointer; stdcall;external 'kernel32' name 'TlsGetValue'; function TlsSetValue(dwTlsIndex : DWord;lpTlsValue : pointer) : LongBool; stdcall;external 'kernel32' name 'TlsSetValue'; function TlsFree(dwTlsIndex : DWord) : LongBool; stdcall;external 'kernel32' name 'TlsFree'; function CreateThread(lpThreadAttributes : pointer; dwStackSize : DWord; lpStartAddress : pointer;lpParameter : pointer; dwCreationFlags : DWord;var lpThreadId : DWord) : Dword; stdcall;external 'kernel32' name 'CreateThread'; procedure ExitThread(dwExitCode : DWord); stdcall;external 'kernel32' name 'ExitThread'; function GlobalAlloc(uFlags:DWord; dwBytes:DWORD):Pointer; stdcall;external 'kernel32' name 'GlobalAlloc'; function GlobalFree(hMem : Pointer):Pointer; stdcall;external 'kernel32' name 'GlobalFree'; procedure Sleep(dwMilliseconds: DWord); stdcall;external 'kernel32' name 'Sleep'; function WinSuspendThread (threadHandle : dword) : dword; stdcall;external 'kernel32' name 'SuspendThread'; function WinResumeThread (threadHandle : dword) : dword; stdcall;external 'kernel32' name 'ResumeThread'; function TerminateThread (threadHandle : dword; var exitCode : dword) : boolean; stdcall;external 'kernel32' name 'TerminateThread'; function GetLastError : dword; stdcall;external 'kernel32' name 'GetLastError'; function WaitForSingleObject (hHandle,Milliseconds: dword): dword; stdcall;external 'kernel32' name 'WaitForSingleObject'; function WinThreadSetPriority (threadHandle : dword; Prio: longint): boolean; stdcall;external 'kernel32' name 'SetThreadPriority'; function WinThreadGetPriority (threadHandle : dword): Integer; stdcall;external 'kernel32' name 'GetThreadPriority'; function WinGetCurrentThreadId : dword; stdcall;external 'kernel32' name 'GetCurrentThread'; function CreateEvent(lpEventAttributes:pointer;bManualReset:longbool;bInitialState:longbool;lpName:pchar):CARDINAL; stdcall; external 'kernel32' name 'CreateEventA'; function CloseHandle(hObject:CARDINAL):LONGBOOL; stdcall; external 'kernel32' name 'CloseHandle'; function ResetEvent(hEvent:CARDINAL):LONGBOOL; stdcall; external 'kernel32' name 'ResetEvent'; function SetEvent(hEvent:CARDINAL):LONGBOOL; stdcall; external 'kernel32' name 'SetEvent'; function PulseEvent(hEvent:THANDLE):CARDINAL {WINBOOL}; stdcall; external 'kernel32' name 'PulseEvent'; CONST WAIT_OBJECT_0 = 0; WAIT_ABANDONED_0 = $80; WAIT_TIMEOUT = $102; WAIT_IO_COMPLETION = $c0; WAIT_ABANDONED = $80; WAIT_FAILED = $ffffffff; {***************************************************************************** Threadvar support *****************************************************************************} {$ifdef HASTHREADVAR} const threadvarblocksize : dword = 0; var TLSKey : Dword; procedure SysInitThreadvar(var offset : dword;size : dword); begin offset:=threadvarblocksize; inc(threadvarblocksize,size); end; function SysRelocateThreadvar(offset : dword) : pointer; begin SysRelocateThreadvar:=TlsGetValue(tlskey)+Offset; end; procedure SysAllocateThreadVars; var dataindex : pointer; begin { we've to allocate the memory from system } { because the FPC heap management uses } { exceptions which use threadvars but } { these aren't allocated yet ... } { allocate room on the heap for the thread vars } dataindex:=pointer(GlobalAlloc(GMEM_FIXED or GMEM_ZEROINIT,threadvarblocksize)); TlsSetValue(tlskey,dataindex); end; procedure SysReleaseThreadVars; begin GlobalFree(TlsGetValue(tlskey)); end; { Include OS independent Threadvar initialization } {$i threadvr.inc} {$endif HASTHREADVAR} {***************************************************************************** Thread starting *****************************************************************************} type pthreadinfo = ^tthreadinfo; tthreadinfo = record f : tthreadfunc; p : pointer; stklen : cardinal; end; procedure DoneThread; begin { Release Threadvars } {$ifdef HASTHREADVAR} SysReleaseThreadVars; {$endif HASTHREADVAR} end; function ThreadMain(param : pointer) : integer; stdcall; var ti : tthreadinfo; begin {$ifdef HASTHREADVAR} { Allocate local thread vars, this must be the first thing, because the exception management and io depends on threadvars } SysAllocateThreadVars; {$endif HASTHREADVAR} { Copy parameter to local data } {$ifdef DEBUG_MT} writeln('New thread started, initialising ...'); {$endif DEBUG_MT} ti:=pthreadinfo(param)^; dispose(pthreadinfo(param)); { Initialize thread } InitThread(ti.stklen); { Start thread function } {$ifdef DEBUG_MT} writeln('Jumping to thread function'); {$endif DEBUG_MT} ThreadMain:=ti.f(ti.p); end; function SysBeginThread(sa : Pointer;stacksize : dword; ThreadFunction : tthreadfunc;p : pointer; creationFlags : dword; var ThreadId : DWord) : DWord; var ti : pthreadinfo; begin {$ifdef DEBUG_MT} writeln('Creating new thread'); {$endif DEBUG_MT} { Initialize multithreading if not done } if not IsMultiThread then begin {$ifdef HASTHREADVAR} { We're still running in single thread mode, setup the TLS } TLSKey:=TlsAlloc; InitThreadVars(@SysRelocateThreadvar); {$endif HASTHREADVAR} 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} writeln('Starting new thread'); {$endif DEBUG_MT} SysBeginThread:=CreateThread(sa,stacksize,@ThreadMain,ti,creationflags,threadid); end; procedure SysEndThread(ExitCode : DWord); begin DoneThread; ExitThread(ExitCode); end; procedure SysThreadSwitch; begin Sleep(0); end; function SysSuspendThread (threadHandle : dword) : dword; begin SysSuspendThread:=WinSuspendThread(threadHandle); end; function SysResumeThread (threadHandle : dword) : dword; begin SysResumeThread:=WinResumeThread(threadHandle); end; function SysKillThread (threadHandle : dword) : dword; var exitCode : dword; begin if not TerminateThread (threadHandle, exitCode) then SysKillThread := GetLastError else SysKillThread := 0; end; function SysWaitForThreadTerminate (threadHandle : dword; TimeoutMs : longint) : dword; begin if timeoutMs = 0 then dec (timeoutMs); // $ffffffff is INFINITE SysWaitForThreadTerminate := WaitForSingleObject(threadHandle, TimeoutMs); end; function SysThreadSetPriority (threadHandle : dword; Prio: longint): boolean; {-15..+15, 0=normal} begin SysThreadSetPriority:=WinThreadSetPriority(threadHandle,Prio); end; function SysThreadGetPriority (threadHandle : dword): Integer; begin SysThreadGetPriority:=WinThreadGetPriority(threadHandle); end; function SysGetCurrentThreadId : dword; begin SysGetCurrentThreadId:=WinGetCurrentThreadId; end; {***************************************************************************** Delphi/Win32 compatibility *****************************************************************************} procedure WinInitCriticalSection(var cs : TRTLCriticalSection); stdcall;external 'kernel32' name 'InitializeCriticalSection'; procedure WinDoneCriticalSection(var cs : TRTLCriticalSection); stdcall;external 'kernel32' name 'DeleteCriticalSection'; procedure WinEnterCriticalSection(var cs : TRTLCriticalSection); stdcall;external 'kernel32' name 'EnterCriticalSection'; procedure WinLeaveCriticalSection(var cs : TRTLCriticalSection); stdcall;external 'kernel32' name 'LeaveCriticalSection'; procedure SySInitCriticalSection(var cs); begin WinInitCriticalSection(PRTLCriticalSection(@cs)^); end; procedure SysDoneCriticalSection(var cs); begin WinDoneCriticalSection(PRTLCriticalSection(@cs)^); end; procedure SysEnterCriticalSection(var cs); begin WinEnterCriticalSection(PRTLCriticalSection(@cs)^); end; procedure SySLeaveCriticalSection(var cs); begin WinLeaveCriticalSection(PRTLCriticalSection(@cs)^); end; {***************************************************************************** Heap Mutex Protection *****************************************************************************} var HeapMutex : TRTLCriticalSection; procedure Win32HeapMutexInit; begin InitCriticalSection(heapmutex); end; procedure Win32HeapMutexDone; begin DoneCriticalSection(heapmutex); end; procedure Win32HeapMutexLock; begin EnterCriticalSection(heapmutex); end; procedure Win32HeapMutexUnlock; begin LeaveCriticalSection(heapmutex); end; const Win32MemoryMutexManager : TMemoryMutexManager = ( MutexInit : @Win32HeapMutexInit; MutexDone : @Win32HeapMutexDone; MutexLock : @Win32HeapMutexLock; MutexUnlock : @Win32HeapMutexUnlock; ); procedure InitHeapMutexes; begin SetMemoryMutexManager(Win32MemoryMutexManager); end; Const wrSignaled = 0; wrTimeout = 1; wrAbandoned= 2; wrError = 3; type Tbasiceventstate=record fhandle : THandle; flasterror : longint; end; plocaleventrec= ^tbasiceventstate; function intBasicEventCreate(EventAttributes : Pointer; AManualReset,InitialState : Boolean;const Name : ansistring):pEventState; begin new(plocaleventrec(result)); plocaleventrec(result)^.FHandle := CreateEvent(EventAttributes, AManualReset, InitialState,PChar(Name)); end; procedure intbasiceventdestroy(state:peventstate); begin closehandle(plocaleventrec(state)^.fhandle); dispose(plocaleventrec(state)); end; procedure intbasiceventResetEvent(state:peventstate); begin ResetEvent(plocaleventrec(state)^.FHandle) end; procedure intbasiceventSetEvent(state:peventstate); begin SetEvent(plocaleventrec(state)^.FHandle); end; function intbasiceventWaitFor(Timeout : Cardinal;state:peventstate) : longint; begin case WaitForSingleObject(plocaleventrec(state)^.fHandle, Timeout) of WAIT_ABANDONED: Result := wrAbandoned; WAIT_OBJECT_0: Result := wrSignaled; WAIT_TIMEOUT: Result := wrTimeout; WAIT_FAILED: begin Result := wrError; plocaleventrec(state)^.FLastError := GetLastError; end; else Result := wrError; end; end; function intRTLEventCreate: PRTLEvent; begin Result := PRTLEVENT(CreateEvent(nil, false, false, nil)); end; procedure intRTLEventDestroy(AEvent: PRTLEvent); begin CloseHandle(THANDLE(AEvent)); end; procedure intRTLEventSetEvent(AEvent: PRTLEvent); begin PulseEvent(THANDLE(AEvent)); end; CONST INFINITE=-1; procedure intRTLEventStartWait(AEvent: PRTLEvent); begin // nothing to do, win32 events stay signalled after being set end; procedure intRTLEventWaitFor(AEvent: PRTLEvent); begin WaitForSingleObject(THANDLE(AEvent), INFINITE); end; Var WinThreadManager : TThreadManager; Procedure SetWinThreadManager; begin With WinThreadManager do begin InitManager :=Nil; DoneManager :=Nil; BeginThread :=@SysBeginThread; EndThread :=@SysEndThread; SuspendThread :=@SysSuspendThread; ResumeThread :=@SysResumeThread; KillThread :=@SysKillThread; ThreadSwitch :=@SysThreadSwitch; WaitForThreadTerminate :=@SysWaitForThreadTerminate; ThreadSetPriority :=@SysThreadSetPriority; ThreadGetPriority :=@SysThreadGetPriority; GetCurrentThreadId :=@SysGetCurrentThreadId; InitCriticalSection :=@SysInitCriticalSection; DoneCriticalSection :=@SysDoneCriticalSection; EnterCriticalSection :=@SysEnterCriticalSection; LeaveCriticalSection :=@SysLeaveCriticalSection; {$ifdef HASTHREADVAR} InitThreadVar :=@SysInitThreadVar; RelocateThreadVar :=@SysRelocateThreadVar; AllocateThreadVars :=@SysAllocateThreadVars; ReleaseThreadVars :=@SysReleaseThreadVars; {$endif HASTHREADVAR} BasicEventCreate :=@intBasicEventCreate; BasicEventDestroy :=@intBasicEventDestroy; BasicEventResetEvent :=@intBasicEventResetEvent; BasicEventSetEvent :=@intBasicEventSetEvent; BasiceventWaitFor :=@intBasiceventWaitFor; RTLEventCreate :=@intRTLEventCreate; RTLEventDestroy :=@intRTLEventDestroy; RTLEventSetEvent :=@intRTLEventSetEvent; RTLEventStartWait :=@intRTLEventStartWait; RTLEventWaitFor :=@intRTLEventWaitFor; end; SetThreadManager(WinThreadManager); InitHeapMutexes; end; initialization SetWinThreadManager; end. { $Log$ Revision 1.15 2005-01-30 21:48:14 marco * stdcall added to few calls Revision 1.14 2004/12/28 14:20:03 marco * tthread patch from neli Revision 1.13 2004/12/26 13:46:45 peter * tthread uses systhrds Revision 1.12 2004/12/22 21:29:24 marco * rtlevent kraam. Checked (compile): Linux, FreeBSD, Darwin, Windows Check work: ask Neli. Revision 1.11 2004/05/23 15:30:13 marco * first try Revision 1.10 2004/01/21 14:15:42 florian * fixed win32 compilation Revision 1.9 2003/11/29 17:34:53 michael + Removed dummy variable from SetCthreadManager Revision 1.8 2003/11/27 10:28:41 michael + Patch from peter to fix make cycle Revision 1.7 2003/11/26 20:10:59 michael + New threadmanager implementation Revision 1.6 2003/10/01 21:00:09 peter * GetCurrentThreadHandle renamed to GetCurrentThreadId Revision 1.5 2003/09/17 15:06:36 peter * stdcall patch Revision 1.4 2003/03/27 17:14:27 armin * more platform independent thread routines, needs to be implemented for unix Revision 1.3 2003/03/24 16:12:01 jonas * BeginThread() now returns the thread handle instead of the threadid (needed because you have to free the handle after your thread is finished, and the threadid is already returned via a var-parameter) Revision 1.2 2002/10/31 13:45:44 carl * threadvar.inc -> threadvr.inc Revision 1.1 2002/10/16 06:27:30 michael + Renamed thread unit to systhrds Revision 1.1 2002/10/14 19:39:18 peter * threads unit added for thread support }