2010-09-18 09:07:10 +00:00
|
|
|
//===-- RegAllocPBQP.h ------------------------------------------*- C++ -*-===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file defines the PBQPBuilder interface, for classes which build PBQP
|
|
|
|
// instances to represent register allocation problems, and the RegAllocPBQP
|
|
|
|
// interface.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#ifndef LLVM_CODEGEN_REGALLOCPBQP_H
|
|
|
|
#define LLVM_CODEGEN_REGALLOCPBQP_H
|
|
|
|
|
|
|
|
#include "llvm/ADT/DenseMap.h"
|
2014-03-03 18:50:05 +00:00
|
|
|
#include "llvm/ADT/SmallVector.h"
|
2010-09-18 09:07:10 +00:00
|
|
|
#include "llvm/CodeGen/MachineFunctionPass.h"
|
2014-03-03 18:50:05 +00:00
|
|
|
#include "llvm/CodeGen/PBQP/RegAllocSolver.h"
|
2010-09-18 09:07:10 +00:00
|
|
|
#include <map>
|
2010-12-08 22:15:32 +00:00
|
|
|
#include <set>
|
2010-09-18 09:07:10 +00:00
|
|
|
|
|
|
|
namespace llvm {
|
|
|
|
|
2010-12-08 22:15:32 +00:00
|
|
|
class LiveIntervals;
|
2013-06-17 19:00:36 +00:00
|
|
|
class MachineBlockFrequencyInfo;
|
2010-09-18 09:07:10 +00:00
|
|
|
class MachineFunction;
|
2012-12-03 16:50:05 +00:00
|
|
|
class TargetRegisterInfo;
|
2010-09-18 09:07:10 +00:00
|
|
|
|
2014-03-03 18:50:05 +00:00
|
|
|
typedef PBQP::RegAlloc::Graph PBQPRAGraph;
|
|
|
|
|
2010-09-18 09:07:10 +00:00
|
|
|
/// This class wraps up a PBQP instance representing a register allocation
|
|
|
|
/// problem, plus the structures necessary to map back from the PBQP solution
|
|
|
|
/// to a register allocation solution. (i.e. The PBQP-node <--> vreg map,
|
|
|
|
/// and the PBQP option <--> storage location map).
|
|
|
|
class PBQPRAProblem {
|
|
|
|
public:
|
|
|
|
|
|
|
|
typedef SmallVector<unsigned, 16> AllowedSet;
|
|
|
|
|
2014-03-03 18:50:05 +00:00
|
|
|
PBQPRAGraph& getGraph() { return graph; }
|
2010-09-18 09:07:10 +00:00
|
|
|
|
2014-03-03 18:50:05 +00:00
|
|
|
const PBQPRAGraph& getGraph() const { return graph; }
|
2010-09-18 09:07:10 +00:00
|
|
|
|
|
|
|
/// Record the mapping between the given virtual register and PBQP node,
|
|
|
|
/// and the set of allowed pregs for the vreg.
|
|
|
|
///
|
|
|
|
/// If you are extending
|
|
|
|
/// PBQPBuilder you are unlikely to need this: Nodes and options for all
|
2014-03-03 18:50:05 +00:00
|
|
|
/// vregs will already have been set up for you by the base class.
|
2010-09-18 09:07:10 +00:00
|
|
|
template <typename AllowedRegsItr>
|
2014-03-03 18:50:05 +00:00
|
|
|
void recordVReg(unsigned vreg, PBQPRAGraph::NodeId nodeId,
|
2010-09-18 09:07:10 +00:00
|
|
|
AllowedRegsItr arBegin, AllowedRegsItr arEnd) {
|
2013-11-09 03:08:56 +00:00
|
|
|
assert(node2VReg.find(nodeId) == node2VReg.end() && "Re-mapping node.");
|
2010-09-18 09:07:10 +00:00
|
|
|
assert(vreg2Node.find(vreg) == vreg2Node.end() && "Re-mapping vreg.");
|
|
|
|
assert(allowedSets[vreg].empty() && "vreg already has pregs.");
|
|
|
|
|
2013-11-09 03:08:56 +00:00
|
|
|
node2VReg[nodeId] = vreg;
|
|
|
|
vreg2Node[vreg] = nodeId;
|
2010-09-18 09:07:10 +00:00
|
|
|
std::copy(arBegin, arEnd, std::back_inserter(allowedSets[vreg]));
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the virtual register corresponding to the given PBQP node.
|
2014-03-03 18:50:05 +00:00
|
|
|
unsigned getVRegForNode(PBQPRAGraph::NodeId nodeId) const;
|
2010-09-18 09:07:10 +00:00
|
|
|
|
|
|
|
/// Get the PBQP node corresponding to the given virtual register.
|
2014-03-03 18:50:05 +00:00
|
|
|
PBQPRAGraph::NodeId getNodeForVReg(unsigned vreg) const;
|
2010-09-18 09:07:10 +00:00
|
|
|
|
|
|
|
/// Returns true if the given PBQP option represents a physical register,
|
|
|
|
/// false otherwise.
|
|
|
|
bool isPRegOption(unsigned vreg, unsigned option) const {
|
|
|
|
// At present we only have spills or pregs, so anything that's not a
|
|
|
|
// spill is a preg. (This might be extended one day to support remat).
|
|
|
|
return !isSpillOption(vreg, option);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns true if the given PBQP option represents spilling, false
|
|
|
|
/// otherwise.
|
|
|
|
bool isSpillOption(unsigned vreg, unsigned option) const {
|
|
|
|
// We hardcode option zero as the spill option.
|
|
|
|
return option == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the allowed set for the given virtual register.
|
|
|
|
const AllowedSet& getAllowedSet(unsigned vreg) const;
|
|
|
|
|
|
|
|
/// Get PReg for option.
|
|
|
|
unsigned getPRegForOption(unsigned vreg, unsigned option) const;
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
2014-03-03 18:50:05 +00:00
|
|
|
typedef std::map<PBQPRAGraph::NodeId, unsigned> Node2VReg;
|
|
|
|
typedef DenseMap<unsigned, PBQPRAGraph::NodeId> VReg2Node;
|
2011-06-07 06:05:58 +00:00
|
|
|
typedef DenseMap<unsigned, AllowedSet> AllowedSetMap;
|
2010-09-18 09:07:10 +00:00
|
|
|
|
2014-03-03 18:50:05 +00:00
|
|
|
PBQPRAGraph graph;
|
2010-09-18 09:07:10 +00:00
|
|
|
Node2VReg node2VReg;
|
|
|
|
VReg2Node vreg2Node;
|
|
|
|
|
|
|
|
AllowedSetMap allowedSets;
|
2014-03-03 18:50:05 +00:00
|
|
|
|
2010-09-18 09:07:10 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/// Builds PBQP instances to represent register allocation problems. Includes
|
|
|
|
/// spill, interference and coalescing costs by default. You can extend this
|
|
|
|
/// class to support additional constraints for your architecture.
|
|
|
|
class PBQPBuilder {
|
|
|
|
private:
|
2012-09-17 06:59:23 +00:00
|
|
|
PBQPBuilder(const PBQPBuilder&) LLVM_DELETED_FUNCTION;
|
|
|
|
void operator=(const PBQPBuilder&) LLVM_DELETED_FUNCTION;
|
2010-09-18 09:07:10 +00:00
|
|
|
public:
|
|
|
|
|
|
|
|
typedef std::set<unsigned> RegSet;
|
2014-03-03 18:50:05 +00:00
|
|
|
|
2010-09-18 09:07:10 +00:00
|
|
|
/// Default constructor.
|
|
|
|
PBQPBuilder() {}
|
|
|
|
|
|
|
|
/// Clean up a PBQPBuilder.
|
|
|
|
virtual ~PBQPBuilder() {}
|
|
|
|
|
|
|
|
/// Build a PBQP instance to represent the register allocation problem for
|
|
|
|
/// the given MachineFunction.
|
2013-04-15 12:06:32 +00:00
|
|
|
virtual PBQPRAProblem *build(MachineFunction *mf, const LiveIntervals *lis,
|
2013-06-17 19:00:36 +00:00
|
|
|
const MachineBlockFrequencyInfo *mbfi,
|
2013-04-15 12:06:32 +00:00
|
|
|
const RegSet &vregs);
|
2010-09-18 09:07:10 +00:00
|
|
|
private:
|
|
|
|
|
|
|
|
void addSpillCosts(PBQP::Vector &costVec, PBQP::PBQPNum spillCost);
|
|
|
|
|
|
|
|
void addInterferenceCosts(PBQP::Matrix &costMat,
|
|
|
|
const PBQPRAProblem::AllowedSet &vr1Allowed,
|
|
|
|
const PBQPRAProblem::AllowedSet &vr2Allowed,
|
|
|
|
const TargetRegisterInfo *tri);
|
|
|
|
};
|
|
|
|
|
2010-09-21 13:19:36 +00:00
|
|
|
/// Extended builder which adds coalescing constraints to a problem.
|
|
|
|
class PBQPBuilderWithCoalescing : public PBQPBuilder {
|
|
|
|
public:
|
2014-03-03 18:50:05 +00:00
|
|
|
|
2010-09-21 13:19:36 +00:00
|
|
|
/// Build a PBQP instance to represent the register allocation problem for
|
|
|
|
/// the given MachineFunction.
|
2014-03-07 09:26:03 +00:00
|
|
|
PBQPRAProblem *build(MachineFunction *mf, const LiveIntervals *lis,
|
|
|
|
const MachineBlockFrequencyInfo *mbfi,
|
|
|
|
const RegSet &vregs) override;
|
2010-09-21 13:19:36 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
void addPhysRegCoalesce(PBQP::Vector &costVec, unsigned pregOption,
|
|
|
|
PBQP::PBQPNum benefit);
|
|
|
|
|
|
|
|
void addVirtRegCoalesce(PBQP::Matrix &costMat,
|
|
|
|
const PBQPRAProblem::AllowedSet &vr1Allowed,
|
|
|
|
const PBQPRAProblem::AllowedSet &vr2Allowed,
|
|
|
|
PBQP::PBQPNum benefit);
|
|
|
|
};
|
|
|
|
|
2014-03-06 05:51:42 +00:00
|
|
|
FunctionPass *
|
|
|
|
createPBQPRegisterAllocator(std::unique_ptr<PBQPBuilder> &builder,
|
2014-04-14 00:51:57 +00:00
|
|
|
char *customPassID = nullptr);
|
2010-09-18 09:07:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* LLVM_CODEGEN_REGALLOCPBQP_H */
|