| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827 | {    Copyright (c) 2000-2002 by Florian Klaempfl    Type checking and register allocation for set/case 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 nset;{$i fpcdefs.inc}interface    uses       cclasses,       node,globtype,globals,       aasmbase,aasmtai,aasmdata,symtype;    type       pcaselabel = ^tcaselabel;       tcaselabel = record          { range }          _low,          _high   : TConstExprInt;          { unique blockid }          blockid : longint;          { left and right tree node }          less,          greater : pcaselabel;       end;       pcaseblock = ^tcaseblock;       tcaseblock = record          { label (only used in pass_generate_code) }          blocklabel : tasmlabel;          { instructions }          statement  : tnode;       end;       tsetelementnode = class(tbinarynode)          constructor create(l,r : tnode);virtual;          function pass_typecheck:tnode;override;          function pass_1 : tnode;override;       end;       tsetelementnodeclass = class of tsetelementnode;       tinnode = class(tbinopnode)          constructor create(l,r : tnode);virtual;reintroduce;          function pass_typecheck:tnode;override;          function pass_1 : tnode;override;       end;       tinnodeclass = class of tinnode;       trangenode = class(tbinarynode)          constructor create(l,r : tnode);virtual;          function pass_typecheck:tnode;override;          function pass_1 : tnode;override;       end;       trangenodeclass = class of trangenode;       tcasenode = class(tunarynode)          labels    : pcaselabel;          blocks    : TFPList;          elseblock : tnode;          constructor create(l:tnode);virtual;          destructor destroy;override;          constructor ppuload(t:tnodetype;ppufile:tcompilerppufile);override;          procedure ppuwrite(ppufile:tcompilerppufile);override;          procedure buildderefimpl;override;          procedure derefimpl;override;          function dogetcopy : tnode;override;          procedure insertintolist(l : tnodelist);override;          function pass_typecheck:tnode;override;          function pass_1 : tnode;override;          function docompare(p: tnode): boolean; override;          procedure addlabel(blockid:longint;l,h : TConstExprInt);          procedure addblock(blockid:longint;instr:tnode);          procedure addelseblock(instr:tnode);       end;       tcasenodeclass = class of tcasenode;    var       csetelementnode : tsetelementnodeclass;       cinnode : tinnodeclass;       crangenode : trangenodeclass;       ccasenode : tcasenodeclass;    { counts the labels }    function case_count_labels(root : pcaselabel) : longint;    { searches the highest label }    function case_get_max(root : pcaselabel) : tconstexprint;    { searches the lowest label }    function case_get_min(root : pcaselabel) : tconstexprint;implementation    uses      systems,      verbose,      symconst,symdef,symsym,symtable,defutil,defcmp,      htypechk,pass_1,      nbas,ncnv,ncon,nld,cgobj,cgbase;{*****************************************************************************                           TSETELEMENTNODE*****************************************************************************}    constructor tsetelementnode.create(l,r : tnode);      begin         inherited create(setelementn,l,r);      end;    function tsetelementnode.pass_typecheck:tnode;      begin         result:=nil;         typecheckpass(left);         if assigned(right) then          typecheckpass(right);         set_varstate(left,vs_read,[vsf_must_be_valid]);         if codegenerror then          exit;         resultdef:=left.resultdef;      end;    function tsetelementnode.pass_1 : tnode;      begin         result:=nil;         firstpass(left);         if assigned(right) then          firstpass(right);         if codegenerror then          exit;         expectloc:=left.expectloc;         calcregisters(self,0,0,0);      end;{*****************************************************************************                              TINNODE*****************************************************************************}    constructor tinnode.create(l,r : tnode);      begin         inherited create(inn,l,r);      end;    function tinnode.pass_typecheck:tnode;      var        t : tnode;        pst : pconstset;        function createsetconst(psd : tsetdef) : pconstset;        var          pcs : pconstset;          pes : tenumsym;          i : longint;        begin          new(pcs);          case psd.elementdef.typ of            enumdef :              begin                pes:=tenumsym(tenumdef(psd.elementdef).firstenum);                while assigned(pes) do                  begin                    include(pcs^,pes.value);                    pes:=pes.nextenum;                  end;              end;            orddef :              begin                for i:=torddef(psd.elementdef).low to torddef(psd.elementdef).high do                  include(pcs^,i);              end;          end;          createsetconst:=pcs;        end;      begin         result:=nil;         resultdef:=booltype;         typecheckpass(right);         set_varstate(right,vs_read,[vsf_must_be_valid]);         if codegenerror then          exit;         { Convert array constructor first to set }         if is_array_constructor(right.resultdef) then          begin            arrayconstructor_to_set(right);            firstpass(right);            if codegenerror then             exit;          end;         if right.resultdef.typ<>setdef then           CGMessage(sym_e_set_expected);         if codegenerror then           exit;         if (right.nodetype=typen) then           begin             { we need to create a setconstn }             pst:=createsetconst(tsetdef(ttypenode(right).resultdef));             t:=csetconstnode.create(pst,ttypenode(right).resultdef);             dispose(pst);             right.free;             right:=t;           end;         typecheckpass(left);         set_varstate(left,vs_read,[vsf_must_be_valid]);         if codegenerror then           exit;         if not assigned(left.resultdef) then           internalerror(20021126);         if (m_tp7 in current_settings.modeswitches) then           begin             { insert a hint that a range check error might occur on non-byte               elements with the in operator.             }             if  (                   (left.resultdef.typ = orddef) and not                   (torddef(left.resultdef).ordtype in [s8bit,u8bit,uchar,bool8bit])                 )                or                 (                   (left.resultdef.typ = enumdef) and                   (tenumdef(left.resultdef).maxval > 255)                 )               then                 CGMessage(type_h_in_range_check);             { type conversion/check }             if assigned(tsetdef(right.resultdef).elementdef) then               inserttypeconv(left,tsetdef(right.resultdef).elementdef);           end         else if not is_ordinal(left.resultdef) or (left.resultdef.size > u32inttype.size) then           begin             CGMessage(type_h_in_range_check);             if is_signed(left.resultdef) then               inserttypeconv(left,s32inttype)             else               inserttypeconv(left,u32inttype);           end         else if assigned(tsetdef(right.resultdef).elementdef) and                 not(is_integer(tsetdef(right.resultdef).elementdef) and                     is_integer(left.resultdef)) then            { Type conversion to check things like 'char in set_of_byte'. }            { Can't use is_subequal because that will fail for            }            { 'widechar in set_of_char'                                   }            { Can't use the type conversion for integers because then     }            { "longint in set_of_byte" will give a range check error      }            { instead of false                                            }            inserttypeconv(left,tsetdef(right.resultdef).elementdef);         { empty set then return false }         if not assigned(tsetdef(right.resultdef).elementdef) or            ((right.nodetype = setconstn) and             (tnormalset(tsetconstnode(right).value_set^) = [])) then          begin            t:=cordconstnode.create(0,booltype,false);            typecheckpass(t);            result:=t;            exit;          end;         { constant evaluation }         if (left.nodetype=ordconstn) then           begin             if (right.nodetype=setconstn) then               begin                 { tordconstnode.value is int64 -> signed -> the expression }                 { below will be converted to longint on 32 bit systems due }                 { to the rule above -> will give range check error if      }                 { value > high(longint) if we don't take the signedness    }                 { into account                                             }                 if is_signed(left.resultdef) then                   t:=cordconstnode.create(byte(tordconstnode(left).value in Tsetconstnode(right).value_set^),                     booltype,true)                 else                   t:=cordconstnode.create(byte(TConstExprUInt(tordconstnode(left).value) in Tsetconstnode(right).value_set^),                     booltype,true);                                    typecheckpass(t);                 result:=t;                 exit;               end             else               begin                 if (Tordconstnode(left).value<int64(tsetdef(right.resultdef).setbase)) or                    (Tordconstnode(left).value>int64(Tsetdef(right.resultdef).setmax)) then                   begin                     t:=cordconstnode.create(0, booltype, true);                     typecheckpass(t);                     result:=t;                     exit;                   end;               end;           end;      end;    function tinnode.pass_1 : tnode;      begin         result:=nil;         expectloc:=LOC_REGISTER;         firstpass(right);         firstpass(left);         if codegenerror then           exit;         left_right_max;         if tsetdef(right.resultdef).settype<>smallset then           begin             if registersint < 3 then               registersint := 3;           end         else           begin              { a smallset needs maybe an misc. register }              if (left.nodetype<>ordconstn) and                 not(right.expectloc in [LOC_CREGISTER,LOC_REGISTER]) and                 (right.registersint<1) then                inc(registersint);           end;      end;{*****************************************************************************                              TRANGENODE*****************************************************************************}    constructor trangenode.create(l,r : tnode);      begin         inherited create(rangen,l,r);      end;    function trangenode.pass_typecheck : tnode;      begin         result:=nil;         typecheckpass(left);         typecheckpass(right);         set_varstate(left,vs_read,[vsf_must_be_valid]);         set_varstate(right,vs_read,[vsf_must_be_valid]);         if codegenerror then           exit;         { both types must be compatible }         if compare_defs(left.resultdef,right.resultdef,left.nodetype)=te_incompatible then           IncompatibleTypes(left.resultdef,right.resultdef);         { Check if only when its a constant set }         if (left.nodetype=ordconstn) and (right.nodetype=ordconstn) then          begin            { upper limit must be greater or equal than lower limit }            if (tordconstnode(left).value>tordconstnode(right).value) and               ((tordconstnode(left).value<0) or (tordconstnode(right).value>=0)) then              CGMessage(parser_e_upper_lower_than_lower);          end;        resultdef:=left.resultdef;      end;    function trangenode.pass_1 : tnode;      begin         result:=nil;         firstpass(left);         firstpass(right);         if codegenerror then           exit;        left_right_max;        expectloc:=left.expectloc;      end;{*****************************************************************************                              Case Helpers*****************************************************************************}    function case_count_labels(root : pcaselabel) : longint;      var         _l : longint;      procedure count(p : pcaselabel);        begin           inc(_l);           if assigned(p^.less) then             count(p^.less);           if assigned(p^.greater) then             count(p^.greater);        end;      begin         _l:=0;         count(root);         case_count_labels:=_l;      end;    function case_get_max(root : pcaselabel) : tconstexprint;      var         hp : pcaselabel;      begin         hp:=root;         while assigned(hp^.greater) do           hp:=hp^.greater;         case_get_max:=hp^._high;      end;    function case_get_min(root : pcaselabel) : tconstexprint;      var         hp : pcaselabel;      begin         hp:=root;         while assigned(hp^.less) do           hp:=hp^.less;         case_get_min:=hp^._low;      end;    procedure deletecaselabels(p : pcaselabel);      begin         if assigned(p^.greater) then           deletecaselabels(p^.greater);         if assigned(p^.less) then           deletecaselabels(p^.less);         dispose(p);      end;    function copycaselabel(p : pcaselabel) : pcaselabel;      var         n : pcaselabel;      begin         new(n);         n^:=p^;         if assigned(p^.greater) then           n^.greater:=copycaselabel(p^.greater);         if assigned(p^.less) then           n^.less:=copycaselabel(p^.less);         copycaselabel:=n;      end;    procedure ppuwritecaselabel(ppufile:tcompilerppufile;p : pcaselabel);      var        b : byte;      begin        ppufile.putexprint(p^._low);        ppufile.putexprint(p^._high);        ppufile.putlongint(p^.blockid);        b:=0;        if assigned(p^.greater) then         b:=b or 1;        if assigned(p^.less) then         b:=b or 2;        ppufile.putbyte(b);        if assigned(p^.greater) then          ppuwritecaselabel(ppufile,p^.greater);        if assigned(p^.less) then          ppuwritecaselabel(ppufile,p^.less);      end;    function ppuloadcaselabel(ppufile:tcompilerppufile):pcaselabel;      var        b : byte;        p : pcaselabel;      begin        new(p);        p^._low:=ppufile.getexprint;        p^._high:=ppufile.getexprint;        p^.blockid:=ppufile.getlongint;        b:=ppufile.getbyte;        if (b and 1)=1 then         p^.greater:=ppuloadcaselabel(ppufile)        else         p^.greater:=nil;        if (b and 2)=2 then         p^.less:=ppuloadcaselabel(ppufile)        else         p^.less:=nil;        ppuloadcaselabel:=p;      end;{*****************************************************************************                              TCASENODE*****************************************************************************}    constructor tcasenode.create(l:tnode);      begin         inherited create(casen,l);         labels:=nil;         blocks:=TFPList.create;         elseblock:=nil;      end;    destructor tcasenode.destroy;      var        i : longint;        hp : pcaseblock;      begin         elseblock.free;         deletecaselabels(labels);         for i:=0 to blocks.count-1 do           begin             pcaseblock(blocks[i])^.statement.free;             hp:=pcaseblock(blocks[i]);             dispose(hp);           end;         blocks.free;         inherited destroy;      end;    constructor tcasenode.ppuload(t:tnodetype;ppufile:tcompilerppufile);      var        cnt,i : longint;      begin        inherited ppuload(t,ppufile);        elseblock:=ppuloadnode(ppufile);        cnt:=ppufile.getlongint();        blocks:=TFPList.create;        for i:=0 to cnt-1 do          addblock(i,ppuloadnode(ppufile));        labels:=ppuloadcaselabel(ppufile);      end;    procedure tcasenode.ppuwrite(ppufile:tcompilerppufile);      var        i : longint;      begin        inherited ppuwrite(ppufile);        ppuwritenode(ppufile,elseblock);        ppufile.putlongint(blocks.count);        for i:=0 to blocks.count-1 do          ppuwritenode(ppufile,pcaseblock(blocks[i])^.statement);        ppuwritecaselabel(ppufile,labels);      end;    procedure tcasenode.buildderefimpl;      var        i : integer;      begin        inherited buildderefimpl;        if assigned(elseblock) then          elseblock.buildderefimpl;        for i:=0 to blocks.count-1 do          pcaseblock(blocks[i])^.statement.buildderefimpl;      end;    procedure tcasenode.derefimpl;      var        i : integer;      begin        inherited derefimpl;        if assigned(elseblock) then          elseblock.derefimpl;        for i:=0 to blocks.count-1 do          pcaseblock(blocks[i])^.statement.derefimpl;      end;    function tcasenode.pass_typecheck : tnode;      begin        result:=nil;        resultdef:=voidtype;      end;    function tcasenode.pass_1 : tnode;      var         old_t_times : longint;         hp : tnode;         i  : integer;      begin         result:=nil;         expectloc:=LOC_VOID;         { evalutes the case expression }         firstpass(left);         set_varstate(left,vs_read,[vsf_must_be_valid]);         if codegenerror then           exit;         registersint:=left.registersint;         registersfpu:=left.registersfpu;{$ifdef SUPPORT_MMX}         registersmmx:=left.registersmmx;{$endif SUPPORT_MMX}         { walk through all instructions }         { estimates the repeat of each instruction }         old_t_times:=cg.t_times;         if not(cs_opt_size in current_settings.optimizerswitches) then           begin              cg.t_times:=cg.t_times div case_count_labels(labels);              if cg.t_times<1 then                cg.t_times:=1;           end;         { first case }         for i:=0 to blocks.count-1 do           begin              firstpass(pcaseblock(blocks[i])^.statement);              { searchs max registers }              hp:=pcaseblock(blocks[i])^.statement;              if hp.registersint>registersint then                registersint:=hp.registersint;              if hp.registersfpu>registersfpu then                registersfpu:=hp.registersfpu;{$ifdef SUPPORT_MMX}              if hp.registersmmx>registersmmx then                registersmmx:=hp.registersmmx;{$endif SUPPORT_MMX}           end;         { may be handle else tree }         if assigned(elseblock) then           begin              firstpass(elseblock);              if registersint<elseblock.registersint then                registersint:=elseblock.registersint;              if registersfpu<elseblock.registersfpu then                registersfpu:=elseblock.registersfpu;{$ifdef SUPPORT_MMX}              if registersmmx<elseblock.registersmmx then                registersmmx:=elseblock.registersmmx;{$endif SUPPORT_MMX}           end;         cg.t_times:=old_t_times;         { there is one register required for the case expression    }         { for 64 bit ints we cheat: the high dword is stored in EDI }         { so we don't need an extra register                        }         if registersint<1 then           registersint:=1;      end;    function tcasenode.dogetcopy : tnode;      var         n : tcasenode;         i : longint;      begin         n:=tcasenode(inherited dogetcopy);         if assigned(elseblock) then           n.elseblock:=elseblock.dogetcopy         else           n.elseblock:=nil;         if assigned(labels) then           n.labels:=copycaselabel(labels)         else           n.labels:=nil;         if assigned(blocks) then           begin             n.blocks:=TFPList.create;             for i:=0 to blocks.count-1 do               begin                 if not assigned(blocks[i]) then                   internalerror(200411302);                 n.addblock(i,pcaseblock(blocks[i])^.statement.dogetcopy);               end;           end         else           n.labels:=nil;         dogetcopy:=n;      end;    procedure tcasenode.insertintolist(l : tnodelist);      begin      end;    function caselabelsequal(n1,n2: pcaselabel): boolean;      begin        result :=          (not assigned(n1) and not assigned(n2)) or          (assigned(n1) and assigned(n2) and           (n1^._low = n2^._low) and           (n1^._high = n2^._high) and           { the rest of the fields don't matter for equality (JM) }           caselabelsequal(n1^.less,n2^.less) and           caselabelsequal(n1^.greater,n2^.greater))      end;    function caseblocksequal(b1,b2:TFPList): boolean;      var        i : longint;      begin        result:=false;        if b1.count<>b2.count then          exit;        for i:=0 to b1.count-1 do          begin            if not pcaseblock(b1[i])^.statement.isequal(pcaseblock(b2[i])^.statement) then              exit;          end;        result:=true;      end;    function tcasenode.docompare(p: tnode): boolean;      begin        result :=          inherited docompare(p) and          caselabelsequal(labels,tcasenode(p).labels) and          caseblocksequal(blocks,tcasenode(p).blocks) and          elseblock.isequal(tcasenode(p).elseblock);      end;    procedure tcasenode.addblock(blockid:longint;instr:tnode);      var        hcaseblock : pcaseblock;      begin        new(hcaseblock);        fillchar(hcaseblock^,sizeof(hcaseblock^),0);        hcaseblock^.statement:=instr;        if blockid>=blocks.count then          blocks.count:=blockid+1;        blocks[blockid]:=hcaseblock;      end;    procedure tcasenode.addelseblock(instr:tnode);      begin        elseblock:=instr;      end;    procedure tcasenode.addlabel(blockid:longint;l,h : TConstExprInt);      var        hcaselabel : pcaselabel;        function insertlabel(var p : pcaselabel):pcaselabel;          begin             if p=nil then               begin                 p:=hcaselabel;                 result:=p;               end             else              if (p^._low>hcaselabel^._low) and                 (p^._low>hcaselabel^._high) then                begin                  if (hcaselabel^.blockid = p^.blockid) and                     (p^._low = hcaselabel^._high + 1) then                    begin                      p^._low := hcaselabel^._low;                      dispose(hcaselabel);                      result:=p;                    end                  else                    result:=insertlabel(p^.less)                end             else               if (p^._high<hcaselabel^._low) and                  (p^._high<hcaselabel^._high) then                 begin                    if (hcaselabel^.blockid = p^.blockid) and                       (p^._high+1 = hcaselabel^._low) then                      begin                        p^._high := hcaselabel^._high;                        dispose(hcaselabel);                        result:=p;                      end                    else                      result:=insertlabel(p^.greater);                 end             else               Message(parser_e_double_caselabel);          end;      begin        new(hcaselabel);        fillchar(hcaselabel^,sizeof(tcaselabel),0);        hcaselabel^.blockid:=blockid;        hcaselabel^._low:=l;        hcaselabel^._high:=h;        insertlabel(labels);      end;begin   csetelementnode:=tsetelementnode;   cinnode:=tinnode;   crangenode:=trangenode;   ccasenode:=tcasenode;end.
 |