mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-04-06 00:28:05 +02:00
* Fix variant record definition for CIF
This commit is contained in:
parent
79ad9fe924
commit
7aaad1868b
@ -41,27 +41,46 @@ begin
|
|||||||
Dispose(t);
|
Dispose(t);
|
||||||
end;
|
end;
|
||||||
|
|
||||||
|
|
||||||
function TypeInfoToFFIType(aTypeInfo: PTypeInfo; aFlags: TParamFlags): pffi_type; forward;
|
function TypeInfoToFFIType(aTypeInfo: PTypeInfo; aFlags: TParamFlags): pffi_type; forward;
|
||||||
|
function ArgIsIndirect(aKind: TTypeKind; aFlags: TParamFlags; aIsResult: Boolean): Boolean; forward;
|
||||||
|
|
||||||
function RecordOrObjectToFFIType(aTypeInfo: PTypeInfo): pffi_type;
|
function RecordOrObjectToFFIType(aTypeInfo: PTypeInfo): pffi_type;
|
||||||
var
|
var
|
||||||
curindex: SizeInt;
|
curindex: SizeInt;
|
||||||
elements: Tpffi_typeArray;
|
elements: Tpffi_typeArray;
|
||||||
|
|
||||||
procedure AddElement(t: pffi_type);
|
function AddElement(t: pffi_type) : Integer;
|
||||||
|
var
|
||||||
|
aCif : ffi_cif;
|
||||||
|
t2 : ffi_type;
|
||||||
|
|
||||||
begin
|
begin
|
||||||
if curindex = Length(elements) then begin
|
Result:=0;
|
||||||
|
if assigned(t) then
|
||||||
|
begin
|
||||||
|
aCIF:=Default(ffi_cif);
|
||||||
|
FillChar(aCIF,SizeOf(aCIF),0);
|
||||||
|
t2:=t^;
|
||||||
|
if ffi_prep_cif(@aCIF, FFI_DEFAULT_ABI, 0, @t2, Nil) = FFI_OK then
|
||||||
|
Result:=t2.size;
|
||||||
|
end;
|
||||||
|
if curindex = Length(elements) then
|
||||||
SetLength(elements, Length(elements) * 2);
|
SetLength(elements, Length(elements) * 2);
|
||||||
end;
|
|
||||||
elements[curindex] := t;
|
elements[curindex] := t;
|
||||||
Inc(curindex);
|
Inc(curindex);
|
||||||
end;
|
end;
|
||||||
|
|
||||||
var
|
var
|
||||||
td, fieldtd: PTypeData;
|
td, fieldtd: PTypeData;
|
||||||
i, j, curoffset, remoffset: SizeInt;
|
i, j, asize,lastoffset, curoffset, remoffset: SizeInt;
|
||||||
field: PManagedField;
|
field: PManagedField;
|
||||||
ffitype: pffi_type;
|
ffitype: pffi_type;
|
||||||
|
{$IFDEF TESTCIFSIZE}
|
||||||
|
aCif : ffi_cif;
|
||||||
|
r2 : ffi_type;
|
||||||
|
{$ENDIF}
|
||||||
|
|
||||||
begin
|
begin
|
||||||
td := GetTypeData(aTypeInfo);
|
td := GetTypeData(aTypeInfo);
|
||||||
if td^.TotalFieldCount = 0 then
|
if td^.TotalFieldCount = 0 then
|
||||||
@ -71,18 +90,21 @@ begin
|
|||||||
FillChar(Result^, SizeOf(Result), 0);
|
FillChar(Result^, SizeOf(Result), 0);
|
||||||
Result^._type := _FFI_TYPE_STRUCT;
|
Result^._type := _FFI_TYPE_STRUCT;
|
||||||
Result^.elements := Nil;
|
Result^.elements := Nil;
|
||||||
|
lastoffset := -1;
|
||||||
curoffset := 0;
|
curoffset := 0;
|
||||||
curindex := 0;
|
curindex := 0;
|
||||||
|
asize := 0;
|
||||||
field := PManagedField(PByte(@td^.TotalFieldCount) + SizeOf(td^.TotalFieldCount));
|
field := PManagedField(PByte(@td^.TotalFieldCount) + SizeOf(td^.TotalFieldCount));
|
||||||
{ assume first that there are no paddings }
|
{ assume first that there are no paddings }
|
||||||
SetLength(elements, td^.TotalFieldCount);
|
SetLength(elements, td^.TotalFieldCount);
|
||||||
for i := 0 to td^.TotalFieldCount - 1 do begin
|
for i := 0 to td^.TotalFieldCount - 1 do begin
|
||||||
{ ToDo: what about fields that are larger that what we have currently? }
|
curoffset := field^.FldOffset;
|
||||||
if field^.FldOffset < curoffset then begin
|
if (curoffset <= lastoffset) then begin
|
||||||
Inc(field);
|
Inc(field);
|
||||||
Continue;
|
Continue;
|
||||||
end;
|
end;
|
||||||
remoffset := field^.FldOffset - curoffset;
|
lastoffset:=field^.FldOffset;
|
||||||
|
remoffset := curoffset-(lastoffset-aSize);
|
||||||
{ insert padding elements }
|
{ insert padding elements }
|
||||||
while remoffset >= SizeOf(QWord) do begin
|
while remoffset >= SizeOf(QWord) do begin
|
||||||
AddElement(@ffi_type_uint64);
|
AddElement(@ffi_type_uint64);
|
||||||
@ -102,21 +124,22 @@ begin
|
|||||||
end;
|
end;
|
||||||
{ now add the real field type (Note: some are handled differently from
|
{ now add the real field type (Note: some are handled differently from
|
||||||
being passed as arguments, so we handle those here) }
|
being passed as arguments, so we handle those here) }
|
||||||
|
aSize:=0;
|
||||||
if field^.TypeRef^.Kind = tkObject then
|
if field^.TypeRef^.Kind = tkObject then
|
||||||
AddElement(RecordOrObjectToFFIType(field^.TypeRef))
|
aSize:=AddElement(RecordOrObjectToFFIType(field^.TypeRef))
|
||||||
else if field^.TypeRef^.Kind = tkSString then begin
|
else if field^.TypeRef^.Kind = tkSString then begin
|
||||||
fieldtd := GetTypeData(field^.TypeRef);
|
fieldtd := GetTypeData(field^.TypeRef);
|
||||||
for j := 0 to fieldtd^.MaxLength + 1 do
|
for j := 0 to fieldtd^.MaxLength + 1 do
|
||||||
AddElement(@ffi_type_uint8);
|
aSize:=aSize+AddElement(@ffi_type_uint8);
|
||||||
end else if field^.TypeRef^.Kind = tkArray then begin
|
end else if field^.TypeRef^.Kind = tkArray then begin
|
||||||
fieldtd := GetTypeData(field^.TypeRef);
|
fieldtd := GetTypeData(field^.TypeRef);
|
||||||
ffitype := TypeInfoToFFIType(fieldtd^.ArrayData.ElType, []);
|
ffitype := TypeInfoToFFIType(fieldtd^.ArrayData.ElType, []);
|
||||||
for j := 0 to fieldtd^.ArrayData.ElCount - 1 do
|
for j := 0 to fieldtd^.ArrayData.ElCount - 1 do
|
||||||
AddElement(ffitype);
|
aSize:=aSize+AddElement(ffitype);
|
||||||
end else
|
end else
|
||||||
AddElement(TypeInfoToFFIType(field^.TypeRef, []));
|
aSize:=AddElement(TypeInfoToFFIType(field^.TypeRef, []));
|
||||||
|
lastoffset:=lastOffset+aSize;
|
||||||
Inc(field);
|
Inc(field);
|
||||||
curoffset := field^.FldOffset;
|
|
||||||
end;
|
end;
|
||||||
{ add a final Nil element }
|
{ add a final Nil element }
|
||||||
AddElement(Nil);
|
AddElement(Nil);
|
||||||
@ -124,6 +147,14 @@ begin
|
|||||||
SetLength(elements, curindex);
|
SetLength(elements, curindex);
|
||||||
{ this is a bit cheeky, but it works }
|
{ this is a bit cheeky, but it works }
|
||||||
Tpffi_typeArray(Result^.elements) := elements;
|
Tpffi_typeArray(Result^.elements) := elements;
|
||||||
|
{$IFDEF TESTCIFSIZE}
|
||||||
|
aCIF:=Default(ffi_cif);
|
||||||
|
r2:=Result^;
|
||||||
|
if ffi_prep_cif(@aCIF, FFI_DEFAULT_ABI, 0, @R2, Nil) = FFI_OK then
|
||||||
|
Writeln('Rec size ',R2.size,' (expected: ',td^.RecSize,')')
|
||||||
|
else
|
||||||
|
Writeln('Fail');
|
||||||
|
{$ENDIF}
|
||||||
end;
|
end;
|
||||||
|
|
||||||
function SetToFFIType(aSize: SizeInt): pffi_type;
|
function SetToFFIType(aSize: SizeInt): pffi_type;
|
||||||
@ -183,7 +214,7 @@ begin
|
|||||||
Result := @ffi_type_void;
|
Result := @ffi_type_void;
|
||||||
if Assigned(aTypeInfo) then begin
|
if Assigned(aTypeInfo) then begin
|
||||||
td := GetTypeData(aTypeInfo);
|
td := GetTypeData(aTypeInfo);
|
||||||
if aFlags * [pfArray, pfOut, pfVar, pfConstRef] <> [] then
|
if ArgIsIndirect(aTypeInfo^.Kind,aFlags,False) then
|
||||||
Result := @ffi_type_pointer
|
Result := @ffi_type_pointer
|
||||||
else
|
else
|
||||||
case aTypeInfo^.Kind of
|
case aTypeInfo^.Kind of
|
||||||
@ -294,7 +325,13 @@ begin
|
|||||||
if (aKind = tkSString) or
|
if (aKind = tkSString) or
|
||||||
(aIsResult and (aKind in ResultTypeNeedsIndirection)) or
|
(aIsResult and (aKind in ResultTypeNeedsIndirection)) or
|
||||||
(aFlags * [pfArray, pfOut, pfVar, pfConstRef] <> []) or
|
(aFlags * [pfArray, pfOut, pfVar, pfConstRef] <> []) or
|
||||||
((aKind = tkUnknown) and (pfConst in aFlags)) then
|
((aKind = tkUnknown) and (pfConst in aFlags))
|
||||||
|
// This is true for all CPUs except sparc64/xtensa and i386/X86_64 on windows.
|
||||||
|
// The latter 2 are handled by the i386-specific invoke, so need not concern us here.
|
||||||
|
{$IF NOT (DEFINED(CPUSPARC64) or DEFINED(CPUXTENSA))}
|
||||||
|
or (aKind=tkVariant)
|
||||||
|
{$ENDIF}
|
||||||
|
then
|
||||||
Result := True;
|
Result := True;
|
||||||
end;
|
end;
|
||||||
|
|
||||||
@ -426,6 +463,7 @@ var
|
|||||||
usevalues, retparam: Boolean;
|
usevalues, retparam: Boolean;
|
||||||
kind: TTypeKind;
|
kind: TTypeKind;
|
||||||
types: ppffi_type;
|
types: ppffi_type;
|
||||||
|
|
||||||
begin
|
begin
|
||||||
if not (fcfStatic in aFlags) and (Length(aArgInfos) = 0) then
|
if not (fcfStatic in aFlags) and (Length(aArgInfos) = 0) then
|
||||||
raise EInvocationError.Create(SErrMissingSelfParam);
|
raise EInvocationError.Create(SErrMissingSelfParam);
|
||||||
|
Loading…
Reference in New Issue
Block a user