mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2024-11-22 18:54:02 +01:00
157 lines
5.2 KiB
C++
157 lines
5.2 KiB
C++
//===-- ProfiledCallGraph.h - Profiled Call Graph ----------------- C++ -*-===//
|
|
//
|
|
// 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef LLVM_TOOLS_LLVM_PROFGEN_PROFILEDCALLGRAPH_H
|
|
#define LLVM_TOOLS_LLVM_PROFGEN_PROFILEDCALLGRAPH_H
|
|
|
|
#include "llvm/ADT/GraphTraits.h"
|
|
#include "llvm/ADT/StringMap.h"
|
|
#include "llvm/ADT/StringRef.h"
|
|
#include "llvm/ProfileData/SampleProf.h"
|
|
#include "llvm/ProfileData/SampleProfReader.h"
|
|
#include "llvm/Transforms/IPO/SampleContextTracker.h"
|
|
#include <queue>
|
|
#include <set>
|
|
|
|
using namespace llvm;
|
|
using namespace sampleprof;
|
|
|
|
namespace llvm {
|
|
namespace sampleprof {
|
|
|
|
struct ProfiledCallGraphNode {
|
|
ProfiledCallGraphNode(StringRef FName = StringRef()) : Name(FName) {}
|
|
StringRef Name;
|
|
|
|
struct ProfiledCallGraphNodeComparer {
|
|
bool operator()(const ProfiledCallGraphNode *L,
|
|
const ProfiledCallGraphNode *R) const {
|
|
return L->Name < R->Name;
|
|
}
|
|
};
|
|
std::set<ProfiledCallGraphNode *, ProfiledCallGraphNodeComparer> Callees;
|
|
};
|
|
|
|
class ProfiledCallGraph {
|
|
public:
|
|
using iterator = std::set<ProfiledCallGraphNode *>::iterator;
|
|
|
|
// Constructor for non-CS profile.
|
|
ProfiledCallGraph(StringMap<FunctionSamples> &ProfileMap) {
|
|
assert(!FunctionSamples::ProfileIsCS && "CS profile is not handled here");
|
|
for (const auto &Samples : ProfileMap) {
|
|
addProfiledCalls(Samples.second);
|
|
}
|
|
}
|
|
|
|
// Constructor for CS profile.
|
|
ProfiledCallGraph(SampleContextTracker &ContextTracker) {
|
|
// BFS traverse the context profile trie to add call edges for calls shown
|
|
// in context.
|
|
std::queue<ContextTrieNode *> Queue;
|
|
for (auto &Child : ContextTracker.getRootContext().getAllChildContext()) {
|
|
ContextTrieNode *Callee = &Child.second;
|
|
addProfiledFunction(Callee->getFuncName());
|
|
Queue.push(Callee);
|
|
}
|
|
|
|
while (!Queue.empty()) {
|
|
ContextTrieNode *Caller = Queue.front();
|
|
Queue.pop();
|
|
// Add calls for context. When AddNodeWithSamplesOnly is true, both caller
|
|
// and callee need to have context profile.
|
|
// Note that callsite target samples are completely ignored since they can
|
|
// conflict with the context edges, which are formed by context
|
|
// compression during profile generation, for cyclic SCCs. This may
|
|
// further result in an SCC order incompatible with the purely
|
|
// context-based one, which may in turn block context-based inlining.
|
|
for (auto &Child : Caller->getAllChildContext()) {
|
|
ContextTrieNode *Callee = &Child.second;
|
|
addProfiledFunction(Callee->getFuncName());
|
|
Queue.push(Callee);
|
|
addProfiledCall(Caller->getFuncName(), Callee->getFuncName());
|
|
}
|
|
}
|
|
}
|
|
|
|
iterator begin() { return Root.Callees.begin(); }
|
|
iterator end() { return Root.Callees.end(); }
|
|
ProfiledCallGraphNode *getEntryNode() { return &Root; }
|
|
void addProfiledFunction(StringRef Name) {
|
|
if (!ProfiledFunctions.count(Name)) {
|
|
// Link to synthetic root to make sure every node is reachable
|
|
// from root. This does not affect SCC order.
|
|
ProfiledFunctions[Name] = ProfiledCallGraphNode(Name);
|
|
Root.Callees.insert(&ProfiledFunctions[Name]);
|
|
}
|
|
}
|
|
|
|
void addProfiledCall(StringRef CallerName, StringRef CalleeName) {
|
|
assert(ProfiledFunctions.count(CallerName));
|
|
auto CalleeIt = ProfiledFunctions.find(CalleeName);
|
|
if (CalleeIt == ProfiledFunctions.end()) {
|
|
return;
|
|
}
|
|
ProfiledFunctions[CallerName].Callees.insert(&CalleeIt->second);
|
|
}
|
|
|
|
void addProfiledCalls(const FunctionSamples &Samples) {
|
|
addProfiledFunction(Samples.getFuncName());
|
|
|
|
for (const auto &Sample : Samples.getBodySamples()) {
|
|
for (const auto &Target : Sample.second.getCallTargets()) {
|
|
addProfiledFunction(Target.first());
|
|
addProfiledCall(Samples.getFuncName(), Target.first());
|
|
}
|
|
}
|
|
|
|
for (const auto &CallsiteSamples : Samples.getCallsiteSamples()) {
|
|
for (const auto &InlinedSamples : CallsiteSamples.second) {
|
|
addProfiledFunction(InlinedSamples.first);
|
|
addProfiledCall(Samples.getFuncName(), InlinedSamples.first);
|
|
addProfiledCalls(InlinedSamples.second);
|
|
}
|
|
}
|
|
}
|
|
|
|
private:
|
|
ProfiledCallGraphNode Root;
|
|
StringMap<ProfiledCallGraphNode> ProfiledFunctions;
|
|
};
|
|
|
|
} // end namespace sampleprof
|
|
|
|
template <> struct GraphTraits<ProfiledCallGraphNode *> {
|
|
using NodeRef = ProfiledCallGraphNode *;
|
|
using ChildIteratorType = std::set<ProfiledCallGraphNode *>::iterator;
|
|
|
|
static NodeRef getEntryNode(NodeRef PCGN) { return PCGN; }
|
|
static ChildIteratorType child_begin(NodeRef N) { return N->Callees.begin(); }
|
|
static ChildIteratorType child_end(NodeRef N) { return N->Callees.end(); }
|
|
};
|
|
|
|
template <>
|
|
struct GraphTraits<ProfiledCallGraph *>
|
|
: public GraphTraits<ProfiledCallGraphNode *> {
|
|
static NodeRef getEntryNode(ProfiledCallGraph *PCG) {
|
|
return PCG->getEntryNode();
|
|
}
|
|
|
|
static ChildIteratorType nodes_begin(ProfiledCallGraph *PCG) {
|
|
return PCG->begin();
|
|
}
|
|
|
|
static ChildIteratorType nodes_end(ProfiledCallGraph *PCG) {
|
|
return PCG->end();
|
|
}
|
|
};
|
|
|
|
} // end namespace llvm
|
|
|
|
#endif
|