Generalize instcombine's support for combining multiple bit checks into a single test. Patch by Dirk Steinke!

llvm-svn: 113423
This commit is contained in:
Owen Anderson 2010-09-08 22:16:17 +00:00
parent 8ead237758
commit 3fe002dfb5
2 changed files with 625 additions and 33 deletions

View File

@ -434,6 +434,270 @@ Value *InstCombiner::FoldLogicalPlusAnd(Value *LHS, Value *RHS,
return Builder->CreateAdd(LHSI->getOperand(0), RHS, "fold"); return Builder->CreateAdd(LHSI->getOperand(0), RHS, "fold");
} }
/// enum for classifying (icmp eq (A & B), C) and (icmp ne (A & B), C)
/// One of A and B is considered the mask, the other the value. This is
/// described as the "AMask" or "BMask" part of the enum. If the enum
/// contains only "Mask", then both A and B can be considered masks.
/// If A is the mask, then it was proven, that (A & C) == C. This
/// is trivial if C == A, or C == 0. If both A and C are constants, this
/// proof is also easy.
/// For the following explanations we assume that A is the mask.
/// The part "AllOnes" declares, that the comparison is true only
/// if (A & B) == A, or all bits of A are set in B.
/// Example: (icmp eq (A & 3), 3) -> FoldMskICmp_AMask_AllOnes
/// The part "AllZeroes" declares, that the comparison is true only
/// if (A & B) == 0, or all bits of A are cleared in B.
/// Example: (icmp eq (A & 3), 0) -> FoldMskICmp_Mask_AllZeroes
/// The part "Mixed" declares, that (A & B) == C and C might or might not
/// contain any number of one bits and zero bits.
/// Example: (icmp eq (A & 3), 1) -> FoldMskICmp_AMask_Mixed
/// The Part "Not" means, that in above descriptions "==" should be replaced
/// by "!=".
/// Example: (icmp ne (A & 3), 3) -> FoldMskICmp_AMask_NotAllOnes
/// If the mask A contains a single bit, then the following is equivalent:
/// (icmp eq (A & B), A) equals (icmp ne (A & B), 0)
/// (icmp ne (A & B), A) equals (icmp eq (A & B), 0)
enum MaskedICmpType {
FoldMskICmp_AMask_AllOnes = 1,
FoldMskICmp_AMask_NotAllOnes = 2,
FoldMskICmp_BMask_AllOnes = 4,
FoldMskICmp_BMask_NotAllOnes = 8,
FoldMskICmp_Mask_AllZeroes = 16,
FoldMskICmp_Mask_NotAllZeroes = 32,
FoldMskICmp_AMask_Mixed = 64,
FoldMskICmp_AMask_NotMixed = 128,
FoldMskICmp_BMask_Mixed = 256,
FoldMskICmp_BMask_NotMixed = 512
};
/// return the set of pattern classes (from MaskedICmpType)
/// that (icmp SCC (A & B), C) satisfies
static unsigned getTypeOfMaskedICmp(Value* A, Value* B, Value* C,
ICmpInst::Predicate SCC)
{
ConstantInt *ACst = dyn_cast<ConstantInt>(A);
ConstantInt *BCst = dyn_cast<ConstantInt>(B);
ConstantInt *CCst = dyn_cast<ConstantInt>(C);
bool icmp_eq = (SCC == ICmpInst::ICMP_EQ);
bool icmp_abit = (ACst != 0 && !ACst->isZero() &&
ACst->getValue().isPowerOf2());
bool icmp_bbit = (BCst != 0 && !BCst->isZero() &&
BCst->getValue().isPowerOf2());
unsigned result = 0;
if (CCst != 0 && CCst->isZero()) {
// if C is zero, then both A and B qualify as mask
result |= (icmp_eq ? (FoldMskICmp_Mask_AllZeroes |
FoldMskICmp_Mask_AllZeroes |
FoldMskICmp_AMask_Mixed |
FoldMskICmp_BMask_Mixed)
: (FoldMskICmp_Mask_NotAllZeroes |
FoldMskICmp_Mask_NotAllZeroes |
FoldMskICmp_AMask_NotMixed |
FoldMskICmp_BMask_NotMixed));
if (icmp_abit)
result |= (icmp_eq ? (FoldMskICmp_AMask_NotAllOnes |
FoldMskICmp_AMask_NotMixed)
: (FoldMskICmp_AMask_AllOnes |
FoldMskICmp_AMask_Mixed));
if (icmp_bbit)
result |= (icmp_eq ? (FoldMskICmp_BMask_NotAllOnes |
FoldMskICmp_BMask_NotMixed)
: (FoldMskICmp_BMask_AllOnes |
FoldMskICmp_BMask_Mixed));
return result;
}
if (A == C) {
result |= (icmp_eq ? (FoldMskICmp_AMask_AllOnes |
FoldMskICmp_AMask_Mixed)
: (FoldMskICmp_AMask_NotAllOnes |
FoldMskICmp_AMask_NotMixed));
if (icmp_abit)
result |= (icmp_eq ? (FoldMskICmp_Mask_NotAllZeroes |
FoldMskICmp_AMask_NotMixed)
: (FoldMskICmp_Mask_AllZeroes |
FoldMskICmp_AMask_Mixed));
}
else if (ACst != 0 && CCst != 0 &&
ConstantExpr::getAnd(ACst, CCst) == CCst) {
result |= (icmp_eq ? FoldMskICmp_AMask_Mixed
: FoldMskICmp_AMask_NotMixed);
}
if (B == C)
{
result |= (icmp_eq ? (FoldMskICmp_BMask_AllOnes |
FoldMskICmp_BMask_Mixed)
: (FoldMskICmp_BMask_NotAllOnes |
FoldMskICmp_BMask_NotMixed));
if (icmp_bbit)
result |= (icmp_eq ? (FoldMskICmp_Mask_NotAllZeroes |
FoldMskICmp_BMask_NotMixed)
: (FoldMskICmp_Mask_AllZeroes |
FoldMskICmp_BMask_Mixed));
}
else if (BCst != 0 && CCst != 0 &&
ConstantExpr::getAnd(BCst, CCst) == CCst) {
result |= (icmp_eq ? FoldMskICmp_BMask_Mixed
: FoldMskICmp_BMask_NotMixed);
}
return result;
}
/// foldLogOpOfMaskedICmpsHelper:
/// handle (icmp(A & B) ==/!= C) &/| (icmp(A & D) ==/!= E)
/// return the set of pattern classes (from MaskedICmpType)
/// that both LHS and RHS satisfy
static unsigned foldLogOpOfMaskedICmpsHelper(Value*& A,
Value*& B, Value*& C,
Value*& D, Value*& E,
ICmpInst *LHS, ICmpInst *RHS) {
ICmpInst::Predicate LHSCC = LHS->getPredicate(), RHSCC = RHS->getPredicate();
if (LHSCC != ICmpInst::ICMP_EQ && LHSCC != ICmpInst::ICMP_NE) return 0;
if (RHSCC != ICmpInst::ICMP_EQ && RHSCC != ICmpInst::ICMP_NE) return 0;
if (LHS->getOperand(0)->getType() != RHS->getOperand(0)->getType()) return 0;
// vectors are not (yet?) supported
if (LHS->getOperand(0)->getType()->isVectorTy()) return 0;
// Here comes the tricky part:
// LHS might be of the form L11 & L12 == X, X == L21 & L22,
// and L11 & L12 == L21 & L22. The same goes for RHS.
// Now we must find those components L** and R**, that are equal, so
// that we can extract the parameters A, B, C, D, and E for the canonical
// above.
Value *L1 = LHS->getOperand(0);
Value *L2 = LHS->getOperand(1);
Value *L11,*L12,*L21,*L22;
if (match(L1, m_And(m_Value(L11), m_Value(L12)))) {
if (!match(L2, m_And(m_Value(L21), m_Value(L22))))
L21 = L22 = 0;
}
else {
if (!match(L2, m_And(m_Value(L11), m_Value(L12))))
return 0;
std::swap(L1, L2);
L21 = L22 = 0;
}
Value *R1 = RHS->getOperand(0);
Value *R2 = RHS->getOperand(1);
Value *R11,*R12;
bool ok = false;
if (match(R1, m_And(m_Value(R11), m_Value(R12)))) {
if (R11 != 0 && (R11 == L11 || R11 == L12 || R11 == L21 || R11 == L22)) {
A = R11; D = R12; E = R2; ok = true;
}
else
if (R12 != 0 && (R12 == L11 || R12 == L12 || R12 == L21 || R12 == L22)) {
A = R12; D = R11; E = R2; ok = true;
}
}
if (!ok && match(R2, m_And(m_Value(R11), m_Value(R12)))) {
if (R11 != 0 && (R11 == L11 || R11 == L12 || R11 == L21 || R11 == L22)) {
A = R11; D = R12; E = R1; ok = true;
}
else
if (R12 != 0 && (R12 == L11 || R12 == L12 || R12 == L21 || R12 == L22)) {
A = R12; D = R11; E = R1; ok = true;
}
else
return 0;
}
if (!ok)
return 0;
if (L11 == A) {
B = L12; C = L2;
}
else if (L12 == A) {
B = L11; C = L2;
}
else if (L21 == A) {
B = L22; C = L1;
}
else if (L22 == A) {
B = L21; C = L1;
}
unsigned left_type = getTypeOfMaskedICmp(A, B, C, LHSCC);
unsigned right_type = getTypeOfMaskedICmp(A, D, E, RHSCC);
return left_type & right_type;
}
/// foldLogOpOfMaskedICmps:
/// try to fold (icmp(A & B) ==/!= C) &/| (icmp(A & D) ==/!= E)
/// into a single (icmp(A & X) ==/!= Y)
static Value* foldLogOpOfMaskedICmps(ICmpInst *LHS, ICmpInst *RHS,
ICmpInst::Predicate NEWCC,
llvm::InstCombiner::BuilderTy* Builder) {
Value *A = 0, *B = 0, *C = 0, *D = 0, *E = 0;
unsigned mask = foldLogOpOfMaskedICmpsHelper(A, B, C, D, E, LHS, RHS);
if (mask == 0) return 0;
if (NEWCC == ICmpInst::ICMP_NE)
mask >>= 1; // treat "Not"-states as normal states
if (mask & FoldMskICmp_Mask_AllZeroes) {
// (icmp eq (A & B), 0) & (icmp eq (A & D), 0)
// -> (icmp eq (A & (B|D)), 0)
Value* newOr = Builder->CreateOr(B, D);
Value* newAnd = Builder->CreateAnd(A, newOr);
// we can't use C as zero, because we might actually handle
// (icmp ne (A & B), B) & (icmp ne (A & D), D)
// with B and D, having a single bit set
Value* zero = Constant::getNullValue(A->getType());
return Builder->CreateICmp(NEWCC, newAnd, zero);
}
else if (mask & FoldMskICmp_BMask_AllOnes) {
// (icmp eq (A & B), B) & (icmp eq (A & D), D)
// -> (icmp eq (A & (B|D)), (B|D))
Value* newOr = Builder->CreateOr(B, D);
Value* newAnd = Builder->CreateAnd(A, newOr);
return Builder->CreateICmp(NEWCC, newAnd, newOr);
}
else if (mask & FoldMskICmp_AMask_AllOnes) {
// (icmp eq (A & B), A) & (icmp eq (A & D), A)
// -> (icmp eq (A & (B&D)), A)
Value* newAnd1 = Builder->CreateAnd(B, D);
Value* newAnd = Builder->CreateAnd(A, newAnd1);
return Builder->CreateICmp(NEWCC, newAnd, A);
}
else if (mask & FoldMskICmp_BMask_Mixed) {
// (icmp eq (A & B), C) & (icmp eq (A & D), E)
// We already know that B & C == C && D & E == E.
// If we can prove that (B & D) & (C ^ E) == 0, that is, the bits of
// C and E, which are shared by both the mask B and the mask D, don't
// contradict, then we can transform to
// -> (icmp eq (A & (B|D)), (C|E))
// Currently, we only handle the case of B, C, D, and E being constant.
ConstantInt *BCst = dyn_cast<ConstantInt>(B);
if (BCst == 0) return 0;
ConstantInt *DCst = dyn_cast<ConstantInt>(D);
if (DCst == 0) return 0;
// we can't simply use C and E, because we might actually handle
// (icmp ne (A & B), B) & (icmp eq (A & D), D)
// with B and D, having a single bit set
ConstantInt *CCst = dyn_cast<ConstantInt>(C);
if (CCst == 0) return 0;
if (LHS->getPredicate() != NEWCC)
CCst = dyn_cast<ConstantInt>( ConstantExpr::getXor(BCst, CCst) );
ConstantInt *ECst = dyn_cast<ConstantInt>(E);
if (ECst == 0) return 0;
if (RHS->getPredicate() != NEWCC)
ECst = dyn_cast<ConstantInt>( ConstantExpr::getXor(DCst, ECst) );
ConstantInt* MCst = dyn_cast<ConstantInt>(
ConstantExpr::getAnd(ConstantExpr::getAnd(BCst, DCst),
ConstantExpr::getXor(CCst, ECst)) );
// if there is a conflict we should actually return a false for the
// whole construct
if (!MCst->isZero())
return 0;
Value* newOr1 = Builder->CreateOr(B, D);
Value* newOr2 = ConstantExpr::getOr(CCst, ECst);
Value* newAnd = Builder->CreateAnd(A, newOr1);
return Builder->CreateICmp(NEWCC, newAnd, newOr2);
}
return 0;
}
/// FoldAndOfICmps - Fold (icmp)&(icmp) if possible. /// FoldAndOfICmps - Fold (icmp)&(icmp) if possible.
Value *InstCombiner::FoldAndOfICmps(ICmpInst *LHS, ICmpInst *RHS) { Value *InstCombiner::FoldAndOfICmps(ICmpInst *LHS, ICmpInst *RHS) {
ICmpInst::Predicate LHSCC = LHS->getPredicate(), RHSCC = RHS->getPredicate(); ICmpInst::Predicate LHSCC = LHS->getPredicate(), RHSCC = RHS->getPredicate();
@ -451,6 +715,13 @@ Value *InstCombiner::FoldAndOfICmps(ICmpInst *LHS, ICmpInst *RHS) {
return getICmpValue(isSigned, Code, Op0, Op1, Builder); return getICmpValue(isSigned, Code, Op0, Op1, Builder);
} }
} }
{
// handle (roughly):
// (icmp eq (A & B), C) & (icmp eq (A & D), E)
Value* fold = foldLogOpOfMaskedICmps(LHS, RHS, ICmpInst::ICMP_EQ, Builder);
if (fold) return fold;
}
// This only handles icmp of constants: (icmp1 A, C1) & (icmp2 B, C2). // This only handles icmp of constants: (icmp1 A, C1) & (icmp2 B, C2).
Value *Val = LHS->getOperand(0), *Val2 = RHS->getOperand(0); Value *Val = LHS->getOperand(0), *Val2 = RHS->getOperand(0);
@ -472,22 +743,6 @@ Value *InstCombiner::FoldAndOfICmps(ICmpInst *LHS, ICmpInst *RHS) {
Value *NewOr = Builder->CreateOr(Val, Val2); Value *NewOr = Builder->CreateOr(Val, Val2);
return Builder->CreateICmp(LHSCC, NewOr, LHSCst); return Builder->CreateICmp(LHSCC, NewOr, LHSCst);
} }
// (icmp ne (A & C1), 0) & (icmp ne (A & C2), 0) -->
// (icmp eq (A & (C1|C2)), (C1|C2)) where C1 and C2 are non-zero POT
if (LHSCC == ICmpInst::ICMP_NE && LHSCst->isZero()) {
Value *Op1 = 0, *Op2 = 0;
ConstantInt *CI1 = 0, *CI2 = 0;
if (match(LHS->getOperand(0), m_And(m_Value(Op1), m_ConstantInt(CI1))) &&
match(RHS->getOperand(0), m_And(m_Value(Op2), m_ConstantInt(CI2)))) {
if (Op1 == Op2 && !CI1->isZero() && !CI2->isZero() &&
CI1->getValue().isPowerOf2() && CI2->getValue().isPowerOf2()) {
Constant *ConstOr = ConstantExpr::getOr(CI1, CI2);
Value *NewAnd = Builder->CreateAnd(Op1, ConstOr);
return Builder->CreateICmp(ICmpInst::ICMP_EQ, NewAnd, ConstOr);
}
}
}
} }
// From here on, we only handle: // From here on, we only handle:
@ -1161,6 +1416,13 @@ Value *InstCombiner::FoldOrOfICmps(ICmpInst *LHS, ICmpInst *RHS) {
} }
} }
{
// handle (roughly):
// (icmp ne (A & B), C) | (icmp ne (A & D), E)
Value* fold = foldLogOpOfMaskedICmps(LHS, RHS, ICmpInst::ICMP_NE, Builder);
if (fold) return fold;
}
// This only handles icmp of constants: (icmp1 A, C1) | (icmp2 B, C2). // This only handles icmp of constants: (icmp1 A, C1) | (icmp2 B, C2).
Value *Val = LHS->getOperand(0), *Val2 = RHS->getOperand(0); Value *Val = LHS->getOperand(0), *Val2 = RHS->getOperand(0);
ConstantInt *LHSCst = dyn_cast<ConstantInt>(LHS->getOperand(1)); ConstantInt *LHSCst = dyn_cast<ConstantInt>(LHS->getOperand(1));
@ -1173,22 +1435,6 @@ Value *InstCombiner::FoldOrOfICmps(ICmpInst *LHS, ICmpInst *RHS) {
Value *NewOr = Builder->CreateOr(Val, Val2); Value *NewOr = Builder->CreateOr(Val, Val2);
return Builder->CreateICmp(LHSCC, NewOr, LHSCst); return Builder->CreateICmp(LHSCC, NewOr, LHSCst);
} }
// (icmp eq (A & C1), 0) | (icmp eq (A & C2), 0) -->
// (icmp ne (A & (C1|C2)), (C1|C2)) where C1 and C2 are non-zero POT
if (LHSCC == ICmpInst::ICMP_EQ && LHSCst->isZero()) {
Value *Op1 = 0, *Op2 = 0;
ConstantInt *CI1 = 0, *CI2 = 0;
if (match(LHS->getOperand(0), m_And(m_Value(Op1), m_ConstantInt(CI1))) &&
match(RHS->getOperand(0), m_And(m_Value(Op2), m_ConstantInt(CI2)))) {
if (Op1 == Op2 && !CI1->isZero() && !CI2->isZero() &&
CI1->getValue().isPowerOf2() && CI2->getValue().isPowerOf2()) {
Constant *ConstOr = ConstantExpr::getOr(CI1, CI2);
Value *NewAnd = Builder->CreateAnd(Op1, ConstOr);
return Builder->CreateICmp(ICmpInst::ICMP_NE, NewAnd, ConstOr);
}
}
}
} }
// From here on, we only handle: // From here on, we only handle:

View File

@ -23,4 +23,350 @@ entry:
%or.cond = or i1 %tobool, %tobool3 ; <i1> [#uses=1] %or.cond = or i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %or.cond, i32 0, i32 1 ; <i32> [#uses=1] %storemerge = select i1 %or.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge ret i32 %storemerge
} }
; tests to check combining (icmp eq (A & B), C) & (icmp eq (A & D), E)
; tests to check if (icmp eq (A & B), 0) is treated like (icmp eq (A & B), B)
; if B is a single bit constant
; (icmp eq (A & B), 0) & (icmp eq (A & D), 0) -> (icmp eq (A & (B|D)), 0)
define i32 @main3(i32 %argc, i8** nocapture %argv) nounwind readnone ssp {
entry:
%and = and i32 %argc, 7 ; <i32> [#uses=1]
%tobool = icmp eq i32 %and, 0 ; <i1> [#uses=1]
%and2 = and i32 %argc, 48 ; <i32> [#uses=1]
%tobool3 = icmp eq i32 %and2, 0 ; <i1> [#uses=1]
%and.cond = and i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %and.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
define i32 @main3b(i32 %argc, i8** nocapture %argv) nounwind readnone ssp {
entry:
%and = and i32 %argc, 7 ; <i32> [#uses=1]
%tobool = icmp eq i32 %and, 0 ; <i1> [#uses=1]
%and2 = and i32 %argc, 16 ; <i32> [#uses=1]
%tobool3 = icmp ne i32 %and2, 16 ; <i1> [#uses=1]
%and.cond = and i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %and.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
define i32 @main3e_like(i32 %argc, i32 %argc2, i32 %argc3, i8** nocapture %argv)
nounwind readnone ssp {
entry:
%and = and i32 %argc, %argc2 ; <i32> [#uses=1]
%tobool = icmp eq i32 %and, 0 ; <i1> [#uses=1]
%and2 = and i32 %argc, %argc3 ; <i32> [#uses=1]
%tobool3 = icmp eq i32 %and2, 0 ; <i1> [#uses=1]
%and.cond = and i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %and.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
; (icmp ne (A & B), 0) | (icmp ne (A & D), 0) -> (icmp ne (A & (B|D)), 0)
define i32 @main3c(i32 %argc, i8** nocapture %argv) nounwind readnone ssp {
entry:
%and = and i32 %argc, 7 ; <i32> [#uses=1]
%tobool = icmp ne i32 %and, 0 ; <i1> [#uses=1]
%and2 = and i32 %argc, 48 ; <i32> [#uses=1]
%tobool3 = icmp ne i32 %and2, 0 ; <i1> [#uses=1]
%or.cond = or i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %or.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
define i32 @main3d(i32 %argc, i8** nocapture %argv) nounwind readnone ssp {
entry:
%and = and i32 %argc, 7 ; <i32> [#uses=1]
%tobool = icmp ne i32 %and, 0 ; <i1> [#uses=1]
%and2 = and i32 %argc, 16 ; <i32> [#uses=1]
%tobool3 = icmp eq i32 %and2, 16 ; <i1> [#uses=1]
%or.cond = or i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %or.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
define i32 @main3f_like(i32 %argc, i32 %argc2, i32 %argc3, i8** nocapture %argv)
nounwind readnone ssp {
entry:
%and = and i32 %argc, %argc2 ; <i32> [#uses=1]
%tobool = icmp ne i32 %and, 0 ; <i1> [#uses=1]
%and2 = and i32 %argc, %argc3 ; <i32> [#uses=1]
%tobool3 = icmp ne i32 %and2, 0 ; <i1> [#uses=1]
%or.cond = or i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %or.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
; (icmp eq (A & B), B) & (icmp eq (A & D), D) -> (icmp eq (A & (B|D)), (B|D))
define i32 @main4(i32 %argc, i8** nocapture %argv) nounwind readnone ssp {
entry:
%and = and i32 %argc, 7 ; <i32> [#uses=1]
%tobool = icmp eq i32 %and, 7 ; <i1> [#uses=1]
%and2 = and i32 %argc, 48 ; <i32> [#uses=1]
%tobool3 = icmp eq i32 %and2, 48 ; <i1> [#uses=1]
%and.cond = and i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %and.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
define i32 @main4b(i32 %argc, i8** nocapture %argv) nounwind readnone ssp {
entry:
%and = and i32 %argc, 7 ; <i32> [#uses=1]
%tobool = icmp eq i32 %and, 7 ; <i1> [#uses=1]
%and2 = and i32 %argc, 16 ; <i32> [#uses=1]
%tobool3 = icmp ne i32 %and2, 0 ; <i1> [#uses=1]
%and.cond = and i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %and.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
define i32 @main4e_like(i32 %argc, i32 %argc2, i32 %argc3, i8** nocapture %argv)
nounwind readnone ssp {
entry:
%and = and i32 %argc, %argc2 ; <i32> [#uses=1]
%tobool = icmp eq i32 %and, %argc2 ; <i1> [#uses=1]
%and2 = and i32 %argc, %argc3 ; <i32> [#uses=1]
%tobool3 = icmp eq i32 %and2, %argc3 ; <i1> [#uses=1]
%and.cond = and i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %and.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
; (icmp ne (A & B), B) | (icmp ne (A & D), D) -> (icmp ne (A & (B|D)), (B|D))
define i32 @main4c(i32 %argc, i8** nocapture %argv) nounwind readnone ssp {
entry:
%and = and i32 %argc, 7 ; <i32> [#uses=1]
%tobool = icmp ne i32 %and, 7 ; <i1> [#uses=1]
%and2 = and i32 %argc, 48 ; <i32> [#uses=1]
%tobool3 = icmp ne i32 %and2, 48 ; <i1> [#uses=1]
%or.cond = or i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %or.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
define i32 @main4d(i32 %argc, i8** nocapture %argv) nounwind readnone ssp {
entry:
%and = and i32 %argc, 7 ; <i32> [#uses=1]
%tobool = icmp ne i32 %and, 7 ; <i1> [#uses=1]
%and2 = and i32 %argc, 16 ; <i32> [#uses=1]
%tobool3 = icmp eq i32 %and2, 0 ; <i1> [#uses=1]
%or.cond = or i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %or.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
define i32 @main4f_like(i32 %argc, i32 %argc2, i32 %argc3, i8** nocapture %argv)
nounwind readnone ssp {
entry:
%and = and i32 %argc, %argc2 ; <i32> [#uses=1]
%tobool = icmp ne i32 %and, %argc2 ; <i1> [#uses=1]
%and2 = and i32 %argc, %argc3 ; <i32> [#uses=1]
%tobool3 = icmp ne i32 %and2, %argc3 ; <i1> [#uses=1]
%or.cond = or i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %or.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
; (icmp eq (A & B), A) & (icmp eq (A & D), A) -> (icmp eq (A & (B&D)), A)
define i32 @main5_like(i32 %argc, i32 %argc2, i8** nocapture %argv)
nounwind readnone ssp {
entry:
%and = and i32 %argc, 7 ; <i32> [#uses=1]
%tobool = icmp eq i32 %and, 7 ; <i1> [#uses=1]
%and2 = and i32 %argc2, 7 ; <i32> [#uses=1]
%tobool3 = icmp eq i32 %and2, 7 ; <i1> [#uses=1]
%and.cond = and i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %and.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
define i32 @main5e_like(i32 %argc, i32 %argc2, i32 %argc3, i8** nocapture %argv)
nounwind readnone ssp {
entry:
%and = and i32 %argc, %argc2 ; <i32> [#uses=1]
%tobool = icmp eq i32 %and, %argc ; <i1> [#uses=1]
%and2 = and i32 %argc, %argc3 ; <i32> [#uses=1]
%tobool3 = icmp eq i32 %and2, %argc ; <i1> [#uses=1]
%and.cond = and i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %and.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
; (icmp ne (A & B), A) | (icmp ne (A & D), A) -> (icmp ne (A & (B&D)), A)
define i32 @main5c_like(i32 %argc, i32 %argc2, i8** nocapture %argv)
nounwind readnone ssp {
entry:
%and = and i32 %argc, 7 ; <i32> [#uses=1]
%tobool = icmp ne i32 %and, 7 ; <i1> [#uses=1]
%and2 = and i32 %argc2, 7 ; <i32> [#uses=1]
%tobool3 = icmp ne i32 %and2, 7 ; <i1> [#uses=1]
%or.cond = or i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %or.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
define i32 @main5f_like(i32 %argc, i32 %argc2, i32 %argc3, i8** nocapture %argv)
nounwind readnone ssp {
entry:
%and = and i32 %argc, %argc2 ; <i32> [#uses=1]
%tobool = icmp ne i32 %and, %argc ; <i1> [#uses=1]
%and2 = and i32 %argc, %argc3 ; <i32> [#uses=1]
%tobool3 = icmp ne i32 %and2, %argc ; <i1> [#uses=1]
%or.cond = or i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %or.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
; (icmp eq (A & B), C) & (icmp eq (A & D), E) -> (icmp eq (A & (B|D)), (C|E))
; if B, C, D, E are constant, and it's possible
define i32 @main6(i32 %argc, i8** nocapture %argv) nounwind readnone ssp {
entry:
%and = and i32 %argc, 7 ; <i32> [#uses=1]
%tobool = icmp eq i32 %and, 3 ; <i1> [#uses=1]
%and2 = and i32 %argc, 48 ; <i32> [#uses=1]
%tobool3 = icmp eq i32 %and2, 16 ; <i1> [#uses=1]
%and.cond = and i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %and.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
define i32 @main6b(i32 %argc, i8** nocapture %argv) nounwind readnone ssp {
entry:
%and = and i32 %argc, 7 ; <i32> [#uses=1]
%tobool = icmp eq i32 %and, 3 ; <i1> [#uses=1]
%and2 = and i32 %argc, 16 ; <i32> [#uses=1]
%tobool3 = icmp ne i32 %and2, 0 ; <i1> [#uses=1]
%and.cond = and i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %and.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
; (icmp ne (A & B), C) | (icmp ne (A & D), E) -> (icmp ne (A & (B|D)), (C|E))
; if B, C, D, E are constant, and it's possible
define i32 @main6c(i32 %argc, i8** nocapture %argv) nounwind readnone ssp {
entry:
%and = and i32 %argc, 7 ; <i32> [#uses=1]
%tobool = icmp ne i32 %and, 3 ; <i1> [#uses=1]
%and2 = and i32 %argc, 48 ; <i32> [#uses=1]
%tobool3 = icmp ne i32 %and2, 16 ; <i1> [#uses=1]
%or.cond = or i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %or.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
define i32 @main6d(i32 %argc, i8** nocapture %argv) nounwind readnone ssp {
entry:
%and = and i32 %argc, 7 ; <i32> [#uses=1]
%tobool = icmp ne i32 %and, 3 ; <i1> [#uses=1]
%and2 = and i32 %argc, 16 ; <i32> [#uses=1]
%tobool3 = icmp eq i32 %and2, 0 ; <i1> [#uses=1]
%or.cond = or i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %or.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
; test parameter permutations
; (B & A) == B & (D & A) == D
define i32 @main7a(i32 %argc, i32 %argc2, i32 %argc3, i8** nocapture %argv)
nounwind readnone ssp {
entry:
%and1 = and i32 %argc2, %argc ; <i32> [#uses=1]
%tobool = icmp eq i32 %and1, %argc2 ; <i1> [#uses=1]
%and2 = and i32 %argc3, %argc ; <i32> [#uses=1]
%tobool3 = icmp eq i32 %and2, %argc3 ; <i1> [#uses=1]
%and.cond = and i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %and.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
; B == (A & B) & D == (A & D)
define i32 @main7b(i32 %argc, i32 %argc2, i32 %argc3, i8** nocapture %argv)
nounwind readnone ssp {
entry:
%and1 = and i32 %argc, %argc2 ; <i32> [#uses=1]
%tobool = icmp eq i32 %argc2, %and1 ; <i1> [#uses=1]
%and2 = and i32 %argc, %argc3 ; <i32> [#uses=1]
%tobool3 = icmp eq i32 %argc3, %and2 ; <i1> [#uses=1]
%and.cond = and i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %and.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
; B == (B & A) & D == (D & A)
define i32 @main7c(i32 %argc, i32 %argc2, i32 %argc3, i8** nocapture %argv)
nounwind readnone ssp {
entry:
%and1 = and i32 %argc2, %argc ; <i32> [#uses=1]
%tobool = icmp eq i32 %argc2, %and1 ; <i1> [#uses=1]
%and2 = and i32 %argc3, %argc ; <i32> [#uses=1]
%tobool3 = icmp eq i32 %argc3, %and2 ; <i1> [#uses=1]
%and.cond = and i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %and.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
; (A & (B & C)) == (B & C) & (A & (D & E)) == (D & E)
define i32 @main7d(i32 %argc, i32 %argc2, i32 %argc3,
i32 %argc4, i32 %argc5, i8** nocapture %argv)
nounwind readnone ssp {
entry:
%bc = and i32 %argc2, %argc4 ; <i32> [#uses=1]
%de = and i32 %argc3, %argc5 ; <i32> [#uses=1]
%and1 = and i32 %argc, %bc ; <i32> [#uses=1]
%tobool = icmp eq i32 %and1, %bc ; <i1> [#uses=1]
%and2 = and i32 %argc, %de ; <i32> [#uses=1]
%tobool3 = icmp eq i32 %and2, %de ; <i1> [#uses=1]
%and.cond = and i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %and.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
; ((B & C) & A) == (B & C) & ((D & E) & A) == (D & E)
define i32 @main7e(i32 %argc, i32 %argc2, i32 %argc3,
i32 %argc4, i32 %argc5, i8** nocapture %argv)
nounwind readnone ssp {
entry:
%bc = and i32 %argc2, %argc4 ; <i32> [#uses=1]
%de = and i32 %argc3, %argc5 ; <i32> [#uses=1]
%and1 = and i32 %bc, %argc ; <i32> [#uses=1]
%tobool = icmp eq i32 %and1, %bc ; <i1> [#uses=1]
%and2 = and i32 %de, %argc ; <i32> [#uses=1]
%tobool3 = icmp eq i32 %and2, %de ; <i1> [#uses=1]
%and.cond = and i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %and.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
; (B & C) == (A & (B & C)) & (D & E) == (A & (D & E))
define i32 @main7f(i32 %argc, i32 %argc2, i32 %argc3,
i32 %argc4, i32 %argc5, i8** nocapture %argv)
nounwind readnone ssp {
entry:
%bc = and i32 %argc2, %argc4 ; <i32> [#uses=1]
%de = and i32 %argc3, %argc5 ; <i32> [#uses=1]
%and1 = and i32 %argc, %bc ; <i32> [#uses=1]
%tobool = icmp eq i32 %bc, %and1 ; <i1> [#uses=1]
%and2 = and i32 %argc, %de ; <i32> [#uses=1]
%tobool3 = icmp eq i32 %de, %and2 ; <i1> [#uses=1]
%and.cond = and i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %and.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}
; (B & C) == ((B & C) & A) & (D & E) == ((D & E) & A)
define i32 @main7g(i32 %argc, i32 %argc2, i32 %argc3,
i32 %argc4, i32 %argc5, i8** nocapture %argv)
nounwind readnone ssp {
entry:
%bc = and i32 %argc2, %argc4 ; <i32> [#uses=1]
%de = and i32 %argc3, %argc5 ; <i32> [#uses=1]
%and1 = and i32 %bc, %argc ; <i32> [#uses=1]
%tobool = icmp eq i32 %bc, %and1 ; <i1> [#uses=1]
%and2 = and i32 %de, %argc ; <i32> [#uses=1]
%tobool3 = icmp eq i32 %de, %and2 ; <i1> [#uses=1]
%and.cond = and i1 %tobool, %tobool3 ; <i1> [#uses=1]
%storemerge = select i1 %and.cond, i32 0, i32 1 ; <i32> [#uses=1]
ret i32 %storemerge
}