2009-11-07 22:00:39 +00:00
|
|
|
//===- ARMBaseInstrInfo.h - ARM Base Instruction Information ----*- C++ -*-===//
|
2009-07-08 16:09:28 +00:00
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file contains the Base ARM implementation of the TargetInstrInfo class.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#ifndef ARMBASEINSTRUCTIONINFO_H
|
|
|
|
#define ARMBASEINSTRUCTIONINFO_H
|
|
|
|
|
|
|
|
#include "ARM.h"
|
2009-07-16 23:26:06 +00:00
|
|
|
#include "llvm/CodeGen/MachineInstrBuilder.h"
|
|
|
|
#include "llvm/Target/TargetInstrInfo.h"
|
2009-07-08 16:09:28 +00:00
|
|
|
|
|
|
|
namespace llvm {
|
2010-07-20 21:17:29 +00:00
|
|
|
class ARMSubtarget;
|
|
|
|
class ARMBaseRegisterInfo;
|
2009-07-08 16:09:28 +00:00
|
|
|
|
|
|
|
/// ARMII - This namespace holds all of the target specific flags that
|
|
|
|
/// instruction info tracks.
|
|
|
|
///
|
|
|
|
namespace ARMII {
|
|
|
|
enum {
|
|
|
|
//===------------------------------------------------------------------===//
|
|
|
|
// Instruction Flags.
|
|
|
|
|
|
|
|
//===------------------------------------------------------------------===//
|
|
|
|
// This four-bit field describes the addressing mode used.
|
|
|
|
|
2010-10-05 18:14:55 +00:00
|
|
|
AddrModeMask = 0x1f,
|
2009-07-08 16:09:28 +00:00
|
|
|
AddrModeNone = 0,
|
|
|
|
AddrMode1 = 1,
|
|
|
|
AddrMode2 = 2,
|
|
|
|
AddrMode3 = 3,
|
|
|
|
AddrMode4 = 4,
|
|
|
|
AddrMode5 = 5,
|
|
|
|
AddrMode6 = 6,
|
|
|
|
AddrModeT1_1 = 7,
|
|
|
|
AddrModeT1_2 = 8,
|
|
|
|
AddrModeT1_4 = 9,
|
|
|
|
AddrModeT1_s = 10, // i8 * 4 for pc and sp relative data
|
|
|
|
AddrModeT2_i12 = 11,
|
|
|
|
AddrModeT2_i8 = 12,
|
|
|
|
AddrModeT2_so = 13,
|
|
|
|
AddrModeT2_pc = 14, // +/- i12 for pc relative data
|
|
|
|
AddrModeT2_i8s4 = 15, // i8 * 4
|
|
|
|
|
|
|
|
// Size* - Flags to keep track of the size of an instruction.
|
2010-10-05 18:14:55 +00:00
|
|
|
SizeShift = 5,
|
2009-07-08 16:09:28 +00:00
|
|
|
SizeMask = 7 << SizeShift,
|
|
|
|
SizeSpecial = 1, // 0 byte pseudo or special case.
|
|
|
|
Size8Bytes = 2,
|
|
|
|
Size4Bytes = 3,
|
|
|
|
Size2Bytes = 4,
|
|
|
|
|
2010-03-13 07:34:35 +00:00
|
|
|
// IndexMode - Unindex, pre-indexed, or post-indexed are valid for load
|
|
|
|
// and store ops only. Generic "updating" flag is used for ld/st multiple.
|
2010-10-05 18:14:55 +00:00
|
|
|
IndexModeShift = 8,
|
2009-07-08 16:09:28 +00:00
|
|
|
IndexModeMask = 3 << IndexModeShift,
|
|
|
|
IndexModePre = 1,
|
|
|
|
IndexModePost = 2,
|
2010-03-13 07:34:35 +00:00
|
|
|
IndexModeUpd = 3,
|
2009-07-08 16:09:28 +00:00
|
|
|
|
|
|
|
//===------------------------------------------------------------------===//
|
|
|
|
// Instruction encoding formats.
|
|
|
|
//
|
2010-10-05 18:14:55 +00:00
|
|
|
FormShift = 10,
|
2009-07-08 16:09:28 +00:00
|
|
|
FormMask = 0x3f << FormShift,
|
|
|
|
|
|
|
|
// Pseudo instructions
|
|
|
|
Pseudo = 0 << FormShift,
|
|
|
|
|
|
|
|
// Multiply instructions
|
|
|
|
MulFrm = 1 << FormShift,
|
|
|
|
|
|
|
|
// Branch instructions
|
|
|
|
BrFrm = 2 << FormShift,
|
|
|
|
BrMiscFrm = 3 << FormShift,
|
|
|
|
|
|
|
|
// Data Processing instructions
|
|
|
|
DPFrm = 4 << FormShift,
|
|
|
|
DPSoRegFrm = 5 << FormShift,
|
|
|
|
|
|
|
|
// Load and Store
|
|
|
|
LdFrm = 6 << FormShift,
|
|
|
|
StFrm = 7 << FormShift,
|
|
|
|
LdMiscFrm = 8 << FormShift,
|
|
|
|
StMiscFrm = 9 << FormShift,
|
|
|
|
LdStMulFrm = 10 << FormShift,
|
|
|
|
|
2010-03-19 17:39:00 +00:00
|
|
|
LdStExFrm = 11 << FormShift,
|
2009-12-11 01:42:04 +00:00
|
|
|
|
2009-07-08 16:09:28 +00:00
|
|
|
// Miscellaneous arithmetic instructions
|
2010-03-19 17:39:00 +00:00
|
|
|
ArithMiscFrm = 12 << FormShift,
|
2010-08-11 00:01:18 +00:00
|
|
|
SatFrm = 13 << FormShift,
|
2009-07-08 16:09:28 +00:00
|
|
|
|
|
|
|
// Extend instructions
|
2010-08-11 00:01:18 +00:00
|
|
|
ExtFrm = 14 << FormShift,
|
2009-07-08 16:09:28 +00:00
|
|
|
|
|
|
|
// VFP formats
|
2010-08-11 00:01:18 +00:00
|
|
|
VFPUnaryFrm = 15 << FormShift,
|
|
|
|
VFPBinaryFrm = 16 << FormShift,
|
|
|
|
VFPConv1Frm = 17 << FormShift,
|
|
|
|
VFPConv2Frm = 18 << FormShift,
|
|
|
|
VFPConv3Frm = 19 << FormShift,
|
|
|
|
VFPConv4Frm = 20 << FormShift,
|
|
|
|
VFPConv5Frm = 21 << FormShift,
|
|
|
|
VFPLdStFrm = 22 << FormShift,
|
|
|
|
VFPLdStMulFrm = 23 << FormShift,
|
|
|
|
VFPMiscFrm = 24 << FormShift,
|
2009-07-08 16:09:28 +00:00
|
|
|
|
|
|
|
// Thumb format
|
2010-08-11 00:01:18 +00:00
|
|
|
ThumbFrm = 25 << FormShift,
|
2009-07-08 16:09:28 +00:00
|
|
|
|
2010-06-25 23:45:37 +00:00
|
|
|
// Miscelleaneous format
|
2010-08-11 00:01:18 +00:00
|
|
|
MiscFrm = 26 << FormShift,
|
2010-06-25 23:45:37 +00:00
|
|
|
|
2010-06-11 21:34:50 +00:00
|
|
|
// NEON formats
|
2010-08-11 00:01:18 +00:00
|
|
|
NGetLnFrm = 27 << FormShift,
|
|
|
|
NSetLnFrm = 28 << FormShift,
|
|
|
|
NDupFrm = 29 << FormShift,
|
|
|
|
NLdStFrm = 30 << FormShift,
|
|
|
|
N1RegModImmFrm= 31 << FormShift,
|
|
|
|
N2RegFrm = 32 << FormShift,
|
|
|
|
NVCVTFrm = 33 << FormShift,
|
|
|
|
NVDupLnFrm = 34 << FormShift,
|
|
|
|
N2RegVShLFrm = 35 << FormShift,
|
|
|
|
N2RegVShRFrm = 36 << FormShift,
|
|
|
|
N3RegFrm = 37 << FormShift,
|
|
|
|
N3RegVShFrm = 38 << FormShift,
|
|
|
|
NVExtFrm = 39 << FormShift,
|
|
|
|
NVMulSLFrm = 40 << FormShift,
|
|
|
|
NVTBLFrm = 41 << FormShift,
|
2009-07-08 16:09:28 +00:00
|
|
|
|
|
|
|
//===------------------------------------------------------------------===//
|
|
|
|
// Misc flags.
|
|
|
|
|
|
|
|
// UnaryDP - Indicates this is a unary data processing instruction, i.e.
|
|
|
|
// it doesn't have a Rn operand.
|
2010-10-05 18:14:55 +00:00
|
|
|
UnaryDP = 1 << 16,
|
2009-07-08 16:09:28 +00:00
|
|
|
|
|
|
|
// Xform16Bit - Indicates this Thumb2 instruction may be transformed into
|
|
|
|
// a 16-bit Thumb instruction if certain conditions are met.
|
2010-10-05 18:14:55 +00:00
|
|
|
Xform16Bit = 1 << 17,
|
2009-07-08 16:09:28 +00:00
|
|
|
|
2009-11-02 00:10:38 +00:00
|
|
|
//===------------------------------------------------------------------===//
|
|
|
|
// Code domain.
|
2010-10-05 18:14:55 +00:00
|
|
|
DomainShift = 18,
|
2009-11-02 00:10:38 +00:00
|
|
|
DomainMask = 3 << DomainShift,
|
|
|
|
DomainGeneral = 0 << DomainShift,
|
|
|
|
DomainVFP = 1 << DomainShift,
|
|
|
|
DomainNEON = 2 << DomainShift,
|
|
|
|
|
2009-07-08 16:09:28 +00:00
|
|
|
//===------------------------------------------------------------------===//
|
|
|
|
// Field shifts - such shifts are used to set field while generating
|
|
|
|
// machine instructions.
|
|
|
|
M_BitShift = 5,
|
|
|
|
ShiftImmShift = 5,
|
|
|
|
ShiftShift = 7,
|
|
|
|
N_BitShift = 7,
|
|
|
|
ImmHiShift = 8,
|
|
|
|
SoRotImmShift = 8,
|
|
|
|
RegRsShift = 8,
|
|
|
|
ExtRotImmShift = 10,
|
|
|
|
RegRdLoShift = 12,
|
|
|
|
RegRdShift = 12,
|
|
|
|
RegRdHiShift = 16,
|
|
|
|
RegRnShift = 16,
|
|
|
|
S_BitShift = 20,
|
|
|
|
W_BitShift = 21,
|
|
|
|
AM3_I_BitShift = 22,
|
|
|
|
D_BitShift = 22,
|
|
|
|
U_BitShift = 23,
|
|
|
|
P_BitShift = 24,
|
|
|
|
I_BitShift = 25,
|
|
|
|
CondShift = 28
|
|
|
|
};
|
2009-07-19 19:16:46 +00:00
|
|
|
}
|
|
|
|
|
2009-07-08 16:09:28 +00:00
|
|
|
class ARMBaseInstrInfo : public TargetInstrInfoImpl {
|
2010-07-20 21:17:29 +00:00
|
|
|
const ARMSubtarget &Subtarget;
|
2009-07-08 16:09:28 +00:00
|
|
|
protected:
|
|
|
|
// Can be only subclassed.
|
2009-11-02 00:10:38 +00:00
|
|
|
explicit ARMBaseInstrInfo(const ARMSubtarget &STI);
|
2009-07-08 16:09:28 +00:00
|
|
|
public:
|
|
|
|
// Return the non-pre/post incrementing version of 'Opc'. Return 0
|
|
|
|
// if there is not such an opcode.
|
|
|
|
virtual unsigned getUnindexedOpcode(unsigned Opc) const =0;
|
|
|
|
|
|
|
|
virtual MachineInstr *convertToThreeAddress(MachineFunction::iterator &MFI,
|
|
|
|
MachineBasicBlock::iterator &MBBI,
|
|
|
|
LiveVariables *LV) const;
|
|
|
|
|
|
|
|
virtual const ARMBaseRegisterInfo &getRegisterInfo() const =0;
|
2009-11-02 00:10:38 +00:00
|
|
|
const ARMSubtarget &getSubtarget() const { return Subtarget; }
|
2009-07-08 16:09:28 +00:00
|
|
|
|
2010-05-22 01:47:14 +00:00
|
|
|
bool spillCalleeSavedRegisters(MachineBasicBlock &MBB,
|
|
|
|
MachineBasicBlock::iterator MI,
|
|
|
|
const std::vector<CalleeSavedInfo> &CSI,
|
|
|
|
const TargetRegisterInfo *TRI) const;
|
|
|
|
|
2009-07-08 16:09:28 +00:00
|
|
|
// Branch analysis.
|
|
|
|
virtual bool AnalyzeBranch(MachineBasicBlock &MBB, MachineBasicBlock *&TBB,
|
|
|
|
MachineBasicBlock *&FBB,
|
|
|
|
SmallVectorImpl<MachineOperand> &Cond,
|
2010-07-22 21:27:00 +00:00
|
|
|
bool AllowModify = false) const;
|
2009-07-08 16:09:28 +00:00
|
|
|
virtual unsigned RemoveBranch(MachineBasicBlock &MBB) const;
|
|
|
|
virtual unsigned InsertBranch(MachineBasicBlock &MBB, MachineBasicBlock *TBB,
|
|
|
|
MachineBasicBlock *FBB,
|
2010-06-17 22:43:56 +00:00
|
|
|
const SmallVectorImpl<MachineOperand> &Cond,
|
|
|
|
DebugLoc DL) const;
|
2009-07-08 16:09:28 +00:00
|
|
|
|
|
|
|
virtual
|
|
|
|
bool ReverseBranchCondition(SmallVectorImpl<MachineOperand> &Cond) const;
|
|
|
|
|
|
|
|
// Predication support.
|
2009-07-10 01:38:27 +00:00
|
|
|
bool isPredicated(const MachineInstr *MI) const {
|
|
|
|
int PIdx = MI->findFirstPredOperandIdx();
|
|
|
|
return PIdx != -1 && MI->getOperand(PIdx).getImm() != ARMCC::AL;
|
|
|
|
}
|
2009-07-08 16:09:28 +00:00
|
|
|
|
|
|
|
ARMCC::CondCodes getPredicate(const MachineInstr *MI) const {
|
|
|
|
int PIdx = MI->findFirstPredOperandIdx();
|
|
|
|
return PIdx != -1 ? (ARMCC::CondCodes)MI->getOperand(PIdx).getImm()
|
|
|
|
: ARMCC::AL;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual
|
|
|
|
bool PredicateInstruction(MachineInstr *MI,
|
|
|
|
const SmallVectorImpl<MachineOperand> &Pred) const;
|
|
|
|
|
|
|
|
virtual
|
|
|
|
bool SubsumesPredicate(const SmallVectorImpl<MachineOperand> &Pred1,
|
|
|
|
const SmallVectorImpl<MachineOperand> &Pred2) const;
|
|
|
|
|
|
|
|
virtual bool DefinesPredicate(MachineInstr *MI,
|
|
|
|
std::vector<MachineOperand> &Pred) const;
|
|
|
|
|
2009-11-21 06:21:52 +00:00
|
|
|
virtual bool isPredicable(MachineInstr *MI) const;
|
|
|
|
|
2009-07-08 16:09:28 +00:00
|
|
|
/// GetInstSize - Returns the size of the specified MachineInstr.
|
|
|
|
///
|
|
|
|
virtual unsigned GetInstSizeInBytes(const MachineInstr* MI) const;
|
|
|
|
|
|
|
|
virtual unsigned isLoadFromStackSlot(const MachineInstr *MI,
|
|
|
|
int &FrameIndex) const;
|
|
|
|
virtual unsigned isStoreToStackSlot(const MachineInstr *MI,
|
|
|
|
int &FrameIndex) const;
|
|
|
|
|
2010-07-11 06:33:54 +00:00
|
|
|
virtual void copyPhysReg(MachineBasicBlock &MBB,
|
|
|
|
MachineBasicBlock::iterator I, DebugLoc DL,
|
|
|
|
unsigned DestReg, unsigned SrcReg,
|
|
|
|
bool KillSrc) const;
|
2009-07-27 03:14:20 +00:00
|
|
|
|
2009-07-08 16:09:28 +00:00
|
|
|
virtual void storeRegToStackSlot(MachineBasicBlock &MBB,
|
|
|
|
MachineBasicBlock::iterator MBBI,
|
|
|
|
unsigned SrcReg, bool isKill, int FrameIndex,
|
2010-05-06 19:06:44 +00:00
|
|
|
const TargetRegisterClass *RC,
|
|
|
|
const TargetRegisterInfo *TRI) const;
|
2009-07-08 16:09:28 +00:00
|
|
|
|
|
|
|
virtual void loadRegFromStackSlot(MachineBasicBlock &MBB,
|
|
|
|
MachineBasicBlock::iterator MBBI,
|
|
|
|
unsigned DestReg, int FrameIndex,
|
2010-05-06 19:06:44 +00:00
|
|
|
const TargetRegisterClass *RC,
|
|
|
|
const TargetRegisterInfo *TRI) const;
|
2009-07-08 16:09:28 +00:00
|
|
|
|
2010-04-26 07:39:25 +00:00
|
|
|
virtual MachineInstr *emitFrameIndexDebugValue(MachineFunction &MF,
|
2010-04-29 01:13:30 +00:00
|
|
|
int FrameIx,
|
2010-04-26 07:39:25 +00:00
|
|
|
uint64_t Offset,
|
|
|
|
const MDNode *MDPtr,
|
|
|
|
DebugLoc DL) const;
|
|
|
|
|
2009-11-08 00:15:23 +00:00
|
|
|
virtual void reMaterialize(MachineBasicBlock &MBB,
|
|
|
|
MachineBasicBlock::iterator MI,
|
|
|
|
unsigned DestReg, unsigned SubIdx,
|
2009-11-14 02:55:43 +00:00
|
|
|
const MachineInstr *Orig,
|
2010-06-02 22:47:25 +00:00
|
|
|
const TargetRegisterInfo &TRI) const;
|
2009-11-08 00:15:23 +00:00
|
|
|
|
2010-01-06 23:47:07 +00:00
|
|
|
MachineInstr *duplicate(MachineInstr *Orig, MachineFunction &MF) const;
|
|
|
|
|
2010-03-03 01:44:33 +00:00
|
|
|
virtual bool produceSameValue(const MachineInstr *MI0,
|
|
|
|
const MachineInstr *MI1) const;
|
2010-06-18 23:09:54 +00:00
|
|
|
|
2010-06-23 23:00:16 +00:00
|
|
|
/// areLoadsFromSameBasePtr - This is used by the pre-regalloc scheduler to
|
|
|
|
/// determine if two loads are loading from the same base address. It should
|
|
|
|
/// only return true if the base pointers are the same and the only
|
|
|
|
/// differences between the two addresses is the offset. It also returns the
|
|
|
|
/// offsets by reference.
|
|
|
|
virtual bool areLoadsFromSameBasePtr(SDNode *Load1, SDNode *Load2,
|
|
|
|
int64_t &Offset1, int64_t &Offset2)const;
|
|
|
|
|
|
|
|
/// shouldScheduleLoadsNear - This is a used by the pre-regalloc scheduler to
|
|
|
|
/// determine (in conjuction with areLoadsFromSameBasePtr) if two loads should
|
|
|
|
/// be scheduled togther. On some targets if two loads are loading from
|
|
|
|
/// addresses in the same cache line, it's better if they are scheduled
|
|
|
|
/// together. This function takes two integers that represent the load offsets
|
|
|
|
/// from the common base address. It returns true if it decides it's desirable
|
|
|
|
/// to schedule the two loads together. "NumLoads" is the number of loads that
|
|
|
|
/// have already been scheduled after Load1.
|
|
|
|
virtual bool shouldScheduleLoadsNear(SDNode *Load1, SDNode *Load2,
|
|
|
|
int64_t Offset1, int64_t Offset2,
|
|
|
|
unsigned NumLoads) const;
|
|
|
|
|
2010-06-18 23:09:54 +00:00
|
|
|
virtual bool isSchedulingBoundary(const MachineInstr *MI,
|
|
|
|
const MachineBasicBlock *MBB,
|
|
|
|
const MachineFunction &MF) const;
|
2010-06-25 22:42:03 +00:00
|
|
|
|
|
|
|
virtual bool isProfitableToIfCvt(MachineBasicBlock &MBB,
|
2010-10-01 22:45:50 +00:00
|
|
|
unsigned NumInstrs,
|
|
|
|
float Prob, float Confidence) const;
|
2010-06-25 22:42:03 +00:00
|
|
|
|
|
|
|
virtual bool isProfitableToIfCvt(MachineBasicBlock &TMBB,unsigned NumT,
|
2010-09-28 18:32:13 +00:00
|
|
|
MachineBasicBlock &FMBB,unsigned NumF,
|
2010-10-01 22:45:50 +00:00
|
|
|
float Probability, float Confidence) const;
|
2010-06-25 22:42:03 +00:00
|
|
|
|
|
|
|
virtual bool isProfitableToDupForIfCvt(MachineBasicBlock &MBB,
|
2010-09-28 18:32:13 +00:00
|
|
|
unsigned NumInstrs,
|
2010-10-01 22:45:50 +00:00
|
|
|
float Probability,
|
|
|
|
float Confidence) const {
|
2010-06-25 22:42:03 +00:00
|
|
|
return NumInstrs && NumInstrs == 1;
|
|
|
|
}
|
2010-08-06 01:32:48 +00:00
|
|
|
|
2010-08-08 05:04:59 +00:00
|
|
|
/// AnalyzeCompare - For a comparison instruction, return the source register
|
|
|
|
/// in SrcReg and the value it compares against in CmpValue. Return true if
|
|
|
|
/// the comparison instruction can be analyzed.
|
|
|
|
virtual bool AnalyzeCompare(const MachineInstr *MI, unsigned &SrcReg,
|
2010-09-21 12:01:15 +00:00
|
|
|
int &CmpMask, int &CmpValue) const;
|
2010-08-06 01:32:48 +00:00
|
|
|
|
2010-09-11 00:13:50 +00:00
|
|
|
/// OptimizeCompareInstr - Convert the instruction to set the zero flag so
|
2010-08-06 01:32:48 +00:00
|
|
|
/// that we can remove a "comparison with zero".
|
2010-09-11 00:13:50 +00:00
|
|
|
virtual bool OptimizeCompareInstr(MachineInstr *CmpInstr, unsigned SrcReg,
|
2010-09-21 12:01:15 +00:00
|
|
|
int CmpMask, int CmpValue,
|
2010-09-10 21:55:43 +00:00
|
|
|
MachineBasicBlock::iterator &MII) const;
|
2010-09-09 18:18:55 +00:00
|
|
|
|
|
|
|
virtual unsigned getNumMicroOps(const MachineInstr *MI,
|
2010-09-10 01:29:16 +00:00
|
|
|
const InstrItineraryData *ItinData) const;
|
2010-10-06 06:27:31 +00:00
|
|
|
|
|
|
|
virtual
|
|
|
|
int getOperandLatency(const InstrItineraryData *ItinData,
|
|
|
|
const MachineInstr *DefMI, unsigned DefIdx,
|
|
|
|
const MachineInstr *UseMI, unsigned UseIdx) const;
|
|
|
|
virtual
|
|
|
|
int getOperandLatency(const InstrItineraryData *ItinData,
|
|
|
|
SDNode *DefNode, unsigned DefIdx,
|
|
|
|
SDNode *UseNode, unsigned UseIdx) const;
|
|
|
|
private:
|
|
|
|
int getOperandLatency(const InstrItineraryData *ItinData,
|
|
|
|
const TargetInstrDesc &DefTID,
|
|
|
|
unsigned DefIdx, unsigned DefAlign,
|
|
|
|
const TargetInstrDesc &UseTID,
|
|
|
|
unsigned UseIdx, unsigned UseAlign) const;
|
2009-07-28 05:48:47 +00:00
|
|
|
};
|
2009-07-27 18:20:05 +00:00
|
|
|
|
2009-07-28 05:48:47 +00:00
|
|
|
static inline
|
|
|
|
const MachineInstrBuilder &AddDefaultPred(const MachineInstrBuilder &MIB) {
|
|
|
|
return MIB.addImm((int64_t)ARMCC::AL).addReg(0);
|
|
|
|
}
|
2009-07-27 18:20:05 +00:00
|
|
|
|
2009-07-28 05:48:47 +00:00
|
|
|
static inline
|
|
|
|
const MachineInstrBuilder &AddDefaultCC(const MachineInstrBuilder &MIB) {
|
|
|
|
return MIB.addReg(0);
|
|
|
|
}
|
2009-07-27 18:25:24 +00:00
|
|
|
|
2009-07-28 05:48:47 +00:00
|
|
|
static inline
|
2009-08-10 02:37:24 +00:00
|
|
|
const MachineInstrBuilder &AddDefaultT1CC(const MachineInstrBuilder &MIB,
|
|
|
|
bool isDead = false) {
|
|
|
|
return MIB.addReg(ARM::CPSR, getDefRegState(true) | getDeadRegState(isDead));
|
2009-07-28 05:48:47 +00:00
|
|
|
}
|
|
|
|
|
2009-08-15 07:59:10 +00:00
|
|
|
static inline
|
|
|
|
const MachineInstrBuilder &AddNoT1CC(const MachineInstrBuilder &MIB) {
|
|
|
|
return MIB.addReg(0);
|
|
|
|
}
|
|
|
|
|
2009-07-28 05:48:47 +00:00
|
|
|
static inline
|
|
|
|
bool isUncondBranchOpcode(int Opc) {
|
|
|
|
return Opc == ARM::B || Opc == ARM::tB || Opc == ARM::t2B;
|
2009-07-08 16:09:28 +00:00
|
|
|
}
|
|
|
|
|
2009-07-28 05:48:47 +00:00
|
|
|
static inline
|
|
|
|
bool isCondBranchOpcode(int Opc) {
|
|
|
|
return Opc == ARM::Bcc || Opc == ARM::tBcc || Opc == ARM::t2Bcc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline
|
|
|
|
bool isJumpTableBranchOpcode(int Opc) {
|
|
|
|
return Opc == ARM::BR_JTr || Opc == ARM::BR_JTm || Opc == ARM::BR_JTadd ||
|
|
|
|
Opc == ARM::tBR_JTr || Opc == ARM::t2BR_JT;
|
|
|
|
}
|
|
|
|
|
2009-10-28 18:26:41 +00:00
|
|
|
static inline
|
|
|
|
bool isIndirectBranchOpcode(int Opc) {
|
2010-03-06 19:39:36 +00:00
|
|
|
return Opc == ARM::BRIND || Opc == ARM::MOVPCRX || Opc == ARM::tBRIND;
|
2009-10-28 18:26:41 +00:00
|
|
|
}
|
|
|
|
|
2009-08-08 03:20:32 +00:00
|
|
|
/// getInstrPredicate - If instruction is predicated, returns its predicate
|
|
|
|
/// condition, otherwise returns AL. It also returns the condition code
|
|
|
|
/// register by reference.
|
2009-09-28 09:14:39 +00:00
|
|
|
ARMCC::CondCodes getInstrPredicate(const MachineInstr *MI, unsigned &PredReg);
|
2009-08-08 03:20:32 +00:00
|
|
|
|
2009-07-28 05:48:47 +00:00
|
|
|
int getMatchingCondBranchOpcode(int Opc);
|
|
|
|
|
|
|
|
/// emitARMRegPlusImmediate / emitT2RegPlusImmediate - Emits a series of
|
|
|
|
/// instructions to materializea destreg = basereg + immediate in ARM / Thumb2
|
|
|
|
/// code.
|
|
|
|
void emitARMRegPlusImmediate(MachineBasicBlock &MBB,
|
|
|
|
MachineBasicBlock::iterator &MBBI, DebugLoc dl,
|
|
|
|
unsigned DestReg, unsigned BaseReg, int NumBytes,
|
|
|
|
ARMCC::CondCodes Pred, unsigned PredReg,
|
|
|
|
const ARMBaseInstrInfo &TII);
|
|
|
|
|
|
|
|
void emitT2RegPlusImmediate(MachineBasicBlock &MBB,
|
|
|
|
MachineBasicBlock::iterator &MBBI, DebugLoc dl,
|
|
|
|
unsigned DestReg, unsigned BaseReg, int NumBytes,
|
|
|
|
ARMCC::CondCodes Pred, unsigned PredReg,
|
|
|
|
const ARMBaseInstrInfo &TII);
|
|
|
|
|
|
|
|
|
2009-08-11 15:33:49 +00:00
|
|
|
/// rewriteARMFrameIndex / rewriteT2FrameIndex -
|
2009-08-27 01:23:50 +00:00
|
|
|
/// Rewrite MI to access 'Offset' bytes from the FP. Return false if the
|
|
|
|
/// offset could not be handled directly in MI, and return the left-over
|
|
|
|
/// portion by reference.
|
|
|
|
bool rewriteARMFrameIndex(MachineInstr &MI, unsigned FrameRegIdx,
|
|
|
|
unsigned FrameReg, int &Offset,
|
|
|
|
const ARMBaseInstrInfo &TII);
|
|
|
|
|
|
|
|
bool rewriteT2FrameIndex(MachineInstr &MI, unsigned FrameRegIdx,
|
|
|
|
unsigned FrameReg, int &Offset,
|
|
|
|
const ARMBaseInstrInfo &TII);
|
2009-07-28 05:48:47 +00:00
|
|
|
|
|
|
|
} // End llvm namespace
|
|
|
|
|
2009-07-08 16:09:28 +00:00
|
|
|
#endif
|