40#define DEBUG_TYPE "wasm-reg-stackify"
47 return "WebAssembly Register Stackify";
74char WebAssemblyRegStackify::ID = 0;
76 "Reorder instructions to use the WebAssembly value stack",
80 return new WebAssemblyRegStackify(OptLevel);
88 if (!
MI->definesRegister(WebAssembly::VALUE_STACK,
nullptr))
94 if (!
MI->readsRegister(WebAssembly::VALUE_STACK,
nullptr))
106 assert(
MI->getOpcode() == TargetOpcode::IMPLICIT_DEF);
108 const auto *RegClass = MRI.
getRegClass(
MI->getOperand(0).getReg());
109 if (RegClass == &WebAssembly::I32RegClass) {
110 MI->setDesc(
TII->get(WebAssembly::CONST_I32));
112 }
else if (RegClass == &WebAssembly::I64RegClass) {
113 MI->setDesc(
TII->get(WebAssembly::CONST_I64));
115 }
else if (RegClass == &WebAssembly::F32RegClass) {
116 MI->setDesc(
TII->get(WebAssembly::CONST_F32));
120 }
else if (RegClass == &WebAssembly::F64RegClass) {
121 MI->setDesc(
TII->get(WebAssembly::CONST_F64));
125 }
else if (RegClass == &WebAssembly::V128RegClass) {
126 MI->setDesc(
TII->get(WebAssembly::CONST_V128_I64x2));
138 bool &Effects,
bool &StackPointer) {
146 if (!GA->isInterposable())
147 GV = GA->getAliasee();
150 if (!
F->doesNotThrow())
152 if (
F->doesNotAccessMemory())
154 if (
F->onlyReadsMemory()) {
170 bool &Effects,
bool &StackPointer) {
173 if (
MI.isDebugInstr() ||
MI.isPosition())
177 if (
MI.mayLoad() && !
MI.isDereferenceableInvariantLoad())
183 }
else if (
MI.hasOrderedMemoryRef()) {
184 switch (
MI.getOpcode()) {
185 case WebAssembly::DIV_S_I32:
186 case WebAssembly::DIV_S_I64:
187 case WebAssembly::REM_S_I32:
188 case WebAssembly::REM_S_I64:
189 case WebAssembly::DIV_U_I32:
190 case WebAssembly::DIV_U_I64:
191 case WebAssembly::REM_U_I32:
192 case WebAssembly::REM_U_I64:
193 case WebAssembly::I32_TRUNC_S_F32:
194 case WebAssembly::I64_TRUNC_S_F32:
195 case WebAssembly::I32_TRUNC_S_F64:
196 case WebAssembly::I64_TRUNC_S_F64:
197 case WebAssembly::I32_TRUNC_U_F32:
198 case WebAssembly::I64_TRUNC_U_F32:
199 case WebAssembly::I32_TRUNC_U_F64:
200 case WebAssembly::I64_TRUNC_U_F64:
218 if (
MI.hasUnmodeledSideEffects()) {
219 switch (
MI.getOpcode()) {
220 case WebAssembly::DIV_S_I32:
221 case WebAssembly::DIV_S_I64:
222 case WebAssembly::REM_S_I32:
223 case WebAssembly::REM_S_I64:
224 case WebAssembly::DIV_U_I32:
225 case WebAssembly::DIV_U_I64:
226 case WebAssembly::REM_U_I32:
227 case WebAssembly::REM_U_I64:
228 case WebAssembly::I32_TRUNC_S_F32:
229 case WebAssembly::I64_TRUNC_S_F32:
230 case WebAssembly::I32_TRUNC_S_F64:
231 case WebAssembly::I64_TRUNC_S_F64:
232 case WebAssembly::I32_TRUNC_U_F32:
233 case WebAssembly::I64_TRUNC_U_F32:
234 case WebAssembly::I32_TRUNC_U_F64:
235 case WebAssembly::I64_TRUNC_U_F64:
248 if ((
MI.getOpcode() == WebAssembly::GLOBAL_SET_I32 ||
249 MI.getOpcode() == WebAssembly::GLOBAL_SET_I64) &&
250 MI.getOperand(0).isSymbol() &&
251 !strcmp(
MI.getOperand(0).getSymbolName(),
"__stack_pointer"))
254 if (
MI.isCall() &&
MI.getOperand(0).isSymbol() &&
255 !strcmp(
MI.getOperand(0).getSymbolName(),
"__wasm_get_stack_pointer"))
267 return Def.isAsCheapAsAMove() &&
TII->isTriviallyReMaterializable(Def);
281 if (LIS !=
nullptr) {
298 if (MFI.isFrameBaseVirtual() && MFI.getFrameBaseVreg() ==
Reg) {
304 bool NeedsRegForDebug =
333 if (Result.valueIn() == DefVNI) {
334 if (!Result.isKill())
380 for (;
I !=
E; ++
I) {
381 for (
const auto &PriorUse :
I->uses()) {
382 if (&PriorUse ==
Use)
384 if (PriorUse.isReg() && SubsequentDef.getReg() == PriorUse.getReg())
393 for (
auto E =
MBB->end(); NextI !=
E && NextI->isDebugInstr(); ++NextI)
412 if (!MO.isReg() || MO.isUndef())
417 if (MO.isDead() && Insert->definesRegister(
Reg,
nullptr) &&
418 !Insert->readsRegister(
Reg,
nullptr))
421 if (
Reg.isPhysical()) {
424 if (
Reg == WebAssembly::ARGUMENTS)
440 bool Read =
false,
Write =
false, Effects =
false, StackPointer =
false;
445 bool HasMutableRegisters = !MutableRegisters.
empty();
446 if (!
Read && !
Write && !Effects && !StackPointer && !HasMutableRegisters)
451 for (--
I;
I !=
D; --
I) {
452 bool InterveningRead =
false;
453 bool InterveningWrite =
false;
454 bool InterveningEffects =
false;
455 bool InterveningStackPointer =
false;
456 query(*
I, InterveningRead, InterveningWrite, InterveningEffects,
457 InterveningStackPointer);
458 if (Effects && InterveningEffects)
460 if (
Read && InterveningWrite)
462 if (
Write && (InterveningRead || InterveningWrite))
464 if (StackPointer && InterveningStackPointer)
467 for (
unsigned Reg : MutableRegisters)
469 if (MO.isReg() && MO.isDef() && MO.getReg() ==
Reg)
495 if (UseVNI != OneUseVNI)
498 if (UseInst == OneUseInst) {
505 while (!MDT.
dominates(OneUseInst, UseInst)) {
522 if (NewUseInst == OneUseInst) {
523 if (&OneUse > &NewUse)
527 UseInst = NewUseInst;
536 if (RC == &WebAssembly::I32RegClass)
537 return WebAssembly::TEE_I32;
538 if (RC == &WebAssembly::I64RegClass)
539 return WebAssembly::TEE_I64;
540 if (RC == &WebAssembly::F32RegClass)
541 return WebAssembly::TEE_F32;
542 if (RC == &WebAssembly::F64RegClass)
543 return WebAssembly::TEE_F64;
544 if (RC == &WebAssembly::V128RegClass)
545 return WebAssembly::TEE_V128;
546 if (RC == &WebAssembly::EXTERNREFRegClass)
547 return WebAssembly::TEE_EXTERNREF;
548 if (RC == &WebAssembly::FUNCREFRegClass)
549 return WebAssembly::TEE_FUNCREF;
550 if (RC == &WebAssembly::EXNREFRegClass)
551 return WebAssembly::TEE_EXNREF;
588 if (LIS !=
nullptr) {
608 for (
auto *
I =
MI->getPrevNode();
I;
I =
I->getPrevNode())
609 if (!
I->isDebugInstr())
736class TreeWalkerState {
738 using mop_reverse_iterator = std::reverse_iterator<mop_iterator>;
739 using RangeTy = iterator_range<mop_reverse_iterator>;
743 explicit TreeWalkerState(MachineInstr *Insert) {
744 const iterator_range<mop_iterator> &
Range =
Insert->explicit_uses();
749 bool done()
const {
return Worklist.empty(); }
751 MachineOperand &pop() {
752 RangeTy &
Range = Worklist.back();
753 MachineOperand &
Op = *
Range.begin();
757 assert((Worklist.empty() || !Worklist.back().empty()) &&
758 "Empty ranges shouldn't remain in the worklist");
763 void pushOperands(MachineInstr *Instr) {
764 const iterator_range<mop_iterator> &
Range(
Instr->explicit_uses());
771 void resetTopOperands(MachineInstr *Instr) {
772 assert(hasRemainingOperands(Instr) &&
773 "Reseting operands should only be done when the instruction has "
774 "an operand still on the stack");
780 bool hasRemainingOperands(
const MachineInstr *Instr)
const {
781 if (Worklist.empty())
783 const RangeTy &
Range = Worklist.back();
793 bool isOnStack(
unsigned Reg)
const {
794 for (
const RangeTy &
Range : Worklist)
795 for (
const MachineOperand &MO :
Range)
796 if (MO.isReg() && MO.getReg() ==
Reg)
804class CommutingState {
810 bool TentativelyCommuting =
false;
811 bool Declined =
false;
815 unsigned Operand0, Operand1;
821 void maybeCommute(MachineInstr *Insert, TreeWalkerState &TreeWalker,
822 const WebAssemblyInstrInfo *
TII) {
823 if (TentativelyCommuting) {
825 "Don't decline commuting until you've finished trying it");
827 TII->commuteInstruction(*Insert,
false, Operand0, Operand1);
828 TentativelyCommuting =
false;
830 }
else if (!Declined && TreeWalker.hasRemainingOperands(Insert)) {
833 if (
TII->findCommutedOpIndices(*Insert, Operand0, Operand1)) {
835 TII->commuteInstruction(*Insert,
false, Operand0, Operand1);
836 TreeWalker.resetTopOperands(Insert);
837 TentativelyCommuting =
true;
846 TentativelyCommuting =
false;
852bool WebAssemblyRegStackify::runOnMachineFunction(
MachineFunction &MF) {
853 LLVM_DEBUG(
dbgs() <<
"********** Register Stackifying **********\n"
854 "********** Function: "
859 WebAssemblyFunctionInfo &MFI = *MF.
getInfo<WebAssemblyFunctionInfo>();
860 const auto *
TII = MF.
getSubtarget<WebAssemblySubtarget>().getInstrInfo();
861 MachineDominatorTree *MDT =
nullptr;
862 LiveIntervals *LIS =
nullptr;
864 MDT = &getAnalysis<MachineDominatorTreeWrapperPass>().getDomTree();
865 LIS = &getAnalysis<LiveIntervalsWrapperPass>().getLIS();
871 for (MachineBasicBlock &
MBB : MF) {
875 MachineInstr *
Insert = &*MII;
878 if (
Insert->isInlineAsm())
882 if (
Insert->isDebugValue())
891 CommutingState Commuting;
892 TreeWalkerState TreeWalker(Insert);
893 while (!TreeWalker.done()) {
894 MachineOperand &
Use = TreeWalker.pop();
901 assert(
Use.isUse() &&
"explicit_uses() should only iterate over uses");
903 "explicit_uses() should only iterate over explicit operands");
922 MachineOperand *
Def =
933 bool CanMove = SameBlock &&
935 !TreeWalker.isOnStack(
Reg);
945 "Stackifying away frame base in unoptimized code not expected");
950 *LIS, MFI, MRI,
TII);
959 if (!CanMove && SameBlock)
960 Commuting.maybeCommute(Insert, TreeWalker,
TII);
968 auto *SubsequentDef =
Insert->defs().begin();
969 auto *SubsequentUse = &
Use;
970 while (SubsequentDef !=
Insert->defs().end() &&
971 SubsequentUse !=
Use.getParent()->uses().end()) {
972 if (!SubsequentDef->isReg() || !SubsequentUse->isReg())
974 Register DefReg = SubsequentDef->getReg();
988 if (
Insert->getOpcode() == TargetOpcode::IMPLICIT_DEF)
994 TreeWalker.pushOperands(Insert);
999 if (Insert != &*MII) {
1010 MF.getRegInfo().addLiveIn(WebAssembly::VALUE_STACK);
1011 for (MachineBasicBlock &
MBB : MF)
1017 SmallVector<unsigned, 0>
Stack;
1018 for (MachineBasicBlock &
MBB : MF) {
1019 for (MachineInstr &
MI :
MBB) {
1020 if (
MI.isDebugInstr())
1022 for (MachineOperand &MO :
reverse(
MI.explicit_uses())) {
1028 "Register stack pop should be paired with a push");
1030 for (MachineOperand &MO :
MI.defs()) {
1035 Stack.push_back(MO.getReg());
1041 "Register stack pushes and pops should be balanced");
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static GCRegistry::Add< StatepointGC > D("statepoint-example", "an example strategy for statepoint")
static GCRegistry::Add< CoreCLRGC > E("coreclr", "CoreCLR-compatible GC")
static Register UseReg(const MachineOperand &MO)
const HexagonInstrInfo * TII
Promote Memory to Register
ConstantRange Range(APInt(BitWidth, Low), APInt(BitWidth, High))
#define INITIALIZE_PASS(passName, arg, name, cfg, analysis)
This file contains the declaration of the WebAssembly-specific manager for DebugValues associated wit...
This file provides WebAssembly-specific target descriptions.
This file declares WebAssembly-specific per-machine-function information.
static bool isSafeToMove(const MachineOperand *Def, const MachineOperand *Use, const MachineInstr *Insert, const WebAssemblyFunctionInfo &MFI, const MachineRegisterInfo &MRI, bool Optimize)
static unsigned getTeeOpcode(const TargetRegisterClass *RC)
Get the appropriate tee opcode for the given register class.
static MachineInstr * rematerializeCheapDef(unsigned Reg, MachineOperand &Op, MachineInstr &Def, MachineBasicBlock::instr_iterator Insert, LiveIntervals &LIS, WebAssemblyFunctionInfo &MFI, MachineRegisterInfo &MRI, const WebAssemblyInstrInfo *TII)
A trivially cloneable instruction; clone it and nest the new copy with the current instruction.
static bool hasSingleUse(unsigned Reg, MachineRegisterInfo &MRI, const MachineFunction &MF, bool Optimize, MachineInstr *Def, LiveIntervals *LIS)
static void imposeStackOrdering(MachineInstr *MI)
static MachineInstr * moveForSingleUse(unsigned Reg, MachineOperand &Op, MachineInstr *Def, MachineBasicBlock &MBB, MachineInstr *Insert, LiveIntervals *LIS, WebAssemblyFunctionInfo &MFI, MachineRegisterInfo &MRI)
A single-use def in the same block with no intervening memory or register dependencies; move the def ...
static void query(const MachineInstr &MI, bool &Read, bool &Write, bool &Effects, bool &StackPointer)
static void shrinkToUses(LiveInterval &LI, LiveIntervals &LIS)
static void convertImplicitDefToConstZero(MachineInstr *MI, MachineRegisterInfo &MRI, const TargetInstrInfo *TII, MachineFunction &MF)
static MachineInstr * getPrevNonDebugInst(MachineInstr *MI)
static bool shouldRematerialize(const MachineInstr &Def, const WebAssemblyInstrInfo *TII)
static MachineInstr * moveAndTeeForMultiUse(unsigned Reg, MachineOperand &Op, MachineInstr *Def, MachineBasicBlock &MBB, MachineInstr *Insert, LiveIntervals &LIS, WebAssemblyFunctionInfo &MFI, MachineRegisterInfo &MRI, const WebAssemblyInstrInfo *TII)
A multiple-use def in the same block with no intervening memory or register dependencies; move the de...
static bool oneUseDominatesOtherUses(unsigned Reg, const MachineOperand &OneUse, const MachineBasicBlock &MBB, const MachineRegisterInfo &MRI, const MachineDominatorTree &MDT, LiveIntervals &LIS, WebAssemblyFunctionInfo &MFI)
Test whether OneUse, a use of Reg, dominates all of Reg's other uses.
static void queryCallee(const MachineInstr &MI, bool &Read, bool &Write, bool &Effects, bool &StackPointer)
This file declares the WebAssembly-specific subclass of TargetSubtarget.
This file contains the declaration of the WebAssembly-specific utility functions.
This file contains the entry points for global functions defined in the LLVM WebAssembly back-end.
Represent the analysis usage information of a pass.
AnalysisUsage & addPreservedID(const void *ID)
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:
This is an important base class in LLVM.
static LLVM_ABI Constant * getNullValue(Type *Ty)
Constructor to create a '0' constant of arbitrary type.
FunctionPass class - This class is used to implement most global optimizations.
DISubprogram * getSubprogram() const
Get the attached subprogram.
LLVMContext & getContext() const
getContext - Return a reference to the LLVMContext associated with this function.
LiveInterval - This class represents the liveness of a register, or stack slot.
MachineInstr * getInstructionFromIndex(SlotIndex index) const
Returns the instruction associated with the given index.
SlotIndex InsertMachineInstrInMaps(MachineInstr &MI)
LLVM_ABI void handleMove(MachineInstr &MI, bool UpdateFlags=false)
Call this method to notify LiveIntervals that instruction MI has been moved within a basic block.
SlotIndex getInstructionIndex(const MachineInstr &Instr) const
Returns the base index of the given instruction.
void RemoveMachineInstrFromMaps(MachineInstr &MI)
LiveInterval & getInterval(Register Reg)
void removeInterval(Register Reg)
Interval removal.
LLVM_ABI bool shrinkToUses(LiveInterval *li, SmallVectorImpl< MachineInstr * > *dead=nullptr)
After removing some uses of a register, shrink its live range to just the remaining uses.
LLVM_ABI void removePhysRegDefAt(MCRegister Reg, SlotIndex Pos)
Remove value numbers and related live segments starting at position Pos that are part of any liverang...
LLVM_ABI void splitSeparateComponents(LiveInterval &LI, SmallVectorImpl< LiveInterval * > &SplitLIs)
Split separate components in LiveInterval LI into separate intervals.
LiveInterval & createAndComputeVirtRegInterval(Register Reg)
Segments::iterator iterator
bool liveAt(SlotIndex index) const
LiveQueryResult Query(SlotIndex Idx) const
Query Liveness at Idx.
VNInfo * getVNInfoBefore(SlotIndex Idx) const
getVNInfoBefore - Return the VNInfo that is live up to but not necessarily including Idx,...
iterator FindSegmentContaining(SlotIndex Idx)
Return an iterator to the segment that contains the specified index, or end() if there is none.
LLVM_ABI void removeSegment(SlotIndex Start, SlotIndex End, bool RemoveDeadValNo=false)
Remove the specified interval from this live range.
VNInfo * getVNInfoAt(SlotIndex Idx) const
getVNInfoAt - Return the VNInfo that is live at Idx, or NULL.
unsigned getNumDefs() const
Return the number of MachineOperands that are register definitions.
static MCRegister from(unsigned Val)
Check the provided unsigned value is a valid MCRegister.
MachineInstrBundleIterator< const MachineInstr > const_iterator
Instructions::iterator instr_iterator
void addLiveIn(MCRegister PhysReg, LaneBitmask LaneMask=LaneBitmask::getAll())
Adds the specified register as a live in.
reverse_iterator rbegin()
MachineInstrBundleIterator< MachineInstr > iterator
Analysis pass which computes a MachineDominatorTree.
DominatorTree Class - Concrete subclass of DominatorTreeBase that is used to compute a normal dominat...
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.
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.
Ty * getInfo()
getInfo - Keep track of various per-function pieces of information for backends that would like to do...
const MachineInstrBuilder & addReg(Register RegNo, RegState Flags={}, unsigned SubReg=0) const
Add a new virtual register operand.
reverse_iterator getReverse() const
Get a reverse iterator to the same node.
Representation of each machine instruction.
mop_range defs()
Returns all explicit operands that are register definitions.
unsigned getOpcode() const
Returns the opcode of this MachineInstr.
const MachineBasicBlock * getParent() const
const MCInstrDesc & getDesc() const
Returns the target instruction descriptor of this MachineInstr.
MachineOperand * mop_iterator
iterator/begin/end - Iterate over all operands of a machine instruction.
LLVM_ABI void dump() const
const MachineOperand & getOperand(unsigned i) const
MachineOperand * findRegisterDefOperand(Register Reg, const TargetRegisterInfo *TRI, bool isDead=false, bool Overlap=false)
Wrapper for findRegisterDefOperandIdx, it returns a pointer to the MachineOperand rather than an inde...
MachineOperand class - Representation of each machine instruction operand.
const GlobalValue * getGlobal() const
static MachineOperand CreateFPImm(const ConstantFP *CFP)
bool isReg() const
isReg - Tests if this is a MO_Register operand.
MachineInstr * getParent()
getParent - Return the instruction that this operand belongs to.
static MachineOperand CreateImm(int64_t Val)
bool isGlobal() const
isGlobal - Tests if this is a MO_GlobalAddress 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)
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
const TargetRegisterClass * getRegClass(Register Reg) const
Return the register class of the specified virtual register.
iterator_range< use_nodbg_iterator > use_nodbg_operands(Register Reg) 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...
bool hasOneUse(Register RegNo) const
hasOneUse - Return true if there is exactly one instruction using the specified register.
bool hasOneDef(Register RegNo) const
Return true if there is exactly one operand defining the specified register.
bool use_empty(Register RegNo) const
use_empty - Return true if there are no instructions using the specified register.
LLVM_ABI bool isPhysRegModified(MCRegister PhysReg, bool SkipNoReturnDef=false) const
Return true if the specified register is modified in this function.
LLVM_ABI MachineInstr * getUniqueVRegDef(Register Reg) const
getUniqueVRegDef - Return the unique machine instr that defines the specified virtual register or nul...
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.
SlotIndex - An opaque wrapper around machine indexes.
SlotIndex getDeadSlot() const
Returns the dead def kill slot for the current instruction.
SlotIndex getRegSlot(bool EC=false) const
Returns the register use/def slot in the current instruction for a normal or early-clobber def.
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.
TargetInstrInfo - Interface to description of machine instruction set.
static const unsigned CommuteAnyOperandIndex
static LLVM_ABI Type * getDoubleTy(LLVMContext &C)
static LLVM_ABI Type * getFloatTy(LLVMContext &C)
A Use represents the edge between a Value definition and its users.
VNInfo - Value Number Information.
SlotIndex def
The index of the defining instruction.
void updateReg(Register Reg)
void cloneSink(MachineInstr *Insert, Register NewReg=Register(), bool CloneDef=true) const
void sink(MachineInstr *Insert)
This class is derived from MachineFunctionInfo and contains private WebAssembly-specific information ...
void stackifyVReg(MachineRegisterInfo &MRI, Register VReg)
unsigned getFrameBaseVreg() const
bool isVRegStackified(Register VReg) const
void clearFrameBaseVreg()
bool isFrameBaseVirtual() const
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
unsigned ID
LLVM IR allows to use arbitrary numbers as calling convention identifiers.
bool isArgument(unsigned Opc)
const MachineOperand & getCalleeOp(const MachineInstr &MI)
Returns the operand number of a callee, assuming the argument is a call instruction.
bool isCatch(unsigned Opc)
NodeAddr< DefNode * > Def
NodeAddr< InstrNode * > Instr
NodeAddr< UseNode * > Use
This is an optimization pass for GlobalISel generic memory operations.
auto drop_begin(T &&RangeOrContainer, size_t N=1)
Return a range covering RangeOrContainer with the first N elements excluded.
void dump(const SparseBitVector< ElementSize > &LHS, raw_ostream &out)
MachineInstrBuilder BuildMI(MachineFunction &MF, const MIMetadata &MIMD, const MCInstrDesc &MCID)
Builder interface. Specify how to create the initial instruction itself.
@ Define
Register definition.
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
FunctionPass * createWebAssemblyRegStackify(CodeGenOptLevel OptLevel)
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.
class LLVM_GSL_OWNER SmallVector
Forward declaration of SmallVector so that calculateSmallVectorDefaultInlinedElements can reference s...
DWARFExpression::Operation Op
LLVM_ABI char & LiveVariablesID
LiveVariables pass - This pass computes the set of blocks in which each variable is life and sets mac...
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
MachineInstr * getVRegDef(MachineRegisterInfo &MRI, Register Reg)
constexpr RegState getUndefRegState(bool B)