mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-04-06 02:47:55 +02:00
* Added support for pure functions
This commit is contained in:
parent
df73ef4f64
commit
1241242ed8
1213
compiler/ncal.pas
1213
compiler/ncal.pas
File diff suppressed because it is too large
Load Diff
@ -1451,7 +1451,12 @@ implementation
|
|||||||
function check_for_sideeffect(var n: tnode; arg: pointer): foreachnoderesult;
|
function check_for_sideeffect(var n: tnode; arg: pointer): foreachnoderesult;
|
||||||
begin
|
begin
|
||||||
result:=fen_false;
|
result:=fen_false;
|
||||||
if (n.nodetype in [assignn,calln,asmn,finalizetempsn]) or
|
if (
|
||||||
|
(n.nodetype = calln) and
|
||||||
|
{ Pure functions by definition do not have side-effects }
|
||||||
|
not (po_pure in tcallnode(n).procdefinition.procoptions)
|
||||||
|
) or
|
||||||
|
(n.nodetype in [assignn,asmn,finalizetempsn]) or
|
||||||
((n.nodetype=inlinen) and
|
((n.nodetype=inlinen) and
|
||||||
tinlinenode(n).may_have_sideeffect_norecurse
|
tinlinenode(n).may_have_sideeffect_norecurse
|
||||||
) or
|
) or
|
||||||
|
@ -2561,7 +2561,7 @@ type
|
|||||||
end;
|
end;
|
||||||
const
|
const
|
||||||
{Should contain the number of procedure directives we support.}
|
{Should contain the number of procedure directives we support.}
|
||||||
num_proc_directives=55;
|
num_proc_directives=56;
|
||||||
proc_direcdata:array[1..num_proc_directives] of proc_dir_rec=
|
proc_direcdata:array[1..num_proc_directives] of proc_dir_rec=
|
||||||
(
|
(
|
||||||
(
|
(
|
||||||
@ -2858,6 +2858,15 @@ const
|
|||||||
mutexclpocall : [pocall_internproc];
|
mutexclpocall : [pocall_internproc];
|
||||||
mutexclpotype : [];
|
mutexclpotype : [];
|
||||||
mutexclpo : [po_external,po_inline]
|
mutexclpo : [po_external,po_inline]
|
||||||
|
),(
|
||||||
|
idtok:_PURE;
|
||||||
|
pd_flags : [pd_interface,pd_implemen,pd_body,pd_notobject,pd_notobjintf,pd_notprocvar];
|
||||||
|
handler : nil;
|
||||||
|
pocall : pocall_none;
|
||||||
|
pooption : [po_pure];
|
||||||
|
mutexclpocall : [pocall_safecall];
|
||||||
|
mutexclpotype : [potype_constructor,potype_destructor,potype_class_constructor,potype_class_destructor];
|
||||||
|
mutexclpo : [po_external,po_interrupt,po_exports,po_overridingmethod,po_virtualmethod,po_iocheck]
|
||||||
),(
|
),(
|
||||||
idtok:_REGISTER;
|
idtok:_REGISTER;
|
||||||
pd_flags : [pd_interface,pd_implemen,pd_body,pd_procvar];
|
pd_flags : [pd_interface,pd_implemen,pd_body,pd_procvar];
|
||||||
|
@ -46,6 +46,7 @@ interface
|
|||||||
procedure swap_tempflags;
|
procedure swap_tempflags;
|
||||||
function store_node_tempflags(var n: tnode; arg: pointer): foreachnoderesult;
|
function store_node_tempflags(var n: tnode; arg: pointer): foreachnoderesult;
|
||||||
procedure CreateInlineInfo;
|
procedure CreateInlineInfo;
|
||||||
|
procedure CreatePurityInfo;
|
||||||
{ returns the node which is the start of the user code, this is needed by the dfa }
|
{ returns the node which is the start of the user code, this is needed by the dfa }
|
||||||
function GetUserCode: tnode;
|
function GetUserCode: tnode;
|
||||||
procedure maybe_add_constructor_wrapper(var tocode: tnode; withexceptblock: boolean);
|
procedure maybe_add_constructor_wrapper(var tocode: tnode; withexceptblock: boolean);
|
||||||
@ -271,6 +272,131 @@ implementation
|
|||||||
end;
|
end;
|
||||||
|
|
||||||
|
|
||||||
|
const
|
||||||
|
SNoPureReasons: array[0..4] of ansistring = (
|
||||||
|
'global variable access',
|
||||||
|
'writable constants',
|
||||||
|
'call to impure function',
|
||||||
|
'pointer dereference',
|
||||||
|
'use of address-of operator'
|
||||||
|
);
|
||||||
|
|
||||||
|
|
||||||
|
function preliminarypurenodescan(var n: tnode; arg: Pointer): foreachnoderesult;
|
||||||
|
begin
|
||||||
|
result:=fen_false;
|
||||||
|
case n.nodetype of
|
||||||
|
loadn:
|
||||||
|
if (TLoadNode(n).symtableentry.typ = staticvarsym) then
|
||||||
|
begin
|
||||||
|
(* if (
|
||||||
|
{ Take no chances }
|
||||||
|
(cs_typed_const_writable in current_settings.localswitches) and
|
||||||
|
(vo_is_typed_const in TAbstractVarSym(TLoadNode(n).symtableentry).varoptions)
|
||||||
|
) then
|
||||||
|
begin
|
||||||
|
result := fen_norecurse_true;
|
||||||
|
PInteger(arg)^ := 1;
|
||||||
|
end
|
||||||
|
else *)if (TAbstractVarSym(TLoadNode(n).symtableentry).varoptions * [vo_is_const(*, vo_is_typed_const*)]) = [] then
|
||||||
|
begin
|
||||||
|
result := fen_norecurse_true;
|
||||||
|
PInteger(arg)^ := 0;
|
||||||
|
end;
|
||||||
|
end;
|
||||||
|
calln:
|
||||||
|
if not(Assigned(TCallNode(n).procdefinition)) or
|
||||||
|
{ Note that the presence of these options don't guarantee success }
|
||||||
|
(([po_compilerproc, po_internconst, po_inline, po_pure] * TCallNode(n).procdefinition.procoptions) = []) then
|
||||||
|
begin
|
||||||
|
result := fen_norecurse_true;
|
||||||
|
PInteger(arg)^ := 2;
|
||||||
|
end;
|
||||||
|
derefn:
|
||||||
|
begin
|
||||||
|
{ Not allowed to dereference a pointer }
|
||||||
|
result := fen_norecurse_true;
|
||||||
|
PInteger(arg)^ := 3;
|
||||||
|
end;
|
||||||
|
addrn:
|
||||||
|
begin
|
||||||
|
{ Not allowed to take the address of a variable }
|
||||||
|
result := fen_norecurse_true;
|
||||||
|
PInteger(arg)^ := 4;
|
||||||
|
end;
|
||||||
|
else
|
||||||
|
;
|
||||||
|
end;
|
||||||
|
end;
|
||||||
|
|
||||||
|
|
||||||
|
function checknodepurity(procdef: tprocdef; coderoot: TNode): boolean;
|
||||||
|
|
||||||
|
procedure _no_pure(const reason: TMsgStr);
|
||||||
|
begin
|
||||||
|
Message2(parser_w_pure_ineligible,tprocdef(procdef).procsym.realname,reason);
|
||||||
|
end;
|
||||||
|
|
||||||
|
var
|
||||||
|
X: Integer;
|
||||||
|
MsgIdx: Integer;
|
||||||
|
begin
|
||||||
|
result := false;
|
||||||
|
{ this code will never be used (only specialisations can be pure),
|
||||||
|
and moreover contains references to defs that are not stored in the
|
||||||
|
ppu file }
|
||||||
|
if df_generic in current_procinfo.procdef.defoptions then
|
||||||
|
exit;
|
||||||
|
|
||||||
|
for X := 0 to procdef.paras.Count - 1 do
|
||||||
|
if not (TAbstractVarSym(procdef.paras[X]).varspez in [vs_value, vs_const, vs_out]) and
|
||||||
|
(
|
||||||
|
{ Make an exception for the function result }
|
||||||
|
(TAbstractVarSym(procdef.paras[X]).varspez <> vs_var) or
|
||||||
|
not (vo_is_funcret in TAbstractVarSym(procdef.paras[X]).varoptions)
|
||||||
|
) then
|
||||||
|
begin
|
||||||
|
_no_pure('parameter ' + (TAbstractVarSym(procdef.paras[X]).vardef).GetTypeName + ' is not passed by value');
|
||||||
|
exit;
|
||||||
|
end;
|
||||||
|
|
||||||
|
MsgIdx := 0;
|
||||||
|
if foreachnodestatic(pm_postprocess, coderoot, @preliminarypurenodescan, @MsgIdx) then
|
||||||
|
begin
|
||||||
|
_no_pure(SNoPureReasons[MsgIdx]);
|
||||||
|
exit;
|
||||||
|
end;
|
||||||
|
|
||||||
|
if pi_is_assembler in current_procinfo.flags then
|
||||||
|
begin
|
||||||
|
_no_pure('assembler');
|
||||||
|
exit;
|
||||||
|
end;
|
||||||
|
if pi_has_assembler_block in current_procinfo.flags then
|
||||||
|
begin
|
||||||
|
_no_pure('assembler');
|
||||||
|
exit;
|
||||||
|
end;
|
||||||
|
if (pi_has_global_goto in current_procinfo.flags) or
|
||||||
|
(pi_has_interproclabel in current_procinfo.flags) then
|
||||||
|
begin
|
||||||
|
_no_pure('global goto');
|
||||||
|
exit;
|
||||||
|
end;
|
||||||
|
if pi_has_inherited in current_procinfo.flags then
|
||||||
|
begin
|
||||||
|
_no_pure('inherited');
|
||||||
|
exit;
|
||||||
|
end;
|
||||||
|
if pi_uses_get_frame in current_procinfo.flags then
|
||||||
|
begin
|
||||||
|
_no_pure('get_frame');
|
||||||
|
exit;
|
||||||
|
end;
|
||||||
|
result:=true;
|
||||||
|
end;
|
||||||
|
|
||||||
|
|
||||||
{****************************************************************************
|
{****************************************************************************
|
||||||
PROCEDURE/FUNCTION BODY PARSING
|
PROCEDURE/FUNCTION BODY PARSING
|
||||||
****************************************************************************}
|
****************************************************************************}
|
||||||
@ -1574,6 +1700,8 @@ implementation
|
|||||||
PrintOption('noreturn');
|
PrintOption('noreturn');
|
||||||
if po_noinline in procdef.procoptions then
|
if po_noinline in procdef.procoptions then
|
||||||
PrintOption('noinline');
|
PrintOption('noinline');
|
||||||
|
if po_pure in procdef.procoptions then
|
||||||
|
PrintOption('pure');
|
||||||
end;
|
end;
|
||||||
|
|
||||||
if Assigned(Code) then
|
if Assigned(Code) then
|
||||||
@ -1738,17 +1866,31 @@ implementation
|
|||||||
|
|
||||||
procedure TCGProcinfo.CreateInlineInfo;
|
procedure TCGProcinfo.CreateInlineInfo;
|
||||||
begin
|
begin
|
||||||
new(procdef.inlininginfo);
|
new(procdef.inlininginfo);
|
||||||
procdef.inlininginfo^.code:=code.getcopy;
|
procdef.inlininginfo^.code:=code.getcopy;
|
||||||
procdef.inlininginfo^.flags:=flags;
|
procdef.inlininginfo^.flags:=flags;
|
||||||
{ The blocknode needs to set an exit label }
|
{ The blocknode needs to set an exit label }
|
||||||
if procdef.inlininginfo^.code.nodetype=blockn then
|
if procdef.inlininginfo^.code.nodetype=blockn then
|
||||||
include(procdef.inlininginfo^.code.flags,nf_block_with_exit);
|
include(procdef.inlininginfo^.code.flags,nf_block_with_exit);
|
||||||
procdef.has_inlininginfo:=true;
|
procdef.has_inlininginfo:=true;
|
||||||
export_local_ref_syms;
|
export_local_ref_syms;
|
||||||
export_local_ref_defs;
|
export_local_ref_defs;
|
||||||
end;
|
end;
|
||||||
|
|
||||||
|
|
||||||
|
procedure TCGProcinfo.CreatePurityInfo;
|
||||||
|
begin
|
||||||
|
new(procdef.purityinfo);
|
||||||
|
if Assigned(procdef.inlininginfo) then
|
||||||
|
{ Reuse the inline nodes to save space }
|
||||||
|
procdef.purityinfo^.code:=procdef.inlininginfo^.code
|
||||||
|
else
|
||||||
|
procdef.purityinfo^.code:=code.getcopy;
|
||||||
|
procdef.purityinfo^.flags:=flags;
|
||||||
|
procdef.has_purityinfo:=true;
|
||||||
|
end;
|
||||||
|
|
||||||
|
|
||||||
procedure searchthreadvar(p: TObject; arg: pointer);
|
procedure searchthreadvar(p: TObject; arg: pointer);
|
||||||
var
|
var
|
||||||
i : longint;
|
i : longint;
|
||||||
@ -2562,6 +2704,16 @@ implementation
|
|||||||
checknodeinlining(procdef) then
|
checknodeinlining(procdef) then
|
||||||
CreateInlineInfo;
|
CreateInlineInfo;
|
||||||
|
|
||||||
|
if (po_pure in procdef.procoptions) then
|
||||||
|
{ Can this routine be made pure? }
|
||||||
|
begin
|
||||||
|
if checknodepurity(procdef, code) then
|
||||||
|
CreatePurityInfo
|
||||||
|
else
|
||||||
|
{ Subroutine is not pure }
|
||||||
|
Exclude(procdef.procoptions, po_pure);
|
||||||
|
end;
|
||||||
|
|
||||||
{ Print the node to tree.log }
|
{ Print the node to tree.log }
|
||||||
if paraprintnodetree <> 0 then
|
if paraprintnodetree <> 0 then
|
||||||
printproc( 'after parsing');
|
printproc( 'after parsing');
|
||||||
|
@ -454,7 +454,9 @@ type
|
|||||||
and returns and it can be called.) }
|
and returns and it can be called.) }
|
||||||
po_wasm_funcref,
|
po_wasm_funcref,
|
||||||
{ WebAssembly suspending external }
|
{ WebAssembly suspending external }
|
||||||
po_wasm_suspending
|
po_wasm_suspending,
|
||||||
|
{ Is a pure function }
|
||||||
|
po_pure
|
||||||
);
|
);
|
||||||
tprocoptions=set of tprocoption;
|
tprocoptions=set of tprocoption;
|
||||||
|
|
||||||
@ -474,7 +476,9 @@ type
|
|||||||
{ compiled with fastmath enabled }
|
{ compiled with fastmath enabled }
|
||||||
pio_fastmath,
|
pio_fastmath,
|
||||||
{ inline is forbidden (calls get_frame) }
|
{ inline is forbidden (calls get_frame) }
|
||||||
pio_inline_forbidden
|
pio_inline_forbidden,
|
||||||
|
{ This is a pure function with a valid node tree }
|
||||||
|
pio_has_purityinfo
|
||||||
);
|
);
|
||||||
timplprocoptions = set of timplprocoption;
|
timplprocoptions = set of timplprocoption;
|
||||||
|
|
||||||
@ -1134,7 +1138,8 @@ inherited_objectoptions : tobjectoptions = [oo_has_virtual,oo_has_private,oo_has
|
|||||||
'objc-related-result-type', {po_objc_related_result_type}
|
'objc-related-result-type', {po_objc_related_result_type}
|
||||||
'po_anonymous', {po_anonymous}
|
'po_anonymous', {po_anonymous}
|
||||||
'"WASMFUNCREF"', {po_wasm_funcref}
|
'"WASMFUNCREF"', {po_wasm_funcref}
|
||||||
'"SUSPENDING"' {po_wasm_suspending}
|
'"SUSPENDING"', {po_wasm_suspending}
|
||||||
|
'"PURE"' {po_pure}
|
||||||
);
|
);
|
||||||
|
|
||||||
implementation
|
implementation
|
||||||
|
@ -21,6 +21,7 @@
|
|||||||
unit symdef;
|
unit symdef;
|
||||||
|
|
||||||
{$i fpcdefs.inc}
|
{$i fpcdefs.inc}
|
||||||
|
{$packenum 1}
|
||||||
|
|
||||||
interface
|
interface
|
||||||
|
|
||||||
@ -858,10 +859,12 @@ interface
|
|||||||
procedure Setinterfacedef(AValue: boolean);virtual;
|
procedure Setinterfacedef(AValue: boolean);virtual;
|
||||||
function Gethasforward: boolean;
|
function Gethasforward: boolean;
|
||||||
procedure Sethasforward(AValue: boolean);
|
procedure Sethasforward(AValue: boolean);
|
||||||
function GetIsEmpty: boolean;
|
function GetIsEmpty: boolean; {$IFDEF USEINLINE}inline;{$ENDIF USEINLINE}
|
||||||
procedure SetIsEmpty(AValue: boolean);
|
procedure SetIsEmpty(AValue: boolean); {$IFDEF USEINLINE}inline;{$ENDIF USEINLINE}
|
||||||
function GetHasInliningInfo: boolean;
|
function GetHasInliningInfo: boolean; {$IFDEF USEINLINE}inline;{$ENDIF USEINLINE}
|
||||||
procedure SetHasInliningInfo(AValue: boolean);
|
procedure SetHasInliningInfo(AValue: boolean); {$IFDEF USEINLINE}inline;{$ENDIF USEINLINE}
|
||||||
|
function GetHasPurityInfo: boolean; {$IFDEF USEINLINE}inline;{$ENDIF USEINLINE}
|
||||||
|
procedure SetHasPurityInfo(AValue: boolean); {$IFDEF USEINLINE}inline;{$ENDIF USEINLINE}
|
||||||
function Getis_implemented: boolean;
|
function Getis_implemented: boolean;
|
||||||
procedure Setis_implemented(AValue: boolean);
|
procedure Setis_implemented(AValue: boolean);
|
||||||
function getwas_anonymous:boolean;
|
function getwas_anonymous:boolean;
|
||||||
@ -901,7 +904,8 @@ interface
|
|||||||
import_name : pshortstring;
|
import_name : pshortstring;
|
||||||
{ info for inlining the subroutine, if this pointer is nil,
|
{ info for inlining the subroutine, if this pointer is nil,
|
||||||
the procedure can't be inlined }
|
the procedure can't be inlined }
|
||||||
inlininginfo : pinlininginfo;
|
inlininginfo,
|
||||||
|
purityinfo : pinlininginfo;
|
||||||
import_nr : word;
|
import_nr : word;
|
||||||
extnumber : word;
|
extnumber : word;
|
||||||
{ set to a value different from tsk_none in case this procdef is for
|
{ set to a value different from tsk_none in case this procdef is for
|
||||||
@ -1000,6 +1004,8 @@ interface
|
|||||||
property isempty: boolean read GetIsEmpty write SetIsEmpty;
|
property isempty: boolean read GetIsEmpty write SetIsEmpty;
|
||||||
{ true if all information required to inline this routine is available }
|
{ true if all information required to inline this routine is available }
|
||||||
property has_inlininginfo: boolean read GetHasInliningInfo write SetHasInliningInfo;
|
property has_inlininginfo: boolean read GetHasInliningInfo write SetHasInliningInfo;
|
||||||
|
{ true if all information required to calculate return values at compile time is available }
|
||||||
|
property has_purityinfo: boolean read GetHasPurityInfo write SetHasPurityInfo;
|
||||||
{ returns the $parentfp parameter for nested routines }
|
{ returns the $parentfp parameter for nested routines }
|
||||||
property parentfpsym: tsym read getparentfpsym;
|
property parentfpsym: tsym read getparentfpsym;
|
||||||
{ true if the implementation part for this procdef has been handled }
|
{ true if the implementation part for this procdef has been handled }
|
||||||
@ -5845,7 +5851,7 @@ implementation
|
|||||||
ppufile.getderef(returndefderef);
|
ppufile.getderef(returndefderef);
|
||||||
proctypeoption:=tproctypeoption(ppufile.getbyte);
|
proctypeoption:=tproctypeoption(ppufile.getbyte);
|
||||||
proccalloption:=tproccalloption(ppufile.getbyte);
|
proccalloption:=tproccalloption(ppufile.getbyte);
|
||||||
ppufile.getset(tppuset8(procoptions));
|
ppufile.getset(tppuset9(procoptions));
|
||||||
|
|
||||||
funcretloc[callerside].init;
|
funcretloc[callerside].init;
|
||||||
if po_explicitparaloc in procoptions then
|
if po_explicitparaloc in procoptions then
|
||||||
@ -5870,7 +5876,7 @@ implementation
|
|||||||
ppufile.do_interface_crc:=false;
|
ppufile.do_interface_crc:=false;
|
||||||
ppufile.putbyte(ord(proctypeoption));
|
ppufile.putbyte(ord(proctypeoption));
|
||||||
ppufile.putbyte(ord(proccalloption));
|
ppufile.putbyte(ord(proccalloption));
|
||||||
ppufile.putset(tppuset8(procoptions));
|
ppufile.putset(tppuset9(procoptions));
|
||||||
ppufile.do_interface_crc:=oldintfcrc;
|
ppufile.do_interface_crc:=oldintfcrc;
|
||||||
|
|
||||||
if (po_explicitparaloc in procoptions) then
|
if (po_explicitparaloc in procoptions) then
|
||||||
@ -6380,7 +6386,7 @@ implementation
|
|||||||
|
|
||||||
function tprocdef.store_localst: boolean;
|
function tprocdef.store_localst: boolean;
|
||||||
begin
|
begin
|
||||||
result:=has_inlininginfo or (df_generic in defoptions);
|
result:=has_inlininginfo or has_purityinfo or (df_generic in defoptions);
|
||||||
end;
|
end;
|
||||||
|
|
||||||
|
|
||||||
@ -6528,13 +6534,13 @@ implementation
|
|||||||
end;
|
end;
|
||||||
|
|
||||||
|
|
||||||
function tprocdef.GetIsEmpty: boolean;
|
function tprocdef.GetIsEmpty: boolean; {$IFDEF USEINLINE}inline;{$ENDIF USEINLINE}
|
||||||
begin
|
begin
|
||||||
result:=pio_empty in implprocoptions;
|
result:=pio_empty in implprocoptions;
|
||||||
end;
|
end;
|
||||||
|
|
||||||
|
|
||||||
procedure tprocdef.SetIsEmpty(AValue: boolean);
|
procedure tprocdef.SetIsEmpty(AValue: boolean); {$IFDEF USEINLINE}inline;{$ENDIF USEINLINE}
|
||||||
begin
|
begin
|
||||||
if AValue then
|
if AValue then
|
||||||
include(implprocoptions,pio_empty)
|
include(implprocoptions,pio_empty)
|
||||||
@ -6543,13 +6549,13 @@ implementation
|
|||||||
end;
|
end;
|
||||||
|
|
||||||
|
|
||||||
function tprocdef.GetHasInliningInfo: boolean;
|
function tprocdef.GetHasInliningInfo: boolean; {$IFDEF USEINLINE}inline;{$ENDIF USEINLINE}
|
||||||
begin
|
begin
|
||||||
result:=pio_has_inlininginfo in implprocoptions;
|
result:=pio_has_inlininginfo in implprocoptions;
|
||||||
end;
|
end;
|
||||||
|
|
||||||
|
|
||||||
procedure tprocdef.SetHasInliningInfo(AValue: boolean);
|
procedure tprocdef.SetHasInliningInfo(AValue: boolean); {$IFDEF USEINLINE}inline;{$ENDIF USEINLINE}
|
||||||
begin
|
begin
|
||||||
if AValue then
|
if AValue then
|
||||||
include(implprocoptions,pio_has_inlininginfo)
|
include(implprocoptions,pio_has_inlininginfo)
|
||||||
@ -6558,6 +6564,21 @@ implementation
|
|||||||
end;
|
end;
|
||||||
|
|
||||||
|
|
||||||
|
function tprocdef.GetHasPurityInfo: boolean; {$IFDEF USEINLINE}inline;{$ENDIF USEINLINE}
|
||||||
|
begin
|
||||||
|
result:=pio_has_purityinfo in implprocoptions;
|
||||||
|
end;
|
||||||
|
|
||||||
|
|
||||||
|
procedure tprocdef.SetHasPurityInfo(AValue: boolean); {$IFDEF USEINLINE}inline;{$ENDIF USEINLINE}
|
||||||
|
begin
|
||||||
|
if AValue then
|
||||||
|
include(implprocoptions,pio_has_purityinfo)
|
||||||
|
else
|
||||||
|
exclude(implprocoptions,pio_has_purityinfo);
|
||||||
|
end;
|
||||||
|
|
||||||
|
|
||||||
procedure tprocdef.Setinterfacedef(AValue: boolean);
|
procedure tprocdef.Setinterfacedef(AValue: boolean);
|
||||||
begin
|
begin
|
||||||
if not assigned(implprocdefinfo) then
|
if not assigned(implprocdefinfo) then
|
||||||
@ -6618,6 +6639,7 @@ implementation
|
|||||||
import_name:=nil;
|
import_name:=nil;
|
||||||
import_nr:=0;
|
import_nr:=0;
|
||||||
inlininginfo:=nil;
|
inlininginfo:=nil;
|
||||||
|
purityinfo:=nil;
|
||||||
deprecatedmsg:=nil;
|
deprecatedmsg:=nil;
|
||||||
genericdecltokenbuf:=nil;
|
genericdecltokenbuf:=nil;
|
||||||
if cs_opt_fastmath in current_settings.optimizerswitches then
|
if cs_opt_fastmath in current_settings.optimizerswitches then
|
||||||
@ -6684,6 +6706,18 @@ implementation
|
|||||||
funcretsym:=nil;
|
funcretsym:=nil;
|
||||||
end;
|
end;
|
||||||
|
|
||||||
|
if has_purityinfo then
|
||||||
|
begin
|
||||||
|
if not has_inlininginfo then
|
||||||
|
ppufile.getderef(funcretsymderef);
|
||||||
|
new(purityinfo);
|
||||||
|
ppufile.getset(tppuset4(purityinfo^.flags));
|
||||||
|
end
|
||||||
|
else
|
||||||
|
begin
|
||||||
|
purityinfo:=nil;
|
||||||
|
end;
|
||||||
|
|
||||||
aliasnames:=TCmdStrList.create;
|
aliasnames:=TCmdStrList.create;
|
||||||
{ count alias names }
|
{ count alias names }
|
||||||
aliasnamescount:=ppufile.getbyte;
|
aliasnamescount:=ppufile.getbyte;
|
||||||
@ -6722,14 +6756,23 @@ implementation
|
|||||||
localst:=nil;
|
localst:=nil;
|
||||||
{ inline stuff }
|
{ inline stuff }
|
||||||
if has_inlininginfo then
|
if has_inlininginfo then
|
||||||
inlininginfo^.code:=ppuloadnodetree(ppufile);
|
begin
|
||||||
|
inlininginfo^.code:=ppuloadnodetree(ppufile);
|
||||||
|
if has_purityinfo then
|
||||||
|
{ Reuse the inline tree to save space }
|
||||||
|
purityinfo^.code:=inlininginfo^.code;
|
||||||
|
end
|
||||||
|
else if has_purityinfo then
|
||||||
|
purityinfo^.code:=ppuloadnodetree(ppufile);
|
||||||
|
|
||||||
{ default values for no persistent data }
|
{ default values for no persistent data }
|
||||||
if (cs_link_deffile in current_settings.globalswitches) and
|
if (cs_link_deffile in current_settings.globalswitches) and
|
||||||
(tf_need_export in target_info.flags) and
|
(tf_need_export in target_info.flags) and
|
||||||
(po_exports in procoptions) then
|
(po_exports in procoptions) then
|
||||||
deffile.AddExport(mangledname);
|
deffile.AddExport(mangledname);
|
||||||
{ Disable po_has_inlining until the derefimpl is done }
|
{ Disable po_has_inlining and purity until the derefimpl is done }
|
||||||
has_inlininginfo:=false;
|
has_inlininginfo:=false;
|
||||||
|
has_purityinfo:=false;
|
||||||
end;
|
end;
|
||||||
|
|
||||||
|
|
||||||
@ -6848,6 +6891,13 @@ implementation
|
|||||||
ppufile.putset(tppuset4(inlininginfo^.flags));
|
ppufile.putset(tppuset4(inlininginfo^.flags));
|
||||||
end;
|
end;
|
||||||
|
|
||||||
|
if has_purityinfo then
|
||||||
|
begin
|
||||||
|
if not has_inlininginfo then
|
||||||
|
ppufile.putderef(funcretsymderef);
|
||||||
|
ppufile.putset(tppuset4(purityinfo^.flags));
|
||||||
|
end;
|
||||||
|
|
||||||
{ count alias names }
|
{ count alias names }
|
||||||
aliasnamescount:=0;
|
aliasnamescount:=0;
|
||||||
item:=TCmdStrListItem(aliasnames.first);
|
item:=TCmdStrListItem(aliasnames.first);
|
||||||
@ -6908,7 +6958,11 @@ implementation
|
|||||||
oldintfcrc:=ppufile.do_crc;
|
oldintfcrc:=ppufile.do_crc;
|
||||||
ppufile.do_crc:=false;
|
ppufile.do_crc:=false;
|
||||||
if has_inlininginfo then
|
if has_inlininginfo then
|
||||||
ppuwritenodetree(ppufile,inlininginfo^.code);
|
ppuwritenodetree(ppufile,inlininginfo^.code)
|
||||||
|
else if has_purityinfo then
|
||||||
|
{ Purity info and the inlining info share the same node tree to save
|
||||||
|
space }
|
||||||
|
ppuwritenodetree(ppufile,purityinfo^.code);
|
||||||
ppufile.do_crc:=oldintfcrc;
|
ppufile.do_crc:=oldintfcrc;
|
||||||
end;
|
end;
|
||||||
|
|
||||||
@ -7282,6 +7336,15 @@ implementation
|
|||||||
funcretsymderef.build(funcretsym);
|
funcretsymderef.build(funcretsym);
|
||||||
inlininginfo^.code.buildderefimpl;
|
inlininginfo^.code.buildderefimpl;
|
||||||
end;
|
end;
|
||||||
|
|
||||||
|
if has_purityinfo then
|
||||||
|
begin
|
||||||
|
if not has_inlininginfo then
|
||||||
|
funcretsymderef.build(funcretsym);
|
||||||
|
|
||||||
|
if not Assigned(inlininginfo) or (purityinfo^.code <> inlininginfo^.code) then
|
||||||
|
purityinfo^.code.buildderefimpl;
|
||||||
|
end;
|
||||||
end;
|
end;
|
||||||
|
|
||||||
|
|
||||||
@ -7303,6 +7366,10 @@ implementation
|
|||||||
if assigned(inlininginfo) then
|
if assigned(inlininginfo) then
|
||||||
has_inlininginfo:=true;
|
has_inlininginfo:=true;
|
||||||
|
|
||||||
|
{ Same with purity info }
|
||||||
|
if assigned(purityinfo) then
|
||||||
|
has_purityinfo:=true;
|
||||||
|
|
||||||
{ Locals }
|
{ Locals }
|
||||||
if store_localst and assigned(localst) then
|
if store_localst and assigned(localst) then
|
||||||
begin
|
begin
|
||||||
@ -7310,20 +7377,23 @@ implementation
|
|||||||
tlocalsymtable(localst).derefimpl(false);
|
tlocalsymtable(localst).derefimpl(false);
|
||||||
end;
|
end;
|
||||||
|
|
||||||
{ Inline }
|
{ Inline }
|
||||||
if has_inlininginfo then
|
if has_inlininginfo then
|
||||||
|
begin
|
||||||
|
inlininginfo^.code.derefimpl;
|
||||||
|
{ funcretsym, this is always located in the localst }
|
||||||
|
funcretsym:=tsym(funcretsymderef.resolve);
|
||||||
|
end;
|
||||||
|
|
||||||
|
{ Pure }
|
||||||
|
if has_purityinfo then
|
||||||
begin
|
begin
|
||||||
inlininginfo^.code.derefimpl;
|
if not has_inlininginfo or (purityinfo^.code <> inlininginfo^.code) then
|
||||||
{ funcretsym, this is always located in the localst }
|
purityinfo^.code.derefimpl;
|
||||||
funcretsym:=tsym(funcretsymderef.resolve);
|
|
||||||
end
|
if not has_inlininginfo then
|
||||||
else
|
{ funcretsym, this is always located in the localst }
|
||||||
begin
|
funcretsym:=tsym(funcretsymderef.resolve);
|
||||||
{ safety }
|
|
||||||
{ Not safe! A unit may be reresolved after its interface has been
|
|
||||||
parsed but before its implementation has been parsed, and in that
|
|
||||||
case the funcretsym is still required!
|
|
||||||
funcretsym:=nil; }
|
|
||||||
end;
|
end;
|
||||||
end;
|
end;
|
||||||
|
|
||||||
|
@ -144,6 +144,7 @@ type
|
|||||||
_LAST,
|
_LAST,
|
||||||
_NAME,
|
_NAME,
|
||||||
_NEAR,
|
_NEAR,
|
||||||
|
_PURE,
|
||||||
_READ,
|
_READ,
|
||||||
_SELF,
|
_SELF,
|
||||||
_SYSV,
|
_SYSV,
|
||||||
@ -491,6 +492,7 @@ const
|
|||||||
(str:'LAST' ;special:false;keyword:[m_none];op:NOTOKEN),
|
(str:'LAST' ;special:false;keyword:[m_none];op:NOTOKEN),
|
||||||
(str:'NAME' ;special:false;keyword:[m_none];op:NOTOKEN),
|
(str:'NAME' ;special:false;keyword:[m_none];op:NOTOKEN),
|
||||||
(str:'NEAR' ;special:false;keyword:[m_none];op:NOTOKEN),
|
(str:'NEAR' ;special:false;keyword:[m_none];op:NOTOKEN),
|
||||||
|
(str:'PURE' ;special:false;keyword:[m_none];op:NOTOKEN),
|
||||||
(str:'READ' ;special:false;keyword:[m_none];op:NOTOKEN),
|
(str:'READ' ;special:false;keyword:[m_none];op:NOTOKEN),
|
||||||
(str:'SELF' ;special:false;keyword:[m_none];op:NOTOKEN), {set inside methods only PM }
|
(str:'SELF' ;special:false;keyword:[m_none];op:NOTOKEN), {set inside methods only PM }
|
||||||
(str:'SYSV' ;special:false;keyword:[m_none];op:NOTOKEN), { Syscall variation on MorphOS }
|
(str:'SYSV' ;special:false;keyword:[m_none];op:NOTOKEN), { Syscall variation on MorphOS }
|
||||||
|
@ -47,7 +47,7 @@ uses
|
|||||||
|
|
||||||
const
|
const
|
||||||
Title = 'PPU-Analyser';
|
Title = 'PPU-Analyser';
|
||||||
Copyright = 'Copyright (c) 1998-2020 by the Free Pascal Development Team';
|
Copyright = 'Copyright (c) 1998-2023 by the Free Pascal Development Team';
|
||||||
|
|
||||||
{ verbosity }
|
{ verbosity }
|
||||||
v_none = $0;
|
v_none = $0;
|
||||||
@ -3095,7 +3095,8 @@ const
|
|||||||
(mask:po_objc_related_result_type; str: 'Objective-C related result type'),
|
(mask:po_objc_related_result_type; str: 'Objective-C related result type'),
|
||||||
(mask:po_anonymous; str: 'Anonymous'),
|
(mask:po_anonymous; str: 'Anonymous'),
|
||||||
(mask:po_wasm_funcref; str: 'WebAssembly funcref'),
|
(mask:po_wasm_funcref; str: 'WebAssembly funcref'),
|
||||||
(mask:po_wasm_suspending; str: 'WebAssembly suspending')
|
(mask:po_wasm_suspending; str: 'WebAssembly suspending'),
|
||||||
|
(mask:po_pure; str: 'Pure')
|
||||||
);
|
);
|
||||||
var
|
var
|
||||||
proctypeoption : tproctypeoption;
|
proctypeoption : tproctypeoption;
|
||||||
@ -3126,7 +3127,7 @@ begin
|
|||||||
writeln;
|
writeln;
|
||||||
proccalloption:=tproccalloption(ppufile.getbyte);
|
proccalloption:=tproccalloption(ppufile.getbyte);
|
||||||
writeln([space,' CallOption : ',proccalloptionStr[proccalloption]]);
|
writeln([space,' CallOption : ',proccalloptionStr[proccalloption]]);
|
||||||
ppufile.getset(tppuset8(procoptions));
|
ppufile.getset(tppuset9(procoptions));
|
||||||
if procoptions<>[] then
|
if procoptions<>[] then
|
||||||
begin
|
begin
|
||||||
if po_classmethod in procoptions then Include(ProcDef.Options, poClassMethod);
|
if po_classmethod in procoptions then Include(ProcDef.Options, poClassMethod);
|
||||||
@ -3411,7 +3412,8 @@ const
|
|||||||
(mask:pio_nested_access; str:'NestedAccess'),
|
(mask:pio_nested_access; str:'NestedAccess'),
|
||||||
(mask:pio_thunk; str:'Thunk'),
|
(mask:pio_thunk; str:'Thunk'),
|
||||||
(mask:pio_fastmath; str:'FastMath'),
|
(mask:pio_fastmath; str:'FastMath'),
|
||||||
(mask:pio_inline_forbidden; str:'InlineForbidden')
|
(mask:pio_inline_forbidden; str:'InlineForbidden'),
|
||||||
|
(mask:pio_has_purityinfo; str:'HasPurityInfo')
|
||||||
);
|
);
|
||||||
var
|
var
|
||||||
i: timplprocoption;
|
i: timplprocoption;
|
||||||
|
Loading…
Reference in New Issue
Block a user