| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212 | {    Tail recursion optimization    Copyright (c) 2006 by 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 opttail;{$i fpcdefs.inc}  interface    uses      symdef,node;    procedure do_opttail(var n : tnode;p : tprocdef);  implementation    uses      globtype,      symconst,symsym,      defcmp,      nutils,nbas,nflw,ncal,nld,ncnv,      pass_1,      paramgr;    procedure do_opttail(var n : tnode;p : tprocdef);      var        labelnode : tlabelnode;      function find_and_replace_tailcalls(var n : tnode) : boolean;        var          usedcallnode : tcallnode;        function is_recursivecall(n : tnode) : boolean;          begin            result:=(n.nodetype=calln) and (tcallnode(n).procdefinition=p);            if result then              usedcallnode:=tcallnode(n)            else              { obsolete type cast? }              result:=((n.nodetype=typeconvn) and (ttypeconvnode(n).convtype=tc_equal) and is_recursivecall(ttypeconvnode(n).left));          end;        function is_resultassignment(n : tnode) : boolean;          begin            result:=((n.nodetype=loadn) and (tloadnode(n).symtableentry=p.funcretsym)) or              ((n.nodetype=typeconvn) and (ttypeconvnode(n).convtype=tc_equal) and is_resultassignment(ttypeconvnode(n).left));          end;        var          calcnodes,          copynodes,          hp : tnode;          nodes,          calcstatements,          copystatements : tstatementnode;          paranode : tcallparanode;          tempnode : ttempcreatenode;          loadnode : tloadnode;          oldnodetree : tnode;        begin          { no tail call found and replaced so far }          result:=false;          if n=nil then            exit;          case n.nodetype of            statementn:              begin                hp:=n;                { search last node }                while assigned(tstatementnode(hp).right) do                  hp:=tstatementnode(hp).right;                result:=find_and_replace_tailcalls(tstatementnode(hp).left);              end;            ifn:              begin                result:=find_and_replace_tailcalls(tifnode(n).right);                { avoid short bool eval here }                result:=find_and_replace_tailcalls(tifnode(n).t1) or result;              end;            assignn:              begin                if is_resultassignment(tbinarynode(n).left) and                   is_recursivecall(tbinarynode(n).right) then                  begin                    { found one! }                    {                    writeln('tail recursion optimization for ',p.mangledname);                    printnode(output,n);                    }                    { create assignments for all parameters }                    { this is hairy to do because one parameter could be used to calculate another one, so                      assign them first to temps and then add them }                    calcnodes:=internalstatements(calcstatements);                    copynodes:=internalstatements(copystatements);                    paranode:=tcallparanode(usedcallnode.left);                    while assigned(paranode) do                      begin                        tempnode:=ctempcreatenode.create(paranode.left.resultdef,paranode.left.resultdef.size,tt_persistent,true);                        addstatement(calcstatements,tempnode);                        addstatement(calcstatements,                          cassignmentnode.create(                            ctemprefnode.create(tempnode),                            paranode.left                            ));                        { "cast" away const varspezs }                        loadnode:=cloadnode.create(paranode.parasym,paranode.parasym.owner);                        include(loadnode.flags,nf_isinternal_ignoreconst);                        addstatement(copystatements,                          cassignmentnode.create(                            loadnode,                            ctemprefnode.create(tempnode)                            ));                        addstatement(copystatements,ctempdeletenode.create_normal_temp(tempnode));                        { reused }                        paranode.left:=nil;                        paranode:=tcallparanode(paranode.right);                      end;                    oldnodetree:=n;                    n:=internalstatements(nodes);                    if assigned(usedcallnode.callinitblock) then                      begin                        addstatement(nodes,usedcallnode.callinitblock);                        usedcallnode.callinitblock:=nil;                      end;                    addstatement(nodes,calcnodes);                    addstatement(nodes,copynodes);                    { create goto }                    addstatement(nodes,cgotonode.create(labelnode.labsym));                    if assigned(usedcallnode.callcleanupblock) then                      begin                        { callcleanupblock should contain only temp. node clean up }                        checktreenodetypes(usedcallnode.callcleanupblock,                          [tempdeleten,blockn,statementn,temprefn,nothingn]);                        addstatement(nodes,usedcallnode.callcleanupblock);                        usedcallnode.callcleanupblock:=nil;                      end;                    oldnodetree.free;                    do_firstpass(n);                    result:=true;                  end;              end;            blockn:              result:=find_and_replace_tailcalls(tblocknode(n).left);          end;        end;      var        s : tstatementnode;        oldnodes : tnode;        i : longint;        labelsym : tlabelsym;      begin        { check if the parameters actually would support tail recursion elimination }        for i:=0 to p.paras.count-1 do          with tparavarsym(p.paras[i]) do            if (varspez in [vs_out,vs_var]) or              ((varspez=vs_const) and               (paramanager.push_addr_param(varspez,vardef,p.proccalloption)) or               { parameters requiring tables are too complicated to handle                 and slow down things anyways so a tail recursion call                 makes no sense               }               vardef.needs_inittable) then               exit;        labelsym:=tlabelsym.create('$opttail');        labelnode:=clabelnode.create(cnothingnode.create,labelsym);        if find_and_replace_tailcalls(n) then          begin            oldnodes:=n;            n:=internalstatements(s);            addstatement(s,labelnode);            addstatement(s,oldnodes);          end        else          labelnode.free;      end;end.
 |