| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854 | {    Copyright (c) 2000-2002 by Florian Klaempfl    Type checking and register allocation for math 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 nmat;{$i fpcdefs.inc}interface    uses       node;    type       tmoddivnode = class(tbinopnode)          function pass_1 : tnode;override;          function det_resulttype:tnode;override;         protected{$ifndef cpu64bit}          { override the following if you want to implement }          { parts explicitely in the code generator (JM)    }          function first_moddiv64bitint: tnode; virtual;{$endif cpu64bit}          function firstoptimize: tnode; virtual;          function first_moddivint: tnode; virtual;       end;       tmoddivnodeclass = class of tmoddivnode;       tshlshrnode = class(tbinopnode)          function pass_1 : tnode;override;          function det_resulttype:tnode;override;{$ifndef cpu64bit}          { override the following if you want to implement }          { parts explicitely in the code generator (CEC)            Should return nil, if everything will be handled            in the code generator          }          function first_shlshr64bitint: tnode; virtual;{$endif cpu64bit}       end;       tshlshrnodeclass = class of tshlshrnode;       tunaryminusnode = class(tunarynode)          constructor create(expr : tnode);virtual;          function pass_1 : tnode;override;          function det_resulttype:tnode;override;       end;       tunaryminusnodeclass = class of tunaryminusnode;       tnotnode = class(tunarynode)          constructor create(expr : tnode);virtual;          function pass_1 : tnode;override;          function det_resulttype:tnode;override;       {$ifdef state_tracking}          function track_state_pass(exec_known:boolean):boolean;override;       {$endif}       end;       tnotnodeclass = class of tnotnode;    var       cmoddivnode : tmoddivnodeclass;       cshlshrnode : tshlshrnodeclass;       cunaryminusnode : tunaryminusnodeclass;       cnotnode : tnotnodeclass;implementation    uses      systems,      verbose,globals,cutils,      globtype,      symconst,symtype,symdef,defutil,      htypechk,pass_1,      cgbase,      ncon,ncnv,ncal,nadd;{****************************************************************************                              TMODDIVNODE ****************************************************************************}    function tmoddivnode.det_resulttype:tnode;      var         hp,t : tnode;         rd,ld : torddef;         rv,lv : tconstexprint;      begin         result:=nil;         resulttypepass(left);         resulttypepass(right);         set_varstate(left,vs_used,[vsf_must_be_valid]);         set_varstate(right,vs_used,[vsf_must_be_valid]);         if codegenerror then           exit;         { we need 2 orddefs always }         if (left.resulttype.def.deftype<>orddef) then           inserttypeconv(right,sinttype);         if (right.resulttype.def.deftype<>orddef) then           inserttypeconv(right,sinttype);         if codegenerror then           exit;         rd:=torddef(right.resulttype.def);         ld:=torddef(left.resulttype.def);         { check for division by zero }         if is_constintnode(right) then           begin             rv:=tordconstnode(right).value;             if (rv=0) then               begin                 Message(parser_e_division_by_zero);                 { recover }                 rv:=1;               end;             if is_constintnode(left) then               begin                 lv:=tordconstnode(left).value;                 case nodetype of                   modn:                     if (torddef(ld).typ <> u64bit) or                        (torddef(rd).typ <> u64bit) then                       t:=genintconstnode(lv mod rv)                     else                       t:=genintconstnode(int64(qword(lv) mod qword(rv)));                   divn:                     if (torddef(ld).typ <> u64bit) or                        (torddef(rd).typ <> u64bit) then                       t:=genintconstnode(lv div rv)                     else                       t:=genintconstnode(int64(qword(lv) div qword(rv)));                 end;                 result:=t;                 exit;              end;            end;         { allow operator overloading }         t:=self;         if isbinaryoverloaded(t) then           begin              result:=t;              exit;           end;         { if one operand is a cardinal and the other is a positive constant, convert the }         { constant to a cardinal as well so we don't have to do a 64bit division (JM)    }         { Do the same for qwords and positive constants as well, otherwise things like   }         { "qword mod 10" are evaluated with int64 as result, which is wrong if the       }         { "qword" was > high(int64) (JM)                                                 }         if (rd.typ in [u32bit,u64bit]) and            is_constintnode(left) and            (tordconstnode(left).value >= 0) then           begin             inserttypeconv(left,right.resulttype);             ld:=torddef(left.resulttype.def);           end;         if (ld.typ in [u32bit,u64bit]) and            is_constintnode(right) and            (tordconstnode(right).value >= 0) then          begin            inserttypeconv(right,left.resulttype);            rd:=torddef(right.resulttype.def);          end;         { when there is one currency value, everything is done           using currency }         if (ld.typ=scurrency) or            (rd.typ=scurrency) then           begin             if (ld.typ<>scurrency) then              inserttypeconv(left,s64currencytype);             if (rd.typ<>scurrency) then              inserttypeconv(right,s64currencytype);             resulttype:=left.resulttype;           end         else{$ifndef cpu64bit}          { when there is one 64bit value, everything is done            in 64bit }          if (is_64bitint(left.resulttype.def) or              is_64bitint(right.resulttype.def)) then           begin             if is_signed(rd) or is_signed(ld) then               begin                  if (ld.typ<>s64bit) then                    inserttypeconv(left,s64inttype);                  if (rd.typ<>s64bit) then                    inserttypeconv(right,s64inttype);               end             else               begin                  if (ld.typ<>u64bit) then                    inserttypeconv(left,u64inttype);                  if (rd.typ<>u64bit) then                    inserttypeconv(right,u64inttype);               end;             resulttype:=left.resulttype;           end         else          { when mixing cardinals and signed numbers, convert everythign to 64bit (JM) }          if ((rd.typ = u32bit) and              is_signed(ld)) or             ((ld.typ = u32bit) and              is_signed(rd)) then           begin              CGMessage(type_w_mixed_signed_unsigned);              if (ld.typ<>s64bit) then                inserttypeconv(left,s64inttype);              if (rd.typ<>s64bit) then                inserttypeconv(right,s64inttype);              resulttype:=left.resulttype;           end         else{$endif cpu64bit}           begin              { Make everything always default singed int }              if not(rd.typ in [torddef(sinttype.def).typ,torddef(uinttype.def).typ]) then                inserttypeconv(right,sinttype);              if not(ld.typ in [torddef(sinttype.def).typ,torddef(uinttype.def).typ]) then                inserttypeconv(left,sinttype);              resulttype:=right.resulttype;           end;         { when the result is currency we need some extra code for           division. this should not be done when the divn node is           created internally }         if (nodetype=divn) and            not(nf_is_currency in flags) and            is_currency(resulttype.def) then          begin            hp:=caddnode.create(muln,getcopy,cordconstnode.create(10000,s64currencytype,false));            include(hp.flags,nf_is_currency);            result:=hp;          end;      end;    function tmoddivnode.first_moddivint: tnode;{$ifdef cpuneedsdiv32helper}      var        procname: string[31];      begin        result := nil;        { otherwise create a call to a helper }        if nodetype = divn then          procname := 'fpc_div_'        else          procname := 'fpc_mod_';        { only qword needs the unsigned code, the          signed code is also used for currency }        if is_signed(resulttype.def) then          procname := procname + 'longint'        else          procname := procname + 'dword';        result := ccallnode.createintern(procname,ccallparanode.create(left,          ccallparanode.create(right,nil)));        left := nil;        right := nil;        firstpass(result);      end;{$else cpuneedsdiv32helper}      begin        result:=nil;      end;{$endif cpuneedsdiv32helper}{$ifndef cpu64bit}    function tmoddivnode.first_moddiv64bitint: tnode;      var        procname: string[31];      begin        result := nil;        { when currency is used set the result of the          parameters to s64bit, so they are not converted }        if is_currency(resulttype.def) then          begin            left.resulttype:=s64inttype;            right.resulttype:=s64inttype;          end;        { otherwise create a call to a helper }        if nodetype = divn then          procname := 'fpc_div_'        else          procname := 'fpc_mod_';        { only qword needs the unsigned code, the          signed code is also used for currency }        if is_signed(resulttype.def) then          procname := procname + 'int64'        else          procname := procname + 'qword';        result := ccallnode.createintern(procname,ccallparanode.create(left,          ccallparanode.create(right,nil)));        left := nil;        right := nil;        firstpass(result);      end;{$endif cpu64bit}    function tmoddivnode.firstoptimize: tnode;      var        power{,shiftval} : longint;        newtype: tnodetype;      begin        result := nil;        { divide/mod a number by a constant which is a power of 2? }        if (cs_optimize in aktglobalswitches) and           (right.nodetype = ordconstn) and{           ((nodetype = divn) or            not is_signed(resulttype.def)) and}           (not is_signed(resulttype.def)) and           ispowerof2(tordconstnode(right).value,power) then          begin            if nodetype = divn then              begin(*                if is_signed(resulttype.def) then                  begin                    if is_64bitint(left.resulttype.def) then                      if not (cs_littlesize in aktglobalswitches) then                        shiftval := 63                      else                        { the shift code is a lot bigger than the call to }                        { the divide helper                               }                        exit                    else                      shiftval := 31;                    { we reuse left twice, so create once a copy of it     }                    { !!! if left is a call is -> call gets executed twice }                    left := caddnode.create(addn,left,                      caddnode.create(andn,                        cshlshrnode.create(sarn,left.getcopy,                          cordconstnode.create(shiftval,sinttype,false)),                        cordconstnode.create(tordconstnode(right).value-1,                          right.resulttype,false)));                    newtype := sarn;                  end                else*)                  newtype := shrn;                tordconstnode(right).value := power;                result := cshlshrnode.create(newtype,left,right)              end            else              begin                dec(tordconstnode(right).value);                result := caddnode.create(andn,left,right);              end;            { left and right are reused }            left := nil;            right := nil;            firstpass(result);            exit;          end;      end;    function tmoddivnode.pass_1 : tnode;      begin         result:=nil;         firstpass(left);         firstpass(right);         if codegenerror then           exit;         { Try to optimize mod/div }         result := firstoptimize;         if assigned(result) then           exit;{$ifndef cpu64bit}         { 64bit }         if (left.resulttype.def.deftype=orddef) and            (right.resulttype.def.deftype=orddef) and            (is_64bitint(left.resulttype.def) or is_64bitint(right.resulttype.def)) then           begin             result := first_moddiv64bitint;             if assigned(result) then               exit;             expectloc:=LOC_REGISTER;             calcregisters(self,2,0,0);           end         else{$endif cpu64bit}           begin             result := first_moddivint;             if assigned(result) then               exit;             left_right_max;             if left.registersint<=right.registersint then              inc(registersint);           end;         expectloc:=LOC_REGISTER;      end;{****************************************************************************                              TSHLSHRNODE ****************************************************************************}    function tshlshrnode.det_resulttype:tnode;      var         t : tnode;      begin         result:=nil;         resulttypepass(left);         resulttypepass(right);         set_varstate(right,vs_used,[vsf_must_be_valid]);         set_varstate(left,vs_used,[vsf_must_be_valid]);         if codegenerror then           exit;         { constant folding }         if is_constintnode(left) and is_constintnode(right) then           begin              case nodetype of                 shrn:                   t:=genintconstnode(tordconstnode(left).value shr tordconstnode(right).value);                 shln:                   t:=genintconstnode(tordconstnode(left).value shl tordconstnode(right).value);              end;              result:=t;              exit;           end;         { allow operator overloading }         t:=self;         if isbinaryoverloaded(t) then           begin              result:=t;              exit;           end;         { calculations for ordinals < 32 bit have to be done in           32 bit for backwards compatibility. That way 'shl 33' is           the same as 'shl 1'. It's ugly but compatible with delphi/tp/gcc }         if (not is_64bit(left.resulttype.def)) and            (torddef(left.resulttype.def).typ<>u32bit) then           inserttypeconv(left,s32inttype);         inserttypeconv(right,sinttype);         resulttype:=left.resulttype;      end;{$ifndef cpu64bit}    function tshlshrnode.first_shlshr64bitint: tnode;      var        procname: string[31];      begin        result := nil;        { otherwise create a call to a helper }        if nodetype = shln then          procname := 'fpc_shl_int64'        else          procname := 'fpc_shr_int64';        { this order of parameters works at least for the arm,          however it should work for any calling conventions (FK) }        result := ccallnode.createintern(procname,ccallparanode.create(right,          ccallparanode.create(left,nil)));        left := nil;        right := nil;        firstpass(result);      end;{$endif cpu64bit}    function tshlshrnode.pass_1 : tnode;      var         regs : longint;      begin         result:=nil;         firstpass(left);         firstpass(right);         if codegenerror then           exit;{$ifndef cpu64bit}         { 64 bit ints have their own shift handling }         if is_64bit(left.resulttype.def) then           begin             result := first_shlshr64bitint;             if assigned(result) then               exit;             regs:=2;           end         else{$endif cpu64bit}           begin             regs:=1           end;         if (right.nodetype<>ordconstn) then           inc(regs);         expectloc:=LOC_REGISTER;         calcregisters(self,regs,0,0);      end;{****************************************************************************                            TUNARYMINUSNODE ****************************************************************************}    constructor tunaryminusnode.create(expr : tnode);      begin         inherited create(unaryminusn,expr);      end;    function tunaryminusnode.det_resulttype : tnode;      var         t : tnode;      begin         result:=nil;         resulttypepass(left);         set_varstate(left,vs_used,[vsf_must_be_valid]);         if codegenerror then           exit;         { constant folding }         if is_constintnode(left) then           begin              result:=genintconstnode(-tordconstnode(left).value);              exit;           end;         if is_constrealnode(left) then           begin              trealconstnode(left).value_real:=-trealconstnode(left).value_real;              result:=left;              left:=nil;              exit;           end;         resulttype:=left.resulttype;         if (left.resulttype.def.deftype=floatdef) then           begin           end{$ifdef SUPPORT_MMX}         else if (cs_mmx in aktlocalswitches) and           is_mmx_able_array(left.resulttype.def) then             begin               { if saturation is on, left.resulttype.def isn't                 "mmx able" (FK)               if (cs_mmx_saturation in aktlocalswitches^) and                 (torddef(tarraydef(resulttype.def).definition).typ in                 [s32bit,u32bit]) then                 CGMessage(type_e_mismatch);               }             end{$endif SUPPORT_MMX}{$ifndef cpu64bit}         else if is_64bitint(left.resulttype.def) then           begin           end{$endif cpu64bit}         else if (left.resulttype.def.deftype=orddef) then           begin              inserttypeconv(left,sinttype);              resulttype:=left.resulttype;           end         else           begin             { allow operator overloading }             t:=self;             if isunaryoverloaded(t) then               begin                  result:=t;                  exit;               end;             CGMessage(type_e_mismatch);           end;      end;    { generic code     }    { overridden by:   }    {   i386           }    function tunaryminusnode.pass_1 : tnode;      begin         result:=nil;         firstpass(left);         if codegenerror then           exit;         registersint:=left.registersint;         registersfpu:=left.registersfpu;{$ifdef SUPPORT_MMX}         registersmmx:=left.registersmmx;{$endif SUPPORT_MMX}         if (left.resulttype.def.deftype=floatdef) then           begin              if (left.expectloc<>LOC_REGISTER) and                 (registersfpu<1) then                registersfpu:=1;              expectloc:=LOC_FPUREGISTER;           end{$ifdef SUPPORT_MMX}         else if (cs_mmx in aktlocalswitches) and           is_mmx_able_array(left.resulttype.def) then             begin               if (left.expectloc<>LOC_MMXREGISTER) and                  (registersmmx<1) then                 registersmmx:=1;             end{$endif SUPPORT_MMX}{$ifndef cpu64bit}         else if is_64bit(left.resulttype.def) then           begin              if (left.expectloc<>LOC_REGISTER) and                 (registersint<2) then                registersint:=2;              expectloc:=LOC_REGISTER;           end{$endif cpu64bit}         else if (left.resulttype.def.deftype=orddef) then           begin              if (left.expectloc<>LOC_REGISTER) and                 (registersint<1) then                registersint:=1;              expectloc:=LOC_REGISTER;           end;      end;{****************************************************************************                               TNOTNODE ****************************************************************************}    const      boolean_reverse:array[ltn..unequaln] of Tnodetype=(        gten,gtn,lten,ltn,unequaln,equaln      );    constructor tnotnode.create(expr : tnode);      begin         inherited create(notn,expr);      end;    function tnotnode.det_resulttype : tnode;      var         t : tnode;         tt : ttype;         v : tconstexprint;      begin         result:=nil;         resulttypepass(left);         set_varstate(left,vs_used,[]);         if codegenerror then           exit;         resulttype:=left.resulttype;         { Try optmimizing ourself away }         if left.nodetype=notn then           begin             { Double not. Remove both }             result:=Tnotnode(left).left;             Tnotnode(left).left:=nil;             exit;           end;         if (left.nodetype in [ltn,lten,equaln,unequaln,gtn,gten]) then          begin            { Not of boolean expression. Turn around the operator and remove              the not. This is not allowed for sets with the gten/lten,              because there is no ltn/gtn support }            if (taddnode(left).left.resulttype.def.deftype<>setdef) or               (left.nodetype in [equaln,unequaln]) then             begin               result:=left;               left.nodetype:=boolean_reverse[left.nodetype];               left:=nil;               exit;             end;          end;         { constant folding }         if (left.nodetype=ordconstn) then           begin              v:=tordconstnode(left).value;              tt:=left.resulttype;              case torddef(left.resulttype.def).typ of                bool8bit,                bool16bit,                bool32bit :                  begin                    { here we do a boolean(byte(..)) type cast because }                    { boolean(<int64>) is buggy in 1.00                }                    v:=byte(not(boolean(byte(v))));                  end;                uchar,                uwidechar,                u8bit,                s8bit,                u16bit,                s16bit,                u32bit,                s32bit,                s64bit,                u64bit :                  begin                    v:=int64(not int64(v)); { maybe qword is required }                    int_to_type(v,tt);                  end;                else                  CGMessage(type_e_mismatch);              end;              t:=cordconstnode.create(v,tt,true);              result:=t;              exit;           end;         if is_boolean(resulttype.def) then           begin           end         else{$ifdef SUPPORT_MMX}           if (cs_mmx in aktlocalswitches) and             is_mmx_able_array(left.resulttype.def) then             begin             end         else{$endif SUPPORT_MMX}{$ifndef cpu64bit}           if is_64bitint(left.resulttype.def) then             begin             end         else{$endif cpu64bit}           if is_integer(left.resulttype.def) then             begin             end         else           begin             { allow operator overloading }             t:=self;             if isunaryoverloaded(t) then               begin                  result:=t;                  exit;               end;             CGMessage(type_e_mismatch);           end;      end;    function tnotnode.pass_1 : tnode;      begin         result:=nil;         firstpass(left);         if codegenerror then           exit;         expectloc:=left.expectloc;         registersint:=left.registersint;{$ifdef SUPPORT_MMX}         registersmmx:=left.registersmmx;{$endif SUPPORT_MMX}         if is_boolean(resulttype.def) then           begin             if (expectloc in [LOC_REFERENCE,LOC_CREFERENCE,LOC_CREGISTER]) then              begin                expectloc:=LOC_REGISTER;                if (registersint<1) then                 registersint:=1;              end;            { before loading it into flags we need to load it into              a register thus 1 register is need PM }{$ifdef cpuflags}             if left.expectloc<>LOC_JUMP then               expectloc:=LOC_FLAGS;{$endif def cpuflags}           end         else{$ifdef SUPPORT_MMX}           if (cs_mmx in aktlocalswitches) and             is_mmx_able_array(left.resulttype.def) then             begin               if (left.expectloc<>LOC_MMXREGISTER) and                 (registersmmx<1) then                 registersmmx:=1;             end         else{$endif SUPPORT_MMX}{$ifndef cpu64bit}           if is_64bit(left.resulttype.def) then             begin                if (expectloc in [LOC_REFERENCE,LOC_CREFERENCE,LOC_CREGISTER]) then                 begin                   expectloc:=LOC_REGISTER;                   if (registersint<2) then                    registersint:=2;                 end;             end         else{$endif cpu64bit}           if is_integer(left.resulttype.def) then             begin               if (left.expectloc<>LOC_REGISTER) and                  (registersint<1) then                 registersint:=1;               expectloc:=LOC_REGISTER;             end;      end;{$ifdef state_tracking}    function Tnotnode.track_state_pass(exec_known:boolean):boolean;      begin        track_state_pass:=true;        if left.track_state_pass(exec_known) then          begin            left.resulttype.def:=nil;            do_resulttypepass(left);          end;      end;{$endif}begin   cmoddivnode:=tmoddivnode;   cshlshrnode:=tshlshrnode;   cunaryminusnode:=tunaryminusnode;   cnotnode:=tnotnode;end.
 |