1*c9157d92SDimitry Andric //===- DomConditionCache.cpp ----------------------------------------------===//
2*c9157d92SDimitry Andric //
3*c9157d92SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*c9157d92SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
5*c9157d92SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*c9157d92SDimitry Andric //
7*c9157d92SDimitry Andric //===----------------------------------------------------------------------===//
8*c9157d92SDimitry Andric 
9*c9157d92SDimitry Andric #include "llvm/Analysis/DomConditionCache.h"
10*c9157d92SDimitry Andric #include "llvm/IR/PatternMatch.h"
11*c9157d92SDimitry Andric 
12*c9157d92SDimitry Andric using namespace llvm;
13*c9157d92SDimitry Andric using namespace llvm::PatternMatch;
14*c9157d92SDimitry Andric 
15*c9157d92SDimitry Andric // TODO: This code is very similar to findAffectedValues() in
16*c9157d92SDimitry Andric // AssumptionCache, but currently specialized to just the patterns that
17*c9157d92SDimitry Andric // computeKnownBits() supports, and without the notion of result elem indices
18*c9157d92SDimitry Andric // that are AC specific. Deduplicate this code once we have a clearer picture
19*c9157d92SDimitry Andric // of how much they can be shared.
findAffectedValues(Value * Cond,SmallVectorImpl<Value * > & Affected)20*c9157d92SDimitry Andric static void findAffectedValues(Value *Cond,
21*c9157d92SDimitry Andric                                SmallVectorImpl<Value *> &Affected) {
22*c9157d92SDimitry Andric   auto AddAffected = [&Affected](Value *V) {
23*c9157d92SDimitry Andric     if (isa<Argument>(V) || isa<GlobalValue>(V)) {
24*c9157d92SDimitry Andric       Affected.push_back(V);
25*c9157d92SDimitry Andric     } else if (auto *I = dyn_cast<Instruction>(V)) {
26*c9157d92SDimitry Andric       Affected.push_back(I);
27*c9157d92SDimitry Andric 
28*c9157d92SDimitry Andric       // Peek through unary operators to find the source of the condition.
29*c9157d92SDimitry Andric       Value *Op;
30*c9157d92SDimitry Andric       if (match(I, m_PtrToInt(m_Value(Op)))) {
31*c9157d92SDimitry Andric         if (isa<Instruction>(Op) || isa<Argument>(Op))
32*c9157d92SDimitry Andric           Affected.push_back(Op);
33*c9157d92SDimitry Andric       }
34*c9157d92SDimitry Andric     }
35*c9157d92SDimitry Andric   };
36*c9157d92SDimitry Andric 
37*c9157d92SDimitry Andric   ICmpInst::Predicate Pred;
38*c9157d92SDimitry Andric   Value *A;
39*c9157d92SDimitry Andric   if (match(Cond, m_ICmp(Pred, m_Value(A), m_Constant()))) {
40*c9157d92SDimitry Andric     AddAffected(A);
41*c9157d92SDimitry Andric 
42*c9157d92SDimitry Andric     if (ICmpInst::isEquality(Pred)) {
43*c9157d92SDimitry Andric       Value *X;
44*c9157d92SDimitry Andric       // (X & C) or (X | C) or (X ^ C).
45*c9157d92SDimitry Andric       // (X << C) or (X >>_s C) or (X >>_u C).
46*c9157d92SDimitry Andric       if (match(A, m_BitwiseLogic(m_Value(X), m_ConstantInt())) ||
47*c9157d92SDimitry Andric           match(A, m_Shift(m_Value(X), m_ConstantInt())))
48*c9157d92SDimitry Andric         AddAffected(X);
49*c9157d92SDimitry Andric     } else {
50*c9157d92SDimitry Andric       Value *X;
51*c9157d92SDimitry Andric       // Handle (A + C1) u< C2, which is the canonical form of A > C3 && A < C4.
52*c9157d92SDimitry Andric       if (match(A, m_Add(m_Value(X), m_ConstantInt())))
53*c9157d92SDimitry Andric         AddAffected(X);
54*c9157d92SDimitry Andric     }
55*c9157d92SDimitry Andric   }
56*c9157d92SDimitry Andric }
57*c9157d92SDimitry Andric 
registerBranch(BranchInst * BI)58*c9157d92SDimitry Andric void DomConditionCache::registerBranch(BranchInst *BI) {
59*c9157d92SDimitry Andric   assert(BI->isConditional() && "Must be conditional branch");
60*c9157d92SDimitry Andric   SmallVector<Value *, 16> Affected;
61*c9157d92SDimitry Andric   findAffectedValues(BI->getCondition(), Affected);
62*c9157d92SDimitry Andric   for (Value *V : Affected) {
63*c9157d92SDimitry Andric     auto &AV = AffectedValues[V];
64*c9157d92SDimitry Andric     if (!is_contained(AV, BI))
65*c9157d92SDimitry Andric       AV.push_back(BI);
66*c9157d92SDimitry Andric   }
67*c9157d92SDimitry Andric }
68