Compare commits

...

18 Commits

Author SHA1 Message Date
kagamma
e18d37cd9a Merge branch 'fv-fixes' into 'main'
Fixes for FV TInputLine's GetData, SetData and DataSize functions

See merge request freepascal.org/fpc/source!581
2025-04-04 02:41:02 +00:00
florian
6c4d218b8d * use for threadvars on RiscV always the size optimization code path as loading addresses is expensive 2025-04-03 23:14:43 +02:00
Michaël Van Canneyt
fc43e66f05 * Wake main thread when a thread is auto freed 2025-04-03 17:12:20 +02:00
Michaël Van Canneyt
1a21ea41b8 * Correctly set current thread 2025-04-03 16:26:31 +02:00
Pierre Muller
3bf5c67485 Revert "Add missing dependency on types unit for math unit"
This reverts commit 1f01ba4bc0.
2025-04-03 12:07:18 +00:00
Pierre Muller
1f01ba4bc0 Add missing dependency on types unit for math unit 2025-04-03 11:58:59 +00:00
Pierre Muller
43538416e3 Handle ADR LDM and STM arm instructions
in taicpu.spilling_get_operation_type method
2025-04-03 11:58:59 +00:00
Michaël Van Canneyt
a797828619 * Some additional thread debugging statements 2025-04-03 11:59:51 +02:00
florian
fb126e32f9 * RtlUnwind destroys register, so save and restore them afterwards 2025-04-02 22:34:21 +02:00
Michaël Van Canneyt
9e9153b2d3 * forgot to commit 2025-04-02 16:27:34 +02:00
Michaël Van Canneyt
736fc12e55 * Allow wasm guest & host to be notified of wasm memory growth 2025-04-02 11:21:03 +02:00
florian
62236ec2bb * proper naming 2025-04-01 22:53:56 +02:00
florian
425ef662cc * patch by Pierre to fix spilling and jump handling of pseudo-instructions 2025-03-31 22:53:40 +02:00
Pierre Muller
282f4aa060 Add definitions of ALUSInt and ALUUInt types in java/jsystemh_types.inc 2025-03-31 13:36:13 +00:00
kagamma
24c7719ad5 Merge branch 'fv-reverts' into 'fv-fixes'
Reverts TInputLine.GetData changes

See merge request kagamma/source!1
2024-12-30 06:48:04 +00:00
kagamma
ef4a94541e Reverts TInputLine.GetData changes 2024-12-30 06:46:29 +00:00
kagamma
c8958f3ab7 Minor correction 2024-01-31 06:21:00 +00:00
kagamma
c2111cab51 Fixes for FV TInputLine's GetData, SetData and DataSize functions 2024-01-31 05:31:35 +00:00
14 changed files with 179 additions and 19 deletions

View File

@ -832,6 +832,21 @@ implementation
result:=operand_read;
A_STREX:
result:=operand_write;
A_LDM:
if opnr=0 then
result:=operand_readwrite
else
result:=operand_write;
A_STM:
if opnr=0 then
result:=operand_readwrite
else
result:=operand_read;
A_ADR:
if opnr=0 then
result:=operand_write
else
result:=operand_read;
else
internalerror(200403151);
end
@ -923,6 +938,21 @@ implementation
result:=operand_read;
A_STREX:
result:=operand_write;
A_LDM:
if opnr=0 then
result:=operand_readwrite
else
result:=operand_write;
A_STM:
if opnr=0 then
result:=operand_readwrite
else
result:=operand_read;
A_ADR:
if opnr=0 then
result:=operand_write
else
result:=operand_read;
else
begin
writeln(opcode);

View File

@ -326,8 +326,12 @@ implementation
else
reference_reset_symbol(tvref,current_asmdata.WeakRefAsmSymbol(gvs.mangledname,AT_DATA),0,sizeof(pint),[]);
{ Enable size optimization with -Os or PIC code is generated and PIC uses GOT }
size_opt:=(cs_opt_size in current_settings.optimizerswitches)
or ((cs_create_pic in current_settings.moduleswitches) and (tf_pic_uses_got in target_info.flags));
size_opt:={$if defined(RISCV)}
true
{$else defined(RISCV)}
(cs_opt_size in current_settings.optimizerswitches)
or ((cs_create_pic in current_settings.moduleswitches) and (tf_pic_uses_got in target_info.flags))
{$endif defined(RISCV)};
hreg_tv_rec:=NR_INVALID;
if size_opt then
begin

View File

@ -507,7 +507,8 @@ uses cutils, cclasses;
result:=operand_read;
// SB type
A_Bxx:
A_BEQZ,A_BNEZ,A_BLEZ,A_BGEZ,A_BLTZ,A_BGTZ,A_BGT,A_BLE,
A_BGTU,A_BLEU,A_Bxx:
result:=operand_read;
// S type

View File

@ -590,7 +590,7 @@ implementation
taicpu(hp1).condition:=C_GE;
end;
DebugMsg('Peephole SltuB2B performed', hp1);
DebugMsg('Peephole SltuB2B 1 performed', hp1);
RemoveInstr(p);
@ -632,7 +632,7 @@ implementation
taicpu(hp1).condition:=C_GE;
end;
DebugMsg('Peephole SltuB2B performed', hp1);
DebugMsg('Peephole SltuB2B 2 performed', hp1);
RemoveInstr(p);

View File

@ -552,6 +552,8 @@ implementation
begin
is_calljmp:=false;
case o of
A_BEQZ,A_BNEZ,A_BLEZ,A_BGEZ,A_BLTZ,A_BGTZ,A_BGT,A_BLE,
A_BGTU,A_BLEU,A_J,A_JR,
A_JAL,A_JALR,A_Bxx,A_CALL:
is_calljmp:=true;
else

View File

@ -1417,18 +1417,24 @@ END;
FUNCTION TInputLine.DataSize: Sw_Word;
VAR DSize: Sw_Word;
BEGIN
DSize := 0; { Preset zero datasize }
{$ifdef FV_UNICODE}
If (Validator <> Nil) AND (Data <> '') Then
DSize := Validator^.Transfer(Data, Nil,
vtDataSize); { Add validator size }
DataSize := SizeOf(Sw_String); { DataSize return the
actual size of the field
in record. Unlike
ShortString field which is
allocated on stack and thus
has variable size,
UnicodeString field's size
is determined by it's
data type size }
{$else FV_UNICODE}
DSize := 0; { Preset zero datasize }
If (Validator <> Nil) AND (Data <> Nil) Then
DSize := Validator^.Transfer(Data^, Nil,
vtDataSize); { Add validator size }
{$endif FV_UNICODE}
If (DSize <> 0) Then DataSize := DSize { Use validtor size }
Else DataSize := MaxLen + 1; { No validator use size }
{$endif FV_UNICODE}
END;
{--TInputLine---------------------------------------------------------------}
@ -1643,15 +1649,20 @@ END;
{---------------------------------------------------------------------------}
PROCEDURE TInputLine.SetData (Var Rec);
BEGIN
{$ifdef FV_UNICODE}
{ We do not check for Data <> Sw_PString_Empty
in the Unicode version, or else this function
will always fail to set the data }
if (Validator = Nil) OR (Validator^.Transfer(
Data Sw_PString_DeRef, @Rec, vtSetData) = 0) Then
Data := Sw_String(Rec);
{$else FV_UNICODE}
If Data <> Sw_PString_Empty Then Begin { Data ptr valid }
If (Validator = Nil) OR (Validator^.Transfer(
Data Sw_PString_DeRef, @Rec, vtSetData) = 0) Then { No validator/data }
{$ifdef FV_UNICODE}
Data := Sw_String(Rec);
{$else FV_UNICODE}
Move(Rec, Data^[0], DataSize); { Set our data }
{$endif FV_UNICODE}
End;
{$endif FV_UNICODE}
SelectAll(True); { Now select all }
END;

View File

@ -27,6 +27,8 @@ begin
P.SourcePath.Add('src');
// Logger
T:=P.Targets.AddUnit('wasm.logger.api.pas');
// Memutils
T:=P.Targets.AddUnit('wasm.memutils.pas');
// Timer
T:=P.Targets.AddUnit('wasm.timer.shared.pas');

View File

@ -0,0 +1,45 @@
{
This file is part of the Free Component Library
Webassembly memory utils.
Copyright (c) 2025 by Michael Van Canneyt michael@freepascal.org
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.
**********************************************************************}
unit wasm.memutils;
{$mode ObjFPC}{$H+}
interface
Type
TWasmGrowMemoryEvent = procedure(aPages : longint) of object;
var
MemGrowNotifyCallBack : TWasmGrowMemoryCallBack;
MemGrowNotifyEvent : TWasmGrowMemoryEvent;
implementation
procedure __wasm_memory_grow_notification(aPages : Longint); external 'wasmmem' name 'wasm_memory_grow_notification' ;
procedure MemNotify(aPages : longint);
begin
__wasm_memory_grow_notification(aPages);
if assigned(MemGrowNotifyCallBack) then
MemGrowNotifyCallBack(aPages);
if assigned(MemGrowNotifyEvent) then
MemGrowNotifyEvent(aPages);
end;
initialization
WasmGrowMemoryCallback:=@MemNotify;
end.

View File

@ -365,6 +365,20 @@ Type
ValUInt = Word;
{$endif CPU16}
{$if defined(CPUINT8)}
ALUSInt = ShortInt;
ALUUInt = Byte;
{$elseif defined(CPUINT16)}
ALUSInt = SmallInt;
ALUUInt = Word;
{$elseif defined(CPUINT32)}
ALUSInt = Longint;
ALUUInt = DWord;
{$elseif defined(CPUINT64)}
ALUSInt = Int64;
ALUUInt = QWord;
{$endif defined(CPUINT64)}
{ NativeInt and NativeUInt are Delphi compatibility types. Even though Delphi
has IntPtr and UIntPtr, the Delphi documentation for NativeInt states that
'The size of NativeInt is equivalent to the size of the pointer on the

View File

@ -71,6 +71,8 @@ begin
end
else
SysOSAlloc:=nil;
if assigned(WasmGrowMemoryCallback) then
WasmGrowMemoryCallback(grow_pages);
end;
{$ifdef FPC_WASM_THREADS}
if InitialHeapCriticalSectionInitialized then

View File

@ -66,17 +66,23 @@ const
sLineBreak = LineEnding;
DefaultTextLineBreakStyle : TTextLineBreakStyle = tlbsLF;
type
TWasmGrowMemoryCallBack = procedure(aGrowPages: longint);
var
argc: longint;
argv: PPAnsiChar;
envp: PPAnsiChar;
___fpc_wasm_suspender: WasmExternRef; section 'WebAssembly.Global';
WasmGrowMemoryCallback : TWasmGrowMemoryCallBack;
function __fpc_get_wasm_suspender: WasmExternRef;
procedure __fpc_set_wasm_suspender(v: WasmExternRef);
property __fpc_wasm_suspender: WasmExternRef read __fpc_get_wasm_suspender write __fpc_set_wasm_suspender;
Procedure DebugWriteln(aString : ShortString);
implementation

View File

@ -399,12 +399,14 @@ procedure WasiAllocateThreadVars; forward;
{$push}{$S-} // no stack checking for this procedure
procedure FPCWasmThreadStartPascal(tid: longint; start_arg: PWasmThread);
begin
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('FPCWasmThreadStartPascal(...)');{$ENDIF}
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('FPCWasmThreadStartPascal('+IntToStr(tid)+','+IntToStr(ptrint(start_arg))+')');{$ENDIF}
start_arg^.ID:=tid;
GlobalCurrentThread:=@start_arg;
GlobalCurrentThread:=start_arg;
GlobalIsMainThread:=0;
GlobalIsWorkerThread:=1;
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('Check : TID='+IntToStr(tid)+', start_arg_id='+IntToStr(start_arg^.ID)+', currentthread= '+IntTostr(ptrint(GetCurrentThreadID))+')');{$ENDIF}
{$IFDEF FPC_WASM_WORKER_THREADS_CAN_WAIT}
GlobalIsThreadBlockable:=1;
{$ELSE FPC_WASM_WORKER_THREADS_CAN_WAIT}
@ -610,6 +612,7 @@ begin
Result:=0;
end;
function WasiGetCurrentThreadId : TThreadID;
begin
Result:=GetSelfThread;

View File

@ -63,7 +63,7 @@ Var
LFreeOnTerminate : Boolean;
begin
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('In threadfunc. Thread object: '+IntToStr(PTrUint(LThread)));{$ENDIF}
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('In threadfunc. Thread object: '+IntToStr(PTrUint(LThread))+' thread id :'+IntToStr(ptrint(Lthread.FThreadID)));{$ENDIF}
try
if LThread.FInitialSuspended then
begin
@ -142,9 +142,11 @@ end;
procedure TThread.SysDestroy;
begin
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('TThread.SysDestroy: enter');{$ENDIF}
{ exception in constructor }
if not assigned(FSuspendEvent) then
exit;
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('TThread.SysDestroy: have suspendevent');{$ENDIF}
{ exception in constructor }
if (FHandle = TThreadID(0)) then
begin
@ -154,25 +156,43 @@ begin
{ Thread itself called destroy ? }
if (FThreadID = GetCurrentThreadID) then
begin
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('TThread.SysDestroy: '+IntToStr(PtrInt(FThreadID))+' = '+IntToStr(PtrInt(GetCurrentThreadID)));{$ENDIF}
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('TThread.SysDestroy: thread itself is freeing');{$ENDIF}
if not(FFreeOnTerminate) and not FFinished then
begin
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('TThread.SysDestroy: error condition');{$ENDIF}
raise EThreadDestroyCalled.Create('A thread cannot destroy itself except by setting FreeOnTerminate and leaving!');
end;
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('TThread.SysDestroy: clearing FreeOnTerminate');{$ENDIF}
FFreeOnTerminate := false;
end
else
begin
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('TThread.SysDestroy: other thread is freeing');{$ENDIF}
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('TThread.SysDestroy: '+IntToStr(PtrInt(FThreadID))+' = '+IntToStr(PtrInt(GetCurrentThreadID)));{$ENDIF}
{ 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
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('TThread.SysDestroy: reaping thread');{$ENDIF}
Terminate;
if (FSuspendedInternal or FInitialSuspended) then
begin
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('TThread.SysDestroy: resuming thread in order to reap');{$ENDIF}
Resume;
end;
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('TThread.SysDestroy: waiting on thread');{$ENDIF}
// Before calling WaitFor, signal main thread with WakeMainThread, so pending checksynchronize calls are handled.
if assigned(WakeMainThread) then
WakeMainThread(Self);
WaitFor;
end;
end;
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('TThread.SysDestroy: destroying RTL suspend event');{$ENDIF}
RtlEventDestroy(FSuspendEvent);
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('TThread.SysDestroy: freeing fatal exception if it exists');{$ENDIF}
FFatalException.Free;
FFatalException := nil;
end;
@ -188,6 +208,7 @@ begin
end
else
begin
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('resuming thread '+IntToStr(ptruint(self)));{$ENDIF}
{ don't compare with ord(true) or ord(longbool(true)), }
{ becaue a longbool's "true" value is anyting <> false }
if FSuspended and
@ -195,7 +216,8 @@ begin
begin
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('resuming '+IntToStr(ptruint(self)));{$ENDIF}
RtlEventSetEvent(FSuspendEvent);
end
end;
{$IFDEF DEBUGWASMTHREADS}DebugWriteln('resumed thread '+IntToStr(ptruint(self)));{$ENDIF}
end
end;

View File

@ -275,6 +275,8 @@ procedure CommonHandler(
var
Exc: TExceptObject;
code: Longint;
_oldebx,_oldedi,_oldesi,
_ebx,_edi,_esi: dword;
begin
if rec.ExceptionCode<>FPC_EXCEPTION_CODE then
begin
@ -297,7 +299,20 @@ begin
Exc.Frames:=rec.ExceptionInformation[3];
end;
asm
movl %ebx,_oldebx
movl %esi,_oldesi
movl %edi,_oldedi
end;
RtlUnwind(@frame,nil,@rec,nil);
asm
movl %ebx,_ebx
movl %esi,_esi
movl %edi,_edi
movl _oldebx,%ebx
movl _oldesi,%esi
movl _oldedi,%edi
end;
Exc.Refcount:=0;
Exc.SEHFrame:=@frame;
@ -312,6 +327,9 @@ begin
movl Exc.FObject,%eax
movl frame,%edx
movl TargetAddr,%ecx // load ebp-based var before changing ebp
movl _ebx,%ebx
movl _esi,%esi
movl _edi,%edi
movl TSEHFrame._EBP(%edx),%ebp
jmpl *%ecx
end;