llvm-project/llvm/lib/Analysis/AssumptionTracker.cpp

111 lines
3.7 KiB
C++

//===- AssumptionTracker.cpp - Track @llvm.assume -------------------------===//
//
// The LLVM Compiler Infrastructure
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
//
// This file contains a pass that keeps track of @llvm.assume intrinsics in
// the functions of a module.
//
//===----------------------------------------------------------------------===//
#include "llvm/Analysis/AssumptionTracker.h"
#include "llvm/IR/CallSite.h"
#include "llvm/IR/Dominators.h"
#include "llvm/IR/Function.h"
#include "llvm/IR/Instructions.h"
#include "llvm/IR/IntrinsicInst.h"
#include "llvm/IR/PatternMatch.h"
#include "llvm/Support/Debug.h"
using namespace llvm;
using namespace llvm::PatternMatch;
void AssumptionTracker::FunctionCallbackVH::deleted() {
AT->forgetCachedAssumptions(cast<Function>(getValPtr()));
// 'this' now dangles!
}
void AssumptionTracker::forgetCachedAssumptions(Function *F) {
auto I = CachedAssumeCalls.find_as(F);
if (I != CachedAssumeCalls.end())
CachedAssumeCalls.erase(I);
}
void AssumptionTracker::CallCallbackVH::deleted() {
assert(F && "delete callback called on dummy handle");
FunctionCallsMap::iterator I = AT->CachedAssumeCalls.find_as(F);
assert(I != AT->CachedAssumeCalls.end() &&
"Function cleared from the map without removing the values?");
I->second->erase(*this);
// 'this' now dangles!
}
AssumptionTracker::FunctionCallsMap::iterator
AssumptionTracker::scanFunction(Function *F) {
auto IP = CachedAssumeCalls.insert(std::make_pair(
FunctionCallbackVH(F, this), llvm::make_unique<CallHandleSet>()));
assert(IP.second && "Scanning function already in the map?");
FunctionCallsMap::iterator I = IP.first;
// Go through all instructions in all blocks, add all calls to @llvm.assume
// to our cache.
for (BasicBlock &B : *F)
for (Instruction &II : B)
if (match(&II, m_Intrinsic<Intrinsic::assume>()))
I->second->insert(CallCallbackVH(&II, this));
return I;
}
void AssumptionTracker::verifyAnalysis() const {
#ifndef NDEBUG
for (const auto &I : CachedAssumeCalls) {
for (const BasicBlock &B : cast<Function>(*I.first))
for (const Instruction &II : B) {
if (match(&II, m_Intrinsic<Intrinsic::assume>())) {
assert(I.second->find_as(&II) != I.second->end() &&
"Assumption in scanned function not in cache");
}
}
}
#endif
}
void AssumptionTracker::registerAssumption(CallInst *CI) {
assert(match(CI, m_Intrinsic<Intrinsic::assume>()) &&
"Registered call does not call @llvm.assume");
assert(CI->getParent() &&
"Cannot register @llvm.assume call not in a basic block");
Function *F = CI->getParent()->getParent();
assert(F && "Cannot register @llvm.assume call not in a function");
FunctionCallsMap::iterator I = CachedAssumeCalls.find_as(F);
if (I == CachedAssumeCalls.end()) {
// If this function has not already been scanned, then don't do anything
// here. This intrinsic will be found, if it still exists, if the list of
// assumptions in this function is requested at some later point. This
// maintains the following invariant: if a function is present in the
// cache, then its list of assumption intrinsic calls is complete.
return;
}
I->second->insert(CallCallbackVH(CI, this));
}
AssumptionTracker::AssumptionTracker() : ImmutablePass(ID) {
initializeAssumptionTrackerPass(*PassRegistry::getPassRegistry());
}
AssumptionTracker::~AssumptionTracker() {}
INITIALIZE_PASS(AssumptionTracker, "assumption-tracker", "Assumption Tracker",
false, true)
char AssumptionTracker::ID = 0;