72#define DEBUG_TYPE "aarch64-machine-sme-abi"
124 Register StatusFlags = AArch64::NoRegister;
125 Register X0Save = AArch64::NoRegister;
131 ZAState NeededState{ZAState::ANY};
133 LiveRegs PhysLiveRegs = LiveRegs::None;
140 ZAState FixedEntryState{ZAState::ANY};
141 ZAState DesiredIncomingState{ZAState::ANY};
142 ZAState DesiredOutgoingState{ZAState::ANY};
143 LiveRegs PhysLiveRegsAtEntry = LiveRegs::None;
144 LiveRegs PhysLiveRegsAtExit = LiveRegs::None;
150 std::optional<MachineBasicBlock::iterator> AfterSMEProloguePt;
151 LiveRegs PhysLiveRegsAfterSMEPrologue = LiveRegs::None;
158 EmitContext() =
default;
163 return *TPIDR2BlockFI;
166 return *TPIDR2BlockFI;
171 if (AgnosticZABufferPtr != AArch64::NoRegister)
172 return AgnosticZABufferPtr;
175 AgnosticZABufferPtr =
176 BufferPtr != AArch64::NoRegister
179 return AgnosticZABufferPtr;
192 bool needsSaveBuffer()
const {
193 assert(!(TPIDR2BlockFI && AgnosticZABufferPtr) &&
194 "Cannot have both a TPIDR2 block and agnostic ZA buffer");
195 return TPIDR2BlockFI || AgnosticZABufferPtr != AArch64::NoRegister;
199 std::optional<int> ZT0SaveFI;
200 std::optional<int> TPIDR2BlockFI;
201 Register AgnosticZABufferPtr = AArch64::NoRegister;
204StringRef getZAStateString(ZAState State) {
205#define MAKE_CASE(V) \
227 return AArch64::MPR128RegClass.contains(SR) ||
228 AArch64::ZTRRegClass.contains(SR);
234static std::pair<ZAState, MachineBasicBlock::iterator>
244 if (
MI.getOpcode() == AArch64::InOutZAUsePseudo)
245 return {ZAState::ACTIVE, std::prev(InsertPt)};
248 if (
MI.getOpcode() == AArch64::RequiresZASavePseudo)
249 return {ZAState::LOCAL_SAVED, std::prev(InsertPt)};
258 if (
MI.getOpcode() == AArch64::RequiresZT0SavePseudo) {
259 return {SMEFnAttrs.
hasZAState() ? ZAState::ACTIVE_ZT0_SAVED
260 : ZAState::LOCAL_COMMITTED,
261 std::prev(InsertPt)};
266 return {ZAOffAtReturn ? ZAState::OFF : ZAState::ACTIVE, InsertPt};
269 for (
auto &MO :
MI.operands()) {
270 if (isZAorZTRegOp(
TRI, MO))
271 return {ZAState::ACTIVE, InsertPt};
274 return {ZAState::ANY, InsertPt};
278 inline static char ID = 0;
285 StringRef getPassName()
const override {
return "Machine SME ABI pass"; }
299 FunctionInfo collectNeededZAStates(
SMEAttrs SMEFnAttrs);
304 const FunctionInfo &FnInfo);
308 void insertStateChanges(EmitContext &,
const FunctionInfo &FnInfo,
329 bool ClearTPIDR2,
bool On);
340 LiveRegs PhysLiveRegs,
bool IsSave);
348 std::pair<MachineBasicBlock::iterator, LiveRegs>
358 if (AFI->getSMEFnAttrs().hasAgnosticZAInterface())
359 return emitFullZASaveRestore(Context,
MBB,
MBBI, PhysLiveRegs,
361 return emitSetupLazySave(Context,
MBB,
MBBI);
365 if (AFI->getSMEFnAttrs().hasAgnosticZAInterface())
366 return emitFullZASaveRestore(Context,
MBB,
MBBI, PhysLiveRegs,
368 return emitRestoreLazySave(Context,
MBB,
MBBI, PhysLiveRegs);
373 if (AFI->getSMEFnAttrs().hasAgnosticZAInterface())
374 return emitAllocateFullZASaveBuffer(Context,
MBB,
MBBI, PhysLiveRegs);
375 return emitAllocateLazySaveBuffer(Context,
MBB,
MBBI);
384 unsigned Marker)
const;
391 void emitError(
const Twine &Message) {
419 LiveRegs PhysLiveRegs = LiveRegs::None;
421 PhysLiveRegs |= LiveRegs::NZCV;
425 PhysLiveRegs |= LiveRegs::W0;
426 if (!LiveUnits.
available(AArch64::W0_HI))
427 PhysLiveRegs |= LiveRegs::W0_HI;
432 if (PhysLiveRegs & LiveRegs::NZCV)
433 LiveUnits.
addReg(AArch64::NZCV);
434 if (PhysLiveRegs & LiveRegs::W0)
435 LiveUnits.
addReg(AArch64::W0);
436 if (PhysLiveRegs & LiveRegs::W0_HI)
437 LiveUnits.
addReg(AArch64::W0_HI);
440[[maybe_unused]]
bool isCallStartOpcode(
unsigned Opc) {
442 case AArch64::TLSDESC_CALLSEQ:
443 case AArch64::TLSDESC_AUTH_CALLSEQ:
444 case AArch64::ADJCALLSTACKDOWN:
451FunctionInfo MachineSMEABI::collectNeededZAStates(
SMEAttrs SMEFnAttrs) {
454 "Expected function to have ZA/ZT0 state!");
457 LiveRegs PhysLiveRegsAfterSMEPrologue = LiveRegs::None;
458 std::optional<MachineBasicBlock::iterator> AfterSMEProloguePt;
461 BlockInfo &
Block = Blocks[
MBB.getNumber()];
463 if (
MBB.isEntryBlock()) {
465 Block.FixedEntryState = ZAState::ENTRY;
466 }
else if (
MBB.isEHPad()) {
468 Block.FixedEntryState = ZAState::LOCAL_COMMITTED;
474 Block.PhysLiveRegsAtExit = getPhysLiveRegs(LiveUnits);
475 auto FirstTerminatorInsertPt =
MBB.getFirstTerminator();
476 auto FirstNonPhiInsertPt =
MBB.getFirstNonPHI();
478 if (
MI.isDebugInstr())
483 LiveRegs PhysLiveRegs = getPhysLiveRegs(LiveUnits);
488 if (
MI.getOpcode() == AArch64::SMEStateAllocPseudo) {
489 AfterSMEProloguePt =
MBBI;
490 PhysLiveRegsAfterSMEPrologue = PhysLiveRegs;
493 auto [NeededState, InsertPt] = getInstNeededZAState(*
TRI,
MI, SMEFnAttrs);
494 assert((InsertPt ==
MBBI || isCallStartOpcode(InsertPt->getOpcode())) &&
495 "Unexpected state change insertion point!");
496 if (
MBBI == FirstTerminatorInsertPt)
497 Block.PhysLiveRegsAtExit = PhysLiveRegs;
498 if (
MBBI == FirstNonPhiInsertPt)
499 Block.PhysLiveRegsAtEntry = PhysLiveRegs;
500 if (NeededState != ZAState::ANY)
501 Block.Insts.push_back({NeededState, InsertPt, PhysLiveRegs});
505 std::reverse(
Block.Insts.begin(),
Block.Insts.end());
509 if (!
Block.Insts.empty()) {
510 Block.DesiredIncomingState =
Block.Insts.front().NeededState;
511 Block.DesiredOutgoingState =
Block.Insts.back().NeededState;
515 return FunctionInfo{std::move(Blocks), AfterSMEProloguePt,
516 PhysLiveRegsAfterSMEPrologue};
522MachineSMEABI::assignBundleZAStates(
const EdgeBundles &Bundles,
523 const FunctionInfo &FnInfo) {
526 std::optional<ZAState> BundleState;
527 for (
unsigned BlockID : Bundles.
getBlocks(
I)) {
528 const BlockInfo &
Block = FnInfo.Blocks[BlockID];
532 if (
Block.FixedEntryState != ZAState::ANY ||
540 BundleState =
Block.DesiredIncomingState;
541 else if (BundleState !=
Block.DesiredIncomingState)
542 BundleState = ZAState::ACTIVE;
545 if (!BundleState || BundleState == ZAState::ANY)
546 BundleState = ZAState::ACTIVE;
548 BundleStates[
I] = *BundleState;
554std::pair<MachineBasicBlock::iterator, LiveRegs>
555MachineSMEABI::findStateChangeInsertionPoint(
560 if (Inst !=
Block.Insts.end()) {
561 InsertPt = Inst->InsertPt;
562 PhysLiveRegs = Inst->PhysLiveRegs;
564 InsertPt =
MBB.getFirstTerminator();
565 PhysLiveRegs =
Block.PhysLiveRegsAtExit;
568 if (PhysLiveRegs == LiveRegs::None)
569 return {InsertPt, PhysLiveRegs};
573 if (Inst ==
Block.Insts.begin()) {
574 PrevStateChangeI =
MBB.begin();
580 PrevStateChangeI = std::prev(Inst)->InsertPt;
585 setPhysLiveRegs(LiveUnits, PhysLiveRegs);
586 auto BestCandidate = std::make_pair(InsertPt, PhysLiveRegs);
589 if (
I->getOpcode() ==
TII->getCallFrameDestroyOpcode() ||
I->isCall())
592 LiveRegs CurrentPhysLiveRegs = getPhysLiveRegs(LiveUnits);
595 if (!(CurrentPhysLiveRegs & LiveRegs::NZCV))
596 BestCandidate = {
I, CurrentPhysLiveRegs};
597 if (CurrentPhysLiveRegs == LiveRegs::None)
600 return BestCandidate;
603void MachineSMEABI::insertStateChanges(EmitContext &Context,
604 const FunctionInfo &FnInfo,
608 const BlockInfo &
Block = FnInfo.Blocks[
MBB.getNumber()];
609 ZAState InState = BundleStates[Bundles.
getBundle(
MBB.getNumber(),
612 ZAState CurrentState =
Block.FixedEntryState;
613 if (CurrentState == ZAState::ANY)
614 CurrentState = InState;
616 for (
auto &Inst :
Block.Insts) {
617 if (CurrentState != Inst.NeededState) {
618 auto [InsertPt, PhysLiveRegs] =
619 findStateChangeInsertionPoint(
MBB,
Block, &Inst);
620 emitStateChange(Context,
MBB, InsertPt, CurrentState, Inst.NeededState,
622 CurrentState = Inst.NeededState;
626 if (
MBB.succ_empty())
631 if (CurrentState != OutState) {
632 auto [InsertPt, PhysLiveRegs] =
633 findStateChangeInsertionPoint(
MBB,
Block,
Block.Insts.end());
634 emitStateChange(Context,
MBB, InsertPt, CurrentState, OutState,
644 return MBBI !=
MBB.end() ?
MBBI->getDebugLoc() :
MBB.back().getDebugLoc();
654 unsigned Marker,
unsigned CallDestroyOpcode) {
655 auto IsMarker = [&](
auto &
MI) {
return MI.getOpcode() == Marker; };
656 auto MarkerInst = std::find_if(
MBBI,
MBB.end(), IsMarker);
657 if (MarkerInst ==
MBB.end())
660 while (++
I !=
MBB.end()) {
661 if (
I->isCall() ||
I->getOpcode() == CallDestroyOpcode)
664 if (
I !=
MBB.end() &&
I->isCall())
670void MachineSMEABI::collectReachableMarkedCalls(
674 assert(Marker == AArch64::InOutZAUsePseudo ||
675 Marker == AArch64::RequiresZASavePseudo ||
676 Marker == AArch64::RequiresZT0SavePseudo);
677 unsigned CallDestroyOpcode =
TII->getCallFrameDestroyOpcode();
678 if (findMarkedCall(StartMBB, StartInst, Calls, Marker, CallDestroyOpcode))
684 while (!Worklist.
empty()) {
690 if (!findMarkedCall(*
MBB,
MBB->begin(), Calls, Marker, CallDestroyOpcode))
714 StringRef StateName = Marker == AArch64::RequiresZT0SavePseudo ?
"ZT0" :
"ZA";
716 return SaveRemark(
DL,
MBB) << SaveName <<
" of " << StateName
717 <<
" emitted in '" << MF->
getName() <<
"'";
722 collectReachableMarkedCalls(
MBB,
MBBI, CallsRequiringSaves, Marker);
727 R <<
" to '" << CalleeName <<
"'";
728 R <<
" requires " << StateName <<
" save";
733void MachineSMEABI::emitSetupLazySave(EmitContext &Context,
738 emitCallSaveRemarks(
MBB,
MBBI,
DL, AArch64::RequiresZASavePseudo,
739 "SMELazySaveZA",
"lazy save");
752 .
addImm(AArch64SysReg::TPIDR2_EL0)
756PhysRegSave MachineSMEABI::createPhysRegSave(
LiveRegs PhysLiveRegs,
760 PhysRegSave RegSave{PhysLiveRegs};
761 if (PhysLiveRegs & LiveRegs::NZCV) {
764 .
addImm(AArch64SysReg::NZCV)
769 if (PhysLiveRegs & LiveRegs::W0) {
771 ? &AArch64::GPR64RegClass
772 : &AArch64::GPR32RegClass);
774 .
addReg(PhysLiveRegs & LiveRegs::W0_HI ? AArch64::X0 : AArch64::W0);
779void MachineSMEABI::restorePhyRegSave(
const PhysRegSave &RegSave,
783 if (RegSave.StatusFlags != AArch64::NoRegister)
785 .
addImm(AArch64SysReg::NZCV)
786 .
addReg(RegSave.StatusFlags)
789 if (RegSave.X0Save != AArch64::NoRegister)
791 RegSave.PhysLiveRegs & LiveRegs::W0_HI ? AArch64::X0 : AArch64::W0)
798 if (LCImpl == RTLIB::Unsupported)
799 emitError(
"cannot lower SME ABI (SME routines unsupported)");
802 if (CC != ExpectedCC)
803 emitError(
"invalid calling convention for SME routine: '" + ImplName +
"'");
809void MachineSMEABI::emitRestoreLazySave(EmitContext &Context,
818 PhysRegSave RegSave = createPhysRegSave(PhysLiveRegs,
MBB,
MBBI,
DL);
822 .
addImm(AArch64SVCR::SVCRZA)
826 .
addImm(AArch64SysReg::TPIDR2_EL0);
837 RestoreZA, RTLIB::SMEABI_TPIDR2_RESTORE,
841 .
addImm(AArch64SysReg::TPIDR2_EL0)
844 restorePhyRegSave(RegSave,
MBB,
MBBI,
DL);
849 bool ClearTPIDR2,
bool On) {
854 .
addImm(AArch64SysReg::TPIDR2_EL0)
859 .
addImm(AArch64SVCR::SVCRZA)
863void MachineSMEABI::emitAllocateLazySaveBuffer(
876 if (Buffer == AArch64::NoRegister) {
884 "Lazy ZA save is not yet supported on Windows");
906 "TPIDR2 block initialization is not supported on big-endian targets");
931 .
addImm(AArch64SysReg::TPIDR2_EL0);
940 CommitZASave, RTLIB::SMEABI_TPIDR2_SAVE,
948 .
addImm(AArch64SVCR::SVCRZA)
960void MachineSMEABI::emitFullZASaveRestore(EmitContext &Context,
963 LiveRegs PhysLiveRegs,
bool IsSave) {
967 emitCallSaveRemarks(
MBB,
MBBI,
DL, AArch64::RequiresZASavePseudo,
968 "SMEFullZASave",
"full save");
970 PhysRegSave RegSave = createPhysRegSave(PhysLiveRegs,
MBB,
MBBI,
DL);
975 .
addReg(Context.getAgnosticZABufferPtr(*MF));
982 IsSave ? RTLIB::SMEABI_SME_SAVE : RTLIB::SMEABI_SME_RESTORE,
985 restorePhyRegSave(RegSave,
MBB,
MBBI,
DL);
988void MachineSMEABI::emitZT0SaveRestore(EmitContext &Context,
998 emitCallSaveRemarks(
MBB,
MBBI,
DL, AArch64::RequiresZT0SavePseudo,
999 "SMEZT0Save",
"spill");
1018void MachineSMEABI::emitAllocateFullZASaveBuffer(
1026 Register BufferPtr = Context.getAgnosticZABufferPtr(*MF);
1029 PhysRegSave RegSave = createPhysRegSave(PhysLiveRegs,
MBB,
MBBI,
DL);
1036 SMEStateSize, RTLIB::SMEABI_SME_STATE_SIZE,
1056 restorePhyRegSave(RegSave,
MBB,
MBBI,
DL);
1062 constexpr uint8_t to(ZAState To)
const {
1063 static_assert(NUM_ZA_STATE < 16,
"expected ZAState to fit in 4-bits");
1068constexpr FromState transitionFrom(ZAState From) {
return FromState{From}; }
1070void MachineSMEABI::emitStateChange(EmitContext &Context,
1073 ZAState From, ZAState To,
1076 if (From == ZAState::ANY || To == ZAState::ANY)
1081 if (From == ZAState::ENTRY && To == ZAState::OFF)
1086 if (From == ZAState::ENTRY) {
1088 "ENTRY state only valid in entry block");
1089 emitSMEPrologue(
MBB,
MBB.getFirstNonPHI());
1090 if (To == ZAState::ACTIVE)
1096 From = ZAState::ACTIVE;
1102 bool HasZAState = IsAgnosticZA || SMEFnAttrs.
hasZAState();
1104 switch (transitionFrom(From).to(To)) {
1106 case transitionFrom(ZAState::ACTIVE).to(ZAState::ACTIVE_ZT0_SAVED):
1107 emitZT0SaveRestore(Context,
MBB, InsertPt,
true);
1109 case transitionFrom(ZAState::ACTIVE_ZT0_SAVED).to(ZAState::ACTIVE):
1110 emitZT0SaveRestore(Context,
MBB, InsertPt,
false);
1114 case transitionFrom(ZAState::ACTIVE).to(ZAState::LOCAL_SAVED):
1115 case transitionFrom(ZAState::ACTIVE_ZT0_SAVED).to(ZAState::LOCAL_SAVED):
1116 if (HasZT0State && From == ZAState::ACTIVE)
1117 emitZT0SaveRestore(Context,
MBB, InsertPt,
true);
1119 emitZASave(Context,
MBB, InsertPt, PhysLiveRegs);
1123 case transitionFrom(ZAState::ACTIVE).to(ZAState::LOCAL_COMMITTED):
1126 assert(HasZT0State && !HasZAState &&
"Expect to only have ZT0 state.");
1127 emitZT0SaveRestore(Context,
MBB, InsertPt,
true);
1128 emitZAMode(
MBB, InsertPt,
false,
false);
1132 case transitionFrom(ZAState::LOCAL_COMMITTED).to(ZAState::OFF):
1133 case transitionFrom(ZAState::LOCAL_COMMITTED).to(ZAState::LOCAL_SAVED):
1138 case transitionFrom(ZAState::LOCAL_COMMITTED).to(ZAState::ACTIVE):
1139 case transitionFrom(ZAState::LOCAL_COMMITTED).to(ZAState::ACTIVE_ZT0_SAVED):
1140 case transitionFrom(ZAState::LOCAL_SAVED).to(ZAState::ACTIVE):
1141 case transitionFrom(ZAState::LOCAL_SAVED).to(ZAState::ACTIVE_ZT0_SAVED):
1143 emitZARestore(Context,
MBB, InsertPt, PhysLiveRegs);
1145 emitZAMode(
MBB, InsertPt,
false,
true);
1146 if (HasZT0State && To == ZAState::ACTIVE)
1147 emitZT0SaveRestore(Context,
MBB, InsertPt,
false);
1151 case transitionFrom(ZAState::ACTIVE).to(ZAState::OFF):
1152 case transitionFrom(ZAState::ACTIVE_ZT0_SAVED).to(ZAState::OFF):
1153 case transitionFrom(ZAState::LOCAL_SAVED).to(ZAState::OFF):
1155 "Did not expect to turn ZA off in shared/agnostic ZA function");
1156 emitZAMode(
MBB, InsertPt, From == ZAState::LOCAL_SAVED,
1161 dbgs() <<
"Error: Transition from " << getZAStateString(From) <<
" to "
1162 << getZAStateString(To) <<
'\n';
1169static bool canElidePrivateZASetup(
const FunctionInfo &FnInfo) {
1170 for (
const BlockInfo &BlockInfo : FnInfo.Blocks) {
1171 for (
const InstInfo &InstInfo : BlockInfo.Insts) {
1172 if (InstInfo.NeededState == ZAState::ACTIVE ||
1173 InstInfo.NeededState == ZAState::ACTIVE_ZT0_SAVED)
1187 SMEAttrs SMEFnAttrs = AFI->getSMEFnAttrs();
1196 assert(MF.getRegInfo().isSSA() &&
"Expected to be run on SSA form!");
1199 ORE = &getAnalysis<MachineOptimizationRemarkEmitterPass>().getORE();
1200 LLI = &getAnalysis<LibcallLoweringInfoWrapper>().getLibcallLowering(
1201 *MF.getFunction().getParent(), *Subtarget);
1202 TII = Subtarget->getInstrInfo();
1203 TRI = Subtarget->getRegisterInfo();
1204 MRI = &MF.getRegInfo();
1207 getAnalysis<EdgeBundlesWrapperLegacy>().getEdgeBundles();
1209 FunctionInfo FnInfo = collectNeededZAStates(SMEFnAttrs);
1216 EmitContext Context;
1217 insertStateChanges(Context, FnInfo, Bundles, BundleStates);
1219 if (Context.needsSaveBuffer()) {
1220 if (FnInfo.AfterSMEProloguePt) {
1224 emitAllocateZASaveBuffer(Context, *
MBBI->getParent(),
MBBI,
1225 FnInfo.PhysLiveRegsAfterSMEPrologue);
1228 emitAllocateZASaveBuffer(
1230 FnInfo.Blocks[EntryBlock.
getNumber()].PhysLiveRegsAtEntry);
1238 return new MachineSMEABI(OptLevel);
static constexpr unsigned ZERO_ALL_ZA_MASK
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
MachineBasicBlock MachineBasicBlock::iterator DebugLoc DL
MachineBasicBlock MachineBasicBlock::iterator MBBI
static GCRegistry::Add< CoreCLRGC > E("coreclr", "CoreCLR-compatible GC")
const HexagonInstrInfo * TII
This file implements the LivePhysRegs utility for tracking liveness of physical registers.
#define ENTRY(ASMNAME, ENUM)
static DebugLoc getDebugLoc(MachineBasicBlock::instr_iterator FirstMI, MachineBasicBlock::instr_iterator LastMI)
Return the first DebugLoc that has line number information, given a range of instructions.
Register const TargetRegisterInfo * TRI
#define INITIALIZE_PASS(passName, arg, name, cfg, analysis)
This file defines the SmallVector class.
AArch64FunctionInfo - This class is derived from MachineFunctionInfo and contains private AArch64-spe...
Register getEarlyAllocSMESaveBuffer() const
SMEAttrs getSMEFnAttrs() const
bool isTargetWindows() const
bool isLittleEndian() const
Represent the analysis usage information of a pass.
AnalysisUsage & addPreservedID(const void *ID)
AnalysisUsage & addRequired()
LLVM_ABI void setPreservesCFG()
This function should be called by the pass, iff they do not:
Represent a constant reference to an array (0 or more elements consecutively in memory),...
This class represents a function call, abstracting a target machine's calling convention.
ArrayRef< unsigned > getBlocks(unsigned Bundle) const
getBlocks - Return an array of blocks that are connected to Bundle.
unsigned getBundle(unsigned N, bool Out) const
getBundle - Return the ingoing (Out = false) or outgoing (Out = true) bundle number for basic block N
unsigned getNumBundles() const
getNumBundles - Return the total number of bundles in the CFG.
FunctionPass class - This class is used to implement most global optimizations.
LLVMContext & getContext() const
getContext - Return a reference to the LLVMContext associated with this function.
const DebugLoc & getDebugLoc() const
Return the debug location for this node as a DebugLoc.
This is an important class for using LLVM in a threaded context.
LLVM_ABI void emitError(const Instruction *I, const Twine &ErrorStr)
emitError - Emit an error message to the currently installed error handler with optional location inf...
Tracks which library functions to use for a particular subtarget.
LLVM_ABI CallingConv::ID getLibcallImplCallingConv(RTLIB::LibcallImpl Call) const
Get the CallingConv that should be used for the specified libcall.
LLVM_ABI RTLIB::LibcallImpl getLibcallImpl(RTLIB::Libcall Call) const
Return the lowering's selection of implementation call for Call.
A set of register units used to track register liveness.
bool available(MCRegister Reg) const
Returns true if no part of physical register Reg is live.
void addReg(MCRegister Reg)
Adds register units covered by physical register Reg.
LLVM_ABI void stepBackward(const MachineInstr &MI)
Updates liveness when stepping backwards over the instruction MI.
LLVM_ABI void addLiveOuts(const MachineBasicBlock &MBB)
Adds registers living out of block MBB.
MachineInstrBundleIterator< const MachineInstr > const_iterator
int getNumber() const
MachineBasicBlocks are uniquely numbered at the function level, unless they're not in a MachineFuncti...
LLVM_ABI iterator getFirstNonPHI()
Returns a pointer to the first instruction in this block that is not a PHINode instruction.
succ_reverse_iterator succ_rbegin()
MachineInstrBundleIterator< MachineInstr > iterator
succ_reverse_iterator succ_rend()
The MachineFrameInfo class represents an abstract stack frame until prolog/epilog code is inserted.
LLVM_ABI int CreateStackObject(uint64_t Size, Align Alignment, bool isSpillSlot, const AllocaInst *Alloca=nullptr, uint8_t ID=0)
Create a new statically sized stack object, returning a nonnegative identifier to represent it.
LLVM_ABI int CreateSpillStackObject(uint64_t Size, Align Alignment)
Create a new statically sized stack object that represents a spill slot, returning a nonnegative iden...
LLVM_ABI int CreateVariableSizedObject(Align Alignment, const AllocaInst *Alloca)
Notify the MachineFrameInfo object that a variable sized object has been created.
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.
StringRef getName() const
getName - Return the name of the corresponding LLVM function.
MachineFrameInfo & getFrameInfo()
getFrameInfo - Return the frame info object for the current function.
MachineRegisterInfo & getRegInfo()
getRegInfo - Return information about the registers currently in use.
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 MachineInstrBuilder & addExternalSymbol(const char *FnName, unsigned TargetFlags=0) const
const MachineInstrBuilder & addReg(Register RegNo, RegState Flags={}, unsigned SubReg=0) const
Add a new virtual register operand.
const MachineInstrBuilder & addImm(int64_t Val) const
Add a new immediate operand.
const MachineInstrBuilder & addFrameIndex(int Idx) const
const MachineInstrBuilder & addRegMask(const uint32_t *Mask) const
const MachineInstrBuilder & addDef(Register RegNo, RegState Flags={}, unsigned SubReg=0) const
Add a virtual register definition operand.
Representation of each machine instruction.
MachineOperand class - Representation of each machine instruction operand.
const GlobalValue * getGlobal() const
bool isReg() const
isReg - Tests if this is a MO_Register operand.
bool isSymbol() const
isSymbol - Tests if this is a MO_ExternalSymbol operand.
bool isGlobal() const
isGlobal - Tests if this is a MO_GlobalAddress operand.
const char * getSymbolName() const
Register getReg() const
getReg - Returns the register number.
MachineRegisterInfo - Keep track of information for virtual and physical registers,...
LLVM_ABI Register createVirtualRegister(const TargetRegisterClass *RegClass, StringRef Name="")
createVirtualRegister - Create and return a new virtual register in the function with the specified r...
Wrapper class representing virtual and physical registers.
constexpr bool isPhysical() const
Return true if the specified register number is in the physical register namespace.
SMEAttrs is a utility class to parse the SME ACLE attributes on functions.
bool hasAgnosticZAInterface() const
bool hasPrivateZAInterface() const
bool hasSharedZAInterface() const
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.
This class consists of common code factored out of the SmallVector class to reduce code duplication b...
typename SuperClass::const_iterator const_iterator
void append(ItTy in_start, ItTy in_end)
Add the specified range to the end of the SmallVector.
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
Represent a constant reference to a string, i.e.
constexpr bool empty() const
Check if the string is empty.
constexpr const char * data() const
Get a pointer to the start of the string (which may not be null terminated).
TargetRegisterInfo base class - We assume that the target defines a static array of TargetRegisterDes...
Twine - A lightweight data structure for efficiently representing the concatenation of temporary valu...
LLVM_ABI StringRef getName() const
Return a constant reference to the value's name.
const ParentTy * getParent() const
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
static unsigned getArithExtendImm(AArch64_AM::ShiftExtendType ET, unsigned Imm)
getArithExtendImm - Encode the extend type and shift amount for an arithmetic instruction: imm: 3-bit...
unsigned ID
LLVM IR allows to use arbitrary numbers as calling convention identifiers.
@ AArch64_SME_ABI_Support_Routines_PreserveMost_From_X0
Preserve X0-X13, X19-X29, SP, Z0-Z31, P0-P15.
@ AArch64_SME_ABI_Support_Routines_PreserveMost_From_X1
Preserve X1-X15, X19-X29, SP, Z0-Z31, P0-P15.
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.
@ Implicit
Not emitted register (e.g. carry, or temporary result).
@ Define
Register definition.
@ LLVM_MARK_AS_BITMASK_ENUM
FunctionPass * createMachineSMEABIPass(CodeGenOptLevel)
LLVM_ABI char & MachineDominatorsID
MachineDominators - This pass is a machine dominators analysis pass.
LLVM_ABI void reportFatalInternalError(Error Err)
Report a fatal error that indicates a bug in LLVM.
LLVM_ABI char & MachineLoopInfoID
MachineLoopInfo - This pass is a loop analysis pass.
bool any_of(R &&range, UnaryPredicate P)
Provide wrappers to std::any_of which take ranges instead of having to pass begin/end explicitly.
auto reverse(ContainerTy &&C)
LLVM_ABI raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
CodeGenOptLevel
Code generation optimization level.
uint16_t MCPhysReg
An unsigned integer type large enough to represent all physical registers, but not necessarily virtua...
This struct is a compact representation of a valid (non-zero power of two) alignment.
static StringRef getLibcallImplName(RTLIB::LibcallImpl CallImpl)
Get the libcall routine name for the specified libcall implementation.