65#define DEBUG_TYPE "hexagon-pei"
156 cl::desc(
"Set the number of scavenger slots"),
161 cl::desc(
"Specify O2(not Os) spill func threshold"),
166 cl::desc(
"Specify Os spill func threshold"),
175 cl::desc(
"Enable stack frame shrink wrapping"));
180 cl::desc(
"Max count of stack frame shrink-wraps"));
184 cl::desc(
"Enable long calls for save-restore stubs."),
195 cl::init(std::numeric_limits<unsigned>::max()));
221 MachineFunctionProperties::Property::NoVRegs);
225 char HexagonCallFrameInformation::ID = 0;
229bool HexagonCallFrameInformation::runOnMachineFunction(
MachineFunction &MF) {
235 HFI.insertCFIInstructions(MF);
240 "Hexagon call frame information",
false,
false)
243 return new HexagonCallFrameInformation();
251 if (Reg < Hexagon::D0 || Reg > Hexagon::D15)
260 if (!RegNo ||
SubReg < RegNo)
270 static_assert(Hexagon::R1 > 0,
271 "Assume physical registers are encoded as positive integers");
276 for (
unsigned I = 1, E = CSI.
size();
I < E; ++
I) {
291 unsigned Opc =
MI.getOpcode();
293 case Hexagon::PS_alloca:
294 case Hexagon::PS_aligna:
317 for (
MCPhysReg S : HRI.subregs_inclusive(R))
322 if (MO.isRegMask()) {
327 const uint32_t *BM = MO.getRegMask();
331 if (!(BM[R/32] & (1u << (R%32))))
346 unsigned RetOpc =
I->getOpcode();
347 return RetOpc == Hexagon::PS_tailcall_i || RetOpc == Hexagon::PS_tailcall_r;
369 case Hexagon::RESTORE_DEALLOC_RET_JMP_V4:
370 case Hexagon::RESTORE_DEALLOC_RET_JMP_V4_PIC:
371 case Hexagon::RESTORE_DEALLOC_RET_JMP_V4_EXT:
372 case Hexagon::RESTORE_DEALLOC_RET_JMP_V4_EXT_PIC:
373 case Hexagon::RESTORE_DEALLOC_BEFORE_TAILCALL_V4_EXT:
374 case Hexagon::RESTORE_DEALLOC_BEFORE_TAILCALL_V4_EXT_PIC:
375 case Hexagon::RESTORE_DEALLOC_BEFORE_TAILCALL_V4:
376 case Hexagon::RESTORE_DEALLOC_BEFORE_TAILCALL_V4_PIC:
389 return F.hasOptSize() && !
F.hasMinSize();
402 static unsigned ShrinkCounter = 0;
427 RPO[
I->getNumber()] = RPON++;
433 unsigned BN = RPO[
I.getNumber()];
436 if (RPO[Succ->getNumber()] <= BN)
444 for (
const MCPhysReg *
P = HRI.getCalleeSavedRegs(&MF); *
P; ++
P)
453 dbgs() <<
"Blocks needing SF: {";
454 for (
auto &
B : SFBlocks)
459 if (SFBlocks.
empty())
464 for (
unsigned i = 1, n = SFBlocks.
size(); i < n; ++i) {
470 for (
unsigned i = 1, n = SFBlocks.
size(); i < n; ++i) {
476 dbgs() <<
"Computed dom block: ";
481 dbgs() <<
", computed pdom block: ";
497 LLVM_DEBUG(
dbgs() <<
"PDom block does not post-dominate dom block\n");
520 findShrunkPrologEpilog(MF, PrologB, EpilogB);
522 bool PrologueStubs =
false;
523 insertCSRSpillsInBlock(*PrologB, CSI, HRI, PrologueStubs);
524 insertPrologueInBlock(*PrologB, PrologueStubs);
525 updateEntryPaths(MF, *PrologB);
528 insertCSRRestoresInBlock(*EpilogB, CSI, HRI);
529 insertEpilogueInBlock(*EpilogB);
532 if (
B.isReturnBlock())
533 insertCSRRestoresInBlock(
B, CSI, HRI);
536 if (
B.isReturnBlock())
537 insertEpilogueInBlock(
B);
555 BitVector DoneT(MaxBN+1), DoneF(MaxBN+1), Path(MaxBN+1);
556 updateExitPaths(*EpilogB, *EpilogB, DoneT, DoneF, Path);
565 assert(
F.hasFnAttribute(Attribute::NoReturn) &&
566 F.getFunction().hasFnAttribute(Attribute::NoUnwind) &&
567 !
F.getFunction().hasFnAttribute(Attribute::UWTable));
583 assert(!MFI.hasVarSizedObjects() &&
584 !HST.getRegisterInfo()->hasStackRealignment(MF));
585 return F.hasFnAttribute(Attribute::NoReturn) &&
586 F.hasFnAttribute(Attribute::NoUnwind) &&
587 !
F.hasFnAttribute(Attribute::UWTable) && HST.noreturnStackElim() &&
588 MFI.getStackSize() == 0;
592 bool PrologueStubs)
const {
597 auto &HRI = *HST.getRegisterInfo();
608 FrameSize = MaxCFA +
alignTo(FrameSize, MaxAlign);
615 Register SP = HRI.getStackRegister();
622 if (
MI.getOpcode() == Hexagon::PS_alloca)
625 for (
auto *
MI : AdjustRegs) {
626 assert((
MI->getOpcode() == Hexagon::PS_alloca) &&
"Expected alloca");
627 expandAlloca(
MI, HII, SP, MaxCF);
628 MI->eraseFromParent();
633 if (MF.getFunction().isVarArg() &&
637 int RegisterSavedAreaSizePlusPadding = (NumVarArgRegs % 2 == 0)
639 : NumVarArgRegs * 4 + 4;
640 if (RegisterSavedAreaSizePlusPadding > 0) {
643 BuildMI(
MBB, InsertPt, dl, HII.get(Hexagon::A2_addi), SP)
645 .
addImm(-RegisterSavedAreaSizePlusPadding)
651 for (
int i = HMFI.getFirstNamedArgFrameIndex(),
652 e = HMFI.getLastNamedArgFrameIndex(); i >=
e; --i) {
657 unsigned LDOpc, STOpc;
661 if (ObjAlign > ObjSize) {
664 else if (ObjSize <= 2)
666 else if (ObjSize <= 4)
668 else if (ObjSize > 4)
672 switch (OpcodeChecker) {
674 LDOpc = Hexagon::L2_loadrb_io;
675 STOpc = Hexagon::S2_storerb_io;
678 LDOpc = Hexagon::L2_loadrh_io;
679 STOpc = Hexagon::S2_storerh_io;
682 LDOpc = Hexagon::L2_loadri_io;
683 STOpc = Hexagon::S2_storeri_io;
687 LDOpc = Hexagon::L2_loadrd_io;
688 STOpc = Hexagon::S2_storerd_io;
692 Register RegUsed = LDOpc == Hexagon::L2_loadrd_io ? Hexagon::D3
694 int LoadStoreCount = ObjSize / OpcodeChecker;
696 if (ObjSize % OpcodeChecker)
704 NumBytes =
alignTo(NumBytes, ObjAlign);
707 while (Count < LoadStoreCount) {
709 BuildMI(
MBB, InsertPt, dl, HII.get(LDOpc), RegUsed)
711 .
addImm(RegisterSavedAreaSizePlusPadding +
712 ObjAlign.
value() * Count + NumBytes)
728 NumBytes =
alignTo(NumBytes, 8);
733 NumBytes = (NumVarArgRegs % 2 == 0) ? NumBytes : NumBytes + 4;
736 BuildMI(
MBB, InsertPt, dl, HII.get(Hexagon::S2_storeri_io))
746 insertAllocframe(
MBB, InsertPt, NumBytes);
748 BuildMI(
MBB, InsertPt, dl, HII.get(Hexagon::A2_andir), SP)
756 BuildMI(
MBB, InsertPt, dl, HII.get(Hexagon::PS_call_stk))
758 }
else if (NumBytes > 0) {
760 BuildMI(
MBB, InsertPt, dl, HII.get(Hexagon::A2_addi), SP)
770 auto &HRI = *HST.getRegisterInfo();
771 Register SP = HRI.getStackRegister();
783 int RegisterSavedAreaSizePlusPadding = (NumVarArgRegs % 2 == 0) ?
784 (NumVarArgRegs * 4) : (NumVarArgRegs * 4 + 4);
785 NumBytes += RegisterSavedAreaSizePlusPadding;
788 BuildMI(
MBB, InsertPt, dl, HII.get(Hexagon::A2_addi), SP)
796 unsigned RetOpc = RetI ? RetI->
getOpcode() : 0;
799 if (RetOpc == Hexagon::EH_RETURN_JMPR) {
800 BuildMI(
MBB, InsertPt, dl, HII.get(Hexagon::L2_deallocframe))
803 BuildMI(
MBB, InsertPt, dl, HII.get(Hexagon::A2_add), SP)
811 if (RetOpc == Hexagon::RESTORE_DEALLOC_RET_JMP_V4 ||
812 RetOpc == Hexagon::RESTORE_DEALLOC_RET_JMP_V4_PIC ||
813 RetOpc == Hexagon::RESTORE_DEALLOC_RET_JMP_V4_EXT ||
814 RetOpc == Hexagon::RESTORE_DEALLOC_RET_JMP_V4_EXT_PIC) {
830 bool NeedsDeallocframe =
true;
833 unsigned COpc = PrevIt->getOpcode();
834 if (COpc == Hexagon::RESTORE_DEALLOC_BEFORE_TAILCALL_V4 ||
835 COpc == Hexagon::RESTORE_DEALLOC_BEFORE_TAILCALL_V4_PIC ||
836 COpc == Hexagon::RESTORE_DEALLOC_BEFORE_TAILCALL_V4_EXT ||
837 COpc == Hexagon::RESTORE_DEALLOC_BEFORE_TAILCALL_V4_EXT_PIC ||
838 COpc == Hexagon::PS_call_nr || COpc == Hexagon::PS_callr_nr)
839 NeedsDeallocframe =
false;
844 if (!NeedsDeallocframe)
850 BuildMI(
MBB, InsertPt, dl, HII.get(Hexagon::L2_deallocframe))
855 unsigned NewOpc = Hexagon::L4_return;
866 int RegisterSavedAreaSizePlusPadding = (NumVarArgRegs % 2 == 0) ?
867 (NumVarArgRegs * 4) : (NumVarArgRegs * 4 + 4);
873 (
I->getOpcode() != Hexagon::RESTORE_DEALLOC_BEFORE_TAILCALL_V4_EXT &&
874 I->getOpcode() != Hexagon::RESTORE_DEALLOC_BEFORE_TAILCALL_V4_EXT_PIC &&
875 I->getOpcode() != Hexagon::RESTORE_DEALLOC_BEFORE_TAILCALL_V4 &&
876 I->getOpcode() != Hexagon::RESTORE_DEALLOC_BEFORE_TAILCALL_V4_PIC))
877 BuildMI(
MBB, InsertPt, dl, HII.get(Hexagon::L2_deallocframe))
880 if (RegisterSavedAreaSizePlusPadding != 0)
881 BuildMI(
MBB, InsertPt, dl, HII.get(Hexagon::A2_addi), SP)
883 .
addImm(RegisterSavedAreaSizePlusPadding);
892 auto &HRI = *HST.getRegisterInfo();
896 const unsigned int ALLOCFRAME_MAX = 16384;
904 Register SP = HRI.getStackRegister();
906 if (NumBytes >= ALLOCFRAME_MAX) {
908 BuildMI(
MBB, InsertPt, dl, HII.get(Hexagon::S2_allocframe))
916 Register SP = HRI.getStackRegister();
917 BuildMI(
MBB, InsertPt, dl, HII.get(Hexagon::A2_addi), SP)
922 BuildMI(
MBB, InsertPt, dl, HII.get(Hexagon::S2_allocframe))
941 for (
unsigned i = 0; i < Worklist.
size(); ++i) {
942 unsigned BN = Worklist[i];
949 Worklist.
insert(SB->getNumber());
958 if (Path[BN] || DoneF[BN])
966 bool ReachedExit =
false;
968 ReachedExit |= updateExitPaths(*SB, RestoreB, DoneT, DoneF, Path);
984 if (ReachedExit && &
MBB != &RestoreB) {
997static std::optional<MachineBasicBlock::iterator>
1004 auto End =
B.instr_end();
1008 if (!
I.isBundle()) {
1009 if (
I.getOpcode() == Hexagon::S2_allocframe)
1010 return std::next(It);
1014 bool HasCall =
false, HasAllocFrame =
false;
1016 while (++
T !=
End &&
T->isBundled()) {
1017 if (
T->getOpcode() == Hexagon::S2_allocframe)
1018 HasAllocFrame =
true;
1019 else if (
T->isCall())
1023 return HasCall ? It : std::next(It);
1025 return std::nullopt;
1031 insertCFIInstructionsAt(
B, *At);
1040 auto &HRI = *HST.getRegisterInfo();
1046 const MCInstrDesc &CFID = HII.get(TargetOpcode::CFI_INSTRUCTION);
1049 bool HasFP =
hasFP(MF);
1052 unsigned DwFPReg = HRI.getDwarfRegNum(HRI.getFrameRegister(),
true);
1053 unsigned DwRAReg = HRI.getDwarfRegNum(HRI.getRARegister(),
true);
1080 Hexagon::R1, Hexagon::R0, Hexagon::R3, Hexagon::R2,
1081 Hexagon::R17, Hexagon::R16, Hexagon::R19, Hexagon::R18,
1082 Hexagon::R21, Hexagon::R20, Hexagon::R23, Hexagon::R22,
1083 Hexagon::R25, Hexagon::R24, Hexagon::R27, Hexagon::R26,
1084 Hexagon::D0, Hexagon::D1, Hexagon::D8, Hexagon::D9,
1085 Hexagon::D10, Hexagon::D11, Hexagon::D12, Hexagon::D13,
1091 for (
unsigned i = 0; RegsToMove[i] != Hexagon::NoRegister; ++i) {
1094 return C.getReg() ==
Reg;
1118 if (Reg < Hexagon::D0 || Reg > Hexagon::D15) {
1119 unsigned DwarfReg = HRI.getDwarfRegNum(Reg,
true);
1131 Register HiReg = HRI.getSubReg(Reg, Hexagon::isub_hi);
1132 Register LoReg = HRI.getSubReg(Reg, Hexagon::isub_lo);
1133 unsigned HiDwarfReg = HRI.getDwarfRegNum(HiReg,
true);
1134 unsigned LoDwarfReg = HRI.getDwarfRegNum(LoReg,
true);
1150 bool HasExtraAlign = HRI.hasStackRealignment(MF);
1166 if (HasAlloca || HasExtraAlign)
1192 bool Stkchk =
false) {
1193 const char * V4SpillToMemoryFunctions[] = {
1194 "__save_r16_through_r17",
1195 "__save_r16_through_r19",
1196 "__save_r16_through_r21",
1197 "__save_r16_through_r23",
1198 "__save_r16_through_r25",
1199 "__save_r16_through_r27" };
1201 const char * V4SpillToMemoryStkchkFunctions[] = {
1202 "__save_r16_through_r17_stkchk",
1203 "__save_r16_through_r19_stkchk",
1204 "__save_r16_through_r21_stkchk",
1205 "__save_r16_through_r23_stkchk",
1206 "__save_r16_through_r25_stkchk",
1207 "__save_r16_through_r27_stkchk" };
1209 const char * V4SpillFromMemoryFunctions[] = {
1210 "__restore_r16_through_r17_and_deallocframe",
1211 "__restore_r16_through_r19_and_deallocframe",
1212 "__restore_r16_through_r21_and_deallocframe",
1213 "__restore_r16_through_r23_and_deallocframe",
1214 "__restore_r16_through_r25_and_deallocframe",
1215 "__restore_r16_through_r27_and_deallocframe" };
1217 const char * V4SpillFromMemoryTailcallFunctions[] = {
1218 "__restore_r16_through_r17_and_deallocframe_before_tailcall",
1219 "__restore_r16_through_r19_and_deallocframe_before_tailcall",
1220 "__restore_r16_through_r21_and_deallocframe_before_tailcall",
1221 "__restore_r16_through_r23_and_deallocframe_before_tailcall",
1222 "__restore_r16_through_r25_and_deallocframe_before_tailcall",
1223 "__restore_r16_through_r27_and_deallocframe_before_tailcall"
1226 const char **SpillFunc =
nullptr;
1230 SpillFunc = Stkchk ? V4SpillToMemoryStkchkFunctions
1231 : V4SpillToMemoryFunctions;
1234 SpillFunc = V4SpillFromMemoryFunctions;
1237 SpillFunc = V4SpillFromMemoryTailcallFunctions;
1240 assert(SpillFunc &&
"Unknown spill kind");
1245 return SpillFunc[0];
1247 return SpillFunc[1];
1249 return SpillFunc[2];
1251 return SpillFunc[3];
1253 return SpillFunc[4];
1255 return SpillFunc[5];
1270 bool HasExtraAlign = HRI.hasStackRealignment(MF);
1275 Register SP = HRI.getStackRegister();
1277 Register AP = HMFI.getStackAlignBaseReg();
1292 bool UseFP =
false, UseAP =
false;
1297 if (NoOpt && !HasExtraAlign)
1302 UseFP |= (HasAlloca || HasExtraAlign);
1313 bool HasFP =
hasFP(MF);
1314 assert((HasFP || !UseFP) &&
"This function must have frame pointer");
1340 if (
Offset > 0 && !HasFP)
1355 if (!UseFP && !UseAP)
1356 RealOffset = FrameSize+
Offset;
1362 bool &PrologueStubs)
const {
1367 PrologueStubs =
false;
1372 if (useSpillFunction(MF, CSI)) {
1373 PrologueStubs =
true;
1385 if (StkOvrFlowEnabled) {
1387 SpillOpc = IsPIC ? Hexagon::SAVE_REGISTERS_CALL_V4STK_EXT_PIC
1388 : Hexagon::SAVE_REGISTERS_CALL_V4STK_EXT;
1390 SpillOpc = IsPIC ? Hexagon::SAVE_REGISTERS_CALL_V4STK_PIC
1391 : Hexagon::SAVE_REGISTERS_CALL_V4STK;
1394 SpillOpc = IsPIC ? Hexagon::SAVE_REGISTERS_CALL_V4_EXT_PIC
1395 : Hexagon::SAVE_REGISTERS_CALL_V4_EXT;
1397 SpillOpc = IsPIC ? Hexagon::SAVE_REGISTERS_CALL_V4_PIC
1398 : Hexagon::SAVE_REGISTERS_CALL_V4;
1406 addCalleeSaveRegistersAsImpOperand(SaveRegsCall, CSI,
false,
true);
1419 int FI =
I.getFrameIdx();
1421 HII.storeRegToStackSlot(
MBB,
MI, Reg, IsKill, FI, RC, &HRI,
Register());
1438 if (useRestoreFunction(MF, CSI)) {
1444 bool IsPIC = HTM.isPositionIndependent();
1455 RetOpc = IsPIC ? Hexagon::RESTORE_DEALLOC_BEFORE_TAILCALL_V4_EXT_PIC
1456 : Hexagon::RESTORE_DEALLOC_BEFORE_TAILCALL_V4_EXT;
1458 RetOpc = IsPIC ? Hexagon::RESTORE_DEALLOC_BEFORE_TAILCALL_V4_PIC
1459 : Hexagon::RESTORE_DEALLOC_BEFORE_TAILCALL_V4;
1468 RetOpc = IsPIC ? Hexagon::RESTORE_DEALLOC_RET_JMP_V4_EXT_PIC
1469 : Hexagon::RESTORE_DEALLOC_RET_JMP_V4_EXT;
1471 RetOpc = IsPIC ? Hexagon::RESTORE_DEALLOC_RET_JMP_V4_PIC
1472 : Hexagon::RESTORE_DEALLOC_RET_JMP_V4;
1478 addCalleeSaveRegistersAsImpOperand(DeallocCall, CSI,
true,
false);
1485 int FI =
I.getFrameIdx();
1486 HII.loadRegFromStackSlot(
MBB,
MI, Reg, FI, RC, &HRI,
Register());
1496 unsigned Opc =
MI.getOpcode();
1498 assert((Opc == Hexagon::ADJCALLSTACKDOWN || Opc == Hexagon::ADJCALLSTACKUP) &&
1499 "Cannot handle this call frame pseudo instruction");
1513 if (!HasAlloca || !NeedsAlign)
1519 AP = AI->getOperand(0).getReg();
1522 HMFI.setStackAlignBaseReg(AP);
1530 auto IsUsed = [&HRI,&
MRI] (
Register Reg) ->
bool {
1532 if (
MRI.isPhysRegUsed(*AI))
1562 BitVector SRegs(Hexagon::NUM_TARGET_REGS);
1592 bool HasResSub =
false;
1618 BitVector TmpSup(Hexagon::NUM_TARGET_REGS);
1624 for (
int x = TmpSup.find_first(); x >= 0; x = TmpSup.find_next(x)) {
1662 int64_t MinOffset = 0;
1664 for (
const SpillSlot *S = FixedSlots; S != FixedSlots+NumFixed; ++S) {
1669 MinOffset = std::min(MinOffset, S->Offset);
1671 SRegs[S->Reg] =
false;
1680 unsigned Size =
TRI->getSpillSize(*RC);
1681 int64_t Off = MinOffset -
Size;
1683 Off &= -Alignment.
value();
1685 MinOffset = std::min(MinOffset, Off);
1691 dbgs() <<
"CS information: {";
1693 int FI =
I.getFrameIdx();
1705 bool MissedReg =
false;
1725 if (!Hexagon::ModRegsRegClass.
contains(DstR) ||
1726 !Hexagon::ModRegsRegClass.
contains(SrcR))
1729 Register TmpR =
MRI.createVirtualRegister(&Hexagon::IntRegsRegClass);
1730 BuildMI(
B, It,
DL, HII.get(TargetOpcode::COPY), TmpR).
add(
MI->getOperand(1));
1731 BuildMI(
B, It,
DL, HII.get(TargetOpcode::COPY), DstR)
1743 if (!
MI->getOperand(0).isFI())
1747 unsigned Opc =
MI->getOpcode();
1749 bool IsKill =
MI->getOperand(2).isKill();
1750 int FI =
MI->getOperand(0).getIndex();
1754 Register TmpR =
MRI.createVirtualRegister(&Hexagon::IntRegsRegClass);
1755 unsigned TfrOpc = (Opc == Hexagon::STriw_pred) ? Hexagon::C2_tfrpr
1756 : Hexagon::A2_tfrcrr;
1761 BuildMI(
B, It,
DL, HII.get(Hexagon::S2_storeri_io))
1776 if (!
MI->getOperand(1).isFI())
1780 unsigned Opc =
MI->getOpcode();
1782 int FI =
MI->getOperand(1).getIndex();
1785 Register TmpR =
MRI.createVirtualRegister(&Hexagon::IntRegsRegClass);
1786 BuildMI(
B, It,
DL, HII.get(Hexagon::L2_loadri_io), TmpR)
1793 unsigned TfrOpc = (Opc == Hexagon::LDriw_pred) ? Hexagon::C2_tfrrp
1794 : Hexagon::A2_tfrrcr;
1807 if (!
MI->getOperand(0).isFI())
1812 bool IsKill =
MI->getOperand(2).isKill();
1813 int FI =
MI->getOperand(0).getIndex();
1814 auto *RC = &Hexagon::HvxVRRegClass;
1820 Register TmpR0 =
MRI.createVirtualRegister(&Hexagon::IntRegsRegClass);
1823 BuildMI(
B, It,
DL, HII.get(Hexagon::A2_tfrsi), TmpR0)
1826 BuildMI(
B, It,
DL, HII.get(Hexagon::V6_vandqrt), TmpR1)
1832 expandStoreVec(
B, std::prev(It),
MRI, HII, NewRegs);
1844 if (!
MI->getOperand(1).isFI())
1849 int FI =
MI->getOperand(1).getIndex();
1850 auto *RC = &Hexagon::HvxVRRegClass;
1855 Register TmpR0 =
MRI.createVirtualRegister(&Hexagon::IntRegsRegClass);
1858 BuildMI(
B, It,
DL, HII.get(Hexagon::A2_tfrsi), TmpR0)
1863 expandLoadVec(
B, std::prev(It),
MRI, HII, NewRegs);
1865 BuildMI(
B, It,
DL, HII.get(Hexagon::V6_vandvrt), DstR)
1882 if (!
MI->getOperand(0).isFI())
1892 for (
auto R =
B.begin(); R != It; ++R) {
1894 LPR.stepForward(*R, Clobbers);
1899 Register SrcLo = HRI.getSubReg(SrcR, Hexagon::vsub_lo);
1900 Register SrcHi = HRI.getSubReg(SrcR, Hexagon::vsub_hi);
1901 bool IsKill =
MI->getOperand(2).isKill();
1902 int FI =
MI->getOperand(0).getIndex();
1904 unsigned Size = HRI.getSpillSize(Hexagon::HvxVRRegClass);
1905 Align NeedAlign = HRI.getSpillAlign(Hexagon::HvxVRRegClass);
1910 if (LPR.contains(SrcLo)) {
1911 StoreOpc = NeedAlign <= HasAlign ? Hexagon::V6_vS32b_ai
1912 : Hexagon::V6_vS32Ub_ai;
1921 if (LPR.contains(SrcHi)) {
1922 StoreOpc = NeedAlign <= HasAlign ? Hexagon::V6_vS32b_ai
1923 : Hexagon::V6_vS32Ub_ai;
1942 if (!
MI->getOperand(1).isFI())
1947 Register DstHi = HRI.getSubReg(DstR, Hexagon::vsub_hi);
1948 Register DstLo = HRI.getSubReg(DstR, Hexagon::vsub_lo);
1949 int FI =
MI->getOperand(1).getIndex();
1951 unsigned Size = HRI.getSpillSize(Hexagon::HvxVRRegClass);
1952 Align NeedAlign = HRI.getSpillAlign(Hexagon::HvxVRRegClass);
1957 LoadOpc = NeedAlign <= HasAlign ? Hexagon::V6_vL32b_ai
1958 : Hexagon::V6_vL32Ub_ai;
1965 LoadOpc = NeedAlign <= HasAlign ? Hexagon::V6_vL32b_ai
1966 : Hexagon::V6_vL32Ub_ai;
1982 if (!
MI->getOperand(0).isFI())
1988 bool IsKill =
MI->getOperand(2).isKill();
1989 int FI =
MI->getOperand(0).getIndex();
1991 Align NeedAlign = HRI.getSpillAlign(Hexagon::HvxVRRegClass);
1993 unsigned StoreOpc = NeedAlign <= HasAlign ? Hexagon::V6_vS32b_ai
1994 : Hexagon::V6_vS32Ub_ai;
2011 if (!
MI->getOperand(1).isFI())
2017 int FI =
MI->getOperand(1).getIndex();
2019 Align NeedAlign = HRI.getSpillAlign(Hexagon::HvxVRRegClass);
2021 unsigned LoadOpc = NeedAlign <= HasAlign ? Hexagon::V6_vL32b_ai
2022 : Hexagon::V6_vL32Ub_ai;
2036 bool Changed =
false;
2038 for (
auto &
B : MF) {
2041 for (
auto I =
B.begin(), E =
B.end();
I != E;
I = NextI) {
2043 NextI = std::next(
I);
2044 unsigned Opc =
MI->getOpcode();
2047 case TargetOpcode::COPY:
2048 Changed |= expandCopy(
B,
I,
MRI, HII, NewRegs);
2050 case Hexagon::STriw_pred:
2051 case Hexagon::STriw_ctr:
2052 Changed |= expandStoreInt(
B,
I,
MRI, HII, NewRegs);
2054 case Hexagon::LDriw_pred:
2055 case Hexagon::LDriw_ctr:
2056 Changed |= expandLoadInt(
B,
I,
MRI, HII, NewRegs);
2058 case Hexagon::PS_vstorerq_ai:
2059 Changed |= expandStoreVecPred(
B,
I,
MRI, HII, NewRegs);
2061 case Hexagon::PS_vloadrq_ai:
2062 Changed |= expandLoadVecPred(
B,
I,
MRI, HII, NewRegs);
2064 case Hexagon::PS_vloadrw_ai:
2065 Changed |= expandLoadVec2(
B,
I,
MRI, HII, NewRegs);
2067 case Hexagon::PS_vstorerw_ai:
2068 Changed |= expandStoreVec2(
B,
I,
MRI, HII, NewRegs);
2082 SavedRegs.
resize(HRI.getNumRegs());
2092 expandSpillMacros(MF, NewRegs);
2094 optimizeSpillSlots(MF, NewRegs);
2098 if (!NewRegs.
empty() || mayOverflowFrameOffset(MF)) {
2104 SpillRCs.
insert(&Hexagon::IntRegsRegClass);
2109 for (
const auto *RC : SpillRCs) {
2113 switch (RC->
getID()) {
2114 case Hexagon::IntRegsRegClassID:
2117 case Hexagon::HvxQRRegClassID:
2121 unsigned S = HRI.getSpillSize(*RC);
2122 Align A = HRI.getSpillAlign(*RC);
2123 for (
unsigned i = 0; i < Num; i++) {
2142 auto F = DeadMap.find({Reg,0});
2143 if (
F == DeadMap.end())
2145 for (
auto &DR :
F->second)
2146 if (DR.contains(FIR))
2168 auto &HII = *HST.getInstrInfo();
2169 auto &HRI = *HST.getRegisterInfo();
2173 using BlockIndexMap =
2174 std::map<MachineBasicBlock *, HexagonBlockRanges::InstrIndexMap>;
2175 using BlockRangeMap =
2176 std::map<MachineBasicBlock *, HexagonBlockRanges::RangeList>;
2184 SlotInfo() =
default;
2187 BlockIndexMap BlockIndexes;
2189 std::map<int,SlotInfo> FIRangeMap;
2198 if (HaveRC ==
nullptr || HaveRC == NewRC)
2203 if (NewRC->hasSubClassEq(HaveRC))
2210 for (
auto &
B : MF) {
2211 std::map<int,IndexType> LastStore, LastLoad;
2214 auto P = BlockIndexes.insert(
2216 auto &IndexMap =
P.first->second;
2218 << IndexMap <<
'\n');
2220 for (
auto &In :
B) {
2222 bool Load = HII.isLoadFromStackSlot(In, LFI) && !HII.isPredicated(In);
2223 bool Store = HII.isStoreToStackSlot(In, SFI) && !HII.isPredicated(In);
2224 if (Load && Store) {
2236 if (Load || Store) {
2237 int TFI =
Load ? LFI : SFI;
2238 unsigned AM = HII.getAddrMode(In);
2239 SlotInfo &
SI = FIRangeMap[TFI];
2243 unsigned OpNum =
Load ? 0 : 2;
2244 auto *RC = HII.getRegClass(
In.getDesc(), OpNum, &HRI, MF);
2245 RC = getCommonRC(
SI.RC, RC);
2253 unsigned S = HII.getMemAccessSize(In);
2254 if (
SI.Size != 0 &&
SI.Size != S)
2260 for (
auto *Mo :
In.memoperands()) {
2261 if (!Mo->isVolatile() && !Mo->isAtomic())
2272 for (
unsigned i = 0, n =
In.getNumOperands(); i < n; ++i) {
2276 int FI =
Op.getIndex();
2279 if (i+1 >= n || !
In.getOperand(i+1).isImm() ||
2280 In.getOperand(i+1).getImm() != 0)
2282 if (BadFIs.
count(FI))
2287 if (LastStore[FI] == IndexType::None)
2288 LastStore[FI] = IndexType::Entry;
2289 LastLoad[FI] =
Index;
2292 if (LastStore[FI] != IndexType::None)
2293 RL.
add(LastStore[FI], LastLoad[FI],
false,
false);
2294 else if (LastLoad[FI] != IndexType::None)
2295 RL.
add(IndexType::Entry, LastLoad[FI],
false,
false);
2296 LastLoad[FI] = IndexType::None;
2297 LastStore[FI] =
Index;
2304 for (
auto &
I : LastLoad) {
2305 IndexType LL =
I.second;
2306 if (LL == IndexType::None)
2308 auto &RL = FIRangeMap[
I.first].Map[&
B];
2309 IndexType &
LS = LastStore[
I.first];
2310 if (LS != IndexType::None)
2311 RL.
add(LS, LL,
false,
false);
2313 RL.
add(IndexType::Entry, LL,
false,
false);
2314 LS = IndexType::None;
2316 for (
auto &
I : LastStore) {
2317 IndexType
LS =
I.second;
2318 if (LS == IndexType::None)
2320 auto &RL = FIRangeMap[
I.first].Map[&
B];
2321 RL.
add(LS, IndexType::None,
false,
false);
2326 for (
auto &
P : FIRangeMap) {
2327 dbgs() <<
"fi#" <<
P.first;
2328 if (BadFIs.
count(
P.first))
2331 if (
P.second.RC !=
nullptr)
2332 dbgs() << HRI.getRegClassName(
P.second.RC) <<
'\n';
2334 dbgs() <<
"<null>\n";
2335 for (
auto &R :
P.second.Map)
2346 std::map<MachineBasicBlock*,std::vector<int>> BlockFIMap;
2348 for (
auto &
P : FIRangeMap) {
2350 if (BadFIs.
count(
P.first))
2352 for (
auto &
B : MF) {
2353 auto F =
P.second.Map.find(&
B);
2355 if (
F ==
P.second.Map.end() ||
F->second.empty())
2358 if (
IR.start() == IndexType::Entry)
2359 LoxFIs.insert(
P.first);
2360 BlockFIMap[&
B].push_back(
P.first);
2365 dbgs() <<
"Block-to-FI map (* -- live-on-exit):\n";
2366 for (
auto &
P : BlockFIMap) {
2367 auto &FIs =
P.second;
2371 for (
auto I : FIs) {
2372 dbgs() <<
" fi#" <<
I;
2373 if (LoxFIs.count(
I))
2385 for (
auto &
B : MF) {
2386 auto F = BlockIndexes.find(&
B);
2387 assert(
F != BlockIndexes.end());
2394 for (
auto FI : BlockFIMap[&
B]) {
2395 if (BadFIs.
count(FI))
2399 for (
auto &
Range : RL) {
2401 if (!IndexType::isInstr(
Range.start()) ||
2402 !IndexType::isInstr(
Range.end()))
2406 assert(
SI.mayStore() &&
"Unexpected start instruction");
2411 SrcOp.getSubReg() };
2412 auto *RC = HII.getRegClass(
SI.getDesc(), 2, &HRI, MF);
2430 if (SrcRR.
Reg != FoundR || SrcRR.
Sub != 0) {
2432 CopyIn =
BuildMI(
B, StartIt,
DL, HII.get(TargetOpcode::COPY), FoundR)
2438 if (LoxFIs.count(FI) && (&
Range == &RL.back())) {
2440 if (
unsigned SR =
SrcOp.getSubReg())
2441 SrcOp.setReg(HRI.getSubReg(FoundR, SR));
2443 SrcOp.setReg(FoundR);
2446 SrcOp.setIsKill(
false);
2453 for (
auto It = StartIt; It != EndIt; It = NextIt) {
2455 NextIt = std::next(It);
2457 if (!HII.isLoadFromStackSlot(
MI, TFI) || TFI != FI)
2460 assert(
MI.getOperand(0).getSubReg() == 0);
2462 if (DstR != FoundR) {
2464 unsigned MemSize = HII.getMemAccessSize(
MI);
2466 unsigned CopyOpc = TargetOpcode::COPY;
2467 if (HII.isSignExtendingLoad(
MI))
2468 CopyOpc = (MemSize == 1) ? Hexagon::A2_sxtb : Hexagon::A2_sxth;
2469 else if (HII.isZeroExtendingLoad(
MI))
2470 CopyOpc = (MemSize == 1) ? Hexagon::A2_zxtb : Hexagon::A2_zxth;
2471 CopyOut =
BuildMI(
B, It,
DL, HII.get(CopyOpc), DstR)
2487void HexagonFrameLowering::expandAlloca(
MachineInstr *AI,
2513 BuildMI(MB, AI,
DL, HII.get(Hexagon::A2_sub), Rd)
2518 BuildMI(MB, AI,
DL, HII.get(Hexagon::A2_sub), SP)
2524 BuildMI(MB, AI,
DL, HII.get(Hexagon::A2_andir), Rd)
2528 BuildMI(MB, AI,
DL, HII.get(Hexagon::A2_andir), SP)
2534 BuildMI(MB, AI,
DL, HII.get(TargetOpcode::COPY), SP)
2539 BuildMI(MB, AI,
DL, HII.get(Hexagon::A2_addi), Rd)
2559 if (
I.getOpcode() == Hexagon::PS_aligna)
2566void HexagonFrameLowering::addCalleeSaveRegistersAsImpOperand(
MachineInstr *
MI,
2567 const CSIVect &CSI,
bool IsDef,
bool IsKill)
const {
2578 const CSIVect &CSI)
const {
2591 BitVector Regs(Hexagon::NUM_TARGET_REGS);
2594 if (!Hexagon::DoubleRegsRegClass.
contains(R))
2598 int F = Regs.find_first();
2599 if (
F != Hexagon::D8)
2602 int N = Regs.find_next(
F);
2603 if (
N >= 0 &&
N !=
F+1)
2612 const CSIVect &CSI)
const {
2613 if (shouldInlineCSR(MF, CSI))
2615 unsigned NumCSI = CSI.size();
2621 return Threshold < NumCSI;
2624bool HexagonFrameLowering::useRestoreFunction(
const MachineFunction &MF,
2625 const CSIVect &CSI)
const {
2626 if (shouldInlineCSR(MF, CSI))
2636 unsigned NumCSI = CSI.size();
2642 return Threshold < NumCSI;
2645bool HexagonFrameLowering::mayOverflowFrameOffset(
MachineFunction &MF)
const {
2650 if (HST.useHVXOps() && StackSize > 256)
2657 bool HasImmStack =
false;
2658 unsigned MinLS = ~0
u;
2663 switch (
MI.getOpcode()) {
2664 case Hexagon::S4_storeirit_io:
2665 case Hexagon::S4_storeirif_io:
2666 case Hexagon::S4_storeiri_io:
2669 case Hexagon::S4_storeirht_io:
2670 case Hexagon::S4_storeirhf_io:
2671 case Hexagon::S4_storeirh_io:
2674 case Hexagon::S4_storeirbt_io:
2675 case Hexagon::S4_storeirbf_io:
2676 case Hexagon::S4_storeirb_io:
2677 if (
MI.getOperand(0).isFI())
2679 MinLS = std::min(MinLS, LS);
2686 return !isUInt<6>(StackSize >> MinLS);
2693struct HexagonFrameSortingObject {
2694 bool IsValid =
false;
2700struct HexagonFrameSortingComparator {
2701 inline bool operator()(
const HexagonFrameSortingObject &
A,
2702 const HexagonFrameSortingObject &
B)
const {
2703 return std::make_tuple(!
A.IsValid,
A.ObjectAlignment,
A.Size) <
2704 std::make_tuple(!
B.IsValid,
B.ObjectAlignment,
B.Size);
2714 if (ObjectsToAllocate.
empty())
2718 int NObjects = ObjectsToAllocate.
size();
2726 if (i != ObjectsToAllocate[j])
2737 SortingObjects[i].IsValid =
true;
2738 SortingObjects[i].Index = i;
2748 for (
auto &Obj : SortingObjects) {
2751 ObjectsToAllocate[--i] = Obj.Index;
unsigned const MachineRegisterInfo * MRI
MachineBasicBlock MachineBasicBlock::iterator DebugLoc DL
This file contains the simple types necessary to represent the attributes associated with functions a...
This file implements the BitVector class.
static GCRegistry::Add< OcamlGC > B("ocaml", "ocaml 3.10-compatible GC")
static GCRegistry::Add< ErlangGC > A("erlang", "erlang-compatible garbage collector")
static RegisterPass< DebugifyModulePass > DM("debugify", "Attach debug info to everything")
This file defines the DenseMap class.
static MachineInstr * getReturn(MachineBasicBlock &MBB)
Returns the "return" instruction from this block, or nullptr if there isn't any.
static cl::opt< unsigned > ShrinkLimit("shrink-frame-limit", cl::init(std::numeric_limits< unsigned >::max()), cl::Hidden, cl::desc("Max count of stack frame shrink-wraps"))
static bool isOptNone(const MachineFunction &MF)
static cl::opt< int > SpillFuncThreshold("spill-func-threshold", cl::Hidden, cl::desc("Specify O2(not Os) spill func threshold"), cl::init(6))
static std::optional< MachineBasicBlock::iterator > findCFILocation(MachineBasicBlock &B)
static cl::opt< bool > EliminateFramePointer("hexagon-fp-elim", cl::init(true), cl::Hidden, cl::desc("Refrain from using FP whenever possible"))
static bool enableAllocFrameElim(const MachineFunction &MF)
static const char * getSpillFunctionFor(Register MaxReg, SpillKind SpillType, bool Stkchk=false)
static bool hasReturn(const MachineBasicBlock &MBB)
Returns true if MBB contains an instruction that returns.
static cl::opt< bool > EnableSaveRestoreLong("enable-save-restore-long", cl::Hidden, cl::desc("Enable long calls for save-restore stubs."), cl::init(false))
static bool needToReserveScavengingSpillSlots(MachineFunction &MF, const HexagonRegisterInfo &HRI, const TargetRegisterClass *RC)
Returns true if there are no caller-saved registers available in class RC.
static bool isOptSize(const MachineFunction &MF)
static Register getMax32BitSubRegister(Register Reg, const TargetRegisterInfo &TRI, bool hireg=true)
Map a register pair Reg to the subregister that has the greater "number", i.e.
static cl::opt< int > SpillFuncThresholdOs("spill-func-threshold-Os", cl::Hidden, cl::desc("Specify Os spill func threshold"), cl::init(1))
static bool needsStackFrame(const MachineBasicBlock &MBB, const BitVector &CSR, const HexagonRegisterInfo &HRI)
Checks if the basic block contains any instruction that needs a stack frame to be already in place.
static cl::opt< bool > DisableDeallocRet("disable-hexagon-dealloc-ret", cl::Hidden, cl::desc("Disable Dealloc Return for Hexagon target"))
static cl::opt< bool > EnableShrinkWrapping("hexagon-shrink-frame", cl::init(true), cl::Hidden, cl::desc("Enable stack frame shrink wrapping"))
static bool hasTailCall(const MachineBasicBlock &MBB)
Returns true if MBB has a machine instructions that indicates a tail call in the block.
static cl::opt< unsigned > NumberScavengerSlots("number-scavenger-slots", cl::Hidden, cl::desc("Set the number of scavenger slots"), cl::init(2))
static Register getMaxCalleeSavedReg(ArrayRef< CalleeSavedInfo > CSI, const TargetRegisterInfo &TRI)
Returns the callee saved register with the largest id in the vector.
static bool isMinSize(const MachineFunction &MF)
static cl::opt< unsigned > SpillOptMax("spill-opt-max", cl::Hidden, cl::init(std::numeric_limits< unsigned >::max()))
static unsigned SpillOptCount
static void dump_registers(BitVector &Regs, const TargetRegisterInfo &TRI)
static bool isRestoreCall(unsigned Opc)
static cl::opt< bool > OptimizeSpillSlots("hexagon-opt-spill", cl::Hidden, cl::init(true), cl::desc("Optimize spill slots"))
static cl::opt< bool > EnableStackOVFSanitizer("enable-stackovf-sanitizer", cl::Hidden, cl::desc("Enable runtime checks for stack overflow."), cl::init(false))
Legalize the Machine IR a function s Machine IR
This file implements the LivePhysRegs utility for tracking liveness of physical registers.
unsigned const TargetRegisterInfo * TRI
static unsigned getReg(const MCDisassembler *D, unsigned RC, unsigned RegNo)
ConstantRange Range(APInt(BitWidth, Low), APInt(BitWidth, High))
#define INITIALIZE_PASS(passName, arg, name, cfg, analysis)
This file builds on the ADT/GraphTraits.h file to build a generic graph post order iterator.
This file declares the machine register scavenger class.
assert(ImpDefSCC.getReg()==AMDGPU::SCC &&ImpDefSCC.isDef())
bool isDead(const MachineInstr &MI, const MachineRegisterInfo &MRI)
static bool contains(SmallPtrSetImpl< ConstantExpr * > &Cache, ConstantExpr *Expr, Constant *C)
This file implements a set that has insertion order iteration characteristics.
This file defines the SmallSet class.
This file defines the SmallVector class.
ArrayRef - Represent a constant reference to an array (0 or more elements consecutively in memory),...
size_t size() const
size - Get the array size.
bool empty() const
empty - Check if the array is empty.
int find_first() const
find_first - Returns the index of the first set bit, -1 if none of the bits are set.
void resize(unsigned N, bool t=false)
resize - Grow or shrink the bitvector.
int find_next(unsigned Prev) const
find_next - Returns the index of the next set bit following the "Prev" bit.
The CalleeSavedInfo class tracks the information need to locate where a callee saved register is in t...
This class represents an Operation in the Expression.
NodeT * findNearestCommonDominator(NodeT *A, NodeT *B) const
Find nearest common dominator basic block for basic block A and B.
bool dominates(const DomTreeNodeBase< NodeT > *A, const DomTreeNodeBase< NodeT > *B) const
dominates - Returns true iff A dominates B.
void recalculate(ParentType &Func)
recalculate - compute a dominator tree for the given function
FunctionPass class - This class is used to implement most global optimizations.
bool hasMinSize() const
Optimize this function for minimum size (-Oz).
bool hasOptNone() const
Do not optimize this function (-O0).
bool isVarArg() const
isVarArg - Return true if this function takes a variable number of arguments.
void replaceInstr(MachineInstr *OldMI, MachineInstr *NewMI)
IndexType getIndex(MachineInstr *MI) const
MachineInstr * getInstr(IndexType Idx) const
void add(IndexType Start, IndexType End, bool Fixed, bool TiedEnd)
const MachineInstr * getAlignaInstr(const MachineFunction &MF) const
void insertCFIInstructions(MachineFunction &MF) const
bool hasFPImpl(const MachineFunction &MF) const override
bool enableCalleeSaveSkip(const MachineFunction &MF) const override
Returns true if the target can safely skip saving callee-saved registers for noreturn nounwind functi...
MachineBasicBlock::iterator eliminateCallFramePseudoInstr(MachineFunction &MF, MachineBasicBlock &MBB, MachineBasicBlock::iterator I) const override
This method is called during prolog/epilog code insertion to eliminate call frame setup and destroy p...
StackOffset getFrameIndexReference(const MachineFunction &MF, int FI, Register &FrameReg) const override
getFrameIndexReference - This method should return the base register and offset used to reference a f...
void emitPrologue(MachineFunction &MF, MachineBasicBlock &MBB) const override
Perform most of the PEI work here:
void determineCalleeSaves(MachineFunction &MF, BitVector &SavedRegs, RegScavenger *RS) const override
This method determines which of the registers reported by TargetRegisterInfo::getCalleeSavedRegs() sh...
void orderFrameObjects(const MachineFunction &MF, SmallVectorImpl< int > &ObjectsToAllocate) const override
Order the symbols in the local stack frame.
void processFunctionBeforeFrameFinalized(MachineFunction &MF, RegScavenger *RS=nullptr) const override
processFunctionBeforeFrameFinalized - This method is called immediately before the specified function...
const SpillSlot * getCalleeSavedSpillSlots(unsigned &NumEntries) const override
getCalleeSavedSpillSlots - This method returns a pointer to an array of pairs, that contains an entry...
bool needsAligna(const MachineFunction &MF) const
bool assignCalleeSavedSpillSlots(MachineFunction &MF, const TargetRegisterInfo *TRI, std::vector< CalleeSavedInfo > &CSI) const override
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.
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.
Hexagon target-specific information for each MachineFunction.
bool isEHReturnCalleeSaveReg(Register Reg) const
const MCPhysReg * getCalleeSavedRegs(const MachineFunction *MF) const override
Code Generation virtual methods...
const MCPhysReg * getCallerSavedRegs(const MachineFunction *MF, const TargetRegisterClass *RC) const
const HexagonInstrInfo * getInstrInfo() const override
bool isEnvironmentMusl() const
A set of physical registers with utility functions to track liveness when walking backward/forward th...
static MCCFIInstruction cfiDefCfa(MCSymbol *L, unsigned Register, int64_t Offset, SMLoc Loc={})
.cfi_def_cfa defines a rule for computing CFA as: take address from Register and add Offset to it.
static MCCFIInstruction createOffset(MCSymbol *L, unsigned Register, int64_t Offset, SMLoc Loc={})
.cfi_offset Previous value of Register is saved at offset Offset from CFA.
MCSymbol * createTempSymbol()
Create a temporary symbol with a unique name.
Describe properties that are true of each instruction in the target description file.
MCRegAliasIterator enumerates all registers aliasing Reg.
MCSymbol - Instances of this class represent a symbol name in the MC file, and MCSymbols are created ...
int getNumber() const
MachineBasicBlocks are uniquely numbered at the function level, unless they're not in a MachineFuncti...
iterator getFirstTerminator()
Returns an iterator to the first terminator instruction of this basic block.
DebugLoc findDebugLoc(instr_iterator MBBI)
Find the next valid DebugLoc starting at MBBI, skipping any debug instructions.
iterator getLastNonDebugInstr(bool SkipPseudoOp=true)
Returns an iterator to the last non-debug instruction in the basic block, or end().
void addLiveIn(MCRegister PhysReg, LaneBitmask LaneMask=LaneBitmask::getAll())
Adds the specified register as a live in.
const MachineFunction * getParent() const
Return the MachineFunction containing this basic block.
instr_iterator erase(instr_iterator I)
Remove an instruction from the instruction list and delete it.
iterator_range< iterator > terminators()
iterator_range< succ_iterator > successors()
bool isLiveIn(MCRegister Reg, LaneBitmask LaneMask=LaneBitmask::getAll()) const
Return true if the specified register is in the live in set.
DominatorTree Class - Concrete subclass of DominatorTreeBase that is used to compute a normal dominat...
bool dominates(const MachineInstr *A, const MachineInstr *B) const
The MachineFrameInfo class represents an abstract stack frame until prolog/epilog code is inserted.
void setMaxCallFrameSize(uint64_t S)
bool hasVarSizedObjects() const
This method may be called any time after instruction selection is complete to determine if the stack ...
bool isObjectPreAllocated(int ObjectIdx) const
Return true if the object was pre-allocated into the local block.
uint64_t getStackSize() const
Return the number of bytes that must be allocated to hold all of the fixed size frame objects.
bool hasCalls() const
Return true if the current function has any function calls.
Align getMaxAlign() const
Return the alignment in bytes that this function must be aligned to, which is greater than the defaul...
uint64_t getMaxCallFrameSize() const
Return the maximum size of a call frame that must be allocated for an outgoing function call.
int CreateSpillStackObject(uint64_t Size, Align Alignment)
Create a new statically sized stack object that represents a spill slot, returning a nonnegative iden...
uint64_t estimateStackSize(const MachineFunction &MF) const
Estimate and return the size of the stack frame.
Align getObjectAlign(int ObjectIdx) const
Return the alignment of the specified stack object.
int64_t getObjectSize(int ObjectIdx) const
Return the size of the specified object.
const std::vector< CalleeSavedInfo > & getCalleeSavedInfo() const
Returns a reference to call saved info vector for the current function.
int getObjectIndexEnd() const
Return one past the maximum frame object index.
int CreateFixedSpillStackObject(uint64_t Size, int64_t SPOffset, bool IsImmutable=false)
Create a spill slot at a fixed location on the stack.
int64_t getObjectOffset(int ObjectIdx) const
Return the assigned stack offset of the specified object from the incoming stack pointer.
void setStackSize(uint64_t Size)
Set the size of the stack.
bool isFixedObjectIndex(int ObjectIdx) const
Returns true if the specified index corresponds to a fixed stack object.
MachineFunctionPass - This class adapts the FunctionPass interface to allow convenient creation of pa...
virtual bool runOnMachineFunction(MachineFunction &MF)=0
runOnMachineFunction - This method must be overloaded to perform the desired machine code transformat...
virtual MachineFunctionProperties getRequiredProperties() const
Properties which a MachineFunction may have at a given point in time.
MachineFunctionProperties & set(Property P)
unsigned addFrameInst(const MCCFIInstruction &Inst)
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.
MachineMemOperand * getMachineMemOperand(MachinePointerInfo PtrInfo, MachineMemOperand::Flags f, LLT MemTy, Align base_alignment, const AAMDNodes &AAInfo=AAMDNodes(), const MDNode *Ranges=nullptr, SyncScope::ID SSID=SyncScope::System, AtomicOrdering Ordering=AtomicOrdering::NotAtomic, AtomicOrdering FailureOrdering=AtomicOrdering::NotAtomic)
getMachineMemOperand - Allocate a new MachineMemOperand.
bool needsFrameMoves() const
True if this function needs frame moves for debug or exceptions.
MachineFrameInfo & getFrameInfo()
getFrameInfo - Return the frame info object for the current function.
MCContext & getContext() const
MachineRegisterInfo & getRegInfo()
getRegInfo - Return information about the registers currently in use.
MachineBasicBlock * getBlockNumbered(unsigned N) const
getBlockNumbered - MachineBasicBlocks are automatically numbered when they are inserted into the mach...
Function & getFunction()
Return the LLVM function that this machine code represents.
unsigned getNumBlockIDs() const
getNumBlockIDs - Return the number of MBB ID's allocated.
Ty * getInfo()
getInfo - Keep track of various per-function pieces of information for backends that would like to do...
const MachineBasicBlock & front() const
const TargetMachine & getTarget() const
getTarget - Return the target machine this machine code is compiled with
const MachineInstrBuilder & addExternalSymbol(const char *FnName, unsigned TargetFlags=0) const
const MachineInstrBuilder & addCFIIndex(unsigned CFIIndex) const
const MachineInstrBuilder & setMIFlag(MachineInstr::MIFlag Flag) const
const MachineInstrBuilder & addImm(int64_t Val) const
Add a new immediate operand.
const MachineInstrBuilder & add(const MachineOperand &MO) const
const MachineInstrBuilder & addFrameIndex(int Idx) const
const MachineInstrBuilder & addReg(Register RegNo, unsigned flags=0, unsigned SubReg=0) const
Add a new virtual register operand.
const MachineInstrBuilder & cloneMemRefs(const MachineInstr &OtherMI) const
const MachineInstrBuilder & addMemOperand(MachineMemOperand *MMO) const
const MachineInstrBuilder & addDef(Register RegNo, unsigned Flags=0, unsigned SubReg=0) const
Add a virtual register definition operand.
instr_iterator getInstrIterator() const
Representation of each machine instruction.
unsigned getOpcode() const
Returns the opcode of this MachineInstr.
bool isReturn(QueryType Type=AnyInBundle) const
const MachineBasicBlock * getParent() const
void addOperand(MachineFunction &MF, const MachineOperand &Op)
Add the specified operand to the instruction.
void copyImplicitOps(MachineFunction &MF, const MachineInstr &MI)
Copy implicit register operands from specified instruction to this instruction.
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
@ MOStore
The memory access writes data.
MachineOperand class - Representation of each machine instruction operand.
Register getReg() const
getReg - Returns the register number.
static MachineOperand CreateReg(Register Reg, bool isDef, bool isImp=false, bool isKill=false, bool isDead=false, bool isUndef=false, bool isEarlyClobber=false, unsigned SubReg=0, bool isDebug=false, bool isInternalRead=false, bool isRenamable=false)
MachinePostDominatorTree - an analysis pass wrapper for DominatorTree used to compute the post-domina...
MachineBasicBlock * findNearestCommonDominator(ArrayRef< MachineBasicBlock * > Blocks) const
Returns the nearest common dominator of the given blocks.
MachineRegisterInfo - Keep track of information for virtual and physical registers,...
PassRegistry - This class manages the registration and intitialization of the pass subsystem as appli...
static PassRegistry * getPassRegistry()
getPassRegistry - Access the global registry object, which is automatically initialized at applicatio...
void addScavengingFrameIndex(int FI)
Add a scavenging frame index.
Wrapper class representing virtual and physical registers.
constexpr bool isValid() const
constexpr bool isPhysical() const
Return true if the specified register number is in the physical register namespace.
A vector that has set insertion semantics.
size_type size() const
Determine the number of elements in the SetVector.
bool insert(const value_type &X)
Insert a new element into the SetVector.
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.
StackOffset holds a fixed and a scalable offset in bytes.
int64_t getFixed() const
Returns the fixed component of the stack.
static StackOffset getFixed(int64_t Fixed)
bool hasFP(const MachineFunction &MF) const
hasFP - Return true if the specified function should have a dedicated frame pointer register.
virtual void determineCalleeSaves(MachineFunction &MF, BitVector &SavedRegs, RegScavenger *RS=nullptr) const
This method determines which of the registers reported by TargetRegisterInfo::getCalleeSavedRegs() sh...
Align getStackAlign() const
getStackAlignment - This method returns the number of bytes to which the stack pointer must be aligne...
Primary interface to the complete machine description for the target machine.
CodeGenOptLevel getOptLevel() const
Returns the optimization level: None, Less, Default, or Aggressive.
bool isPositionIndependent() const
unsigned getID() const
Return the register class ID number.
bool hasSubClassEq(const TargetRegisterClass *RC) const
Returns true if RC is a sub-class of or equal to this class.
ArrayRef< MCPhysReg > getRawAllocationOrder(const MachineFunction &MF) const
Returns the preferred order for allocating registers from this register class in MF.
TargetRegisterInfo base class - We assume that the target defines a static array of TargetRegisterDes...
self_iterator getIterator()
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
@ C
The default llvm calling convention, compatible with C.
unsigned ID
LLVM IR allows to use arbitrary numbers as calling convention identifiers.
@ Kill
The last use of a register.
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.
void stable_sort(R &&Range)
MachineInstrBuilder BuildMI(MachineFunction &MF, const MIMetadata &MIMD, const MCInstrDesc &MCID)
Builder interface. Specify how to create the initial instruction itself.
raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
FunctionPass * createHexagonCallFrameInformation()
void initializeHexagonCallFrameInformationPass(PassRegistry &)
unsigned getKillRegState(bool B)
uint64_t alignTo(uint64_t Size, Align A)
Returns a multiple of A needed to store Size bytes.
auto find_if(R &&Range, UnaryPredicate P)
Provide wrappers to std::find_if which take ranges instead of having to pass begin/end explicitly.
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.
Printable printMBBReference(const MachineBasicBlock &MBB)
Prints a machine basic block reference.
This struct is a compact representation of a valid (non-zero power of two) alignment.
uint64_t value() const
This is a hole in the type system and should not be abused.
static RegisterSet expandToSubRegs(RegisterRef R, const MachineRegisterInfo &MRI, const TargetRegisterInfo &TRI)
std::map< RegisterRef, RangeList > RegToRangeMap
static MachinePointerInfo getStack(MachineFunction &MF, int64_t Offset, uint8_t ID=0)
Stack pointer relative access.