[llvm-commits] CVS: llvm/lib/CodeGen/SelectionDAG/SelectionDAG.cpp

Chris Lattner lattner at cs.uiuc.edu
Thu Sep 22 17:56:04 PDT 2005



Changes in directory llvm/lib/CodeGen/SelectionDAG:

SelectionDAG.cpp updated: 1.192 -> 1.193
---
Log message:

Turn (X^C1) == C2 into X == C1^C2 iff X&~C1 = 0 (and move a function)

This happens all the time on PPC for bool values, e.g. eliminating a xori 
in inverted-bool-compares.ll.

This should be added to the dag combiner as well.


---
Diffs of the changes:  (+86 -72)

 SelectionDAG.cpp |  158 +++++++++++++++++++++++++++++--------------------------
 1 files changed, 86 insertions(+), 72 deletions(-)


Index: llvm/lib/CodeGen/SelectionDAG/SelectionDAG.cpp
diff -u llvm/lib/CodeGen/SelectionDAG/SelectionDAG.cpp:1.192 llvm/lib/CodeGen/SelectionDAG/SelectionDAG.cpp:1.193
--- llvm/lib/CodeGen/SelectionDAG/SelectionDAG.cpp:1.192	Fri Sep  9 18:00:07 2005
+++ llvm/lib/CodeGen/SelectionDAG/SelectionDAG.cpp	Thu Sep 22 19:55:52 2005
@@ -580,6 +580,80 @@
   return SDOperand(Reg, 0);
 }
 
+/// MaskedValueIsZero - Return true if 'V & Mask' is known to be zero.  We use
+/// this predicate to simplify operations downstream.  V and Mask are known to
+/// be the same type.
+static bool MaskedValueIsZero(const SDOperand &Op, uint64_t Mask,
+                              const TargetLowering &TLI) {
+  unsigned SrcBits;
+  if (Mask == 0) return true;
+  
+  // If we know the result of a setcc has the top bits zero, use this info.
+  switch (Op.getOpcode()) {
+    case ISD::Constant:
+      return (cast<ConstantSDNode>(Op)->getValue() & Mask) == 0;
+      
+    case ISD::SETCC:
+      return ((Mask & 1) == 0) &&
+      TLI.getSetCCResultContents() == TargetLowering::ZeroOrOneSetCCResult;
+      
+    case ISD::ZEXTLOAD:
+      SrcBits = MVT::getSizeInBits(cast<VTSDNode>(Op.getOperand(3))->getVT());
+      return (Mask & ((1ULL << SrcBits)-1)) == 0; // Returning only the zext bits.
+    case ISD::ZERO_EXTEND:
+      SrcBits = MVT::getSizeInBits(Op.getOperand(0).getValueType());
+      return MaskedValueIsZero(Op.getOperand(0),Mask & ((1ULL << SrcBits)-1),TLI);
+    case ISD::AssertZext:
+      SrcBits = MVT::getSizeInBits(cast<VTSDNode>(Op.getOperand(1))->getVT());
+      return (Mask & ((1ULL << SrcBits)-1)) == 0; // Returning only the zext bits.
+    case ISD::AND:
+      // (X & C1) & C2 == 0   iff   C1 & C2 == 0.
+      if (ConstantSDNode *AndRHS = dyn_cast<ConstantSDNode>(Op.getOperand(1)))
+        return MaskedValueIsZero(Op.getOperand(0),AndRHS->getValue() & Mask, TLI);
+      
+      // FALL THROUGH
+    case ISD::OR:
+    case ISD::XOR:
+      return MaskedValueIsZero(Op.getOperand(0), Mask, TLI) &&
+      MaskedValueIsZero(Op.getOperand(1), Mask, TLI);
+    case ISD::SELECT:
+      return MaskedValueIsZero(Op.getOperand(1), Mask, TLI) &&
+      MaskedValueIsZero(Op.getOperand(2), Mask, TLI);
+    case ISD::SELECT_CC:
+      return MaskedValueIsZero(Op.getOperand(2), Mask, TLI) &&
+      MaskedValueIsZero(Op.getOperand(3), Mask, TLI);
+    case ISD::SRL:
+      // (ushr X, C1) & C2 == 0   iff  X & (C2 << C1) == 0
+      if (ConstantSDNode *ShAmt = dyn_cast<ConstantSDNode>(Op.getOperand(1))) {
+        uint64_t NewVal = Mask << ShAmt->getValue();
+        SrcBits = MVT::getSizeInBits(Op.getValueType());
+        if (SrcBits != 64) NewVal &= (1ULL << SrcBits)-1;
+        return MaskedValueIsZero(Op.getOperand(0), NewVal, TLI);
+      }
+      return false;
+    case ISD::SHL:
+      // (ushl X, C1) & C2 == 0   iff  X & (C2 >> C1) == 0
+      if (ConstantSDNode *ShAmt = dyn_cast<ConstantSDNode>(Op.getOperand(1))) {
+        uint64_t NewVal = Mask >> ShAmt->getValue();
+        return MaskedValueIsZero(Op.getOperand(0), NewVal, TLI);
+      }
+      return false;
+    case ISD::CTTZ:
+    case ISD::CTLZ:
+    case ISD::CTPOP:
+      // Bit counting instructions can not set the high bits of the result
+      // register.  The max number of bits sets depends on the input.
+      return (Mask & (MVT::getSizeInBits(Op.getValueType())*2-1)) == 0;
+      
+      // TODO we could handle some SRA cases here.
+    default: break;
+  }
+  
+  return false;
+}
+
+
+
 SDOperand SelectionDAG::SimplifySetCC(MVT::ValueType VT, SDOperand N1,
                                       SDOperand N2, ISD::CondCode Cond) {
   // These setcc operations always fold.
@@ -816,6 +890,18 @@
 
       // FIXME: move this stuff to the DAG Combiner when it exists!
 
+      // Turn (X^C1) == C2 into X == C1^C2 iff X&~C1 = 0.  Common for condcodes.
+      if (N1.getOpcode() == ISD::XOR)
+        if (ConstantSDNode *XORC = dyn_cast<ConstantSDNode>(N1.getOperand(1)))
+          if (ConstantSDNode *RHSC = dyn_cast<ConstantSDNode>(N2)) {
+            // If we know that all of the inverted bits are zero, don't bother
+            // performing the inversion.
+            if (MaskedValueIsZero(N1.getOperand(0), ~XORC->getValue(), TLI))
+              return getSetCC(VT, N1.getOperand(0),
+                              getConstant(XORC->getValue()^RHSC->getValue(),
+                                          N1.getValueType()), Cond);
+          }
+      
       // Simplify (X+Z) == X -->  Z == 0
       if (N1.getOperand(0) == N2)
         return getSetCC(VT, N1.getOperand(1),
@@ -1126,78 +1212,6 @@
   return SDOperand(N, 0);
 }
 
-/// MaskedValueIsZero - Return true if 'V & Mask' is known to be zero.  We use
-/// this predicate to simplify operations downstream.  V and Mask are known to
-/// be the same type.
-static bool MaskedValueIsZero(const SDOperand &Op, uint64_t Mask,
-                              const TargetLowering &TLI) {
-  unsigned SrcBits;
-  if (Mask == 0) return true;
-
-  // If we know the result of a setcc has the top bits zero, use this info.
-  switch (Op.getOpcode()) {
-  case ISD::Constant:
-    return (cast<ConstantSDNode>(Op)->getValue() & Mask) == 0;
-
-  case ISD::SETCC:
-    return ((Mask & 1) == 0) &&
-           TLI.getSetCCResultContents() == TargetLowering::ZeroOrOneSetCCResult;
-
-  case ISD::ZEXTLOAD:
-    SrcBits = MVT::getSizeInBits(cast<VTSDNode>(Op.getOperand(3))->getVT());
-    return (Mask & ((1ULL << SrcBits)-1)) == 0; // Returning only the zext bits.
-  case ISD::ZERO_EXTEND:
-    SrcBits = MVT::getSizeInBits(Op.getOperand(0).getValueType());
-    return MaskedValueIsZero(Op.getOperand(0),Mask & ((1ULL << SrcBits)-1),TLI);
-  case ISD::AssertZext:
-    SrcBits = MVT::getSizeInBits(cast<VTSDNode>(Op.getOperand(1))->getVT());
-    return (Mask & ((1ULL << SrcBits)-1)) == 0; // Returning only the zext bits.
-  case ISD::AND:
-    // (X & C1) & C2 == 0   iff   C1 & C2 == 0.
-    if (ConstantSDNode *AndRHS = dyn_cast<ConstantSDNode>(Op.getOperand(1)))
-      return MaskedValueIsZero(Op.getOperand(0),AndRHS->getValue() & Mask, TLI);
-
-    // FALL THROUGH
-  case ISD::OR:
-  case ISD::XOR:
-    return MaskedValueIsZero(Op.getOperand(0), Mask, TLI) &&
-           MaskedValueIsZero(Op.getOperand(1), Mask, TLI);
-  case ISD::SELECT:
-    return MaskedValueIsZero(Op.getOperand(1), Mask, TLI) &&
-           MaskedValueIsZero(Op.getOperand(2), Mask, TLI);
-  case ISD::SELECT_CC:
-    return MaskedValueIsZero(Op.getOperand(2), Mask, TLI) &&
-           MaskedValueIsZero(Op.getOperand(3), Mask, TLI);
-  case ISD::SRL:
-    // (ushr X, C1) & C2 == 0   iff  X & (C2 << C1) == 0
-    if (ConstantSDNode *ShAmt = dyn_cast<ConstantSDNode>(Op.getOperand(1))) {
-      uint64_t NewVal = Mask << ShAmt->getValue();
-      SrcBits = MVT::getSizeInBits(Op.getValueType());
-      if (SrcBits != 64) NewVal &= (1ULL << SrcBits)-1;
-      return MaskedValueIsZero(Op.getOperand(0), NewVal, TLI);
-    }
-    return false;
-  case ISD::SHL:
-    // (ushl X, C1) & C2 == 0   iff  X & (C2 >> C1) == 0
-    if (ConstantSDNode *ShAmt = dyn_cast<ConstantSDNode>(Op.getOperand(1))) {
-      uint64_t NewVal = Mask >> ShAmt->getValue();
-      return MaskedValueIsZero(Op.getOperand(0), NewVal, TLI);
-    }
-    return false;
-  case ISD::CTTZ:
-  case ISD::CTLZ:
-  case ISD::CTPOP:
-    // Bit counting instructions can not set the high bits of the result
-    // register.  The max number of bits sets depends on the input.
-    return (Mask & (MVT::getSizeInBits(Op.getValueType())*2-1)) == 0;
-    
-    // TODO we could handle some SRA cases here.
-  default: break;
-  }
-
-  return false;
-}
-
 
 
 SDOperand SelectionDAG::getNode(unsigned Opcode, MVT::ValueType VT,






More information about the llvm-commits mailing list