aboutsummaryrefslogtreecommitdiff
path: root/lib/Target/X86/X86ISelLowering.cpp
diff options
context:
space:
mode:
authorDale Johannesen <dalej@apple.com>2009-06-01 23:27:20 +0000
committerDale Johannesen <dalej@apple.com>2009-06-01 23:27:20 +0000
commit4150d83abe90a5da4ddf86433b7bf4329acfa57c (patch)
treef6d49f1a1eadb573ca9969173c76b891c035eb53 /lib/Target/X86/X86ISelLowering.cpp
parent83138998513faed0c1d31e712ac6e6d5e6ee8d91 (diff)
Make the implicit inputs and outputs of target-independent
ADDC/ADDE use MVT::i1 (later, whatever it gets legalized to) instead of MVT::Flag. Remove CARRY_FALSE in favor of 0; adjust all target-independent code to use this format. Most targets will still produce a Flag-setting target-dependent version when selection is done. X86 is converted to use i32 instead, which means TableGen needs to produce different code in xxxGenDAGISel.inc. This keys off the new supportsHasI1 bit in xxxInstrInfo, currently set only for X86; in principle this is temporary and should go away when all other targets have been converted. All relevant X86 instruction patterns are modified to represent setting and using EFLAGS explicitly. The same can be done on other targets. The immediate behavior change is that an ADC/ADD pair are no longer tightly coupled in the X86 scheduler; they can be separated by instructions that don't clobber the flags (MOV). I will soon add some peephole optimizations based on using other instructions that set the flags to feed into ADC. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@72707 91177308-0d34-0410-b5e6-96231b3b80d8
Diffstat (limited to 'lib/Target/X86/X86ISelLowering.cpp')
-rw-r--r--lib/Target/X86/X86ISelLowering.cpp45
1 files changed, 45 insertions, 0 deletions
diff --git a/lib/Target/X86/X86ISelLowering.cpp b/lib/Target/X86/X86ISelLowering.cpp
index 882ee3a01f..14406b5df1 100644
--- a/lib/Target/X86/X86ISelLowering.cpp
+++ b/lib/Target/X86/X86ISelLowering.cpp
@@ -190,6 +190,28 @@ X86TargetLowering::X86TargetLowering(X86TargetMachine &TM)
setOperationAction(ISD::BIT_CONVERT , MVT::i32 , Expand);
}
+ // ADDE and SUBE are lowered to local versions that contain EFLAGS explicitly.
+ // ADDC and SUBC are lowered to local versions so EFLAGS will be an i32
+ // rather than the Flag used by the generic patterns.
+ setOperationAction(ISD::ADDC , MVT::i8 , Custom);
+ setOperationAction(ISD::ADDC , MVT::i16 , Custom);
+ setOperationAction(ISD::ADDC , MVT::i32 , Custom);
+ setOperationAction(ISD::SUBC , MVT::i8 , Custom);
+ setOperationAction(ISD::SUBC , MVT::i16 , Custom);
+ setOperationAction(ISD::SUBC , MVT::i32 , Custom);
+ setOperationAction(ISD::ADDE , MVT::i8 , Custom);
+ setOperationAction(ISD::ADDE , MVT::i16 , Custom);
+ setOperationAction(ISD::ADDE , MVT::i32 , Custom);
+ setOperationAction(ISD::SUBE , MVT::i8 , Custom);
+ setOperationAction(ISD::SUBE , MVT::i16 , Custom);
+ setOperationAction(ISD::SUBE , MVT::i32 , Custom);
+ if (Subtarget->is64Bit()) {
+ setOperationAction(ISD::ADDC , MVT::i64 , Custom);
+ setOperationAction(ISD::SUBC , MVT::i64 , Custom);
+ setOperationAction(ISD::ADDE , MVT::i64 , Custom);
+ setOperationAction(ISD::SUBE , MVT::i64 , Custom);
+ }
+
// Scalar integer divide and remainder are lowered to use operations that
// produce two results, to match the available instructions. This exposes
// the two-result form to trivial CSE, which is able to combine x/y and x%y
@@ -6475,6 +6497,21 @@ SDValue X86TargetLowering::LowerXALUO(SDValue Op, SelectionDAG &DAG) {
return Sum;
}
+SDValue X86TargetLowering::LowerADDSUBE(SDValue Op, SelectionDAG &DAG) {
+ DebugLoc dl = Op.getDebugLoc();
+ SDVTList VTs = DAG.getVTList(Op.getValueType(), MVT::i32);
+ return DAG.getNode(Op.getOpcode()==ISD::ADDE ? X86ISD::ADDE : X86ISD::SUBE,
+ dl, VTs, Op.getOperand(0), Op.getOperand(1),
+ Op.getOperand(2).getValue(1));
+}
+
+SDValue X86TargetLowering::LowerADDSUBC(SDValue Op, SelectionDAG &DAG) {
+ DebugLoc dl = Op.getDebugLoc();
+ SDVTList VTs = DAG.getVTList(Op.getValueType(), MVT::i32);
+ return DAG.getNode(Op.getOpcode()==ISD::ADDC ? X86ISD::ADD : X86ISD::SUB,
+ dl, VTs, Op.getOperand(0), Op.getOperand(1));
+}
+
SDValue X86TargetLowering::LowerCMP_SWAP(SDValue Op, SelectionDAG &DAG) {
MVT T = Op.getValueType();
DebugLoc dl = Op.getDebugLoc();
@@ -6543,6 +6580,10 @@ SDValue X86TargetLowering::LowerLOAD_SUB(SDValue Op, SelectionDAG &DAG) {
SDValue X86TargetLowering::LowerOperation(SDValue Op, SelectionDAG &DAG) {
switch (Op.getOpcode()) {
default: assert(0 && "Should not custom lower this!");
+ case ISD::ADDC:
+ case ISD::SUBC: return LowerADDSUBC(Op,DAG);
+ case ISD::ADDE:
+ case ISD::SUBE: return LowerADDSUBE(Op,DAG);
case ISD::ATOMIC_CMP_SWAP: return LowerCMP_SWAP(Op,DAG);
case ISD::ATOMIC_LOAD_SUB: return LowerLOAD_SUB(Op,DAG);
case ISD::BUILD_VECTOR: return LowerBUILD_VECTOR(Op, DAG);
@@ -6791,6 +6832,10 @@ const char *X86TargetLowering::getTargetNodeName(unsigned Opcode) const {
case X86ISD::INC: return "X86ISD::INC";
case X86ISD::DEC: return "X86ISD::DEC";
case X86ISD::MUL_IMM: return "X86ISD::MUL_IMM";
+ case X86ISD::ADDE: return "X86ISD::ADDE";
+ case X86ISD::SUBE: return "X86ISD::SUBE";
+ case X86ISD::ADDC: return "X86ISD::ADDC";
+ case X86ISD::SUBC: return "X86ISD::SUBC";
}
}