mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-15 23:31:37 +00:00
[PM] Actually add the new pass manager support for the assumption cache.
I had already factored this analysis specifically to enable doing this, but hadn't actually committed the necessary wiring to get at this from the new pass manager. This also nicely shows how the separate cache object can be directly managed by the new pass manager. This analysis didn't have any direct tests and so I've added a printer pass and a boring test case. I chose to print the i1 value which is being assumed rather than the call to llvm.assume as that seems much more useful for testing... but suggestions on an even better printing strategy welcome. My main goal was to make sure things actually work. =] git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@226868 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
1552e50011
commit
9a78a64776
@ -27,6 +27,11 @@
|
|||||||
|
|
||||||
namespace llvm {
|
namespace llvm {
|
||||||
|
|
||||||
|
// FIXME: Replace this brittle forward declaration with the include of the new
|
||||||
|
// PassManager.h when doing so doesn't break the PassManagerBuilder.
|
||||||
|
template <typename IRUnitT> class AnalysisManager;
|
||||||
|
class PreservedAnalyses;
|
||||||
|
|
||||||
/// \brief A cache of @llvm.assume calls within a function.
|
/// \brief A cache of @llvm.assume calls within a function.
|
||||||
///
|
///
|
||||||
/// This cache provides fast lookup of assumptions within a function by caching
|
/// This cache provides fast lookup of assumptions within a function by caching
|
||||||
@ -88,6 +93,42 @@ public:
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/// \brief A function analysis which provides an \c AssumptionCache.
|
||||||
|
///
|
||||||
|
/// This analysis is intended for use with the new pass manager and will vend
|
||||||
|
/// assumption caches for a given function.
|
||||||
|
class AssumptionAnalysis {
|
||||||
|
static char PassID;
|
||||||
|
|
||||||
|
public:
|
||||||
|
typedef AssumptionCache Result;
|
||||||
|
|
||||||
|
/// \brief Opaque, unique identifier for this analysis pass.
|
||||||
|
static void *ID() { return (void *)&PassID; }
|
||||||
|
|
||||||
|
/// \brief Provide a name for the analysis for debugging and logging.
|
||||||
|
static StringRef name() { return "AssumptionAnalysis"; }
|
||||||
|
|
||||||
|
AssumptionAnalysis() {}
|
||||||
|
AssumptionAnalysis(const AssumptionAnalysis &Arg) {}
|
||||||
|
AssumptionAnalysis(AssumptionAnalysis &&Arg) {}
|
||||||
|
AssumptionAnalysis &operator=(const AssumptionAnalysis &RHS) { return *this; }
|
||||||
|
AssumptionAnalysis &operator=(AssumptionAnalysis &&RHS) { return *this; }
|
||||||
|
|
||||||
|
AssumptionCache run(Function &F) { return AssumptionCache(F); }
|
||||||
|
};
|
||||||
|
|
||||||
|
/// \brief Printer pass for the \c AssumptionAnalysis results.
|
||||||
|
class AssumptionPrinterPass {
|
||||||
|
raw_ostream &OS;
|
||||||
|
|
||||||
|
public:
|
||||||
|
explicit AssumptionPrinterPass(raw_ostream &OS) : OS(OS) {}
|
||||||
|
PreservedAnalyses run(Function &F, AnalysisManager<Function> *AM);
|
||||||
|
|
||||||
|
static StringRef name() { return "AssumptionPrinterPass"; }
|
||||||
|
};
|
||||||
|
|
||||||
/// \brief An immutable pass that tracks lazily created \c AssumptionCache
|
/// \brief An immutable pass that tracks lazily created \c AssumptionCache
|
||||||
/// objects.
|
/// objects.
|
||||||
///
|
///
|
||||||
|
@ -18,6 +18,7 @@
|
|||||||
#include "llvm/IR/Function.h"
|
#include "llvm/IR/Function.h"
|
||||||
#include "llvm/IR/Instructions.h"
|
#include "llvm/IR/Instructions.h"
|
||||||
#include "llvm/IR/IntrinsicInst.h"
|
#include "llvm/IR/IntrinsicInst.h"
|
||||||
|
#include "llvm/IR/PassManager.h"
|
||||||
#include "llvm/IR/PatternMatch.h"
|
#include "llvm/IR/PatternMatch.h"
|
||||||
#include "llvm/Support/Debug.h"
|
#include "llvm/Support/Debug.h"
|
||||||
using namespace llvm;
|
using namespace llvm;
|
||||||
@ -73,6 +74,20 @@ void AssumptionCache::registerAssumption(CallInst *CI) {
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
char AssumptionAnalysis::PassID;
|
||||||
|
|
||||||
|
PreservedAnalyses AssumptionPrinterPass::run(Function &F,
|
||||||
|
AnalysisManager<Function> *AM) {
|
||||||
|
AssumptionCache &AC = AM->getResult<AssumptionAnalysis>(F);
|
||||||
|
|
||||||
|
OS << "Cached assumptions for function: " << F.getName() << "\n";
|
||||||
|
for (auto &VH : AC.assumptions())
|
||||||
|
if (VH)
|
||||||
|
OS << " " << *cast<CallInst>(VH)->getArgOperand(0) << "\n";
|
||||||
|
|
||||||
|
return PreservedAnalyses::all();
|
||||||
|
}
|
||||||
|
|
||||||
void AssumptionCacheTracker::FunctionCallbackVH::deleted() {
|
void AssumptionCacheTracker::FunctionCallbackVH::deleted() {
|
||||||
auto I = ACT->AssumptionCaches.find_as(cast<Function>(getValPtr()));
|
auto I = ACT->AssumptionCaches.find_as(cast<Function>(getValPtr()));
|
||||||
if (I != ACT->AssumptionCaches.end())
|
if (I != ACT->AssumptionCaches.end())
|
||||||
|
22
test/Analysis/AssumptionCache/basic.ll
Normal file
22
test/Analysis/AssumptionCache/basic.ll
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
; RUN: opt < %s -disable-output -passes='print<assumptions>' 2>&1 | FileCheck %s
|
||||||
|
|
||||||
|
target datalayout = "e-i64:64-f80:128-n8:16:32:64-S128"
|
||||||
|
|
||||||
|
declare void @llvm.assume(i1)
|
||||||
|
|
||||||
|
define void @test1(i32 %a) {
|
||||||
|
; CHECK-LABEL: Cached assumptions for function: test1
|
||||||
|
; CHECK-NEXT: icmp ne i32 %{{.*}}, 0
|
||||||
|
; CHECK-NEXT: icmp slt i32 %{{.*}}, 0
|
||||||
|
; CHECK-NEXT: icmp sgt i32 %{{.*}}, 0
|
||||||
|
|
||||||
|
entry:
|
||||||
|
%cond1 = icmp ne i32 %a, 0
|
||||||
|
call void @llvm.assume(i1 %cond1)
|
||||||
|
%cond2 = icmp slt i32 %a, 0
|
||||||
|
call void @llvm.assume(i1 %cond2)
|
||||||
|
%cond3 = icmp sgt i32 %a, 0
|
||||||
|
call void @llvm.assume(i1 %cond3)
|
||||||
|
|
||||||
|
ret void
|
||||||
|
}
|
@ -50,6 +50,7 @@ CGSCC_PASS("no-op-cgscc", NoOpCGSCCPass())
|
|||||||
#ifndef FUNCTION_ANALYSIS
|
#ifndef FUNCTION_ANALYSIS
|
||||||
#define FUNCTION_ANALYSIS(NAME, CREATE_PASS)
|
#define FUNCTION_ANALYSIS(NAME, CREATE_PASS)
|
||||||
#endif
|
#endif
|
||||||
|
FUNCTION_ANALYSIS("assumptions", AssumptionAnalysis())
|
||||||
FUNCTION_ANALYSIS("domtree", DominatorTreeAnalysis())
|
FUNCTION_ANALYSIS("domtree", DominatorTreeAnalysis())
|
||||||
FUNCTION_ANALYSIS("loops", LoopAnalysis())
|
FUNCTION_ANALYSIS("loops", LoopAnalysis())
|
||||||
FUNCTION_ANALYSIS("no-op-function", NoOpFunctionAnalysis())
|
FUNCTION_ANALYSIS("no-op-function", NoOpFunctionAnalysis())
|
||||||
@ -61,6 +62,7 @@ FUNCTION_ANALYSIS("no-op-function", NoOpFunctionAnalysis())
|
|||||||
FUNCTION_PASS("invalidate<all>", InvalidateAllAnalysesPass())
|
FUNCTION_PASS("invalidate<all>", InvalidateAllAnalysesPass())
|
||||||
FUNCTION_PASS("no-op-function", NoOpFunctionPass())
|
FUNCTION_PASS("no-op-function", NoOpFunctionPass())
|
||||||
FUNCTION_PASS("print", PrintFunctionPass(dbgs()))
|
FUNCTION_PASS("print", PrintFunctionPass(dbgs()))
|
||||||
|
FUNCTION_PASS("print<assumptions>", AssumptionPrinterPass(dbgs()))
|
||||||
FUNCTION_PASS("print<domtree>", DominatorTreePrinterPass(dbgs()))
|
FUNCTION_PASS("print<domtree>", DominatorTreePrinterPass(dbgs()))
|
||||||
FUNCTION_PASS("print<loops>", LoopPrinterPass(dbgs()))
|
FUNCTION_PASS("print<loops>", LoopPrinterPass(dbgs()))
|
||||||
FUNCTION_PASS("verify", VerifierPass())
|
FUNCTION_PASS("verify", VerifierPass())
|
||||||
|
@ -15,6 +15,7 @@
|
|||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
|
|
||||||
#include "Passes.h"
|
#include "Passes.h"
|
||||||
|
#include "llvm/Analysis/AssumptionCache.h"
|
||||||
#include "llvm/Analysis/CGSCCPassManager.h"
|
#include "llvm/Analysis/CGSCCPassManager.h"
|
||||||
#include "llvm/Analysis/LazyCallGraph.h"
|
#include "llvm/Analysis/LazyCallGraph.h"
|
||||||
#include "llvm/Analysis/LoopInfo.h"
|
#include "llvm/Analysis/LoopInfo.h"
|
||||||
|
Loading…
x
Reference in New Issue
Block a user