blob: 57e856c8fc5051dc7afc56ce866a202e41397d4f [file] [log] [blame] [edit]
#pragma once
#include "llvm/ADT/ArrayRef.h"
#include "llvm/ADT/DenseMap.h"
#include "llvm/ADT/SetVector.h"
#include "llvm/IR/BasicBlock.h"
#include <map>
#include <set>
#include <vector>
namespace llvm {
class AllocaInst;
class BasicBlock;
class Function;
class Instruction;
class Use;
class Value;
} // namespace llvm
typedef std::set<llvm::BasicBlock *> BasicBlockSet;
typedef llvm::SetVector<llvm::Instruction *> InstructionSetVector;
// Compute live values at specified instructions.
class LiveValues {
public:
LiveValues(llvm::ArrayRef<llvm::Instruction *> computeLiveAt);
// Compute live values at specified instructions (computeLiveAt)
void run();
// Returns all values that are live at the index.
const InstructionSetVector &getLiveValues(unsigned int index) const {
return m_liveSets[index];
}
// Returns all live values, excluding allocas.
const InstructionSetVector &getAllLiveValues() const { return m_allLiveSet; }
// Update the live sets using the map
void remapLiveValues(
llvm::DenseMap<llvm::Instruction *, llvm::Instruction *> &imap);
typedef llvm::SetVector<unsigned int> Indices;
// Return all indices at which the given value is live.
const Indices *getIndicesWhereLive(const llvm::Value *value) const;
// For the two given values, check if they are both live at any of the
// marker instructions. This does not perform a true "lifetime overlap"
// test, it considers values to be disjoint if they have disjoint sets of
// markers.
// For example, value A is live at call sites 0, 1, 2, value B is live at
// 3, 4, where A is used for the last time between 2 and 3 and B is defined
// before that use. A and B will be considered "disjoint" in the sense of
// this method, even though the lifetimes of their values overlap.
bool liveInDisjointRegions(const llvm::Value *valueA,
const llvm::Value *valueB) const;
// Return true if the given value is live at the given index.
bool getLiveAtIndex(const llvm::Value *value, unsigned int index) const;
// Update the analysis manually. Use only if you know exactly what you are
// doing and document the reason thoroughly.
void setLiveAtIndex(llvm::Value *value, unsigned int index, bool live);
void setLiveAtAllIndices(llvm::Value *value, bool live);
void setIndicesWhereLive(llvm::Value *value, const Indices *indices);
private:
llvm::Function *m_function = nullptr;
std::vector<InstructionSetVector> m_liveSets;
InstructionSetVector m_allLiveSet;
llvm::SmallSet<llvm::BasicBlock *, 8> m_activeBlocks;
llvm::DenseMap<llvm::Instruction *, unsigned int> m_computeLiveAtIndex;
llvm::DenseMap<const llvm::Value *, Indices> m_liveAtIndices;
typedef llvm::SmallSet<llvm::BasicBlock *, 8> BlockSet;
void markLiveRange(llvm::Instruction *value, llvm::BasicBlock::iterator begin,
llvm::BasicBlock::iterator end);
void upAndMark(llvm::Instruction *v, llvm::Use &use, BlockSet &scanned);
};