mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-10-24 20:31:36 +02:00

while waiting on a thread that isn't finished, but the waiting happens on the main thread. This fixes the hang in test tb0636.pp
297 lines
8.0 KiB
PHP
297 lines
8.0 KiB
PHP
{
|
|
This file is part of the Free Component Library (FCL)
|
|
Copyright (c) 1999-2000 by the Free Pascal development team
|
|
|
|
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.
|
|
|
|
**********************************************************************}
|
|
{****************************************************************************}
|
|
{* TThread *}
|
|
{****************************************************************************}
|
|
|
|
|
|
{$ifdef FPC_WASM_THREADS}
|
|
procedure TThread.CallOnTerminate;
|
|
|
|
begin
|
|
FOnTerminate(self);
|
|
end;
|
|
|
|
|
|
function TThread.GetPriority: TThreadPriority;
|
|
|
|
begin
|
|
GetPriority:=tpNormal;
|
|
end;
|
|
|
|
|
|
procedure TThread.SetPriority(Value: TThreadPriority);
|
|
|
|
begin
|
|
// Not supported
|
|
end;
|
|
|
|
|
|
procedure TThread.SetSuspended(Value: Boolean);
|
|
|
|
begin
|
|
if Value <> FSuspended then
|
|
if Value then
|
|
Suspend
|
|
else
|
|
Resume;
|
|
end;
|
|
|
|
|
|
procedure TThread.DoTerminate;
|
|
|
|
begin
|
|
if Assigned(FOnTerminate) then
|
|
Synchronize(@CallOnTerminate);
|
|
end;
|
|
|
|
|
|
function ThreadFunc(parameter: Pointer): ptrint;
|
|
|
|
Var
|
|
LThread : TThread Absolute parameter;
|
|
LFreeOnTerminate : Boolean;
|
|
|
|
begin
|
|
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('In threadfunc. Thread object: '+IntToStr(PTrUint(LThread)));{$ENDIF}
|
|
try
|
|
if LThread.FInitialSuspended then
|
|
begin
|
|
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('thread '+IntToStr(ptruint(LThread))+' waiting for RTLEvent '+IntToStr(ptruint(LThread.FSuspendEvent)));{$ENDIF}
|
|
RtlEventWaitFor(LThread.FSuspendEvent);
|
|
if (LThread.FTerminated) then
|
|
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('initially created suspended, but already terminated'){$ENDIF}
|
|
else if LThread.FSuspended then
|
|
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('thread '+IntToStr(PtrUint(LThread))+' initially created suspended, resumed, but still suspended?!'){$ENDIF}
|
|
else
|
|
begin
|
|
LThread.FInitialSuspended := false;
|
|
CurrentThreadVar := LThread;
|
|
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('going into LThread.Execute (1)');{$ENDIF}
|
|
LThread.Execute;
|
|
end
|
|
end
|
|
else
|
|
begin
|
|
// The suspend internal is needed due to bug 16884
|
|
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('Suspending internally');{$ENDIF}
|
|
LThread.FSuspendedInternal:=True;
|
|
RtlEventWaitFor(LThread.FSuspendEvent);
|
|
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('Internal suspend done.');{$ENDIF}
|
|
CurrentThreadVar := LThread;
|
|
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('going into LThread.Execute (2)');{$ENDIF}
|
|
LThread.Execute;
|
|
end;
|
|
except
|
|
on e: exception do
|
|
begin
|
|
LThread.FFatalException := TObject(AcquireExceptionObject);
|
|
if e is EThreadDestroyCalled then
|
|
LThread.FFreeOnTerminate := true;
|
|
end;
|
|
end;
|
|
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('thread done running');{$ENDIF}
|
|
Result := LThread.FReturnValue;
|
|
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('Result is '+IntToStr(Result));{$ENDIF}
|
|
LFreeOnTerminate := LThread.FreeOnTerminate;
|
|
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('Calling doterminate');{$ENDIF}
|
|
LThread.DoTerminate;
|
|
LThread.FFinished := True;
|
|
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('Thread Set to finished');{$ENDIF}
|
|
if LFreeOnTerminate then
|
|
begin
|
|
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('Thread '+IntToStr(ptruint(lthread))+' should be freed');{$ENDIF}
|
|
LThread.Free;
|
|
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('Thread freed');{$ENDIF}
|
|
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('Thread func calling EndThread');{$ENDIF}
|
|
EndThread(Result);
|
|
end;
|
|
end;
|
|
|
|
procedure TThread.SysCreate(CreateSuspended: Boolean; const StackSize: SizeUInt);
|
|
|
|
begin
|
|
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('In TThread.SysCreate');{$ENDIF}
|
|
FSuspendEvent := RtlEventCreate;
|
|
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('TThread.SysCreate: Created suspend event');{$ENDIF}
|
|
FSuspended := CreateSuspended;
|
|
FThreadReaped := false;
|
|
FInitialSuspended := CreateSuspended;
|
|
FSuspendedInternal := not CreateSuspended;
|
|
FFatalException := nil;
|
|
FHandle:=BeginThread(@ThreadFunc, Pointer(Self), FThreadID, StackSize);
|
|
if FHandle = TThreadID(0) then
|
|
begin
|
|
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('TThread.SysCreate: Failed to create thread');{$ENDIF}
|
|
raise EThread.create('Failed to create new thread');
|
|
end;
|
|
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('TThread.SysCreate: thread created');{$ENDIF}
|
|
end;
|
|
|
|
|
|
procedure TThread.SysDestroy;
|
|
|
|
begin
|
|
{ exception in constructor }
|
|
if not assigned(FSuspendEvent) then
|
|
exit;
|
|
{ exception in constructor }
|
|
if (FHandle = TThreadID(0)) then
|
|
begin
|
|
RtlEventDestroy(FSuspendEvent);
|
|
exit;
|
|
end;
|
|
{ Thread itself called destroy ? }
|
|
if (FThreadID = GetCurrentThreadID) then
|
|
begin
|
|
if not(FFreeOnTerminate) and not FFinished then
|
|
raise EThreadDestroyCalled.Create('A thread cannot destroy itself except by setting FreeOnTerminate and leaving!');
|
|
FFreeOnTerminate := false;
|
|
end
|
|
else
|
|
begin
|
|
{ avoid recursion}
|
|
FFreeOnTerminate := false;
|
|
{ you can't join yourself, so only for FThreadID<>GetCurrentThreadID }
|
|
{ and you can't join twice -> make sure we didn't join already }
|
|
if not FThreadReaped then
|
|
begin
|
|
Terminate;
|
|
if (FSuspendedInternal or FInitialSuspended) then
|
|
Resume;
|
|
WaitFor;
|
|
end;
|
|
end;
|
|
RtlEventDestroy(FSuspendEvent);
|
|
FFatalException.Free;
|
|
FFatalException := nil;
|
|
end;
|
|
|
|
|
|
procedure TThread.Resume;
|
|
|
|
begin
|
|
if FSuspendedInternal and (InterLockedExchange(longint(FSuspendedInternal),ord(false)) = longint(longbool(true))) then
|
|
begin
|
|
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('resuming thread after TThread construction '+IntToStr(ptruint(self)));{$ENDIF}
|
|
RtlEventSetEvent(FSuspendEvent);
|
|
end
|
|
else
|
|
begin
|
|
{ don't compare with ord(true) or ord(longbool(true)), }
|
|
{ becaue a longbool's "true" value is anyting <> false }
|
|
if FSuspended and
|
|
(InterLockedExchange(longint(FSuspended),longint(false)) <> longint(longbool(false))) then
|
|
begin
|
|
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('resuming '+IntToStr(ptruint(self)));{$ENDIF}
|
|
RtlEventSetEvent(FSuspendEvent);
|
|
end
|
|
end
|
|
end;
|
|
|
|
|
|
procedure TThread.Suspend;
|
|
|
|
begin
|
|
if FThreadID<>GetCurrentThreadID then
|
|
Raise EThread.create('Suspending one thread from inside another one is unsupported (because it is unsafe and deadlock prone) by *nix and posix operating systems');
|
|
{ don't compare with ord(true) or ord(longbool(true)), }
|
|
{ becaue a longbool's "true" value is anyting <> false }
|
|
if not FSuspended and
|
|
(InterLockedExchange(longint(FSuspended),longint(longbool(true))) = longint(longbool(false))) then
|
|
RtlEventWaitFor(FSuspendEvent)
|
|
end;
|
|
|
|
|
|
function TThread.WaitFor: Integer;
|
|
|
|
begin
|
|
If (MainThreadID=GetCurrentThreadID) then
|
|
{
|
|
FFinished is set after DoTerminate, which does a synchronize of OnTerminate,
|
|
so make sure synchronize works (or indeed any other synchronize that may be
|
|
in progress)
|
|
}
|
|
While not FFinished do
|
|
CheckSynchronize(100);
|
|
WaitFor:=WaitForThreadTerminate(FThreadID,-1);
|
|
{ should actually check for errors in WaitForThreadTerminate, but no }
|
|
{ error api is defined for that function }
|
|
FThreadReaped:=true;
|
|
end;
|
|
|
|
{$else FPC_WASM_THREADS}
|
|
procedure TThread.CallOnTerminate;
|
|
|
|
begin
|
|
end;
|
|
|
|
|
|
function TThread.GetPriority: TThreadPriority;
|
|
|
|
begin
|
|
GetPriority:=tpNormal;
|
|
end;
|
|
|
|
|
|
procedure TThread.SetPriority(Value: TThreadPriority);
|
|
|
|
begin
|
|
end;
|
|
|
|
|
|
procedure TThread.SetSuspended(Value: Boolean);
|
|
|
|
begin
|
|
end;
|
|
|
|
|
|
procedure TThread.DoTerminate;
|
|
|
|
begin
|
|
end;
|
|
|
|
|
|
procedure TThread.SysCreate(CreateSuspended: Boolean; const StackSize: SizeUInt);
|
|
|
|
begin
|
|
{IsMultiThread := TRUE; }
|
|
end;
|
|
|
|
|
|
procedure TThread.SysDestroy;
|
|
|
|
begin
|
|
end;
|
|
|
|
|
|
procedure TThread.Resume;
|
|
|
|
begin
|
|
end;
|
|
|
|
|
|
procedure TThread.Suspend;
|
|
|
|
begin
|
|
end;
|
|
|
|
|
|
function TThread.WaitFor: Integer;
|
|
|
|
begin
|
|
WaitFor:=0;
|
|
end;
|
|
{$endif FPC_WASM_THREADS}
|
|
|