Procedure signature encapsulation in RTTI implementation.

This commit is contained in:
Henrique Gottardi Werlang 2025-03-12 20:21:13 -03:00
parent b795f4ed7a
commit 86083aa850

View File

@ -24,7 +24,7 @@ uses
JSApi.JS, System.RTLConsts, System.Types, System.SysUtils, System.TypInfo; JSApi.JS, System.RTLConsts, System.Types, System.SysUtils, System.TypInfo;
{$ELSE} {$ELSE}
JS, RTLConsts, Types, SysUtils, TypInfo; JS, RTLConsts, Types, SysUtils, TypInfo;
{$ENDIF} {$ENDIF}
resourcestring resourcestring
SErrInvokeInvalidCodeAddr = 'CodeAddress is not a function'; SErrInvokeInvalidCodeAddr = 'CodeAddress is not a function';
@ -232,12 +232,32 @@ type
TRttiParameterArray = specialize TArray<TRttiParameter>; TRttiParameterArray = specialize TArray<TRttiParameter>;
TRttiProcedureSignature = class(TRttiObject)
private
FFlags: TProcedureFlags;
FParameters: TRttiParameterArray;
FReturnType: TRttiType;
function GetProcedureSignature: TProcedureSignature;
procedure LoadFlags;
procedure LoadParameters;
public
constructor Create(const Signature: TProcedureSignature);
class function Invoke(const Instance: TValue; const Args: array of TValue): TValue;
property Flags: TProcedureFlags read FFlags;
property Parameters: TRttiParameterArray read FParameters;
property ProcedureSignature: TProcedureSignature read GetProcedureSignature;
property ReturnType: TRttiType read FReturnType;
end;
{ TRttiMethod } { TRttiMethod }
TRttiMethod = class(TRttiMember) TRttiMethod = class(TRttiMember)
private private
FParameters: TRttiParameterArray; FProcedureSignature: TRttiProcedureSignature;
FParametersLoaded: Boolean;
function GetIsAsyncCall: Boolean; function GetIsAsyncCall: Boolean;
function GetIsClassMethod: Boolean; function GetIsClassMethod: Boolean;
@ -250,11 +270,12 @@ type
function GetMethodKind: TMethodKind; function GetMethodKind: TMethodKind;
function GetMethodTypeInfo: TTypeMemberMethod; function GetMethodTypeInfo: TTypeMemberMethod;
function GetProcedureFlags: TProcedureFlags; function GetProcedureFlags: TProcedureFlags;
function GetProcedureSignature: TRttiProcedureSignature;
function GetReturnType: TRttiType; function GetReturnType: TRttiType;
procedure LoadParameters;
protected protected
function GetName: String; override; function GetName: String; override;
property ProcedureSignature: TRttiProcedureSignature read GetProcedureSignature;
public public
function GetParameters: TRttiParameterArray; function GetParameters: TRttiParameterArray;
function Invoke(const Instance: TValue; const Args: array of TValue): TValue; function Invoke(const Instance: TValue; const Args: array of TValue): TValue;
@ -2173,7 +2194,7 @@ end;
function TRttiMethod.GetIsAsyncCall: Boolean; function TRttiMethod.GetIsAsyncCall: Boolean;
begin begin
Result := (pfAsync in GetProcedureFlags) or Assigned(ReturnType) and ReturnType.IsInstanceExternal and (ReturnType.AsInstanceExternal.ExternalName = 'Promise'); Result := pfAsync in GetProcedureFlags;
end; end;
function TRttiMethod.GetIsSafeCall: Boolean; function TRttiMethod.GetIsSafeCall: Boolean;
@ -2187,26 +2208,13 @@ begin
end; end;
function TRttiMethod.GetProcedureFlags: TProcedureFlags; function TRttiMethod.GetProcedureFlags: TProcedureFlags;
const
PROCEDURE_FLAGS: array[TProcedureFlag] of NativeInt = (1, 2, 4, 8, 16);
var
Flag: TProcedureFlag;
ProcedureFlags: NativeInt;
begin begin
ProcedureFlags := MethodTypeInfo.ProcSig.Flags; Result := ProcedureSignature.Flags;
Result := [];
for Flag := Low(PROCEDURE_FLAGS) to High(PROCEDURE_FLAGS) do
if PROCEDURE_FLAGS[Flag] and ProcedureFlags > 0 then
Result := Result + [Flag];
end; end;
function TRttiMethod.GetReturnType: TRttiType; function TRttiMethod.GetReturnType: TRttiType;
begin begin
Result := Pool.GetType(MethodTypeInfo.ProcSig.ResultType); Result := ProcedureSignature.ReturnType;
end; end;
function TRttiMethod.GetName: String; function TRttiMethod.GetName: String;
@ -2217,48 +2225,17 @@ begin
Result := Result.SubString(0, Result.IndexOf('$')); Result := Result.SubString(0, Result.IndexOf('$'));
end; end;
procedure TRttiMethod.LoadParameters; function TRttiMethod.GetProcedureSignature: TRttiProcedureSignature;
const
FLAGS_CONVERSION: array[TParamFlag] of NativeInt = (1, 2, 4, 8, 16, 32);
var
A: Integer;
Flag: TParamFlag;
Param: TProcedureParam;
RttiParam: TRttiParameter;
MethodParams: TProcedureParams;
begin begin
FParametersLoaded := True; if not Assigned(FProcedureSignature) then
MethodParams := MethodTypeInfo.ProcSig.Params; FProcedureSignature := TRttiProcedureSignature.Create;
SetLength(FParameters, Length(MethodParams)); Result := FProcedureSignature;
for A := Low(FParameters) to High(FParameters) do
begin
Param := MethodParams[A];
RttiParam := TRttiParameter.Create(Self, Param);
RttiParam.FName := Param.Name;
RttiParam.FParamType := Pool.GetType(Param.TypeInfo);
for Flag := Low(FLAGS_CONVERSION) to High(FLAGS_CONVERSION) do
if FLAGS_CONVERSION[Flag] and Param.Flags > 0 then
RttiParam.FFlags := RttiParam.FFlags + [Flag];
FParameters[A] := RttiParam;
end;
end; end;
function TRttiMethod.GetParameters: TRttiParameterArray; function TRttiMethod.GetParameters: TRttiParameterArray;
begin begin
if not FParametersLoaded then Result := ProcedureSignature.Parameters;
LoadParameters;
Result := FParameters;
end; end;
function TRttiMethod.Invoke(const Instance: TValue; const Args: array of TValue): TValue; function TRttiMethod.Invoke(const Instance: TValue; const Args: array of TValue): TValue;
@ -2885,8 +2862,84 @@ begin
Result := TValue.specialize From<specialize TArray<T>>(arr); Result := TValue.specialize From<specialize TArray<T>>(arr);
end; end;
{ TRttiProcedureSignature }
constructor TRttiProcedureSignature.Create(const Signature: TProcedureSignature);
begin
inherited Create(nil, Signature);
FReturnType := Pool.GetType(Signature.ResultType);
LoadFlags;
LoadParameters;
end;
procedure TRttiProcedureSignature.LoadFlags;
const
PROCEDURE_FLAGS: array[TProcedureFlag] of NativeInt = (1, 2, 4, 8, 16);
var
Flag: TProcedureFlag;
ProcedureFlags: NativeInt;
begin
ProcedureFlags := ProcedureSignature.Flags;
FFlags := [];
for Flag := Low(PROCEDURE_FLAGS) to High(PROCEDURE_FLAGS) do
if PROCEDURE_FLAGS[Flag] and ProcedureFlags > 0 then
FFlags := FFlags + [Flag];
if Assigned(ReturnType) and ReturnType.IsInstanceExternal and (ReturnType.AsInstanceExternal.ExternalName = 'Promise') then
FFlags := FFlags + [pfAsync];
end;
procedure TRttiProcedureSignature.LoadParameters;
const
FLAGS_CONVERSION: array[TParamFlag] of NativeInt = (1, 2, 4, 8, 16, 32);
var
A: Integer;
Flag: TParamFlag;
Param: TProcedureParam;
RttiParam: TRttiParameter;
MethodParams: TProcedureParams;
begin
MethodParams := ProcedureSignature.Params;
SetLength(FParameters, Length(MethodParams));
for A := Low(FParameters) to High(FParameters) do
begin
Param := MethodParams[A];
RttiParam := TRttiParameter.Create(Self, Param);
RttiParam.FName := Param.Name;
RttiParam.FParamType := Pool.GetType(Param.TypeInfo);
for Flag := Low(FLAGS_CONVERSION) to High(FLAGS_CONVERSION) do
if FLAGS_CONVERSION[Flag] and Param.Flags > 0 then
RttiParam.FFlags := RttiParam.FFlags + [Flag];
FParameters[A] := RttiParam;
end;
end;
function TRttiProcedureSignature.GetProcedureSignature: TProcedureSignature;
begin
Result := TProcedureSignature(inherited Handle);
end;
class function TRttiProcedureSignature.Invoke(const Instance: TValue; const Args: array of TValue): TValue;
begin
end;
end. end.