|
@@ -0,0 +1,275 @@
|
|
|
+{
|
|
|
+ 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;
|
|
|
+ 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;
|
|
|
+
|
|
|
+{*****************************************************************************
|
|
|
+ TSparcAddNode
|
|
|
+*****************************************************************************}
|
|
|
+
|
|
|
+ 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_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_swapped 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 tavraddnode.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
|
|
|
+ current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg(A_CMP,left.location.register,right.location.register));
|
|
|
+ location.resflags:=F_EQ;
|
|
|
+ end;
|
|
|
+ unequaln:
|
|
|
+ begin
|
|
|
+ current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg(A_CMP,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);
|
|
|
+ current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg_reg(A_AND,tmpreg,left.location.register,right.location.register));
|
|
|
+ current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg(A_CMP,tmpreg,right.location.register));
|
|
|
+ location.resflags:=F_EQ;
|
|
|
+ end;
|
|
|
+ else
|
|
|
+ internalerror(2004012401);
|
|
|
+ end;
|
|
|
+ }
|
|
|
+ end;
|
|
|
+
|
|
|
+
|
|
|
+ procedure tavraddnode.second_cmp64bit;
|
|
|
+ var
|
|
|
+ unsigned : boolean;
|
|
|
+ oldnodetype : tnodetype;
|
|
|
+ begin
|
|
|
+ {
|
|
|
+ pass_left_right;
|
|
|
+ force_reg_left_right(false,false);
|
|
|
+
|
|
|
+ unsigned:=not(is_signed(left.resultdef)) or
|
|
|
+ not(is_signed(right.resultdef));
|
|
|
+
|
|
|
+ { 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);
|
|
|
+ current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg(A_CMP,left.location.register64.reghi,right.location.register64.reghi));
|
|
|
+ current_asmdata.CurrAsmList.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);
|
|
|
+ current_asmdata.CurrAsmList.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(current_asmdata.CurrAsmList,getresflags(false),current_procinfo.CurrTrueLabel);
|
|
|
+ { cheat a little bit for the negative test }
|
|
|
+ toggleflag(nf_swapped);
|
|
|
+ cg.a_jmp_flags(current_asmdata.CurrAsmList,getresflags(false),current_procinfo.CurrFalseLabel);
|
|
|
+ toggleflag(nf_swapped);
|
|
|
+ end;
|
|
|
+ lten,gten:
|
|
|
+ begin
|
|
|
+ oldnodetype:=nodetype;
|
|
|
+ if nodetype=lten then
|
|
|
+ nodetype:=ltn
|
|
|
+ else
|
|
|
+ nodetype:=gtn;
|
|
|
+ cg.a_jmp_flags(current_asmdata.CurrAsmList,getresflags(unsigned),current_procinfo.CurrTrueLabel);
|
|
|
+ { cheat for the negative test }
|
|
|
+ if nodetype=ltn then
|
|
|
+ nodetype:=gtn
|
|
|
+ else
|
|
|
+ nodetype:=ltn;
|
|
|
+ cg.a_jmp_flags(current_asmdata.CurrAsmList,getresflags(unsigned),current_procinfo.CurrFalseLabel);
|
|
|
+ nodetype:=oldnodetype;
|
|
|
+ end;
|
|
|
+ end;
|
|
|
+ current_asmdata.CurrAsmList.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(current_asmdata.CurrAsmList,getresflags(true),current_procinfo.CurrTrueLabel);
|
|
|
+ cg.a_jmp_always(current_asmdata.CurrAsmList,current_procinfo.CurrFalseLabel);
|
|
|
+ end;
|
|
|
+ }
|
|
|
+ end;
|
|
|
+
|
|
|
+
|
|
|
+ function tavraddnode.pass_1 : tnode;
|
|
|
+ begin
|
|
|
+ result:=inherited pass_1;
|
|
|
+
|
|
|
+ { 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;
|
|
|
+ end;
|
|
|
+
|
|
|
+
|
|
|
+ procedure tavraddnode.second_cmpordinal;
|
|
|
+ var
|
|
|
+ unsigned : boolean;
|
|
|
+ tmpreg : tregister;
|
|
|
+ b : byte;
|
|
|
+ begin
|
|
|
+ {
|
|
|
+ pass_left_right;
|
|
|
+ force_reg_left_right(true,true);
|
|
|
+
|
|
|
+ unsigned:=not(is_signed(left.resultdef)) or
|
|
|
+ not(is_signed(right.resultdef));
|
|
|
+
|
|
|
+ if right.location.loc = LOC_CONSTANT then
|
|
|
+ begin
|
|
|
+ if is_shifter_const(right.location.value,b) then
|
|
|
+ current_asmdata.CurrAsmList.concat(taicpu.op_reg_const(A_CMP,left.location.register,right.location.value))
|
|
|
+ else
|
|
|
+ begin
|
|
|
+ tmpreg:=cg.getintregister(current_asmdata.CurrAsmList,location.size);
|
|
|
+ cg.a_load_const_reg(current_asmdata.CurrAsmList,OS_INT,
|
|
|
+ right.location.value,tmpreg);
|
|
|
+ current_asmdata.CurrAsmList.concat(taicpu.op_reg_reg(A_CMP,left.location.register,tmpreg));
|
|
|
+ end;
|
|
|
+ end
|
|
|
+ else
|
|
|
+ current_asmdata.CurrAsmList.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:=tavraddnode;
|
|
|
+end.
|