mirror of
				https://github.com/c64scene-ar/llvm-6502.git
				synced 2025-11-04 05:17:07 +00:00 
			
		
		
		
	Also make it const safe and remove superfluous casting. NFC. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@220616 91177308-0d34-0410-b5e6-96231b3b80d8
		
			
				
	
	
		
			111 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			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;
 | 
						|
 |