2014-04-12 01:20:58 +02:00
|
|
|
//===- BlockFrequencyInfo.cpp - Block Frequency Analysis ------------------===//
|
2011-06-23 23:56:59 +02:00
|
|
|
//
|
2019-01-19 09:50:56 +01:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2011-06-23 23:56:59 +02:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// Loops should be simplified before this analysis.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2011-07-25 21:25:40 +02:00
|
|
|
#include "llvm/Analysis/BlockFrequencyInfo.h"
|
2017-07-21 23:37:46 +02:00
|
|
|
#include "llvm/ADT/APInt.h"
|
|
|
|
#include "llvm/ADT/None.h"
|
|
|
|
#include "llvm/ADT/iterator.h"
|
2014-04-12 01:20:58 +02:00
|
|
|
#include "llvm/Analysis/BlockFrequencyInfoImpl.h"
|
2012-12-03 17:50:05 +01:00
|
|
|
#include "llvm/Analysis/BranchProbabilityInfo.h"
|
2011-06-23 23:56:59 +02:00
|
|
|
#include "llvm/Analysis/LoopInfo.h"
|
2014-03-04 12:45:46 +01:00
|
|
|
#include "llvm/IR/CFG.h"
|
2017-07-21 23:37:46 +02:00
|
|
|
#include "llvm/IR/Function.h"
|
|
|
|
#include "llvm/IR/PassManager.h"
|
Sink all InitializePasses.h includes
This file lists every pass in LLVM, and is included by Pass.h, which is
very popular. Every time we add, remove, or rename a pass in LLVM, it
caused lots of recompilation.
I found this fact by looking at this table, which is sorted by the
number of times a file was changed over the last 100,000 git commits
multiplied by the number of object files that depend on it in the
current checkout:
recompiles touches affected_files header
342380 95 3604 llvm/include/llvm/ADT/STLExtras.h
314730 234 1345 llvm/include/llvm/InitializePasses.h
307036 118 2602 llvm/include/llvm/ADT/APInt.h
213049 59 3611 llvm/include/llvm/Support/MathExtras.h
170422 47 3626 llvm/include/llvm/Support/Compiler.h
162225 45 3605 llvm/include/llvm/ADT/Optional.h
158319 63 2513 llvm/include/llvm/ADT/Triple.h
140322 39 3598 llvm/include/llvm/ADT/StringRef.h
137647 59 2333 llvm/include/llvm/Support/Error.h
131619 73 1803 llvm/include/llvm/Support/FileSystem.h
Before this change, touching InitializePasses.h would cause 1345 files
to recompile. After this change, touching it only causes 550 compiles in
an incremental rebuild.
Reviewers: bkramer, asbirlea, bollu, jdoerfert
Differential Revision: https://reviews.llvm.org/D70211
2019-11-13 22:15:01 +01:00
|
|
|
#include "llvm/InitializePasses.h"
|
2017-07-21 23:37:46 +02:00
|
|
|
#include "llvm/Pass.h"
|
2013-11-14 03:27:46 +01:00
|
|
|
#include "llvm/Support/CommandLine.h"
|
|
|
|
#include "llvm/Support/GraphWriter.h"
|
2017-07-21 23:37:46 +02:00
|
|
|
#include "llvm/Support/raw_ostream.h"
|
|
|
|
#include <algorithm>
|
|
|
|
#include <cassert>
|
|
|
|
#include <string>
|
2011-06-23 23:56:59 +02:00
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
|
2014-04-22 04:48:03 +02:00
|
|
|
#define DEBUG_TYPE "block-freq"
|
|
|
|
|
2016-06-28 06:07:03 +02:00
|
|
|
static cl::opt<GVDAGType> ViewBlockFreqPropagationDAG(
|
|
|
|
"view-block-freq-propagation-dags", cl::Hidden,
|
|
|
|
cl::desc("Pop up a window to show a dag displaying how block "
|
|
|
|
"frequencies propagation through the CFG."),
|
|
|
|
cl::values(clEnumValN(GVDT_None, "none", "do not display graphs."),
|
|
|
|
clEnumValN(GVDT_Fraction, "fraction",
|
|
|
|
"display a graph using the "
|
|
|
|
"fractional block frequency representation."),
|
|
|
|
clEnumValN(GVDT_Integer, "integer",
|
|
|
|
"display a graph using the raw "
|
|
|
|
"integer fractional block frequency representation."),
|
|
|
|
clEnumValN(GVDT_Count, "count", "display a graph using the real "
|
2016-10-08 21:41:06 +02:00
|
|
|
"profile count if available.")));
|
2016-06-28 06:07:03 +02:00
|
|
|
|
2021-05-07 20:15:43 +02:00
|
|
|
namespace llvm {
|
2016-06-28 08:58:21 +02:00
|
|
|
cl::opt<std::string>
|
|
|
|
ViewBlockFreqFuncName("view-bfi-func-name", cl::Hidden,
|
|
|
|
cl::desc("The option to specify "
|
|
|
|
"the name of the function "
|
|
|
|
"whose CFG will be displayed."));
|
|
|
|
|
|
|
|
cl::opt<unsigned>
|
|
|
|
ViewHotFreqPercent("view-hot-freq-percent", cl::init(10), cl::Hidden,
|
|
|
|
cl::desc("An integer in percent used to specify "
|
|
|
|
"the hot blocks/edges to be displayed "
|
|
|
|
"in red: a block or edge whose frequency "
|
|
|
|
"is no less than the max frequency of the "
|
|
|
|
"function multiplied by this percent."));
|
2013-11-14 03:27:46 +01:00
|
|
|
|
2017-09-13 19:20:38 +02:00
|
|
|
// Command line option to turn on CFG dot or text dump after profile annotation.
|
|
|
|
cl::opt<PGOViewCountsType> PGOViewCounts(
|
|
|
|
"pgo-view-counts", cl::Hidden,
|
|
|
|
cl::desc("A boolean option to show CFG dag or text with "
|
|
|
|
"block profile counts and branch probabilities "
|
|
|
|
"right after PGO profile annotation step. The "
|
|
|
|
"profile counts are computed using branch "
|
|
|
|
"probabilities from the runtime profile data and "
|
|
|
|
"block frequency propagation algorithm. To view "
|
|
|
|
"the raw counts from the profile, use option "
|
|
|
|
"-pgo-view-raw-counts instead. To limit graph "
|
|
|
|
"display to only one function, use filtering option "
|
|
|
|
"-view-bfi-func-name."),
|
|
|
|
cl::values(clEnumValN(PGOVCT_None, "none", "do not show."),
|
|
|
|
clEnumValN(PGOVCT_Graph, "graph", "show a graph."),
|
|
|
|
clEnumValN(PGOVCT_Text, "text", "show in text.")));
|
2017-01-23 19:58:24 +01:00
|
|
|
|
2017-08-26 02:31:00 +02:00
|
|
|
static cl::opt<bool> PrintBlockFreq(
|
|
|
|
"print-bfi", cl::init(false), cl::Hidden,
|
|
|
|
cl::desc("Print the block frequency info."));
|
|
|
|
|
|
|
|
cl::opt<std::string> PrintBlockFreqFuncName(
|
|
|
|
"print-bfi-func-name", cl::Hidden,
|
|
|
|
cl::desc("The option to specify the name of the function "
|
|
|
|
"whose block frequency info is printed."));
|
2021-05-07 20:15:43 +02:00
|
|
|
} // namespace llvm
|
2017-08-26 02:31:00 +02:00
|
|
|
|
2013-11-14 03:27:46 +01:00
|
|
|
namespace llvm {
|
|
|
|
|
2017-01-23 19:58:24 +01:00
|
|
|
static GVDAGType getGVDT() {
|
2017-09-13 19:20:38 +02:00
|
|
|
if (PGOViewCounts == PGOVCT_Graph)
|
2017-01-23 19:58:24 +01:00
|
|
|
return GVDT_Count;
|
|
|
|
return ViewBlockFreqPropagationDAG;
|
|
|
|
}
|
|
|
|
|
2013-11-14 03:27:46 +01:00
|
|
|
template <>
|
|
|
|
struct GraphTraits<BlockFrequencyInfo *> {
|
2017-07-21 23:37:46 +02:00
|
|
|
using NodeRef = const BasicBlock *;
|
2020-03-10 19:33:02 +01:00
|
|
|
using ChildIteratorType = const_succ_iterator;
|
2017-07-21 23:37:46 +02:00
|
|
|
using nodes_iterator = pointer_iterator<Function::const_iterator>;
|
2013-11-14 03:27:46 +01:00
|
|
|
|
2016-08-31 18:48:13 +02:00
|
|
|
static NodeRef getEntryNode(const BlockFrequencyInfo *G) {
|
Analysis: Remove implicit ilist iterator conversions
Remove implicit ilist iterator conversions from LLVMAnalysis.
I came across something really scary in `llvm::isKnownNotFullPoison()`
which relied on `Instruction::getNextNode()` being completely broken
(not surprising, but scary nevertheless). This function is documented
(and coded to) return `nullptr` when it gets to the sentinel, but with
an `ilist_half_node` as a sentinel, the sentinel check looks into some
other memory and we don't recognize we've hit the end.
Rooting out these scary cases is the reason I'm removing the implicit
conversions before doing anything else with `ilist`; I'm not at all
surprised that clients rely on badness.
I found another scary case -- this time, not relying on badness, just
bad (but I guess getting lucky so far) -- in
`ObjectSizeOffsetEvaluator::compute_()`. Here, we save out the
insertion point, do some things, and then restore it. Previously, we
let the iterator auto-convert to `Instruction*`, and then set it back
using the `Instruction*` version:
Instruction *PrevInsertPoint = Builder.GetInsertPoint();
/* Logic that may change insert point */
if (PrevInsertPoint)
Builder.SetInsertPoint(PrevInsertPoint);
The check for `PrevInsertPoint` doesn't protect correctly against bad
accesses. If the insertion point has been set to the end of a basic
block (i.e., `SetInsertPoint(SomeBB)`), then `GetInsertPoint()` returns
an iterator pointing at the list sentinel. The version of
`SetInsertPoint()` that's getting called will then call
`PrevInsertPoint->getParent()`, which explodes horribly. The only
reason this hasn't blown up is that it's fairly unlikely the builder is
adding to the end of the block; usually, we're adding instructions
somewhere before the terminator.
llvm-svn: 249925
2015-10-10 02:53:03 +02:00
|
|
|
return &G->getFunction()->front();
|
2013-11-14 03:27:46 +01:00
|
|
|
}
|
2017-07-21 23:37:46 +02:00
|
|
|
|
2016-08-22 23:09:30 +02:00
|
|
|
static ChildIteratorType child_begin(const NodeRef N) {
|
2013-11-14 03:27:46 +01:00
|
|
|
return succ_begin(N);
|
|
|
|
}
|
2017-07-21 23:37:46 +02:00
|
|
|
|
2016-08-22 23:09:30 +02:00
|
|
|
static ChildIteratorType child_end(const NodeRef N) { return succ_end(N); }
|
2017-07-21 23:37:46 +02:00
|
|
|
|
2013-11-14 03:27:46 +01:00
|
|
|
static nodes_iterator nodes_begin(const BlockFrequencyInfo *G) {
|
2016-08-19 23:20:13 +02:00
|
|
|
return nodes_iterator(G->getFunction()->begin());
|
2013-11-14 03:27:46 +01:00
|
|
|
}
|
2017-07-21 23:37:46 +02:00
|
|
|
|
2013-11-14 03:27:46 +01:00
|
|
|
static nodes_iterator nodes_end(const BlockFrequencyInfo *G) {
|
2016-08-19 23:20:13 +02:00
|
|
|
return nodes_iterator(G->getFunction()->end());
|
2013-11-14 03:27:46 +01:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2017-07-21 23:37:46 +02:00
|
|
|
using BFIDOTGTraitsBase =
|
|
|
|
BFIDOTGraphTraitsBase<BlockFrequencyInfo, BranchProbabilityInfo>;
|
2013-11-14 03:27:46 +01:00
|
|
|
|
2016-06-28 05:41:29 +02:00
|
|
|
template <>
|
|
|
|
struct DOTGraphTraits<BlockFrequencyInfo *> : public BFIDOTGTraitsBase {
|
|
|
|
explicit DOTGraphTraits(bool isSimple = false)
|
|
|
|
: BFIDOTGTraitsBase(isSimple) {}
|
2013-11-14 03:27:46 +01:00
|
|
|
|
|
|
|
std::string getNodeLabel(const BasicBlock *Node,
|
|
|
|
const BlockFrequencyInfo *Graph) {
|
2016-06-28 05:41:29 +02:00
|
|
|
|
2017-01-23 19:58:24 +01:00
|
|
|
return BFIDOTGTraitsBase::getNodeLabel(Node, Graph, getGVDT());
|
2016-06-28 05:41:29 +02:00
|
|
|
}
|
|
|
|
|
2016-06-28 08:58:21 +02:00
|
|
|
std::string getNodeAttributes(const BasicBlock *Node,
|
|
|
|
const BlockFrequencyInfo *Graph) {
|
|
|
|
return BFIDOTGTraitsBase::getNodeAttributes(Node, Graph,
|
|
|
|
ViewHotFreqPercent);
|
|
|
|
}
|
|
|
|
|
2016-06-28 05:41:29 +02:00
|
|
|
std::string getEdgeAttributes(const BasicBlock *Node, EdgeIter EI,
|
|
|
|
const BlockFrequencyInfo *BFI) {
|
2016-06-28 08:58:21 +02:00
|
|
|
return BFIDOTGTraitsBase::getEdgeAttributes(Node, EI, BFI, BFI->getBPI(),
|
|
|
|
ViewHotFreqPercent);
|
2013-11-14 03:27:46 +01:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
} // end namespace llvm
|
|
|
|
|
2017-07-21 23:37:46 +02:00
|
|
|
BlockFrequencyInfo::BlockFrequencyInfo() = default;
|
2015-07-17 01:23:35 +02:00
|
|
|
|
|
|
|
BlockFrequencyInfo::BlockFrequencyInfo(const Function &F,
|
|
|
|
const BranchProbabilityInfo &BPI,
|
|
|
|
const LoopInfo &LI) {
|
|
|
|
calculate(F, BPI, LI);
|
|
|
|
}
|
|
|
|
|
2016-05-05 23:13:27 +02:00
|
|
|
BlockFrequencyInfo::BlockFrequencyInfo(BlockFrequencyInfo &&Arg)
|
|
|
|
: BFI(std::move(Arg.BFI)) {}
|
|
|
|
|
|
|
|
BlockFrequencyInfo &BlockFrequencyInfo::operator=(BlockFrequencyInfo &&RHS) {
|
|
|
|
releaseMemory();
|
|
|
|
BFI = std::move(RHS.BFI);
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2016-07-13 07:01:48 +02:00
|
|
|
// Explicitly define the default constructor otherwise it would be implicitly
|
|
|
|
// defined at the first ODR-use which is the BFI member in the
|
|
|
|
// LazyBlockFrequencyInfo header. The dtor needs the BlockFrequencyInfoImpl
|
|
|
|
// template instantiated which is not available in the header.
|
2017-07-21 23:37:46 +02:00
|
|
|
BlockFrequencyInfo::~BlockFrequencyInfo() = default;
|
2016-07-13 07:01:48 +02:00
|
|
|
|
2017-01-15 07:32:49 +01:00
|
|
|
bool BlockFrequencyInfo::invalidate(Function &F, const PreservedAnalyses &PA,
|
|
|
|
FunctionAnalysisManager::Invalidator &) {
|
|
|
|
// Check whether the analysis, all analyses on functions, or the function's
|
|
|
|
// CFG have been preserved.
|
|
|
|
auto PAC = PA.getChecker<BlockFrequencyAnalysis>();
|
|
|
|
return !(PAC.preserved() || PAC.preservedSet<AllAnalysesOn<Function>>() ||
|
|
|
|
PAC.preservedSet<CFGAnalyses>());
|
|
|
|
}
|
|
|
|
|
2015-07-15 01:40:50 +02:00
|
|
|
void BlockFrequencyInfo::calculate(const Function &F,
|
|
|
|
const BranchProbabilityInfo &BPI,
|
|
|
|
const LoopInfo &LI) {
|
2014-03-25 19:01:38 +01:00
|
|
|
if (!BFI)
|
|
|
|
BFI.reset(new ImplType);
|
2015-07-15 21:58:26 +02:00
|
|
|
BFI->calculate(F, BPI, LI);
|
2016-06-28 06:07:03 +02:00
|
|
|
if (ViewBlockFreqPropagationDAG != GVDT_None &&
|
|
|
|
(ViewBlockFreqFuncName.empty() ||
|
|
|
|
F.getName().equals(ViewBlockFreqFuncName))) {
|
2013-11-14 03:27:46 +01:00
|
|
|
view();
|
2016-06-28 06:07:03 +02:00
|
|
|
}
|
2017-08-26 02:31:00 +02:00
|
|
|
if (PrintBlockFreq &&
|
|
|
|
(PrintBlockFreqFuncName.empty() ||
|
|
|
|
F.getName().equals(PrintBlockFreqFuncName))) {
|
|
|
|
print(dbgs());
|
|
|
|
}
|
2011-10-19 12:12:41 +02:00
|
|
|
}
|
|
|
|
|
2011-12-20 21:03:10 +01:00
|
|
|
BlockFrequency BlockFrequencyInfo::getBlockFreq(const BasicBlock *BB) const {
|
2014-03-25 19:01:38 +01:00
|
|
|
return BFI ? BFI->getBlockFreq(BB) : 0;
|
2011-06-23 23:56:59 +02:00
|
|
|
}
|
2013-11-14 03:27:46 +01:00
|
|
|
|
2016-03-23 19:18:26 +01:00
|
|
|
Optional<uint64_t>
|
2019-04-24 21:51:16 +02:00
|
|
|
BlockFrequencyInfo::getBlockProfileCount(const BasicBlock *BB,
|
|
|
|
bool AllowSynthetic) const {
|
2016-06-22 19:12:12 +02:00
|
|
|
if (!BFI)
|
2016-03-23 19:18:26 +01:00
|
|
|
return None;
|
2016-06-22 19:12:12 +02:00
|
|
|
|
2019-04-24 21:51:16 +02:00
|
|
|
return BFI->getBlockProfileCount(*getFunction(), BB, AllowSynthetic);
|
2016-03-23 19:18:26 +01:00
|
|
|
}
|
|
|
|
|
2016-08-02 04:15:45 +02:00
|
|
|
Optional<uint64_t>
|
|
|
|
BlockFrequencyInfo::getProfileCountFromFreq(uint64_t Freq) const {
|
|
|
|
if (!BFI)
|
|
|
|
return None;
|
|
|
|
return BFI->getProfileCountFromFreq(*getFunction(), Freq);
|
|
|
|
}
|
|
|
|
|
2017-11-02 23:26:51 +01:00
|
|
|
bool BlockFrequencyInfo::isIrrLoopHeader(const BasicBlock *BB) {
|
|
|
|
assert(BFI && "Expected analysis to be available");
|
|
|
|
return BFI->isIrrLoopHeader(BB);
|
|
|
|
}
|
|
|
|
|
2016-06-22 19:12:12 +02:00
|
|
|
void BlockFrequencyInfo::setBlockFreq(const BasicBlock *BB, uint64_t Freq) {
|
2015-10-15 16:59:40 +02:00
|
|
|
assert(BFI && "Expected analysis to be available");
|
|
|
|
BFI->setBlockFreq(BB, Freq);
|
|
|
|
}
|
|
|
|
|
2017-01-19 19:53:16 +01:00
|
|
|
void BlockFrequencyInfo::setBlockFreqAndScale(
|
|
|
|
const BasicBlock *ReferenceBB, uint64_t Freq,
|
|
|
|
SmallPtrSetImpl<BasicBlock *> &BlocksToScale) {
|
|
|
|
assert(BFI && "Expected analysis to be available");
|
|
|
|
// Use 128 bits APInt to avoid overflow.
|
|
|
|
APInt NewFreq(128, Freq);
|
|
|
|
APInt OldFreq(128, BFI->getBlockFreq(ReferenceBB).getFrequency());
|
|
|
|
APInt BBFreq(128, 0);
|
|
|
|
for (auto *BB : BlocksToScale) {
|
|
|
|
BBFreq = BFI->getBlockFreq(BB).getFrequency();
|
|
|
|
// Multiply first by NewFreq and then divide by OldFreq
|
|
|
|
// to minimize loss of precision.
|
|
|
|
BBFreq *= NewFreq;
|
|
|
|
// udiv is an expensive operation in the general case. If this ends up being
|
|
|
|
// a hot spot, one of the options proposed in
|
|
|
|
// https://reviews.llvm.org/D28535#650071 could be used to avoid this.
|
|
|
|
BBFreq = BBFreq.udiv(OldFreq);
|
|
|
|
BFI->setBlockFreq(BB, BBFreq.getLimitedValue());
|
|
|
|
}
|
|
|
|
BFI->setBlockFreq(ReferenceBB, Freq);
|
|
|
|
}
|
|
|
|
|
2013-11-14 03:27:46 +01:00
|
|
|
/// Pop up a ghostview window with the current block frequency propagation
|
|
|
|
/// rendered using dot.
|
2019-01-09 20:12:38 +01:00
|
|
|
void BlockFrequencyInfo::view(StringRef title) const {
|
|
|
|
ViewGraph(const_cast<BlockFrequencyInfo *>(this), title);
|
2013-11-14 03:27:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
const Function *BlockFrequencyInfo::getFunction() const {
|
2014-04-21 19:57:07 +02:00
|
|
|
return BFI ? BFI->getFunction() : nullptr;
|
2013-11-14 03:27:46 +01:00
|
|
|
}
|
2013-12-14 01:06:03 +01:00
|
|
|
|
2016-06-28 05:41:29 +02:00
|
|
|
const BranchProbabilityInfo *BlockFrequencyInfo::getBPI() const {
|
|
|
|
return BFI ? &BFI->getBPI() : nullptr;
|
|
|
|
}
|
|
|
|
|
2013-12-14 01:06:03 +01:00
|
|
|
raw_ostream &BlockFrequencyInfo::
|
|
|
|
printBlockFreq(raw_ostream &OS, const BlockFrequency Freq) const {
|
2014-03-25 19:01:38 +01:00
|
|
|
return BFI ? BFI->printBlockFreq(OS, Freq) : OS;
|
2013-12-14 01:06:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
raw_ostream &
|
|
|
|
BlockFrequencyInfo::printBlockFreq(raw_ostream &OS,
|
|
|
|
const BasicBlock *BB) const {
|
2014-03-25 19:01:38 +01:00
|
|
|
return BFI ? BFI->printBlockFreq(OS, BB) : OS;
|
2013-12-14 01:06:03 +01:00
|
|
|
}
|
2013-12-20 23:11:11 +01:00
|
|
|
|
|
|
|
uint64_t BlockFrequencyInfo::getEntryFreq() const {
|
2014-03-25 19:01:38 +01:00
|
|
|
return BFI ? BFI->getEntryFreq() : 0;
|
2013-12-20 23:11:11 +01:00
|
|
|
}
|
2015-07-15 01:40:50 +02:00
|
|
|
|
|
|
|
void BlockFrequencyInfo::releaseMemory() { BFI.reset(); }
|
|
|
|
|
|
|
|
void BlockFrequencyInfo::print(raw_ostream &OS) const {
|
|
|
|
if (BFI)
|
|
|
|
BFI->print(OS);
|
|
|
|
}
|
|
|
|
|
2020-04-03 19:40:26 +02:00
|
|
|
void BlockFrequencyInfo::verifyMatch(BlockFrequencyInfo &Other) const {
|
|
|
|
if (BFI)
|
|
|
|
BFI->verifyMatch(*Other.BFI);
|
|
|
|
}
|
|
|
|
|
2015-07-15 01:40:50 +02:00
|
|
|
INITIALIZE_PASS_BEGIN(BlockFrequencyInfoWrapperPass, "block-freq",
|
|
|
|
"Block Frequency Analysis", true, true)
|
2015-07-16 00:48:29 +02:00
|
|
|
INITIALIZE_PASS_DEPENDENCY(BranchProbabilityInfoWrapperPass)
|
2015-07-15 01:40:50 +02:00
|
|
|
INITIALIZE_PASS_DEPENDENCY(LoopInfoWrapperPass)
|
|
|
|
INITIALIZE_PASS_END(BlockFrequencyInfoWrapperPass, "block-freq",
|
|
|
|
"Block Frequency Analysis", true, true)
|
|
|
|
|
|
|
|
char BlockFrequencyInfoWrapperPass::ID = 0;
|
|
|
|
|
|
|
|
BlockFrequencyInfoWrapperPass::BlockFrequencyInfoWrapperPass()
|
|
|
|
: FunctionPass(ID) {
|
|
|
|
initializeBlockFrequencyInfoWrapperPassPass(*PassRegistry::getPassRegistry());
|
|
|
|
}
|
|
|
|
|
2017-07-21 23:37:46 +02:00
|
|
|
BlockFrequencyInfoWrapperPass::~BlockFrequencyInfoWrapperPass() = default;
|
2015-07-15 01:40:50 +02:00
|
|
|
|
|
|
|
void BlockFrequencyInfoWrapperPass::print(raw_ostream &OS,
|
|
|
|
const Module *) const {
|
|
|
|
BFI.print(OS);
|
|
|
|
}
|
|
|
|
|
|
|
|
void BlockFrequencyInfoWrapperPass::getAnalysisUsage(AnalysisUsage &AU) const {
|
2015-07-16 00:48:29 +02:00
|
|
|
AU.addRequired<BranchProbabilityInfoWrapperPass>();
|
2015-07-15 01:40:50 +02:00
|
|
|
AU.addRequired<LoopInfoWrapperPass>();
|
|
|
|
AU.setPreservesAll();
|
|
|
|
}
|
|
|
|
|
|
|
|
void BlockFrequencyInfoWrapperPass::releaseMemory() { BFI.releaseMemory(); }
|
|
|
|
|
|
|
|
bool BlockFrequencyInfoWrapperPass::runOnFunction(Function &F) {
|
2015-07-16 00:48:29 +02:00
|
|
|
BranchProbabilityInfo &BPI =
|
|
|
|
getAnalysis<BranchProbabilityInfoWrapperPass>().getBPI();
|
2015-07-15 01:40:50 +02:00
|
|
|
LoopInfo &LI = getAnalysis<LoopInfoWrapperPass>().getLoopInfo();
|
|
|
|
BFI.calculate(F, BPI, LI);
|
|
|
|
return false;
|
|
|
|
}
|
2016-05-05 23:13:27 +02:00
|
|
|
|
2016-11-23 18:53:26 +01:00
|
|
|
AnalysisKey BlockFrequencyAnalysis::Key;
|
2016-05-05 23:13:27 +02:00
|
|
|
BlockFrequencyInfo BlockFrequencyAnalysis::run(Function &F,
|
2016-08-09 02:28:15 +02:00
|
|
|
FunctionAnalysisManager &AM) {
|
2016-05-05 23:13:27 +02:00
|
|
|
BlockFrequencyInfo BFI;
|
|
|
|
BFI.calculate(F, AM.getResult<BranchProbabilityAnalysis>(F),
|
|
|
|
AM.getResult<LoopAnalysis>(F));
|
|
|
|
return BFI;
|
|
|
|
}
|
|
|
|
|
|
|
|
PreservedAnalyses
|
2016-08-09 02:28:15 +02:00
|
|
|
BlockFrequencyPrinterPass::run(Function &F, FunctionAnalysisManager &AM) {
|
2016-05-05 23:13:27 +02:00
|
|
|
OS << "Printing analysis results of BFI for function "
|
|
|
|
<< "'" << F.getName() << "':"
|
|
|
|
<< "\n";
|
|
|
|
AM.getResult<BlockFrequencyAnalysis>(F).print(OS);
|
|
|
|
return PreservedAnalyses::all();
|
|
|
|
}
|