| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304 | {    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,defutil,      nutils,nbas,nflw,ncal,nld,ncnv,nmem,      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 has_copyback_paras(call: tcallnode): boolean;          var            n: tcallparanode;          begin            n:=tcallparanode(call.left);            result:=false;            while assigned(n) do              begin                if assigned(n.fparacopyback) then                  begin                    result:=true;                    exit;                  end;                n:=tcallparanode(n.right);              end;          end;        function is_optimizable_recursivecall(n : tnode) : boolean;          begin            result:=              (n.nodetype=calln) and              (tcallnode(n).procdefinition=p) and              (not assigned(tcallnode(n).methodpointer)) and              (not has_copyback_paras(tcallnode(n)));            if result then              usedcallnode:=tcallnode(n)            else              { obsolete type cast? }              result:=((n.nodetype=typeconvn) and (ttypeconvnode(n).convtype=tc_equal) and is_optimizable_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;          useaddr : boolean;        begin          { no tail call found and replaced so far }          result:=false;          if n=nil then            exit;          usedcallnode:=nil;          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;            calln,            assignn:              begin                if ((n.nodetype=calln) and is_optimizable_recursivecall(n) and is_void(n.resultdef)) or                   ((n.nodetype=assignn) and is_resultassignment(tbinarynode(n).left) and                   is_optimizable_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                        if assigned(paranode.fparainit) then                          begin                            addstatement(calcstatements,paranode.fparainit);                            paranode.fparainit:=nil;                          end;                        useaddr:=(paranode.parasym.varspez in [vs_var,vs_constref]) or                          ((paranode.parasym.varspez=vs_const) and                          paramanager.push_addr_param(paranode.parasym.varspez,paranode.parasym.vardef,p.proccalloption)) or                          ((paranode.parasym.varspez=vs_value) and                          is_open_array(paranode.parasym.vardef));                        if useaddr then                          begin                            tempnode:=ctempcreatenode.create(voidpointertype,voidpointertype.size,tt_persistent,true);                            addstatement(calcstatements,tempnode);                            addstatement(calcstatements,                              cassignmentnode.create(                                ctemprefnode.create(tempnode),                                caddrnode.create_internal(paranode.left)                                ));                          end                        else                          begin                            tempnode:=ctempcreatenode.create(paranode.left.resultdef,paranode.left.resultdef.size,tt_persistent,true);                            addstatement(calcstatements,tempnode);                            addstatement(calcstatements,                              cassignmentnode.create_internal(                                ctemprefnode.create(tempnode),                                paranode.left                                ));                          end;                        { "cast" away const varspezs }                        loadnode:=cloadnode.create(paranode.parasym,paranode.parasym.owner);                        include(tloadnode(loadnode).loadnodeflags,loadnf_isinternal_ignoreconst);                        { load the address of the symbol instead of symbol }                        if useaddr then                          include(tloadnode(loadnode).loadnodeflags,loadnf_load_addr);                        addstatement(copystatements,                          cassignmentnode.create_internal(                            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);            else              ;          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=vs_out) or               { parameters requiring tables are too complicated to handle                 and slow down things anyways so a tail recursion call                 makes no sense               }               is_managed_type(vardef) then              begin{$ifdef debug_opttail}        writeln('====================================================================================');        write('callnode: ',p.gettypename,' tail call opt disabled because parameter ',i,' ',realname,' ');        if is_managed_type(vardef) then          write('is managed ')        else if (varspez=vs_out) then          write('is out parameter');        writeln;        writeln('====================================================================================');{$endif debug_opttail}                exit;              end;{$ifdef fix_opttail}        { check if the local parameters should prevent tail recursion elimination }        for i:=0 to p.localst.count-1 do          with tabstractvarsym(p.localst[i]) do            if is_managed_type(vardef) then              begin{$ifdef debug_opttail}        writeln('====================================================================================');        writeln('callnode: ',p.gettypename,' tail call opt disabled because local ',i,' ',realname,' is managed');        writeln('====================================================================================');{$endif debug_opttail}                exit;              end;{$endif fix_opttail}        labelsym:=clabelsym.create('$opttail');        labelnode:=clabelnode.create(cnothingnode.create,labelsym);        if find_and_replace_tailcalls(n) then          begin{$ifdef debug_opttail}        writeln('====================================================================================');        writeln('Tail call optimization found:');        writeln('====================================================================================');        write('callnode: ',p.gettypename);        writeln('====================================================================================');{$endif debug_opttail}            oldnodes:=n;            n:=internalstatements(s);            addstatement(s,labelnode);            addstatement(s,oldnodes);{$ifdef debug_opttail}        writeln('====================================================================================');        write('Tail call replaced by: ');        printnode(n);        writeln('====================================================================================');        writeln;{$endif debug_opttail}          end        else          labelnode.free;      end;end.
 |