| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288 | {    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, nadd;type  tsubnodetype = (    addsstringcharoptn,  { shorstring + char }    addsstringcsstringoptn   { shortstring + constant shortstring }  );  taddoptnode = class(taddnode)     subnodetype: tsubnodetype;     constructor create(ts: tsubnodetype; l,r : tnode); virtual;     { pass_1 will be overridden by the separate subclasses    }     { By default, pass_2 is the same as for addnode           }     { Only if there's a processor specific implementation, it }     { will be overridden.                                     }     function getcopy: 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 det_resulttype: tnode; override;    function pass_1: tnode; override;    function getcopy: 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;  end;  taddsstringcharoptnodeclass = class of taddsstringcharoptnode;  { add a constant string to a short string }  taddsstringcsstringoptnode = class(taddsstringoptnode)    constructor create(l,r : tnode); virtual;    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 is_addsstringoptnode(p: tnode): boolean;var   caddsstringcharoptnode: taddsstringcharoptnodeclass;   caddsstringcsstringoptnode: taddsstringcsstringoptnodeclass;implementationuses cutils, htypechk, defutil, defcmp, globtype, globals, cpubase, ncnv, ncon,ncal,     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_2 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.getcopy: tnode;var  hp: taddoptnode;begin  hp := taddoptnode(inherited getcopy);  hp.subnodetype := subnodetype;  getcopy := hp;end;function taddoptnode.docompare(p: tnode): boolean;begin  docompare :=    inherited docompare(p) and    (subnodetype = taddoptnode(p).subnodetype);end;{*****************************************************************************                        TADDSSTRINGOPTNODE*****************************************************************************}function taddsstringoptnode.det_resulttype: tnode;begin  result := nil;  updatecurmaxlen;  { left and right are already firstpass'ed by taddnode.pass_1 }  if not is_shortstring(left.resulttype.def) then   inserttypeconv(left,cshortstringtype);  if not is_shortstring(right.resulttype.def) then   inserttypeconv(right,cshortstringtype);  resulttype := left.resulttype;end;function taddsstringoptnode.pass_1: tnode;begin  pass_1 := nil;  expectloc:= LOC_REFERENCE;  calcregisters(self,0,0,0);  { here we call STRCONCAT or STRCMP or STRCOPY }  include(current_procinfo.flags,pi_do_call);end;function taddsstringoptnode.getcopy: tnode;var  hp: taddsstringoptnode;begin  hp := taddsstringoptnode(inherited getcopy);  hp.curmaxlen := curmaxlen;  getcopy := 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)          else            internalerror(291220001);        end      else curmaxlen := 255;    end  else if (left.nodetype = stringconstn) then    curmaxlen := min(tstringconstnode(left).len,255)  else if is_char(left.resulttype.def) 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.resulttype.def.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_optimize in aktglobalswitches) and{   the shortstring will be gotten through conversion if necessary (JM)    is_shortstring(p.left.resulttype.def) and }    ((p.nodetype = addn) and     is_char(p.right.resulttype.def));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_optimize in aktglobalswitches) and{   the shortstring will be gotten through conversion if necessary (JM)    is_shortstring(p.left.resulttype.def) 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;begin  caddsstringcharoptnode := taddsstringcharoptnode;  caddsstringcsstringoptnode := taddsstringcsstringoptnode;end.
 |