| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605 | {    Copyright (c) 1998-2002 by Florian Klaempfl    Generate PowerPC assembler 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 nppcmat;{$i fpcdefs.inc}interface    uses      node,nmat;    type      tppcmoddivnode = class(tmoddivnode)         function pass_1: tnode;override;         procedure pass_generate_code;override;      end;      tppcshlshrnode = class(tshlshrnode)         procedure pass_generate_code;override;         { everything will be handled in pass_2 }         function first_shlshr64bitint: tnode; override;      end;      tppcunaryminusnode = class(tunaryminusnode)         procedure pass_generate_code;override;      end;      tppcnotnode = class(tnotnode)         procedure pass_generate_code;override;      end;implementation    uses      globtype,systems,constexp,      cutils,verbose,globals,      symconst,symdef,      aasmbase,aasmcpu,aasmtai,aasmdata,      defutil,      cgbase,cgutils,cgobj,hlcgobj,pass_2,      ncon,procinfo,      cpubase,      ncgutil,cgcpu;{*****************************************************************************                             TPPCMODDIVNODE*****************************************************************************}    function tppcmoddivnode.pass_1: tnode;      begin        result := inherited pass_1;        if not assigned(result) then          include(current_procinfo.flags,pi_do_call);      end;    procedure tppcmoddivnode.pass_generate_code;      const                    { signed   overflow }        divops: array[boolean, boolean] of tasmop =          ((A_DIVWU,A_DIVWU_),(A_DIVW,A_DIVWO_));        zerocond: tasmcond = (dirhint: DH_Plus; simple: true; cond:C_NE; cr: RS_CR1);      var         power  : longint;         op         : tasmop;         numerator,         divider,         resultreg  : tregister;         size       : Tcgsize;         hl : tasmlabel;         done: boolean;                  procedure genOrdConstNodeDiv;         const             negops : array[boolean] of tasmop = (A_NEG, A_NEGO);         var             divreg : tregister;         begin             if (tordconstnode(right).value = 0) then begin                 internalerror(2005061701);             end else if (tordconstnode(right).value = 1) then begin                cg.a_load_reg_reg(current_asmdata.CurrAsmList, OS_INT, OS_INT, numerator, resultreg);             end else if (tordconstnode(right).value = int64(-1)) then begin                // note: only in the signed case possible..., may overflow                current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg(negops[cs_check_overflow in current_settings.localswitches], resultreg, numerator));             end else if (ispowerof2(tordconstnode(right).value, power)) then begin                if (is_signed(right.resultdef)) then begin                    { From "The PowerPC Compiler Writer's Guide", pg. 52ff          }                    cg.a_op_const_reg_reg(current_asmdata.CurrAsmList, OP_SAR, OS_INT, power,                        numerator, resultreg);                    current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg(A_ADDZE, resultreg, resultreg));                end else begin                    cg.a_op_const_reg_reg(current_asmdata.CurrAsmList, OP_SHR, OS_INT, power, numerator, resultreg)                end;             end else begin                 cg.g_div_const_reg_reg(current_asmdata.CurrAsmList,def_cgsize(resultdef),                     tordconstnode(right).value.svalue,numerator,resultreg);             end;             done := true;         end;         procedure genOrdConstNodeMod;         var             modreg, maskreg, tempreg : tregister;         begin             if (tordconstnode(right).value = 0) then begin                 internalerror(2005061702);             end else if (abs(tordconstnode(right).value.svalue) = 1) then begin                // x mod +/-1 is always zero                cg.a_load_const_reg(current_asmdata.CurrAsmList, OS_INT, 0, resultreg);             end else if (ispowerof2(tordconstnode(right).value, power)) then begin                 if (is_signed(right.resultdef)) then begin                                          tempreg := cg.getintregister(current_asmdata.CurrAsmList, OS_INT);                     maskreg := cg.getintregister(current_asmdata.CurrAsmList, OS_INT);                     modreg := cg.getintregister(current_asmdata.CurrAsmList, OS_INT);                     cg.a_load_const_reg(current_asmdata.CurrAsmList, OS_INT, abs(tordconstnode(right).value.svalue)-1, modreg);                     cg.a_op_const_reg_reg(current_asmdata.CurrAsmList, OP_SAR, OS_INT, 31, numerator, maskreg);                     cg.a_op_reg_reg_reg(current_asmdata.CurrAsmList, OP_AND, OS_INT, numerator, modreg, tempreg);                     current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg_reg(A_ANDC, maskreg, maskreg, modreg));                     current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg_const(A_SUBFIC, modreg, tempreg, 0));                     current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg_reg(A_SUBFE, modreg, modreg, modreg));                     cg.a_op_reg_reg_reg(current_asmdata.CurrAsmList, OP_AND, OS_INT, modreg, maskreg, maskreg);                     cg.a_op_reg_reg_reg(current_asmdata.CurrAsmList, OP_OR, OS_INT, maskreg, tempreg, resultreg);                 end else begin                     cg.a_op_const_reg_reg(current_asmdata.CurrAsmList, OP_AND, OS_INT, tordconstnode(right).value.svalue-1, numerator, resultreg);                 end;             end else begin                 genOrdConstNodeDiv();                 cg.a_op_const_reg_reg(current_asmdata.CurrAsmList, OP_MUL, OS_INT, tordconstnode(right).value.svalue, resultreg, resultreg);                 cg.a_op_reg_reg_reg(current_asmdata.CurrAsmList, OP_SUB, OS_INT, resultreg, numerator, resultreg);             end;         end;               begin         secondpass(left);         secondpass(right);         location_copy(location,left.location);         { put numerator in register }         size:=def_cgsize(left.resultdef);         hlcg.location_force_reg(current_asmdata.CurrAsmList,left.location,           left.resultdef,left.resultdef,true);         location_copy(location,left.location);         numerator := location.register;         resultreg := location.register;         if (location.loc = LOC_CREGISTER) then begin           location.loc := LOC_REGISTER;           location.register := cg.getintregister(current_asmdata.CurrAsmList,size);           resultreg := location.register;         end else if (nodetype = modn) or (right.nodetype = ordconstn) then begin           // for a modulus op, and for const nodes we need the result register           // to be an extra register           resultreg := cg.getintregister(current_asmdata.CurrAsmList,size);         end;         done := false;         if (right.nodetype = ordconstn) then begin           if (nodetype = divn) then             genOrdConstNodeDiv           else             genOrdConstNodeMod;           done := true;         end;         if (not done) then begin             { load divider in a register if necessary }             hlcg.location_force_reg(current_asmdata.CurrAsmList,right.location,               right.resultdef,right.resultdef,true);             if (right.nodetype <> ordconstn) then               current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg_const(A_CMPWI,NR_CR1,                 right.location.register,0));             divider := right.location.register;             { needs overflow checking, (-maxlongint-1) div (-1) overflows! }             op := divops[is_signed(right.resultdef),                          cs_check_overflow in current_settings.localswitches];             current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg_reg(op,resultreg,numerator,               divider));           if (nodetype = modn) then             begin               current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg_reg(A_MULLW,resultreg,                 divider,resultreg));               current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg_reg(A_SUB,location.register,                 numerator,resultreg));               resultreg := location.register;             end;           end;        { set result location }        location.loc:=LOC_REGISTER;        location.register:=resultreg;        if right.nodetype <> ordconstn then          begin            current_asmdata.getjumplabel(hl);            current_asmdata.CurrAsmList.concat(taicpu.op_cond_sym(A_BC,zerocond,hl));            cg.a_call_name(current_asmdata.CurrAsmList,'FPC_DIVBYZERO',false);            cg.a_label(current_asmdata.CurrAsmList,hl);          end;        { unsigned division/module can only overflow in case of division by zero }        { (but checking this overflow flag is more convoluted than performing a  }        {  simple comparison with 0)                                             }        if is_signed(right.resultdef) then          cg.g_overflowcheck(current_asmdata.CurrAsmList,location,resultdef);      end;{*****************************************************************************                             TPPCSHLRSHRNODE*****************************************************************************}    function tppcshlshrnode.first_shlshr64bitint: tnode;      begin        result := nil;      end;    procedure tppcshlshrnode.pass_generate_code;      var         resultreg, hregister1,hregister2,         hreg64hi,hreg64lo : tregister;         op : topcg;         asmop1, asmop2: tasmop;         shiftval: aint;      begin         secondpass(left);         secondpass(right);         if is_64bitint(left.resultdef) then           begin             hlcg.location_force_reg(current_asmdata.CurrAsmList,left.location,               left.resultdef,left.resultdef,true);             location_copy(location,left.location);             hreg64hi := location.register64.reghi;             hreg64lo := location.register64.reglo;             if (location.loc = LOC_CREGISTER) then               begin                 location.loc := LOC_REGISTER;                 location.register64.reghi := cg.getintregister(current_asmdata.CurrAsmList,OS_32);                 location.register64.reglo := cg.getintregister(current_asmdata.CurrAsmList,OS_32);               end;             if (right.nodetype = ordconstn) then               begin                 shiftval := tordconstnode(right).value.svalue;                 shiftval := shiftval and 63;{              I think the statements below is much more correct instead of the hack above,              but then we fail tshlshr.pp :/                 if shiftval > 63 then                   begin                     cg.a_load_const_reg(current_asmdata.CurrAsmList,OS_32,0,location.register64.reglo);                     cg.a_load_const_reg(current_asmdata.CurrAsmList,OS_32,0,location.register64.reglo);                   end                 else }                 if shiftval = 0 then                   begin                     cg.a_load_reg_reg(current_asmdata.CurrAsmList,OS_32,OS_32,left.location.register64.reghi,location.register64.reghi);                     cg.a_load_reg_reg(current_asmdata.CurrAsmList,OS_32,OS_32,left.location.register64.reglo,location.register64.reglo);                   end                 else if shiftval > 31 then                   begin                     if nodetype = shln then                       begin                         cg.a_op_const_reg_reg(current_asmdata.CurrAsmList,OP_SHL,OS_32,                           shiftval and 31,hreg64lo,location.register64.reghi);                         cg.a_load_const_reg(current_asmdata.CurrAsmList,OS_32,0,location.register64.reglo);                       end                     else                       begin                         cg.a_op_const_reg_reg(current_asmdata.CurrAsmList,OP_SHR,OS_32,                           shiftval and 31,hreg64hi,location.register64.reglo);                         cg.a_load_const_reg(current_asmdata.CurrAsmList,OS_32,0,location.register64.reghi);                       end;                   end                 else                   begin                     if nodetype = shln then                       begin                         current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg_const_const_const(                           A_RLWINM,location.register64.reghi,hreg64hi,shiftval,                           0,31-shiftval));                         current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg_const_const_const(                           A_RLWIMI,location.register64.reghi,hreg64lo,shiftval,                           32-shiftval,31));                         current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg_const_const_const(                           A_RLWINM,location.register64.reglo,hreg64lo,shiftval,                           0,31-shiftval));                       end                     else                       begin                         current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg_const_const_const(                           A_RLWINM,location.register64.reglo,hreg64lo,32-shiftval,                           shiftval,31));                         current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg_const_const_const(                           A_RLWIMI,location.register64.reglo,hreg64hi,32-shiftval,                           0,shiftval-1));                         current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg_const_const_const(                           A_RLWINM,location.register64.reghi,hreg64hi,32-shiftval,                           shiftval,31));                       end;                   end;               end             else               { no constant shiftcount }               begin                 hlcg.location_force_reg(current_asmdata.CurrAsmList,right.location,right.resultdef,s32inttype,true);                 hregister1 := right.location.register;                 if nodetype = shln then                   begin                     asmop1 := A_SLW;                     asmop2 := A_SRW;                   end                 else                   begin                     asmop1 := A_SRW;                     asmop2 := A_SLW;                     resultreg := hreg64hi;                     hreg64hi := hreg64lo;                     hreg64lo := resultreg;                     resultreg := location.register64.reghi;                     location.register64.reghi := location.register64.reglo;                     location.register64.reglo := resultreg;                   end;                 cg.getcpuregister(current_asmdata.CurrAsmList,NR_R0);                 current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg_const(A_SUBFIC,                   NR_R0,hregister1,32));                 current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg_reg(asmop1,                   location.register64.reghi,hreg64hi,hregister1));                 current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg_reg(asmop2,                   NR_R0,hreg64lo,NR_R0));                 current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg_reg(A_OR,                   location.register64.reghi,location.register64.reghi,NR_R0));                 current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg_const(A_SUBI,                   NR_R0,hregister1,32));                 current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg_reg(asmop1,                   NR_R0,hreg64lo,NR_R0));                 current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg_reg(A_OR,                   location.register64.reghi,location.register64.reghi,NR_R0));                 current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg_reg(asmop1,                   location.register64.reglo,hreg64lo,hregister1));                 cg.ungetcpuregister(current_asmdata.CurrAsmList,NR_R0);                 if nodetype = shrn then                   begin                     resultreg := location.register64.reghi;                     location.register64.reghi := location.register64.reglo;                     location.register64.reglo := resultreg;                   end;               end           end         else           begin             { load left operators in a register }             hlcg.location_force_reg(current_asmdata.CurrAsmList,left.location,left.resultdef,left.resultdef,true);             location_copy(location,left.location);             resultreg := location.register;             hregister1 := location.register;             location.loc := LOC_REGISTER;             resultreg := cg.getintregister(current_asmdata.CurrAsmList,location.size);             location.register := resultreg;             { determine operator }             if nodetype=shln then               op:=OP_SHL             else               op:=OP_SHR;             { shifting by a constant directly coded: }             if (right.nodetype=ordconstn) then               cg.a_op_const_reg_reg(current_asmdata.CurrAsmList,op,location.size,                 tordconstnode(right).value.svalue and 31,hregister1,resultreg)             else               begin                 { load shift count in a register if necessary }                 hlcg.location_force_reg(current_asmdata.CurrAsmList,right.location,right.resultdef,right.resultdef,true);                 hregister2 := right.location.register;                 cg.a_op_reg_reg_reg(current_asmdata.CurrAsmList,op,location.size,hregister2,                  hregister1,resultreg);               end;           end;      end;{*****************************************************************************                          TPPCUNARYMINUSNODE*****************************************************************************}    procedure tppcunaryminusnode.pass_generate_code;      var        src1: tregister;        op: tasmop;      begin         src1:=NR_NO;         secondpass(left);         if is_64bit(left.resultdef) then           begin             hlcg.location_force_reg(current_asmdata.CurrAsmList,left.location,left.resultdef,left.resultdef,true);             location_copy(location,left.location);             if (location.loc = LOC_CREGISTER) then               begin                 location.register64.reglo := cg.getintregister(current_asmdata.CurrAsmList,OS_INT);                 location.register64.reghi := cg.getintregister(current_asmdata.CurrAsmList,OS_INT);                 location.loc := LOC_REGISTER;               end;             current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg_const(A_SUBFIC,               location.register64.reglo,left.location.register64.reglo,0));             if not(cs_check_overflow in current_settings.localswitches) then               current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg(A_SUBFZE,                 location.register64.reghi,left.location.register64.reghi))             else               current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg(A_SUBFZEO_,                 location.register64.reghi,left.location.register64.reghi));           end         else           begin              if left.location.loc in [LOC_SUBSETREG,LOC_CSUBSETREG,LOC_SUBSETREF,LOC_CSUBSETREF] then                hlcg.location_force_reg(current_asmdata.CurrAsmList,left.location,left.resultdef,left.resultdef,true);              location_copy(location,left.location);              location.loc:=LOC_REGISTER;              case left.location.loc of                LOC_FPUREGISTER, LOC_REGISTER:                  begin                    src1 := left.location.register;                    location.register := src1;                  end;                LOC_CFPUREGISTER, LOC_CREGISTER:                  begin                     src1 := left.location.register;                     if left.location.loc = LOC_CREGISTER then                       location.register := cg.getintregister(current_asmdata.CurrAsmList,OS_INT)                     else                       location.register := cg.getfpuregister(current_asmdata.CurrAsmList,location.size);                  end;                LOC_REFERENCE,LOC_CREFERENCE:                  begin                     if (left.resultdef.typ=floatdef) then                       begin                          src1 := cg.getfpuregister(current_asmdata.CurrAsmList,def_cgsize(left.resultdef));                          location.register := src1;                          cg.a_loadfpu_ref_reg(current_asmdata.CurrAsmList,                            left.location.size,left.location.size,                            left.location.reference,src1);                       end                     else                       begin                          src1 := cg.getintregister(current_asmdata.CurrAsmList,OS_32);                          location.register:= src1;                          cg.a_load_ref_reg(current_asmdata.CurrAsmList,OS_32,OS_32,                            left.location.reference,src1);                       end;                  end;              end;              { choose appropriate operand }              if left.resultdef.typ <> floatdef then                begin                  if not(cs_check_overflow in current_settings.localswitches) then                    op := A_NEG                  else                    op := A_NEGO_;                  location.loc := LOC_REGISTER;                end              else                begin                  op := A_FNEG;                  location.loc := LOC_FPUREGISTER;                end;              { emit operation }              current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg(op,location.register,src1));           end;{ Here was a problem...     }{ Operand to be negated always     }{ seems to be converted to signed  }{ 32-bit before doing neg!!     }{ So this is useless...     }{ that's not true: -2^31 gives an overflow error if it is negated (FK) }        cg.g_overflowcheck(current_asmdata.CurrAsmList,location,resultdef);      end;{*****************************************************************************                               TPPCNOTNODE*****************************************************************************}    procedure tppcnotnode.pass_generate_code;      var         hl : tasmlabel;         tmpreg: tregister;      begin         if is_boolean(resultdef) then          begin            { if the location is LOC_JUMP, we do the secondpass after the              labels are allocated            }            if left.expectloc=LOC_JUMP then              begin                hl:=current_procinfo.CurrTrueLabel;                current_procinfo.CurrTrueLabel:=current_procinfo.CurrFalseLabel;                current_procinfo.CurrFalseLabel:=hl;                secondpass(left);                if left.location.loc<>LOC_JUMP then                  internalerror(2012081303);                maketojumpbool(current_asmdata.CurrAsmList,left,lr_load_regvars);                hl:=current_procinfo.CurrTrueLabel;                current_procinfo.CurrTrueLabel:=current_procinfo.CurrFalseLabel;                current_procinfo.CurrFalseLabel:=hl;                location.loc:=LOC_JUMP;              end            else              begin                secondpass(left);                case left.location.loc of                  LOC_FLAGS :                    begin                      location_copy(location,left.location);                      inverse_flags(location.resflags);                    end;                  LOC_REGISTER, LOC_CREGISTER,                  LOC_REFERENCE, LOC_CREFERENCE,                  LOC_SUBSETREG, LOC_CSUBSETREG,                   LOC_SUBSETREF, LOC_CSUBSETREF:                    begin                      hlcg.location_force_reg(current_asmdata.CurrAsmList,left.location,left.resultdef,left.resultdef,true);                      tmpreg:=left.location.register;{$ifndef cpu64bitalu}                      { 64 bit pascal booleans have their truth value stored in                        the lower 32 bits; with cbools, it can be anywhere }                      if (left.location.size in [OS_64,OS_S64]) and                         not is_pasbool(left.resultdef) then                        begin                          tmpreg:=cg.getintregister(current_asmdata.CurrAsmList,OS_32);                          cg.a_op_reg_reg_reg(current_asmdata.CurrAsmList,OP_OR,OS_32,left.location.register64.reglo,left.location.register64.reghi,tmpreg);                        end;{$endif not cpu64bitalu}                      current_asmdata.CurrAsmList.concat(taicpu.op_reg_const(A_CMPWI,tmpreg,0));                      location_reset(location,LOC_FLAGS,OS_NO);                      location.resflags.cr:=RS_CR0;                      location.resflags.flag:=F_EQ;                   end;                  else                    internalerror(2003042401);                end;              end;          end         else if is_64bitint(left.resultdef) then           begin             secondpass(left);             hlcg.location_force_reg(current_asmdata.CurrAsmList,left.location,left.resultdef,left.resultdef,false);             location_copy(location,left.location);             { perform the NOT operation }             current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg(A_NOT,location.register64.reghi,               location.register64.reghi));             current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg(A_NOT,location.register64.reglo,               location.register64.reglo));           end         else           begin             secondpass(left);             hlcg.location_force_reg(current_asmdata.CurrAsmList,left.location,left.resultdef,left.resultdef,true);             location_copy(location,left.location);             location.loc := LOC_REGISTER;             location.register := cg.getintregister(current_asmdata.CurrAsmList,OS_INT);             { perform the NOT operation }             cg.a_op_reg_reg(current_asmdata.CurrAsmList,OP_NOT,def_cgsize(resultdef),left.location.register,               location.register);          end;      end;begin   cmoddivnode:=tppcmoddivnode;   cshlshrnode:=tppcshlshrnode;   cunaryminusnode:=tppcunaryminusnode;   cnotnode:=tppcnotnode;end.
 |