35#define DEBUG_TYPE "M68k-instr-info"
37#define GET_INSTRINFO_CTOR_DTOR
38#include "M68kGenInstrInfo.inc"
41void M68kInstrInfo::anchor() {}
46 Subtarget(STI), RI(STI) {}
87 bool AllowModify)
const {
90 std::pair<MachineBasicBlock::reverse_iterator, MachineBasicBlock *>{
94 std::vector<std::reference_wrapper<llvm::MachineInstr>> EraseList;
96 std::for_each(EraseList.begin(), EraseList.end(),
97 [](
auto &ref) { ref.get().eraseFromParent(); });
102 for (
auto iter =
MBB.
rbegin(); iter !=
MBB.
rend(); iter = std::next(iter)) {
104 unsigned Opcode = iter->getOpcode();
106 if (iter->isDebugInstr())
111 if (!isUnpredicatedTerminator(*iter))
116 if (!iter->isBranch())
120 if (Opcode == M68k::BRA8 || Opcode == M68k::BRA16) {
121 if (!iter->getOperand(0).isMBB())
123 UncondBranch = {iter, iter->getOperand(0).getMBB()};
126 TBB = UncondBranch.second;
141 UncondBranch = {
MBB.
rend(),
nullptr};
162 if (!iter->getOperand(0).isMBB())
174 if (UncondBranch.first !=
MBB.
rend()) {
176 assert(std::next(UncondBranch.first) == iter &&
"Wrong block layout.");
201 .
addMBB(UncondBranch.second);
203 EraseList.push_back(*iter);
204 EraseList.push_back(*UncondBranch.first);
206 TBB = UncondBranch.second;
212 TBB = CondBranchTarget;
213 FBB = UncondBranch.second;
217 UncondBranch = {
MBB.
rend(),
nullptr};
221 TBB = CondBranchTarget;
236 if (!iter->getOperand(0).isMBB())
238 auto NewTBB = iter->getOperand(0).getMBB();
239 if (OldBranchCode == BranchCode &&
TBB == NewTBB)
253 bool AllowModify)
const {
258 int *BytesRemoved)
const {
259 assert(!BytesRemoved &&
"code size not handled");
266 if (
I->isDebugValue())
268 if (
I->getOpcode() != M68k::BRA8 &&
272 I->eraseFromParent();
284 assert(
TBB &&
"InsertBranch must not be told to insert a fallthrough");
286 "M68k branch conditions have one component!");
287 assert(!BytesAdded &&
"code size not handled");
291 assert(!FBB &&
"Unconditional branch with multiple successors!");
297 bool FallThru = FBB ==
nullptr;
316 if (
From == MVT::i8) {
319 if (To == MVT::i32) {
320 R =
RI.getSubReg(Reg, M68k::MxSubRegIndex16Lo);
321 assert(R &&
"No viable SUB register available");
354 bool IsAddressReg =
false;
356 const auto *DR32 =
RI.getRegClass(M68k::DR32RegClassID);
357 const auto *AR32 =
RI.getRegClass(M68k::AR32RegClassID);
358 const auto *AR16 =
RI.getRegClass(M68k::AR16RegClassID);
360 if (AR16->contains(Reg) || AR32->contains(Reg))
365 if (MVTSize == MVT::i8 || (!IsAddressReg && Imm >= -128 && Imm <= 127)) {
371 assert(SReg &&
"No viable MEGA register available");
377 MIB->
setDesc(
get(MVTSize == MVT::i16 ? M68k::MOV16ri : M68k::MOV32ri));
385 unsigned Move = MVTDst == MVT::i16 ? M68k::MOV16rr : M68k::MOV32rr;
389 assert(Dst != Src &&
"You cannot use the same Regs with MOVX_RR");
393 const auto *RCDst =
TRI.getMaximalPhysRegClass(Dst, MVTDst);
394 const auto *RCSrc =
TRI.getMaximalPhysRegClass(Src, MVTSrc);
396 assert(RCDst && RCSrc &&
"Wrong use of MOVX_RR");
397 assert(RCDst != RCSrc &&
"You cannot use the same Reg Classes with MOVX_RR");
402 assert(SSrc &&
"No viable MEGA register available");
423 MVT MVTDst,
MVT MVTSrc)
const {
428 if (MVTDst == MVT::i16)
429 Move = M68k::MOV16rr;
431 Move = M68k::MOV32rr;
436 assert(Dst != Src &&
"You cannot use the same Regs with MOVSX_RR");
440 const auto *RCDst =
TRI.getMaximalPhysRegClass(Dst, MVTDst);
441 const auto *RCSrc =
TRI.getMaximalPhysRegClass(Src, MVTSrc);
443 assert(RCDst && RCSrc &&
"Wrong use of MOVSX_RR");
444 assert(RCDst != RCSrc &&
"You cannot use the same Reg Classes with MOVSX_RR");
449 assert(SSrc &&
"No viable MEGA register available");
485 RI.getSubReg(Dst, MVTSrc == MVT::i8 ? M68k::MxSubRegIndex8Lo
486 : M68k::MxSubRegIndex16Lo);
487 assert(SubDst &&
"No viable SUB register available");
537 Opd.getReg(), M68k::MxSubRegIndex8Lo, &M68k::DR16RegClass));
545 auto XR32 =
RI.getRegClass(M68k::XR32RegClassID);
562 if (!XR32->contains(Reg)) {
564 assert(Reg &&
"Has not meaningful MEGA register");
598 assert(
Desc.getNumOperands() == 3 &&
"Expected two-addr instruction.");
613 switch (
MI.getOpcode()) {
628 case M68k::SETCS_C8d:
630 case M68k::SETCS_C16d:
632 case M68k::SETCS_C32d:
646 const unsigned NameIndices = M68kInstrNameIndices[
MI->getOpcode()];
647 StringRef InstrName(&M68kInstrNameData[NameIndices]);
653 return Regex(
"[A-Z]+(8|16|32)k[a-z](_TC)?$").
match(InstrName);
657 if (OperandNo ==
MI->getNumExplicitOperands() - 1)
658 return Regex(
"[A-Z]+(8|16|32)[a-z]k(_TC)?$").
match(InstrName);
670 if (M68k::XR32RegClass.
contains(DstReg, SrcReg))
672 else if (M68k::XR16RegClass.
contains(DstReg, SrcReg))
674 else if (M68k::DR8RegClass.
contains(DstReg, SrcReg))
690 if (M68k::DR8RegClass.
contains(SrcReg)) {
691 if (M68k::XR16RegClass.
contains(DstReg))
692 Opc = M68k::MOVXd16d8;
693 else if (M68k::XR32RegClass.
contains(DstReg))
694 Opc = M68k::MOVXd32d8;
695 }
else if (M68k::XR16RegClass.
contains(SrcReg) &&
696 M68k::XR32RegClass.
contains(DstReg))
697 Opc = M68k::MOVXd32d16;
705 bool FromCCR = SrcReg == M68k::CCR;
706 bool FromSR = SrcReg == M68k::SR;
707 bool ToCCR = DstReg == M68k::CCR;
708 bool ToSR = DstReg == M68k::SR;
712 "Need DR8 register to copy CCR");
716 "Need DR8 register to copy CCR");
718 }
else if (FromSR || ToSR)
728 <<
RI.getName(DstReg) <<
'\n');
736 switch (
TRI->getRegSizeInBits(*RC)) {
740 if (M68k::DR8RegClass.hasSubClassEq(RC))
741 return load ? M68k::MOV8dp : M68k::MOV8pd;
742 if (M68k::CCRCRegClass.hasSubClassEq(RC))
743 return load ? M68k::MOV16cp : M68k::MOV16pc;
747 assert(M68k::XR16RegClass.hasSubClassEq(RC) &&
"Unknown 2-byte regclass");
748 return load ? M68k::MOVM16mp_P : M68k::MOVM16pm_P;
750 assert(M68k::XR32RegClass.hasSubClassEq(RC) &&
"Unknown 4-byte regclass");
751 return load ? M68k::MOVM32mp_P : M68k::MOVM32pm_P;
769 unsigned SubIdx,
unsigned &
Size,
784 "Stack slot is too small to store");
802 "Stack slot is too small to load");
818 if (GlobalBaseReg != 0)
819 return GlobalBaseReg;
833 return GlobalBaseReg;
836std::pair<unsigned, unsigned>
838 return std::make_pair(TF, 0u);
843 using namespace M68kII;
844 static const std::pair<unsigned, const char *> TargetFlags[] = {
845 {MO_ABSOLUTE_ADDRESS,
"m68k-absolute"},
846 {MO_PC_RELATIVE_ADDRESS,
"m68k-pcrel"},
847 {MO_GOT,
"m68k-got"},
848 {MO_GOTOFF,
"m68k-gotoff"},
849 {MO_GOTPCREL,
"m68k-gotpcrel"},
850 {MO_PLT,
"m68k-plt"},
851 {MO_TLSGD,
"m68k-tlsgd"},
852 {MO_TLSLD,
"m68k-tlsld"},
853 {MO_TLSLDM,
"m68k-tlsldm"},
854 {MO_TLSIE,
"m68k-tlsie"},
855 {MO_TLSLE,
"m68k-tlsle"}};
860#define DEBUG_TYPE "m68k-create-global-base-reg"
862#define PASS_NAME "M68k PIC Global Base Reg Initialization"
874 unsigned GlobalBaseReg = MxFI->getGlobalBaseReg();
877 if (GlobalBaseReg == 0)
898char M68kGlobalBaseReg::ID = 0;
904 return new M68kGlobalBaseReg();
AMDGPU Mark last scratch load
MachineBasicBlock MachineBasicBlock::iterator DebugLoc DL
MachineBasicBlock MachineBasicBlock::iterator MBBI
BlockVerifier::State From
const HexagonInstrInfo * TII
This file implements the LivePhysRegs utility for tracking liveness of physical registers.
This file exposes functions that may be used with BuildMI from the MachineInstrBuilder....
static M68k::CondCode getCondFromBranchOpc(unsigned BrOpc)
static bool Expand2AddrUndef(MachineInstrBuilder &MIB, const MCInstrDesc &Desc)
Expand a single-def pseudo instruction to a two-addr instruction with two undef reads of the register...
This file contains the M68k implementation of the TargetInstrInfo class.
This file contains the declarations for the code emitter which are useful outside of the emitter itse...
This file declares the M68k specific subclass of MachineFunctionInfo.
This file declares the M68k specific subclass of TargetMachine.
unsigned const TargetRegisterInfo * TRI
#define INITIALIZE_PASS(passName, arg, name, cfg, analysis)
const SmallVectorImpl< MachineOperand > MachineBasicBlock * TBB
const SmallVectorImpl< MachineOperand > & Cond
assert(ImpDefSCC.getReg()==AMDGPU::SCC &&ImpDefSCC.isDef())
This file defines the make_scope_exit function, which executes user-defined cleanup logic at scope ex...
static SPCC::CondCodes GetOppositeBranchCondition(SPCC::CondCodes CC)
static bool contains(SmallPtrSetImpl< ConstantExpr * > &Cache, ConstantExpr *Expr, Constant *C)
static unsigned getStoreRegOpcode(Register SrcReg, const TargetRegisterClass *RC, bool IsStackAligned, const X86Subtarget &STI)
static unsigned getLoadRegOpcode(Register DestReg, const TargetRegisterClass *RC, bool IsStackAligned, const X86Subtarget &STI)
static unsigned getLoadStoreRegOpcode(Register Reg, const TargetRegisterClass *RC, bool IsStackAligned, const X86Subtarget &STI, bool Load)
static unsigned GetCondBranchFromCond(XCore::CondCode CC)
GetCondBranchFromCond - Return the Branch instruction opcode that matches the cc.
Represent the analysis usage information of a pass.
void setPreservesCFG()
This function should be called by the pass, iff they do not:
ArrayRef - Represent a constant reference to an array (0 or more elements consecutively in memory),...
FunctionPass class - This class is used to implement most global optimizations.
unsigned getGlobalBaseReg(MachineFunction *MF) const
Return a virtual register initialized with the global base register value.
const M68kSubtarget & Subtarget
bool ExpandMOVI(MachineInstrBuilder &MIB, MVT MVTSize) const
Move immediate to register.
bool ExpandMOVSZX_RR(MachineInstrBuilder &MIB, bool IsSigned, MVT MVTDst, MVT MVTSrc) const
Move from register and extend.
const M68kRegisterInfo & getRegisterInfo() const
TargetInstrInfo is a superset of MRegister info.
const M68kRegisterInfo RI
bool analyzeBranch(MachineBasicBlock &MBB, MachineBasicBlock *&TBB, MachineBasicBlock *&FBB, SmallVectorImpl< MachineOperand > &Cond, bool AllowModify) const override
ArrayRef< std::pair< unsigned, const char * > > getSerializableDirectMachineOperandTargetFlags() const override
bool expandPostRAPseudo(MachineInstr &MI) const override
unsigned insertBranch(MachineBasicBlock &MBB, MachineBasicBlock *TBB, MachineBasicBlock *FBB, ArrayRef< MachineOperand > Cond, const DebugLoc &DL, int *BytesAdded=nullptr) const override
void loadRegFromStackSlot(MachineBasicBlock &MBB, MachineBasicBlock::iterator MI, Register DestReg, int FrameIndex, const TargetRegisterClass *RC, const TargetRegisterInfo *TRI, Register VReg) const override
void storeRegToStackSlot(MachineBasicBlock &MBB, MachineBasicBlock::iterator MI, Register SrcReg, bool IsKill, int FrameIndex, const TargetRegisterClass *RC, const TargetRegisterInfo *TRI, Register VReg) const override
std::pair< unsigned, unsigned > decomposeMachineOperandsTargetFlags(unsigned TF) const override
bool AnalyzeBranchImpl(MachineBasicBlock &MBB, MachineBasicBlock *&TBB, MachineBasicBlock *&FBB, SmallVectorImpl< MachineOperand > &Cond, bool AllowModify) const
bool isPCRelRegisterOperandLegal(const MachineOperand &MO) const override
bool ExpandMOVX_RR(MachineInstrBuilder &MIB, MVT MVTDst, MVT MVTSrc) const
Move across register classes without extension.
bool ExpandMOVEM(MachineInstrBuilder &MIB, const MCInstrDesc &Desc, bool IsRM) const
Expand all MOVEM pseudos into real MOVEMs.
unsigned removeBranch(MachineBasicBlock &MBB, int *BytesRemoved=nullptr) const override
bool ExpandPUSH_POP(MachineInstrBuilder &MIB, const MCInstrDesc &Desc, bool IsPush) const
Push/Pop to/from stack.
M68kInstrInfo(const M68kSubtarget &STI)
void AddZExt(MachineBasicBlock &MBB, MachineBasicBlock::iterator I, DebugLoc DL, unsigned Reg, MVT From, MVT To) const
Add appropriate ZExt nodes.
void copyPhysReg(MachineBasicBlock &MBB, MachineBasicBlock::iterator MI, const DebugLoc &DL, MCRegister DestReg, MCRegister SrcReg, bool KillSrc) const override
bool ExpandCCR(MachineInstrBuilder &MIB, bool IsToCCR) const
Moves to/from CCR.
bool ExpandMOVSZX_RM(MachineInstrBuilder &MIB, bool IsSigned, const MCInstrDesc &Desc, MVT MVTDst, MVT MVTSrc) const
Move from memory and extend.
bool getStackSlotRange(const TargetRegisterClass *RC, unsigned SubIdx, unsigned &Size, unsigned &Offset, const MachineFunction &MF) const override
void AddSExt(MachineBasicBlock &MBB, MachineBasicBlock::iterator I, DebugLoc DL, unsigned Reg, MVT From, MVT To) const
Add appropriate SExt nodes.
void setGlobalBaseReg(unsigned Reg)
unsigned getGlobalBaseReg() const
unsigned getMatchingMegaReg(unsigned Reg, const TargetRegisterClass *RC) const
Return a mega-register of the specified register Reg so its sub-register of index SubIdx is Reg,...
int getSpillRegisterOrder(unsigned Reg) const
Return spill order index of a register, if there is none then trap.
unsigned getStackRegister() const
const M68kInstrInfo * getInstrInfo() const override
Describe properties that are true of each instruction in the target description file.
Wrapper class representing physical registers. Should be passed by value.
instr_iterator insert(instr_iterator I, MachineInstr *M)
Insert MI into the instruction list before I, possibly inside a bundle.
void push_back(MachineInstr *MI)
DebugLoc rfindDebugLoc(reverse_instr_iterator MBBI)
Has exact same behavior as findDebugLoc (it also searches towards the end of this MBB) except that th...
DebugLoc findDebugLoc(instr_iterator MBBI)
Find the next valid DebugLoc starting at MBBI, skipping any debug instructions.
bool isLayoutSuccessor(const MachineBasicBlock *MBB) const
Return true if the specified MBB will be emitted immediately after this block, such that if this bloc...
const MachineFunction * getParent() const
Return the MachineFunction containing this basic block.
reverse_iterator rbegin()
The MachineFrameInfo class represents an abstract stack frame until prolog/epilog code is inserted.
int64_t getObjectSize(int ObjectIdx) const
Return the size of the specified object.
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.
MachineFrameInfo & getFrameInfo()
getFrameInfo - Return the frame info object for the current function.
MachineRegisterInfo & getRegInfo()
getRegInfo - Return information about the registers currently in use.
Ty * getInfo()
getInfo - Keep track of various per-function pieces of information for backends that would like to do...
const MachineBasicBlock & front() const
const MachineInstrBuilder & addExternalSymbol(const char *FnName, unsigned TargetFlags=0) const
const MachineInstrBuilder & addImm(int64_t Val) const
Add a new immediate operand.
const MachineInstrBuilder & add(const MachineOperand &MO) const
const MachineInstrBuilder & addReg(Register RegNo, unsigned flags=0, unsigned SubReg=0) const
Add a new virtual register operand.
const MachineInstrBuilder & addMBB(MachineBasicBlock *MBB, unsigned TargetFlags=0) const
const MachineInstrBuilder & copyImplicitOps(const MachineInstr &OtherMI) const
Copy all the implicit operands from OtherMI onto this one.
MachineInstr * getInstr() const
If conversion operators fail, use this method to get the MachineInstr explicitly.
Representation of each machine instruction.
const MachineBasicBlock * getParent() const
void setDesc(const MCInstrDesc &TID)
Replace the instruction descriptor (thus opcode) of the current instruction with a new one.
const DebugLoc & getDebugLoc() const
Returns the debug location id of this MachineInstr.
void eraseFromParent()
Unlink 'this' from the containing basic block and delete it.
const MachineOperand & getOperand(unsigned i) const
MachineOperand class - Representation of each machine instruction operand.
unsigned getOperandNo() const
Returns the index of this operand in the instruction that it belongs to.
bool isReg() const
isReg - Tests if this is a MO_Register operand.
void setReg(Register Reg)
Change the register this operand corresponds to.
MachineInstr * getParent()
getParent - Return the instruction that this operand belongs to.
static MachineOperand CreateImm(int64_t Val)
Register getReg() const
getReg - Returns the register number.
MachineRegisterInfo - Keep track of information for virtual and physical registers,...
Register createVirtualRegister(const TargetRegisterClass *RegClass, StringRef Name="")
createVirtualRegister - Create and return a new virtual register in the function with the specified r...
bool match(StringRef String, SmallVectorImpl< StringRef > *Matches=nullptr, std::string *Error=nullptr) const
matches - Match the regex against a given String.
Wrapper class representing virtual and physical registers.
This class consists of common code factored out of the SmallVector class to reduce code duplication b...
StringRef - Represent a constant reference to a string, i.e.
TargetRegisterInfo base class - We assume that the target defines a static array of TargetRegisterDes...
#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.
@ MO_GOTPCREL
On a symbol operand this indicates that the immediate is offset to the GOT entry for the symbol name ...
static const MachineInstrBuilder & addFrameReference(const MachineInstrBuilder &MIB, int FI, int Offset=0)
addFrameReference - This function is used to add a reference to the base of an abstract object on the...
static M68k::CondCode GetCondFromBranchOpc(unsigned Opcode)
@ Implicit
Not emitted register (e.g. carry, or temporary result).
@ Undef
Value of the register doesn't matter.
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.
detail::scope_exit< std::decay_t< Callable > > make_scope_exit(Callable &&F)
FunctionPass * createM68kGlobalBaseRegPass()
This pass initializes a global base register for PIC on M68k.
decltype(auto) get(const PointerIntPair< PointerTy, IntBits, IntType, PtrTraits, Info > &Pair)
raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
@ And
Bitwise or logical AND of integers.
unsigned getKillRegState(bool B)
Description of the encoding of one expression Op.