mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-03 13:31:05 +00:00
3666e7f4c1
This adds an immutable pass, AssumptionTracker, which keeps a cache of @llvm.assume call instructions within a module. It uses callback value handles to keep stale functions and intrinsics out of the map, and it relies on any code that creates new @llvm.assume calls to notify it of the new instructions. The benefit is that code needing to find @llvm.assume intrinsics can do so directly, without scanning the function, thus allowing the cost of @llvm.assume handling to be negligible when none are present. The current design is intended to be lightweight. We don't keep track of anything until we need a list of assumptions in some function. The first time this happens, we scan the function. After that, we add/remove @llvm.assume calls from the cache in response to registration calls and ValueHandle callbacks. There are no new direct test cases for this pass, but because it calls it validation function upon module finalization, we'll pick up detectable inconsistencies from the other tests that touch @llvm.assume calls. This pass will be used by follow-up commits that make use of @llvm.assume. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@217334 91177308-0d34-0410-b5e6-96231b3b80d8
114 lines
3.9 KiB
C++
114 lines
3.9 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) {
|
|
CachedAssumeCalls.erase(F);
|
|
}
|
|
|
|
void AssumptionTracker::CallCallbackVH::deleted() {
|
|
assert(F && "delete callback called on dummy handle");
|
|
FunctionCallsMap::iterator I = AT->CachedAssumeCalls.find(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),
|
|
std::unique_ptr<CallHandleSet>(
|
|
new 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(cast<Value>(&II), m_Intrinsic<Intrinsic::assume>(m_Value())))
|
|
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) {
|
|
Instruction *C = const_cast<Instruction*>(&II);
|
|
if (match(C, m_Intrinsic<Intrinsic::assume>(m_Value()))) {
|
|
assert(I.second->count(CallCallbackVH(C,
|
|
const_cast<AssumptionTracker*>(this))) &&
|
|
"Assumption in scanned function not in cache");
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void AssumptionTracker::registerAssumption(CallInst *CI) {
|
|
assert(match(cast<Value>(CI),
|
|
m_Intrinsic<Intrinsic::assume>(m_Value())) &&
|
|
"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(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;
|
|
|