| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352 | {    Copyright (c) 2000-2002 by Florian Klaempfl    Code generation for add nodes on the ARM    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 narmadd;{$i fpcdefs.inc}interface    uses       node,ncgadd,cpubase;    type       tarmaddnode = class(tcgaddnode)       private          function  GetResFlags(unsigned:Boolean):TResFlags;       protected          function pass_1 : tnode;override;          procedure second_addfloat;override;          procedure second_cmpfloat;override;          procedure second_cmpordinal;override;          procedure second_cmpsmallset;override;          procedure second_cmp64bit;override;       end;  implementation    uses      globtype,systems,      cutils,verbose,globals,      symconst,symdef,paramgr,      aasmbase,aasmtai,aasmcpu,defutil,htypechk,      cgbase,cgutils,cgcpu,      cpuinfo,pass_1,pass_2,regvars,      cpupara,      ncon,nset,nadd,      ncgutil,tgobj,rgobj,rgcpu,cgobj,cg64f32;{*****************************************************************************                               TSparcAddNode*****************************************************************************}    function tarmaddnode.GetResFlags(unsigned:Boolean):TResFlags;      begin        case NodeType of          equaln:            GetResFlags:=F_EQ;          unequaln:            GetResFlags:=F_NE;          else            if not(unsigned) then              begin                if nf_swaped in flags then                  case NodeType of                    ltn:                      GetResFlags:=F_GT;                    lten:                      GetResFlags:=F_GE;                    gtn:                      GetResFlags:=F_LT;                    gten:                      GetResFlags:=F_LE;                  end                else                  case NodeType of                    ltn:                      GetResFlags:=F_LT;                    lten:                      GetResFlags:=F_LE;                    gtn:                      GetResFlags:=F_GT;                    gten:                      GetResFlags:=F_GE;                  end;              end            else              begin                if nf_swaped in Flags then                  case NodeType of                    ltn:                      GetResFlags:=F_HI;                    lten:                      GetResFlags:=F_CS;                    gtn:                      GetResFlags:=F_CC;                    gten:                      GetResFlags:=F_LS;                  end                else                  case NodeType of                    ltn:                      GetResFlags:=F_CC;                    lten:                      GetResFlags:=F_LS;                    gtn:                      GetResFlags:=F_HI;                    gten:                      GetResFlags:=F_CS;                  end;              end;        end;      end;    procedure tarmaddnode.second_addfloat;      var        op : TAsmOp;      begin        case aktfputype of          fpu_fpa,          fpu_fpa10,          fpu_fpa11:            begin              pass_left_right;              if (nf_swaped in flags) then                swapleftright;              case nodetype of                addn :                  op:=A_ADF;                muln :                  op:=A_MUF;                subn :                  op:=A_SUF;                slashn :                  op:=A_DVF;                else                  internalerror(200308313);              end;              { force fpureg as location, left right doesn't matter                as both will be in a fpureg }              location_force_fpureg(exprasmlist,left.location,true);              location_force_fpureg(exprasmlist,right.location,(left.location.loc<>LOC_CFPUREGISTER));              location_reset(location,LOC_FPUREGISTER,def_cgsize(resulttype.def));              if left.location.loc<>LOC_CFPUREGISTER then                location.register:=left.location.register              else                location.register:=right.location.register;              exprasmlist.concat(setoppostfix(taicpu.op_reg_reg_reg(op,                 location.register,left.location.register,right.location.register),                 cgsize2fpuoppostfix[def_cgsize(resulttype.def)]));              location.loc:=LOC_FPUREGISTER;            end;          fpu_soft:            { this case should be handled already by pass1 }            internalerror(200308252);          else            internalerror(200308251);        end;      end;    procedure tarmaddnode.second_cmpfloat;      begin        pass_left_right;        if (nf_swaped in flags) then          swapleftright;        { force fpureg as location, left right doesn't matter          as both will be in a fpureg }        location_force_fpureg(exprasmlist,left.location,true);        location_force_fpureg(exprasmlist,right.location,true);        location_reset(location,LOC_FLAGS,OS_NO);        location.resflags:=getresflags(true);        if nodetype in [equaln,unequaln] then          exprasmlist.concat(setoppostfix(taicpu.op_reg_reg(A_CMF,             left.location.register,right.location.register),             cgsize2fpuoppostfix[def_cgsize(resulttype.def)]))        else          exprasmlist.concat(setoppostfix(taicpu.op_reg_reg(A_CMFE,             left.location.register,right.location.register),             cgsize2fpuoppostfix[def_cgsize(resulttype.def)]));        location_reset(location,LOC_FLAGS,OS_NO);        location.resflags:=getresflags(false);      end;    procedure tarmaddnode.second_cmpsmallset;      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              exprasmlist.concat(taicpu.op_reg_reg(A_CMP,left.location.register,right.location.register));              location.resflags:=F_EQ;            end;          unequaln:            begin              exprasmlist.concat(taicpu.op_reg_reg(A_CMP,left.location.register,right.location.register));              location.resflags:=F_NE;            end;          lten,          gten:            begin              if (not(nf_swaped in flags) and                  (nodetype = lten)) or                 ((nf_swaped in flags) and                  (nodetype = gten)) then                swapleftright;              tmpreg:=cg.getintregister(exprasmlist,location.size);              exprasmlist.concat(taicpu.op_reg_reg_reg(A_AND,tmpreg,left.location.register,right.location.register));              exprasmlist.concat(taicpu.op_reg_reg(A_CMP,tmpreg,right.location.register));              location.resflags:=F_EQ;            end;          else            internalerror(2004012401);        end;      end;    procedure tarmaddnode.second_cmp64bit;      var        unsigned : boolean;        tmpreg : tregister;        oldnodetype : tnodetype;      begin        pass_left_right;        force_reg_left_right(false,false);        unsigned:=not(is_signed(left.resulttype.def)) or                  not(is_signed(right.resulttype.def));        { operation requiring proper N, Z and C flags ? }        if unsigned or (nodetype in [equaln,unequaln]) then          begin            location_reset(location,LOC_FLAGS,OS_NO);            location.resflags:=getresflags(unsigned);            exprasmlist.concat(taicpu.op_reg_reg(A_CMP,left.location.register64.reghi,right.location.register64.reghi));            exprasmlist.concat(setcondition(taicpu.op_reg_reg(A_CMP,left.location.register64.reglo,right.location.register64.reglo),C_EQ));          end        else        { operation requiring proper N, Z and V flags ? }          begin            location_reset(location,LOC_JUMP,OS_NO);            exprasmlist.concat(taicpu.op_reg_reg(A_CMP,left.location.register64.reghi,right.location.register64.reghi));            { the jump the sequence is a little bit hairy }            case nodetype of               ltn,gtn:                 begin                    cg.a_jmp_flags(exprasmlist,getresflags(false),truelabel);                    { cheat a little bit for the negative test }                    toggleflag(nf_swaped);                    cg.a_jmp_flags(exprasmlist,getresflags(false),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(unsigned),truelabel);                    { cheat for the negative test }                    if nodetype=ltn then                      nodetype:=gtn                    else                      nodetype:=ltn;                    cg.a_jmp_flags(exprasmlist,getresflags(unsigned),falselabel);                    nodetype:=oldnodetype;                 end;            end;            exprasmlist.concat(taicpu.op_reg_reg(A_CMP,left.location.register64.reglo,right.location.register64.reglo));            { the comparisaion of the low dword have to be               always unsigned!                            }            cg.a_jmp_flags(exprasmlist,getresflags(true),truelabel);            cg.a_jmp_always(exprasmlist,falselabel);          end;      end;    function tarmaddnode.pass_1 : tnode;      begin        result:=inherited pass_1;        { handling boolean expressions }        if not(assigned(result)) and           (             not(is_boolean(left.resulttype.def)) or             not(is_boolean(right.resulttype.def)) or             is_dynamic_array(left.resulttype.def)           ) then          expectloc:=LOC_FLAGS;      end;    procedure tarmaddnode.second_cmpordinal;      var        unsigned : boolean;        tmpreg : tregister;        b : byte;      begin        pass_left_right;        force_reg_left_right(true,true);        unsigned:=not(is_signed(left.resulttype.def)) or                  not(is_signed(right.resulttype.def));        if right.location.loc = LOC_CONSTANT then          begin             if is_shifter_const(right.location.value,b) then               exprasmlist.concat(taicpu.op_reg_const(A_CMP,left.location.register,right.location.value))             else               begin                 tmpreg:=cg.getintregister(exprasmlist,location.size);                 cg.a_load_const_reg(exprasmlist,OS_INT,                   right.location.value,tmpreg);                 exprasmlist.concat(taicpu.op_reg_reg(A_CMP,left.location.register,tmpreg));               end;          end        else          exprasmlist.concat(taicpu.op_reg_reg(A_CMP,left.location.register,right.location.register));        location_reset(location,LOC_FLAGS,OS_NO);        location.resflags:=getresflags(unsigned);      end;begin  caddnode:=tarmaddnode;end.
 |