BFI: Improve assertion message, since it's actually firing
[oota-llvm.git] / lib / Analysis / AssumptionTracker.cpp
1 //===- AssumptionTracker.cpp - Track @llvm.assume -------------------------===//
2 //
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 //
10 // This file contains a pass that keeps track of @llvm.assume intrinsics in
11 // the functions of a module.
12 //
13 //===----------------------------------------------------------------------===//
14
15 #include "llvm/Analysis/AssumptionTracker.h"
16 #include "llvm/IR/CallSite.h"
17 #include "llvm/IR/Dominators.h"
18 #include "llvm/IR/Function.h"
19 #include "llvm/IR/Instructions.h"
20 #include "llvm/IR/IntrinsicInst.h"
21 #include "llvm/IR/PatternMatch.h"
22 #include "llvm/Support/Debug.h"
23 using namespace llvm;
24 using namespace llvm::PatternMatch;
25
26 void AssumptionTracker::FunctionCallbackVH::deleted() {
27   AT->forgetCachedAssumptions(cast<Function>(getValPtr()));
28   // 'this' now dangles!
29 }
30
31 void AssumptionTracker::forgetCachedAssumptions(Function *F) {
32   CachedAssumeCalls.erase(F);
33 }
34
35 void AssumptionTracker::CallCallbackVH::deleted() {
36   assert(F && "delete callback called on dummy handle");
37   FunctionCallsMap::iterator I = AT->CachedAssumeCalls.find(F);
38   assert(I != AT->CachedAssumeCalls.end() &&
39          "Function cleared from the map without removing the values?");
40
41   I->second->erase(*this);
42   // 'this' now dangles!
43 }
44
45 AssumptionTracker::FunctionCallsMap::iterator
46 AssumptionTracker::scanFunction(Function *F) {
47   auto IP =
48     CachedAssumeCalls.insert(std::make_pair(FunctionCallbackVH(F, this),
49                                             std::unique_ptr<CallHandleSet>(
50                                               new CallHandleSet())));
51   assert(IP.second && "Scanning function already in the map?");
52
53   FunctionCallsMap::iterator I = IP.first;
54
55   // Go through all instructions in all blocks, add all calls to @llvm.assume
56   // to our cache.
57   for (BasicBlock &B : *F)
58     for (Instruction &II : B)
59       if (match(cast<Value>(&II), m_Intrinsic<Intrinsic::assume>(m_Value())))
60         I->second->insert(CallCallbackVH(&II, this));
61
62   return I;
63 }
64
65 void AssumptionTracker::verifyAnalysis() const {
66 #ifndef NDEBUG
67   for (const auto &I : CachedAssumeCalls) {
68     for (const BasicBlock &B : cast<Function>(*I.first))
69       for (const Instruction &II : B) {
70         Instruction *C = const_cast<Instruction*>(&II);
71         if (match(C, m_Intrinsic<Intrinsic::assume>(m_Value()))) {
72           assert(I.second->count(CallCallbackVH(C,
73                    const_cast<AssumptionTracker*>(this))) &&
74                  "Assumption in scanned function not in cache");
75         }
76     }
77   }
78 #endif
79 }
80
81 void AssumptionTracker::registerAssumption(CallInst *CI) {
82   assert(match(cast<Value>(CI),
83                m_Intrinsic<Intrinsic::assume>(m_Value())) &&
84          "Registered call does not call @llvm.assume");
85   assert(CI->getParent() &&
86          "Cannot register @llvm.assume call not in a basic block");
87
88   Function *F = CI->getParent()->getParent();
89   assert(F && "Cannot register @llvm.assume call not in a function");
90
91   FunctionCallsMap::iterator I = CachedAssumeCalls.find(F);
92   if (I == CachedAssumeCalls.end()) {
93     // If this function has not already been scanned, then don't do anything
94     // here. This intrinsic will be found, if it still exists, if the list of
95     // assumptions in this function is requested at some later point. This
96     // maintains the following invariant: if a function is present in the
97     // cache, then its list of assumption intrinsic calls is complete.
98     return;
99   }
100
101   I->second->insert(CallCallbackVH(CI, this));
102 }
103
104 AssumptionTracker::AssumptionTracker() : ImmutablePass(ID) {
105   initializeAssumptionTrackerPass(*PassRegistry::getPassRegistry());
106 }
107
108 AssumptionTracker::~AssumptionTracker() {}
109
110 INITIALIZE_PASS(AssumptionTracker, "assumption-tracker", "Assumption Tracker",
111                 false, true)
112 char AssumptionTracker::ID = 0;
113