mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-08-15 22:28:18 +00:00
initial support for branches
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@29854 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
@@ -174,8 +174,7 @@ void ARMAsmPrinter::printOperand(const MachineInstr *MI, int opNum) {
|
|||||||
O << "#" << (int)MO.getImmedValue();
|
O << "#" << (int)MO.getImmedValue();
|
||||||
break;
|
break;
|
||||||
case MachineOperand::MO_MachineBasicBlock:
|
case MachineOperand::MO_MachineBasicBlock:
|
||||||
assert(0 && "not implemented");
|
printBasicBlockLabel(MO.getMachineBasicBlock());
|
||||||
abort();
|
|
||||||
return;
|
return;
|
||||||
case MachineOperand::MO_GlobalAddress: {
|
case MachineOperand::MO_GlobalAddress: {
|
||||||
GlobalValue *GV = MO.getGlobal();
|
GlobalValue *GV = MO.getGlobal();
|
||||||
|
@@ -53,6 +53,7 @@ ARMTargetLowering::ARMTargetLowering(TargetMachine &TM)
|
|||||||
|
|
||||||
setOperationAction(ISD::SETCC, MVT::i32, Expand);
|
setOperationAction(ISD::SETCC, MVT::i32, Expand);
|
||||||
setOperationAction(ISD::SELECT_CC, MVT::i32, Custom);
|
setOperationAction(ISD::SELECT_CC, MVT::i32, Custom);
|
||||||
|
setOperationAction(ISD::BR_CC, MVT::i32, Custom);
|
||||||
|
|
||||||
setSchedulingPreference(SchedulingForRegPressure);
|
setSchedulingPreference(SchedulingForRegPressure);
|
||||||
computeRegisterProperties();
|
computeRegisterProperties();
|
||||||
@@ -71,7 +72,9 @@ namespace llvm {
|
|||||||
|
|
||||||
CMP,
|
CMP,
|
||||||
|
|
||||||
SELECT
|
SELECT,
|
||||||
|
|
||||||
|
BR
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -83,6 +86,7 @@ const char *ARMTargetLowering::getTargetNodeName(unsigned Opcode) const {
|
|||||||
case ARMISD::RET_FLAG: return "ARMISD::RET_FLAG";
|
case ARMISD::RET_FLAG: return "ARMISD::RET_FLAG";
|
||||||
case ARMISD::SELECT: return "ARMISD::SELECT";
|
case ARMISD::SELECT: return "ARMISD::SELECT";
|
||||||
case ARMISD::CMP: return "ARMISD::CMP";
|
case ARMISD::CMP: return "ARMISD::CMP";
|
||||||
|
case ARMISD::BR: return "ARMISD::BR";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -312,6 +316,19 @@ static SDOperand LowerSELECT_CC(SDOperand Op, SelectionDAG &DAG) {
|
|||||||
return DAG.getNode(ARMISD::SELECT, MVT::i32, FalseVal, TrueVal, Cmp);
|
return DAG.getNode(ARMISD::SELECT, MVT::i32, FalseVal, TrueVal, Cmp);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static SDOperand LowerBR_CC(SDOperand Op, SelectionDAG &DAG) {
|
||||||
|
SDOperand Chain = Op.getOperand(0);
|
||||||
|
ISD::CondCode CC = cast<CondCodeSDNode>(Op.getOperand(1))->get();
|
||||||
|
SDOperand LHS = Op.getOperand(2);
|
||||||
|
SDOperand RHS = Op.getOperand(3);
|
||||||
|
SDOperand Dest = Op.getOperand(4);
|
||||||
|
|
||||||
|
assert(CC == ISD::SETNE);
|
||||||
|
|
||||||
|
SDOperand Cmp = DAG.getNode(ARMISD::CMP, MVT::Flag, LHS, RHS);
|
||||||
|
return DAG.getNode(ARMISD::BR, MVT::Other, Chain, Dest, Cmp);
|
||||||
|
}
|
||||||
|
|
||||||
SDOperand ARMTargetLowering::LowerOperation(SDOperand Op, SelectionDAG &DAG) {
|
SDOperand ARMTargetLowering::LowerOperation(SDOperand Op, SelectionDAG &DAG) {
|
||||||
switch (Op.getOpcode()) {
|
switch (Op.getOpcode()) {
|
||||||
default:
|
default:
|
||||||
@@ -329,6 +346,8 @@ SDOperand ARMTargetLowering::LowerOperation(SDOperand Op, SelectionDAG &DAG) {
|
|||||||
return LowerRET(Op, DAG);
|
return LowerRET(Op, DAG);
|
||||||
case ISD::SELECT_CC:
|
case ISD::SELECT_CC:
|
||||||
return LowerSELECT_CC(Op, DAG);
|
return LowerSELECT_CC(Op, DAG);
|
||||||
|
case ISD::BR_CC:
|
||||||
|
return LowerBR_CC(Op, DAG);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -37,6 +37,8 @@ class InstARM<dag ops, string asmstr, list<dag> pattern> : Instruction {
|
|||||||
let Pattern = pattern;
|
let Pattern = pattern;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
def brtarget : Operand<OtherVT>;
|
||||||
|
|
||||||
def SDT_ARMCallSeq : SDTypeProfile<0, 1, [ SDTCisVT<0, i32> ]>;
|
def SDT_ARMCallSeq : SDTypeProfile<0, 1, [ SDTCisVT<0, i32> ]>;
|
||||||
def callseq_start : SDNode<"ISD::CALLSEQ_START", SDT_ARMCallSeq,
|
def callseq_start : SDNode<"ISD::CALLSEQ_START", SDT_ARMCallSeq,
|
||||||
[SDNPHasChain, SDNPOutFlag]>;
|
[SDNPHasChain, SDNPOutFlag]>;
|
||||||
@@ -50,6 +52,9 @@ def retflag : SDNode<"ARMISD::RET_FLAG", SDTRet,
|
|||||||
[SDNPHasChain, SDNPOptInFlag]>;
|
[SDNPHasChain, SDNPOptInFlag]>;
|
||||||
def armselect : SDNode<"ARMISD::SELECT", SDTIntBinOp, [SDNPInFlag, SDNPOutFlag]>;
|
def armselect : SDNode<"ARMISD::SELECT", SDTIntBinOp, [SDNPInFlag, SDNPOutFlag]>;
|
||||||
|
|
||||||
|
def SDTarmbr : SDTypeProfile<0, 1, [SDTCisVT<0, OtherVT>]>;
|
||||||
|
def armbr : SDNode<"ARMISD::BR", SDTarmbr, [SDNPHasChain, SDNPInFlag]>;
|
||||||
|
|
||||||
def SDTVoidBinOp : SDTypeProfile<0, 2, [SDTCisSameAs<0, 1>]>;
|
def SDTVoidBinOp : SDTypeProfile<0, 2, [SDTCisSameAs<0, 1>]>;
|
||||||
def armcmp : SDNode<"ARMISD::CMP", SDTVoidBinOp, [SDNPOutFlag]>;
|
def armcmp : SDNode<"ARMISD::CMP", SDTVoidBinOp, [SDNPOutFlag]>;
|
||||||
|
|
||||||
@@ -107,6 +112,10 @@ let isTwoAddress = 1 in {
|
|||||||
[(set IntRegs:$dst, (armselect IntRegs:$true, IntRegs:$false))]>;
|
[(set IntRegs:$dst, (armselect IntRegs:$true, IntRegs:$false))]>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
def bne : InstARM<(ops brtarget:$dst),
|
||||||
|
"bne $dst",
|
||||||
|
[(armbr bb:$dst)]>;
|
||||||
|
|
||||||
def cmp : InstARM<(ops IntRegs:$a, IntRegs:$b),
|
def cmp : InstARM<(ops IntRegs:$a, IntRegs:$b),
|
||||||
"cmp $a, $b",
|
"cmp $a, $b",
|
||||||
[(armcmp IntRegs:$a, IntRegs:$b)]>;
|
[(armcmp IntRegs:$a, IntRegs:$b)]>;
|
||||||
|
13
test/CodeGen/ARM/branch.ll
Normal file
13
test/CodeGen/ARM/branch.ll
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
; RUN: llvm-as < %s | llc -march=arm
|
||||||
|
void %f(int %a, int* %v) {
|
||||||
|
entry:
|
||||||
|
%tmp = seteq int %a, 0 ; <bool> [#uses=1]
|
||||||
|
br bool %tmp, label %cond_true, label %return
|
||||||
|
|
||||||
|
cond_true: ; preds = %entry
|
||||||
|
store int 0, int* %v
|
||||||
|
ret void
|
||||||
|
|
||||||
|
return: ; preds = %entry
|
||||||
|
ret void
|
||||||
|
}
|
Reference in New Issue
Block a user