| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504 | {    Copyright (c) 1998-2002 by Jonas Maebe    This unit implements optimized nodes    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 nopt;{$i fpcdefs.inc}interfaceuses node,nbas,nadd,constexp;type  tsubnodetype = (    addsstringcharoptn,  { shorstring + char }    addsstringcsstringoptn   { shortstring + constant shortstring }  );  taddoptnode = class(taddnode)     subnodetype: tsubnodetype;     constructor create(ts: tsubnodetype; l,r : tnode); virtual; reintroduce;     { pass_1 will be overridden by the separate subclasses    }     { By default, pass_generate_code is the same as for addnode           }     { Only if there's a processor specific implementation, it }     { will be overridden.                                     }     function dogetcopy: tnode; override;     function docompare(p: tnode): boolean; override;  end;  taddsstringoptnode = class(taddoptnode)    { maximum length of the string until now, allows us to skip a compare }    { sometimes (it's initialized/updated by calling updatecurmaxlen)     }    curmaxlen: byte;    { pass_1 must be overridden, otherwise we get an endless loop }    function pass_typecheck: tnode; override;    function pass_1: tnode; override;    function dogetcopy: tnode; override;    function docompare(p: tnode): boolean; override;   protected    procedure updatecurmaxlen;  end;  { add a char to a shortstring }  taddsstringcharoptnode = class(taddsstringoptnode)    constructor create(l,r : tnode); virtual; reintroduce;  end;  taddsstringcharoptnodeclass = class of taddsstringcharoptnode;  { add a constant string to a short string }  taddsstringcsstringoptnode = class(taddsstringoptnode)    constructor create(l,r : tnode); virtual; reintroduce;    function pass_1: tnode; override;  end;  taddsstringcsstringoptnodeclass = class of taddsstringcsstringoptnode;function canbeaddsstringcharoptnode(p: taddnode): boolean;function genaddsstringcharoptnode(p: taddnode): tnode;function canbeaddsstringcsstringoptnode(p: taddnode): boolean;function genaddsstringcsstringoptnode(p: taddnode): tnode;function canbemultistringadd(p: taddnode): boolean;function genmultistringadd(p: taddnode): tnode;function canbemultidynarrayadd(p: taddnode): boolean;function genmultidynarrayadd(p: taddnode): tnode;function is_addsstringoptnode(p: tnode): boolean;var   caddsstringcharoptnode: taddsstringcharoptnodeclass;   caddsstringcsstringoptnode: taddsstringcsstringoptnodeclass;implementationuses cutils, systems,     htypechk, defutil, defcmp, globtype, globals, cpubase, compinnr,     ncnv, ncon, ncal, ninl, nld, nmem,     verbose, symconst,symdef, cgbase, procinfo;{*****************************************************************************                             TADDOPTNODE*****************************************************************************}constructor taddoptnode.create(ts: tsubnodetype; l,r : tnode);begin  { we need to keep the addn nodetype, otherwise taddnode.pass_generate_code will be }  { confused. Comparison for equal nodetypes therefore has to be         }  { implemented using the classtype() method (JM)                        }  inherited create(addn,l,r);  subnodetype := ts;end;function taddoptnode.dogetcopy: tnode;var  hp: taddoptnode;begin  hp := taddoptnode(inherited dogetcopy);  hp.subnodetype := subnodetype;  dogetcopy := hp;end;function taddoptnode.docompare(p: tnode): boolean;begin  docompare :=    inherited docompare(p) and    (subnodetype = taddoptnode(p).subnodetype);end;{*****************************************************************************                        TADDSSTRINGOPTNODE*****************************************************************************}function taddsstringoptnode.pass_typecheck: tnode;begin  result := nil;  updatecurmaxlen;  { left and right are already firstpass'ed by taddnode.pass_1 }  if not is_shortstring(left.resultdef) then   inserttypeconv(left,cshortstringtype);  if not is_shortstring(right.resultdef) then   inserttypeconv(right,cshortstringtype);  resultdef := left.resultdef;end;function taddsstringoptnode.pass_1: tnode;begin  pass_1 := nil;  expectloc:= LOC_REFERENCE;  { here we call STRCONCAT or STRCMP or STRCOPY }  include(current_procinfo.flags,pi_do_call);end;function taddsstringoptnode.dogetcopy: tnode;var  hp: taddsstringoptnode;begin  hp := taddsstringoptnode(inherited dogetcopy);  hp.curmaxlen := curmaxlen;  dogetcopy := hp;end;function taddsstringoptnode.docompare(p: tnode): boolean;begin  docompare :=    inherited docompare(p) and    (curmaxlen = taddsstringcharoptnode(p).curmaxlen);end;function is_addsstringoptnode(p: tnode): boolean;begin  is_addsstringoptnode :=    p.inheritsfrom(taddsstringoptnode);end;procedure taddsstringoptnode.updatecurmaxlen;begin  if is_addsstringoptnode(left) then    begin      { made it a separate block so no other if's are processed (would be a }      { simple waste of time) (JM)                                          }      if (taddsstringoptnode(left).curmaxlen < 255) then        case subnodetype of          addsstringcharoptn:            curmaxlen := succ(taddsstringoptnode(left).curmaxlen);          addsstringcsstringoptn:            curmaxlen := min(taddsstringoptnode(left).curmaxlen +                              tstringconstnode(right).len,255)        end      else        curmaxlen := 255;    end  else if (left.nodetype = stringconstn) then    curmaxlen := min(tstringconstnode(left).len,255)  else if is_char(left.resultdef) then    curmaxlen := 1  else if (left.nodetype = typeconvn) then    begin      case ttypeconvnode(left).convtype of        tc_char_2_string:          curmaxlen := 1;{       doesn't work yet, don't know why (JM)        tc_chararray_2_string:          curmaxlen :=            min(ttypeconvnode(left).left.resultdef.size,255); }        else curmaxlen := 255;      end;    end  else    curmaxlen := 255;end;{*****************************************************************************                        TADDSSTRINGCHAROPTNODE*****************************************************************************}constructor taddsstringcharoptnode.create(l,r : tnode);begin  inherited create(addsstringcharoptn,l,r);end;{*****************************************************************************                        TADDSSTRINGCSSTRINGOPTNODE*****************************************************************************}constructor taddsstringcsstringoptnode.create(l,r : tnode);begin  inherited create(addsstringcsstringoptn,l,r);end;function taddsstringcsstringoptnode.pass_1: tnode;begin  { create the call to the concat routine both strings as arguments }  result := ccallnode.createintern('fpc_shortstr_append_shortstr',    ccallparanode.create(left,ccallparanode.create(right,nil)));  left:=nil;  right:=nil;end;{*****************************************************************************                                HELPERS*****************************************************************************}function canbeaddsstringcharoptnode(p: taddnode): boolean;begin  canbeaddsstringcharoptnode :=    (cs_opt_level1 in current_settings.optimizerswitches) and{   the shortstring will be gotten through conversion if necessary (JM)    is_shortstring(p.left.resultdef) and }    ((p.nodetype = addn) and     is_char(p.right.resultdef));end;function genaddsstringcharoptnode(p: taddnode): tnode;var  hp: tnode;begin  hp := caddsstringcharoptnode.create(p.left.getcopy,p.right.getcopy);  hp.flags := p.flags;  genaddsstringcharoptnode := hp;end;function canbeaddsstringcsstringoptnode(p: taddnode): boolean;begin  canbeaddsstringcsstringoptnode :=    (cs_opt_level1 in current_settings.optimizerswitches) and{   the shortstring will be gotten through conversion if necessary (JM)    is_shortstring(p.left.resultdef) and }    ((p.nodetype = addn) and     (p.right.nodetype = stringconstn));end;function genaddsstringcsstringoptnode(p: taddnode): tnode;var  hp: tnode;begin  hp := caddsstringcsstringoptnode.create(p.left.getcopy,p.right.getcopy);  hp.flags := p.flags;  genaddsstringcsstringoptnode := hp;end;function canbemultistringadd(p: taddnode): boolean;var  hp : tnode;  i  : longint;begin  result:=false;  if p.resultdef.typ<>stringdef then    exit;  i:=0;  hp:=p;  while assigned(hp) and (hp.nodetype=addn) do    begin      inc(i);      hp:=taddnode(hp).left;    end;  result:=(i>1);end;function genmultistringadd(p: taddnode): tnode;var  hp,sn : tnode;  arrp  : tarrayconstructornode;  newstatement : tstatementnode;  tempnode    : ttempcreatenode;  is_shortstr : boolean;  para : tcallparanode;begin  arrp:=nil;  hp:=p;  is_shortstr:=is_shortstring(p.resultdef);  while assigned(hp) and (hp.nodetype=addn) do    begin      sn:=taddnode(hp).right.getcopy;      inserttypeconv(sn,p.resultdef);      if is_shortstr then        begin          sn:=caddrnode.create(sn);          include(taddrnode(sn).addrnodeflags,anf_typedaddr);          include(sn.flags,nf_internal);        end;      arrp:=carrayconstructornode.create(sn,arrp);      hp:=taddnode(hp).left;    end;  sn:=hp.getcopy;  inserttypeconv(sn,p.resultdef);  if is_shortstr then    begin      sn:=caddrnode.create(sn);      include(sn.flags,nf_internal);    end;  arrp:=carrayconstructornode.create(sn,arrp);  arrp.allow_array_constructor:=true;  if assigned(aktassignmentnode) and     (aktassignmentnode.right=p) and     (aktassignmentnode.left.resultdef=p.resultdef) and     valid_for_var(aktassignmentnode.left,false) then    begin      para:=ccallparanode.create(              arrp,              ccallparanode.create(aktassignmentnode.left.getcopy,nil)            );      if is_ansistring(p.resultdef) then        para:=ccallparanode.create(                cordconstnode.create(                  { don't use getparaencoding(), we have to know                    when the result is rawbytestring }                  tstringdef(p.resultdef).encoding,                  u16inttype,                  true                ),                para              );      result:=ccallnode.createintern(                'fpc_'+tstringdef(p.resultdef).stringtypname+'_concat_multi',                para              );      include(aktassignmentnode.flags,nf_assign_done_in_right);    end  else    begin      result:=internalstatements(newstatement);      tempnode:=ctempcreatenode.create(p.resultdef,p.resultdef.size,tt_persistent ,true);      addstatement(newstatement,tempnode);      { initialize the temp, since it will be passed to a        var-parameter (and finalization, which is performed by the        ttempcreate node and which takes care of the initialization        on native targets, is a noop on managed VM targets) }      if (target_info.system in systems_managed_vm) and         is_managed_type(p.resultdef) then        addstatement(newstatement,cinlinenode.create(in_setlength_x,          false,          ccallparanode.create(genintconstnode(0),            ccallparanode.create(ctemprefnode.create(tempnode),nil))));      para:=ccallparanode.create(              arrp,              ccallparanode.create(ctemprefnode.create(tempnode),nil)            );      if is_ansistring(p.resultdef) then        para:=ccallparanode.create(                cordconstnode.create(                  { don't use getparaencoding(), we have to know                    when the result is rawbytestring }                  tstringdef(p.resultdef).encoding,                  u16inttype,                  true                ),                para              );      addstatement(        newstatement,        ccallnode.createintern(          'fpc_'+tstringdef(p.resultdef).stringtypname+'_concat_multi',          para        )      );      addstatement(newstatement,ctempdeletenode.create_normal_temp(tempnode));      addstatement(newstatement,ctemprefnode.create(tempnode));    end;end;function canbemultidynarrayadd(p: taddnode): boolean;var  hp : tnode;  i  : longint;begin  result:=false;  if not(is_dynamic_array(p.resultdef)) then    exit;  i:=0;  hp:=p;  while assigned(hp) and (hp.nodetype=addn) do    begin      inc(i);      hp:=taddnode(hp).left;    end;  result:=(i>1);end;function genmultidynarrayadd(p: taddnode): tnode;var  hp,sn : tnode;  arrp  : tarrayconstructornode;  newstatement : tstatementnode;  tempnode    : ttempcreatenode;  para : tcallparanode;begin  arrp:=nil;  hp:=p;  while assigned(hp) and (hp.nodetype=addn) do    begin      sn:=ctypeconvnode.create_internal(taddnode(hp).right.getcopy,voidpointertype);      arrp:=carrayconstructornode.create(sn,arrp);      hp:=taddnode(hp).left;    end;  sn:=ctypeconvnode.create_internal(hp.getcopy,voidpointertype);  arrp:=carrayconstructornode.create(sn,arrp);  arrp.allow_array_constructor:=true;  if assigned(aktassignmentnode) and     (aktassignmentnode.right=p) and     (aktassignmentnode.left.resultdef=p.resultdef) and     valid_for_var(aktassignmentnode.left,false) then    begin      para:=ccallparanode.create(              arrp,            ccallparanode.create(              caddrnode.create_internal(crttinode.create(tstoreddef(p.resultdef),initrtti,rdt_normal)),            ccallparanode.create(              ctypeconvnode.create_internal(aktassignmentnode.left.getcopy,voidpointertype),nil)          ));      result:=ccallnode.createintern(                'fpc_dynarray_concat_multi',                para              );      include(aktassignmentnode.flags,nf_assign_done_in_right);    end  else    begin      result:=internalstatements(newstatement);      tempnode:=ctempcreatenode.create(p.resultdef,p.resultdef.size,tt_persistent ,true);      addstatement(newstatement,tempnode);      { initialize the temp, since it will be passed to a        var-parameter (and finalization, which is performed by the        ttempcreate node and which takes care of the initialization        on native targets, is a noop on managed VM targets) }      if (target_info.system in systems_managed_vm) and         is_managed_type(p.resultdef) then        addstatement(newstatement,cinlinenode.create(in_setlength_x,          false,          ccallparanode.create(genintconstnode(0),            ccallparanode.create(ctemprefnode.create(tempnode),nil))));      para:=ccallparanode.create(              arrp,            ccallparanode.create(              caddrnode.create_internal(crttinode.create(tstoreddef(p.resultdef),initrtti,rdt_normal)),            ccallparanode.create(              ctypeconvnode.create_internal(ctemprefnode.create(tempnode),voidpointertype),nil)          ));      addstatement(        newstatement,        ccallnode.createintern(          'fpc_dynarray_concat_multi',          para        )      );      addstatement(newstatement,ctempdeletenode.create_normal_temp(tempnode));      addstatement(newstatement,ctemprefnode.create(tempnode));    end;end;begin  caddsstringcharoptnode := taddsstringcharoptnode;  caddsstringcsstringoptnode := taddsstringcsstringoptnode;end.
 |