//===- Profile.h - XRay Profile Abstraction -------------------------------===//
//
// The LLVM Compiler Infrastructure
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
//
// Defines the XRay Profile class representing the latency profile generated by
// XRay's profiling mode.
//
//===----------------------------------------------------------------------===//
#ifndef LLVM_XRAY_PROFILE_H
#define LLVM_XRAY_PROFILE_H
#include "llvm/ADT/DenseMap.h"
#include "llvm/ADT/SmallVector.h"
#include "llvm/ADT/StringRef.h"
#include "llvm/Support/Error.h"
#include <list>
#include <utility>
#include <vector>
namespace llvm {
namespace xray {
class Profile;
// We forward declare the Trace type for turning a Trace into a Profile.
class Trace;
/// This function will attempt to load an XRay Profiling Mode profile from the
/// provided |Filename|.
///
/// For any errors encountered in the loading of the profile data from
/// |Filename|, this function will return an Error condition appropriately.
Expected<Profile> loadProfile(StringRef Filename);
/// This algorithm will merge two Profile instances into a single Profile
/// instance, aggregating blocks by Thread ID.
Profile mergeProfilesByThread(const Profile &L, const Profile &R);
/// This algorithm will merge two Profile instances into a single Profile
/// instance, aggregating blocks by function call stack.
Profile mergeProfilesByStack(const Profile &L, const Profile &R);
/// This function takes a Trace and creates a Profile instance from it.
Expected<Profile> profileFromTrace(const Trace &T);
/// Profile instances are thread-compatible.
class Profile {
public:
using ThreadID = uint64_t;
using PathID = unsigned;
using FuncID = int32_t;
struct Data {
uint64_t CallCount;
uint64_t CumulativeLocalTime;
};
struct Block {
ThreadID Thread;
std::vector<std::pair<PathID, Data>> PathData;
};
/// Provides a sequence of function IDs from a previously interned PathID.
///
/// Returns an error if |P| had not been interned before into the Profile.
///
Expected<std::vector<FuncID>> expandPath(PathID P) const;
/// The stack represented in |P| must be in stack order (leaf to root). This
/// will always return the same PathID for |P| that has the same sequence.
PathID internPath(ArrayRef<FuncID> P);
/// Appends a fully-formed Block instance into the Profile.
///
/// Returns an error condition in the following cases:
///
/// - The PathData component of the Block is empty
///
Error addBlock(Block &&B);
Profile() = default;
~Profile() = default;
Profile(Profile &&O) noexcept
: Blocks(std::move(O.Blocks)), NodeStorage(std::move(O.NodeStorage)),
Roots(std::move(O.Roots)), PathIDMap(std::move(O.PathIDMap)),
NextID(O.NextID) {}
Profile &operator=(Profile &&O) noexcept {
Blocks = std::move(O.Blocks);
NodeStorage = std::move(O.NodeStorage);
Roots = std::move(O.Roots);
PathIDMap = std::move(O.PathIDMap);
NextID = O.NextID;
return *this;
}
Profile(const Profile &);
Profile &operator=(const Profile &);
friend void swap(Profile &L, Profile &R) {
using std::swap;
swap(L.Blocks, R.Blocks);
swap(L.NodeStorage, R.NodeStorage);
swap(L.Roots, R.Roots);
swap(L.PathIDMap, R.PathIDMap);
swap(L.NextID, R.NextID);
}
private:
using BlockList = std::list<Block>;
struct TrieNode {
FuncID Func = 0;
std::vector<TrieNode *> Callees{};
TrieNode *Caller = nullptr;
PathID ID = 0;
};
// List of blocks associated with a Profile.
BlockList Blocks;
// List of TrieNode elements we've seen.
std::list<TrieNode> NodeStorage;
// List of call stack roots.
SmallVector<TrieNode *, 4> Roots;
// Reverse mapping between a PathID to a TrieNode*.
DenseMap<PathID, TrieNode *> PathIDMap;
// Used to identify paths.
PathID NextID = 1;
public:
using const_iterator = BlockList::const_iterator;
const_iterator begin() const { return Blocks.begin(); }
const_iterator end() const { return Blocks.end(); }
bool empty() const { return Blocks.empty(); }
};
} // namespace xray
} // namespace llvm
#endif