InstCombine: Reduce "X shift (A srem B)" to "X shift (A urem B)" iff B is positive.

This allows to transform the rem in "1 << ((int)x % 8);" to an and.


git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@120028 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Benjamin Kramer
2010-11-23 18:52:42 +00:00
parent e104f1bccb
commit b70ebd2aa3
3 changed files with 28 additions and 10 deletions

View File

@@ -53,6 +53,21 @@ Instruction *InstCombiner::commonShiftTransforms(BinaryOperator &I) {
if (ConstantInt *CUI = dyn_cast<ConstantInt>(Op1))
if (Instruction *Res = FoldShiftByConstant(Op0, CUI, I))
return Res;
// X shift (A srem B) -> X shift (A urem B) iff B is positive.
// Because shifts by negative values are undefined.
if (BinaryOperator *BO = dyn_cast<BinaryOperator>(Op1))
if (BO->getOpcode() == Instruction::SRem && BO->getType()->isIntegerTy()) {
// Make sure the divisor's sign bit is zero.
APInt Mask = APInt::getSignBit(BO->getType()->getPrimitiveSizeInBits());
if (MaskedValueIsZero(BO->getOperand(1), Mask)) {
Value *URem = Builder->CreateURem(BO->getOperand(0), BO->getOperand(1),
BO->getName());
I.setOperand(1, URem);
return &I;
}
}
return 0;
}