Index: llvm/include/llvm/IR/PatternMatch.h =================================================================== --- llvm/include/llvm/IR/PatternMatch.h +++ llvm/include/llvm/IR/PatternMatch.h @@ -50,10 +50,75 @@ return const_cast(P).match(V); } +/// Match Value \p V against Pattern \P. Store a number of matched operations +/// with multiple uses into \p UseComplexity if provided. Match fails if the +/// complexity is higher than MaxComplexity. +template +bool match(Val *V, const Pattern &P, int *UseComplexity, + int MaxComplexity = INT_MAX) { + if (MaxComplexity < 0 || !const_cast(P).match(V)) + return false; + int Complexity = (int)P.getUseComplexity(false); + if (UseComplexity) + *UseComplexity = Complexity; + return Complexity <= MaxComplexity; +} + template bool match(ArrayRef Mask, const Pattern &P) { return const_cast(P).match(Mask); } +template +using has_get_use_complexity_t = + decltype(std::declval().getUseComplexity(std::declval())); + +template +std::enable_if_t::value, unsigned> +getMatchUseComplexity(T V, bool Used) { + return V.getUseComplexity(Used); +} + +template +std::enable_if_t::value, unsigned> +getMatchUseComplexity(T V, bool Used) { + return 0; +} + +struct ValueUseTracker { + bool IsOneUse = false; + + template void trackUse(OpTy *V) { IsOneUse = V->hasOneUse(); } + + template bool trackUse(OpTy *V, bool Match) { + if (Match) + trackUse(V); + return Match; + } + + unsigned getUseComplexity(bool Used) const { + return (IsOneUse && !Used) ? 0 : 1; + } +}; + +template +struct UseTracker : ValueUseTracker { + unsigned getUseComplexity(T0 Op, bool Used) const { + Used |= !ValueUseTracker::IsOneUse; + return ValueUseTracker::getUseComplexity(Used) + + getMatchUseComplexity(Op, Used); + } + + unsigned getUseComplexity(T0 Op0, T1 Op1, bool Used) const { + Used |= !ValueUseTracker::IsOneUse; + return getUseComplexity(Op0, Used) + getMatchUseComplexity(Op1, Used); + } + + unsigned getUseComplexity(T0 Op0, T1 Op1, T2 Op2, bool Used) const { + Used |= !ValueUseTracker::IsOneUse; + return getUseComplexity(Op1, Used) + getMatchUseComplexity(Op2, Used); + } +}; + template struct OneUse_match { SubPattern_t SubPattern; @@ -179,16 +244,26 @@ template struct match_combine_or { LTy L; RTy R; + bool IsLeft = false; match_combine_or(const LTy &Left, const RTy &Right) : L(Left), R(Right) {} template bool match(ITy *V) { - if (L.match(V)) + if (L.match(V)) { + IsLeft = true; return true; - if (R.match(V)) + } + if (R.match(V)) { + IsLeft = false; return true; + } return false; } + + unsigned getUseComplexity(bool Used) const { + return IsLeft ? getMatchUseComplexity(L, Used) + : getMatchUseComplexity(R, Used); + } }; template struct match_combine_and { @@ -203,6 +278,11 @@ return true; return false; } + + unsigned getUseComplexity(bool Used) const { + return std::max(getMatchUseComplexity(L, Used), + getMatchUseComplexity(R, Used)); + } }; /// Combine two pattern matchers matching L || R @@ -715,19 +795,34 @@ } }; +template +struct inst_bind_ty : bind_ty, UseTracker { + inst_bind_ty(Class *&V) : bind_ty(V) {} + + template bool match(ITy *V) { + return ValueUseTracker::trackUse(V, bind_ty::match(V)); + } + + unsigned getUseComplexity(bool Used) const { + return UseTracker::getUseComplexity(bind_ty::VR, Used); + } +}; + /// Match a value, capturing it if we match. inline bind_ty m_Value(Value *&V) { return V; } inline bind_ty m_Value(const Value *&V) { return V; } /// Match an instruction, capturing it if we match. -inline bind_ty m_Instruction(Instruction *&I) { return I; } +inline inst_bind_ty m_Instruction(Instruction *&I) { return I; } /// Match a unary operator, capturing it if we match. -inline bind_ty m_UnOp(UnaryOperator *&I) { return I; } +inline inst_bind_ty m_UnOp(UnaryOperator *&I) { return I; } /// Match a binary operator, capturing it if we match. -inline bind_ty m_BinOp(BinaryOperator *&I) { return I; } +inline inst_bind_ty m_BinOp(BinaryOperator *&I) { return I; } /// Match a with overflow intrinsic, capturing it if we match. -inline bind_ty m_WithOverflowInst(WithOverflowInst *&I) { return I; } -inline bind_ty +inline inst_bind_ty m_WithOverflowInst(WithOverflowInst *&I) { + return I; +} +inline inst_bind_ty m_WithOverflowInst(const WithOverflowInst *&I) { return I; } @@ -907,7 +1002,7 @@ // Matcher for any binary operator. // template -struct AnyBinaryOp_match { +struct AnyBinaryOp_match : UseTracker { LHS_t L; RHS_t R; @@ -917,11 +1012,16 @@ template bool match(OpTy *V) { if (auto *I = dyn_cast(V)) - return (L.match(I->getOperand(0)) && R.match(I->getOperand(1))) || - (Commutable && L.match(I->getOperand(1)) && - R.match(I->getOperand(0))); + return ValueUseTracker::trackUse( + V, ((L.match(I->getOperand(0)) && R.match(I->getOperand(1))) || + (Commutable && L.match(I->getOperand(1)) && + R.match(I->getOperand(0))))); return false; } + + unsigned getUseComplexity(bool Used) const { + return UseTracker::getUseComplexity(L, R, Used); + } }; template @@ -933,16 +1033,21 @@ // Matcher for any unary operator. // TODO fuse unary, binary matcher into n-ary matcher // -template struct AnyUnaryOp_match { +template struct AnyUnaryOp_match : UseTracker { OP_t X; AnyUnaryOp_match(const OP_t &X) : X(X) {} template bool match(OpTy *V) { - if (auto *I = dyn_cast(V)) - return X.match(I->getOperand(0)); + if (auto *I = dyn_cast(V)) { + return ValueUseTracker::trackUse(V, X.match(I->getOperand(0))); + } return false; } + + unsigned getUseComplexity(bool Used) const { + return UseTracker::getMatchUseComplexity(X, Used); + } }; template inline AnyUnaryOp_match m_UnOp(const OP_t &X) { @@ -955,7 +1060,7 @@ template -struct BinaryOp_match { +struct BinaryOp_match : UseTracker { LHS_t L; RHS_t R; @@ -978,7 +1083,13 @@ return false; } - template bool match(OpTy *V) { return match(Opcode, V); } + template bool match(OpTy *V) { + return ValueUseTracker::trackUse(V, match(Opcode, V)); + } + + unsigned getUseComplexity(bool Used) const { + return UseTracker::getUseComplexity(L, R, Used); + } }; template @@ -1005,7 +1116,7 @@ return BinaryOp_match(L, R); } -template struct FNeg_match { +template struct FNeg_match : UseTracker { Op_t X; FNeg_match(const Op_t &Op) : X(Op) {} @@ -1014,7 +1125,7 @@ if (!FPMO) return false; if (FPMO->getOpcode() == Instruction::FNeg) - return X.match(FPMO->getOperand(0)); + return ValueUseTracker::trackUse(V, X.match(FPMO->getOperand(0))); if (FPMO->getOpcode() == Instruction::FSub) { if (FPMO->hasNoSignedZeros()) { @@ -1027,11 +1138,15 @@ return false; } - return X.match(FPMO->getOperand(1)); + return ValueUseTracker::trackUse(V, X.match(FPMO->getOperand(1))); } return false; } + + unsigned getUseComplexity(bool Used) const { + return UseTracker::getMatchUseComplexity(X, Used); + } }; /// Match 'fneg X' as 'fsub -0.0, X'. @@ -1134,7 +1249,7 @@ template -struct OverflowingBinaryOp_match { +struct OverflowingBinaryOp_match : UseTracker { LHS_t L; RHS_t R; @@ -1151,10 +1266,15 @@ if ((WrapFlags & OverflowingBinaryOperator::NoSignedWrap) && !Op->hasNoSignedWrap()) return false; - return L.match(Op->getOperand(0)) && R.match(Op->getOperand(1)); + return ValueUseTracker::trackUse(V, L.match(Op->getOperand(0)) && + R.match(Op->getOperand(1))); } return false; } + + unsigned getUseComplexity(bool Used) const { + return UseTracker::getUseComplexity(L, R, Used); + } }; template @@ -1232,7 +1352,8 @@ : BinaryOp_match(LHS, RHS), Opcode(Opcode) {} template bool match(OpTy *V) { - return BinaryOp_match::match(Opcode, V); + return ValueUseTracker::trackUse( + V, BinaryOp_match::match(Opcode, V)); } }; @@ -1247,7 +1368,7 @@ // Class that matches a group of binary opcodes. // template -struct BinOpPred_match : Predicate { +struct BinOpPred_match : Predicate, UseTracker { LHS_t L; RHS_t R; @@ -1255,13 +1376,19 @@ template bool match(OpTy *V) { if (auto *I = dyn_cast(V)) - return this->isOpType(I->getOpcode()) && L.match(I->getOperand(0)) && - R.match(I->getOperand(1)); + return ValueUseTracker::trackUse(V, this->isOpType(I->getOpcode()) && + L.match(I->getOperand(0)) && + R.match(I->getOperand(1))); if (auto *CE = dyn_cast(V)) - return this->isOpType(CE->getOpcode()) && L.match(CE->getOperand(0)) && - R.match(CE->getOperand(1)); + return ValueUseTracker::trackUse(V, this->isOpType(CE->getOpcode()) && + L.match(CE->getOperand(0)) && + R.match(CE->getOperand(1))); return false; } + + unsigned getUseComplexity(bool Used) const { + return UseTracker::getUseComplexity(L, R, Used); + } }; struct is_shift_op { @@ -1353,6 +1480,10 @@ return PEO->isExact() && SubPattern.match(V); return false; } + + unsigned getUseComplexity(bool Used) const { + return SubPattern.getUseComplexity(Used); + } }; template inline Exact_match m_Exact(const T &SubPattern) { @@ -1365,7 +1496,7 @@ template -struct CmpClass_match { +struct CmpClass_match : UseTracker { PredicateTy &Predicate; LHS_t L; RHS_t R; @@ -1379,15 +1510,21 @@ if (auto *I = dyn_cast(V)) { if (L.match(I->getOperand(0)) && R.match(I->getOperand(1))) { Predicate = I->getPredicate(); + ValueUseTracker::trackUse(V); return true; } else if (Commutable && L.match(I->getOperand(1)) && R.match(I->getOperand(0))) { Predicate = I->getSwappedPredicate(); + ValueUseTracker::trackUse(V); return true; } } return false; } + + unsigned getUseComplexity(bool Used) const { + return UseTracker::getUseComplexity(L, R, Used); + } }; template @@ -1413,7 +1550,7 @@ // /// Matches instructions with Opcode and three operands. -template struct OneOps_match { +template struct OneOps_match : UseTracker { T0 Op1; OneOps_match(const T0 &Op1) : Op1(Op1) {} @@ -1421,14 +1558,19 @@ template bool match(OpTy *V) { if (V->getValueID() == Value::InstructionVal + Opcode) { auto *I = cast(V); - return Op1.match(I->getOperand(0)); + return ValueUseTracker::trackUse(V, Op1.match(I->getOperand(0))); } return false; } + + unsigned getUseComplexity(bool Used) const { + return UseTracker::getUseComplexity(Op1, Used); + } }; /// Matches instructions with Opcode and three operands. -template struct TwoOps_match { +template +struct TwoOps_match : UseTracker { T0 Op1; T1 Op2; @@ -1437,15 +1579,20 @@ template bool match(OpTy *V) { if (V->getValueID() == Value::InstructionVal + Opcode) { auto *I = cast(V); - return Op1.match(I->getOperand(0)) && Op2.match(I->getOperand(1)); + return ValueUseTracker::trackUse(V, Op1.match(I->getOperand(0)) && + Op2.match(I->getOperand(1))); } return false; } + + unsigned getUseComplexity(bool Used) const { + return UseTracker::getUseComplexity(Op1, Op2, Used); + } }; /// Matches instructions with Opcode and three operands. template -struct ThreeOps_match { +struct ThreeOps_match : UseTracker { T0 Op1; T1 Op2; T2 Op3; @@ -1456,11 +1603,16 @@ template bool match(OpTy *V) { if (V->getValueID() == Value::InstructionVal + Opcode) { auto *I = cast(V); - return Op1.match(I->getOperand(0)) && Op2.match(I->getOperand(1)) && - Op3.match(I->getOperand(2)); + return ValueUseTracker::trackUse(V, Op1.match(I->getOperand(0)) && + Op2.match(I->getOperand(1)) && + Op3.match(I->getOperand(2))); } return false; } + + unsigned getUseComplexity(bool Used) const { + return UseTracker::getUseComplexity(Op1, Op2, Op3, Used); + } }; /// Matches SelectInst. @@ -1501,7 +1653,8 @@ } /// Matches shuffle. -template struct Shuffle_match { +template +struct Shuffle_match : UseTracker { T0 Op1; T1 Op2; T2 Mask; @@ -1511,11 +1664,16 @@ template bool match(OpTy *V) { if (auto *I = dyn_cast(V)) { - return Op1.match(I->getOperand(0)) && Op2.match(I->getOperand(1)) && - Mask.match(I->getShuffleMask()); + return ValueUseTracker::trackUse(V, Op1.match(I->getOperand(0)) && + Op2.match(I->getOperand(1)) && + Mask.match(I->getShuffleMask())); } return false; } + + unsigned getUseComplexity(bool Used) const { + return UseTracker::getUseComplexity(Op1, Op2, Mask, Used); + } }; struct m_Mask { @@ -1583,16 +1741,22 @@ // Matchers for CastInst classes // -template struct CastClass_match { +template +struct CastClass_match : UseTracker { Op_t Op; CastClass_match(const Op_t &OpMatch) : Op(OpMatch) {} template bool match(OpTy *V) { if (auto *O = dyn_cast(V)) - return O->getOpcode() == Opcode && Op.match(O->getOperand(0)); + return ValueUseTracker::trackUse(V, O->getOpcode() == Opcode && + Op.match(O->getOperand(0))); return false; } + + unsigned getUseComplexity(bool Used) const { + return UseTracker::getUseComplexity(Op, Used); + } }; /// Matches BitCast. @@ -1752,7 +1916,7 @@ template -struct MaxMin_match { +struct MaxMin_match : UseTracker { using PredType = Pred_t; LHS_t L; RHS_t R; @@ -1769,8 +1933,9 @@ (IID == Intrinsic::umax && Pred_t::match(ICmpInst::ICMP_UGT)) || (IID == Intrinsic::umin && Pred_t::match(ICmpInst::ICMP_ULT))) { Value *LHS = II->getOperand(0), *RHS = II->getOperand(1); - return (L.match(LHS) && R.match(RHS)) || - (Commutable && L.match(RHS) && R.match(LHS)); + return ValueUseTracker::trackUse( + V, (L.match(LHS) && R.match(RHS)) || + (Commutable && L.match(RHS) && R.match(LHS))); } } // Look for "(x pred y) ? x : y" or "(x pred y) ? y : x". @@ -1795,8 +1960,13 @@ if (!Pred_t::match(Pred)) return false; // It does! Bind the operands. - return (L.match(LHS) && R.match(RHS)) || - (Commutable && L.match(RHS) && R.match(LHS)); + return ValueUseTracker::trackUse( + V, (L.match(LHS) && R.match(RHS)) || + (Commutable && L.match(RHS) && R.match(LHS))); + } + + unsigned getUseComplexity(bool Used) const { + return UseTracker::getUseComplexity(L, R, Used); } }; @@ -1957,7 +2127,7 @@ // template -struct UAddWithOverflow_match { +struct UAddWithOverflow_match : UseTracker { LHS_t L; RHS_t R; Sum_t S; @@ -1965,7 +2135,7 @@ UAddWithOverflow_match(const LHS_t &L, const RHS_t &R, const Sum_t &S) : L(L), R(R), S(S) {} - template bool match(OpTy *V) { + template bool match_impl(OpTy *V) { Value *ICmpLHS, *ICmpRHS; ICmpInst::Predicate Pred; if (!m_ICmp(Pred, m_Value(ICmpLHS), m_Value(ICmpRHS)).match(V)) @@ -2013,6 +2183,14 @@ return false; } + + template bool match(OpTy *V) { + return ValueUseTracker::trackUse(V, match_impl(V)); + } + + unsigned getUseComplexity(bool Used) const { + return UseTracker::getUseComplexity(L, R, Used); + } }; /// Match an icmp instruction checking for unsigned overflow on addition. @@ -2285,7 +2463,7 @@ return m_c_Xor(V, m_AllOnes()); } -template struct NotForbidUndef_match { +template struct NotForbidUndef_match : UseTracker { ValTy Val; NotForbidUndef_match(const ValTy &V) : Val(V) {} @@ -2296,11 +2474,15 @@ Value *X; const APInt *C; if (m_Xor(m_Value(X), m_APIntForbidUndef(C)).match(V) && C->isAllOnes()) - return Val.match(X); + return ValueUseTracker::trackUse(V, Val.match(X)); if (m_Xor(m_APIntForbidUndef(C), m_Value(X)).match(V) && C->isAllOnes()) - return Val.match(X); + return ValueUseTracker::trackUse(V, Val.match(X)); return false; } + + unsigned getUseComplexity(bool Used) const { + return UseTracker::getUseComplexity(Val, Used); + } }; /// Matches a bitwise 'not' as 'xor V, -1' or 'xor -1, V'. For vectors, the @@ -2360,7 +2542,7 @@ return BinaryOp_match(L, R); } -template struct Signum_match { +template struct Signum_match : UseTracker { Opnd_t Val; Signum_match(const Opnd_t &V) : Val(V) {} @@ -2386,7 +2568,12 @@ auto RHS = m_LShr(m_Neg(m_Value(OpR)), m_SpecificInt(ShiftWidth)); auto Signum = m_Or(LHS, RHS); - return Signum.match(V) && OpL == OpR && Val.match(OpL); + return ValueUseTracker::trackUse(V, Signum.match(V) && OpL == OpR && + Val.match(OpL)); + } + + unsigned getUseComplexity(bool Used) const { + return UseTracker::getUseComplexity(Val, Used); } }; @@ -2400,7 +2587,8 @@ return Signum_match(V); } -template struct ExtractValue_match { +template +struct ExtractValue_match : UseTracker { Opnd_t Val; ExtractValue_match(const Opnd_t &V) : Val(V) {} @@ -2410,10 +2598,14 @@ if (Ind != -1 && !(I->getNumIndices() == 1 && I->getIndices()[0] == (unsigned)Ind)) return false; - return Val.match(I->getAggregateOperand()); + return ValueUseTracker::trackUse(V, Val.match(I->getAggregateOperand())); } return false; } + + unsigned getUseComplexity(bool Used) const { + return UseTracker::getUseComplexity(Val, Used); + } }; /// Match a single index ExtractValue instruction. @@ -2431,7 +2623,8 @@ } /// Matcher for a single index InsertValue instruction. -template struct InsertValue_match { +template +struct InsertValue_match : UseTracker { T0 Op0; T1 Op1; @@ -2439,11 +2632,16 @@ template bool match(OpTy *V) { if (auto *I = dyn_cast(V)) { - return Op0.match(I->getOperand(0)) && Op1.match(I->getOperand(1)) && - I->getNumIndices() == 1 && Ind == I->getIndices()[0]; + return ValueUseTracker::trackUse( + V, Op0.match(I->getOperand(0)) && Op1.match(I->getOperand(1)) && + I->getNumIndices() == 1 && Ind == I->getIndices()[0]); } return false; } + + unsigned getUseComplexity(bool Used) const { + return UseTracker::getUseComplexity(Op0, Op1, Used); + } }; /// Matches a single index InsertValue instruction. @@ -2486,7 +2684,7 @@ } template -struct LogicalOp_match { +struct LogicalOp_match : UseTracker { LHS L; RHS R; @@ -2500,8 +2698,9 @@ if (I->getOpcode() == Opcode) { auto *Op0 = I->getOperand(0); auto *Op1 = I->getOperand(1); - return (L.match(Op0) && R.match(Op1)) || - (Commutable && L.match(Op1) && R.match(Op0)); + return ValueUseTracker::trackUse( + V, (L.match(Op0) && R.match(Op1)) || + (Commutable && L.match(Op1) && R.match(Op0))); } if (auto *Select = dyn_cast(I)) { @@ -2511,19 +2710,25 @@ if (Opcode == Instruction::And) { auto *C = dyn_cast(FVal); if (C && C->isNullValue()) - return (L.match(Cond) && R.match(TVal)) || - (Commutable && L.match(TVal) && R.match(Cond)); + return ValueUseTracker::trackUse( + V, (L.match(Cond) && R.match(TVal)) || + (Commutable && L.match(TVal) && R.match(Cond))); } else { assert(Opcode == Instruction::Or); auto *C = dyn_cast(TVal); if (C && C->isOneValue()) - return (L.match(Cond) && R.match(FVal)) || - (Commutable && L.match(FVal) && R.match(Cond)); + return ValueUseTracker::trackUse( + V, (L.match(Cond) && R.match(FVal)) || + (Commutable && L.match(FVal) && R.match(Cond))); } } return false; } + + unsigned getUseComplexity(bool Used) const { + return UseTracker::getUseComplexity(L, R, Used); + } }; /// Matches L && R either in the form of L & R or L ? R : false. Index: llvm/lib/Transforms/InstCombine/InstCombineAndOrXor.cpp =================================================================== --- llvm/lib/Transforms/InstCombine/InstCombineAndOrXor.cpp +++ llvm/lib/Transforms/InstCombine/InstCombineAndOrXor.cpp @@ -1728,24 +1728,23 @@ Value *Op0 = I.getOperand(0), *Op1 = I.getOperand(1); Value *A, *B, *C, *X, *Y; + int UseComplexity; // (~(A | B) & C) | ... --> ... // (~(A & B) | C) & ... --> ... - // TODO: One use checks are conservative. We just need to check that a total - // number of multiple used values does not exceed reduction - // in operations. if (match(Op0, m_c_BinOp(FlippedOpcode, m_CombineAnd(m_Value(X), m_Not(m_BinOp(Opcode, m_Value(A), m_Value(B)))), - m_Value(C)))) { + m_Value(C)), + &UseComplexity)) { // (~(A | B) & C) | (~(A | C) & B) --> (B ^ C) & ~A // (~(A & B) | C) & (~(A & C) | B) --> ~((B ^ C) & A) if (match(Op1, - m_OneUse(m_c_BinOp(FlippedOpcode, - m_OneUse(m_Not(m_c_BinOp(Opcode, m_Specific(A), - m_Specific(C)))), - m_Specific(B))))) { + m_c_BinOp(FlippedOpcode, + m_Not(m_c_BinOp(Opcode, m_Specific(A), m_Specific(C))), + m_Specific(B)), + nullptr, 4 - UseComplexity)) { Value *Xor = Builder.CreateXor(B, C); return (Opcode == Instruction::Or) ? BinaryOperator::CreateAnd(Xor, Builder.CreateNot(A)) @@ -1755,10 +1754,10 @@ // (~(A | B) & C) | (~(B | C) & A) --> (A ^ C) & ~B // (~(A & B) | C) & (~(B & C) | A) --> ~((A ^ C) & B) if (match(Op1, - m_OneUse(m_c_BinOp(FlippedOpcode, - m_OneUse(m_Not(m_c_BinOp(Opcode, m_Specific(B), - m_Specific(C)))), - m_Specific(A))))) { + m_c_BinOp(FlippedOpcode, + m_Not(m_c_BinOp(Opcode, m_Specific(B), m_Specific(C))), + m_Specific(A)), + nullptr, 4 - UseComplexity)) { Value *Xor = Builder.CreateXor(A, C); return (Opcode == Instruction::Or) ? BinaryOperator::CreateAnd(Xor, Builder.CreateNot(B)) @@ -1767,15 +1766,15 @@ // (~(A | B) & C) | ~(A | C) --> ~((B & C) | A) // (~(A & B) | C) & ~(A & C) --> ~((B | C) & A) - if (match(Op1, m_OneUse(m_Not(m_OneUse( - m_c_BinOp(Opcode, m_Specific(A), m_Specific(C))))))) + if (match(Op1, m_Not(m_c_BinOp(Opcode, m_Specific(A), m_Specific(C))), + nullptr, 3 - UseComplexity)) return BinaryOperator::CreateNot(Builder.CreateBinOp( Opcode, Builder.CreateBinOp(FlippedOpcode, B, C), A)); // (~(A | B) & C) | ~(B | C) --> ~((A & C) | B) // (~(A & B) | C) & ~(B & C) --> ~((A | C) & B) - if (match(Op1, m_OneUse(m_Not(m_OneUse( - m_c_BinOp(Opcode, m_Specific(B), m_Specific(C))))))) + if (match(Op1, m_Not(m_c_BinOp(Opcode, m_Specific(B), m_Specific(C))), + nullptr, 3 - UseComplexity)) return BinaryOperator::CreateNot(Builder.CreateBinOp( Opcode, Builder.CreateBinOp(FlippedOpcode, A, C), B)); @@ -1783,11 +1782,13 @@ // Note, the pattern with swapped and/or is not handled because the // result is more undefined than a source: // (~(A & B) | C) & ~(C & (A ^ B)) --> (A ^ B ^ C) | ~(A | C) is invalid. - if (Opcode == Instruction::Or && Op0->hasOneUse() && - match(Op1, m_OneUse(m_Not(m_CombineAnd( - m_Value(Y), - m_c_BinOp(Opcode, m_Specific(C), - m_c_Xor(m_Specific(A), m_Specific(B)))))))) { + if (Opcode == Instruction::Or && + match( + Op1, + m_Not(m_CombineAnd( + m_Value(Y), m_c_BinOp(Opcode, m_Specific(C), + m_c_Xor(m_Specific(A), m_Specific(B))))), + nullptr, 5 - UseComplexity)) { // X = ~(A | B) // Y = (C | (A ^ B) Value *Or = cast(X)->getOperand(0); @@ -1797,30 +1798,35 @@ // (~A & B & C) | ... --> ... // (~A | B | C) | ... --> ... - // TODO: One use checks are conservative. We just need to check that a total - // number of multiple used values does not exceed reduction - // in operations. if (match(Op0, - m_OneUse(m_c_BinOp(FlippedOpcode, - m_BinOp(FlippedOpcode, m_Value(B), m_Value(C)), - m_CombineAnd(m_Value(X), m_Not(m_Value(A)))))) || - match(Op0, m_OneUse(m_c_BinOp( - FlippedOpcode, - m_c_BinOp(FlippedOpcode, m_Value(C), - m_CombineAnd(m_Value(X), m_Not(m_Value(A)))), - m_Value(B))))) { + m_c_BinOp(FlippedOpcode, + m_BinOp(FlippedOpcode, m_Value(B), m_Value(C)), + m_CombineAnd(m_Value(X), m_Not(m_Value(A)))), + &UseComplexity) || + match(Op0, + m_c_BinOp(FlippedOpcode, + m_c_BinOp(FlippedOpcode, m_Value(C), + m_CombineAnd(m_Value(X), m_Not(m_Value(A)))), + m_Value(B)), + &UseComplexity)) { // X = ~A // (~A & B & C) | ~(A | B | C) --> ~(A | (B ^ C)) // (~A | B | C) & ~(A & B & C) --> (~A | (B ^ C)) - if (match(Op1, m_OneUse(m_Not(m_c_BinOp( - Opcode, m_c_BinOp(Opcode, m_Specific(A), m_Specific(B)), - m_Specific(C))))) || - match(Op1, m_OneUse(m_Not(m_c_BinOp( - Opcode, m_c_BinOp(Opcode, m_Specific(B), m_Specific(C)), - m_Specific(A))))) || - match(Op1, m_OneUse(m_Not(m_c_BinOp( - Opcode, m_c_BinOp(Opcode, m_Specific(A), m_Specific(C)), - m_Specific(B)))))) { + if (match(Op1, + m_Not(m_c_BinOp(Opcode, + m_c_BinOp(Opcode, m_Specific(A), m_Specific(B)), + m_Specific(C))), + nullptr, 5 - UseComplexity) || + match(Op1, + m_Not(m_c_BinOp(Opcode, + m_c_BinOp(Opcode, m_Specific(B), m_Specific(C)), + m_Specific(A))), + nullptr, 5 - UseComplexity) || + match(Op1, + m_Not(m_c_BinOp(Opcode, + m_c_BinOp(Opcode, m_Specific(A), m_Specific(C)), + m_Specific(B))), + nullptr, 5 - UseComplexity)) { Value *Xor = Builder.CreateXor(B, C); return (Opcode == Instruction::Or) ? BinaryOperator::CreateNot(Builder.CreateOr(Xor, A)) @@ -1829,16 +1835,16 @@ // (~A & B & C) | ~(A | B) --> (C | ~B) & ~A // (~A | B | C) & ~(A & B) --> (C & ~B) | ~A - if (match(Op1, m_OneUse(m_Not(m_OneUse( - m_c_BinOp(Opcode, m_Specific(A), m_Specific(B))))))) + if (match(Op1, m_Not(m_c_BinOp(Opcode, m_Specific(A), m_Specific(B))), + nullptr, 3 - UseComplexity)) return BinaryOperator::Create( FlippedOpcode, Builder.CreateBinOp(Opcode, C, Builder.CreateNot(B)), X); // (~A & B & C) | ~(A | C) --> (B | ~C) & ~A // (~A | B | C) & ~(A & C) --> (B & ~C) | ~A - if (match(Op1, m_OneUse(m_Not(m_OneUse( - m_c_BinOp(Opcode, m_Specific(A), m_Specific(C))))))) + if (match(Op1, m_Not(m_c_BinOp(Opcode, m_Specific(A), m_Specific(C))), + nullptr, 3 - UseComplexity)) return BinaryOperator::Create( FlippedOpcode, Builder.CreateBinOp(Opcode, B, Builder.CreateNot(C)), X); Index: llvm/test/Transforms/InstCombine/and-xor-or.ll =================================================================== --- llvm/test/Transforms/InstCombine/and-xor-or.ll +++ llvm/test/Transforms/InstCombine/and-xor-or.ll @@ -914,13 +914,11 @@ define i32 @or_not_and_extra_not_use2(i32 %a, i32 %b, i32 %c) { ; CHECK-LABEL: @or_not_and_extra_not_use2( -; CHECK-NEXT: [[OR1:%.*]] = or i32 [[A:%.*]], [[B:%.*]] -; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[OR1]], -1 -; CHECK-NEXT: [[AND1:%.*]] = and i32 [[NOT1]], [[C:%.*]] -; CHECK-NEXT: [[OR2:%.*]] = or i32 [[A]], [[C]] +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[A:%.*]], [[C:%.*]] ; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[OR2]], -1 -; CHECK-NEXT: [[AND2:%.*]] = and i32 [[NOT2]], [[B]] -; CHECK-NEXT: [[OR3:%.*]] = or i32 [[AND1]], [[AND2]] +; CHECK-NEXT: [[TMP1:%.*]] = xor i32 [[B:%.*]], [[C]] +; CHECK-NEXT: [[TMP2:%.*]] = xor i32 [[A]], -1 +; CHECK-NEXT: [[OR3:%.*]] = and i32 [[TMP1]], [[TMP2]] ; CHECK-NEXT: call void @use(i32 [[NOT2]]) ; CHECK-NEXT: ret i32 [[OR3]] ; @@ -959,13 +957,12 @@ define i32 @or_not_and_extra_and_use2(i32 %a, i32 %b, i32 %c) { ; CHECK-LABEL: @or_not_and_extra_and_use2( -; CHECK-NEXT: [[OR1:%.*]] = or i32 [[A:%.*]], [[B:%.*]] -; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[OR1]], -1 -; CHECK-NEXT: [[AND1:%.*]] = and i32 [[NOT1]], [[C:%.*]] -; CHECK-NEXT: [[OR2:%.*]] = or i32 [[A]], [[C]] +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[A:%.*]], [[C:%.*]] ; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[OR2]], -1 -; CHECK-NEXT: [[AND2:%.*]] = and i32 [[NOT2]], [[B]] -; CHECK-NEXT: [[OR3:%.*]] = or i32 [[AND1]], [[AND2]] +; CHECK-NEXT: [[AND2:%.*]] = and i32 [[NOT2]], [[B:%.*]] +; CHECK-NEXT: [[TMP1:%.*]] = xor i32 [[B]], [[C]] +; CHECK-NEXT: [[TMP2:%.*]] = xor i32 [[A]], -1 +; CHECK-NEXT: [[OR3:%.*]] = and i32 [[TMP1]], [[TMP2]] ; CHECK-NEXT: call void @use(i32 [[AND2]]) ; CHECK-NEXT: ret i32 [[OR3]] ; @@ -1020,6 +1017,233 @@ ret i32 %or3 } +; Even though there are only 2 uses all the instructions have to remain +; since used values are outermost subexpressions. +define i32 @or_not_and_2_uses_rhs_and_lhs(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @or_not_and_2_uses_rhs_and_lhs( +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[A:%.*]], [[B:%.*]] +; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[OR1]], -1 +; CHECK-NEXT: [[AND1:%.*]] = and i32 [[NOT1]], [[C:%.*]] +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[A]], [[C]] +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[OR2]], -1 +; CHECK-NEXT: [[AND2:%.*]] = and i32 [[NOT2]], [[B]] +; CHECK-NEXT: [[OR3:%.*]] = or i32 [[AND1]], [[AND2]] +; CHECK-NEXT: call void @use(i32 [[AND1]]) +; CHECK-NEXT: call void @use(i32 [[AND2]]) +; CHECK-NEXT: ret i32 [[OR3]] +; + %or1 = or i32 %a, %b + %not1 = xor i32 %or1, -1 + %and1 = and i32 %not1, %c + + %or2 = or i32 %a, %c + %not2 = xor i32 %or2, -1 + %and2 = and i32 %not2, %b + %or3 = or i32 %and1, %and2 + + call void @use(i32 %and1) + call void @use(i32 %and2) + ret i32 %or3 +} + +define i32 @or_not_and_4_uses1(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @or_not_and_4_uses1( +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[A:%.*]], [[B:%.*]] +; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[OR1]], -1 +; CHECK-NEXT: [[AND1:%.*]] = and i32 [[NOT1]], [[C:%.*]] +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[A]], [[C]] +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[OR2]], -1 +; CHECK-NEXT: [[AND2:%.*]] = and i32 [[NOT2]], [[B]] +; CHECK-NEXT: [[OR3:%.*]] = or i32 [[AND1]], [[AND2]] +; CHECK-NEXT: call void @use(i32 [[OR1]]) +; CHECK-NEXT: call void @use(i32 [[NOT1]]) +; CHECK-NEXT: call void @use(i32 [[OR2]]) +; CHECK-NEXT: call void @use(i32 [[AND2]]) +; CHECK-NEXT: ret i32 [[OR3]] +; + %or1 = or i32 %a, %b + %not1 = xor i32 %or1, -1 + %and1 = and i32 %not1, %c + + %or2 = or i32 %a, %c + %not2 = xor i32 %or2, -1 + %and2 = and i32 %not2, %b + %or3 = or i32 %and1, %and2 + + call void @use(i32 %or1) + call void @use(i32 %not1) + + call void @use(i32 %or2) + call void @use(i32 %and2) + ret i32 %or3 +} + +define i32 @or_not_and_4_uses2(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @or_not_and_4_uses2( +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[A:%.*]], [[B:%.*]] +; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[OR1]], -1 +; CHECK-NEXT: [[AND1:%.*]] = and i32 [[NOT1]], [[C:%.*]] +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[A]], [[C]] +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[OR2]], -1 +; CHECK-NEXT: [[AND2:%.*]] = and i32 [[NOT2]], [[B]] +; CHECK-NEXT: [[OR3:%.*]] = or i32 [[AND1]], [[AND2]] +; CHECK-NEXT: call void @use(i32 [[OR1]]) +; CHECK-NEXT: call void @use(i32 [[AND1]]) +; CHECK-NEXT: call void @use(i32 [[OR2]]) +; CHECK-NEXT: call void @use(i32 [[AND2]]) +; CHECK-NEXT: ret i32 [[OR3]] +; + %or1 = or i32 %a, %b + %not1 = xor i32 %or1, -1 + %and1 = and i32 %not1, %c + + %or2 = or i32 %a, %c + %not2 = xor i32 %or2, -1 + %and2 = and i32 %not2, %b + %or3 = or i32 %and1, %and2 + + call void @use(i32 %or1) + call void @use(i32 %and1) + + call void @use(i32 %or2) + call void @use(i32 %and2) + ret i32 %or3 +} + +define i32 @or_not_and_4_uses3(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @or_not_and_4_uses3( +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[A:%.*]], [[B:%.*]] +; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[OR1]], -1 +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[A]], [[C:%.*]] +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[OR2]], -1 +; CHECK-NEXT: [[TMP1:%.*]] = xor i32 [[B]], [[C]] +; CHECK-NEXT: [[TMP2:%.*]] = xor i32 [[A]], -1 +; CHECK-NEXT: [[OR3:%.*]] = and i32 [[TMP1]], [[TMP2]] +; CHECK-NEXT: call void @use(i32 [[OR1]]) +; CHECK-NEXT: call void @use(i32 [[NOT1]]) +; CHECK-NEXT: call void @use(i32 [[OR2]]) +; CHECK-NEXT: call void @use(i32 [[NOT2]]) +; CHECK-NEXT: ret i32 [[OR3]] +; + %or1 = or i32 %a, %b + %not1 = xor i32 %or1, -1 + %and1 = and i32 %not1, %c + + %or2 = or i32 %a, %c + %not2 = xor i32 %or2, -1 + %and2 = and i32 %not2, %b + %or3 = or i32 %and1, %and2 + + call void @use(i32 %or1) + call void @use(i32 %not1) + + call void @use(i32 %or2) + call void @use(i32 %not2) + ret i32 %or3 +} + +define i32 @or_not_and_5_uses1(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @or_not_and_5_uses1( +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[A:%.*]], [[B:%.*]] +; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[OR1]], -1 +; CHECK-NEXT: [[AND1:%.*]] = and i32 [[NOT1]], [[C:%.*]] +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[A]], [[C]] +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[OR2]], -1 +; CHECK-NEXT: [[AND2:%.*]] = and i32 [[NOT2]], [[B]] +; CHECK-NEXT: [[OR3:%.*]] = or i32 [[AND1]], [[AND2]] +; CHECK-NEXT: call void @use(i32 [[OR1]]) +; CHECK-NEXT: call void @use(i32 [[NOT1]]) +; CHECK-NEXT: call void @use(i32 [[AND1]]) +; CHECK-NEXT: call void @use(i32 [[OR2]]) +; CHECK-NEXT: call void @use(i32 [[AND2]]) +; CHECK-NEXT: ret i32 [[OR3]] +; + %or1 = or i32 %a, %b + %not1 = xor i32 %or1, -1 + %and1 = and i32 %not1, %c + + %or2 = or i32 %a, %c + %not2 = xor i32 %or2, -1 + %and2 = and i32 %not2, %b + %or3 = or i32 %and1, %and2 + + call void @use(i32 %or1) + call void @use(i32 %not1) + call void @use(i32 %and1) + + call void @use(i32 %or2) + call void @use(i32 %and2) + ret i32 %or3 +} + +define i32 @or_not_and_5_uses2(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @or_not_and_5_uses2( +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[A:%.*]], [[B:%.*]] +; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[OR1]], -1 +; CHECK-NEXT: [[AND1:%.*]] = and i32 [[NOT1]], [[C:%.*]] +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[A]], [[C]] +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[OR2]], -1 +; CHECK-NEXT: [[AND2:%.*]] = and i32 [[NOT2]], [[B]] +; CHECK-NEXT: [[OR3:%.*]] = or i32 [[AND1]], [[AND2]] +; CHECK-NEXT: call void @use(i32 [[OR1]]) +; CHECK-NEXT: call void @use(i32 [[AND1]]) +; CHECK-NEXT: call void @use(i32 [[OR2]]) +; CHECK-NEXT: call void @use(i32 [[NOT2]]) +; CHECK-NEXT: call void @use(i32 [[AND2]]) +; CHECK-NEXT: ret i32 [[OR3]] +; + %or1 = or i32 %a, %b + %not1 = xor i32 %or1, -1 + %and1 = and i32 %not1, %c + + %or2 = or i32 %a, %c + %not2 = xor i32 %or2, -1 + %and2 = and i32 %not2, %b + %or3 = or i32 %and1, %and2 + + call void @use(i32 %or1) + call void @use(i32 %and1) + + call void @use(i32 %or2) + call void @use(i32 %not2) + call void @use(i32 %and2) + ret i32 %or3 +} + +define i32 @or_not_and_5_uses3(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @or_not_and_5_uses3( +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[A:%.*]], [[B:%.*]] +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[A]], [[C:%.*]] +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[OR2]], -1 +; CHECK-NEXT: [[AND2:%.*]] = and i32 [[NOT2]], [[B]] +; CHECK-NEXT: [[TMP1:%.*]] = xor i32 [[B]], [[C]] +; CHECK-NEXT: [[TMP2:%.*]] = xor i32 [[A]], -1 +; CHECK-NEXT: [[OR3:%.*]] = and i32 [[TMP1]], [[TMP2]] +; CHECK-NEXT: call void @use(i32 [[OR1]]) +; CHECK-NEXT: call void @use(i32 [[OR2]]) +; CHECK-NEXT: call void @use(i32 [[NOT2]]) +; CHECK-NEXT: call void @use(i32 [[AND2]]) +; CHECK-NEXT: call void @use(i32 [[OR3]]) +; CHECK-NEXT: ret i32 [[OR3]] +; + %or1 = or i32 %a, %b + %not1 = xor i32 %or1, -1 + %and1 = and i32 %not1, %c + + %or2 = or i32 %a, %c + %not2 = xor i32 %or2, -1 + %and2 = and i32 %not2, %b + %or3 = or i32 %and1, %and2 + + call void @use(i32 %or1) + + call void @use(i32 %or2) + call void @use(i32 %not2) + call void @use(i32 %and2) + call void @use(i32 %or3) + ret i32 %or3 +} + define i32 @or_not_and_wrong_c(i32 %a, i32 %b, i32 %c, i32 %d) { ; CHECK-LABEL: @or_not_and_wrong_c( ; CHECK-NEXT: [[OR1:%.*]] = or i32 [[A:%.*]], [[B:%.*]] @@ -1277,13 +1501,11 @@ define i32 @and_not_or_extra_not_use2(i32 %a, i32 %b, i32 %c) { ; CHECK-LABEL: @and_not_or_extra_not_use2( -; CHECK-NEXT: [[AND1:%.*]] = and i32 [[A:%.*]], [[B:%.*]] -; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[AND1]], -1 -; CHECK-NEXT: [[OR1:%.*]] = or i32 [[NOT1]], [[C:%.*]] -; CHECK-NEXT: [[AND2:%.*]] = and i32 [[A]], [[C]] +; CHECK-NEXT: [[AND2:%.*]] = and i32 [[A:%.*]], [[C:%.*]] ; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[AND2]], -1 -; CHECK-NEXT: [[OR2:%.*]] = or i32 [[NOT2]], [[B]] -; CHECK-NEXT: [[AND3:%.*]] = and i32 [[OR1]], [[OR2]] +; CHECK-NEXT: [[TMP1:%.*]] = xor i32 [[B:%.*]], [[C]] +; CHECK-NEXT: [[TMP2:%.*]] = and i32 [[TMP1]], [[A]] +; CHECK-NEXT: [[AND3:%.*]] = xor i32 [[TMP2]], -1 ; CHECK-NEXT: call void @use(i32 [[NOT2]]) ; CHECK-NEXT: ret i32 [[AND3]] ; @@ -1322,13 +1544,12 @@ define i32 @and_not_or_extra_and_use2(i32 %a, i32 %b, i32 %c) { ; CHECK-LABEL: @and_not_or_extra_and_use2( -; CHECK-NEXT: [[AND1:%.*]] = and i32 [[A:%.*]], [[B:%.*]] -; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[AND1]], -1 -; CHECK-NEXT: [[OR1:%.*]] = or i32 [[NOT1]], [[C:%.*]] -; CHECK-NEXT: [[AND2:%.*]] = and i32 [[A]], [[C]] +; CHECK-NEXT: [[AND2:%.*]] = and i32 [[A:%.*]], [[C:%.*]] ; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[AND2]], -1 -; CHECK-NEXT: [[OR2:%.*]] = or i32 [[NOT2]], [[B]] -; CHECK-NEXT: [[AND3:%.*]] = and i32 [[OR1]], [[OR2]] +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[NOT2]], [[B:%.*]] +; CHECK-NEXT: [[TMP1:%.*]] = xor i32 [[B]], [[C]] +; CHECK-NEXT: [[TMP2:%.*]] = and i32 [[TMP1]], [[A]] +; CHECK-NEXT: [[AND3:%.*]] = xor i32 [[TMP2]], -1 ; CHECK-NEXT: call void @use(i32 [[OR2]]) ; CHECK-NEXT: ret i32 [[AND3]] ; @@ -1383,6 +1604,126 @@ ret i32 %and3 } +define i32 @and_not_or_4_uses1(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @and_not_or_4_uses1( +; CHECK-NEXT: [[AND1:%.*]] = and i32 [[A:%.*]], [[B:%.*]] +; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[AND1]], -1 +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[NOT1]], [[C:%.*]] +; CHECK-NEXT: [[AND2:%.*]] = and i32 [[A]], [[C]] +; CHECK-NEXT: [[TMP1:%.*]] = xor i32 [[B]], [[C]] +; CHECK-NEXT: [[TMP2:%.*]] = and i32 [[TMP1]], [[A]] +; CHECK-NEXT: [[AND3:%.*]] = xor i32 [[TMP2]], -1 +; CHECK-NEXT: call void @use(i32 [[AND1]]) +; CHECK-NEXT: call void @use(i32 [[NOT1]]) +; CHECK-NEXT: call void @use(i32 [[OR1]]) +; CHECK-NEXT: call void @use(i32 [[AND2]]) +; CHECK-NEXT: ret i32 [[AND3]] +; + %and1 = and i32 %a, %b + %not1 = xor i32 %and1, -1 + %or1 = or i32 %not1, %c + %and2 = and i32 %a, %c + %not2 = xor i32 %and2, -1 + %or2 = or i32 %not2, %b + %and3 = and i32 %or1, %or2 + call void @use(i32 %and1) + call void @use(i32 %not1) + call void @use(i32 %or1) + call void @use(i32 %and2) + ret i32 %and3 +} + +define i32 @and_not_or_4_uses2(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @and_not_or_4_uses2( +; CHECK-NEXT: [[AND1:%.*]] = and i32 [[A:%.*]], [[B:%.*]] +; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[AND1]], -1 +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[NOT1]], [[C:%.*]] +; CHECK-NEXT: [[AND2:%.*]] = and i32 [[A]], [[C]] +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[AND2]], -1 +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[NOT2]], [[B]] +; CHECK-NEXT: [[AND3:%.*]] = and i32 [[OR1]], [[OR2]] +; CHECK-NEXT: call void @use(i32 [[NOT1]]) +; CHECK-NEXT: call void @use(i32 [[AND2]]) +; CHECK-NEXT: call void @use(i32 [[OR1]]) +; CHECK-NEXT: call void @use(i32 [[OR2]]) +; CHECK-NEXT: ret i32 [[AND3]] +; + %and1 = and i32 %a, %b + %not1 = xor i32 %and1, -1 + %or1 = or i32 %not1, %c + %and2 = and i32 %a, %c + %not2 = xor i32 %and2, -1 + %or2 = or i32 %not2, %b + %and3 = and i32 %or1, %or2 + call void @use(i32 %not1) + call void @use(i32 %and2) + call void @use(i32 %or1) + call void @use(i32 %or2) + ret i32 %and3 +} + +define i32 @and_not_or_5_uses1(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @and_not_or_5_uses1( +; CHECK-NEXT: [[AND1:%.*]] = and i32 [[A:%.*]], [[B:%.*]] +; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[AND1]], -1 +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[NOT1]], [[C:%.*]] +; CHECK-NEXT: [[AND2:%.*]] = and i32 [[A]], [[C]] +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[AND2]], -1 +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[NOT2]], [[B]] +; CHECK-NEXT: [[AND3:%.*]] = and i32 [[OR1]], [[OR2]] +; CHECK-NEXT: call void @use(i32 [[AND1]]) +; CHECK-NEXT: call void @use(i32 [[NOT1]]) +; CHECK-NEXT: call void @use(i32 [[OR1]]) +; CHECK-NEXT: call void @use(i32 [[AND2]]) +; CHECK-NEXT: call void @use(i32 [[NOT2]]) +; CHECK-NEXT: ret i32 [[AND3]] +; + %and1 = and i32 %a, %b + %not1 = xor i32 %and1, -1 + %or1 = or i32 %not1, %c + %and2 = and i32 %a, %c + %not2 = xor i32 %and2, -1 + %or2 = or i32 %not2, %b + %and3 = and i32 %or1, %or2 + call void @use(i32 %and1) + call void @use(i32 %not1) + call void @use(i32 %or1) + call void @use(i32 %and2) + call void @use(i32 %not2) + ret i32 %and3 +} + +define i32 @and_not_or_5_uses2(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @and_not_or_5_uses2( +; CHECK-NEXT: [[AND1:%.*]] = and i32 [[A:%.*]], [[B:%.*]] +; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[AND1]], -1 +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[NOT1]], [[C:%.*]] +; CHECK-NEXT: [[AND2:%.*]] = and i32 [[A]], [[C]] +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[AND2]], -1 +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[NOT2]], [[B]] +; CHECK-NEXT: [[AND3:%.*]] = and i32 [[OR1]], [[OR2]] +; CHECK-NEXT: call void @use(i32 [[AND1]]) +; CHECK-NEXT: call void @use(i32 [[NOT1]]) +; CHECK-NEXT: call void @use(i32 [[OR1]]) +; CHECK-NEXT: call void @use(i32 [[AND2]]) +; CHECK-NEXT: call void @use(i32 [[OR2]]) +; CHECK-NEXT: ret i32 [[AND3]] +; + %and1 = and i32 %a, %b + %not1 = xor i32 %and1, -1 + %or1 = or i32 %not1, %c + %and2 = and i32 %a, %c + %not2 = xor i32 %and2, -1 + %or2 = or i32 %not2, %b + %and3 = and i32 %or1, %or2 + call void @use(i32 %and1) + call void @use(i32 %not1) + call void @use(i32 %or1) + call void @use(i32 %and2) + call void @use(i32 %or2) + ret i32 %and3 +} + define i32 @and_not_or_wrong_c(i32 %a, i32 %b, i32 %c, i32 %d) { ; CHECK-LABEL: @and_not_or_wrong_c( ; CHECK-NEXT: [[AND1:%.*]] = and i32 [[A:%.*]], [[B:%.*]] @@ -1563,10 +1904,9 @@ ; CHECK-LABEL: @or_and_not_not_extra_not_use1( ; CHECK-NEXT: [[OR1:%.*]] = or i32 [[B:%.*]], [[A:%.*]] ; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[OR1]], -1 -; CHECK-NEXT: [[OR2:%.*]] = or i32 [[A]], [[C:%.*]] -; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[OR2]], -1 -; CHECK-NEXT: [[AND:%.*]] = and i32 [[NOT2]], [[B]] -; CHECK-NEXT: [[OR3:%.*]] = or i32 [[AND]], [[NOT1]] +; CHECK-NEXT: [[TMP1:%.*]] = and i32 [[C:%.*]], [[B]] +; CHECK-NEXT: [[TMP2:%.*]] = or i32 [[TMP1]], [[A]] +; CHECK-NEXT: [[OR3:%.*]] = xor i32 [[TMP2]], -1 ; CHECK-NEXT: call void @use(i32 [[NOT1]]) ; CHECK-NEXT: ret i32 [[OR3]] ; @@ -1621,14 +1961,12 @@ ret i32 %or3 } -define i32 @or_and_not_not_extra_or_use1(i32 %a, i32 %b, i32 %c) { -; CHECK-LABEL: @or_and_not_not_extra_or_use1( -; CHECK-NEXT: [[OR1:%.*]] = or i32 [[B:%.*]], [[A:%.*]] -; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[OR1]], -1 -; CHECK-NEXT: [[OR2:%.*]] = or i32 [[A]], [[C:%.*]] -; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[OR2]], -1 -; CHECK-NEXT: [[AND:%.*]] = and i32 [[NOT2]], [[B]] -; CHECK-NEXT: [[OR3:%.*]] = or i32 [[AND]], [[NOT1]] +define i32 @or_and_not_not_extra_or_use1(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @or_and_not_not_extra_or_use1( +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[B:%.*]], [[A:%.*]] +; CHECK-NEXT: [[TMP1:%.*]] = and i32 [[C:%.*]], [[B]] +; CHECK-NEXT: [[TMP2:%.*]] = or i32 [[TMP1]], [[A]] +; CHECK-NEXT: [[OR3:%.*]] = xor i32 [[TMP2]], -1 ; CHECK-NEXT: call void @use(i32 [[OR1]]) ; CHECK-NEXT: ret i32 [[OR3]] ; @@ -1661,8 +1999,7 @@ ret i32 %or3 } -; Check the use limit. It can be adjusted in the future in terms of -; LHS and RHS uses distribution to be more flexible. +; Check the use limit. define i32 @or_and_not_not_2_extra_uses(i32 %a, i32 %b, i32 %c) { ; CHECK-LABEL: @or_and_not_not_2_extra_uses( ; CHECK-NEXT: [[OR1:%.*]] = or i32 [[B:%.*]], [[A:%.*]] @@ -1862,10 +2199,9 @@ ; CHECK-LABEL: @and_or_not_not_extra_not_use1( ; CHECK-NEXT: [[AND1:%.*]] = and i32 [[B:%.*]], [[A:%.*]] ; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[AND1]], -1 -; CHECK-NEXT: [[AND2:%.*]] = and i32 [[A]], [[C:%.*]] -; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[AND2]], -1 -; CHECK-NEXT: [[OR:%.*]] = or i32 [[NOT2]], [[B]] -; CHECK-NEXT: [[AND3:%.*]] = xor i32 [[AND1]], [[OR]] +; CHECK-NEXT: [[TMP1:%.*]] = or i32 [[C:%.*]], [[B]] +; CHECK-NEXT: [[TMP2:%.*]] = and i32 [[TMP1]], [[A]] +; CHECK-NEXT: [[AND3:%.*]] = xor i32 [[TMP2]], -1 ; CHECK-NEXT: call void @use(i32 [[NOT1]]) ; CHECK-NEXT: ret i32 [[AND3]] ; @@ -1923,10 +2259,9 @@ define i32 @and_or_not_not_extra_or_use1(i32 %a, i32 %b, i32 %c) { ; CHECK-LABEL: @and_or_not_not_extra_or_use1( ; CHECK-NEXT: [[AND1:%.*]] = and i32 [[B:%.*]], [[A:%.*]] -; CHECK-NEXT: [[AND2:%.*]] = and i32 [[A]], [[C:%.*]] -; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[AND2]], -1 -; CHECK-NEXT: [[OR:%.*]] = or i32 [[NOT2]], [[B]] -; CHECK-NEXT: [[AND3:%.*]] = xor i32 [[AND1]], [[OR]] +; CHECK-NEXT: [[TMP1:%.*]] = or i32 [[C:%.*]], [[B]] +; CHECK-NEXT: [[TMP2:%.*]] = and i32 [[TMP1]], [[A]] +; CHECK-NEXT: [[AND3:%.*]] = xor i32 [[TMP2]], -1 ; CHECK-NEXT: call void @use(i32 [[AND1]]) ; CHECK-NEXT: ret i32 [[AND3]] ; @@ -2188,8 +2523,8 @@ ; CHECK-NEXT: [[AND1:%.*]] = and i32 [[NOT1]], [[A:%.*]] ; CHECK-NEXT: [[XOR1:%.*]] = xor i32 [[B]], [[C]] ; CHECK-NEXT: [[OR2:%.*]] = or i32 [[XOR1]], [[A]] -; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[OR2]], -1 -; CHECK-NEXT: [[OR3:%.*]] = or i32 [[AND1]], [[NOT2]] +; CHECK-NEXT: [[TMP1:%.*]] = and i32 [[OR1]], [[OR2]] +; CHECK-NEXT: [[OR3:%.*]] = xor i32 [[TMP1]], -1 ; CHECK-NEXT: call void @use(i32 [[AND1]]) ; CHECK-NEXT: ret i32 [[OR3]] ; @@ -2249,12 +2584,253 @@ define i32 @and_not_or_or_not_or_xor_use6(i32 %a, i32 %b, i32 %c) { ; CHECK-LABEL: @and_not_or_or_not_or_xor_use6( ; CHECK-NEXT: [[OR1:%.*]] = or i32 [[B:%.*]], [[C:%.*]] +; CHECK-NEXT: [[XOR1:%.*]] = xor i32 [[B]], [[C]] +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[XOR1]], [[A:%.*]] +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[OR2]], -1 +; CHECK-NEXT: [[TMP1:%.*]] = and i32 [[OR1]], [[OR2]] +; CHECK-NEXT: [[OR3:%.*]] = xor i32 [[TMP1]], -1 +; CHECK-NEXT: call void @use(i32 [[NOT2]]) +; CHECK-NEXT: ret i32 [[OR3]] +; + %or1 = or i32 %b, %c + %not1 = xor i32 %or1, -1 + %and1 = and i32 %not1, %a + %xor1 = xor i32 %b, %c + %or2 = or i32 %xor1, %a + %not2 = xor i32 %or2, -1 + %or3 = or i32 %and1, %not2 + call void @use(i32 %not2) + ret i32 %or3 +} + +define i32 @and_not_or_or_not_or_xor_4_uses1(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @and_not_or_or_not_or_xor_4_uses1( +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[B:%.*]], [[C:%.*]] +; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[OR1]], -1 +; CHECK-NEXT: [[XOR1:%.*]] = xor i32 [[B]], [[C]] +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[XOR1]], [[A:%.*]] +; CHECK-NEXT: [[TMP1:%.*]] = and i32 [[OR1]], [[OR2]] +; CHECK-NEXT: [[OR3:%.*]] = xor i32 [[TMP1]], -1 +; CHECK-NEXT: call void @use(i32 [[OR1]]) +; CHECK-NEXT: call void @use(i32 [[NOT1]]) +; CHECK-NEXT: call void @use(i32 [[XOR1]]) +; CHECK-NEXT: call void @use(i32 [[OR2]]) +; CHECK-NEXT: ret i32 [[OR3]] +; + %or1 = or i32 %b, %c + %not1 = xor i32 %or1, -1 + %and1 = and i32 %not1, %a + %xor1 = xor i32 %b, %c + %or2 = or i32 %xor1, %a + %not2 = xor i32 %or2, -1 + %or3 = or i32 %and1, %not2 + call void @use(i32 %or1) + call void @use(i32 %not1) + call void @use(i32 %xor1) + call void @use(i32 %or2) + ret i32 %or3 +} + +define i32 @and_not_or_or_not_or_xor_5_uses1(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @and_not_or_or_not_or_xor_5_uses1( +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[B:%.*]], [[C:%.*]] +; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[OR1]], -1 +; CHECK-NEXT: [[AND1:%.*]] = and i32 [[NOT1]], [[A:%.*]] +; CHECK-NEXT: [[XOR1:%.*]] = xor i32 [[B]], [[C]] +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[XOR1]], [[A]] +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[OR2]], -1 +; CHECK-NEXT: [[OR3:%.*]] = or i32 [[AND1]], [[NOT2]] +; CHECK-NEXT: call void @use(i32 [[NOT1]]) +; CHECK-NEXT: call void @use(i32 [[AND1]]) +; CHECK-NEXT: call void @use(i32 [[XOR1]]) +; CHECK-NEXT: call void @use(i32 [[OR2]]) +; CHECK-NEXT: call void @use(i32 [[NOT2]]) +; CHECK-NEXT: ret i32 [[OR3]] +; + %or1 = or i32 %b, %c + %not1 = xor i32 %or1, -1 + %and1 = and i32 %not1, %a + %xor1 = xor i32 %b, %c + %or2 = or i32 %xor1, %a + %not2 = xor i32 %or2, -1 + %or3 = or i32 %and1, %not2 + call void @use(i32 %not1) + call void @use(i32 %and1) + call void @use(i32 %xor1) + call void @use(i32 %or2) + call void @use(i32 %not2) + ret i32 %or3 +} + +define i32 @and_not_or_or_not_or_xor_5_uses2(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @and_not_or_or_not_or_xor_5_uses2( +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[B:%.*]], [[C:%.*]] +; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[OR1]], -1 +; CHECK-NEXT: [[AND1:%.*]] = and i32 [[NOT1]], [[A:%.*]] +; CHECK-NEXT: [[XOR1:%.*]] = xor i32 [[B]], [[C]] +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[XOR1]], [[A]] +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[OR2]], -1 +; CHECK-NEXT: [[OR3:%.*]] = or i32 [[AND1]], [[NOT2]] +; CHECK-NEXT: call void @use(i32 [[OR1]]) +; CHECK-NEXT: call void @use(i32 [[AND1]]) +; CHECK-NEXT: call void @use(i32 [[XOR1]]) +; CHECK-NEXT: call void @use(i32 [[OR2]]) +; CHECK-NEXT: call void @use(i32 [[NOT2]]) +; CHECK-NEXT: ret i32 [[OR3]] +; + %or1 = or i32 %b, %c + %not1 = xor i32 %or1, -1 + %and1 = and i32 %not1, %a + %xor1 = xor i32 %b, %c + %or2 = or i32 %xor1, %a + %not2 = xor i32 %or2, -1 + %or3 = or i32 %and1, %not2 + call void @use(i32 %or1) + call void @use(i32 %and1) + call void @use(i32 %xor1) + call void @use(i32 %or2) + call void @use(i32 %not2) + ret i32 %or3 +} + +define i32 @and_not_or_or_not_or_xor_5_uses3(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @and_not_or_or_not_or_xor_5_uses3( +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[B:%.*]], [[C:%.*]] +; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[OR1]], -1 +; CHECK-NEXT: [[XOR1:%.*]] = xor i32 [[B]], [[C]] +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[XOR1]], [[A:%.*]] +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[OR2]], -1 +; CHECK-NEXT: [[TMP1:%.*]] = and i32 [[OR1]], [[OR2]] +; CHECK-NEXT: [[OR3:%.*]] = xor i32 [[TMP1]], -1 +; CHECK-NEXT: call void @use(i32 [[OR1]]) +; CHECK-NEXT: call void @use(i32 [[NOT1]]) +; CHECK-NEXT: call void @use(i32 [[XOR1]]) +; CHECK-NEXT: call void @use(i32 [[OR2]]) +; CHECK-NEXT: call void @use(i32 [[NOT2]]) +; CHECK-NEXT: ret i32 [[OR3]] +; + %or1 = or i32 %b, %c + %not1 = xor i32 %or1, -1 + %and1 = and i32 %not1, %a + %xor1 = xor i32 %b, %c + %or2 = or i32 %xor1, %a + %not2 = xor i32 %or2, -1 + %or3 = or i32 %and1, %not2 + call void @use(i32 %or1) + call void @use(i32 %not1) + call void @use(i32 %xor1) + call void @use(i32 %or2) + call void @use(i32 %not2) + ret i32 %or3 +} + +define i32 @and_not_or_or_not_or_xor_5_uses4(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @and_not_or_or_not_or_xor_5_uses4( +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[B:%.*]], [[C:%.*]] +; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[OR1]], -1 +; CHECK-NEXT: [[AND1:%.*]] = and i32 [[NOT1]], [[A:%.*]] +; CHECK-NEXT: [[XOR1:%.*]] = xor i32 [[B]], [[C]] +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[XOR1]], [[A]] +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[OR2]], -1 +; CHECK-NEXT: [[OR3:%.*]] = or i32 [[AND1]], [[NOT2]] +; CHECK-NEXT: call void @use(i32 [[OR1]]) +; CHECK-NEXT: call void @use(i32 [[NOT1]]) +; CHECK-NEXT: call void @use(i32 [[AND1]]) +; CHECK-NEXT: call void @use(i32 [[OR2]]) +; CHECK-NEXT: call void @use(i32 [[NOT2]]) +; CHECK-NEXT: ret i32 [[OR3]] +; + %or1 = or i32 %b, %c + %not1 = xor i32 %or1, -1 + %and1 = and i32 %not1, %a + %xor1 = xor i32 %b, %c + %or2 = or i32 %xor1, %a + %not2 = xor i32 %or2, -1 + %or3 = or i32 %and1, %not2 + call void @use(i32 %or1) + call void @use(i32 %not1) + call void @use(i32 %and1) + call void @use(i32 %or2) + call void @use(i32 %not2) + ret i32 %or3 +} + +define i32 @and_not_or_or_not_or_xor_5_uses5(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @and_not_or_or_not_or_xor_5_uses5( +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[B:%.*]], [[C:%.*]] +; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[OR1]], -1 +; CHECK-NEXT: [[AND1:%.*]] = and i32 [[NOT1]], [[A:%.*]] +; CHECK-NEXT: [[XOR1:%.*]] = xor i32 [[B]], [[C]] +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[XOR1]], [[A]] +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[OR2]], -1 +; CHECK-NEXT: [[OR3:%.*]] = or i32 [[AND1]], [[NOT2]] +; CHECK-NEXT: call void @use(i32 [[OR1]]) +; CHECK-NEXT: call void @use(i32 [[NOT1]]) +; CHECK-NEXT: call void @use(i32 [[AND1]]) +; CHECK-NEXT: call void @use(i32 [[XOR1]]) +; CHECK-NEXT: call void @use(i32 [[NOT2]]) +; CHECK-NEXT: ret i32 [[OR3]] +; + %or1 = or i32 %b, %c + %not1 = xor i32 %or1, -1 + %and1 = and i32 %not1, %a + %xor1 = xor i32 %b, %c + %or2 = or i32 %xor1, %a + %not2 = xor i32 %or2, -1 + %or3 = or i32 %and1, %not2 + call void @use(i32 %or1) + call void @use(i32 %not1) + call void @use(i32 %and1) + call void @use(i32 %xor1) + call void @use(i32 %not2) + ret i32 %or3 +} + +define i32 @and_not_or_or_not_or_xor_5_uses6(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @and_not_or_or_not_or_xor_5_uses6( +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[B:%.*]], [[C:%.*]] +; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[OR1]], -1 +; CHECK-NEXT: [[AND1:%.*]] = and i32 [[NOT1]], [[A:%.*]] +; CHECK-NEXT: [[XOR1:%.*]] = xor i32 [[B]], [[C]] +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[XOR1]], [[A]] +; CHECK-NEXT: [[TMP1:%.*]] = and i32 [[OR1]], [[OR2]] +; CHECK-NEXT: [[OR3:%.*]] = xor i32 [[TMP1]], -1 +; CHECK-NEXT: call void @use(i32 [[OR1]]) +; CHECK-NEXT: call void @use(i32 [[NOT1]]) +; CHECK-NEXT: call void @use(i32 [[AND1]]) +; CHECK-NEXT: call void @use(i32 [[XOR1]]) +; CHECK-NEXT: call void @use(i32 [[OR2]]) +; CHECK-NEXT: ret i32 [[OR3]] +; + %or1 = or i32 %b, %c + %not1 = xor i32 %or1, -1 + %and1 = and i32 %not1, %a + %xor1 = xor i32 %b, %c + %or2 = or i32 %xor1, %a + %not2 = xor i32 %or2, -1 + %or3 = or i32 %and1, %not2 + call void @use(i32 %or1) + call void @use(i32 %not1) + call void @use(i32 %and1) + call void @use(i32 %xor1) + call void @use(i32 %or2) + ret i32 %or3 +} + +define i32 @and_not_or_or_not_or_xor_6_uses(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @and_not_or_or_not_or_xor_6_uses( +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[B:%.*]], [[C:%.*]] ; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[OR1]], -1 ; CHECK-NEXT: [[AND1:%.*]] = and i32 [[NOT1]], [[A:%.*]] ; CHECK-NEXT: [[XOR1:%.*]] = xor i32 [[B]], [[C]] ; CHECK-NEXT: [[OR2:%.*]] = or i32 [[XOR1]], [[A]] ; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[OR2]], -1 ; CHECK-NEXT: [[OR3:%.*]] = or i32 [[AND1]], [[NOT2]] +; CHECK-NEXT: call void @use(i32 [[OR1]]) +; CHECK-NEXT: call void @use(i32 [[NOT1]]) +; CHECK-NEXT: call void @use(i32 [[AND1]]) +; CHECK-NEXT: call void @use(i32 [[XOR1]]) +; CHECK-NEXT: call void @use(i32 [[OR2]]) ; CHECK-NEXT: call void @use(i32 [[NOT2]]) ; CHECK-NEXT: ret i32 [[OR3]] ; @@ -2265,6 +2841,11 @@ %or2 = or i32 %xor1, %a %not2 = xor i32 %or2, -1 %or3 = or i32 %and1, %not2 + call void @use(i32 %or1) + call void @use(i32 %not1) + call void @use(i32 %and1) + call void @use(i32 %xor1) + call void @use(i32 %or2) call void @use(i32 %not2) ret i32 %or3 } @@ -2737,10 +3318,9 @@ ; CHECK-NEXT: [[OR1:%.*]] = or i32 [[B:%.*]], [[A:%.*]] ; CHECK-NEXT: [[OR2:%.*]] = or i32 [[OR1]], [[C:%.*]] ; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[OR2]], -1 -; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[A]], -1 -; CHECK-NEXT: [[AND1:%.*]] = and i32 [[NOT2]], [[B]] -; CHECK-NEXT: [[AND2:%.*]] = and i32 [[AND1]], [[C]] -; CHECK-NEXT: [[OR3:%.*]] = or i32 [[AND2]], [[NOT1]] +; CHECK-NEXT: [[TMP1:%.*]] = xor i32 [[C]], [[B]] +; CHECK-NEXT: [[TMP2:%.*]] = or i32 [[TMP1]], [[A]] +; CHECK-NEXT: [[OR3:%.*]] = xor i32 [[TMP2]], -1 ; CHECK-NEXT: call void @use(i32 [[NOT1]]) ; CHECK-NEXT: ret i32 [[OR3]] ; @@ -2798,13 +3378,12 @@ define i32 @not_and_and_or_not_or_or_use6(i32 %a, i32 %b, i32 %c) { ; CHECK-LABEL: @not_and_and_or_not_or_or_use6( -; CHECK-NEXT: [[OR1:%.*]] = or i32 [[B:%.*]], [[A:%.*]] -; CHECK-NEXT: [[OR2:%.*]] = or i32 [[OR1]], [[C:%.*]] -; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[OR2]], -1 -; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[A]], -1 -; CHECK-NEXT: [[AND1:%.*]] = and i32 [[NOT2]], [[B]] -; CHECK-NEXT: [[AND2:%.*]] = and i32 [[AND1]], [[C]] -; CHECK-NEXT: [[OR3:%.*]] = or i32 [[AND2]], [[NOT1]] +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[A:%.*]], -1 +; CHECK-NEXT: [[AND1:%.*]] = and i32 [[NOT2]], [[B:%.*]] +; CHECK-NEXT: [[AND2:%.*]] = and i32 [[AND1]], [[C:%.*]] +; CHECK-NEXT: [[TMP1:%.*]] = xor i32 [[C]], [[B]] +; CHECK-NEXT: [[TMP2:%.*]] = or i32 [[TMP1]], [[A]] +; CHECK-NEXT: [[OR3:%.*]] = xor i32 [[TMP2]], -1 ; CHECK-NEXT: call void @use(i32 [[AND2]]) ; CHECK-NEXT: ret i32 [[OR3]] ; @@ -3027,9 +3606,8 @@ ; CHECK-NEXT: [[AND2:%.*]] = and i32 [[AND1]], [[C:%.*]] ; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[AND2]], -1 ; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[A]], -1 -; CHECK-NEXT: [[OR1:%.*]] = or i32 [[NOT2]], [[B]] -; CHECK-NEXT: [[OR2:%.*]] = or i32 [[OR1]], [[C]] -; CHECK-NEXT: [[AND3:%.*]] = xor i32 [[AND2]], [[OR2]] +; CHECK-NEXT: [[TMP1:%.*]] = xor i32 [[C]], [[B]] +; CHECK-NEXT: [[AND3:%.*]] = or i32 [[TMP1]], [[NOT2]] ; CHECK-NEXT: call void @use(i32 [[NOT1]]) ; CHECK-NEXT: ret i32 [[AND3]] ; @@ -3085,12 +3663,129 @@ define i32 @not_or_or_and_not_and_and_use6(i32 %a, i32 %b, i32 %c) { ; CHECK-LABEL: @not_or_or_and_not_and_and_use6( +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[A:%.*]], -1 +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[NOT2]], [[B:%.*]] +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[OR1]], [[C:%.*]] +; CHECK-NEXT: [[TMP1:%.*]] = xor i32 [[C]], [[B]] +; CHECK-NEXT: [[AND3:%.*]] = or i32 [[TMP1]], [[NOT2]] +; CHECK-NEXT: call void @use(i32 [[OR2]]) +; CHECK-NEXT: ret i32 [[AND3]] +; + %and1 = and i32 %b, %a + %and2 = and i32 %and1, %c + %not1 = xor i32 %and2, -1 + %not2 = xor i32 %a, -1 + %or1 = or i32 %not2, %b + %or2 = or i32 %or1, %c + %and3 = and i32 %or2, %not1 + call void @use(i32 %or2) + ret i32 %and3 +} + +define i32 @not_or_or_and_not_and_and_4_uses1(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @not_or_or_and_not_and_and_4_uses1( +; CHECK-NEXT: [[AND1:%.*]] = and i32 [[B:%.*]], [[A:%.*]] +; CHECK-NEXT: [[AND2:%.*]] = and i32 [[AND1]], [[C:%.*]] +; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[AND2]], -1 +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[A]], -1 +; CHECK-NEXT: [[TMP1:%.*]] = xor i32 [[C]], [[B]] +; CHECK-NEXT: [[AND3:%.*]] = or i32 [[TMP1]], [[NOT2]] +; CHECK-NEXT: call void @use(i32 [[AND1]]) +; CHECK-NEXT: call void @use(i32 [[AND2]]) +; CHECK-NEXT: call void @use(i32 [[NOT1]]) +; CHECK-NEXT: call void @use(i32 [[NOT2]]) +; CHECK-NEXT: ret i32 [[AND3]] +; + %and1 = and i32 %b, %a + %and2 = and i32 %and1, %c + %not1 = xor i32 %and2, -1 + %not2 = xor i32 %a, -1 + %or1 = or i32 %not2, %b + %or2 = or i32 %or1, %c + %and3 = and i32 %or2, %not1 + call void @use(i32 %and1) + call void @use(i32 %and2) + call void @use(i32 %not1) + call void @use(i32 %not2) + ret i32 %and3 +} + +define i32 @not_or_or_and_not_and_and_5_uses1(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @not_or_or_and_not_and_and_5_uses1( +; CHECK-NEXT: [[AND1:%.*]] = and i32 [[B:%.*]], [[A:%.*]] +; CHECK-NEXT: [[AND2:%.*]] = and i32 [[AND1]], [[C:%.*]] +; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[AND2]], -1 +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[A]], -1 +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[NOT2]], [[B]] +; CHECK-NEXT: [[TMP1:%.*]] = xor i32 [[C]], [[B]] +; CHECK-NEXT: [[AND3:%.*]] = or i32 [[TMP1]], [[NOT2]] +; CHECK-NEXT: call void @use(i32 [[AND1]]) +; CHECK-NEXT: call void @use(i32 [[AND2]]) +; CHECK-NEXT: call void @use(i32 [[NOT1]]) +; CHECK-NEXT: call void @use(i32 [[NOT2]]) +; CHECK-NEXT: call void @use(i32 [[OR1]]) +; CHECK-NEXT: ret i32 [[AND3]] +; + %and1 = and i32 %b, %a + %and2 = and i32 %and1, %c + %not1 = xor i32 %and2, -1 + %not2 = xor i32 %a, -1 + %or1 = or i32 %not2, %b + %or2 = or i32 %or1, %c + %and3 = and i32 %or2, %not1 + call void @use(i32 %and1) + call void @use(i32 %and2) + call void @use(i32 %not1) + call void @use(i32 %not2) + call void @use(i32 %or1) + ret i32 %and3 +} + +define i32 @not_or_or_and_not_and_and_5_uses2(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @not_or_or_and_not_and_and_5_uses2( +; CHECK-NEXT: [[AND1:%.*]] = and i32 [[B:%.*]], [[A:%.*]] +; CHECK-NEXT: [[AND2:%.*]] = and i32 [[AND1]], [[C:%.*]] +; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[AND2]], -1 +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[A]], -1 +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[NOT2]], [[B]] +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[OR1]], [[C]] +; CHECK-NEXT: [[AND3:%.*]] = xor i32 [[AND2]], [[OR2]] +; CHECK-NEXT: call void @use(i32 [[AND2]]) +; CHECK-NEXT: call void @use(i32 [[NOT1]]) +; CHECK-NEXT: call void @use(i32 [[NOT2]]) +; CHECK-NEXT: call void @use(i32 [[OR1]]) +; CHECK-NEXT: call void @use(i32 [[OR2]]) +; CHECK-NEXT: ret i32 [[AND3]] +; + %and1 = and i32 %b, %a + %and2 = and i32 %and1, %c + %not1 = xor i32 %and2, -1 + %not2 = xor i32 %a, -1 + %or1 = or i32 %not2, %b + %or2 = or i32 %or1, %c + %and3 = and i32 %or2, %not1 + call void @use(i32 %and2) + call void @use(i32 %not1) + call void @use(i32 %not2) + call void @use(i32 %or1) + call void @use(i32 %or2) + ret i32 %and3 +} + +define i32 @not_or_or_and_not_and_and_6_uses(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @not_or_or_and_not_and_and_6_uses( ; CHECK-NEXT: [[AND1:%.*]] = and i32 [[B:%.*]], [[A:%.*]] ; CHECK-NEXT: [[AND2:%.*]] = and i32 [[AND1]], [[C:%.*]] +; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[AND2]], -1 ; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[A]], -1 ; CHECK-NEXT: [[OR1:%.*]] = or i32 [[NOT2]], [[B]] ; CHECK-NEXT: [[OR2:%.*]] = or i32 [[OR1]], [[C]] ; CHECK-NEXT: [[AND3:%.*]] = xor i32 [[AND2]], [[OR2]] +; CHECK-NEXT: call void @use(i32 [[AND1]]) +; CHECK-NEXT: call void @use(i32 [[AND2]]) +; CHECK-NEXT: call void @use(i32 [[NOT1]]) +; CHECK-NEXT: call void @use(i32 [[NOT2]]) +; CHECK-NEXT: call void @use(i32 [[OR1]]) ; CHECK-NEXT: call void @use(i32 [[OR2]]) ; CHECK-NEXT: ret i32 [[AND3]] ; @@ -3101,6 +3796,11 @@ %or1 = or i32 %not2, %b %or2 = or i32 %or1, %c %and3 = and i32 %or2, %not1 + call void @use(i32 %and1) + call void @use(i32 %and2) + call void @use(i32 %not1) + call void @use(i32 %not2) + call void @use(i32 %or1) call void @use(i32 %or2) ret i32 %and3 } @@ -3292,11 +3992,10 @@ define i32 @not_and_and_or_no_or_use5(i32 %a, i32 %b, i32 %c) { ; CHECK-LABEL: @not_and_and_or_no_or_use5( ; CHECK-NEXT: [[OR1:%.*]] = or i32 [[B:%.*]], [[A:%.*]] -; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[OR1]], -1 ; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[A]], -1 -; CHECK-NEXT: [[AND1:%.*]] = and i32 [[NOT2]], [[B]] -; CHECK-NEXT: [[AND2:%.*]] = and i32 [[AND1]], [[C:%.*]] -; CHECK-NEXT: [[OR2:%.*]] = or i32 [[AND2]], [[NOT1]] +; CHECK-NEXT: [[TMP1:%.*]] = xor i32 [[B]], -1 +; CHECK-NEXT: [[TMP2:%.*]] = or i32 [[TMP1]], [[C:%.*]] +; CHECK-NEXT: [[OR2:%.*]] = and i32 [[TMP2]], [[NOT2]] ; CHECK-NEXT: call void @use(i32 [[OR1]]) ; CHECK-NEXT: ret i32 [[OR2]] ; @@ -3315,9 +4014,9 @@ ; CHECK-NEXT: [[OR1:%.*]] = or i32 [[B:%.*]], [[A:%.*]] ; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[OR1]], -1 ; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[A]], -1 -; CHECK-NEXT: [[AND1:%.*]] = and i32 [[NOT2]], [[B]] -; CHECK-NEXT: [[AND2:%.*]] = and i32 [[AND1]], [[C:%.*]] -; CHECK-NEXT: [[OR2:%.*]] = or i32 [[AND2]], [[NOT1]] +; CHECK-NEXT: [[TMP1:%.*]] = xor i32 [[B]], -1 +; CHECK-NEXT: [[TMP2:%.*]] = or i32 [[TMP1]], [[C:%.*]] +; CHECK-NEXT: [[OR2:%.*]] = and i32 [[TMP2]], [[NOT2]] ; CHECK-NEXT: call void @use(i32 [[NOT1]]) ; CHECK-NEXT: ret i32 [[OR2]] ; @@ -3353,12 +4052,113 @@ define i32 @not_and_and_or_no_or_use8(i32 %a, i32 %b, i32 %c) { ; CHECK-LABEL: @not_and_and_or_no_or_use8( +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[A:%.*]], -1 +; CHECK-NEXT: [[AND1:%.*]] = and i32 [[NOT2]], [[B:%.*]] +; CHECK-NEXT: [[AND2:%.*]] = and i32 [[AND1]], [[C:%.*]] +; CHECK-NEXT: [[TMP1:%.*]] = xor i32 [[B]], -1 +; CHECK-NEXT: [[TMP2:%.*]] = or i32 [[TMP1]], [[C]] +; CHECK-NEXT: [[OR2:%.*]] = and i32 [[TMP2]], [[NOT2]] +; CHECK-NEXT: call void @use(i32 [[AND2]]) +; CHECK-NEXT: ret i32 [[OR2]] +; + %or1 = or i32 %b, %a + %not1 = xor i32 %or1, -1 + %not2 = xor i32 %a, -1 + %and1 = and i32 %not2, %b + %and2 = and i32 %and1, %c + %or2 = or i32 %and2, %not1 + call void @use(i32 %and2) + ret i32 %or2 +} + +define i32 @not_and_and_or_no_or_3_uses1(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @not_and_and_or_no_or_3_uses1( +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[B:%.*]], [[A:%.*]] +; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[OR1]], -1 +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[A]], -1 +; CHECK-NEXT: [[TMP1:%.*]] = xor i32 [[B]], -1 +; CHECK-NEXT: [[TMP2:%.*]] = or i32 [[TMP1]], [[C:%.*]] +; CHECK-NEXT: [[OR2:%.*]] = and i32 [[TMP2]], [[NOT2]] +; CHECK-NEXT: call void @use(i32 [[OR1]]) +; CHECK-NEXT: call void @use(i32 [[NOT1]]) +; CHECK-NEXT: call void @use(i32 [[NOT2]]) +; CHECK-NEXT: ret i32 [[OR2]] +; + %or1 = or i32 %b, %a + %not1 = xor i32 %or1, -1 + %not2 = xor i32 %a, -1 + %and1 = and i32 %not2, %b + %and2 = and i32 %and1, %c + %or2 = or i32 %and2, %not1 + call void @use(i32 %or1) + call void @use(i32 %not1) + call void @use(i32 %not2) + ret i32 %or2 +} + +define i32 @not_and_and_or_no_or_3_uses2(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @not_and_and_or_no_or_3_uses2( +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[A:%.*]], -1 +; CHECK-NEXT: [[AND1:%.*]] = and i32 [[NOT2]], [[B:%.*]] +; CHECK-NEXT: [[AND2:%.*]] = and i32 [[AND1]], [[C:%.*]] +; CHECK-NEXT: [[TMP1:%.*]] = xor i32 [[B]], -1 +; CHECK-NEXT: [[TMP2:%.*]] = or i32 [[TMP1]], [[C]] +; CHECK-NEXT: [[OR2:%.*]] = and i32 [[TMP2]], [[NOT2]] +; CHECK-NEXT: call void @use(i32 [[AND1]]) +; CHECK-NEXT: call void @use(i32 [[AND2]]) +; CHECK-NEXT: call void @use(i32 [[OR2]]) +; CHECK-NEXT: ret i32 [[OR2]] +; + %or1 = or i32 %b, %a + %not1 = xor i32 %or1, -1 + %not2 = xor i32 %a, -1 + %and1 = and i32 %not2, %b + %and2 = and i32 %and1, %c + %or2 = or i32 %and2, %not1 + call void @use(i32 %and1) + call void @use(i32 %and2) + call void @use(i32 %or2) + ret i32 %or2 +} + +define i32 @not_and_and_or_no_or_4_uses1(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @not_and_and_or_no_or_4_uses1( +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[B:%.*]], [[A:%.*]] +; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[OR1]], -1 +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[A]], -1 +; CHECK-NEXT: [[AND1:%.*]] = and i32 [[NOT2]], [[B]] +; CHECK-NEXT: [[AND2:%.*]] = and i32 [[AND1]], [[C:%.*]] +; CHECK-NEXT: [[OR2:%.*]] = or i32 [[AND2]], [[NOT1]] +; CHECK-NEXT: call void @use(i32 [[OR1]]) +; CHECK-NEXT: call void @use(i32 [[NOT1]]) +; CHECK-NEXT: call void @use(i32 [[NOT2]]) +; CHECK-NEXT: call void @use(i32 [[AND1]]) +; CHECK-NEXT: ret i32 [[OR2]] +; + %or1 = or i32 %b, %a + %not1 = xor i32 %or1, -1 + %not2 = xor i32 %a, -1 + %and1 = and i32 %not2, %b + %and2 = and i32 %and1, %c + %or2 = or i32 %and2, %not1 + call void @use(i32 %or1) + call void @use(i32 %not1) + call void @use(i32 %not2) + call void @use(i32 %and1) + ret i32 %or2 +} + +define i32 @not_and_and_or_no_or_4_uses2(i32 %a, i32 %b, i32 %c) { +; CHECK-LABEL: @not_and_and_or_no_or_4_uses2( ; CHECK-NEXT: [[OR1:%.*]] = or i32 [[B:%.*]], [[A:%.*]] ; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[OR1]], -1 ; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[A]], -1 ; CHECK-NEXT: [[AND1:%.*]] = and i32 [[NOT2]], [[B]] ; CHECK-NEXT: [[AND2:%.*]] = and i32 [[AND1]], [[C:%.*]] ; CHECK-NEXT: [[OR2:%.*]] = or i32 [[AND2]], [[NOT1]] +; CHECK-NEXT: call void @use(i32 [[OR1]]) +; CHECK-NEXT: call void @use(i32 [[NOT1]]) +; CHECK-NEXT: call void @use(i32 [[NOT2]]) ; CHECK-NEXT: call void @use(i32 [[AND2]]) ; CHECK-NEXT: ret i32 [[OR2]] ; @@ -3368,6 +4168,9 @@ %and1 = and i32 %not2, %b %and2 = and i32 %and1, %c %or2 = or i32 %and2, %not1 + call void @use(i32 %or1) + call void @use(i32 %not1) + call void @use(i32 %not2) call void @use(i32 %and2) ret i32 %or2 } @@ -3559,11 +4362,10 @@ define i32 @not_or_or_and_no_and_use5(i32 %a, i32 %b, i32 %c) { ; CHECK-LABEL: @not_or_or_and_no_and_use5( ; CHECK-NEXT: [[AND1:%.*]] = and i32 [[B:%.*]], [[A:%.*]] -; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[AND1]], -1 ; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[A]], -1 -; CHECK-NEXT: [[OR1:%.*]] = or i32 [[NOT2]], [[B]] -; CHECK-NEXT: [[OR2:%.*]] = or i32 [[OR1]], [[C:%.*]] -; CHECK-NEXT: [[AND2:%.*]] = and i32 [[OR2]], [[NOT1]] +; CHECK-NEXT: [[TMP1:%.*]] = xor i32 [[B]], -1 +; CHECK-NEXT: [[TMP2:%.*]] = and i32 [[TMP1]], [[C:%.*]] +; CHECK-NEXT: [[AND2:%.*]] = or i32 [[TMP2]], [[NOT2]] ; CHECK-NEXT: call void @use(i32 [[AND1]]) ; CHECK-NEXT: ret i32 [[AND2]] ; @@ -3582,9 +4384,9 @@ ; CHECK-NEXT: [[AND1:%.*]] = and i32 [[B:%.*]], [[A:%.*]] ; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[AND1]], -1 ; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[A]], -1 -; CHECK-NEXT: [[OR1:%.*]] = or i32 [[NOT2]], [[B]] -; CHECK-NEXT: [[OR2:%.*]] = or i32 [[OR1]], [[C:%.*]] -; CHECK-NEXT: [[AND2:%.*]] = and i32 [[OR2]], [[NOT1]] +; CHECK-NEXT: [[TMP1:%.*]] = xor i32 [[B]], -1 +; CHECK-NEXT: [[TMP2:%.*]] = and i32 [[TMP1]], [[C:%.*]] +; CHECK-NEXT: [[AND2:%.*]] = or i32 [[TMP2]], [[NOT2]] ; CHECK-NEXT: call void @use(i32 [[NOT1]]) ; CHECK-NEXT: ret i32 [[AND2]] ; @@ -3620,12 +4422,12 @@ define i32 @not_or_or_and_no_and_use8(i32 %a, i32 %b, i32 %c) { ; CHECK-LABEL: @not_or_or_and_no_and_use8( -; CHECK-NEXT: [[AND1:%.*]] = and i32 [[B:%.*]], [[A:%.*]] -; CHECK-NEXT: [[NOT1:%.*]] = xor i32 [[AND1]], -1 -; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[A]], -1 -; CHECK-NEXT: [[OR1:%.*]] = or i32 [[NOT2]], [[B]] +; CHECK-NEXT: [[NOT2:%.*]] = xor i32 [[A:%.*]], -1 +; CHECK-NEXT: [[OR1:%.*]] = or i32 [[NOT2]], [[B:%.*]] ; CHECK-NEXT: [[OR2:%.*]] = or i32 [[OR1]], [[C:%.*]] -; CHECK-NEXT: [[AND2:%.*]] = and i32 [[OR2]], [[NOT1]] +; CHECK-NEXT: [[TMP1:%.*]] = xor i32 [[B]], -1 +; CHECK-NEXT: [[TMP2:%.*]] = and i32 [[TMP1]], [[C]] +; CHECK-NEXT: [[AND2:%.*]] = or i32 [[TMP2]], [[NOT2]] ; CHECK-NEXT: call void @use(i32 [[OR2]]) ; CHECK-NEXT: ret i32 [[AND2]] ;