| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464 | {    Copyright (c) 2000-2002 by Florian Klaempfl and Jonas Maebe    Code generation for add nodes on the PowerPC    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 nppcadd;{$i fpcdefs.inc}interface    uses       node,nadd,ncgadd,cpubase;    type       tppcaddnode = class(tcgaddnode)          function pass_1: tnode; override;          procedure pass_2;override;         private          procedure pass_left_and_right;          procedure load_left_right(cmpop, load_constants: boolean);          function  getresflags : tresflags;          procedure emit_compare(unsigned : boolean);          procedure second_addfloat;override;          procedure second_addboolean;override;          procedure second_addsmallset;override;{$ifdef SUPPORT_MMX}          procedure second_addmmx;override;{$endif SUPPORT_MMX}          procedure second_add64bit;override;       end;  implementation    uses      globtype,systems,      cutils,verbose,globals,      symconst,symdef,paramgr,      aasmbase,aasmtai,aasmcpu,defutil,htypechk,      cgbase,cpuinfo,pass_1,pass_2,regvars,      cpupara,cgcpu,cgutils,      ncon,nset,      ncgutil,tgobj,rgobj,rgcpu,cgobj,cg64f32;{*****************************************************************************                                  Pass 1*****************************************************************************}    function tppcaddnode.pass_1: tnode;      begin        resulttypepass(left);        if (nodetype in [equaln,unequaln]) and           (left.resulttype.def.deftype = orddef) and           is_64bit(left.resulttype.def) then          begin            result := nil;            firstpass(left);            firstpass(right);            expectloc := LOC_FLAGS;            calcregisters(self,2,0,0);            exit;          end;        result := inherited pass_1;      end;{*****************************************************************************                                  Helpers*****************************************************************************}    procedure tppcaddnode.pass_left_and_right;      begin        { calculate the operator which is more difficult }        firstcomplex(self);        { in case of constant put it to the left }        if (left.nodetype=ordconstn) then         swapleftright;        secondpass(left);        secondpass(right);      end;    procedure tppcaddnode.load_left_right(cmpop, load_constants: boolean);      procedure load_node(var n: tnode);        begin          case n.location.loc of            LOC_REGISTER:              if not cmpop then                begin                  location.register := n.location.register;                  if is_64bit(n.resulttype.def) then                    location.register64.reghi := n.location.register64.reghi;                end;            LOC_REFERENCE,LOC_CREFERENCE:              begin                location_force_reg(exprasmlist,n.location,def_cgsize(n.resulttype.def),false);                if not cmpop then                  begin                    location.register := n.location.register;                    if is_64bit(n.resulttype.def) then                      location.register64.reghi := n.location.register64.reghi;                  end;              end;            LOC_CONSTANT:              begin                if load_constants then                  begin                    location_force_reg(exprasmlist,n.location,def_cgsize(n.resulttype.def),false);                    if not cmpop then                      begin                        location.register := n.location.register;                        if is_64bit(n.resulttype.def) then                          location.register64.reghi := n.location.register64.reghi;                      end;                  end;              end;          end;        end;      begin        load_node(left);        load_node(right);        if not(cmpop) and           (location.register = NR_NO) then         begin           location.register := cg.getintregister(exprasmlist,OS_INT);           if is_64bit(resulttype.def) then             location.register64.reghi := cg.getintregister(exprasmlist,OS_INT);         end;      end;    function tppcaddnode.getresflags : tresflags;      begin        if (left.resulttype.def.deftype <> floatdef) then          result.cr := RS_CR0        else          result.cr := RS_CR1;        case nodetype of          equaln : result.flag:=F_EQ;          unequaln : result.flag:=F_NE;        else          if nf_swaped in flags then            case nodetype of              ltn : result.flag:=F_GT;              lten : result.flag:=F_GE;              gtn : result.flag:=F_LT;              gten : result.flag:=F_LE;            end          else            case nodetype of              ltn : result.flag:=F_LT;              lten : result.flag:=F_LE;              gtn : result.flag:=F_GT;              gten : result.flag:=F_GE;            end;        end      end;    procedure tppcaddnode.emit_compare(unsigned: boolean);      var        op : tasmop;        tmpreg : tregister;        useconst : boolean;      begin        // get the constant on the right if there is one        if (left.location.loc = LOC_CONSTANT) then          swapleftright;        // can we use an immediate, or do we have to load the        // constant in a register first?        if (right.location.loc = LOC_CONSTANT) then          begin{$ifdef dummy}            if (right.location.size in [OS_64,OS_S64]) and (hi(right.location.value64)<>0) and ((hi(right.location.value64)<>$ffffffff) or unsigned) then              internalerror(2002080301);{$endif extdebug}            if (nodetype in [equaln,unequaln]) then              if (unsigned and                  (aword(right.location.value) > high(word))) or                 (not unsigned and                  (aint(right.location.value) < low(smallint)) or                   (aint(right.location.value) > high(smallint))) then                { we can then maybe use a constant in the 'othersigned' case                 (the sign doesn't matter for // equal/unequal)}                unsigned := not unsigned;            if (unsigned and                (aword(right.location.value) <= high(word))) or               (not(unsigned) and                (aint(right.location.value) >= low(smallint)) and                (aint(right.location.value) <= high(smallint))) then               useconst := true            else              begin                useconst := false;                tmpreg := cg.getintregister(exprasmlist,OS_INT);                cg.a_load_const_reg(exprasmlist,OS_INT,                    right.location.value,tmpreg);               end          end        else          useconst := false;        location.loc := LOC_FLAGS;        location.resflags := getresflags;        if not unsigned then          if useconst then            op := A_CMPWI          else            op := A_CMPW        else          if useconst then            op := A_CMPLWI          else            op := A_CMPLW;        if (right.location.loc = LOC_CONSTANT) then          begin            if useconst then              exprasmlist.concat(taicpu.op_reg_const(op,left.location.register,longint(right.location.value)))            else              exprasmlist.concat(taicpu.op_reg_reg(op,left.location.register,tmpreg));          end        else          exprasmlist.concat(taicpu.op_reg_reg(op,            left.location.register,right.location.register));      end;{*****************************************************************************                                AddBoolean*****************************************************************************}    procedure tppcaddnode.second_addboolean;      var        cgop      : TOpCg;        cgsize  : TCgSize;        cmpop,        isjump  : boolean;        otl,ofl : tasmlabel;      begin        { calculate the operator which is more difficult }        firstcomplex(self);        cmpop:=false;        if (torddef(left.resulttype.def).typ=bool8bit) or           (torddef(right.resulttype.def).typ=bool8bit) then         cgsize:=OS_8        else          if (torddef(left.resulttype.def).typ=bool16bit) or             (torddef(right.resulttype.def).typ=bool16bit) then           cgsize:=OS_16        else           cgsize:=OS_32;        if (cs_full_boolean_eval in aktlocalswitches) or           (nodetype in [unequaln,ltn,lten,gtn,gten,equaln,xorn]) then          begin            if left.nodetype in [ordconstn,realconstn] then             swapleftright;            isjump:=(left.expectloc=LOC_JUMP);            if isjump then              begin                 otl:=truelabel;                 objectlibrary.getlabel(truelabel);                 ofl:=falselabel;                 objectlibrary.getlabel(falselabel);              end;            secondpass(left);            if left.location.loc in [LOC_FLAGS,LOC_JUMP] then             location_force_reg(exprasmlist,left.location,cgsize,false);            if isjump then             begin               truelabel:=otl;               falselabel:=ofl;             end            else if left.location.loc=LOC_JUMP then              internalerror(2003122901);            isjump:=(right.expectloc=LOC_JUMP);            if isjump then              begin                 otl:=truelabel;                 objectlibrary.getlabel(truelabel);                 ofl:=falselabel;                 objectlibrary.getlabel(falselabel);              end;            secondpass(right);            if right.location.loc in [LOC_FLAGS,LOC_JUMP] then             location_force_reg(exprasmlist,right.location,cgsize,false);            if isjump then             begin               truelabel:=otl;               falselabel:=ofl;             end            else if right.location.loc=LOC_JUMP then              internalerror(200312292);            cmpop := nodetype in [ltn,lten,gtn,gten,equaln,unequaln];            { set result location }            if not cmpop then              location_reset(location,LOC_REGISTER,def_cgsize(resulttype.def))             else              location_reset(location,LOC_FLAGS,OS_NO);            load_left_right(cmpop,false);            if (left.location.loc = LOC_CONSTANT) then              swapleftright;            { compare the }            case nodetype of              ltn,lten,gtn,gten,              equaln,unequaln :                begin                  if (right.location.loc <> LOC_CONSTANT) then                    exprasmlist.concat(taicpu.op_reg_reg(A_CMPLW,                      left.location.register,right.location.register))                  else                    exprasmlist.concat(taicpu.op_reg_const(A_CMPLWI,                      left.location.register,longint(right.location.value)));                  location.resflags := getresflags;                end;              else                begin                  case nodetype of                    xorn :                      cgop:=OP_XOR;                    orn :                      cgop:=OP_OR;                    andn :                      cgop:=OP_AND;                    else                      internalerror(200203247);                  end;                  if right.location.loc <> LOC_CONSTANT then                    cg.a_op_reg_reg_reg(exprasmlist,cgop,OS_INT,                      left.location.register,right.location.register,                      location.register)                  else                    cg.a_op_const_reg_reg(exprasmlist,cgop,OS_INT,                      right.location.value,left.location.register,                      location.register);                end;            end;         end        else         begin           // just to make sure we free the right registers           cmpop := true;           case nodetype of             andn,             orn :               begin                 location_reset(location,LOC_JUMP,OS_NO);                 case nodetype of                   andn :                     begin                        otl:=truelabel;                        objectlibrary.getlabel(truelabel);                        secondpass(left);                        maketojumpbool(exprasmlist,left,lr_load_regvars);                        cg.a_label(exprasmlist,truelabel);                        truelabel:=otl;                     end;                   orn :                     begin                        ofl:=falselabel;                        objectlibrary.getlabel(falselabel);                        secondpass(left);                        maketojumpbool(exprasmlist,left,lr_load_regvars);                        cg.a_label(exprasmlist,falselabel);                        falselabel:=ofl;                     end;                   else                     internalerror(200403181);                 end;                 secondpass(right);                 maketojumpbool(exprasmlist,right,lr_load_regvars);               end;           end;         end;      end;{*****************************************************************************                                AddFloat*****************************************************************************}    procedure tppcaddnode.second_addfloat;      var        op    : TAsmOp;        cmpop : boolean;      begin        pass_left_and_right;        cmpop:=false;        case nodetype of          addn :            op:=A_FADD;          muln :            op:=A_FMUL;          subn :            op:=A_FSUB;          slashn :            op:=A_FDIV;          ltn,lten,gtn,gten,          equaln,unequaln :            begin              op:=A_FCMPO;              cmpop:=true;            end;          else            internalerror(200403182);        end;        // get the operands in the correct order, there are no special cases        // here, everything is register-based        if nf_swaped in flags then          swapleftright;        // put both operands in a register        location_force_fpureg(exprasmlist,right.location,true);        location_force_fpureg(exprasmlist,left.location,true);        // initialize de result        if not cmpop then          begin            location_reset(location,LOC_FPUREGISTER,def_cgsize(resulttype.def));            if left.location.loc = LOC_FPUREGISTER then              location.register := left.location.register            else if right.location.loc = LOC_FPUREGISTER then              location.register := right.location.register            else              location.register := cg.getfpuregister(exprasmlist,location.size);          end        else         begin           location_reset(location,LOC_FLAGS,OS_NO);           location.resflags := getresflags;         end;        // emit the actual operation        if not cmpop then          begin            exprasmlist.concat(taicpu.op_reg_reg_reg(op,              location.register,left.location.register,              right.location.register))          end        else          begin            exprasmlist.concat(taicpu.op_reg_reg_reg(op,              newreg(R_SPECIALREGISTER,location.resflags.cr,R_SUBNONE),left.location.register,right.location.register))          end;      end;{*****************************************************************************                                AddSmallSet*****************************************************************************}    procedure tppcaddnode.second_addsmallset;      var        cgop   : TOpCg;        tmpreg : tregister;        opdone,        cmpop  : boolean;      begin        pass_left_and_right;        { when a setdef is passed, it has to be a smallset }        if ((left.resulttype.def.deftype=setdef) and            (tsetdef(left.resulttype.def).settype<>smallset)) or           ((right.resulttype.def.deftype=setdef) and            (tsetdef(right.resulttype.def).settype<>smallset)) then         internalerror(200203301);        opdone := false;        cmpop:=nodetype in [equaln,unequaln,lten,gten];        { set result location }        if not cmpop then          location_reset(location,LOC_REGISTER,def_cgsize(resulttype.def))         else          location_reset(location,LOC_FLAGS,OS_NO);        load_left_right(cmpop,false);        if not(cmpop) and           (location.register = NR_NO) then          location.register := cg.getintregister(exprasmlist,OS_INT);        case nodetype of          addn :            begin              if (nf_swaped in flags) and (left.nodetype=setelementn) then                swapleftright;              { are we adding set elements ? }              if right.nodetype=setelementn then                begin                  { no range support for smallsets! }                  if assigned(tsetelementnode(right).right) then                   internalerror(43244);                  if (right.location.loc = LOC_CONSTANT) then                    cg.a_op_const_reg_reg(exprasmlist,OP_OR,OS_INT,                      aint(aword(1) shl aword(right.location.value)),                      left.location.register,location.register)                  else                    begin                      tmpreg := cg.getintregister(exprasmlist,OS_INT);                      cg.a_load_const_reg(exprasmlist,OS_INT,1,tmpreg);                      cg.a_op_reg_reg(exprasmlist,OP_SHL,OS_INT,                        right.location.register,tmpreg);                      if left.location.loc <> LOC_CONSTANT then                        cg.a_op_reg_reg_reg(exprasmlist,OP_OR,OS_INT,tmpreg,                          left.location.register,location.register)                      else                        cg.a_op_const_reg_reg(exprasmlist,OP_OR,OS_INT,                          left.location.value,tmpreg,location.register);                    end;                  opdone := true;                end              else                cgop := OP_OR;            end;          symdifn :            cgop:=OP_XOR;          muln :            cgop:=OP_AND;          subn :            begin              cgop:=OP_AND;              if (not(nf_swaped in flags)) then                if (right.location.loc=LOC_CONSTANT) then                  right.location.value := not(right.location.value)                else                  opdone := true              else if (left.location.loc=LOC_CONSTANT) then                left.location.value := not(left.location.value)              else                 begin                   swapleftright;                   opdone := true;                 end;              if opdone then                begin                  if left.location.loc = LOC_CONSTANT then                    begin                      tmpreg := cg.getintregister(exprasmlist,OS_INT);                      cg.a_load_const_reg(exprasmlist,OS_INT,                        left.location.value,tmpreg);                      exprasmlist.concat(taicpu.op_reg_reg_reg(A_ANDC,                        location.register,tmpreg,right.location.register));                    end                  else                    exprasmlist.concat(taicpu.op_reg_reg_reg(A_ANDC,                      location.register,left.location.register,                      right.location.register));                end;            end;          equaln,          unequaln :            begin              emit_compare(true);              opdone := true;            end;          lten,gten:            begin              If (not(nf_swaped in flags) and                  (nodetype = lten)) or                 ((nf_swaped in flags) and                  (nodetype = gten)) then                swapleftright;              // now we have to check whether left >= right              tmpreg := cg.getintregister(exprasmlist,OS_INT);              if left.location.loc = LOC_CONSTANT then                begin                  cg.a_op_const_reg_reg(exprasmlist,OP_AND,OS_INT,                    not(left.location.value),right.location.register,tmpreg);                  exprasmlist.concat(taicpu.op_reg_const(A_CMPWI,tmpreg,0));                  // the two instructions above should be folded together by                  // the peepholeoptimizer                end              else                begin                  if right.location.loc = LOC_CONSTANT then                    begin                      cg.a_load_const_reg(exprasmlist,OS_INT,                        right.location.value,tmpreg);                      exprasmlist.concat(taicpu.op_reg_reg_reg(A_ANDC_,tmpreg,                        tmpreg,left.location.register));                    end                  else                    exprasmlist.concat(taicpu.op_reg_reg_reg(A_ANDC_,tmpreg,                      right.location.register,left.location.register));                end;              location.resflags.cr := RS_CR0;              location.resflags.flag := F_EQ;              opdone := true;            end;          else            internalerror(2002072701);        end;        if not opdone then          begin            // these are all commutative operations            if (left.location.loc = LOC_CONSTANT) then              swapleftright;            if (right.location.loc = LOC_CONSTANT) then              cg.a_op_const_reg_reg(exprasmlist,cgop,OS_INT,                right.location.value,left.location.register,                location.register)            else              cg.a_op_reg_reg_reg(exprasmlist,cgop,OS_INT,                right.location.register,left.location.register,                location.register);          end;      end;{*****************************************************************************                                Add64bit*****************************************************************************}    procedure tppcaddnode.second_add64bit;      var        op         : TOpCG;        op1,op2    : TAsmOp;        cmpop,        unsigned   : boolean;      procedure emit_cmp64_hi;        var          oldleft, oldright: tlocation;        begin          // put the high part of the location in the low part          location_copy(oldleft,left.location);          location_copy(oldright,right.location);          if left.location.loc = LOC_CONSTANT then            left.location.value64 := left.location.value64 shr 32          else            left.location.register64.reglo := left.location.register64.reghi;          if right.location.loc = LOC_CONSTANT then            right.location.value64 := right.location.value64 shr 32          else            right.location.register64.reglo := right.location.register64.reghi;          // and call the normal emit_compare          emit_compare(unsigned);          location_copy(left.location,oldleft);          location_copy(right.location,oldright);        end;      procedure emit_cmp64_lo;        begin          emit_compare(true);        end;      procedure firstjmp64bitcmp;        var          oldnodetype: tnodetype;        begin{$ifdef OLDREGVARS}           load_all_regvars(exprasmlist);{$endif OLDREGVARS}           { the jump the sequence is a little bit hairy }           case nodetype of              ltn,gtn:                begin                   cg.a_jmp_flags(exprasmlist,getresflags,truelabel);                   { cheat a little bit for the negative test }                   toggleflag(nf_swaped);                   cg.a_jmp_flags(exprasmlist,getresflags,falselabel);                   toggleflag(nf_swaped);                end;              lten,gten:                begin                   oldnodetype:=nodetype;                   if nodetype=lten then                     nodetype:=ltn                   else                     nodetype:=gtn;                   cg.a_jmp_flags(exprasmlist,getresflags,truelabel);                   { cheat for the negative test }                   if nodetype=ltn then                     nodetype:=gtn                   else                     nodetype:=ltn;                   cg.a_jmp_flags(exprasmlist,getresflags,falselabel);                   nodetype:=oldnodetype;                end;              equaln:                begin                  nodetype := unequaln;                  cg.a_jmp_flags(exprasmlist,getresflags,falselabel);                  nodetype := equaln;                end;              unequaln:                begin                  cg.a_jmp_flags(exprasmlist,getresflags,truelabel);                end;           end;        end;      procedure secondjmp64bitcmp;        begin           { the jump the sequence is a little bit hairy }           case nodetype of              ltn,gtn,lten,gten:                begin                   { the comparison of the low dword always has }                   { to be always unsigned!                     }                   cg.a_jmp_flags(exprasmlist,getresflags,truelabel);                   cg.a_jmp_always(exprasmlist,falselabel);                end;              equaln:                begin                   nodetype := unequaln;                   cg.a_jmp_flags(exprasmlist,getresflags,falselabel);                   cg.a_jmp_always(exprasmlist,truelabel);                   nodetype := equaln;                end;              unequaln:                begin                   cg.a_jmp_flags(exprasmlist,getresflags,truelabel);                   cg.a_jmp_always(exprasmlist,falselabel);                end;           end;        end;    var      tempreg64: tregister64;      begin        firstcomplex(self);        pass_left_and_right;        cmpop:=false;        unsigned:=((left.resulttype.def.deftype=orddef) and                   (torddef(left.resulttype.def).typ=u64bit)) or                  ((right.resulttype.def.deftype=orddef) and                   (torddef(right.resulttype.def).typ=u64bit));        case nodetype of          addn :            begin              op:=OP_ADD;            end;          subn :            begin              op:=OP_SUB;              if (nf_swaped in flags) then                swapleftright;            end;          ltn,lten,          gtn,gten,          equaln,unequaln:            begin              op:=OP_NONE;              cmpop:=true;            end;          xorn:            op:=OP_XOR;          orn:            op:=OP_OR;          andn:            op:=OP_AND;          muln:            begin              { should be handled in pass_1 (JM) }              internalerror(200109051);            end;          else            internalerror(2002072705);        end;        if not cmpop then          location_reset(location,LOC_REGISTER,def_cgsize(resulttype.def));        load_left_right(cmpop,(cs_check_overflow in aktlocalswitches) and            (nodetype in [addn,subn]));        if not(cs_check_overflow in aktlocalswitches) or           not(nodetype in [addn,subn]) then          begin            case nodetype of              ltn,lten,              gtn,gten:                begin                  emit_cmp64_hi;                  firstjmp64bitcmp;                  emit_cmp64_lo;                  secondjmp64bitcmp;                end;              equaln,unequaln:                begin                  // instead of doing a complicated compare, do                  // (left.hi xor right.hi) or (left.lo xor right.lo)                  // (somewhate optimized so that no superfluous 'mr's are                  //  generated)                  if (left.location.loc = LOC_CONSTANT) then                    swapleftright;                  if (right.location.loc = LOC_CONSTANT) then                    begin                      if left.location.loc = LOC_REGISTER then                        begin                          tempreg64.reglo := left.location.register64.reglo;                          tempreg64.reghi := left.location.register64.reghi;                        end                      else                        begin                          if (aint(right.location.value64) <> 0) then                            tempreg64.reglo := cg.getintregister(exprasmlist,OS_32)                          else                            tempreg64.reglo := left.location.register64.reglo;                          if ((right.location.value64 shr 32) <> 0) then                            tempreg64.reghi := cg.getintregister(exprasmlist,OS_32)                          else                            tempreg64.reghi := left.location.register64.reghi;                        end;                      if (aint(right.location.value64) <> 0) then                        { negative values can be handled using SUB, }                        { positive values < 65535 using XOR.        }                        if (longint(right.location.value64) >= -32767) and                           (longint(right.location.value64) < 0) then                          cg.a_op_const_reg_reg(exprasmlist,OP_SUB,OS_INT,                            aint(right.location.value64),                            left.location.register64.reglo,tempreg64.reglo)                        else                          cg.a_op_const_reg_reg(exprasmlist,OP_XOR,OS_INT,                            aint(right.location.value64),                            left.location.register64.reglo,tempreg64.reglo);                      if ((right.location.value64 shr 32) <> 0) then                        if (longint(right.location.value64 shr 32) >= -32767) and                           (longint(right.location.value64 shr 32) < 0) then                          cg.a_op_const_reg_reg(exprasmlist,OP_SUB,OS_INT,                            aint(right.location.value64 shr 32),                            left.location.register64.reghi,tempreg64.reghi)                        else                          cg.a_op_const_reg_reg(exprasmlist,OP_XOR,OS_INT,                            aint(right.location.value64 shr 32),                            left.location.register64.reghi,tempreg64.reghi);                    end                  else                    begin                       tempreg64.reglo := cg.getintregister(exprasmlist,OS_INT);                       tempreg64.reghi := cg.getintregister(exprasmlist,OS_INT);                       cg64.a_op64_reg_reg_reg(exprasmlist,OP_XOR,location.size,                         left.location.register64,right.location.register64,                         tempreg64);                    end;                  cg.a_reg_alloc(exprasmlist,NR_R0);                  exprasmlist.concat(taicpu.op_reg_reg_reg(A_OR_,NR_R0,                    tempreg64.reglo,tempreg64.reghi));                  cg.a_reg_dealloc(exprasmlist,NR_R0);                  location_reset(location,LOC_FLAGS,OS_NO);                  location.resflags := getresflags;                end;              xorn,orn,andn,addn:                begin                  if (location.register64.reglo = NR_NO) then                    begin                      location.register64.reglo := cg.getintregister(exprasmlist,OS_INT);                      location.register64.reghi := cg.getintregister(exprasmlist,OS_INT);                    end;                  if (left.location.loc = LOC_CONSTANT) then                    swapleftright;                  if (right.location.loc = LOC_CONSTANT) then                    cg64.a_op64_const_reg_reg(exprasmlist,op,location.size,right.location.value64,                      left.location.register64,location.register64)                  else                    cg64.a_op64_reg_reg_reg(exprasmlist,op,location.size,right.location.register64,                      left.location.register64,location.register64);                end;              subn:                begin                  if left.location.loc <> LOC_CONSTANT then                    begin                      if (location.register64.reglo = NR_NO) then                        begin                         location.register64.reglo := cg.getintregister(exprasmlist,OS_INT);                         location.register64.reghi := cg.getintregister(exprasmlist,OS_INT);                      end;                      if right.location.loc <> LOC_CONSTANT then                        // reg64 - reg64                        cg64.a_op64_reg_reg_reg(exprasmlist,OP_SUB,location.size,                          right.location.register64,left.location.register64,                          location.register64)                      else                        // reg64 - const64                        cg64.a_op64_const_reg_reg(exprasmlist,OP_SUB,location.size,                          right.location.value64,left.location.register64,                          location.register64)                    end                  else if ((left.location.value64 shr 32) = 0) then                    begin                      if (location.register64.reglo = NR_NO) then                        begin                         location.register64.reglo := cg.getintregister(exprasmlist,OS_INT);                         location.register64.reghi := cg.getintregister(exprasmlist,OS_INT);                      end;                      if (int64(left.location.value64) >= low(smallint)) and                         (int64(left.location.value64) <= high(smallint)) then                        begin                          // consts16 - reg64                          exprasmlist.concat(taicpu.op_reg_reg_const(A_SUBFIC,                            location.register64.reglo,right.location.register64.reglo,                            left.location.value));                        end                      else                        begin                          // const32 - reg64                          location_force_reg(exprasmlist,left.location,                            OS_32,true);                          exprasmlist.concat(taicpu.op_reg_reg_reg(A_SUBC,                            location.register64.reglo,left.location.register64.reglo,                            right.location.register64.reglo));                        end;                      exprasmlist.concat(taicpu.op_reg_reg(A_SUBFZE,                        location.register64.reghi,right.location.register64.reghi));                    end                  else if (aint(left.location.value64) = 0) then                    begin                      // (const32 shl 32) - reg64                      if (location.register64.reglo = NR_NO) then                        begin                         location.register64.reglo := cg.getintregister(exprasmlist,OS_INT);                         location.register64.reghi := cg.getintregister(exprasmlist,OS_INT);                      end;                      exprasmlist.concat(taicpu.op_reg_reg_const(A_SUBFIC,                        location.register64.reglo,right.location.register64.reglo,0));                      left.location.value64 := left.location.value64 shr 32;                      location_force_reg(exprasmlist,left.location,OS_32,true);                      exprasmlist.concat(taicpu.op_reg_reg_reg(A_SUBFE,                        location.register64.reghi,right.location.register64.reghi,                        left.location.register));                    end                  else                    begin                      // const64 - reg64                      location_force_reg(exprasmlist,left.location,                        def_cgsize(left.resulttype.def),false);                      if (left.location.loc = LOC_REGISTER) then                        location.register64 := left.location.register64                      else if (location.register64.reglo = NR_NO) then                        begin                         location.register64.reglo := cg.getintregister(exprasmlist,OS_INT);                         location.register64.reghi := cg.getintregister(exprasmlist,OS_INT);                        end;                      cg64.a_op64_reg_reg_reg(exprasmlist,OP_SUB,location.size,                        right.location.register64,left.location.register64,                        location.register64);                     end;                end;              else                internalerror(2002072803);            end;          end        else          begin            if is_signed(resulttype.def) then              begin                case nodetype of                  addn:                    begin                      op1 := A_ADDC;                      op2 := A_ADDEO;                    end;                  subn:                    begin                      op1 := A_SUBC;                      op2 := A_SUBFEO;                    end;                  else                    internalerror(2002072806);                end              end            else              begin                case nodetype of                  addn:                    begin                      op1 := A_ADDC;                      op2 := A_ADDE;                    end;                  subn:                    begin                      op1 := A_SUBC;                      op2 := A_SUBFE;                    end;                end;              end;            exprasmlist.concat(taicpu.op_reg_reg_reg(op1,location.register64.reglo,              left.location.register64.reglo,right.location.register64.reglo));            exprasmlist.concat(taicpu.op_reg_reg_reg(op2,location.register64.reghi,              right.location.register64.reghi,left.location.register64.reghi));            if not(is_signed(resulttype.def)) then              if nodetype = addn then                exprasmlist.concat(taicpu.op_reg_reg(A_CMPLW,location.register64.reghi,left.location.register64.reghi))              else                exprasmlist.concat(taicpu.op_reg_reg(A_CMPLW,left.location.register64.reghi,location.register64.reghi));            cg.g_overflowcheck(exprasmlist,location,resulttype.def);          end;        { set result location }        { (emit_compare sets it to LOC_FLAGS for compares, so set the }        {  real location only now) (JM)                               }        if cmpop and           not(nodetype in [equaln,unequaln]) then          location_reset(location,LOC_JUMP,OS_NO);      end;{*****************************************************************************                                AddMMX*****************************************************************************}{$ifdef SUPPORT_MMX}    procedure ti386addnode.second_addmmx;      var        op         : TAsmOp;        cmpop      : boolean;        mmxbase    : tmmxtype;        hregister  : tregister;      begin        pass_left_and_right;        cmpop:=false;        mmxbase:=mmx_type(left.resulttype.def);        case nodetype of          addn :            begin              if (cs_mmx_saturation in aktlocalswitches) then                begin                   case mmxbase of                      mmxs8bit:                        op:=A_PADDSB;                      mmxu8bit:                        op:=A_PADDUSB;                      mmxs16bit,mmxfixed16:                        op:=A_PADDSB;                      mmxu16bit:                        op:=A_PADDUSW;                   end;                end              else                begin                   case mmxbase of                      mmxs8bit,mmxu8bit:                        op:=A_PADDB;                      mmxs16bit,mmxu16bit,mmxfixed16:                        op:=A_PADDW;                      mmxs32bit,mmxu32bit:                        op:=A_PADDD;                   end;                end;            end;          muln :            begin               case mmxbase of                  mmxs16bit,mmxu16bit:                    op:=A_PMULLW;                  mmxfixed16:                    op:=A_PMULHW;               end;            end;          subn :            begin              if (cs_mmx_saturation in aktlocalswitches) then                begin                   case mmxbase of                      mmxs8bit:                        op:=A_PSUBSB;                      mmxu8bit:                        op:=A_PSUBUSB;                      mmxs16bit,mmxfixed16:                        op:=A_PSUBSB;                      mmxu16bit:                        op:=A_PSUBUSW;                   end;                end              else                begin                   case mmxbase of                      mmxs8bit,mmxu8bit:                        op:=A_PSUBB;                      mmxs16bit,mmxu16bit,mmxfixed16:                        op:=A_PSUBW;                      mmxs32bit,mmxu32bit:                        op:=A_PSUBD;                   end;                end;            end;          xorn:            op:=A_PXOR;          orn:            op:=A_POR;          andn:            op:=A_PAND;          else            internalerror(200403183);        end;        { left and right no register?  }        { then one must be demanded    }        if (left.location.loc<>LOC_MMXREGISTER) then         begin           if (right.location.loc=LOC_MMXREGISTER) then            begin              location_swap(left.location,right.location);              toggleflag(nf_swaped);            end           else            begin              { register variable ? }              if (left.location.loc=LOC_CMMXREGISTER) then               begin                 hregister:=rg.getregistermm(exprasmlist);                 emit_reg_reg(A_MOVQ,S_NO,left.location.register,hregister);               end              else               begin                 if not(left.location.loc in [LOC_REFERENCE,LOC_CREFERENCE]) then                  internalerror(200203245);                 location_release(exprasmlist,left.location);                 hregister:=rg.getregistermm(exprasmlist);                 emit_ref_reg(A_MOVQ,S_NO,left.location.reference,hregister);               end;              location_reset(left.location,LOC_MMXREGISTER,OS_NO);              left.location.register:=hregister;            end;         end;        { at this point, left.location.loc should be LOC_MMXREGISTER }        if right.location.loc<>LOC_MMXREGISTER then         begin           if (nodetype=subn) and (nf_swaped in flags) then            begin              if right.location.loc=LOC_CMMXREGISTER then               begin                 emit_reg_reg(A_MOVQ,S_NO,right.location.register,R_MM7);                 emit_reg_reg(op,S_NO,left.location.register,R_MM7);                 emit_reg_reg(A_MOVQ,S_NO,R_MM7,left.location.register);               end              else               begin                 if not(left.location.loc in [LOC_REFERENCE,LOC_CREFERENCE]) then                  internalerror(200203247);                 emit_ref_reg(A_MOVQ,S_NO,right.location.reference,R_MM7);                 emit_reg_reg(op,S_NO,left.location.register,R_MM7);                 emit_reg_reg(A_MOVQ,S_NO,R_MM7,left.location.register);                 location_release(exprasmlist,right.location);               end;            end           else            begin              if (right.location.loc=LOC_CMMXREGISTER) then               begin                 emit_reg_reg(op,S_NO,right.location.register,left.location.register);               end              else               begin                 if not(right.location.loc in [LOC_REFERENCE,LOC_CREFERENCE]) then                  internalerror(200203246);                 emit_ref_reg(op,S_NO,right.location.reference,left.location.register);                 location_release(exprasmlist,right.location);               end;            end;          end        else          begin            { right.location=LOC_MMXREGISTER }            if (nodetype=subn) and (nf_swaped in flags) then             begin               emit_reg_reg(op,S_NO,left.location.register,right.location.register);               location_swap(left.location,right.location);               toggleflag(nf_swaped);             end            else             begin               emit_reg_reg(op,S_NO,right.location.register,left.location.register);             end;          end;        location_freetemp(exprasmlist,right.location);        location_release(exprasmlist,right.location);        if cmpop then         begin           location_freetemp(exprasmlist,left.location);           location_release(exprasmlist,left.location);         end;        set_result_location(cmpop,true);      end;{$endif SUPPORT_MMX}{*****************************************************************************                                pass_2*****************************************************************************}    procedure tppcaddnode.pass_2;    { is also being used for xor, and "mul", "sub, or and comparative }    { operators                                                }      var         cgop       : topcg;         op         : tasmop;         tmpreg     : tregister;         hl         : tasmlabel;         cmpop      : boolean;         { true, if unsigned types are compared }         unsigned : boolean;      begin         { to make it more readable, string and set (not smallset!) have their           own procedures }         case left.resulttype.def.deftype of           orddef :             begin               { handling boolean expressions }               if is_boolean(left.resulttype.def) and                  is_boolean(right.resulttype.def) then                 begin                   second_addboolean;                   exit;                 end               { 64bit operations }               else if is_64bit(left.resulttype.def) then                 begin                   second_add64bit;                   exit;                 end;             end;           stringdef :             begin               internalerror(2002072402);               exit;             end;           setdef :             begin               { normalsets are already handled in pass1 }               if (tsetdef(left.resulttype.def).settype<>smallset) then                internalerror(200109041);               second_addsmallset;               exit;             end;           arraydef :             begin{$ifdef SUPPORT_MMX}               if is_mmx_able_array(left.resulttype.def) then                begin                  second_addmmx;                  exit;                end;{$endif SUPPORT_MMX}             end;           floatdef :             begin               second_addfloat;               exit;             end;         end;         { defaults }         cmpop:=nodetype in [ltn,lten,gtn,gten,equaln,unequaln];         unsigned:=not(is_signed(left.resulttype.def)) or                   not(is_signed(right.resulttype.def));         pass_left_and_right;         { Convert flags to register first }         { can any of these things be in the flags actually?? (JM) }         if (left.location.loc = LOC_FLAGS) or            (right.location.loc = LOC_FLAGS) then           internalerror(2002072602);         { set result location }         if not cmpop then           location_reset(location,LOC_REGISTER,def_cgsize(resulttype.def))          else           location_reset(location,LOC_FLAGS,OS_NO);         load_left_right(cmpop, (cs_check_overflow in aktlocalswitches) and            (nodetype in [addn,subn,muln]));         if (location.register = NR_NO) and            not(cmpop) then           location.register := cg.getintregister(exprasmlist,OS_INT);         if not(cs_check_overflow in aktlocalswitches) or            (cmpop) or            (nodetype in [orn,andn,xorn]) then           begin             case nodetype of               addn, muln, xorn, orn, andn:                 begin                   case nodetype of                     addn:                       cgop := OP_ADD;                     muln:                       if unsigned then                         cgop := OP_MUL                       else                         cgop := OP_IMUL;                     xorn:                       cgop := OP_XOR;                     orn:                       cgop := OP_OR;                     andn:                       cgop := OP_AND;                   end;                   if (left.location.loc = LOC_CONSTANT) then                     swapleftright;                   if (right.location.loc <> LOC_CONSTANT) then                     cg.a_op_reg_reg_reg(exprasmlist,cgop,OS_INT,                       left.location.register,right.location.register,                       location.register)                   else                     cg.a_op_const_reg_reg(exprasmlist,cgop,OS_INT,                       right.location.value,left.location.register,                     location.register);                 end;               subn:                 begin                   if (nf_swaped in flags) then                     swapleftright;                   if left.location.loc <> LOC_CONSTANT then                     if right.location.loc <> LOC_CONSTANT then                       cg.a_op_reg_reg_reg(exprasmlist,OP_SUB,OS_INT,                         right.location.register,left.location.register,                         location.register)                     else                       cg.a_op_const_reg_reg(exprasmlist,OP_SUB,OS_INT,                         right.location.value,left.location.register,                         location.register)                   else                     if (longint(left.location.value) >= low(smallint)) and                        (longint(left.location.value) <= high(smallint)) then                       begin                         exprasmlist.concat(taicpu.op_reg_reg_const(A_SUBFIC,                           location.register,right.location.register,                           longint(left.location.value)));                       end                     else                       begin                         tmpreg := cg.getintregister(exprasmlist,OS_INT);                         cg.a_load_const_reg(exprasmlist,OS_INT,                           left.location.value,tmpreg);                         cg.a_op_reg_reg_reg(exprasmlist,OP_SUB,OS_INT,                           right.location.register,tmpreg,location.register);                       end;                 end;               ltn,lten,gtn,gten,equaln,unequaln :                 begin                   emit_compare(unsigned);                 end;             end;           end         else           // overflow checking is on and we have an addn, subn or muln           begin             if is_signed(resulttype.def) then               begin                 case nodetype of                   addn:                     op := A_ADDO;                   subn:                     begin                       op := A_SUBO;                       if (nf_swaped in flags) then                         swapleftright;                     end;                   muln:                     op := A_MULLWO;                   else                     internalerror(2002072601);                 end;                 exprasmlist.concat(taicpu.op_reg_reg_reg(op,location.register,                   left.location.register,right.location.register));                 cg.g_overflowcheck(exprasmlist,location,resulttype.def);              end             else              begin                case nodetype of                  addn:                    begin                      exprasmlist.concat(taicpu.op_reg_reg_reg(A_ADD,location.register,                        left.location.register,right.location.register));                      exprasmlist.concat(taicpu.op_reg_reg(A_CMPLW,location.register,left.location.register));                      cg.g_overflowcheck(exprasmlist,location,resulttype.def);                    end;                  subn:                    begin                      if nf_swaped in flags then                        swapleftright;                      exprasmlist.concat(taicpu.op_reg_reg_reg(A_SUB,location.register,                        left.location.register,right.location.register));                      exprasmlist.concat(taicpu.op_reg_reg(A_CMPLW,left.location.register,location.register));                      cg.g_overflowcheck(exprasmlist,location,resulttype.def);                    end;                  muln:                    begin                      { calculate the upper 32 bits of the product, = 0 if no overflow }                      cg.a_reg_alloc(exprasmlist,NR_R0);                      exprasmlist.concat(taicpu.op_reg_reg_reg(A_MULHWU_,NR_R0,                        left.location.register,right.location.register));                      cg.a_reg_dealloc(exprasmlist,NR_R0);                      { calculate the real result }                      exprasmlist.concat(taicpu.op_reg_reg_reg(A_MULLW,location.register,                        left.location.register,right.location.register));                      { g_overflowcheck generates a OC_AE instead of OC_EQ :/ }                      objectlibrary.getlabel(hl);                      tcgppc(cg).a_jmp_cond(exprasmlist,OC_EQ,hl);                      cg.a_call_name(exprasmlist,'FPC_OVERFLOW');                      cg.a_label(exprasmlist,hl);                    end;                end;              end;           end;      end;begin   caddnode:=tppcaddnode;end.
 |