45#include "llvm/Config/llvm-config.h"
60#define DEBUG_TYPE "regalloc"
62STATISTIC(NumSpilledRanges,
"Number of spilled live ranges");
63STATISTIC(NumSnippets,
"Number of spilled snippets");
65STATISTIC(NumSpillsRemoved,
"Number of spills removed");
67STATISTIC(NumReloadsRemoved,
"Number of reloads removed");
68STATISTIC(NumFolded,
"Number of folded stack accesses");
70STATISTIC(NumRemats,
"Number of rematerialized defs for spilling");
75 cl::desc(
"Restrict remat for statepoint operands"));
100 using MergeableSpillsMap =
102 MergeableSpillsMap MergeableSpills;
112 void rmRedundantSpills(
132 : MF(mf), LIS(Analyses.LIS), LSS(Analyses.LSS), MDT(Analyses.MDT),
133 VRM(vrm),
MRI(mf.getRegInfo()),
TII(*mf.getSubtarget().getInstrInfo()),
134 TRI(*mf.getSubtarget().getRegisterInfo()), MBFI(Analyses.MBFI),
135 IPA(LIS, mf.getNumBlockIDs()) {}
137 void addToMergeableSpills(
MachineInstr &Spill,
int StackSlot,
139 bool rmFromMergeableSpills(
MachineInstr &Spill,
int StackSlot);
140 void hoistAllSpills();
144class InlineSpiller :
public Spiller {
178 HoistSpillHelper HSpiller;
183 ~InlineSpiller()
override =
default;
188 : MF(MF), LIS(Analyses.LIS), LSS(Analyses.LSS), VRM(VRM),
189 MRI(MF.getRegInfo()),
TII(*MF.getSubtarget().getInstrInfo()),
190 TRI(*MF.getSubtarget().getRegisterInfo()), HSpiller(Analyses, MF, VRM),
200 void collectRegsToSpill();
211 void reMaterializeAll();
214 bool foldMemoryOperand(
ArrayRef<std::pair<MachineInstr *, unsigned>>,
227void Spiller::anchor() {}
233 return new InlineSpiller(Analyses, MF, VRM, VRAI);
252 if (!
TII.isCopyInstr(
MI))
275 "expected to see first instruction in bundle");
279 while (
I->isBundledWithSucc()) {
281 auto CopyInst =
TII.isCopyInstr(
MI);
307 if (MO.getReg().isVirtual())
314bool InlineSpiller::isSnippet(
const LiveInterval &SnipLI) {
328 if (!LIS.intervalIsInOneMBB(SnipLI))
334 for (
auto *VNI : SnipLI.
vnis()) {
336 if (
MI->getOpcode() == TargetOpcode::STATEPOINT)
346 RI =
MRI.reg_bundle_nodbg_begin(SnipLI.
reg()),
347 E =
MRI.reg_bundle_nodbg_end();
377void InlineSpiller::collectRegsToSpill() {
381 RegsToSpill.assign(1, Reg);
382 SnippetCopies.clear();
383 RegsReplaced.clear();
392 if (!isSibling(SnipReg))
395 if (!isSnippet(SnipLI))
397 SnippetCopies.insert(&
MI);
398 if (isRegToSpill(SnipReg))
400 RegsToSpill.push_back(SnipReg);
406bool InlineSpiller::isSibling(
Register Reg) {
407 return Reg.isVirtual() && VRM.getOriginal(Reg) == Original;
429bool InlineSpiller::hoistSpillInsideBB(
LiveInterval &SpillLI,
434 assert(VNI && VNI->
def ==
Idx.getRegSlot() &&
"Not defined by copy");
448 assert(StackInt &&
"No stack slot assigned yet.");
451 StackInt->MergeValueInAsValue(OrigLI, OrigVNI, StackInt->getValNumInfo(0));
453 << *StackInt <<
'\n');
457 eliminateRedundantSpills(SrcLI, SrcVNI);
465 assert(
DefMI &&
"Defining instruction disappeared");
473 LIS.InsertMachineInstrRangeInMaps(MIS.begin(), MII);
482 if (MIS.begin() == MII)
483 HSpiller.addToMergeableSpills(*MII, StackSlot, Original);
491 assert(VNI &&
"Missing value");
493 WorkList.
push_back(std::make_pair(&SLI, VNI));
494 assert(StackInt &&
"No stack slot assigned yet.");
501 << VNI->
def <<
" in " << *LI <<
'\n');
504 if (isRegToSpill(Reg))
508 StackInt->MergeValueInAsValue(*LI, VNI, StackInt->getValNumInfo(0));
509 LLVM_DEBUG(
dbgs() <<
"Merged to stack int: " << *StackInt <<
'\n');
514 if (!
MI.mayStore() && !
TII.isCopyInstr(
MI))
522 if (isSibling(DstReg)) {
525 assert(DstVNI &&
"Missing defined value");
526 assert(DstVNI->
def ==
Idx.getRegSlot() &&
"Wrong copy def slot");
528 WorkList.
push_back(std::make_pair(&DstLI, DstVNI));
538 MI.setDesc(
TII.get(TargetOpcode::KILL));
539 DeadDefs.push_back(&
MI);
541 if (HSpiller.rmFromMergeableSpills(
MI, StackSlot))
545 }
while (!WorkList.
empty());
556 WorkList.
push_back(std::make_pair(LI, VNI));
559 if (!UsedValues.insert(VNI).second)
567 WorkList.
push_back(std::make_pair(LI, PVNI));
574 if (!SnippetCopies.count(
MI))
576 LiveInterval &SnipLI = LIS.getInterval(
MI->getOperand(1).getReg());
577 assert(isRegToSpill(SnipLI.
reg()) &&
"Unexpected register in copy");
579 assert(SnipVNI &&
"Snippet undefined before copy");
580 WorkList.
push_back(std::make_pair(&SnipLI, SnipVNI));
581 }
while (!WorkList.
empty());
584bool InlineSpiller::canGuaranteeAssignmentAfterRemat(
Register VReg,
603 if (
MI.getOpcode() != TargetOpcode::STATEPOINT)
609 EndIdx =
MI.getNumOperands();
639 if (SnippetCopies.count(&
MI))
645 RM.OrigMI = LIS.getInstructionFromIndex(OrigVNI->
def);
647 if (!Edit->canRematerializeAt(RM, OrigVNI, UseIdx,
false)) {
648 markValueUsed(&VirtReg, ParentVNI);
656 markValueUsed(&VirtReg, ParentVNI);
663 if (
RM.OrigMI->canFoldAsLoad() &&
664 foldMemoryOperand(Ops,
RM.OrigMI)) {
665 Edit->markRematerialized(
RM.ParentVNI);
672 if (!canGuaranteeAssignmentAfterRemat(VirtReg.
reg(),
MI)) {
673 markValueUsed(&VirtReg, ParentVNI);
679 Register NewVReg = Edit->createFrom(Original);
683 Edit->rematerializeAt(*
MI.getParent(),
MI, NewVReg, RM,
TRI);
687 auto *NewMI = LIS.getInstructionFromIndex(DefIdx);
688 NewMI->setDebugLoc(
MI.getDebugLoc());
692 << *LIS.getInstructionFromIndex(DefIdx));
695 for (
const auto &OpPair : Ops) {
710void InlineSpiller::reMaterializeAll() {
711 if (!Edit->anyRematerializable())
717 bool anyRemat =
false;
722 if (
MI.isDebugValue())
725 assert(!
MI.isDebugInstr() &&
"Did not expect to find a use in debug "
726 "instruction that isn't a DBG_VALUE");
728 anyRemat |= reMaterializeFor(LI,
MI);
741 MI->addRegisterDead(Reg, &
TRI);
742 if (!
MI->allDefsAreDead())
745 DeadDefs.push_back(
MI);
749 if (
MI->isBundledWithSucc() && !
MI->isBundledWithPred()) {
751 EndIt =
MI->getParent()->instr_end();
754 bool OnlyDeadCopies =
true;
756 It != EndIt && It->isBundledWithPred(); ++It) {
758 auto DestSrc =
TII.isCopyInstr(*It);
759 bool IsCopyToDeadReg =
760 DestSrc && DestSrc->Destination->getReg() ==
Reg;
761 if (!IsCopyToDeadReg) {
762 OnlyDeadCopies =
false;
766 if (OnlyDeadCopies) {
768 It != EndIt && It->isBundledWithPred(); ++It) {
769 It->addRegisterDead(Reg, &
TRI);
771 DeadDefs.push_back(&*It);
780 if (DeadDefs.empty())
782 LLVM_DEBUG(
dbgs() <<
"Remat created " << DeadDefs.size() <<
" dead defs.\n");
783 Edit->eliminateDeadDefs(DeadDefs, RegsToSpill);
791 unsigned ResultPos = 0;
793 if (
MRI.reg_nodbg_empty(Reg)) {
794 Edit->eraseVirtReg(Reg);
795 RegsReplaced.push_back(Reg);
799 assert(LIS.hasInterval(Reg) &&
800 (!LIS.getInterval(Reg).empty() || !
MRI.reg_nodbg_empty(Reg)) &&
801 "Empty and not used live-range?!");
803 RegsToSpill[ResultPos++] =
Reg;
805 RegsToSpill.erase(RegsToSpill.begin() + ResultPos, RegsToSpill.end());
807 <<
" registers to spill after remat.\n");
818 bool IsLoad = InstrReg;
823 if (InstrReg != Reg || FI != StackSlot)
827 HSpiller.rmFromMergeableSpills(*
MI, StackSlot);
830 LIS.RemoveMachineInstrFromMaps(*
MI);
831 MI->eraseFromParent();
844#if !defined(NDEBUG) || defined(LLVM_ENABLE_DUMP)
850 const char *
const header,
852 char NextLine =
'\n';
853 char SlotIndent =
'\t';
855 if (std::next(
B) == E) {
860 dbgs() <<
'\t' << header <<
": " << NextLine;
871 Idx =
Idx.getRegSlot(
true);
874 dbgs() << SlotIndent <<
Idx <<
'\t' << *
I;
886foldMemoryOperand(
ArrayRef<std::pair<MachineInstr *, unsigned>> Ops,
892 if (Ops.back().first !=
MI ||
MI->isBundled())
895 bool WasCopy =
TII.isCopyInstr(*MI).has_value();
904 bool UntieRegs =
MI->getOpcode() == TargetOpcode::STATEPOINT;
908 bool SpillSubRegs =
TII.isSubregFoldable() ||
909 MI->getOpcode() == TargetOpcode::STATEPOINT ||
910 MI->getOpcode() == TargetOpcode::PATCHPOINT ||
911 MI->getOpcode() == TargetOpcode::STACKMAP;
916 for (
const auto &OpPair : Ops) {
917 unsigned Idx = OpPair.second;
918 assert(
MI == OpPair.first &&
"Instruction conflict during operand folding");
935 if (LoadMI && MO.
isDef())
938 if (UntieRegs || !
MI->isRegTiedToDefOperand(
Idx))
951 for (
unsigned Idx : FoldOps) {
955 unsigned Tied =
MI->findTiedOperandIdx(
Idx);
962 MI->untieRegOperand(
Idx);
966 LoadMI ?
TII.foldMemoryOperand(*
MI, FoldOps, *LoadMI, &LIS)
967 :
TII.foldMemoryOperand(*
MI, FoldOps, StackSlot, &LIS, &VRM);
970 for (
auto Tied : TiedOps)
971 MI->tieOperands(Tied.first, Tied.second);
980 if (!Reg ||
Reg.isVirtual() ||
MRI.isReserved(Reg)) {
997 HSpiller.rmFromMergeableSpills(*
MI, FI))
1001 if (
MI->isCandidateForAdditionalCallInfo())
1002 MI->getMF()->moveAdditionalCallInfo(
MI, FoldMI);
1009 if (
MI->peekDebugInstrNum() && Ops[0].second == 0) {
1011 auto MakeSubstitution = [
this,FoldMI,
MI,&Ops]() {
1013 unsigned OldOperandNum = Ops[0].second;
1015 unsigned OldNum =
MI->getDebugInstrNum();
1016 MF.makeDebugValueSubstitution({OldNum, OldOperandNum},
1021 if (Ops.size() == 1 && Op0.
isDef()) {
1023 }
else if (Ops.size() == 2 && Op0.
isDef() &&
MI->getOperand(1).isTied() &&
1024 Op0.
getReg() ==
MI->getOperand(1).getReg()) {
1027 }
else if (
MI->peekDebugInstrNum()) {
1033 MF.substituteDebugValuesForInst(*
MI, *FoldMI, Ops[0].second);
1036 MI->eraseFromParent();
1039 assert(!MIS.empty() &&
"Unexpected empty span of instructions!");
1051 if (MO.
getReg() == ImpReg)
1060 else if (Ops.front().second == 0) {
1065 if (std::distance(MIS.begin(), MIS.end()) <= 1)
1066 HSpiller.addToMergeableSpills(*FoldMI, StackSlot, Original);
1072void InlineSpiller::insertReload(
Register NewVReg,
1092 if (!Def.isImplicitDef())
1098 return Def.getOperand(0).getSubReg();
1102void InlineSpiller::insertSpill(
Register NewVReg,
bool isKill,
1106 assert(!
MI->isTerminator() &&
"Inserting a spill after a terminator");
1121 BuildMI(
MBB, SpillBefore,
MI->getDebugLoc(),
TII.get(TargetOpcode::KILL))
1135 if (IsRealSpill && std::distance(Spill, MIS.end()) <= 1)
1136 HSpiller.addToMergeableSpills(*Spill, StackSlot, Original);
1140void InlineSpiller::spillAroundUses(
Register Reg) {
1147 if (
MI.isDebugValue()) {
1156 assert(!
MI.isDebugInstr() &&
"Did not expect to find a use in debug "
1157 "instruction that isn't a DBG_VALUE");
1160 if (SnippetCopies.count(&
MI))
1164 if (coalesceStackAccess(&
MI, Reg))
1180 if (SibReg && isSibling(SibReg)) {
1182 if (isRegToSpill(SibReg)) {
1184 SnippetCopies.insert(&
MI);
1188 if (hoistSpillInsideBB(OldLI,
MI)) {
1190 MI.getOperand(0).setIsDead();
1191 DeadDefs.push_back(&
MI);
1203 if (foldMemoryOperand(Ops))
1208 Register NewVReg = Edit->createFrom(Reg);
1211 insertReload(NewVReg,
Idx, &
MI);
1214 bool hasLiveDef =
false;
1215 for (
const auto &OpPair : Ops) {
1219 if (!OpPair.first->isRegTiedToDefOperand(OpPair.second))
1231 insertSpill(NewVReg,
true, &
MI);
1236void InlineSpiller::spillAll() {
1239 StackSlot = VRM.assignVirt2StackSlot(Original);
1240 StackInt = &LSS.getOrCreateInterval(StackSlot,
MRI.getRegClass(Original));
1241 StackInt->getNextValue(
SlotIndex(), LSS.getVNInfoAllocator());
1243 StackInt = &LSS.getInterval(StackSlot);
1245 if (Original != Edit->getReg())
1246 VRM.assignVirt2StackSlot(Edit->getReg(), StackSlot);
1248 assert(StackInt->getNumValNums() == 1 &&
"Bad stack interval values");
1250 StackInt->MergeSegmentsInAsValue(LIS.
getInterval(Reg),
1252 LLVM_DEBUG(
dbgs() <<
"Merged spilled regs: " << *StackInt <<
'\n');
1256 spillAroundUses(Reg);
1260 VRM.assignVirt2StackSlot(Reg, StackSlot);
1264 if (!DeadDefs.empty()) {
1265 LLVM_DEBUG(
dbgs() <<
"Eliminating " << DeadDefs.size() <<
" dead defs\n");
1266 Edit->eliminateDeadDefs(DeadDefs, RegsToSpill);
1273 assert(SnippetCopies.count(&
MI) &&
"Remaining use wasn't a snippet copy");
1276 MI.eraseFromBundle();
1282 Edit->eraseVirtReg(Reg);
1289 "Trying to spill a stack slot.");
1291 Original = VRM.getOriginal(edit.
getReg());
1292 StackSlot = VRM.getStackSlot(Original);
1297 <<
':' << edit.
getParent() <<
"\nFrom original "
1300 "Attempting to spill already spilled value.");
1301 assert(DeadDefs.empty() &&
"Previous spill didn't remove dead defs");
1303 collectRegsToSpill();
1307 if (!RegsToSpill.empty())
1310 Edit->calculateRegClassAndHint(MF, VRAI);
1314void InlineSpiller::postOptimization() { HSpiller.hoistAllSpills(); }
1317void HoistSpillHelper::addToMergeableSpills(
MachineInstr &Spill,
int StackSlot,
1318 unsigned Original) {
1323 if (!StackSlotToOrigLI.contains(StackSlot)) {
1324 auto LI = std::make_unique<LiveInterval>(OrigLI.
reg(), OrigLI.
weight());
1325 LI->
assign(OrigLI, Allocator);
1326 StackSlotToOrigLI[StackSlot] = std::move(LI);
1329 VNInfo *OrigVNI = StackSlotToOrigLI[StackSlot]->getVNInfoAt(
Idx.getRegSlot());
1330 std::pair<int, VNInfo *> MIdx = std::make_pair(StackSlot, OrigVNI);
1331 MergeableSpills[MIdx].insert(&Spill);
1336bool HoistSpillHelper::rmFromMergeableSpills(
MachineInstr &Spill,
1338 auto It = StackSlotToOrigLI.find(StackSlot);
1339 if (It == StackSlotToOrigLI.end())
1342 VNInfo *OrigVNI = It->second->getVNInfoAt(
Idx.getRegSlot());
1343 std::pair<int, VNInfo *> MIdx = std::make_pair(StackSlot, OrigVNI);
1344 return MergeableSpills[MIdx].erase(&Spill);
1357 LLVM_DEBUG(
dbgs() <<
"can't spill in root block - def after LIP\n");
1364 for (
const Register &SibReg : Siblings) {
1377void HoistSpillHelper::rmRedundantSpills(
1384 for (
auto *
const CurrentSpill : Spills) {
1391 MachineInstr *SpillToRm = (CIdx > PIdx) ? CurrentSpill : PrevSpill;
1392 MachineInstr *SpillToKeep = (CIdx > PIdx) ? PrevSpill : CurrentSpill;
1394 SpillBBToSpill[MDT.getNode(
Block)] = SpillToKeep;
1396 SpillBBToSpill[MDT.getNode(
Block)] = CurrentSpill;
1399 for (
auto *
const SpillToRm : SpillsToRm)
1400 Spills.
erase(SpillToRm);
1409void HoistSpillHelper::getVisitOrders(
1433 for (
auto *
const Spill : Spills) {
1437 while (
Node != RootIDomNode) {
1441 SpillToRm = SpillBBToSpill[MDT[
Block]];
1461 SpillsToKeep[MDT[
Block]] = 0;
1464 NodesOnPath.
clear();
1474 if (WorkSet.
count(Child))
1477 }
while (idx != Orders.
size());
1479 "Orders have different size with WorkSet");
1484 for (; RIt != Orders.
rend(); RIt++)
1485 LLVM_DEBUG(
dbgs() <<
"BB" << (*RIt)->getBlock()->getNumber() <<
",");
1493void HoistSpillHelper::runHoistSpills(
1509 rmRedundantSpills(Spills, SpillsToRm, SpillBBToSpill);
1512 getVisitOrders(Root, Spills, Orders, SpillsToRm, SpillsToKeep,
1519 using NodesCostPair =
1520 std::pair<SmallPtrSet<MachineDomTreeNode *, 16>,
BlockFrequency>;
1528 for (; RIt != Orders.
rend(); RIt++) {
1532 if (SpillsToKeep.
contains(*RIt) && !SpillsToKeep[*RIt]) {
1533 SpillsInSubTreeMap[*RIt].first.
insert(*RIt);
1535 SpillsInSubTreeMap[*RIt].second = MBFI.getBlockFreq(
Block);
1542 if (!SpillsInSubTreeMap.
contains(Child))
1550 SpillsInSubTreeMap[*RIt].first;
1552 SubTreeCost += SpillsInSubTreeMap[Child].second;
1553 auto BI = SpillsInSubTreeMap[Child].first.
begin();
1554 auto EI = SpillsInSubTreeMap[Child].first.
end();
1555 SpillsInSubTree.
insert(BI, EI);
1556 SpillsInSubTreeMap.
erase(Child);
1560 SpillsInSubTreeMap[*RIt].first;
1563 if (SpillsInSubTree.
empty())
1568 if (!isSpillCandBB(OrigLI, OrigVNI, *
Block, LiveReg))
1576 if (SubTreeCost > MBFI.getBlockFreq(
Block) * MarginProb) {
1578 for (
auto *
const SpillBB : SpillsInSubTree) {
1581 if (SpillsToKeep.
contains(SpillBB) && !SpillsToKeep[SpillBB]) {
1586 SpillsToKeep.
erase(SpillBB);
1590 SpillsToKeep[*RIt] = LiveReg;
1592 dbgs() <<
"spills in BB: ";
1593 for (
const auto Rspill : SpillsInSubTree)
1594 dbgs() << Rspill->getBlock()->getNumber() <<
" ";
1595 dbgs() <<
"were promoted to BB" << (*RIt)->getBlock()->getNumber()
1598 SpillsInSubTree.clear();
1599 SpillsInSubTree.insert(*RIt);
1600 SubTreeCost = MBFI.getBlockFreq(
Block);
1605 for (
const auto &Ent : SpillsToKeep) {
1607 SpillsToIns[Ent.first->getBlock()] = Ent.second;
1627void HoistSpillHelper::hoistAllSpills() {
1631 for (
unsigned i = 0, e =
MRI.getNumVirtRegs(); i != e; ++i) {
1633 Register Original = VRM.getPreSplitReg(Reg);
1634 if (!
MRI.def_empty(Reg))
1635 Virt2SiblingsMap[Original].insert(Reg);
1639 for (
auto &Ent : MergeableSpills) {
1640 int Slot = Ent.first.first;
1642 VNInfo *OrigVNI = Ent.first.second;
1644 if (Ent.second.empty())
1648 dbgs() <<
"\nFor Slot" <<
Slot <<
" and VN" << OrigVNI->
id <<
":\n"
1649 <<
"Equal spills in BB: ";
1650 for (
const auto spill : EqValSpills)
1651 dbgs() << spill->getParent()->getNumber() <<
" ";
1660 runHoistSpills(OrigLI, *OrigVNI, EqValSpills, SpillsToRm, SpillsToIns);
1663 dbgs() <<
"Finally inserted spills in BB: ";
1664 for (
const auto &Ispill : SpillsToIns)
1665 dbgs() << Ispill.first->getNumber() <<
" ";
1666 dbgs() <<
"\nFinally removed spills in BB: ";
1667 for (
const auto Rspill : SpillsToRm)
1668 dbgs() << Rspill->getParent()->getNumber() <<
" ";
1674 if (!SpillsToIns.empty() || !SpillsToRm.empty())
1676 StackIntvl.getValNumInfo(0));
1679 for (
auto const &Insert : SpillsToIns) {
1693 NumSpills -= SpillsToRm.size();
1694 for (
auto *
const RMEnt : SpillsToRm) {
1695 RMEnt->setDesc(
TII.get(TargetOpcode::KILL));
1696 for (
unsigned i = RMEnt->getNumOperands(); i; --i) {
1699 RMEnt->removeOperand(i - 1);
1702 Edit.eliminateDeadDefs(SpillsToRm, {});
1709 if (VRM.hasPhys(Old))
1710 VRM.assignVirt2Phys(New, VRM.getPhys(Old));
1712 VRM.assignVirt2StackSlot(New, VRM.getStackSlot(Old));
1715 if (VRM.hasShape(Old))
1716 VRM.assignVirt2Shape(New, VRM.getShape(Old));
unsigned const MachineRegisterInfo * MRI
MachineInstrBuilder & UseMI
MachineInstrBuilder MachineInstrBuilder & DefMI
static GCRegistry::Add< OcamlGC > B("ocaml", "ocaml 3.10-compatible GC")
#define LLVM_DUMP_METHOD
Mark debug helper function definitions like dump() that should not be stripped from debug builds.
Returns the sub type a function will return at a given Idx Should correspond to the result type of an ExtractValue instruction executed with just that one unsigned Idx
This file defines the DenseMap class.
const HexagonInstrInfo * TII
static LLVM_DUMP_METHOD void dumpMachineInstrRangeWithSlotIndex(MachineBasicBlock::iterator B, MachineBasicBlock::iterator E, LiveIntervals const &LIS, const char *const header, Register VReg=Register())
static Register isCopyOfBundle(const MachineInstr &FirstMI, Register Reg, const TargetInstrInfo &TII)
Check for a copy bundle as formed by SplitKit.
static bool isRealSpill(const MachineInstr &Def)
Check if Def fully defines a VReg with an undefined value.
static cl::opt< bool > RestrictStatepointRemat("restrict-statepoint-remat", cl::init(false), cl::Hidden, cl::desc("Restrict remat for statepoint operands"))
static Register isCopyOf(const MachineInstr &MI, Register Reg, const TargetInstrInfo &TII)
isFullCopyOf - If MI is a COPY to or from Reg, return the other register, otherwise return 0.
static void getVDefInterval(const MachineInstr &MI, LiveIntervals &LIS)
unsigned const TargetRegisterInfo * TRI
This file implements a map that provides insertion order iteration.
assert(ImpDefSCC.getReg()==AMDGPU::SCC &&ImpDefSCC.isDef())
This file implements a set that has insertion order iteration characteristics.
This file defines the SmallPtrSet class.
This file defines the SmallVector class.
This file defines the 'Statistic' class, which is designed to be an easy way to expose various metric...
#define STATISTIC(VARNAME, DESC)
ArrayRef - Represent a constant reference to an array (0 or more elements consecutively in memory),...
Allocate memory in an ever growing pool, as if by bump-pointer.
bool erase(const KeyT &Val)
bool contains(const_arg_type_t< KeyT > Val) const
Return true if the specified key is in the map, false otherwise.
std::pair< iterator, bool > insert(const std::pair< KeyT, ValueT > &KV)
Base class for the actual dominator tree node.
iterator_range< iterator > children()
DomTreeNodeBase * getIDom() const
void storeRegToStackSlot(MachineBasicBlock &MBB, MachineBasicBlock::iterator MBBI, Register SrcReg, bool isKill, int FrameIndex, const TargetRegisterClass *RC, const TargetRegisterInfo *TRI, Register VReg) const override
Store the specified register of the given register class to the specified stack frame index.
Register isLoadFromStackSlot(const MachineInstr &MI, int &FrameIndex) const override
TargetInstrInfo overrides.
void loadRegFromStackSlot(MachineBasicBlock &MBB, MachineBasicBlock::iterator MBBI, Register DestReg, int FrameIndex, const TargetRegisterClass *RC, const TargetRegisterInfo *TRI, Register VReg) const override
Load the specified register of the given register class from the specified stack frame index.
Register isStoreToStackSlot(const MachineInstr &MI, int &FrameIndex) const override
If the specified machine instruction is a direct store to a stack slot, return the virtual or physica...
Determines the latest safe point in a block in which we can insert a split, spill or other instructio...
LiveInterval - This class represents the liveness of a register, or stack slot.
bool isSpillable() const
isSpillable - Can this interval be spilled?
SlotIndex InsertMachineInstrInMaps(MachineInstr &MI)
SlotIndexes * getSlotIndexes() const
SlotIndex getInstructionIndex(const MachineInstr &Instr) const
Returns the base index of the given instruction.
VNInfo::Allocator & getVNInfoAllocator()
LiveInterval & getInterval(Register Reg)
void InsertMachineInstrRangeInMaps(MachineBasicBlock::iterator B, MachineBasicBlock::iterator E)
void removePhysRegDefAt(MCRegister Reg, SlotIndex Pos)
Remove value numbers and related live segments starting at position Pos that are part of any liverang...
MachineBasicBlock * getMBBFromIndex(SlotIndex index) const
SlotIndex ReplaceMachineInstrInMaps(MachineInstr &MI, MachineInstr &NewMI)
Result of a LiveRange query.
bool isKill() const
Return true if the live-in value is killed by this instruction.
Callback methods for LiveRangeEdit owners.
virtual void LRE_DidCloneVirtReg(Register New, Register Old)
Called after cloning a virtual register.
const LiveInterval & getParent() const
VNInfo * getValNumInfo(unsigned ValNo)
getValNumInfo - Returns pointer to the specified val#.
void MergeValueInAsValue(const LiveRange &RHS, const VNInfo *RHSValNo, VNInfo *LHSValNo)
MergeValueInAsValue - Merge all of the segments of a specific val# in RHS into this live range as the...
iterator_range< vni_iterator > vnis()
LiveQueryResult Query(SlotIndex Idx) const
Query Liveness at Idx.
VNInfo * getVNInfoBefore(SlotIndex Idx) const
getVNInfoBefore - Return the VNInfo that is live up to but not necessarily including Idx,...
unsigned getNumValNums() const
void assign(const LiveRange &Other, BumpPtrAllocator &Allocator)
Copies values numbers and live segments from Other into this range.
VNInfo * getVNInfoAt(SlotIndex Idx) const
getVNInfoAt - Return the VNInfo that is live at Idx, or NULL.
MIBundleOperands - Iterate over all operands in a bundle of machine instructions.
iterator SkipPHIsLabelsAndDebug(iterator I, Register Reg=Register(), bool SkipPseudoOp=true)
Return the first instruction in MBB after I that is not a PHI, label or debug.
Instructions::iterator instr_iterator
Instructions::const_iterator const_instr_iterator
instr_iterator erase(instr_iterator I)
Remove an instruction from the instruction list and delete it.
iterator_range< pred_iterator > predecessors()
MachineBlockFrequencyInfo pass uses BlockFrequencyInfoImpl implementation to estimate machine basic b...
DominatorTree Class - Concrete subclass of DominatorTreeBase that is used to compute a normal dominat...
static const unsigned int DebugOperandMemNumber
A reserved operand number representing the instructions memory operand, for instructions that have a ...
const MachineInstrBuilder & addReg(Register RegNo, unsigned flags=0, unsigned SubReg=0) const
Add a new virtual register operand.
MachineInstrSpan provides an interface to get an iteration range containing the instruction it was in...
Representation of each machine instruction.
const MachineBasicBlock * getParent() const
unsigned getNumOperands() const
Retuns the total number of operands.
bool isBundledWithPred() const
Return true if this instruction is part of a bundle, and it is not the first instruction in the bundl...
void removeOperand(unsigned OpNo)
Erase an operand from an instruction, leaving it with one fewer operand than it started with.
bool isBundledWithSucc() const
Return true if this instruction is part of a bundle, and it is not the last instruction in the bundle...
unsigned getDebugInstrNum()
Fetch the instruction number of this MachineInstr.
const MachineOperand & getOperand(unsigned i) const
bool isBundled() const
Return true if this instruction part of a bundle.
MachineOperand class - Representation of each machine instruction operand.
unsigned getSubReg() const
bool readsReg() const
readsReg - Returns true if this operand reads the previous value of its register.
bool isReg() const
isReg - Tests if this is a MO_Register operand.
void setReg(Register Reg)
Change the register this operand corresponds to.
void setIsKill(bool Val=true)
void setIsUndef(bool Val=true)
bool isEarlyClobber() const
Register getReg() const
getReg - Returns the register number.
defusechain_iterator - This class provides iterator support for machine operands in the function that...
MachineRegisterInfo - Keep track of information for virtual and physical registers,...
This class implements a map that also provides access to all stored values in a deterministic order.
Wrapper class representing virtual and physical registers.
static constexpr bool isStackSlot(unsigned Reg)
isStackSlot - Sometimes it is useful to be able to store a non-negative frame index in a variable tha...
static Register index2VirtReg(unsigned Index)
Convert a 0-based index to a virtual register number.
SlotIndex - An opaque wrapper around machine indexes.
static bool isSameInstr(SlotIndex A, SlotIndex B)
isSameInstr - Return true if A and B refer to the same instruction.
SlotIndex getBaseIndex() const
Returns the base index for associated with this index.
SlotIndex getRegSlot(bool EC=false) const
Returns the register use/def slot in the current instruction for a normal or early-clobber def.
void removeSingleMachineInstrFromMaps(MachineInstr &MI)
Removes a single machine instruction MI from the mapping.
size_type count(ConstPtrType Ptr) const
count - Return 1 if the specified pointer is in the set, 0 otherwise.
std::pair< iterator, bool > insert(PtrType Ptr)
Inserts Ptr if and only if there is no element in the container equal to Ptr.
SmallPtrSet - This class implements a set which is optimized for holding SmallSize or less elements.
A SetVector that performs no allocations if smaller than a certain size.
This class consists of common code factored out of the SmallVector class to reduce code duplication b...
reference emplace_back(ArgTypes &&... Args)
void push_back(const T &Elt)
reverse_iterator rbegin()
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
virtual void spill(LiveRangeEdit &LRE)=0
spill - Spill the LRE.getParent() live interval.
virtual ArrayRef< Register > getReplacedRegs()=0
Return registers that were not spilled, but otherwise replaced (e.g.
virtual ArrayRef< Register > getSpilledRegs()=0
Return the registers that were spilled.
virtual void postOptimization()
MI-level Statepoint operands.
bool isFoldableReg(Register Reg) const
Return true if Reg is used only in operands which can be folded to stack usage.
TargetInstrInfo - Interface to description of machine instruction set.
TargetRegisterInfo base class - We assume that the target defines a static array of TargetRegisterDes...
VNInfo - Value Number Information.
bool isUnused() const
Returns true if this value is unused.
unsigned id
The ID number of this value.
SlotIndex def
The index of the defining instruction.
bool isPHIDef() const
Returns true if this value is defined by a PHI instruction (or was, PHI instructions may have been el...
Calculate auxiliary information for a virtual register such as its spill weight and allocation hint.
static constexpr int NO_STACK_SLOT
self_iterator getIterator()
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
Reg
All possible values of the reg field in the ModR/M byte.
initializer< Ty > init(const Ty &Val)
This is an optimization pass for GlobalISel generic memory operations.
MachineInstrBuilder BuildMI(MachineFunction &MF, const MIMetadata &MIMD, const MCInstrDesc &MCID)
Builder interface. Specify how to create the initial instruction itself.
iterator_range< T > make_range(T x, T y)
Convenience function for iterating over sub-ranges.
PhysRegInfo AnalyzePhysRegInBundle(const MachineInstr &MI, Register Reg, const TargetRegisterInfo *TRI)
AnalyzePhysRegInBundle - Analyze how the current instruction or bundle uses a physical register.
iterator_range< early_inc_iterator_impl< detail::IterOfRange< RangeT > > > make_early_inc_range(RangeT &&Range)
Make a range that does early increment to allow mutation of the underlying range without disrupting i...
raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
Spiller * createInlineSpiller(const Spiller::RequiredAnalyses &Analyses, MachineFunction &MF, VirtRegMap &VRM, VirtRegAuxInfo &VRAI)
Create and return a spiller that will insert spill code directly instead of deferring though VirtRegM...
VirtRegInfo AnalyzeVirtRegInBundle(MachineInstr &MI, Register Reg, SmallVectorImpl< std::pair< MachineInstr *, unsigned > > *Ops=nullptr)
AnalyzeVirtRegInBundle - Analyze how the current instruction or bundle uses a virtual register.
unsigned getKillRegState(bool B)
MachineInstr * buildDbgValueForSpill(MachineBasicBlock &BB, MachineBasicBlock::iterator I, const MachineInstr &Orig, int FrameIndex, Register SpillReg)
Clone a DBG_VALUE whose value has been spilled to FrameIndex.
bool is_contained(R &&Range, const E &Element)
Returns true if Element is found in Range.
Printable printReg(Register Reg, const TargetRegisterInfo *TRI=nullptr, unsigned SubIdx=0, const MachineRegisterInfo *MRI=nullptr)
Prints virtual and physical registers with or without a TRI instance.
Remat - Information needed to rematerialize at a specific location.
Information about how a physical register Reg is used by a set of operands.
bool FullyDefined
Reg or a super-register is defined.
VirtRegInfo - Information about a virtual register used by a set of operands.
bool Reads
Reads - One of the operands read the virtual register.
bool Tied
Tied - Uses and defs must use the same register.
bool Writes
Writes - One of the operands writes the virtual register.