| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289 | {    Copyright (c) 2008 by Florian Klaempfl    Code generation for add nodes on the AVR    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 navradd;{$i fpcdefs.inc}interface    uses       node,ncgadd,cpubase;    type       TAVRAddNode = class(tcgaddnode)       private         function  GetResFlags(unsigned:Boolean):TResFlags;       protected         function pass_1 : tnode;override;         procedure second_cmpordinal;override;         procedure second_cmpsmallset;override;         procedure second_cmp64bit;override;         procedure second_cmp;       end;  implementation    uses      globtype,systems,      cutils,verbose,globals,      symconst,symdef,paramgr,      aasmbase,aasmtai,aasmdata,aasmcpu,defutil,htypechk,      cgbase,cgutils,cgcpu,      cpuinfo,pass_1,pass_2,regvars,procinfo,      cpupara,      ncon,nset,nadd,      ncgutil,tgobj,rgobj,rgcpu,cgobj,cg64f32,      hlcgobj;{*****************************************************************************                               TAVRAddNode*****************************************************************************}    function tavraddnode.GetResFlags(unsigned:Boolean):TResFlags;      begin        case NodeType of          equaln:            GetResFlags:=F_EQ;          unequaln:            GetResFlags:=F_NE;          else            if not(unsigned) then              begin                if nf_swapped in flags then                  case NodeType of                    ltn:                      GetResFlags:=F_NotPossible;                    lten:                      GetResFlags:=F_GE;                    gtn:                      GetResFlags:=F_LT;                    gten:                      GetResFlags:=F_NotPossible;                    else                      internalerror(2014082020);                  end                else                  case NodeType of                    ltn:                      GetResFlags:=F_LT;                    lten:                      GetResFlags:=F_NotPossible;                    gtn:                      GetResFlags:=F_NotPossible;                    gten:                      GetResFlags:=F_GE;                    else                      internalerror(2014082021);                  end;              end            else              begin                if nf_swapped in Flags then                  case NodeType of                    ltn:                      GetResFlags:=F_NotPossible;                    lten:                      GetResFlags:=F_SH;                    gtn:                      GetResFlags:=F_LO;                    gten:                      GetResFlags:=F_NotPossible;                    else                      internalerror(2014082022);                  end                else                  case NodeType of                    ltn:                      GetResFlags:=F_LO;                    lten:                      GetResFlags:=F_NotPossible;                    gtn:                      GetResFlags:=F_NotPossible;                    gten:                      GetResFlags:=F_SH;                    else                      internalerror(2014082023);                  end;              end;        end;      end;    procedure tavraddnode.second_cmpsmallset;      procedure gencmp(tmpreg1,tmpreg2 : tregister);        var          i : byte;        begin          current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg(A_CP,tmpreg1,tmpreg2));          for i:=2 to tcgsize2size[left.location.size] do            begin              tmpreg1:=GetNextReg(tmpreg1);              tmpreg2:=GetNextReg(tmpreg2);              current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg(A_CPC,tmpreg1,tmpreg2));            end;        end;      var        tmpreg : tregister;      begin        pass_left_right;        location_reset(location,LOC_FLAGS,OS_NO);        force_reg_left_right(false,false);        case nodetype of          equaln:            begin              gencmp(left.location.register,right.location.register);              location.resflags:=F_EQ;            end;          unequaln:            begin              gencmp(left.location.register,right.location.register);              location.resflags:=F_NE;            end;          lten,          gten:            begin              if (not(nf_swapped in flags) and                  (nodetype = lten)) or                 ((nf_swapped in flags) and                  (nodetype = gten)) then                swapleftright;              tmpreg:=cg.getintregister(current_asmdata.CurrAsmList,location.size);              cg.a_op_reg_reg_reg(current_asmdata.CurrAsmList,OP_AND,location.size,                left.location.register,right.location.register,tmpreg);              gencmp(tmpreg,right.location.register);              location.resflags:=F_EQ;            end;          else            internalerror(2004012401);        end;      end;    procedure tavraddnode.second_cmp;      var        unsigned : boolean;        tmpreg1,tmpreg2 : tregister;        i : longint;      begin        pass_left_right;        force_reg_left_right(true,true);        unsigned:=not(is_signed(left.resultdef)) or                  not(is_signed(right.resultdef));        if getresflags(unsigned)=F_NotPossible then          begin            swapleftright;            { if we have to swap back and left is a constant, force it to a register because we cannot generate              the needed code using a constant }            if left.location.loc=LOC_CONSTANT then              hlcg.location_force_reg(current_asmdata.CurrAsmList,left.location,left.resultdef,left.resultdef,false);          end;        if right.location.loc=LOC_CONSTANT then          begin            { decrease register pressure on registers >= r16 }            if (right.location.value and $ff)=0 then              current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg(A_CP,left.location.register,NR_R1))            else              current_asmdata.CurrAsmList.concat(taicpu.op_reg_const(A_CPI,left.location.register,right.location.value and $ff))          end        else          current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg(A_CP,left.location.register,right.location.register));        tmpreg1:=left.location.register;        tmpreg2:=right.location.register;        for i:=2 to tcgsize2size[left.location.size] do          begin            if i=5 then              begin                tmpreg1:=left.location.registerhi;                if right.location.loc<>LOC_CONSTANT then                  tmpreg2:=right.location.registerhi;              end            else              begin                tmpreg1:=GetNextReg(tmpreg1);                if right.location.loc<>LOC_CONSTANT then                  tmpreg2:=GetNextReg(tmpreg2);              end;            if right.location.loc=LOC_CONSTANT then              begin                tmpreg2:=cg.getintregister(current_asmdata.CurrAsmList,OS_8);                cg.a_load_const_reg(current_asmdata.CurrAsmList,OS_8,(right.location.value64 shr ((i-1)*8)) and $ff,tmpreg2);                current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg(A_CPC,tmpreg1,tmpreg2));              end            else              begin                current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg(A_CPC,tmpreg1,tmpreg2));              end;          end;        location_reset(location,LOC_FLAGS,OS_NO);        location.resflags:=getresflags(unsigned);      end;    procedure tavraddnode.second_cmp64bit;      begin        second_cmp;      end;    function tavraddnode.pass_1 : tnode;      begin        result:=inherited pass_1;{$ifdef dummy}        if not(assigned(result)) then          begin            unsigned:=not(is_signed(left.resultdef)) or              not(is_signed(right.resultdef));            if is_64bit(left.resultdef) and              ((nodetype in [equaln,unequaln]) or               (unsigned and (nodetype in [ltn,lten,gtn,gten]))              ) then              expectloc:=LOC_FLAGS;          end;        { handling boolean expressions }        if not(assigned(result)) and           (             not(is_boolean(left.resultdef)) or             not(is_boolean(right.resultdef)) or             is_dynamic_array(left.resultdef)           ) then          expectloc:=LOC_FLAGS;{$endif dummy}      end;    procedure tavraddnode.second_cmpordinal;      begin        second_cmp;      end;begin  caddnode:=tavraddnode;end.
 |