fpc/compiler/optconstprop.pas
J. Gareth "Curious Kit" Moreton ffe97bb7d9 * do_optconstpropagate and do_optdeadstoreelim now set their
"changed" parameter properly and is now an out type.
  * Optimisations on calls to said functions
2024-09-22 12:13:14 +00:00

424 lines
17 KiB
ObjectPascal

{
Constant propagation across statements
Copyright (c) 2005-2012 by Jeppe Johansen and Florian Klaempfl
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
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. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
****************************************************************************
}
unit optconstprop;
{$i fpcdefs.inc}
{ $define DEBUG_CONSTPROP}
interface
uses
node;
{ does constant propagation for rootnode
The approach is simple: It search for constant assignment statements. As soon as such
a statement is found, the following statements are search if they contain
a usage of the assigned variable. If this is a the case, the variable is
replaced by the constant. This does not work across points where the
program flow joins so e.g.
if ... then
...
a:=1;
...
else
...
a:=1;
...
writeln(a);
will not result in any constant propagation.
}
function do_optconstpropagate(var rootnode : tnode;out changed: boolean) : tnode;
implementation
uses
globtype, globals,
pass_1,procinfo,compinnr,
symsym, symconst,
nutils, nbas, ncnv, nld, nflw, ncal, ninl,
optbase, optutils;
function check_written(var n: tnode; arg: pointer): foreachnoderesult;
begin
result:=fen_false;
if n.isequal(tnode(arg)) and
((n.flags*[nf_write,nf_modify])<>[]) then
begin
result:=fen_norecurse_true;
end;
end;
{ propagates the constant assignment passed in arg into n, it returns true if
the search can continue with the next statement }
function replaceBasicAssign(var n: tnode; arg: tnode; var tree_modified: boolean): boolean;
var
st2, oldnode: tnode;
old: pnode;
changed, tree_modified2, tree_modified3: boolean;
written, tree_modified4, tree_modified5: Boolean;
begin
result:=true;
if n = nil then
exit;
tree_modified:=false;
tree_modified2:=false;
tree_modified3:=false;
tree_modified4:=false;
tree_modified5:=false;
{ while it might be usefull, to use foreach to iterate all nodes, it is safer to
iterate manually here so we have full controll how all nodes are processed }
{ We cannot analyze beyond those nodes, so we terminate to be on the safe side }
if (n.nodetype in [addrn,derefn,asmn,casen,whilerepeatn,labeln,continuen,breakn,
tryexceptn,raisen,tryfinallyn,onn,loadparentfpn,loadvmtaddrn,guidconstn,rttin,addoptn,asn,goton,
objcselectorn,objcprotocoln,
arrayconstructorn,arrayconstructorrangen]) then
exit(false)
else if n.nodetype=assignn then
begin
tree_modified:=false;
{ we can propage the constant in both branches because the evaluation order is not defined }
result:=replaceBasicAssign(tassignmentnode(n).right, arg, tree_modified);
{ do not use the intuitive way result:=result and replace... because this would prevent
replaceBasicAssign being called if the result is already false }
result:=replaceBasicAssign(tassignmentnode(n).left, arg, tree_modified2) and result;
tree_modified:=tree_modified or tree_modified2;
{ but we have to check if left writes to the currently searched variable ... }
written:=foreachnodestatic(pm_postprocess, tassignmentnode(n).left, @check_written, tassignmentnode(arg).left);
{ ... if this is the case, we have to stop searching }
result:=result and not(written);
end
else if n.isequal(tassignmentnode(arg).left) and ((n.flags*[nf_write,nf_modify])=[]) then
begin
n.Free;
n:=tassignmentnode(arg).right.getcopy;
inserttypeconv_internal(n, tassignmentnode(arg).left.resultdef);
tree_modified:=true;
end
else if n.nodetype=statementn then
result:=replaceBasicAssign(tstatementnode(n).left, arg, tree_modified)
else if n.nodetype=forn then
begin
result:=replaceBasicAssign(tfornode(n).right, arg, tree_modified);
if result then
begin
{ play safe and set the result which is check below }
result:=replaceBasicAssign(tfornode(n).t1, arg, tree_modified2);
tree_modified:=tree_modified or tree_modified2;
if result and (pi_dfaavailable in current_procinfo.flags) and
{ play safe }
assigned(tfornode(n).t2.optinfo) and assigned(tassignmentnode(arg).left.optinfo) then
begin
CalcDefSum(tfornode(n).t2);
{ the constant can propagete if is is not the counter variable ... }
if not(tassignmentnode(arg).left.isequal(actualtargetnode(@tfornode(n).left)^)) and
{ if it is a temprefn or its address is not taken in case of loadn }
((tassignmentnode(arg).left.nodetype=temprefn) or not(tabstractvarsym(tloadnode(tassignmentnode(arg).left).symtableentry).addr_taken)) and
{ and no definition in the loop? }
not(DFASetIn(tfornode(n).t2.optinfo^.defsum,tassignmentnode(arg).left.optinfo^.index)) then
begin
result:=replaceBasicAssign(tfornode(n).t2, arg, tree_modified3);
tree_modified:=tree_modified or tree_modified3;
end
else
result:=false;
end
else
result:=false;
end;
end
else if n.nodetype=blockn then
begin
changed:=false;
st2:=tstatementnode(tblocknode(n).statements);
old:=@tblocknode(n).statements;
while assigned(st2) do
begin
repeat
oldnode:=st2;
tree_modified2:=false;
if not replaceBasicAssign(st2, arg, tree_modified2) then
begin
old^:=st2;
oldnode:=nil;
changed:=changed or tree_modified2;
result:=false;
break;
end
else
old^:=st2;
changed:=changed or tree_modified2;
until oldnode=st2;
if oldnode = nil then
break;
old:=@tstatementnode(st2).next;
st2:=tstatementnode(st2).next;
end;
tree_modified:=changed;
end
else if n.nodetype=ifn then
begin
result:=replaceBasicAssign(tifnode(n).left, arg, tree_modified);
if result then
begin
if assigned(tifnode(n).t1) then
begin
{ we can propagate the constant in both branches of an if statement
because even if the the branch writes to it, the else branch gets the
unmodified value }
result:=replaceBasicAssign(tifnode(n).right, arg, tree_modified2);
{ do not use the intuitive way result:=result and replace... because this would prevent
replaceBasicAssign being called if the result is already false }
result:=replaceBasicAssign(tifnode(n).t1, arg, tree_modified3) and result;
tree_modified:=tree_modified or tree_modified2 or tree_modified3;
end
else
begin
result:=replaceBasicAssign(tifnode(n).right, arg, tree_modified2);
tree_modified:=tree_modified or tree_modified2;
end;
end;
end
else if n.nodetype=inlinen then
begin
{ constant inc'ed/dec'ed? }
if (tinlinenode(n).inlinenumber=in_dec_x) or (tinlinenode(n).inlinenumber=in_inc_x) then
begin
if tnode(tassignmentnode(arg).left).isequal(tcallparanode(tinlinenode(n).left).left) and
{ Internal Inc/Dec flags are created through a tree transformation from
a previous ConstProp pass. Setting it prevents an infinite loop where
Inc/Dec nodes are converted into an Add/Sub tree, and then converted
back to Inc/Dec by the forced firstpass run }
not (nf_internal in n.flags) and
(
not(assigned(tcallparanode(tinlinenode(n).left).right)) or
(tcallparanode(tcallparanode(tinlinenode(n).left).right).left.nodetype=ordconstn)
) then
begin
{ if the node just being searched is inc'ed/dec'ed then replace the inc/dec
by add/sub and force a second replacement pass }
oldnode:=n;
n:=tinlinenode(n).getaddsub_for_incdec;
Include(n.flags, nf_internal);
oldnode.free;
tree_modified:=true;
{ do not continue, value changed, if further const. propagations are possible, this is done
by the next pass }
result:=false;
exit;
end;
{ inc/dec might have a side effect, so stop here for now }
result:=false;
exit;
end
else if might_have_sideeffects(n) then
exit(false);
result:=replaceBasicAssign(tunarynode(n).left, arg, tree_modified);
end
else if n.nodetype=calln then
begin
{ only propagate simply variables which are regable, this means that their address is not
taken }
if (tassignmentnode(arg).left.nodetype=loadn) and
(tabstractvarsym(tloadnode(tassignmentnode(arg).left).symtableentry).varregable in [vr_fpureg,vr_mmreg,vr_intreg]) then
begin
result:=replaceBasicAssign(tnode(tcallnode(n).callinitblock), arg, tree_modified);
result:=result and replaceBasicAssign(tcallnode(n).left, arg, tree_modified2);
result:=result and replaceBasicAssign(tcallnode(n).vmt_entry, arg, tree_modified3);
result:=result and replaceBasicAssign(tcallnode(n).right, arg, tree_modified4);
result:=result and replaceBasicAssign(tnode(tcallnode(n).callcleanupblock), arg, tree_modified5);
tree_modified:=tree_modified or tree_modified2 or tree_modified3 or tree_modified4 or tree_modified5;
end
else
result:=false;
exit;
end
else if n.InheritsFrom(tbinarynode) then
begin
result:=replaceBasicAssign(tbinarynode(n).left, arg, tree_modified);
if result then
result:=replaceBasicAssign(tbinarynode(n).right, arg, tree_modified2);
tree_modified:=tree_modified or tree_modified2;
end
else if n.InheritsFrom(tunarynode) then
begin
result:=replaceBasicAssign(tunarynode(n).left, arg, tree_modified);
end;
if n.nodetype<>callparan then
begin
if tree_modified then
exclude(n.transientflags,tnf_pass1_done);
do_firstpass(n);
end;
end;
function propagate(var n: tnode; arg: pointer): foreachnoderesult;
var
l,
st, st2, oldnode: tnode;
old: pnode;
a: tassignmentnode;
tree_mod, changed: boolean;
begin
result:=fen_true;
changed:=false;
PBoolean(arg)^:=false;
if not assigned(n) then
exit;
if n.nodetype in [calln] then
exit(fen_norecurse_true);
if n.nodetype=blockn then
begin
st:=tblocknode(n).statements;
while assigned(st) and
(st.nodetype=statementn) and
assigned(tstatementnode(st).statement) do
begin
if tstatementnode(st).statement.nodetype=assignn then
begin
a:=tassignmentnode(tstatementnode(st).statement);
l:=a.left;
if ((((l.nodetype=loadn) and
{ its address cannot have escaped the current routine }
not(tabstractvarsym(tloadnode(l).symtableentry).addr_taken)) and
((
(tloadnode(l).symtableentry.typ=localvarsym) and
(tloadnode(l).symtable=current_procinfo.procdef.localst)
) or
((tloadnode(l).symtableentry.typ=paravarsym) and
(tloadnode(l).symtable=current_procinfo.procdef.parast)
) or
((tloadnode(l).symtableentry.typ=staticvarsym) and
(tloadnode(l).symtable.symtabletype=staticsymtable)
)
)) or
(l.nodetype = temprefn)) and
(is_constintnode(a.right) or
is_constboolnode(a.right) or
is_constcharnode(a.right) or
is_constwidecharnode(a.right) or
is_constwidestringnode(a.right) or
is_constenumnode(a.right) or
is_conststringnode(a.right) or
is_constpointernode(a.right)) then
begin
{$ifdef DEBUG_CONSTPROP}
writeln('******************************* propagating ***********************************');
printnode(a);
writeln('*******************************************************************************');
{$endif DEBUG_CONSTPROP}
st2:=tstatementnode(tstatementnode(st).right);
old:=@tstatementnode(st).right;
while assigned(st2) do
begin
repeat
oldnode:=st2;
{ Simple assignment of constant found }
tree_mod:=false;
if not replaceBasicAssign(st2, a, tree_mod) then
begin
old^:=st2;
oldnode:=nil;
changed:=changed or tree_mod;
break;
end
else
old^:=st2;
changed:=changed or tree_mod;
until oldnode=st2;
if oldnode = nil then
break;
old:=@tstatementnode(st2).next;
st2:=tstatementnode(st2).next;
end;
end;
end;
st:=tstatementnode(st).next;
end;
end;
PBoolean(arg)^:=changed;
end;
function do_optconstpropagate(var rootnode: tnode;out changed: boolean): tnode;
var
iteration_changed: Boolean;
begin
changed:=false;
repeat
iteration_changed:=false;
{$ifdef DEBUG_CONSTPROP}
writeln('************************ before constant propagation ***************************');
printnode(rootnode);
{$endif DEBUG_CONSTPROP}
foreachnodestatic(pm_postandagain, rootnode, @propagate, @iteration_changed);
changed:=changed or iteration_changed;
if iteration_changed then
doinlinesimplify(rootnode);
{$ifdef DEBUG_CONSTPROP}
writeln('************************ after constant propagation ***************************');
printnode(rootnode);
writeln('*******************************************************************************');
{$endif DEBUG_CONSTPROP}
until not(cs_opt_level3 in current_settings.optimizerswitches) or not(iteration_changed);
result:=rootnode;
end;
end.