103#define DEBUG_TYPE "peephole-opt"
107 cl::desc(
"Aggressive extension optimization"));
111 cl::desc(
"Disable the peephole optimizer"));
118 cl::desc(
"Disable advanced copy optimization"));
122 cl::desc(
"Disable non-allocatable physical register copy optimization"));
128 cl::desc(
"Limit the length of PHI chains to lookup"));
134 cl::desc(
"Maximum length of recurrence chain when evaluating the benefit "
135 "of commuting operands"));
137STATISTIC(NumReuse,
"Number of extension results reused");
139STATISTIC(NumImmFold,
"Number of move immediate folded");
142STATISTIC(NumUncoalescableCopies,
"Number of uncoalescable copies optimized");
143STATISTIC(NumRewrittenCopies,
"Number of copies rewritten");
144STATISTIC(NumNAPhysCopies,
"Number of non-allocatable physical copies removed");
148class ValueTrackerResult;
149class RecurrenceInstr;
155 int CurrentSrcIdx = 0;
158 virtual ~Rewriter() =
default;
190 virtual bool RewriteCurrentSource(
Register NewReg,
unsigned NewSubReg) = 0;
194class CopyRewriter :
public Rewriter {
197 assert(
MI.isCopy() &&
"Expected copy instruction");
199 ~CopyRewriter()
override =
default;
203 if (++CurrentSrcIdx > 1)
207 const MachineOperand &MOSrc = CopyLike.getOperand(CurrentSrcIdx);
210 const MachineOperand &MODef = CopyLike.getOperand(0);
215 bool RewriteCurrentSource(
Register NewReg,
unsigned NewSubReg)
override {
216 MachineOperand &MOSrc = CopyLike.getOperand(CurrentSrcIdx);
225class UncoalescableRewriter :
public Rewriter {
229 UncoalescableRewriter(MachineInstr &
MI) :
Rewriter(
MI) {
230 NumDefs =
MI.getDesc().getNumDefs();
240 if (CurrentSrcIdx == NumDefs)
243 while (CopyLike.getOperand(CurrentSrcIdx).isDead()) {
245 if (CurrentSrcIdx == NumDefs)
251 const MachineOperand &MODef = CopyLike.getOperand(CurrentSrcIdx);
258 bool RewriteCurrentSource(
Register NewReg,
unsigned NewSubReg)
override {
264class InsertSubregRewriter :
public Rewriter {
267 assert(
MI.isInsertSubreg() &&
"Invalid instruction");
284 if (CurrentSrcIdx == 2)
288 const MachineOperand &MOInsertedReg = CopyLike.getOperand(2);
290 const MachineOperand &MODef = CopyLike.getOperand(0);
298 (
unsigned)CopyLike.getOperand(3).getImm());
302 bool RewriteCurrentSource(
Register NewReg,
unsigned NewSubReg)
override {
303 if (CurrentSrcIdx != 2)
306 MachineOperand &MO = CopyLike.getOperand(CurrentSrcIdx);
314class ExtractSubregRewriter :
public Rewriter {
315 const TargetInstrInfo &TII;
318 ExtractSubregRewriter(MachineInstr &
MI,
const TargetInstrInfo &TII)
320 assert(
MI.isExtractSubreg() &&
"Invalid instruction");
331 if (CurrentSrcIdx == 1)
335 const MachineOperand &MOExtractedReg = CopyLike.getOperand(1);
344 const MachineOperand &MODef = CopyLike.getOperand(0);
349 bool RewriteCurrentSource(
Register NewReg,
unsigned NewSubReg)
override {
351 if (CurrentSrcIdx != 1)
354 CopyLike.getOperand(CurrentSrcIdx).setReg(NewReg);
365 CopyLike.removeOperand(2);
367 CopyLike.setDesc(TII.get(TargetOpcode::COPY));
370 CopyLike.getOperand(CurrentSrcIdx + 1).setImm(NewSubReg);
376class RegSequenceRewriter :
public Rewriter {
379 assert(
MI.isRegSequence() &&
"Invalid instruction");
403 if (
static_cast<unsigned>(CurrentSrcIdx) >= CopyLike.getNumOperands())
406 const MachineOperand &MOInsertedReg = CopyLike.getOperand(CurrentSrcIdx);
407 Src.Reg = MOInsertedReg.
getReg();
412 Dst.SubReg = CopyLike.getOperand(CurrentSrcIdx + 1).getImm();
414 const MachineOperand &MODef = CopyLike.getOperand(0);
416 assert(MODef.
getSubReg() == 0 &&
"cannot have subregister def in SSA");
420 bool RewriteCurrentSource(
Register NewReg,
unsigned NewSubReg)
override {
421 MachineOperand &MO = CopyLike.getOperand(CurrentSrcIdx);
429 const TargetInstrInfo *TII =
nullptr;
430 const TargetRegisterInfo *TRI =
nullptr;
431 MachineRegisterInfo *MRI =
nullptr;
432 MachineDominatorTree *DT =
nullptr;
433 MachineLoopInfo *MLI =
nullptr;
436 PeepholeOptimizer(MachineDominatorTree *DT, MachineLoopInfo *MLI)
437 : DT(DT), MLI(MLI) {}
439 bool run(MachineFunction &MF);
441 using RewriteMapTy = SmallDenseMap<RegSubRegPair, ValueTrackerResult>;
444 using RecurrenceCycle = SmallVector<RecurrenceInstr, 4>;
447 bool optimizeCmpInstr(MachineInstr &
MI, MachineFunction &MF,
448 SmallPtrSet<MachineInstr *, 16> &LocalMIs);
449 bool optimizeExtInstr(MachineInstr &
MI, MachineBasicBlock &
MBB,
450 SmallPtrSetImpl<MachineInstr *> &LocalMIs);
451 bool optimizeSelect(MachineInstr &
MI,
452 SmallPtrSetImpl<MachineInstr *> &LocalMIs);
453 bool optimizeCondBranch(MachineInstr &
MI);
455 bool optimizeCoalescableCopyImpl(
Rewriter &&CpyRewriter);
456 bool optimizeCoalescableCopy(MachineInstr &
MI);
457 bool optimizeUncoalescableCopy(MachineInstr &
MI,
458 SmallPtrSetImpl<MachineInstr *> &LocalMIs);
459 bool optimizeRecurrence(MachineInstr &
PHI);
460 bool findNextSource(
const TargetRegisterClass *DefRC,
unsigned DefSubReg,
462 bool isMoveImmediate(MachineInstr &
MI, SmallSet<Register, 4> &ImmDefRegs,
463 DenseMap<Register, MachineInstr *> &ImmDefMIs);
464 bool foldImmediate(MachineInstr &
MI, SmallSet<Register, 4> &ImmDefRegs,
465 DenseMap<Register, MachineInstr *> &ImmDefMIs,
473 const SmallSet<Register, 2> &TargetReg,
474 RecurrenceCycle &RC);
481 bool foldRedundantCopy(MachineInstr &
MI);
492 foldRedundantNAPhysCopy(MachineInstr &
MI,
493 DenseMap<Register, MachineInstr *> &NAPhysToVirtMIs);
495 bool isLoadFoldable(MachineInstr &
MI,
496 SmallSet<Register, 16> &FoldAsLoadDefCandidates);
501 MachineInstr *foldLoadInto(MachineFunction &MF, MachineInstr &
MI,
503 SmallPtrSet<MachineInstr *, 16> &LocalMIs);
507 static bool isCoalescableCopy(
const MachineInstr &
MI) {
510 return MI.isCopy() ||
512 MI.isExtractSubreg()));
517 static bool isUncoalescableCopy(
const MachineInstr &
MI) {
519 MI.isInsertSubregLike() ||
520 MI.isExtractSubregLike()));
523 MachineInstr &rewriteSource(MachineInstr &CopyLike,
RegSubRegPair Def,
524 RewriteMapTy &RewriteMap);
528 DenseMap<RegSubRegPair, MachineInstr *> CopySrcMIs;
531 void MF_HandleInsertion(MachineInstr &
MI)
override {}
538 unsigned SrcSubReg =
MI.getOperand(1).getSubReg();
539 if (!SrcReg.
isVirtual() && !MRI->isConstantPhysReg(SrcReg))
548 void deleteChangedCopy(MachineInstr &
MI) {
550 if (!getCopySrc(
MI, SrcPair))
553 auto It = CopySrcMIs.find(SrcPair);
554 if (It != CopySrcMIs.end() && It->second == &
MI)
555 CopySrcMIs.erase(It);
558 void MF_HandleRemoval(MachineInstr &
MI)
override { deleteChangedCopy(
MI); }
560 void MF_HandleChangeDesc(MachineInstr &
MI,
const MCInstrDesc &TID)
override {
561 deleteChangedCopy(
MI);
569 PeepholeOptimizerLegacy() : MachineFunctionPass(ID) {}
571 bool runOnMachineFunction(MachineFunction &MF)
override;
573 void getAnalysisUsage(AnalysisUsage &AU)
const override {
584 MachineFunctionProperties getRequiredProperties()
const override {
585 return MachineFunctionProperties().setIsSSA();
595class RecurrenceInstr {
597 using IndexPair = std::pair<unsigned, unsigned>;
599 RecurrenceInstr(MachineInstr *MI) : MI(MI) {}
600 RecurrenceInstr(MachineInstr *MI,
unsigned Idx1,
unsigned Idx2)
601 : MI(MI), CommutePair(std::make_pair(Idx1, Idx2)) {}
603 MachineInstr *getMI()
const {
return MI; }
604 std::optional<IndexPair> getCommutePair()
const {
return CommutePair; }
608 std::optional<IndexPair> CommutePair;
614class ValueTrackerResult {
620 const MachineInstr *Inst =
nullptr;
623 ValueTrackerResult() =
default;
625 ValueTrackerResult(
Register Reg,
unsigned SubReg) { addSource(
Reg, SubReg); }
627 bool isValid()
const {
return getNumSources() > 0; }
629 void setInst(
const MachineInstr *
I) { Inst =
I; }
630 const MachineInstr *getInst()
const {
return Inst; }
637 void addSource(
Register SrcReg,
unsigned SrcSubReg) {
641 void setSource(
int Idx,
Register SrcReg,
unsigned SrcSubReg) {
642 assert(Idx < getNumSources() &&
"Reg pair source out of index");
646 int getNumSources()
const {
return RegSrcs.size(); }
651 assert(Idx < getNumSources() &&
"Reg source out of index");
652 return RegSrcs[Idx].Reg;
655 unsigned getSrcSubReg(
int Idx)
const {
656 assert(Idx < getNumSources() &&
"SubReg source out of index");
657 return RegSrcs[Idx].SubReg;
661 if (
Other.getInst() != getInst())
664 if (
Other.getNumSources() != getNumSources())
667 for (
int i = 0, e =
Other.getNumSources(); i != e; ++i)
668 if (
Other.getSrcReg(i) != getSrcReg(i) ||
669 Other.getSrcSubReg(i) != getSrcSubReg(i))
694 const MachineInstr *Def =
nullptr;
706 const MachineRegisterInfo &MRI;
709 const TargetInstrInfo *TII;
712 ValueTrackerResult getNextSourceImpl();
715 ValueTrackerResult getNextSourceFromCopy();
718 ValueTrackerResult getNextSourceFromBitcast();
721 ValueTrackerResult getNextSourceFromRegSequence();
724 ValueTrackerResult getNextSourceFromInsertSubreg();
727 ValueTrackerResult getNextSourceFromExtractSubreg();
730 ValueTrackerResult getNextSourceFromSubregToReg();
733 ValueTrackerResult getNextSourceFromPHI();
745 ValueTracker(
Register Reg,
unsigned DefSubReg,
const MachineRegisterInfo &MRI,
746 const TargetInstrInfo *TII =
nullptr)
747 : DefSubReg(DefSubReg), Reg(Reg), MRI(MRI), TII(TII) {
748 if (!Reg.isPhysical()) {
749 Def = MRI.getVRegDef(Reg);
750 DefIdx = MRI.def_begin(Reg).getOperandNo();
759 ValueTrackerResult getNextSource();
764char PeepholeOptimizerLegacy::ID = 0;
769 "Peephole Optimizations",
false,
false)
783bool PeepholeOptimizer::optimizeExtInstr(
788 if (!
TII->isCoalescableExtInstr(
MI, SrcReg, DstReg, SubIdx))
801 DstRC =
TRI->getSubClassWithSubReg(DstRC, SubIdx);
811 TRI->getSubClassWithSubReg(MRI->
getRegClass(SrcReg), SubIdx) !=
nullptr;
817 ReachedBBs.insert(UI.getParent());
825 bool ExtendLife =
true;
827 MachineInstr *UseMI = UseMO.getParent();
831 if (UseMI->isPHI()) {
837 if (UseSrcSubIdx && UseMO.getSubReg() != SubIdx)
857 if (
UseMI->getOpcode() == TargetOpcode::SUBREG_TO_REG)
861 if (UseMBB == &
MBB) {
863 if (!LocalMIs.count(
UseMI))
864 Uses.push_back(&UseMO);
865 }
else if (ReachedBBs.count(UseMBB)) {
868 Uses.push_back(&UseMO);
872 ExtendedUses.push_back(&UseMO);
881 if (ExtendLife && !ExtendedUses.empty())
883 Uses.append(ExtendedUses.begin(), ExtendedUses.end());
888 SmallPtrSet<MachineBasicBlock *, 4> PHIBBs;
893 for (MachineInstr &UI : MRI->use_nodbg_instructions(DstReg))
895 PHIBBs.insert(UI.getParent());
897 const TargetRegisterClass *RC = MRI->getRegClass(SrcReg);
898 for (MachineOperand *UseMO : Uses) {
899 MachineInstr *UseMI = UseMO->getParent();
900 MachineBasicBlock *UseMBB = UseMI->getParent();
901 if (PHIBBs.count(UseMBB))
906 MRI->clearKillFlags(DstReg);
907 MRI->constrainRegClass(DstReg, DstRC);
925 RC = MRI->getRegClass(UseMI->getOperand(0).getReg());
927 Register NewVR = MRI->createVirtualRegister(RC);
928 BuildMI(*UseMBB, UseMI, UseMI->getDebugLoc(),
929 TII->get(TargetOpcode::COPY), NewVR)
930 .addReg(DstReg, {}, SubIdx);
934 UseMO->setReg(NewVR);
947bool PeepholeOptimizer::optimizeCmpInstr(
953 int64_t CmpMask, CmpValue;
960 if (!
TII->optimizeCompareInstr(
MI, SrcReg, SrcReg2, CmpMask, CmpValue, MRI))
970 MachineInstr *LoadMI = MRI->
getVRegDef(SrcReg);
977 [](
const MachineInstr &
I) { return I.isLoadFoldBarrier(); }))
978 foldLoadInto(MF, *FlagProducer, SrcReg, LocalMIs);
985bool PeepholeOptimizer::optimizeSelect(
986 MachineInstr &
MI, SmallPtrSetImpl<MachineInstr *> &LocalMIs) {
987 assert(
MI.isSelect() &&
"Should only be called when MI->isSelect() is true");
988 if (!
TII->optimizeSelect(
MI, LocalMIs))
991 MI.eraseFromParent();
997bool PeepholeOptimizer::optimizeCondBranch(MachineInstr &
MI) {
998 return TII->optimizeCondBranch(
MI);
1014bool PeepholeOptimizer::findNextSource(
const TargetRegisterClass *DefRC,
1017 RewriteMapTy &RewriteMap) {
1026 unsigned PHICount = 0;
1033 ValueTracker ValTracker(CurSrcPair.
Reg, CurSrcPair.
SubReg, *MRI,
TII);
1038 ValueTrackerResult Res = ValTracker.getNextSource();
1044 auto [InsertPt, WasInserted] = RewriteMap.try_emplace(CurSrcPair, Res);
1047 const ValueTrackerResult &CurSrcRes = InsertPt->second;
1049 assert(CurSrcRes == Res &&
"ValueTrackerResult found must match");
1052 if (CurSrcRes.getNumSources() > 1) {
1054 <<
"findNextSource: found PHI cycle, aborting...\n");
1062 unsigned NumSrcs = Res.getNumSources();
1070 for (
unsigned i = 0; i < NumSrcs; ++i)
1075 CurSrcPair = Res.getSrc(0);
1084 const TargetRegisterClass *SrcRC = MRI->
getRegClass(CurSrcPair.
Reg);
1085 if (!
TRI->shouldRewriteCopySrc(DefRC, DefSubReg, SrcRC,
1091 if (PHICount > 0 && CurSrcPair.
SubReg != 0)
1097 }
while (!SrcToLook.
empty());
1100 return CurSrcPair.
Reg !=
Reg;
1112 assert(!SrcRegs.
empty() &&
"No sources to create a PHI instruction?");
1117 assert(SrcRegs[0].SubReg == 0 &&
"should not have subreg operand");
1121 TII.get(TargetOpcode::PHI), NewVR);
1123 unsigned MBBOpIdx = 2;
1125 MIB.
addReg(RegPair.Reg, {}, RegPair.SubReg);
1146 const PeepholeOptimizer::RewriteMapTy &RewriteMap,
1147 bool HandleMultipleSources =
true) {
1150 ValueTrackerResult Res = RewriteMap.
lookup(LookupSrc);
1156 unsigned NumSrcs = Res.getNumSources();
1158 LookupSrc.
Reg = Res.getSrcReg(0);
1159 LookupSrc.
SubReg = Res.getSrcSubReg(0);
1164 if (!HandleMultipleSources)
1170 for (
unsigned i = 0; i < NumSrcs; ++i) {
1171 RegSubRegPair PHISrc(Res.getSrcReg(i), Res.getSrcSubReg(i));
1189bool PeepholeOptimizer::optimizeCoalescableCopyImpl(
Rewriter &&CpyRewriter) {
1195 while (CpyRewriter.getNextRewritableSource(TrackPair, Dst)) {
1196 if (Dst.Reg.isPhysical()) {
1204 const TargetRegisterClass *DefRC = MRI->
getRegClass(Dst.Reg);
1207 RewriteMapTy RewriteMap;
1210 if (!findNextSource(DefRC, Dst.SubReg, TrackPair, RewriteMap))
1218 "should not rewrite source to original value");
1227 const TargetRegisterClass *WithSubRC =
1228 TRI->getSubClassWithSubReg(RC, NewSrc.
SubReg);
1235 if (CpyRewriter.RewriteCurrentSource(NewSrc.
Reg, NewSrc.
SubReg)) {
1247 NumRewrittenCopies +=
Changed;
1262bool PeepholeOptimizer::optimizeCoalescableCopy(MachineInstr &
MI) {
1263 assert(isCoalescableCopy(
MI) &&
"Invalid argument");
1264 assert(
MI.getDesc().getNumDefs() == 1 &&
1265 "Coalescer can understand multiple defs?!");
1266 const MachineOperand &MODef =
MI.getOperand(0);
1271 switch (
MI.getOpcode()) {
1272 case TargetOpcode::COPY:
1273 return optimizeCoalescableCopyImpl(CopyRewriter(
MI));
1274 case TargetOpcode::INSERT_SUBREG:
1275 return optimizeCoalescableCopyImpl(InsertSubregRewriter(
MI));
1276 case TargetOpcode::EXTRACT_SUBREG:
1277 return optimizeCoalescableCopyImpl(ExtractSubregRewriter(
MI, *
TII));
1278 case TargetOpcode::REG_SEQUENCE:
1279 return optimizeCoalescableCopyImpl(RegSequenceRewriter(
MI));
1282 if (
MI.isBitcast() ||
MI.isRegSequenceLike() ||
MI.isInsertSubregLike() ||
1283 MI.isExtractSubregLike())
1284 return optimizeCoalescableCopyImpl(UncoalescableRewriter(
MI));
1294MachineInstr &PeepholeOptimizer::rewriteSource(MachineInstr &CopyLike,
1296 RewriteMapTy &RewriteMap) {
1297 assert(!
Def.Reg.isPhysical() &&
"We do not rewrite physical registers");
1307 const TargetRegisterClass *NewSrcRC = MRI->
getRegClass(NewSrc.
Reg);
1308 const TargetRegisterClass *WithSubRC =
1309 TRI->getSubClassWithSubReg(NewSrcRC, NewSrc.
SubReg);
1318 MachineInstr *NewCopy =
1320 TII->get(TargetOpcode::COPY), NewVReg)
1352bool PeepholeOptimizer::optimizeUncoalescableCopy(
1353 MachineInstr &
MI, SmallPtrSetImpl<MachineInstr *> &LocalMIs) {
1354 assert(isUncoalescableCopy(
MI) &&
"Invalid argument");
1355 UncoalescableRewriter CpyRewriter(
MI);
1360 RewriteMapTy RewriteMap;
1364 while (CpyRewriter.getNextRewritableSource(Src, Def)) {
1367 if (
Def.Reg.isPhysical())
1377 if (!findNextSource(DefRC,
Def.SubReg, Def, RewriteMap))
1386 MachineInstr &NewCopy = rewriteSource(
MI, Def, RewriteMap);
1387 LocalMIs.
insert(&NewCopy);
1392 MI.eraseFromParent();
1393 ++NumUncoalescableCopies;
1400bool PeepholeOptimizer::isLoadFoldable(
1401 MachineInstr &
MI, SmallSet<Register, 16> &FoldAsLoadDefCandidates) {
1402 if (!
MI.canFoldAsLoad() || !
MI.mayLoad())
1404 const MCInstrDesc &MCID =
MI.getDesc();
1421PeepholeOptimizer::foldLoadInto(MachineFunction &MF, MachineInstr &
MI,
1423 SmallPtrSet<MachineInstr *, 16> &LocalMIs) {
1425 MachineInstr *
DefMI =
nullptr;
1426 MachineInstr *CopyMI =
nullptr;
1427 MachineInstr *FoldMI =
TII->optimizeLoadInstr(
MI, MRI,
Reg,
DefMI, CopyMI);
1436 if (
MI.shouldUpdateAdditionalCallInfo())
1438 MI.eraseFromParent();
1445bool PeepholeOptimizer::isMoveImmediate(
1446 MachineInstr &
MI, SmallSet<Register, 4> &ImmDefRegs,
1447 DenseMap<Register, MachineInstr *> &ImmDefMIs) {
1448 const MCInstrDesc &MCID =
MI.getDesc();
1449 if (MCID.
getNumDefs() != 1 || !
MI.getOperand(0).isReg())
1456 if (!
MI.isMoveImmediate() && !
TII->getConstValDefinedInReg(
MI,
Reg, ImmVal))
1467bool PeepholeOptimizer::foldImmediate(
1468 MachineInstr &
MI, SmallSet<Register, 4> &ImmDefRegs,
1469 DenseMap<Register, MachineInstr *> &ImmDefMIs,
bool &
Deleted) {
1471 for (
unsigned i = 0, e =
MI.getDesc().getNumOperands(); i != e; ++i) {
1472 MachineOperand &MO =
MI.getOperand(i);
1481 assert(
II != ImmDefMIs.
end() &&
"couldn't find immediate definition");
1482 if (
TII->foldImmediate(
MI, *
II->second,
Reg, MRI)) {
1494 MI.eraseFromParent();
1518bool PeepholeOptimizer::foldRedundantCopy(MachineInstr &
MI) {
1519 assert(
MI.isCopy() &&
"expected a COPY machine instruction");
1522 if (!getCopySrc(
MI, SrcPair))
1529 if (CopySrcMIs.
insert(std::make_pair(SrcPair, &
MI)).second) {
1534 MachineInstr *PrevCopy = CopySrcMIs.
find(SrcPair)->second;
1537 "Unexpected mismatching subreg!");
1555bool PeepholeOptimizer::isNAPhysCopy(
Register Reg) {
1559bool PeepholeOptimizer::foldRedundantNAPhysCopy(
1560 MachineInstr &
MI, DenseMap<Register, MachineInstr *> &NAPhysToVirtMIs) {
1561 assert(
MI.isCopy() &&
"expected a COPY machine instruction");
1568 if (isNAPhysCopy(SrcReg) && DstReg.
isVirtual()) {
1572 NAPhysToVirtMIs.
insert({SrcReg, &
MI});
1576 if (!(SrcReg.
isVirtual() && isNAPhysCopy(DstReg)))
1580 auto PrevCopy = NAPhysToVirtMIs.
find(DstReg);
1581 if (PrevCopy == NAPhysToVirtMIs.
end()) {
1584 LLVM_DEBUG(
dbgs() <<
"NAPhysCopy: intervening clobber forbids erasing "
1590 if (PrevDstReg == SrcReg) {
1603 NAPhysToVirtMIs.
erase(PrevCopy);
1612bool PeepholeOptimizer::findTargetRecurrence(
1613 Register Reg,
const SmallSet<Register, 2> &TargetRegs,
1614 RecurrenceCycle &RC) {
1632 unsigned Idx =
MI.findRegisterUseOperandIdx(
Reg,
nullptr);
1636 if (
MI.getDesc().getNumDefs() != 1)
1639 MachineOperand &DefOp =
MI.getOperand(0);
1646 unsigned TiedUseIdx;
1647 if (!
MI.isRegTiedToUseOperand(0, &TiedUseIdx))
1650 if (Idx == TiedUseIdx) {
1651 RC.push_back(RecurrenceInstr(&
MI));
1652 return findTargetRecurrence(DefOp.
getReg(), TargetRegs, RC);
1656 if (
TII->findCommutedOpIndices(
MI, Idx, CommIdx) && CommIdx == TiedUseIdx) {
1657 RC.push_back(RecurrenceInstr(&
MI, Idx, CommIdx));
1658 return findTargetRecurrence(DefOp.
getReg(), TargetRegs, RC);
1683bool PeepholeOptimizer::optimizeRecurrence(MachineInstr &
PHI) {
1684 SmallSet<Register, 2> TargetRegs;
1685 for (
unsigned Idx = 1; Idx <
PHI.getNumOperands(); Idx += 2) {
1686 MachineOperand &MO =
PHI.getOperand(Idx);
1693 if (findTargetRecurrence(
PHI.getOperand(0).getReg(), TargetRegs, RC)) {
1697 for (
auto &RI : RC) {
1699 auto CP = RI.getCommutePair();
1702 TII->commuteInstruction(*(RI.getMI()),
false, (*CP).first,
1719 PeepholeOptimizer Impl(DT, MLI);
1731bool PeepholeOptimizerLegacy::runOnMachineFunction(
MachineFunction &MF) {
1735 ? &getAnalysis<MachineDominatorTreeWrapperPass>().getDomTree()
1737 auto *MLI = &getAnalysis<MachineLoopInfoWrapperPass>().getLI();
1738 PeepholeOptimizer Impl(DT, MLI);
1739 return Impl.run(MF);
1744 LLVM_DEBUG(
dbgs() <<
"********** PEEPHOLE OPTIMIZER **********\n");
1758 bool SeenMoveImm =
false;
1791 if (
MI->isDebugInstr())
1794 if (
MI->isPosition())
1797 if (IsLoopHeader &&
MI->isPHI()) {
1798 if (optimizeRecurrence(*
MI)) {
1804 if (!
MI->isCopy()) {
1805 for (
const MachineOperand &MO :
MI->operands()) {
1809 if (MO.
isDef() && isNAPhysCopy(
Reg)) {
1811 if (Def != NAPhysToVirtMIs.
end()) {
1815 <<
"NAPhysCopy: invalidating because of " << *
MI);
1816 NAPhysToVirtMIs.
erase(Def);
1821 NAPhysToVirtMIs.
remove_if([&](
const auto &RegMI) {
1825 <<
"NAPhysCopy: invalidating because of " << *
MI);
1832 if (
MI->isImplicitDef() ||
MI->isKill())
1835 if (
MI->isInlineAsm() ||
MI->hasUnmodeledSideEffects()) {
1842 NAPhysToVirtMIs.
clear();
1845 if (
MI->isCompare() && optimizeCmpInstr(*
MI, MF, LocalMIs)) {
1851 if ((isUncoalescableCopy(*
MI) &&
1852 optimizeUncoalescableCopy(*
MI, LocalMIs)) ||
1853 (
MI->isSelect() && optimizeSelect(*
MI, LocalMIs))) {
1860 if (
MI->isConditionalBranch() && optimizeCondBranch(*
MI)) {
1865 if (isCoalescableCopy(*
MI) && optimizeCoalescableCopy(*
MI)) {
1871 if (
MI->isCopy() && (foldRedundantCopy(*
MI) ||
1872 foldRedundantNAPhysCopy(*
MI, NAPhysToVirtMIs))) {
1875 MI->eraseFromParent();
1880 if (isMoveImmediate(*
MI, ImmDefRegs, ImmDefMIs)) {
1902 if (!isLoadFoldable(*
MI, FoldAsLoadDefCandidates) &&
1903 !FoldAsLoadDefCandidates.
empty()) {
1910 const MCInstrDesc &MIDesc =
MI->getDesc();
1911 for (
unsigned i = MIDesc.
getNumDefs(); i !=
MI->getNumOperands(); ++i) {
1912 const MachineOperand &MOp =
MI->getOperand(i);
1916 if (FoldAsLoadDefCandidates.
count(FoldAsLoadDefReg)) {
1919 Register FoldedReg = FoldAsLoadDefReg;
1920 if (MachineInstr *FoldMI =
1921 foldLoadInto(MF, *
MI, FoldAsLoadDefReg, LocalMIs)) {
1922 FoldAsLoadDefCandidates.
erase(FoldedReg);
1934 if (
MI->isLoadFoldBarrier()) {
1936 FoldAsLoadDefCandidates.
clear();
1941 MF.resetDelegate(
this);
1945ValueTrackerResult ValueTracker::getNextSourceFromCopy() {
1946 assert(
Def->isCopy() &&
"Invalid definition");
1951 assert(
Def->getNumOperands() -
Def->getNumImplicitOperands() == 2 &&
1952 "Invalid number of operands");
1953 assert(!
Def->hasImplicitDef() &&
"Only implicit uses are allowed");
1954 assert(!
Def->getOperand(DefIdx).getSubReg() &&
"no subregister defs in SSA");
1957 const MachineOperand &Src =
Def->getOperand(1);
1959 return ValueTrackerResult();
1962 unsigned SubReg = Src.getSubReg();
1965 SubReg =
TRI->composeSubRegIndices(SubReg, DefSubReg);
1969 const TargetRegisterClass *RegRC = MRI.
getRegClass(SrcReg);
1970 if (!
TRI->isSubRegValidForRegClass(RegRC, SubReg))
1971 return ValueTrackerResult();
1973 if (!
TRI->getSubReg(SrcReg, SubReg))
1974 return ValueTrackerResult();
1978 return ValueTrackerResult(SrcReg, SubReg);
1981ValueTrackerResult ValueTracker::getNextSourceFromBitcast() {
1982 assert(
Def->isBitcast() &&
"Invalid definition");
1985 if (
Def->mayRaiseFPException() ||
Def->hasUnmodeledSideEffects())
1986 return ValueTrackerResult();
1989 if (
Def->getDesc().getNumDefs() != 1)
1990 return ValueTrackerResult();
1992 assert(!
Def->getOperand(DefIdx).getSubReg() &&
"no subregister defs in SSA");
1994 unsigned SrcIdx =
Def->getNumOperands();
1995 for (
unsigned OpIdx = DefIdx + 1, EndOpIdx = SrcIdx;
OpIdx != EndOpIdx;
1997 const MachineOperand &MO =
Def->getOperand(
OpIdx);
2003 assert(!MO.
isDef() &&
"We should have skipped all the definitions by now");
2004 if (SrcIdx != EndOpIdx)
2006 return ValueTrackerResult();
2012 if (SrcIdx >=
Def->getNumOperands())
2013 return ValueTrackerResult();
2015 const MachineOperand &DefOp =
Def->getOperand(DefIdx);
2020 if (
UseMI.isSubregToReg())
2021 return ValueTrackerResult();
2024 const MachineOperand &Src =
Def->getOperand(SrcIdx);
2026 return ValueTrackerResult();
2027 return ValueTrackerResult(Src.getReg(), Src.getSubReg());
2030ValueTrackerResult ValueTracker::getNextSourceFromRegSequence() {
2031 assert((
Def->isRegSequence() ||
Def->isRegSequenceLike()) &&
2032 "Invalid definition");
2034 assert(!
Def->getOperand(DefIdx).getSubReg() &&
"illegal subregister def");
2037 if (!
TII->getRegSequenceInputs(*Def, DefIdx, RegSeqInputRegs))
2038 return ValueTrackerResult();
2046 if (RegSeqInput.SubIdx == DefSubReg)
2047 return ValueTrackerResult(RegSeqInput.Reg, RegSeqInput.SubReg);
2055 LaneBitmask DefMask =
TRI->getSubRegIndexLaneMask(DefSubReg);
2056 LaneBitmask ThisOpRegMask =
TRI->getSubRegIndexLaneMask(RegSeqInput.SubIdx);
2062 if ((DefMask & ThisOpRegMask) != DefMask)
2065 unsigned ReverseDefCompose =
2066 TRI->reverseComposeSubRegIndices(RegSeqInput.SubIdx, DefSubReg);
2067 if (!ReverseDefCompose)
2070 unsigned ComposedDefInSrcReg1 =
2071 TRI->composeSubRegIndices(RegSeqInput.SubReg, ReverseDefCompose);
2077 const TargetRegisterClass *SrcRC = MRI.
getRegClass(RegSeqInput.Reg);
2078 if (!
TRI->isSubRegValidForRegClass(SrcRC, ComposedDefInSrcReg1))
2079 return ValueTrackerResult();
2081 return ValueTrackerResult(RegSeqInput.Reg, ComposedDefInSrcReg1);
2087 return ValueTrackerResult();
2090ValueTrackerResult ValueTracker::getNextSourceFromInsertSubreg() {
2091 assert((
Def->isInsertSubreg() ||
Def->isInsertSubregLike()) &&
2092 "Invalid definition");
2093 assert(!
Def->getOperand(DefIdx).getSubReg() &&
"no subreg defs in SSA");
2097 if (!
TII->getInsertSubregInputs(*Def, DefIdx, BaseReg, InsertedReg))
2098 return ValueTrackerResult();
2107 if (InsertedReg.
SubIdx == DefSubReg) {
2108 return ValueTrackerResult(InsertedReg.
Reg, InsertedReg.
SubReg);
2113 const MachineOperand &MODef =
Def->getOperand(DefIdx);
2119 return ValueTrackerResult();
2124 if ((
TRI->getSubRegIndexLaneMask(DefSubReg) &
2125 TRI->getSubRegIndexLaneMask(InsertedReg.
SubIdx))
2127 return ValueTrackerResult();
2130 return ValueTrackerResult(
BaseReg.Reg, DefSubReg);
2133ValueTrackerResult ValueTracker::getNextSourceFromExtractSubreg() {
2134 assert((
Def->isExtractSubreg() ||
Def->isExtractSubregLike()) &&
2135 "Invalid definition");
2142 return ValueTrackerResult();
2145 if (!
TII->getExtractSubregInputs(*Def, DefIdx, ExtractSubregInputReg))
2146 return ValueTrackerResult();
2150 if (ExtractSubregInputReg.
SubReg)
2151 return ValueTrackerResult();
2153 return ValueTrackerResult(ExtractSubregInputReg.
Reg,
2154 ExtractSubregInputReg.
SubIdx);
2157ValueTrackerResult ValueTracker::getNextSourceFromSubregToReg() {
2158 assert(
Def->isSubregToReg() &&
"Invalid definition");
2166 if (DefSubReg !=
Def->getOperand(2).getImm())
2167 return ValueTrackerResult();
2170 if (
Def->getOperand(1).getSubReg())
2171 return ValueTrackerResult();
2173 return ValueTrackerResult(
Def->getOperand(1).getReg(),
2174 Def->getOperand(2).getImm());
2178ValueTrackerResult ValueTracker::getNextSourceFromPHI() {
2179 assert(
Def->isPHI() &&
"Invalid definition");
2180 ValueTrackerResult Res;
2183 for (
unsigned i = 1, e =
Def->getNumOperands(); i < e; i += 2) {
2184 const MachineOperand &MO =
Def->getOperand(i);
2189 return ValueTrackerResult();
2196ValueTrackerResult ValueTracker::getNextSourceImpl() {
2197 assert(Def &&
"This method needs a valid definition");
2199 assert(((
Def->getOperand(DefIdx).isDef() &&
2200 (DefIdx < Def->
getDesc().getNumDefs() ||
2201 Def->getDesc().isVariadic())) ||
2202 Def->getOperand(DefIdx).isImplicit()) &&
2205 return getNextSourceFromCopy();
2206 if (
Def->isBitcast())
2207 return getNextSourceFromBitcast();
2211 return ValueTrackerResult();
2212 if (
Def->isRegSequence() ||
Def->isRegSequenceLike())
2213 return getNextSourceFromRegSequence();
2214 if (
Def->isInsertSubreg() ||
Def->isInsertSubregLike())
2215 return getNextSourceFromInsertSubreg();
2216 if (
Def->isExtractSubreg() ||
Def->isExtractSubregLike())
2217 return getNextSourceFromExtractSubreg();
2218 if (
Def->isSubregToReg())
2219 return getNextSourceFromSubregToReg();
2221 return getNextSourceFromPHI();
2222 return ValueTrackerResult();
2225ValueTrackerResult ValueTracker::getNextSource() {
2229 return ValueTrackerResult();
2231 ValueTrackerResult Res = getNextSourceImpl();
2232 if (Res.isValid()) {
2236 bool OneRegSrc = Res.getNumSources() == 1;
2238 Reg = Res.getSrcReg(0);
2250 DefSubReg = Res.getSrcSubReg(0);
for(const MachineOperand &MO :llvm::drop_begin(OldMI.operands(), Desc.getNumOperands()))
MachineInstrBuilder & UseMI
MachineInstrBuilder MachineInstrBuilder & DefMI
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
This file defines the DenseMap class.
const HexagonInstrInfo * TII
A common definition of LaneBitmask for use in TableGen and CodeGen.
TargetInstrInfo::RegSubRegPair RegSubRegPair
Register const TargetRegisterInfo * TRI
Promote Memory to Register
MachineInstr unsigned OpIdx
uint64_t IntrinsicInst * II
#define INITIALIZE_PASS_DEPENDENCY(depName)
#define INITIALIZE_PASS_END(passName, arg, name, cfg, analysis)
#define INITIALIZE_PASS_BEGIN(passName, arg, name, cfg, analysis)
static cl::opt< unsigned > RewritePHILimit("rewrite-phi-limit", cl::Hidden, cl::init(10), cl::desc("Limit the length of PHI chains to lookup"))
static cl::opt< bool > DisablePeephole("disable-peephole", cl::Hidden, cl::init(false), cl::desc("Disable the peephole optimizer"))
static cl::opt< unsigned > MaxRecurrenceChain("recurrence-chain-limit", cl::Hidden, cl::init(3), cl::desc("Maximum length of recurrence chain when evaluating the benefit " "of commuting operands"))
static cl::opt< bool > DisableNAPhysCopyOpt("disable-non-allocatable-phys-copy-opt", cl::Hidden, cl::init(false), cl::desc("Disable non-allocatable physical register copy optimization"))
static bool isVirtualRegisterOperand(MachineOperand &MO)
\bried Returns true if MO is a virtual register operand.
static MachineInstr & insertPHI(MachineRegisterInfo &MRI, const TargetInstrInfo &TII, const SmallVectorImpl< RegSubRegPair > &SrcRegs, MachineInstr &OrigPHI)
Insert a PHI instruction with incoming edges SrcRegs that are guaranteed to have the same register cl...
static cl::opt< bool > Aggressive("aggressive-ext-opt", cl::Hidden, cl::desc("Aggressive extension optimization"))
static cl::opt< bool > DisableAdvCopyOpt("disable-adv-copy-opt", cl::Hidden, cl::init(false), cl::desc("Disable advanced copy optimization"))
Specifiy whether or not the value tracking looks through complex instructions.
TargetInstrInfo::RegSubRegPairAndIdx RegSubRegPairAndIdx
static RegSubRegPair getNewSource(MachineRegisterInfo *MRI, const TargetInstrInfo *TII, RegSubRegPair Def, const PeepholeOptimizer::RewriteMapTy &RewriteMap, bool HandleMultipleSources=true)
Given a Def.Reg and Def.SubReg pair, use RewriteMap to find the new source to use for rewrite.
Remove Loads Into Fake Uses
static bool isValid(const char C)
Returns true if C is a valid mangled character: <0-9a-zA-Z_>.
This file defines the SmallPtrSet class.
This file defines the SmallSet 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)
Virtual Register Rewriter
PassT::Result & getResult(IRUnitT &IR, ExtraArgTs... ExtraArgs)
Get the result of an analysis pass for a given IR unit.
AnalysisUsage & addRequired()
AnalysisUsage & addPreserved()
Add the specified Pass class to the set of analyses preserved by this pass.
LLVM_ABI void setPreservesCFG()
This function should be called by the pass, iff they do not:
Represents analyses that only rely on functions' control flow.
ValueT lookup(const_arg_type_t< KeyT > Val) const
Return the entry for the specified key, or a default constructed value if no such entry exists.
iterator find(const_arg_type_t< KeyT > Val)
bool erase(const KeyT &Val)
bool remove_if(Predicate Pred)
Remove entries that match the given predicate.
std::pair< iterator, bool > insert(const std::pair< KeyT, ValueT > &KV)
bool analyzeCompare(const MachineInstr &MI, Register &SrcReg, Register &SrcReg2, int64_t &Mask, int64_t &Value) const override
For a comparison instruction, return the source registers in SrcReg and SrcReg2 if having two registe...
bool isLoopHeader(const BlockT *BB) const
unsigned getNumDefs() const
Return the number of MachineOperands that are register definitions.
An RAII based helper class to modify MachineFunctionProperties when running pass.
MachineInstrBundleIterator< MachineInstr > iterator
Analysis pass which computes a MachineDominatorTree.
Analysis pass which computes a MachineDominatorTree.
bool dominates(const MachineInstr *A, const MachineInstr *B) const
MachineFunctionPass - This class adapts the FunctionPass interface to allow convenient creation of pa...
void getAnalysisUsage(AnalysisUsage &AU) const override
getAnalysisUsage - Subclasses that override getAnalysisUsage must call this.
void moveAdditionalCallInfo(const MachineInstr *Old, const MachineInstr *New)
Move the call site info from Old to \New call site info.
const TargetSubtargetInfo & getSubtarget() const
getSubtarget - Return the subtarget for which this machine code is being compiled.
StringRef getName() const
getName - Return the name of the corresponding LLVM function.
MachineRegisterInfo & getRegInfo()
getRegInfo - Return information about the registers currently in use.
Function & getFunction()
Return the LLVM function that this machine code represents.
void setDelegate(Delegate *delegate)
Set the delegate.
const MachineInstrBuilder & addReg(Register RegNo, RegState Flags={}, unsigned SubReg=0) const
Add a new virtual register operand.
const MachineInstrBuilder & addMBB(MachineBasicBlock *MBB, unsigned TargetFlags=0) const
Representation of each machine instruction.
const MachineBasicBlock * getParent() const
bool mayLoad(QueryType Type=AnyInBundle) const
Return true if this instruction could possibly read memory.
const DebugLoc & getDebugLoc() const
Returns the debug location id of this MachineInstr.
const MachineOperand & getOperand(unsigned i) const
LLVM_ABI MachineInstrBundleIterator< MachineInstr > eraseFromParent()
Unlink 'this' from the containing basic block and delete it.
bool canFoldAsLoad(QueryType Type=IgnoreBundle) const
Return true for instructions that can be folded as memory operands in other instructions.
Analysis pass that exposes the MachineLoopInfo for a machine function.
MachineOperand class - Representation of each machine instruction operand.
void setSubReg(unsigned subReg)
unsigned getSubReg() const
bool isReg() const
isReg - Tests if this is a MO_Register operand.
bool isRegMask() const
isRegMask - Tests if this is a MO_RegisterMask operand.
MachineBasicBlock * getMBB() const
LLVM_ABI void setReg(Register Reg)
Change the register this operand corresponds to.
MachineInstr * getParent()
getParent - Return the instruction that this operand belongs to.
void setIsUndef(bool Val=true)
Register getReg() const
getReg - Returns the register number.
static bool clobbersPhysReg(const uint32_t *RegMask, MCRegister PhysReg)
clobbersPhysReg - Returns true if this RegMask clobbers PhysReg.
const uint32_t * getRegMask() const
getRegMask - Returns a bit mask of registers preserved by this RegMask operand.
unsigned getOperandNo() const
getOperandNo - Return the operand # of this MachineOperand in its MachineInstr.
MachineRegisterInfo - Keep track of information for virtual and physical registers,...
LLVM_ABI bool hasOneNonDBGUse(Register RegNo) const
hasOneNonDBGUse - Return true if there is exactly one non-Debug use of the specified register.
use_nodbg_iterator use_nodbg_begin(Register RegNo) const
LLVM_ABI void markUsesInDebugValueAsUndef(Register Reg) const
markUsesInDebugValueAsUndef - Mark every DBG_VALUE referencing the specified register as undefined wh...
const TargetRegisterClass * getRegClass(Register Reg) const
Return the register class of the specified virtual register.
LLVM_ABI void clearKillFlags(Register Reg) const
clearKillFlags - Iterate over all the uses of the given register and clear the kill flag from the Mac...
LLVM_ABI MachineInstr * getVRegDef(Register Reg) const
getVRegDef - Return the machine instr that defines the specified virtual register or null if none is ...
iterator_range< use_nodbg_iterator > use_nodbg_operands(Register Reg) const
def_iterator def_begin(Register RegNo) const
LLVM_ABI Register createVirtualRegister(const TargetRegisterClass *RegClass, StringRef Name="")
createVirtualRegister - Create and return a new virtual register in the function with the specified r...
use_instr_nodbg_iterator use_instr_nodbg_begin(Register RegNo) const
LLVM_ABI bool hasOneNonDBGUser(Register RegNo) const
hasOneNonDBGUse - Return true if there is exactly one non-Debug instruction using the specified regis...
bool isAllocatable(MCRegister PhysReg) const
isAllocatable - Returns true when PhysReg belongs to an allocatable register class and it hasn't been...
defusechain_iterator< false, true, false, true, false > def_iterator
def_iterator/def_begin/def_end - Walk all defs of the specified register.
iterator_range< use_instr_nodbg_iterator > use_nodbg_instructions(Register Reg) const
static def_iterator def_end()
const TargetRegisterInfo * getTargetRegisterInfo() const
LLVM_ABI const TargetRegisterClass * constrainRegClass(Register Reg, const TargetRegisterClass *RC, unsigned MinNumRegs=0)
constrainRegClass - Constrain the register class of the specified virtual register to be a common sub...
LLVM_ABI void replaceRegWith(Register FromReg, Register ToReg)
replaceRegWith - Replace all instances of FromReg with ToReg in the machine function.
LLVM_ABI PreservedAnalyses run(MachineFunction &MF, MachineFunctionAnalysisManager &MFAM)
static PreservedAnalyses all()
Construct a special preserved set that preserves all passes.
Wrapper class representing virtual and physical registers.
constexpr bool isVirtual() const
Return true if the specified register number is in the virtual register namespace.
constexpr bool isPhysical() const
Return true if the specified register number is in the physical register namespace.
A templated base class for SmallPtrSet which provides the typesafe interface that is common across al...
bool erase(PtrType Ptr)
Remove pointer from the set.
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.
SmallSet - This maintains a set of unique values, optimizing for the case when the set is small (less...
size_type count(const T &V) const
count - Return 1 if the element is in the set, 0 otherwise.
std::pair< const_iterator, bool > insert(const T &V)
insert - Insert an element into the set if it isn't already there.
This class consists of common code factored out of the SmallVector class to reduce code duplication b...
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
TargetInstrInfo - Interface to description of machine instruction set.
static const unsigned CommuteAnyOperandIndex
virtual const TargetInstrInfo * getInstrInfo() const
virtual const TargetRegisterInfo * getRegisterInfo() const =0
Return the target's register information.
self_iterator getIterator()
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
MCInstrDesc const & getDesc(MCInstrInfo const &MCII, MCInst const &MCI)
initializer< Ty > init(const Ty &Val)
DXILDebugInfoMap run(Module &M)
NodeAddr< DefNode * > Def
BaseReg
Stack frame base register. Bit 0 of FREInfo.Info.
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.
AnalysisManager< MachineFunction > MachineFunctionAnalysisManager
bool operator==(const AddressRangeValuePair &LHS, const AddressRangeValuePair &RHS)
LLVM_ABI char & PeepholeOptimizerLegacyID
PeepholeOptimizer - This pass performs peephole optimizations - like extension and comparison elimina...
LLVM_ABI PreservedAnalyses getMachineFunctionPassPreservedAnalyses()
Returns the minimum set of Analyses that all machine function passes must preserve.
LLVM_ABI raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
bool none_of(R &&Range, UnaryPredicate P)
Provide wrappers to std::none_of which take ranges instead of having to pass begin/end explicitly.
class LLVM_GSL_OWNER SmallVector
Forward declaration of SmallVector so that calculateSmallVectorDefaultInlinedElements can reference s...
A pair composed of a pair of a register and a sub-register index, and another sub-register index.
A pair composed of a register and a sub-register index.