Postponing building of VMT until inheritance chain is specialized

This commit does 3 changes:
1. Introduce new option `oo_inherits_not_specialized` indicating if
   somewhere in the inheritance chain of an object there is a non
   specialized generic parameter
2. Avoid building the VMT for an object which has a generic parameter in
   the inheritance chain (fixes #40983)
3. When no vmt is build `insert_struct_hidden_paras` usually called as
   part of `build_vmt` will be called seperately to add missing
   parameters
This commit is contained in:
Frederic Kehrein 2024-11-02 22:13:12 +01:00 committed by FPK
parent 60690e379e
commit 05b73f1523
11 changed files with 145 additions and 6 deletions

View File

@ -1134,7 +1134,11 @@ implementation
{ Build VMT indexes, skip for type renaming and forward classes } { Build VMT indexes, skip for type renaming and forward classes }
if not istyperenaming and if not istyperenaming and
not(oo_is_forward in tobjectdef(hdef).objectoptions) then not(oo_is_forward in tobjectdef(hdef).objectoptions) then
build_vmt(tobjectdef(hdef)); if not (oo_inherits_not_specialized in tobjectdef(hdef).objectoptions) then
build_vmt(tobjectdef(hdef))
else
{ update the procdevs to add hidden self param }
insert_struct_hidden_paras(tobjectdef(hdef));
{ In case of an objcclass, verify that all methods have a message { In case of an objcclass, verify that all methods have a message
name set. We only check this now, because message names can be set name set. We only check this now, because message names can be set

View File

@ -2058,8 +2058,11 @@ uses
if oo_is_forward in tobjectdef(result).objectoptions then if oo_is_forward in tobjectdef(result).objectoptions then
add_forward_generic_def(result,context) add_forward_generic_def(result,context)
else if not (oo_inherits_not_specialized in tobjectdef(result).objectoptions) then
build_vmt(tobjectdef(result))
else else
build_vmt(tobjectdef(result)); { update the procdevs to add hidden self param }
insert_struct_hidden_paras(tobjectdef(result));
end; end;
{ handle params, calling convention, etc } { handle params, calling convention, etc }
procvardef: procvardef:

View File

@ -48,7 +48,7 @@ const
CurrentPPUVersion = 208; CurrentPPUVersion = 208;
{ for any other changes to the ppu format, increase this version number { for any other changes to the ppu format, increase this version number
(it's a cardinal) } (it's a cardinal) }
CurrentPPULongVersion = 26; CurrentPPULongVersion = 27;
{ unit flags } { unit flags }
uf_big_endian = $000004; uf_big_endian = $000004;

View File

@ -585,7 +585,8 @@ type
oo_has_new_destructor,{ the object/class declares a destructor (apart from potentially inherting one from the parent) } oo_has_new_destructor,{ the object/class declares a destructor (apart from potentially inherting one from the parent) }
oo_is_funcref, { interface has a single Invoke method that can be directly called } oo_is_funcref, { interface has a single Invoke method that can be directly called }
oo_is_invokable, { interface that is invokable like a function } oo_is_invokable, { interface that is invokable like a function }
oo_is_capturer { the class is the capturer for anonymous functions (or converted proc(var)s) } oo_is_capturer, { the class is the capturer for anonymous functions (or converted proc(var)s) }
oo_inherits_not_specialized { the class inherits from a not yet specialized type }
); );
tobjectoptions=set of tobjectoption; tobjectoptions=set of tobjectoption;
@ -895,7 +896,7 @@ const
{$ifndef jvm} {$ifndef jvm}
inherited_objectoptions : tobjectoptions = [oo_has_virtual,oo_has_private,oo_has_protected, inherited_objectoptions : tobjectoptions = [oo_has_virtual,oo_has_private,oo_has_protected,
oo_has_strictprotected,oo_has_strictprivate,oo_has_constructor,oo_has_destructor, oo_has_strictprotected,oo_has_strictprivate,oo_has_constructor,oo_has_destructor,
oo_can_have_published]; oo_can_have_published,oo_inherits_not_specialized];
{$else not jvm} {$else not jvm}
{ constructors are not inherited in Java } { constructors are not inherited in Java }
inherited_objectoptions : tobjectoptions = [oo_has_virtual,oo_has_private,oo_has_protected, inherited_objectoptions : tobjectoptions = [oo_has_virtual,oo_has_private,oo_has_protected,

View File

@ -8327,6 +8327,9 @@ implementation
exit; exit;
{ inherit options and status } { inherit options and status }
objectoptions:=objectoptions+(c.objectoptions*inherited_objectoptions); objectoptions:=objectoptions+(c.objectoptions*inherited_objectoptions);
{ check if parent is a generic parameter }
if sp_generic_para in c.typesym.symoptions then
objectoptions:=objectoptions+[oo_inherits_not_specialized];
{ initially has the same number of abstract methods as the parent } { initially has the same number of abstract methods as the parent }
abstractcnt:=c.abstractcnt; abstractcnt:=c.abstractcnt;
{ add the data of the anchestor class/object } { add the data of the anchestor class/object }

View File

@ -3302,7 +3302,8 @@ const
(mask:oo_has_new_destructor; str:'HasNewDestructor'), (mask:oo_has_new_destructor; str:'HasNewDestructor'),
(mask:oo_is_funcref; str:'IsFuncRef'), (mask:oo_is_funcref; str:'IsFuncRef'),
(mask:oo_is_invokable; str:'IsInvokable'), (mask:oo_is_invokable; str:'IsInvokable'),
(mask:oo_is_capturer; str:'IsCapturer') (mask:oo_is_capturer; str:'IsCapturer'),
(mask:oo_inherits_not_specialized; str:'InheritedNotSpecialized')
); );
var var
i : longint; i : longint;

15
tests/test/tgeneric112.pp Normal file
View File

@ -0,0 +1,15 @@
{$Mode ObjFPC}{$H+}
type
generic TTest<T:class> = class(T)
public
function Foo:Integer;override;
end;
function TTest.Foo:Integer;
begin
Result:=42;
end;
begin
end.

26
tests/test/tgeneric113.pp Normal file
View File

@ -0,0 +1,26 @@
{$Mode ObjFPC}{$H+}
type
TBase = class
public
function Foo:Integer;virtual;abstract;
end;
generic TTest<T:class> = class(T)
public
function Foo:Integer;override;
end;
function TTest.Foo:Integer;
begin
Result:=42;
end;
var
b: TBase;
begin
b:=specialize TTest<TBase>.Create;
if b.Foo<>42 then
Halt(1);
WriteLn('Ok');
end.

26
tests/test/tgeneric114.pp Normal file
View File

@ -0,0 +1,26 @@
{ %FAIL }
{$Mode ObjFPC}{$H+}
type
TBase = class
public
end;
generic TTest<T:class> = class(T)
public
function Foo:Integer;override;
end;
function TTest.Foo:Integer;
begin
Result:=42;
end;
var
b: TBase;
begin
b:=specialize TTest<TBase>.Create;
if b.Foo<>42 then
Halt(1);
WriteLn('Ok');
end.

30
tests/test/tgeneric115.pp Normal file
View File

@ -0,0 +1,30 @@
{$Mode ObjFPC}{$H+}
type
TBase = class
public
function Foo:Integer;virtual;abstract;
end;
generic TTest<T:class> = class(T)
public
function Foo:Integer;override;
end;
generic TTest2<T:class> = class(specialize TTest<T>)
public
end;
function TTest.Foo:Integer;
begin
Result:=42;
end;
var
b: TBase;
begin
b:=specialize TTest2<TBase>.Create;
if b.Foo<>42 then
Halt(1);
WriteLn('Ok');
end.

30
tests/test/tgeneric116.pp Normal file
View File

@ -0,0 +1,30 @@
{$Mode ObjFPC}{$H+}
type
TBase = class
public
function Foo:Integer;virtual;abstract;
end;
generic TTest<T:class> = class(T)
public
end;
generic TTest2<T:class> = class(specialize TTest<T>)
public
function Foo:Integer;override;
end;
function TTest2.Foo:Integer;
begin
Result:=42;
end;
var
b: TBase;
begin
b:=specialize TTest2<TBase>.Create;
if b.Foo<>42 then
Halt(1);
WriteLn('Ok');
end.