2011-05-10 02:16:46 +08:00
|
|
|
//===- MipsInstrInfo.td - Target Description for Mips Target -*- tablegen -*-=//
|
2007-06-06 15:42:06 +08:00
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
2007-12-30 04:36:04 +08:00
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
2007-06-06 15:42:06 +08:00
|
|
|
//
|
2011-04-16 05:51:11 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2011-05-10 02:16:46 +08:00
|
|
|
//
|
|
|
|
// This file contains the Mips implementation of the TargetInstrInfo class.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
2007-06-06 15:42:06 +08:00
|
|
|
|
|
|
|
|
2011-04-16 05:51:11 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2007-06-06 15:42:06 +08:00
|
|
|
// Mips profiles and nodes
|
2011-04-16 05:51:11 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2007-06-06 15:42:06 +08:00
|
|
|
|
Several changes to Mips backend, experimental fp support being the most
important.
- Cleanup in the Subtarget info with addition of new features, not all support
yet, but they allow the future inclusion of features easier. Among new features,
we have : Arch family info (mips1, mips2, ...), ABI info (o32, eabi), 64-bit
integer
and float registers, allegrex vector FPU (VFPU), single float only support.
- TargetMachine now detects allegrex core.
- Added allegrex (Mips32r2) sext_inreg instructions.
- *Added Float Point Instructions*, handling single float only, and
aliased accesses for 32-bit FPUs.
- Some cleanup in FP instruction formats and FP register classes.
- Calling conventions improved to support mips 32-bit EABI.
- Added Asm Printer support for fp cond codes.
- Added support for sret copy to a return register.
- EABI support added into LowerCALL and FORMAL_ARGS.
- MipsFunctionInfo now keeps a virtual register per function to track the
sret on function entry until function ret.
- MipsInstrInfo FP support into methods (isMoveInstr, isLoadFromStackSlot, ...),
FP cond codes mapping and initial FP Branch Analysis.
- Two new Mips SDNode to handle fp branch and compare instructions : FPBrcond,
FPCmp
- MipsTargetLowering : handling different FP classes, Allegrex support, sret
return copy, no homing location within EABI, non 32-bit stack objects
arguments, and asm constraint for float.
llvm-svn: 53146
2008-07-06 03:05:21 +08:00
|
|
|
def SDT_MipsJmpLink : SDTypeProfile<0, 1, [SDTCisVT<0, iPTR>]>;
|
2010-11-10 01:25:34 +08:00
|
|
|
def SDT_MipsCMov : SDTypeProfile<1, 4, [SDTCisSameAs<0, 1>,
|
2011-04-16 05:00:26 +08:00
|
|
|
SDTCisSameAs<1, 2>,
|
|
|
|
SDTCisSameAs<3, 4>,
|
|
|
|
SDTCisInt<4>]>;
|
Several changes to Mips backend, experimental fp support being the most
important.
- Cleanup in the Subtarget info with addition of new features, not all support
yet, but they allow the future inclusion of features easier. Among new features,
we have : Arch family info (mips1, mips2, ...), ABI info (o32, eabi), 64-bit
integer
and float registers, allegrex vector FPU (VFPU), single float only support.
- TargetMachine now detects allegrex core.
- Added allegrex (Mips32r2) sext_inreg instructions.
- *Added Float Point Instructions*, handling single float only, and
aliased accesses for 32-bit FPUs.
- Some cleanup in FP instruction formats and FP register classes.
- Calling conventions improved to support mips 32-bit EABI.
- Added Asm Printer support for fp cond codes.
- Added support for sret copy to a return register.
- EABI support added into LowerCALL and FORMAL_ARGS.
- MipsFunctionInfo now keeps a virtual register per function to track the
sret on function entry until function ret.
- MipsInstrInfo FP support into methods (isMoveInstr, isLoadFromStackSlot, ...),
FP cond codes mapping and initial FP Branch Analysis.
- Two new Mips SDNode to handle fp branch and compare instructions : FPBrcond,
FPCmp
- MipsTargetLowering : handling different FP classes, Allegrex support, sret
return copy, no homing location within EABI, non 32-bit stack objects
arguments, and asm constraint for float.
llvm-svn: 53146
2008-07-06 03:05:21 +08:00
|
|
|
def SDT_MipsCallSeqStart : SDCallSeqStart<[SDTCisVT<0, i32>]>;
|
|
|
|
def SDT_MipsCallSeqEnd : SDCallSeqEnd<[SDTCisVT<0, i32>, SDTCisVT<1, i32>]>;
|
2011-03-05 01:51:39 +08:00
|
|
|
def SDT_MipsMAddMSub : SDTypeProfile<0, 4,
|
2011-01-19 03:29:17 +08:00
|
|
|
[SDTCisVT<0, i32>, SDTCisSameAs<0, 1>,
|
2011-03-05 01:51:39 +08:00
|
|
|
SDTCisSameAs<1, 2>,
|
2011-01-19 03:29:17 +08:00
|
|
|
SDTCisSameAs<2, 3>]>;
|
2011-03-05 05:03:24 +08:00
|
|
|
def SDT_MipsDivRem : SDTypeProfile<0, 2,
|
2011-10-04 05:06:13 +08:00
|
|
|
[SDTCisInt<0>,
|
2011-03-05 05:03:24 +08:00
|
|
|
SDTCisSameAs<0, 1>]>;
|
Several changes to Mips backend, experimental fp support being the most
important.
- Cleanup in the Subtarget info with addition of new features, not all support
yet, but they allow the future inclusion of features easier. Among new features,
we have : Arch family info (mips1, mips2, ...), ABI info (o32, eabi), 64-bit
integer
and float registers, allegrex vector FPU (VFPU), single float only support.
- TargetMachine now detects allegrex core.
- Added allegrex (Mips32r2) sext_inreg instructions.
- *Added Float Point Instructions*, handling single float only, and
aliased accesses for 32-bit FPUs.
- Some cleanup in FP instruction formats and FP register classes.
- Calling conventions improved to support mips 32-bit EABI.
- Added Asm Printer support for fp cond codes.
- Added support for sret copy to a return register.
- EABI support added into LowerCALL and FORMAL_ARGS.
- MipsFunctionInfo now keeps a virtual register per function to track the
sret on function entry until function ret.
- MipsInstrInfo FP support into methods (isMoveInstr, isLoadFromStackSlot, ...),
FP cond codes mapping and initial FP Branch Analysis.
- Two new Mips SDNode to handle fp branch and compare instructions : FPBrcond,
FPCmp
- MipsTargetLowering : handling different FP classes, Allegrex support, sret
return copy, no homing location within EABI, non 32-bit stack objects
arguments, and asm constraint for float.
llvm-svn: 53146
2008-07-06 03:05:21 +08:00
|
|
|
|
2011-05-31 10:53:58 +08:00
|
|
|
def SDT_MipsThreadPointer : SDTypeProfile<1, 0, [SDTCisPtrTy<0>]>;
|
|
|
|
|
2011-07-20 07:30:50 +08:00
|
|
|
def SDT_Sync : SDTypeProfile<0, 1, [SDTCisVT<0, i32>]>;
|
2011-06-21 08:40:49 +08:00
|
|
|
|
2011-09-23 07:31:54 +08:00
|
|
|
def SDT_Ext : SDTypeProfile<1, 3, [SDTCisInt<0>, SDTCisSameAs<0, 1>,
|
|
|
|
SDTCisVT<2, i32>, SDTCisSameAs<2, 3>]>;
|
|
|
|
def SDT_Ins : SDTypeProfile<1, 4, [SDTCisInt<0>, SDTCisSameAs<0, 1>,
|
|
|
|
SDTCisVT<2, i32>, SDTCisSameAs<2, 3>,
|
2011-08-17 10:05:42 +08:00
|
|
|
SDTCisSameAs<0, 4>]>;
|
|
|
|
|
2012-06-02 08:03:12 +08:00
|
|
|
def SDTMipsLoadLR : SDTypeProfile<1, 2,
|
|
|
|
[SDTCisInt<0>, SDTCisPtrTy<1>,
|
|
|
|
SDTCisSameAs<0, 2>]>;
|
|
|
|
|
2007-06-06 15:42:06 +08:00
|
|
|
// Call
|
2010-11-10 01:25:34 +08:00
|
|
|
def MipsJmpLink : SDNode<"MipsISD::JmpLink",SDT_MipsJmpLink,
|
2010-12-24 02:28:41 +08:00
|
|
|
[SDNPHasChain, SDNPOutGlue, SDNPOptInGlue,
|
2010-03-19 13:33:51 +08:00
|
|
|
SDNPVariadic]>;
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2012-10-20 04:59:39 +08:00
|
|
|
// Tail call
|
|
|
|
def MipsTailCall : SDNode<"MipsISD::TailCall", SDT_MipsJmpLink,
|
|
|
|
[SDNPHasChain, SDNPOptInGlue, SDNPVariadic]>;
|
|
|
|
|
2010-11-10 01:25:34 +08:00
|
|
|
// Hi and Lo nodes are used to handle global addresses. Used on
|
|
|
|
// MipsISelLowering to lower stuff like GlobalAddress, ExternalSymbol
|
2007-11-05 11:02:32 +08:00
|
|
|
// static model. (nothing to do with Mips Registers Hi and Lo)
|
2008-07-22 02:52:34 +08:00
|
|
|
def MipsHi : SDNode<"MipsISD::Hi", SDTIntUnaryOp>;
|
|
|
|
def MipsLo : SDNode<"MipsISD::Lo", SDTIntUnaryOp>;
|
|
|
|
def MipsGPRel : SDNode<"MipsISD::GPRel", SDTIntUnaryOp>;
|
2007-10-09 10:55:31 +08:00
|
|
|
|
2011-05-31 10:53:58 +08:00
|
|
|
// TlsGd node is used to handle General Dynamic TLS
|
|
|
|
def MipsTlsGd : SDNode<"MipsISD::TlsGd", SDTIntUnaryOp>;
|
|
|
|
|
|
|
|
// TprelHi and TprelLo nodes are used to handle Local Exec TLS
|
|
|
|
def MipsTprelHi : SDNode<"MipsISD::TprelHi", SDTIntUnaryOp>;
|
|
|
|
def MipsTprelLo : SDNode<"MipsISD::TprelLo", SDTIntUnaryOp>;
|
|
|
|
|
|
|
|
// Thread pointer
|
|
|
|
def MipsThreadPointer: SDNode<"MipsISD::ThreadPointer", SDT_MipsThreadPointer>;
|
|
|
|
|
2007-10-26 12:00:13 +08:00
|
|
|
// Return
|
2012-07-10 08:19:06 +08:00
|
|
|
def MipsRet : SDNode<"MipsISD::Ret", SDTNone, [SDNPHasChain, SDNPOptInGlue]>;
|
2007-06-06 15:42:06 +08:00
|
|
|
|
|
|
|
// These are target-independent nodes, but have target-specific formats.
|
Several changes to Mips backend, experimental fp support being the most
important.
- Cleanup in the Subtarget info with addition of new features, not all support
yet, but they allow the future inclusion of features easier. Among new features,
we have : Arch family info (mips1, mips2, ...), ABI info (o32, eabi), 64-bit
integer
and float registers, allegrex vector FPU (VFPU), single float only support.
- TargetMachine now detects allegrex core.
- Added allegrex (Mips32r2) sext_inreg instructions.
- *Added Float Point Instructions*, handling single float only, and
aliased accesses for 32-bit FPUs.
- Some cleanup in FP instruction formats and FP register classes.
- Calling conventions improved to support mips 32-bit EABI.
- Added Asm Printer support for fp cond codes.
- Added support for sret copy to a return register.
- EABI support added into LowerCALL and FORMAL_ARGS.
- MipsFunctionInfo now keeps a virtual register per function to track the
sret on function entry until function ret.
- MipsInstrInfo FP support into methods (isMoveInstr, isLoadFromStackSlot, ...),
FP cond codes mapping and initial FP Branch Analysis.
- Two new Mips SDNode to handle fp branch and compare instructions : FPBrcond,
FPCmp
- MipsTargetLowering : handling different FP classes, Allegrex support, sret
return copy, no homing location within EABI, non 32-bit stack objects
arguments, and asm constraint for float.
llvm-svn: 53146
2008-07-06 03:05:21 +08:00
|
|
|
def callseq_start : SDNode<"ISD::CALLSEQ_START", SDT_MipsCallSeqStart,
|
2012-08-24 22:43:27 +08:00
|
|
|
[SDNPHasChain, SDNPSideEffect, SDNPOutGlue]>;
|
Several changes to Mips backend, experimental fp support being the most
important.
- Cleanup in the Subtarget info with addition of new features, not all support
yet, but they allow the future inclusion of features easier. Among new features,
we have : Arch family info (mips1, mips2, ...), ABI info (o32, eabi), 64-bit
integer
and float registers, allegrex vector FPU (VFPU), single float only support.
- TargetMachine now detects allegrex core.
- Added allegrex (Mips32r2) sext_inreg instructions.
- *Added Float Point Instructions*, handling single float only, and
aliased accesses for 32-bit FPUs.
- Some cleanup in FP instruction formats and FP register classes.
- Calling conventions improved to support mips 32-bit EABI.
- Added Asm Printer support for fp cond codes.
- Added support for sret copy to a return register.
- EABI support added into LowerCALL and FORMAL_ARGS.
- MipsFunctionInfo now keeps a virtual register per function to track the
sret on function entry until function ret.
- MipsInstrInfo FP support into methods (isMoveInstr, isLoadFromStackSlot, ...),
FP cond codes mapping and initial FP Branch Analysis.
- Two new Mips SDNode to handle fp branch and compare instructions : FPBrcond,
FPCmp
- MipsTargetLowering : handling different FP classes, Allegrex support, sret
return copy, no homing location within EABI, non 32-bit stack objects
arguments, and asm constraint for float.
llvm-svn: 53146
2008-07-06 03:05:21 +08:00
|
|
|
def callseq_end : SDNode<"ISD::CALLSEQ_END", SDT_MipsCallSeqEnd,
|
2012-08-24 22:43:27 +08:00
|
|
|
[SDNPHasChain, SDNPSideEffect,
|
|
|
|
SDNPOptInGlue, SDNPOutGlue]>;
|
2007-11-13 08:44:25 +08:00
|
|
|
|
2011-01-19 03:29:17 +08:00
|
|
|
// MAdd*/MSub* nodes
|
|
|
|
def MipsMAdd : SDNode<"MipsISD::MAdd", SDT_MipsMAddMSub,
|
|
|
|
[SDNPOptInGlue, SDNPOutGlue]>;
|
|
|
|
def MipsMAddu : SDNode<"MipsISD::MAddu", SDT_MipsMAddMSub,
|
|
|
|
[SDNPOptInGlue, SDNPOutGlue]>;
|
|
|
|
def MipsMSub : SDNode<"MipsISD::MSub", SDT_MipsMAddMSub,
|
|
|
|
[SDNPOptInGlue, SDNPOutGlue]>;
|
|
|
|
def MipsMSubu : SDNode<"MipsISD::MSubu", SDT_MipsMAddMSub,
|
|
|
|
[SDNPOptInGlue, SDNPOutGlue]>;
|
|
|
|
|
2011-03-05 05:03:24 +08:00
|
|
|
// DivRem(u) nodes
|
|
|
|
def MipsDivRem : SDNode<"MipsISD::DivRem", SDT_MipsDivRem,
|
|
|
|
[SDNPOutGlue]>;
|
|
|
|
def MipsDivRemU : SDNode<"MipsISD::DivRemU", SDT_MipsDivRem,
|
|
|
|
[SDNPOutGlue]>;
|
|
|
|
|
2011-06-08 02:00:14 +08:00
|
|
|
// Target constant nodes that are not part of any isel patterns and remain
|
|
|
|
// unchanged can cause instructions with illegal operands to be emitted.
|
|
|
|
// Wrapper node patterns give the instruction selector a chance to replace
|
|
|
|
// target constant nodes that would otherwise remain unchanged with ADDiu
|
|
|
|
// nodes. Without these wrapper node patterns, the following conditional move
|
|
|
|
// instrucion is emitted when function cmov2 in test/CodeGen/Mips/cmov.ll is
|
2012-02-28 15:46:26 +08:00
|
|
|
// compiled:
|
2011-06-08 02:00:14 +08:00
|
|
|
// movn %got(d)($gp), %got(c)($gp), $4
|
|
|
|
// This instruction is illegal since movn can take only register operands.
|
|
|
|
|
2012-02-25 06:34:47 +08:00
|
|
|
def MipsWrapper : SDNode<"MipsISD::Wrapper", SDTIntBinOp>;
|
2011-05-28 09:07:07 +08:00
|
|
|
|
2012-08-24 22:43:27 +08:00
|
|
|
def MipsSync : SDNode<"MipsISD::Sync", SDT_Sync, [SDNPHasChain,SDNPSideEffect]>;
|
2011-07-20 07:30:50 +08:00
|
|
|
|
2011-08-17 10:05:42 +08:00
|
|
|
def MipsExt : SDNode<"MipsISD::Ext", SDT_Ext>;
|
|
|
|
def MipsIns : SDNode<"MipsISD::Ins", SDT_Ins>;
|
|
|
|
|
2012-06-02 08:03:12 +08:00
|
|
|
def MipsLWL : SDNode<"MipsISD::LWL", SDTMipsLoadLR,
|
|
|
|
[SDNPHasChain, SDNPMayLoad, SDNPMemOperand]>;
|
|
|
|
def MipsLWR : SDNode<"MipsISD::LWR", SDTMipsLoadLR,
|
|
|
|
[SDNPHasChain, SDNPMayLoad, SDNPMemOperand]>;
|
|
|
|
def MipsSWL : SDNode<"MipsISD::SWL", SDTStore,
|
|
|
|
[SDNPHasChain, SDNPMayStore, SDNPMemOperand]>;
|
|
|
|
def MipsSWR : SDNode<"MipsISD::SWR", SDTStore,
|
|
|
|
[SDNPHasChain, SDNPMayStore, SDNPMemOperand]>;
|
|
|
|
def MipsLDL : SDNode<"MipsISD::LDL", SDTMipsLoadLR,
|
|
|
|
[SDNPHasChain, SDNPMayLoad, SDNPMemOperand]>;
|
|
|
|
def MipsLDR : SDNode<"MipsISD::LDR", SDTMipsLoadLR,
|
|
|
|
[SDNPHasChain, SDNPMayLoad, SDNPMemOperand]>;
|
|
|
|
def MipsSDL : SDNode<"MipsISD::SDL", SDTStore,
|
|
|
|
[SDNPHasChain, SDNPMayStore, SDNPMemOperand]>;
|
|
|
|
def MipsSDR : SDNode<"MipsISD::SDR", SDTStore,
|
|
|
|
[SDNPHasChain, SDNPMayStore, SDNPMemOperand]>;
|
|
|
|
|
2011-04-16 05:51:11 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2007-10-09 10:55:31 +08:00
|
|
|
// Mips Instruction Predicate Definitions.
|
2011-04-16 05:51:11 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2012-04-18 02:03:21 +08:00
|
|
|
def HasSEInReg : Predicate<"Subtarget.hasSEInReg()">,
|
|
|
|
AssemblerPredicate<"FeatureSEInReg">;
|
|
|
|
def HasBitCount : Predicate<"Subtarget.hasBitCount()">,
|
|
|
|
AssemblerPredicate<"FeatureBitCount">;
|
|
|
|
def HasSwap : Predicate<"Subtarget.hasSwap()">,
|
|
|
|
AssemblerPredicate<"FeatureSwap">;
|
|
|
|
def HasCondMov : Predicate<"Subtarget.hasCondMov()">,
|
|
|
|
AssemblerPredicate<"FeatureCondMov">;
|
2012-11-16 05:17:13 +08:00
|
|
|
def HasFPIdx : Predicate<"Subtarget.hasFPIdx()">,
|
|
|
|
AssemblerPredicate<"FeatureFPIdx">;
|
2012-04-18 02:03:21 +08:00
|
|
|
def HasMips32 : Predicate<"Subtarget.hasMips32()">,
|
|
|
|
AssemblerPredicate<"FeatureMips32">;
|
|
|
|
def HasMips32r2 : Predicate<"Subtarget.hasMips32r2()">,
|
|
|
|
AssemblerPredicate<"FeatureMips32r2">;
|
|
|
|
def HasMips64 : Predicate<"Subtarget.hasMips64()">,
|
|
|
|
AssemblerPredicate<"FeatureMips64">;
|
|
|
|
def NotMips64 : Predicate<"!Subtarget.hasMips64()">,
|
|
|
|
AssemblerPredicate<"!FeatureMips64">;
|
|
|
|
def HasMips64r2 : Predicate<"Subtarget.hasMips64r2()">,
|
|
|
|
AssemblerPredicate<"FeatureMips64r2">;
|
|
|
|
def IsN64 : Predicate<"Subtarget.isABI_N64()">,
|
|
|
|
AssemblerPredicate<"FeatureN64">;
|
|
|
|
def NotN64 : Predicate<"!Subtarget.isABI_N64()">,
|
|
|
|
AssemblerPredicate<"!FeatureN64">;
|
2012-05-25 02:32:33 +08:00
|
|
|
def InMips16Mode : Predicate<"Subtarget.inMips16Mode()">,
|
|
|
|
AssemblerPredicate<"FeatureMips16">;
|
2012-04-18 02:03:21 +08:00
|
|
|
def RelocStatic : Predicate<"TM.getRelocationModel() == Reloc::Static">,
|
|
|
|
AssemblerPredicate<"FeatureMips32">;
|
|
|
|
def RelocPIC : Predicate<"TM.getRelocationModel() == Reloc::PIC_">,
|
|
|
|
AssemblerPredicate<"FeatureMips32">;
|
|
|
|
def NoNaNsFPMath : Predicate<"TM.Options.NoNaNsFPMath">,
|
|
|
|
AssemblerPredicate<"FeatureMips32">;
|
2012-12-07 11:06:09 +08:00
|
|
|
def HasStdEnc : Predicate<"Subtarget.hasStandardEncoding()">,
|
|
|
|
AssemblerPredicate<"!FeatureMips16">;
|
2012-05-22 11:10:09 +08:00
|
|
|
|
2012-06-15 05:03:23 +08:00
|
|
|
class MipsPat<dag pattern, dag result> : Pat<pattern, result> {
|
2012-12-07 11:06:09 +08:00
|
|
|
let Predicates = [HasStdEnc];
|
2012-06-15 05:03:23 +08:00
|
|
|
}
|
|
|
|
|
2012-12-13 08:32:01 +08:00
|
|
|
class IsCommutable {
|
|
|
|
bit isCommutable = 1;
|
|
|
|
}
|
|
|
|
|
2012-10-20 05:11:03 +08:00
|
|
|
class IsBranch {
|
|
|
|
bit isBranch = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
class IsReturn {
|
|
|
|
bit isReturn = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
class IsCall {
|
|
|
|
bit isCall = 1;
|
|
|
|
}
|
|
|
|
|
2012-10-20 05:14:34 +08:00
|
|
|
class IsTailCall {
|
|
|
|
bit isCall = 1;
|
|
|
|
bit isTerminator = 1;
|
|
|
|
bit isReturn = 1;
|
|
|
|
bit isBarrier = 1;
|
|
|
|
bit hasExtraSrcRegAllocReq = 1;
|
|
|
|
bit isCodeGenOnly = 1;
|
|
|
|
}
|
|
|
|
|
2012-11-01 02:37:55 +08:00
|
|
|
class IsAsCheapAsAMove {
|
|
|
|
bit isAsCheapAsAMove = 1;
|
|
|
|
}
|
|
|
|
|
2012-11-03 08:53:12 +08:00
|
|
|
class NeverHasSideEffects {
|
|
|
|
bit neverHasSideEffects = 1;
|
|
|
|
}
|
|
|
|
|
2012-05-22 11:10:09 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Instruction format superclass
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
include "MipsInstrFormats.td"
|
2007-10-09 10:55:31 +08:00
|
|
|
|
2011-04-16 05:51:11 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2007-10-09 10:55:31 +08:00
|
|
|
// Mips Operand, Complex Patterns and Transformations Definitions.
|
2011-04-16 05:51:11 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2007-10-09 10:55:31 +08:00
|
|
|
|
2007-06-06 15:42:06 +08:00
|
|
|
// Instruction operand types
|
2011-11-12 06:58:42 +08:00
|
|
|
def jmptarget : Operand<OtherVT> {
|
|
|
|
let EncoderMethod = "getJumpTargetOpValue";
|
|
|
|
}
|
|
|
|
def brtarget : Operand<OtherVT> {
|
|
|
|
let EncoderMethod = "getBranchTargetOpValue";
|
|
|
|
let OperandType = "OPERAND_PCREL";
|
2012-04-18 02:03:21 +08:00
|
|
|
let DecoderMethod = "DecodeBranchTarget";
|
2011-11-12 06:58:42 +08:00
|
|
|
}
|
2011-11-24 06:19:28 +08:00
|
|
|
def calltarget : Operand<iPTR> {
|
|
|
|
let EncoderMethod = "getJumpTargetOpValue";
|
|
|
|
}
|
2011-11-11 12:03:54 +08:00
|
|
|
def calltarget64: Operand<i64>;
|
2012-04-18 02:03:21 +08:00
|
|
|
def simm16 : Operand<i32> {
|
|
|
|
let DecoderMethod= "DecodeSimm16";
|
|
|
|
}
|
2011-10-11 08:11:12 +08:00
|
|
|
def simm16_64 : Operand<i64>;
|
2007-10-26 12:00:13 +08:00
|
|
|
def shamt : Operand<i32>;
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2008-08-13 15:13:40 +08:00
|
|
|
// Unsigned Operand
|
|
|
|
def uimm16 : Operand<i32> {
|
|
|
|
let PrintMethod = "printUnsignedImm";
|
|
|
|
}
|
|
|
|
|
2012-08-18 04:16:42 +08:00
|
|
|
def MipsMemAsmOperand : AsmOperandClass {
|
|
|
|
let Name = "Mem";
|
|
|
|
let ParserMethod = "parseMemOperand";
|
|
|
|
}
|
|
|
|
|
2007-06-06 15:42:06 +08:00
|
|
|
// Address operand
|
|
|
|
def mem : Operand<i32> {
|
|
|
|
let PrintMethod = "printMemOperand";
|
2011-07-08 02:57:00 +08:00
|
|
|
let MIOperandInfo = (ops CPURegs, simm16);
|
2011-10-19 01:50:36 +08:00
|
|
|
let EncoderMethod = "getMemEncoding";
|
2012-08-18 04:16:42 +08:00
|
|
|
let ParserMatchClass = MipsMemAsmOperand;
|
2007-06-06 15:42:06 +08:00
|
|
|
}
|
|
|
|
|
2011-10-11 08:11:12 +08:00
|
|
|
def mem64 : Operand<i64> {
|
|
|
|
let PrintMethod = "printMemOperand";
|
|
|
|
let MIOperandInfo = (ops CPU64Regs, simm16_64);
|
2012-06-28 07:13:42 +08:00
|
|
|
let EncoderMethod = "getMemEncoding";
|
2012-08-18 04:16:42 +08:00
|
|
|
let ParserMatchClass = MipsMemAsmOperand;
|
2011-10-11 08:11:12 +08:00
|
|
|
}
|
|
|
|
|
2011-07-08 04:54:20 +08:00
|
|
|
def mem_ea : Operand<i32> {
|
|
|
|
let PrintMethod = "printMemOperandEA";
|
|
|
|
let MIOperandInfo = (ops CPURegs, simm16);
|
2011-10-19 01:50:36 +08:00
|
|
|
let EncoderMethod = "getMemEncoding";
|
|
|
|
}
|
|
|
|
|
2011-11-11 12:06:38 +08:00
|
|
|
def mem_ea_64 : Operand<i64> {
|
|
|
|
let PrintMethod = "printMemOperandEA";
|
|
|
|
let MIOperandInfo = (ops CPU64Regs, simm16_64);
|
|
|
|
let EncoderMethod = "getMemEncoding";
|
|
|
|
}
|
|
|
|
|
2011-10-19 01:50:36 +08:00
|
|
|
// size operand of ext instruction
|
|
|
|
def size_ext : Operand<i32> {
|
|
|
|
let EncoderMethod = "getSizeExtEncoding";
|
2012-04-18 02:03:21 +08:00
|
|
|
let DecoderMethod = "DecodeExtSize";
|
2011-10-19 01:50:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// size operand of ins instruction
|
|
|
|
def size_ins : Operand<i32> {
|
|
|
|
let EncoderMethod = "getSizeInsEncoding";
|
2012-04-18 02:03:21 +08:00
|
|
|
let DecoderMethod = "DecodeInsSize";
|
2011-07-08 04:54:20 +08:00
|
|
|
}
|
|
|
|
|
2007-06-06 15:42:06 +08:00
|
|
|
// Transformation Function - get the lower 16 bits.
|
|
|
|
def LO16 : SDNodeXForm<imm, [{
|
2011-12-08 04:10:24 +08:00
|
|
|
return getImm(N, N->getZExtValue() & 0xFFFF);
|
2007-06-06 15:42:06 +08:00
|
|
|
}]>;
|
|
|
|
|
|
|
|
// Transformation Function - get the higher 16 bits.
|
|
|
|
def HI16 : SDNodeXForm<imm, [{
|
2011-12-08 04:10:24 +08:00
|
|
|
return getImm(N, (N->getZExtValue() >> 16) & 0xFFFF);
|
2007-06-06 15:42:06 +08:00
|
|
|
}]>;
|
|
|
|
|
|
|
|
// Node immediate fits as 16-bit sign extended on target immediate.
|
|
|
|
// e.g. addi, andi
|
2010-08-19 07:56:46 +08:00
|
|
|
def immSExt16 : PatLeaf<(imm), [{ return isInt<16>(N->getSExtValue()); }]>;
|
2007-06-06 15:42:06 +08:00
|
|
|
|
|
|
|
// Node immediate fits as 16-bit zero extended on target immediate.
|
|
|
|
// The LO16 param means that only the lower 16 bits of the node
|
|
|
|
// immediate are caught.
|
|
|
|
// e.g. addiu, sltiu
|
|
|
|
def immZExt16 : PatLeaf<(imm), [{
|
2009-08-12 04:47:22 +08:00
|
|
|
if (N->getValueType(0) == MVT::i32)
|
2008-09-13 00:56:44 +08:00
|
|
|
return (uint32_t)N->getZExtValue() == (unsigned short)N->getZExtValue();
|
2007-10-26 12:00:13 +08:00
|
|
|
else
|
2008-09-13 00:56:44 +08:00
|
|
|
return (uint64_t)N->getZExtValue() == (unsigned short)N->getZExtValue();
|
2007-06-06 15:42:06 +08:00
|
|
|
}], LO16>;
|
|
|
|
|
2011-12-20 04:21:18 +08:00
|
|
|
// Immediate can be loaded with LUi (32-bit int with lower 16-bit cleared).
|
2012-01-04 11:09:26 +08:00
|
|
|
def immLow16Zero : PatLeaf<(imm), [{
|
2011-12-20 04:21:18 +08:00
|
|
|
int64_t Val = N->getSExtValue();
|
|
|
|
return isInt<32>(Val) && !(Val & 0xffff);
|
|
|
|
}]>;
|
|
|
|
|
2007-06-06 15:42:06 +08:00
|
|
|
// shamt field must fit in 5 bits.
|
2011-10-18 02:01:00 +08:00
|
|
|
def immZExt5 : ImmLeaf<i32, [{return Imm == (Imm & 0x1f);}]>;
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2007-10-26 12:00:13 +08:00
|
|
|
// Mips Address Mode! SDNode frameindex could possibily be a match
|
2007-06-06 15:42:06 +08:00
|
|
|
// since load and store instructions from stack used it.
|
2012-05-25 02:32:33 +08:00
|
|
|
def addr :
|
|
|
|
ComplexPattern<iPTR, 2, "SelectAddr", [frameindex], [SDNPWantParent]>;
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2011-04-16 05:51:11 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2007-06-06 15:42:06 +08:00
|
|
|
// Instructions specific format
|
2011-04-16 05:51:11 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2012-10-06 09:17:37 +08:00
|
|
|
/// Move Control Registers From/To CPU Registers
|
|
|
|
def MFC0_3OP : MFC3OP<0x10, 0, (outs CPURegs:$rt),
|
|
|
|
(ins CPURegs:$rd, uimm16:$sel),"mfc0\t$rt, $rd, $sel">;
|
|
|
|
def : InstAlias<"mfc0 $rt, $rd", (MFC0_3OP CPURegs:$rt, CPURegs:$rd, 0)>;
|
|
|
|
|
|
|
|
def MTC0_3OP : MFC3OP<0x10, 4, (outs CPURegs:$rd, uimm16:$sel),
|
|
|
|
(ins CPURegs:$rt),"mtc0\t$rt, $rd, $sel">;
|
|
|
|
def : InstAlias<"mtc0 $rt, $rd", (MTC0_3OP CPURegs:$rd, 0, CPURegs:$rt)>;
|
|
|
|
|
|
|
|
def MFC2_3OP : MFC3OP<0x12, 0, (outs CPURegs:$rt),
|
|
|
|
(ins CPURegs:$rd, uimm16:$sel),"mfc2\t$rt, $rd, $sel">;
|
|
|
|
def : InstAlias<"mfc2 $rt, $rd", (MFC2_3OP CPURegs:$rt, CPURegs:$rd, 0)>;
|
|
|
|
|
|
|
|
def MTC2_3OP : MFC3OP<0x12, 4, (outs CPURegs:$rd, uimm16:$sel),
|
|
|
|
(ins CPURegs:$rt),"mtc2\t$rt, $rd, $sel">;
|
|
|
|
def : InstAlias<"mtc2 $rt, $rd", (MTC2_3OP CPURegs:$rd, 0, CPURegs:$rt)>;
|
|
|
|
|
2011-10-12 07:12:12 +08:00
|
|
|
// Arithmetic and logical instructions with 3 register operands.
|
2011-10-12 07:05:46 +08:00
|
|
|
class ArithLogicR<bits<6> op, bits<6> func, string instr_asm, SDNode OpNode,
|
|
|
|
InstrItinClass itin, RegisterClass RC, bit isComm = 0>:
|
|
|
|
FR<op, func, (outs RC:$rd), (ins RC:$rs, RC:$rt),
|
|
|
|
!strconcat(instr_asm, "\t$rd, $rs, $rt"),
|
|
|
|
[(set RC:$rd, (OpNode RC:$rs, RC:$rt))], itin> {
|
|
|
|
let shamt = 0;
|
2011-05-13 01:42:08 +08:00
|
|
|
let isCommutable = isComm;
|
2012-04-19 02:52:10 +08:00
|
|
|
let isReMaterializable = 1;
|
2011-05-13 01:42:08 +08:00
|
|
|
}
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2011-10-12 07:43:48 +08:00
|
|
|
class ArithOverflowR<bits<6> op, bits<6> func, string instr_asm,
|
2011-10-12 07:05:46 +08:00
|
|
|
InstrItinClass itin, RegisterClass RC, bit isComm = 0>:
|
|
|
|
FR<op, func, (outs RC:$rd), (ins RC:$rs, RC:$rt),
|
|
|
|
!strconcat(instr_asm, "\t$rd, $rs, $rt"), [], itin> {
|
|
|
|
let shamt = 0;
|
2011-05-13 01:42:08 +08:00
|
|
|
let isCommutable = isComm;
|
|
|
|
}
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2011-10-12 07:38:52 +08:00
|
|
|
// Arithmetic and logical instructions with 2 register operands.
|
|
|
|
class ArithLogicI<bits<6> op, string instr_asm, SDNode OpNode,
|
|
|
|
Operand Od, PatLeaf imm_type, RegisterClass RC> :
|
2011-10-19 01:50:36 +08:00
|
|
|
FI<op, (outs RC:$rt), (ins RC:$rs, Od:$imm16),
|
|
|
|
!strconcat(instr_asm, "\t$rt, $rs, $imm16"),
|
2012-04-19 02:52:10 +08:00
|
|
|
[(set RC:$rt, (OpNode RC:$rs, imm_type:$imm16))], IIAlu> {
|
|
|
|
let isReMaterializable = 1;
|
|
|
|
}
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2008-08-13 15:13:40 +08:00
|
|
|
class ArithOverflowI<bits<6> op, string instr_asm, SDNode OpNode,
|
2011-10-12 07:38:52 +08:00
|
|
|
Operand Od, PatLeaf imm_type, RegisterClass RC> :
|
2011-10-19 01:50:36 +08:00
|
|
|
FI<op, (outs RC:$rt), (ins RC:$rs, Od:$imm16),
|
|
|
|
!strconcat(instr_asm, "\t$rt, $rs, $imm16"), [], IIAlu>;
|
2008-08-13 15:13:40 +08:00
|
|
|
|
2007-06-06 15:42:06 +08:00
|
|
|
// Arithmetic Multiply ADD/SUB
|
2011-01-19 03:29:17 +08:00
|
|
|
let rd = 0, shamt = 0, Defs = [HI, LO], Uses = [HI, LO] in
|
2011-05-13 01:42:08 +08:00
|
|
|
class MArithR<bits<6> func, string instr_asm, SDNode op, bit isComm = 0> :
|
2011-01-19 03:29:17 +08:00
|
|
|
FR<0x1c, func, (outs), (ins CPURegs:$rs, CPURegs:$rt),
|
2011-03-05 01:51:39 +08:00
|
|
|
!strconcat(instr_asm, "\t$rs, $rt"),
|
2011-05-13 01:42:08 +08:00
|
|
|
[(op CPURegs:$rs, CPURegs:$rt, LO, HI)], IIImul> {
|
2011-10-12 08:56:06 +08:00
|
|
|
let rd = 0;
|
|
|
|
let shamt = 0;
|
2011-05-13 01:42:08 +08:00
|
|
|
let isCommutable = isComm;
|
|
|
|
}
|
2007-06-06 15:42:06 +08:00
|
|
|
|
|
|
|
// Logical
|
2011-10-12 09:05:13 +08:00
|
|
|
class LogicNOR<bits<6> op, bits<6> func, string instr_asm, RegisterClass RC>:
|
|
|
|
FR<op, func, (outs RC:$rd), (ins RC:$rs, RC:$rt),
|
2011-10-12 08:56:06 +08:00
|
|
|
!strconcat(instr_asm, "\t$rd, $rs, $rt"),
|
2011-10-12 09:05:13 +08:00
|
|
|
[(set RC:$rd, (not (or RC:$rs, RC:$rt)))], IIAlu> {
|
2011-10-12 08:56:06 +08:00
|
|
|
let shamt = 0;
|
|
|
|
let isCommutable = 1;
|
|
|
|
}
|
2007-06-06 15:42:06 +08:00
|
|
|
|
|
|
|
// Shifts
|
2011-10-18 02:06:56 +08:00
|
|
|
class shift_rotate_imm<bits<6> func, bits<5> isRotate, string instr_asm,
|
|
|
|
SDNode OpNode, PatFrag PF, Operand ImmOpnd,
|
|
|
|
RegisterClass RC>:
|
|
|
|
FR<0x00, func, (outs RC:$rd), (ins RC:$rt, ImmOpnd:$shamt),
|
2011-10-12 08:56:06 +08:00
|
|
|
!strconcat(instr_asm, "\t$rd, $rt, $shamt"),
|
2011-10-18 02:06:56 +08:00
|
|
|
[(set RC:$rd, (OpNode RC:$rt, PF:$shamt))], IIAlu> {
|
|
|
|
let rs = isRotate;
|
2010-12-10 01:32:30 +08:00
|
|
|
}
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2011-10-18 02:06:56 +08:00
|
|
|
// 32-bit shift instructions.
|
|
|
|
class shift_rotate_imm32<bits<6> func, bits<5> isRotate, string instr_asm,
|
|
|
|
SDNode OpNode>:
|
|
|
|
shift_rotate_imm<func, isRotate, instr_asm, OpNode, immZExt5, shamt, CPURegs>;
|
|
|
|
|
2011-10-18 02:17:58 +08:00
|
|
|
class shift_rotate_reg<bits<6> func, bits<5> isRotate, string instr_asm,
|
|
|
|
SDNode OpNode, RegisterClass RC>:
|
2011-11-08 02:59:49 +08:00
|
|
|
FR<0x00, func, (outs RC:$rd), (ins CPURegs:$rs, RC:$rt),
|
2011-10-12 08:56:06 +08:00
|
|
|
!strconcat(instr_asm, "\t$rd, $rt, $rs"),
|
2011-11-08 02:59:49 +08:00
|
|
|
[(set RC:$rd, (OpNode RC:$rt, CPURegs:$rs))], IIAlu> {
|
2011-10-12 08:56:06 +08:00
|
|
|
let shamt = isRotate;
|
2010-12-10 01:32:30 +08:00
|
|
|
}
|
2007-06-06 15:42:06 +08:00
|
|
|
|
|
|
|
// Load Upper Imediate
|
2011-11-08 03:10:49 +08:00
|
|
|
class LoadUpper<bits<6> op, string instr_asm, RegisterClass RC, Operand Imm>:
|
|
|
|
FI<op, (outs RC:$rt), (ins Imm:$imm16),
|
2012-11-03 08:26:02 +08:00
|
|
|
!strconcat(instr_asm, "\t$rt, $imm16"), [], IIAlu>, IsAsCheapAsAMove {
|
2011-10-12 08:56:06 +08:00
|
|
|
let rs = 0;
|
2012-04-03 10:51:09 +08:00
|
|
|
let neverHasSideEffects = 1;
|
2012-04-19 02:52:10 +08:00
|
|
|
let isReMaterializable = 1;
|
2011-10-12 08:56:06 +08:00
|
|
|
}
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2011-10-19 01:50:36 +08:00
|
|
|
class FMem<bits<6> op, dag outs, dag ins, string asmstr, list<dag> pattern,
|
|
|
|
InstrItinClass itin>: FFI<op, outs, ins, asmstr, pattern> {
|
|
|
|
bits<21> addr;
|
|
|
|
let Inst{25-21} = addr{20-16};
|
|
|
|
let Inst{15-0} = addr{15-0};
|
2012-04-18 02:03:21 +08:00
|
|
|
let DecoderMethod = "DecodeMem";
|
2011-10-19 01:50:36 +08:00
|
|
|
}
|
|
|
|
|
2007-10-26 12:00:13 +08:00
|
|
|
// Memory Load/Store
|
2011-09-10 04:45:50 +08:00
|
|
|
let canFoldAsLoad = 1 in
|
2011-10-11 08:11:12 +08:00
|
|
|
class LoadM<bits<6> op, string instr_asm, PatFrag OpNode, RegisterClass RC,
|
|
|
|
Operand MemOpnd, bit Pseudo>:
|
2011-10-19 01:50:36 +08:00
|
|
|
FMem<op, (outs RC:$rt), (ins MemOpnd:$addr),
|
2011-10-12 08:56:06 +08:00
|
|
|
!strconcat(instr_asm, "\t$rt, $addr"),
|
|
|
|
[(set RC:$rt, (OpNode addr:$addr))], IILoad> {
|
2011-10-08 10:24:10 +08:00
|
|
|
let isPseudo = Pseudo;
|
|
|
|
}
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2011-10-11 08:11:12 +08:00
|
|
|
class StoreM<bits<6> op, string instr_asm, PatFrag OpNode, RegisterClass RC,
|
|
|
|
Operand MemOpnd, bit Pseudo>:
|
2011-10-19 01:50:36 +08:00
|
|
|
FMem<op, (outs), (ins RC:$rt, MemOpnd:$addr),
|
2011-10-12 08:56:06 +08:00
|
|
|
!strconcat(instr_asm, "\t$rt, $addr"),
|
|
|
|
[(OpNode RC:$rt, addr:$addr)], IIStore> {
|
2011-10-08 10:24:10 +08:00
|
|
|
let isPseudo = Pseudo;
|
|
|
|
}
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2011-10-11 08:11:12 +08:00
|
|
|
// 32-bit load.
|
|
|
|
multiclass LoadM32<bits<6> op, string instr_asm, PatFrag OpNode,
|
|
|
|
bit Pseudo = 0> {
|
|
|
|
def #NAME# : LoadM<op, instr_asm, OpNode, CPURegs, mem, Pseudo>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[NotN64, HasStdEnc]>;
|
2011-10-11 08:11:12 +08:00
|
|
|
def _P8 : LoadM<op, instr_asm, OpNode, CPURegs, mem64, Pseudo>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[IsN64, HasStdEnc]> {
|
2012-04-18 02:03:21 +08:00
|
|
|
let DecoderNamespace = "Mips64";
|
|
|
|
let isCodeGenOnly = 1;
|
|
|
|
}
|
2012-02-28 15:46:26 +08:00
|
|
|
}
|
2011-10-11 08:11:12 +08:00
|
|
|
|
|
|
|
// 64-bit load.
|
|
|
|
multiclass LoadM64<bits<6> op, string instr_asm, PatFrag OpNode,
|
|
|
|
bit Pseudo = 0> {
|
|
|
|
def #NAME# : LoadM<op, instr_asm, OpNode, CPU64Regs, mem, Pseudo>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[NotN64, HasStdEnc]>;
|
2011-10-11 08:11:12 +08:00
|
|
|
def _P8 : LoadM<op, instr_asm, OpNode, CPU64Regs, mem64, Pseudo>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[IsN64, HasStdEnc]> {
|
2012-04-18 02:03:21 +08:00
|
|
|
let DecoderNamespace = "Mips64";
|
|
|
|
let isCodeGenOnly = 1;
|
|
|
|
}
|
2012-02-28 15:46:26 +08:00
|
|
|
}
|
2011-10-11 08:11:12 +08:00
|
|
|
|
|
|
|
// 32-bit store.
|
|
|
|
multiclass StoreM32<bits<6> op, string instr_asm, PatFrag OpNode,
|
|
|
|
bit Pseudo = 0> {
|
|
|
|
def #NAME# : StoreM<op, instr_asm, OpNode, CPURegs, mem, Pseudo>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[NotN64, HasStdEnc]>;
|
2011-10-11 08:11:12 +08:00
|
|
|
def _P8 : StoreM<op, instr_asm, OpNode, CPURegs, mem64, Pseudo>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[IsN64, HasStdEnc]> {
|
2012-04-18 02:03:21 +08:00
|
|
|
let DecoderNamespace = "Mips64";
|
|
|
|
let isCodeGenOnly = 1;
|
|
|
|
}
|
2011-10-11 08:11:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// 64-bit store.
|
|
|
|
multiclass StoreM64<bits<6> op, string instr_asm, PatFrag OpNode,
|
|
|
|
bit Pseudo = 0> {
|
|
|
|
def #NAME# : StoreM<op, instr_asm, OpNode, CPU64Regs, mem, Pseudo>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[NotN64, HasStdEnc]>;
|
2011-10-11 08:11:12 +08:00
|
|
|
def _P8 : StoreM<op, instr_asm, OpNode, CPU64Regs, mem64, Pseudo>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[IsN64, HasStdEnc]> {
|
2012-04-18 02:03:21 +08:00
|
|
|
let DecoderNamespace = "Mips64";
|
|
|
|
let isCodeGenOnly = 1;
|
|
|
|
}
|
2011-10-11 08:11:12 +08:00
|
|
|
}
|
|
|
|
|
2012-06-02 08:04:19 +08:00
|
|
|
// Load/Store Left/Right
|
|
|
|
let canFoldAsLoad = 1 in
|
|
|
|
class LoadLeftRight<bits<6> op, string instr_asm, SDNode OpNode,
|
|
|
|
RegisterClass RC, Operand MemOpnd> :
|
|
|
|
FMem<op, (outs RC:$rt), (ins MemOpnd:$addr, RC:$src),
|
|
|
|
!strconcat(instr_asm, "\t$rt, $addr"),
|
|
|
|
[(set RC:$rt, (OpNode addr:$addr, RC:$src))], IILoad> {
|
|
|
|
string Constraints = "$src = $rt";
|
|
|
|
}
|
|
|
|
|
|
|
|
class StoreLeftRight<bits<6> op, string instr_asm, SDNode OpNode,
|
|
|
|
RegisterClass RC, Operand MemOpnd>:
|
|
|
|
FMem<op, (outs), (ins RC:$rt, MemOpnd:$addr),
|
|
|
|
!strconcat(instr_asm, "\t$rt, $addr"), [(OpNode RC:$rt, addr:$addr)],
|
|
|
|
IIStore>;
|
|
|
|
|
|
|
|
// 32-bit load left/right.
|
|
|
|
multiclass LoadLeftRightM32<bits<6> op, string instr_asm, SDNode OpNode> {
|
|
|
|
def #NAME# : LoadLeftRight<op, instr_asm, OpNode, CPURegs, mem>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[NotN64, HasStdEnc]>;
|
2012-06-02 08:04:19 +08:00
|
|
|
def _P8 : LoadLeftRight<op, instr_asm, OpNode, CPURegs, mem64>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[IsN64, HasStdEnc]> {
|
2012-06-02 08:04:19 +08:00
|
|
|
let DecoderNamespace = "Mips64";
|
|
|
|
let isCodeGenOnly = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// 64-bit load left/right.
|
|
|
|
multiclass LoadLeftRightM64<bits<6> op, string instr_asm, SDNode OpNode> {
|
|
|
|
def #NAME# : LoadLeftRight<op, instr_asm, OpNode, CPU64Regs, mem>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[NotN64, HasStdEnc]>;
|
2012-06-02 08:04:19 +08:00
|
|
|
def _P8 : LoadLeftRight<op, instr_asm, OpNode, CPU64Regs, mem64>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[IsN64, HasStdEnc]> {
|
2012-06-02 08:04:19 +08:00
|
|
|
let DecoderNamespace = "Mips64";
|
|
|
|
let isCodeGenOnly = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// 32-bit store left/right.
|
|
|
|
multiclass StoreLeftRightM32<bits<6> op, string instr_asm, SDNode OpNode> {
|
|
|
|
def #NAME# : StoreLeftRight<op, instr_asm, OpNode, CPURegs, mem>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[NotN64, HasStdEnc]>;
|
2012-06-02 08:04:19 +08:00
|
|
|
def _P8 : StoreLeftRight<op, instr_asm, OpNode, CPURegs, mem64>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[IsN64, HasStdEnc]> {
|
2012-06-02 08:04:19 +08:00
|
|
|
let DecoderNamespace = "Mips64";
|
|
|
|
let isCodeGenOnly = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// 64-bit store left/right.
|
|
|
|
multiclass StoreLeftRightM64<bits<6> op, string instr_asm, SDNode OpNode> {
|
|
|
|
def #NAME# : StoreLeftRight<op, instr_asm, OpNode, CPU64Regs, mem>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[NotN64, HasStdEnc]>;
|
2012-06-02 08:04:19 +08:00
|
|
|
def _P8 : StoreLeftRight<op, instr_asm, OpNode, CPU64Regs, mem64>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[IsN64, HasStdEnc]> {
|
2012-04-18 02:03:21 +08:00
|
|
|
let DecoderNamespace = "Mips64";
|
|
|
|
let isCodeGenOnly = 1;
|
|
|
|
}
|
2011-11-24 06:19:28 +08:00
|
|
|
}
|
|
|
|
|
2007-06-06 15:42:06 +08:00
|
|
|
// Conditional Branch
|
2011-10-12 02:49:17 +08:00
|
|
|
class CBranch<bits<6> op, string instr_asm, PatFrag cond_op, RegisterClass RC>:
|
2011-12-06 11:34:48 +08:00
|
|
|
BranchBase<op, (outs), (ins RC:$rs, RC:$rt, brtarget:$imm16),
|
|
|
|
!strconcat(instr_asm, "\t$rs, $rt, $imm16"),
|
|
|
|
[(brcond (i32 (cond_op RC:$rs, RC:$rt)), bb:$imm16)], IIBranch> {
|
2011-10-12 02:49:17 +08:00
|
|
|
let isBranch = 1;
|
|
|
|
let isTerminator = 1;
|
|
|
|
let hasDelaySlot = 1;
|
2012-06-14 09:17:59 +08:00
|
|
|
let Defs = [AT];
|
2011-10-12 02:49:17 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
class CBranchZero<bits<6> op, bits<5> _rt, string instr_asm, PatFrag cond_op,
|
|
|
|
RegisterClass RC>:
|
2011-12-06 11:34:48 +08:00
|
|
|
BranchBase<op, (outs), (ins RC:$rs, brtarget:$imm16),
|
|
|
|
!strconcat(instr_asm, "\t$rs, $imm16"),
|
|
|
|
[(brcond (i32 (cond_op RC:$rs, 0)), bb:$imm16)], IIBranch> {
|
2011-10-12 02:49:17 +08:00
|
|
|
let rt = _rt;
|
|
|
|
let isBranch = 1;
|
|
|
|
let isTerminator = 1;
|
|
|
|
let hasDelaySlot = 1;
|
2012-06-14 09:17:59 +08:00
|
|
|
let Defs = [AT];
|
2007-10-26 12:00:13 +08:00
|
|
|
}
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2007-10-26 12:00:13 +08:00
|
|
|
// SetCC
|
2011-10-12 02:53:46 +08:00
|
|
|
class SetCC_R<bits<6> op, bits<6> func, string instr_asm, PatFrag cond_op,
|
|
|
|
RegisterClass RC>:
|
|
|
|
FR<op, func, (outs CPURegs:$rd), (ins RC:$rs, RC:$rt),
|
|
|
|
!strconcat(instr_asm, "\t$rd, $rs, $rt"),
|
|
|
|
[(set CPURegs:$rd, (cond_op RC:$rs, RC:$rt))],
|
2011-10-12 08:56:06 +08:00
|
|
|
IIAlu> {
|
|
|
|
let shamt = 0;
|
|
|
|
}
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2011-10-12 02:53:46 +08:00
|
|
|
class SetCC_I<bits<6> op, string instr_asm, PatFrag cond_op, Operand Od,
|
|
|
|
PatLeaf imm_type, RegisterClass RC>:
|
2011-10-19 01:50:36 +08:00
|
|
|
FI<op, (outs CPURegs:$rt), (ins RC:$rs, Od:$imm16),
|
|
|
|
!strconcat(instr_asm, "\t$rt, $rs, $imm16"),
|
|
|
|
[(set CPURegs:$rt, (cond_op RC:$rs, imm_type:$imm16))],
|
2010-11-10 01:25:34 +08:00
|
|
|
IIAlu>;
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2011-12-13 06:39:35 +08:00
|
|
|
// Jump
|
2012-10-20 05:30:15 +08:00
|
|
|
class JumpFJ<bits<6> op, DAGOperand opnd, string instr_asm,
|
|
|
|
SDPatternOperator operator, SDPatternOperator targetoperator>:
|
|
|
|
FJ<op, (outs), (ins opnd:$target), !strconcat(instr_asm, "\t$target"),
|
|
|
|
[(operator targetoperator:$target)], IIBranch> {
|
2011-12-13 06:39:35 +08:00
|
|
|
let isTerminator=1;
|
|
|
|
let isBarrier=1;
|
|
|
|
let hasDelaySlot = 1;
|
2012-04-18 02:03:21 +08:00
|
|
|
let DecoderMethod = "DecodeJumpTarget";
|
2012-06-14 09:17:59 +08:00
|
|
|
let Defs = [AT];
|
2011-12-13 06:39:35 +08:00
|
|
|
}
|
|
|
|
|
2007-06-06 15:42:06 +08:00
|
|
|
// Unconditional branch
|
2011-12-06 11:34:48 +08:00
|
|
|
class UncondBranch<bits<6> op, string instr_asm>:
|
|
|
|
BranchBase<op, (outs), (ins brtarget:$imm16),
|
|
|
|
!strconcat(instr_asm, "\t$imm16"), [(br bb:$imm16)], IIBranch> {
|
|
|
|
let rs = 0;
|
|
|
|
let rt = 0;
|
|
|
|
let isBranch = 1;
|
|
|
|
let isTerminator = 1;
|
|
|
|
let isBarrier = 1;
|
|
|
|
let hasDelaySlot = 1;
|
2012-12-07 11:06:09 +08:00
|
|
|
let Predicates = [RelocPIC, HasStdEnc];
|
2012-06-14 09:17:59 +08:00
|
|
|
let Defs = [AT];
|
2011-12-06 11:34:48 +08:00
|
|
|
}
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2012-07-10 08:19:06 +08:00
|
|
|
// Base class for indirect branch and return instruction classes.
|
|
|
|
let isTerminator=1, isBarrier=1, hasDelaySlot = 1 in
|
2012-10-20 05:11:03 +08:00
|
|
|
class JumpFR<RegisterClass RC, SDPatternOperator operator = null_frag>:
|
|
|
|
FR<0, 0x8, (outs), (ins RC:$rs), "jr\t$rs", [(operator RC:$rs)], IIBranch> {
|
2011-10-12 08:56:06 +08:00
|
|
|
let rt = 0;
|
|
|
|
let rd = 0;
|
|
|
|
let shamt = 0;
|
|
|
|
}
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2012-07-10 08:19:06 +08:00
|
|
|
// Indirect branch
|
2012-10-20 05:11:03 +08:00
|
|
|
class IndirectBranch<RegisterClass RC>: JumpFR<RC, brind> {
|
2012-07-10 08:19:06 +08:00
|
|
|
let isBranch = 1;
|
|
|
|
let isIndirectBranch = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return instruction
|
2012-10-20 05:11:03 +08:00
|
|
|
class RetBase<RegisterClass RC>: JumpFR<RC> {
|
2012-07-10 08:19:06 +08:00
|
|
|
let isReturn = 1;
|
|
|
|
let isCodeGenOnly = 1;
|
|
|
|
let hasCtrlDep = 1;
|
|
|
|
let hasExtraSrcRegAllocReq = 1;
|
|
|
|
}
|
|
|
|
|
2007-06-06 15:42:06 +08:00
|
|
|
// Jump and Link (Call)
|
2012-07-10 08:19:06 +08:00
|
|
|
let isCall=1, hasDelaySlot=1, Defs = [RA] in {
|
2007-10-26 12:00:13 +08:00
|
|
|
class JumpLink<bits<6> op, string instr_asm>:
|
2012-07-14 04:44:29 +08:00
|
|
|
FJ<op, (outs), (ins calltarget:$target),
|
2010-11-10 01:25:34 +08:00
|
|
|
!strconcat(instr_asm, "\t$target"), [(MipsJmpLink imm:$target)],
|
2012-04-18 02:03:21 +08:00
|
|
|
IIBranch> {
|
|
|
|
let DecoderMethod = "DecodeJumpTarget";
|
|
|
|
}
|
2007-08-18 10:37:46 +08:00
|
|
|
|
2012-01-04 11:02:47 +08:00
|
|
|
class JumpLinkReg<bits<6> op, bits<6> func, string instr_asm,
|
|
|
|
RegisterClass RC>:
|
2012-07-14 04:44:29 +08:00
|
|
|
FR<op, func, (outs), (ins RC:$rs),
|
2012-01-04 11:02:47 +08:00
|
|
|
!strconcat(instr_asm, "\t$rs"), [(MipsJmpLink RC:$rs)], IIBranch> {
|
2011-10-12 08:56:06 +08:00
|
|
|
let rt = 0;
|
|
|
|
let rd = 31;
|
|
|
|
let shamt = 0;
|
|
|
|
}
|
2007-08-18 10:37:46 +08:00
|
|
|
|
2012-01-04 11:02:47 +08:00
|
|
|
class BranchLink<string instr_asm, bits<5> _rt, RegisterClass RC>:
|
2012-07-14 04:44:29 +08:00
|
|
|
FI<0x1, (outs), (ins RC:$rs, brtarget:$imm16),
|
2012-01-04 11:02:47 +08:00
|
|
|
!strconcat(instr_asm, "\t$rs, $imm16"), [], IIBranch> {
|
|
|
|
let rt = _rt;
|
|
|
|
}
|
2007-08-18 10:37:46 +08:00
|
|
|
}
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2007-10-26 12:00:13 +08:00
|
|
|
// Mul, Div
|
2011-10-18 02:21:24 +08:00
|
|
|
class Mult<bits<6> func, string instr_asm, InstrItinClass itin,
|
|
|
|
RegisterClass RC, list<Register> DefRegs>:
|
|
|
|
FR<0x00, func, (outs), (ins RC:$rs, RC:$rt),
|
2011-10-12 08:56:06 +08:00
|
|
|
!strconcat(instr_asm, "\t$rs, $rt"), [], itin> {
|
|
|
|
let rd = 0;
|
|
|
|
let shamt = 0;
|
|
|
|
let isCommutable = 1;
|
2011-10-18 02:21:24 +08:00
|
|
|
let Defs = DefRegs;
|
2012-04-03 10:51:09 +08:00
|
|
|
let neverHasSideEffects = 1;
|
2011-10-12 08:56:06 +08:00
|
|
|
}
|
2011-03-05 05:03:24 +08:00
|
|
|
|
2011-10-18 02:21:24 +08:00
|
|
|
class Mult32<bits<6> func, string instr_asm, InstrItinClass itin>:
|
|
|
|
Mult<func, instr_asm, itin, CPURegs, [HI, LO]>;
|
|
|
|
|
|
|
|
class Div<SDNode op, bits<6> func, string instr_asm, InstrItinClass itin,
|
|
|
|
RegisterClass RC, list<Register> DefRegs>:
|
|
|
|
FR<0x00, func, (outs), (ins RC:$rs, RC:$rt),
|
|
|
|
!strconcat(instr_asm, "\t$$zero, $rs, $rt"),
|
|
|
|
[(op RC:$rs, RC:$rt)], itin> {
|
2011-10-12 08:56:06 +08:00
|
|
|
let rd = 0;
|
|
|
|
let shamt = 0;
|
2011-10-18 02:21:24 +08:00
|
|
|
let Defs = DefRegs;
|
2011-03-05 05:03:24 +08:00
|
|
|
}
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2011-10-18 02:21:24 +08:00
|
|
|
class Div32<SDNode op, bits<6> func, string instr_asm, InstrItinClass itin>:
|
|
|
|
Div<op, func, instr_asm, itin, CPURegs, [HI, LO]>;
|
|
|
|
|
2007-10-26 12:00:13 +08:00
|
|
|
// Move from Hi/Lo
|
2011-10-18 02:24:15 +08:00
|
|
|
class MoveFromLOHI<bits<6> func, string instr_asm, RegisterClass RC,
|
|
|
|
list<Register> UseRegs>:
|
|
|
|
FR<0x00, func, (outs RC:$rd), (ins),
|
2011-10-12 08:56:06 +08:00
|
|
|
!strconcat(instr_asm, "\t$rd"), [], IIHiLo> {
|
|
|
|
let rs = 0;
|
|
|
|
let rt = 0;
|
|
|
|
let shamt = 0;
|
2011-10-18 02:24:15 +08:00
|
|
|
let Uses = UseRegs;
|
2012-04-03 10:51:09 +08:00
|
|
|
let neverHasSideEffects = 1;
|
2011-10-12 08:56:06 +08:00
|
|
|
}
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2011-10-18 02:24:15 +08:00
|
|
|
class MoveToLOHI<bits<6> func, string instr_asm, RegisterClass RC,
|
|
|
|
list<Register> DefRegs>:
|
|
|
|
FR<0x00, func, (outs), (ins RC:$rs),
|
2011-10-12 08:56:06 +08:00
|
|
|
!strconcat(instr_asm, "\t$rs"), [], IIHiLo> {
|
|
|
|
let rt = 0;
|
|
|
|
let rd = 0;
|
|
|
|
let shamt = 0;
|
2011-10-18 02:24:15 +08:00
|
|
|
let Defs = DefRegs;
|
2012-04-03 10:51:09 +08:00
|
|
|
let neverHasSideEffects = 1;
|
2011-10-04 03:28:44 +08:00
|
|
|
}
|
2008-08-03 03:42:36 +08:00
|
|
|
|
2012-08-07 07:29:06 +08:00
|
|
|
class EffectiveAddress<bits<6> opc, string instr_asm, RegisterClass RC, Operand Mem> :
|
|
|
|
FMem<opc, (outs RC:$rt), (ins Mem:$addr),
|
|
|
|
instr_asm, [(set RC:$rt, addr:$addr)], IIAlu> {
|
|
|
|
let isCodeGenOnly = 1;
|
|
|
|
}
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2008-08-08 14:16:31 +08:00
|
|
|
// Count Leading Ones/Zeros in Word
|
2011-10-18 02:26:37 +08:00
|
|
|
class CountLeading0<bits<6> func, string instr_asm, RegisterClass RC>:
|
|
|
|
FR<0x1c, func, (outs RC:$rd), (ins RC:$rs),
|
|
|
|
!strconcat(instr_asm, "\t$rd, $rs"),
|
|
|
|
[(set RC:$rd, (ctlz RC:$rs))], IIAlu>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[HasBitCount, HasStdEnc]> {
|
2011-10-18 02:26:37 +08:00
|
|
|
let shamt = 0;
|
|
|
|
let rt = rd;
|
|
|
|
}
|
|
|
|
|
|
|
|
class CountLeading1<bits<6> func, string instr_asm, RegisterClass RC>:
|
|
|
|
FR<0x1c, func, (outs RC:$rd), (ins RC:$rs),
|
|
|
|
!strconcat(instr_asm, "\t$rd, $rs"),
|
|
|
|
[(set RC:$rd, (ctlz (not RC:$rs)))], IIAlu>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[HasBitCount, HasStdEnc]> {
|
2010-11-10 10:13:22 +08:00
|
|
|
let shamt = 0;
|
|
|
|
let rt = rd;
|
|
|
|
}
|
2008-08-08 14:16:31 +08:00
|
|
|
|
|
|
|
// Sign Extend in Register.
|
2012-01-25 05:41:09 +08:00
|
|
|
class SignExtInReg<bits<5> sa, string instr_asm, ValueType vt,
|
|
|
|
RegisterClass RC>:
|
|
|
|
FR<0x1f, 0x20, (outs RC:$rd), (ins RC:$rt),
|
2011-10-12 08:56:06 +08:00
|
|
|
!strconcat(instr_asm, "\t$rd, $rt"),
|
2012-01-25 05:41:09 +08:00
|
|
|
[(set RC:$rd, (sext_inreg RC:$rt, vt))], NoItinerary> {
|
2011-10-12 08:56:06 +08:00
|
|
|
let rs = 0;
|
|
|
|
let shamt = sa;
|
2012-12-07 11:06:09 +08:00
|
|
|
let Predicates = [HasSEInReg, HasStdEnc];
|
2011-10-12 08:56:06 +08:00
|
|
|
}
|
Several changes to Mips backend, experimental fp support being the most
important.
- Cleanup in the Subtarget info with addition of new features, not all support
yet, but they allow the future inclusion of features easier. Among new features,
we have : Arch family info (mips1, mips2, ...), ABI info (o32, eabi), 64-bit
integer
and float registers, allegrex vector FPU (VFPU), single float only support.
- TargetMachine now detects allegrex core.
- Added allegrex (Mips32r2) sext_inreg instructions.
- *Added Float Point Instructions*, handling single float only, and
aliased accesses for 32-bit FPUs.
- Some cleanup in FP instruction formats and FP register classes.
- Calling conventions improved to support mips 32-bit EABI.
- Added Asm Printer support for fp cond codes.
- Added support for sret copy to a return register.
- EABI support added into LowerCALL and FORMAL_ARGS.
- MipsFunctionInfo now keeps a virtual register per function to track the
sret on function entry until function ret.
- MipsInstrInfo FP support into methods (isMoveInstr, isLoadFromStackSlot, ...),
FP cond codes mapping and initial FP Branch Analysis.
- Two new Mips SDNode to handle fp branch and compare instructions : FPBrcond,
FPCmp
- MipsTargetLowering : handling different FP classes, Allegrex support, sret
return copy, no homing location within EABI, non 32-bit stack objects
arguments, and asm constraint for float.
llvm-svn: 53146
2008-07-06 03:05:21 +08:00
|
|
|
|
2011-12-21 07:47:44 +08:00
|
|
|
// Subword Swap
|
|
|
|
class SubwordSwap<bits<6> func, bits<5> sa, string instr_asm, RegisterClass RC>:
|
|
|
|
FR<0x1f, func, (outs RC:$rd), (ins RC:$rt),
|
|
|
|
!strconcat(instr_asm, "\t$rd, $rt"), [], NoItinerary> {
|
2011-10-12 08:56:06 +08:00
|
|
|
let rs = 0;
|
|
|
|
let shamt = sa;
|
2012-12-07 11:06:09 +08:00
|
|
|
let Predicates = [HasSwap, HasStdEnc];
|
2012-04-03 10:51:09 +08:00
|
|
|
let neverHasSideEffects = 1;
|
2011-10-12 08:56:06 +08:00
|
|
|
}
|
Several changes to Mips backend, experimental fp support being the most
important.
- Cleanup in the Subtarget info with addition of new features, not all support
yet, but they allow the future inclusion of features easier. Among new features,
we have : Arch family info (mips1, mips2, ...), ABI info (o32, eabi), 64-bit
integer
and float registers, allegrex vector FPU (VFPU), single float only support.
- TargetMachine now detects allegrex core.
- Added allegrex (Mips32r2) sext_inreg instructions.
- *Added Float Point Instructions*, handling single float only, and
aliased accesses for 32-bit FPUs.
- Some cleanup in FP instruction formats and FP register classes.
- Calling conventions improved to support mips 32-bit EABI.
- Added Asm Printer support for fp cond codes.
- Added support for sret copy to a return register.
- EABI support added into LowerCALL and FORMAL_ARGS.
- MipsFunctionInfo now keeps a virtual register per function to track the
sret on function entry until function ret.
- MipsInstrInfo FP support into methods (isMoveInstr, isLoadFromStackSlot, ...),
FP cond codes mapping and initial FP Branch Analysis.
- Two new Mips SDNode to handle fp branch and compare instructions : FPBrcond,
FPCmp
- MipsTargetLowering : handling different FP classes, Allegrex support, sret
return copy, no homing location within EABI, non 32-bit stack objects
arguments, and asm constraint for float.
llvm-svn: 53146
2008-07-06 03:05:21 +08:00
|
|
|
|
2011-05-31 10:53:58 +08:00
|
|
|
// Read Hardware
|
2011-12-08 07:31:26 +08:00
|
|
|
class ReadHardware<RegisterClass CPURegClass, RegisterClass HWRegClass>
|
|
|
|
: FR<0x1f, 0x3b, (outs CPURegClass:$rt), (ins HWRegClass:$rd),
|
|
|
|
"rdhwr\t$rt, $rd", [], IIAlu> {
|
2011-05-31 10:53:58 +08:00
|
|
|
let rs = 0;
|
|
|
|
let shamt = 0;
|
|
|
|
}
|
|
|
|
|
2011-08-18 06:59:46 +08:00
|
|
|
// Ext and Ins
|
2011-12-06 05:14:28 +08:00
|
|
|
class ExtBase<bits<6> _funct, string instr_asm, RegisterClass RC>:
|
2012-02-28 15:46:26 +08:00
|
|
|
FR<0x1f, _funct, (outs RC:$rt), (ins RC:$rs, uimm16:$pos, size_ext:$sz),
|
2011-12-06 05:14:28 +08:00
|
|
|
!strconcat(instr_asm, " $rt, $rs, $pos, $sz"),
|
|
|
|
[(set RC:$rt, (MipsExt RC:$rs, imm:$pos, imm:$sz))], NoItinerary> {
|
2011-08-18 06:59:46 +08:00
|
|
|
bits<5> pos;
|
2011-08-19 00:30:49 +08:00
|
|
|
bits<5> sz;
|
|
|
|
let rd = sz;
|
2011-08-18 06:59:46 +08:00
|
|
|
let shamt = pos;
|
2012-12-07 11:06:09 +08:00
|
|
|
let Predicates = [HasMips32r2, HasStdEnc];
|
2011-12-06 05:14:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
class InsBase<bits<6> _funct, string instr_asm, RegisterClass RC>:
|
|
|
|
FR<0x1f, _funct, (outs RC:$rt),
|
|
|
|
(ins RC:$rs, uimm16:$pos, size_ins:$sz, RC:$src),
|
|
|
|
!strconcat(instr_asm, " $rt, $rs, $pos, $sz"),
|
|
|
|
[(set RC:$rt, (MipsIns RC:$rs, imm:$pos, imm:$sz, RC:$src))],
|
|
|
|
NoItinerary> {
|
|
|
|
bits<5> pos;
|
|
|
|
bits<5> sz;
|
|
|
|
let rd = sz;
|
|
|
|
let shamt = pos;
|
2012-12-07 11:06:09 +08:00
|
|
|
let Predicates = [HasMips32r2, HasStdEnc];
|
2011-12-06 05:14:28 +08:00
|
|
|
let Constraints = "$src = $rt";
|
2011-08-18 06:59:46 +08:00
|
|
|
}
|
|
|
|
|
2011-07-20 08:23:01 +08:00
|
|
|
// Atomic instructions with 2 source operands (ATOMIC_SWAP & ATOMIC_LOAD_*).
|
2011-11-11 12:14:30 +08:00
|
|
|
class Atomic2Ops<PatFrag Op, string Opstr, RegisterClass DRC,
|
|
|
|
RegisterClass PRC> :
|
2012-08-01 03:13:07 +08:00
|
|
|
PseudoSE<(outs DRC:$dst), (ins PRC:$ptr, DRC:$incr),
|
|
|
|
!strconcat("atomic_", Opstr, "\t$dst, $ptr, $incr"),
|
|
|
|
[(set DRC:$dst, (Op PRC:$ptr, DRC:$incr))]>;
|
2011-11-11 12:14:30 +08:00
|
|
|
|
|
|
|
multiclass Atomic2Ops32<PatFrag Op, string Opstr> {
|
2012-05-22 11:10:09 +08:00
|
|
|
def #NAME# : Atomic2Ops<Op, Opstr, CPURegs, CPURegs>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[NotN64, HasStdEnc]>;
|
2012-05-22 11:10:09 +08:00
|
|
|
def _P8 : Atomic2Ops<Op, Opstr, CPURegs, CPU64Regs>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[IsN64, HasStdEnc]> {
|
2012-04-18 02:03:21 +08:00
|
|
|
let DecoderNamespace = "Mips64";
|
|
|
|
}
|
2011-11-11 12:14:30 +08:00
|
|
|
}
|
2011-07-20 08:23:01 +08:00
|
|
|
|
|
|
|
// Atomic Compare & Swap.
|
2011-11-11 12:14:30 +08:00
|
|
|
class AtomicCmpSwap<PatFrag Op, string Width, RegisterClass DRC,
|
|
|
|
RegisterClass PRC> :
|
2012-08-01 03:13:07 +08:00
|
|
|
PseudoSE<(outs DRC:$dst), (ins PRC:$ptr, DRC:$cmp, DRC:$swap),
|
|
|
|
!strconcat("atomic_cmp_swap_", Width, "\t$dst, $ptr, $cmp, $swap"),
|
|
|
|
[(set DRC:$dst, (Op PRC:$ptr, DRC:$cmp, DRC:$swap))]>;
|
2011-11-11 12:14:30 +08:00
|
|
|
|
|
|
|
multiclass AtomicCmpSwap32<PatFrag Op, string Width> {
|
2012-05-22 11:10:09 +08:00
|
|
|
def #NAME# : AtomicCmpSwap<Op, Width, CPURegs, CPURegs>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[NotN64, HasStdEnc]>;
|
2012-05-22 11:10:09 +08:00
|
|
|
def _P8 : AtomicCmpSwap<Op, Width, CPURegs, CPU64Regs>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[IsN64, HasStdEnc]> {
|
2012-04-18 02:03:21 +08:00
|
|
|
let DecoderNamespace = "Mips64";
|
|
|
|
}
|
2011-11-11 12:14:30 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
class LLBase<bits<6> Opc, string opstring, RegisterClass RC, Operand Mem> :
|
|
|
|
FMem<Opc, (outs RC:$rt), (ins Mem:$addr),
|
|
|
|
!strconcat(opstring, "\t$rt, $addr"), [], IILoad> {
|
|
|
|
let mayLoad = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
class SCBase<bits<6> Opc, string opstring, RegisterClass RC, Operand Mem> :
|
|
|
|
FMem<Opc, (outs RC:$dst), (ins RC:$rt, Mem:$addr),
|
|
|
|
!strconcat(opstring, "\t$rt, $addr"), [], IIStore> {
|
|
|
|
let mayStore = 1;
|
|
|
|
let Constraints = "$rt = $dst";
|
|
|
|
}
|
2011-07-20 08:23:01 +08:00
|
|
|
|
2011-04-16 05:51:11 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2007-06-06 15:42:06 +08:00
|
|
|
// Pseudo instructions
|
2011-04-16 05:51:11 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2012-07-10 08:19:06 +08:00
|
|
|
// Return RA.
|
|
|
|
let isReturn=1, isTerminator=1, hasDelaySlot=1, isBarrier=1, hasCtrlDep=1 in
|
2012-08-01 03:13:07 +08:00
|
|
|
def RetRA : PseudoSE<(outs), (ins), "", [(MipsRet)]>;
|
2012-07-10 08:19:06 +08:00
|
|
|
|
2012-08-01 03:13:07 +08:00
|
|
|
let Defs = [SP], Uses = [SP], hasSideEffects = 1 in {
|
|
|
|
def ADJCALLSTACKDOWN : MipsPseudo<(outs), (ins i32imm:$amt),
|
Several changes to Mips backend, experimental fp support being the most
important.
- Cleanup in the Subtarget info with addition of new features, not all support
yet, but they allow the future inclusion of features easier. Among new features,
we have : Arch family info (mips1, mips2, ...), ABI info (o32, eabi), 64-bit
integer
and float registers, allegrex vector FPU (VFPU), single float only support.
- TargetMachine now detects allegrex core.
- Added allegrex (Mips32r2) sext_inreg instructions.
- *Added Float Point Instructions*, handling single float only, and
aliased accesses for 32-bit FPUs.
- Some cleanup in FP instruction formats and FP register classes.
- Calling conventions improved to support mips 32-bit EABI.
- Added Asm Printer support for fp cond codes.
- Added support for sret copy to a return register.
- EABI support added into LowerCALL and FORMAL_ARGS.
- MipsFunctionInfo now keeps a virtual register per function to track the
sret on function entry until function ret.
- MipsInstrInfo FP support into methods (isMoveInstr, isLoadFromStackSlot, ...),
FP cond codes mapping and initial FP Branch Analysis.
- Two new Mips SDNode to handle fp branch and compare instructions : FPBrcond,
FPCmp
- MipsTargetLowering : handling different FP classes, Allegrex support, sret
return copy, no homing location within EABI, non 32-bit stack objects
arguments, and asm constraint for float.
llvm-svn: 53146
2008-07-06 03:05:21 +08:00
|
|
|
"!ADJCALLSTACKDOWN $amt",
|
2008-10-12 06:08:30 +08:00
|
|
|
[(callseq_start timm:$amt)]>;
|
2012-08-01 03:13:07 +08:00
|
|
|
def ADJCALLSTACKUP : MipsPseudo<(outs), (ins i32imm:$amt1, i32imm:$amt2),
|
Several changes to Mips backend, experimental fp support being the most
important.
- Cleanup in the Subtarget info with addition of new features, not all support
yet, but they allow the future inclusion of features easier. Among new features,
we have : Arch family info (mips1, mips2, ...), ABI info (o32, eabi), 64-bit
integer
and float registers, allegrex vector FPU (VFPU), single float only support.
- TargetMachine now detects allegrex core.
- Added allegrex (Mips32r2) sext_inreg instructions.
- *Added Float Point Instructions*, handling single float only, and
aliased accesses for 32-bit FPUs.
- Some cleanup in FP instruction formats and FP register classes.
- Calling conventions improved to support mips 32-bit EABI.
- Added Asm Printer support for fp cond codes.
- Added support for sret copy to a return register.
- EABI support added into LowerCALL and FORMAL_ARGS.
- MipsFunctionInfo now keeps a virtual register per function to track the
sret on function entry until function ret.
- MipsInstrInfo FP support into methods (isMoveInstr, isLoadFromStackSlot, ...),
FP cond codes mapping and initial FP Branch Analysis.
- Two new Mips SDNode to handle fp branch and compare instructions : FPBrcond,
FPCmp
- MipsTargetLowering : handling different FP classes, Allegrex support, sret
return copy, no homing location within EABI, non 32-bit stack objects
arguments, and asm constraint for float.
llvm-svn: 53146
2008-07-06 03:05:21 +08:00
|
|
|
"!ADJCALLSTACKUP $amt1",
|
2008-10-12 06:08:30 +08:00
|
|
|
[(callseq_end timm:$amt1, timm:$amt2)]>;
|
2007-09-12 03:55:27 +08:00
|
|
|
}
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2007-10-26 12:00:13 +08:00
|
|
|
// When handling PIC code the assembler needs .cpload and .cprestore
|
|
|
|
// directives. If the real instructions corresponding these directives
|
|
|
|
// are used, we have the same behavior, but get also a bunch of warnings
|
2007-10-09 10:55:31 +08:00
|
|
|
// from the assembler.
|
2012-04-03 10:51:09 +08:00
|
|
|
let neverHasSideEffects = 1 in
|
2012-08-01 03:13:07 +08:00
|
|
|
def CPRESTORE : PseudoSE<(outs), (ins i32imm:$loc, CPURegs:$gp),
|
|
|
|
".cprestore\t$loc", []>;
|
2008-06-06 08:58:26 +08:00
|
|
|
|
2011-05-31 10:54:07 +08:00
|
|
|
let usesCustomInserter = 1 in {
|
2011-11-11 12:14:30 +08:00
|
|
|
defm ATOMIC_LOAD_ADD_I8 : Atomic2Ops32<atomic_load_add_8, "load_add_8">;
|
|
|
|
defm ATOMIC_LOAD_ADD_I16 : Atomic2Ops32<atomic_load_add_16, "load_add_16">;
|
|
|
|
defm ATOMIC_LOAD_ADD_I32 : Atomic2Ops32<atomic_load_add_32, "load_add_32">;
|
|
|
|
defm ATOMIC_LOAD_SUB_I8 : Atomic2Ops32<atomic_load_sub_8, "load_sub_8">;
|
|
|
|
defm ATOMIC_LOAD_SUB_I16 : Atomic2Ops32<atomic_load_sub_16, "load_sub_16">;
|
|
|
|
defm ATOMIC_LOAD_SUB_I32 : Atomic2Ops32<atomic_load_sub_32, "load_sub_32">;
|
|
|
|
defm ATOMIC_LOAD_AND_I8 : Atomic2Ops32<atomic_load_and_8, "load_and_8">;
|
|
|
|
defm ATOMIC_LOAD_AND_I16 : Atomic2Ops32<atomic_load_and_16, "load_and_16">;
|
|
|
|
defm ATOMIC_LOAD_AND_I32 : Atomic2Ops32<atomic_load_and_32, "load_and_32">;
|
|
|
|
defm ATOMIC_LOAD_OR_I8 : Atomic2Ops32<atomic_load_or_8, "load_or_8">;
|
|
|
|
defm ATOMIC_LOAD_OR_I16 : Atomic2Ops32<atomic_load_or_16, "load_or_16">;
|
|
|
|
defm ATOMIC_LOAD_OR_I32 : Atomic2Ops32<atomic_load_or_32, "load_or_32">;
|
|
|
|
defm ATOMIC_LOAD_XOR_I8 : Atomic2Ops32<atomic_load_xor_8, "load_xor_8">;
|
|
|
|
defm ATOMIC_LOAD_XOR_I16 : Atomic2Ops32<atomic_load_xor_16, "load_xor_16">;
|
|
|
|
defm ATOMIC_LOAD_XOR_I32 : Atomic2Ops32<atomic_load_xor_32, "load_xor_32">;
|
|
|
|
defm ATOMIC_LOAD_NAND_I8 : Atomic2Ops32<atomic_load_nand_8, "load_nand_8">;
|
|
|
|
defm ATOMIC_LOAD_NAND_I16 : Atomic2Ops32<atomic_load_nand_16, "load_nand_16">;
|
|
|
|
defm ATOMIC_LOAD_NAND_I32 : Atomic2Ops32<atomic_load_nand_32, "load_nand_32">;
|
|
|
|
|
|
|
|
defm ATOMIC_SWAP_I8 : Atomic2Ops32<atomic_swap_8, "swap_8">;
|
|
|
|
defm ATOMIC_SWAP_I16 : Atomic2Ops32<atomic_swap_16, "swap_16">;
|
|
|
|
defm ATOMIC_SWAP_I32 : Atomic2Ops32<atomic_swap_32, "swap_32">;
|
|
|
|
|
|
|
|
defm ATOMIC_CMP_SWAP_I8 : AtomicCmpSwap32<atomic_cmp_swap_8, "8">;
|
|
|
|
defm ATOMIC_CMP_SWAP_I16 : AtomicCmpSwap32<atomic_cmp_swap_16, "16">;
|
|
|
|
defm ATOMIC_CMP_SWAP_I32 : AtomicCmpSwap32<atomic_cmp_swap_32, "32">;
|
2011-05-31 10:54:07 +08:00
|
|
|
}
|
|
|
|
|
2011-04-16 05:51:11 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2007-06-06 15:42:06 +08:00
|
|
|
// Instruction definition
|
2011-04-16 05:51:11 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2007-06-06 15:42:06 +08:00
|
|
|
|
Implement methods that enable expansion of load immediate
macro instruction (li) in the assembler.
We have identified three possible expansions depending on
the size of immediate operand:
1) for 0 ≤ j ≤ 65535.
li d,j =>
ori d,$zero,j
2) for −32768 ≤ j < 0.
li d,j =>
addiu d,$zero,j
3) for any other value of j that is representable as a 32-bit integer.
li d,j =>
lui d,hi16(j)
ori d,d,lo16(j)
All of the above have been implemented in ths patch.
Contributer: Vladimir Medic
llvm-svn: 165199
2012-10-04 12:03:53 +08:00
|
|
|
class LoadImm32< string instr_asm, Operand Od, RegisterClass RC> :
|
|
|
|
MipsAsmPseudoInst<(outs RC:$rt), (ins Od:$imm32),
|
Initial assembler implementation of Mips load address macro
This patch provides initial implementation of load address
macro instruction for Mips. We have implemented two kinds
of expansions with their variations depending on the size
of immediate operand:
1) load address with immediate value directly:
* la d,j => addiu d,$zero,j (for -32768 <= j <= 65535)
* la d,j => lui d,hi16(j)
ori d,d,lo16(j) (for any other 32 bit value of j)
2) load load address with register offset value
* la d,j(s) => addiu d,s,j (for -32768 <= j <= 65535)
* la d,j(s) => lui d,hi16(j) (for any other 32 bit value of j)
ori d,d,lo16(j)
addu d,d,s
This patch does not cover the case when the address is loaded
from the value of the label or function.
Contributer: Vladimir Medic
llvm-svn: 165561
2012-10-10 07:29:45 +08:00
|
|
|
!strconcat(instr_asm, "\t$rt, $imm32")> ;
|
|
|
|
def LoadImm32Reg : LoadImm32<"li", shamt,CPURegs>;
|
|
|
|
|
|
|
|
class LoadAddress<string instr_asm, Operand MemOpnd, RegisterClass RC> :
|
|
|
|
MipsAsmPseudoInst<(outs RC:$rt), (ins MemOpnd:$addr),
|
|
|
|
!strconcat(instr_asm, "\t$rt, $addr")> ;
|
|
|
|
def LoadAddr32Reg : LoadAddress<"la", mem, CPURegs>;
|
|
|
|
|
|
|
|
class LoadAddressImm<string instr_asm, Operand Od, RegisterClass RC> :
|
|
|
|
MipsAsmPseudoInst<(outs RC:$rt), (ins Od:$imm32),
|
|
|
|
!strconcat(instr_asm, "\t$rt, $imm32")> ;
|
|
|
|
def LoadAddr32Imm : LoadAddressImm<"la", shamt,CPURegs>;
|
|
|
|
|
2011-04-16 05:51:11 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2007-08-18 10:37:46 +08:00
|
|
|
// MipsI Instructions
|
2011-04-16 05:51:11 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2008-07-31 00:58:59 +08:00
|
|
|
/// Arithmetic Instructions (ALU Immediate)
|
2012-11-01 02:37:55 +08:00
|
|
|
def ADDiu : ArithLogicI<0x09, "addiu", add, simm16, immSExt16, CPURegs>,
|
|
|
|
IsAsCheapAsAMove;
|
2011-10-12 07:38:52 +08:00
|
|
|
def ADDi : ArithOverflowI<0x08, "addi", add, simm16, immSExt16, CPURegs>;
|
2011-10-12 02:53:46 +08:00
|
|
|
def SLTi : SetCC_I<0x0a, "slti", setlt, simm16, immSExt16, CPURegs>;
|
|
|
|
def SLTiu : SetCC_I<0x0b, "sltiu", setult, simm16, immSExt16, CPURegs>;
|
2011-10-12 07:38:52 +08:00
|
|
|
def ANDi : ArithLogicI<0x0c, "andi", and, uimm16, immZExt16, CPURegs>;
|
|
|
|
def ORi : ArithLogicI<0x0d, "ori", or, uimm16, immZExt16, CPURegs>;
|
|
|
|
def XORi : ArithLogicI<0x0e, "xori", xor, uimm16, immZExt16, CPURegs>;
|
2011-11-08 03:10:49 +08:00
|
|
|
def LUi : LoadUpper<0x0f, "lui", CPURegs, uimm16>;
|
2008-07-31 00:58:59 +08:00
|
|
|
|
|
|
|
/// Arithmetic Instructions (3-Operand, R-Type)
|
2011-10-12 07:05:46 +08:00
|
|
|
def ADDu : ArithLogicR<0x00, 0x21, "addu", add, IIAlu, CPURegs, 1>;
|
|
|
|
def SUBu : ArithLogicR<0x00, 0x23, "subu", sub, IIAlu, CPURegs>;
|
2011-10-12 07:43:48 +08:00
|
|
|
def ADD : ArithOverflowR<0x00, 0x20, "add", IIAlu, CPURegs, 1>;
|
|
|
|
def SUB : ArithOverflowR<0x00, 0x22, "sub", IIAlu, CPURegs>;
|
2011-10-12 02:53:46 +08:00
|
|
|
def SLT : SetCC_R<0x00, 0x2a, "slt", setlt, CPURegs>;
|
|
|
|
def SLTu : SetCC_R<0x00, 0x2b, "sltu", setult, CPURegs>;
|
2011-10-12 07:05:46 +08:00
|
|
|
def AND : ArithLogicR<0x00, 0x24, "and", and, IIAlu, CPURegs, 1>;
|
|
|
|
def OR : ArithLogicR<0x00, 0x25, "or", or, IIAlu, CPURegs, 1>;
|
|
|
|
def XOR : ArithLogicR<0x00, 0x26, "xor", xor, IIAlu, CPURegs, 1>;
|
2011-10-12 09:05:13 +08:00
|
|
|
def NOR : LogicNOR<0x00, 0x27, "nor", CPURegs>;
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2008-07-31 00:58:59 +08:00
|
|
|
/// Shift Instructions
|
2011-10-18 02:06:56 +08:00
|
|
|
def SLL : shift_rotate_imm32<0x00, 0x00, "sll", shl>;
|
|
|
|
def SRL : shift_rotate_imm32<0x02, 0x00, "srl", srl>;
|
|
|
|
def SRA : shift_rotate_imm32<0x03, 0x00, "sra", sra>;
|
2011-10-18 02:17:58 +08:00
|
|
|
def SLLV : shift_rotate_reg<0x04, 0x00, "sllv", shl, CPURegs>;
|
|
|
|
def SRLV : shift_rotate_reg<0x06, 0x00, "srlv", srl, CPURegs>;
|
|
|
|
def SRAV : shift_rotate_reg<0x07, 0x00, "srav", sra, CPURegs>;
|
2010-12-10 01:32:30 +08:00
|
|
|
|
|
|
|
// Rotate Instructions
|
2012-12-07 11:06:09 +08:00
|
|
|
let Predicates = [HasMips32r2, HasStdEnc] in {
|
2011-10-18 02:06:56 +08:00
|
|
|
def ROTR : shift_rotate_imm32<0x02, 0x01, "rotr", rotr>;
|
2011-10-18 02:17:58 +08:00
|
|
|
def ROTRV : shift_rotate_reg<0x06, 0x01, "rotrv", rotr, CPURegs>;
|
2010-12-10 01:32:30 +08:00
|
|
|
}
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2008-07-31 00:58:59 +08:00
|
|
|
/// Load and Store Instructions
|
2011-10-08 10:24:10 +08:00
|
|
|
/// aligned
|
2011-10-11 08:11:12 +08:00
|
|
|
defm LB : LoadM32<0x20, "lb", sextloadi8>;
|
|
|
|
defm LBu : LoadM32<0x24, "lbu", zextloadi8>;
|
2012-09-15 09:52:08 +08:00
|
|
|
defm LH : LoadM32<0x21, "lh", sextloadi16>;
|
|
|
|
defm LHu : LoadM32<0x25, "lhu", zextloadi16>;
|
|
|
|
defm LW : LoadM32<0x23, "lw", load>;
|
2011-10-11 08:11:12 +08:00
|
|
|
defm SB : StoreM32<0x28, "sb", truncstorei8>;
|
2012-09-15 09:52:08 +08:00
|
|
|
defm SH : StoreM32<0x29, "sh", truncstorei16>;
|
|
|
|
defm SW : StoreM32<0x2b, "sw", store>;
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2012-06-02 08:04:19 +08:00
|
|
|
/// load/store left/right
|
|
|
|
defm LWL : LoadLeftRightM32<0x22, "lwl", MipsLWL>;
|
|
|
|
defm LWR : LoadLeftRightM32<0x26, "lwr", MipsLWR>;
|
|
|
|
defm SWL : StoreLeftRightM32<0x2a, "swl", MipsSWL>;
|
|
|
|
defm SWR : StoreLeftRightM32<0x2e, "swr", MipsSWR>;
|
2011-11-24 06:19:28 +08:00
|
|
|
|
2011-07-20 07:30:50 +08:00
|
|
|
let hasSideEffects = 1 in
|
2012-08-01 02:55:01 +08:00
|
|
|
def SYNC : InstSE<(outs), (ins i32imm:$stype), "sync $stype",
|
|
|
|
[(MipsSync imm:$stype)], NoItinerary, FrmOther>
|
2011-07-20 07:30:50 +08:00
|
|
|
{
|
2011-10-19 01:50:36 +08:00
|
|
|
bits<5> stype;
|
|
|
|
let Opcode = 0;
|
2011-07-20 07:30:50 +08:00
|
|
|
let Inst{25-11} = 0;
|
2011-10-19 01:50:36 +08:00
|
|
|
let Inst{10-6} = stype;
|
2011-07-20 07:30:50 +08:00
|
|
|
let Inst{5-0} = 15;
|
|
|
|
}
|
|
|
|
|
2011-05-31 10:54:07 +08:00
|
|
|
/// Load-linked, Store-conditional
|
2012-05-22 11:10:09 +08:00
|
|
|
def LL : LLBase<0x30, "ll", CPURegs, mem>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[NotN64, HasStdEnc]>;
|
2012-05-22 11:10:09 +08:00
|
|
|
def LL_P8 : LLBase<0x30, "ll", CPURegs, mem64>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[IsN64, HasStdEnc]> {
|
2012-04-18 02:03:21 +08:00
|
|
|
let DecoderNamespace = "Mips64";
|
|
|
|
}
|
|
|
|
|
2012-05-22 11:10:09 +08:00
|
|
|
def SC : SCBase<0x38, "sc", CPURegs, mem>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[NotN64, HasStdEnc]>;
|
2012-05-22 11:10:09 +08:00
|
|
|
def SC_P8 : SCBase<0x38, "sc", CPURegs, mem64>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[IsN64, HasStdEnc]> {
|
2012-04-18 02:03:21 +08:00
|
|
|
let DecoderNamespace = "Mips64";
|
|
|
|
}
|
2011-05-31 10:54:07 +08:00
|
|
|
|
2008-07-31 00:58:59 +08:00
|
|
|
/// Jump and Branch Instructions
|
2012-10-20 05:30:15 +08:00
|
|
|
def J : JumpFJ<0x02, jmptarget, "j", br, bb>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[RelocStatic, HasStdEnc]>, IsBranch;
|
2012-07-10 08:19:06 +08:00
|
|
|
def JR : IndirectBranch<CPURegs>;
|
2011-12-06 11:34:48 +08:00
|
|
|
def B : UncondBranch<0x04, "b">;
|
2011-10-12 02:49:17 +08:00
|
|
|
def BEQ : CBranch<0x04, "beq", seteq, CPURegs>;
|
|
|
|
def BNE : CBranch<0x05, "bne", setne, CPURegs>;
|
|
|
|
def BGEZ : CBranchZero<0x01, 1, "bgez", setge, CPURegs>;
|
|
|
|
def BGTZ : CBranchZero<0x07, 0, "bgtz", setgt, CPURegs>;
|
2011-10-19 01:50:36 +08:00
|
|
|
def BLEZ : CBranchZero<0x06, 0, "blez", setle, CPURegs>;
|
2011-10-12 02:49:17 +08:00
|
|
|
def BLTZ : CBranchZero<0x01, 0, "bltz", setlt, CPURegs>;
|
2007-08-18 10:37:46 +08:00
|
|
|
|
2012-07-21 11:30:44 +08:00
|
|
|
let rt = 0, rs = 0, isBranch = 1, isTerminator = 1, isBarrier = 1,
|
|
|
|
hasDelaySlot = 1, Defs = [RA] in
|
|
|
|
def BAL_BR: FI<0x1, (outs), (ins brtarget:$imm16), "bal\t$imm16", [], IIBranch>;
|
|
|
|
|
2012-03-02 06:27:29 +08:00
|
|
|
def JAL : JumpLink<0x03, "jal">;
|
|
|
|
def JALR : JumpLinkReg<0x00, 0x09, "jalr", CPURegs>;
|
|
|
|
def BGEZAL : BranchLink<"bgezal", 0x11, CPURegs>;
|
|
|
|
def BLTZAL : BranchLink<"bltzal", 0x10, CPURegs>;
|
2012-10-20 05:30:15 +08:00
|
|
|
def TAILCALL : JumpFJ<0x02, calltarget, "j", MipsTailCall, imm>, IsTailCall;
|
2012-10-20 05:14:34 +08:00
|
|
|
def TAILCALL_R : JumpFR<CPURegs, MipsTailCall>, IsTailCall;
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2012-07-10 08:19:06 +08:00
|
|
|
def RET : RetBase<CPURegs>;
|
2008-07-31 00:58:59 +08:00
|
|
|
|
2010-11-10 01:25:34 +08:00
|
|
|
/// Multiply and Divide Instructions.
|
2011-10-18 02:21:24 +08:00
|
|
|
def MULT : Mult32<0x18, "mult", IIImul>;
|
|
|
|
def MULTu : Mult32<0x19, "multu", IIImul>;
|
|
|
|
def SDIV : Div32<MipsDivRem, 0x1a, "div", IIIdiv>;
|
|
|
|
def UDIV : Div32<MipsDivRemU, 0x1b, "divu", IIIdiv>;
|
2008-08-03 03:42:36 +08:00
|
|
|
|
2011-10-18 02:24:15 +08:00
|
|
|
def MTHI : MoveToLOHI<0x11, "mthi", CPURegs, [HI]>;
|
|
|
|
def MTLO : MoveToLOHI<0x13, "mtlo", CPURegs, [LO]>;
|
|
|
|
def MFHI : MoveFromLOHI<0x10, "mfhi", CPURegs, [HI]>;
|
|
|
|
def MFLO : MoveFromLOHI<0x12, "mflo", CPURegs, [LO]>;
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2008-07-31 00:58:59 +08:00
|
|
|
/// Sign Ext In Register Instructions.
|
2012-01-25 05:41:09 +08:00
|
|
|
def SEB : SignExtInReg<0x10, "seb", i8, CPURegs>;
|
|
|
|
def SEH : SignExtInReg<0x18, "seh", i16, CPURegs>;
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2008-08-08 14:16:31 +08:00
|
|
|
/// Count Leading
|
2011-10-18 02:26:37 +08:00
|
|
|
def CLZ : CountLeading0<0x20, "clz", CPURegs>;
|
|
|
|
def CLO : CountLeading1<0x21, "clo", CPURegs>;
|
2008-08-13 15:13:40 +08:00
|
|
|
|
2011-12-21 07:47:44 +08:00
|
|
|
/// Word Swap Bytes Within Halfwords
|
|
|
|
def WSBH : SubwordSwap<0x20, 0x2, "wsbh", CPURegs>;
|
2008-08-13 15:13:40 +08:00
|
|
|
|
2008-07-31 00:58:59 +08:00
|
|
|
/// No operation
|
|
|
|
let addr=0 in
|
|
|
|
def NOP : FJ<0, (outs), (ins), "nop", [], IIAlu>;
|
|
|
|
|
2007-10-26 12:00:13 +08:00
|
|
|
// FrameIndexes are legalized when they are operands from load/store
|
2007-09-25 04:15:11 +08:00
|
|
|
// instructions. The same not happens for stack address copies, so an
|
|
|
|
// add op with mem ComplexPattern is used and the stack address copy
|
|
|
|
// can be matched. It's similar to Sparc LEA_ADDRi
|
2012-08-07 07:29:06 +08:00
|
|
|
def LEA_ADDiu : EffectiveAddress<0x09,"addiu\t$rt, $addr", CPURegs, mem_ea>;
|
2007-09-25 04:15:11 +08:00
|
|
|
|
2011-01-19 03:29:17 +08:00
|
|
|
// MADD*/MSUB*
|
2011-05-13 01:42:08 +08:00
|
|
|
def MADD : MArithR<0, "madd", MipsMAdd, 1>;
|
|
|
|
def MADDU : MArithR<1, "maddu", MipsMAddu, 1>;
|
2011-01-19 03:29:17 +08:00
|
|
|
def MSUB : MArithR<4, "msub", MipsMSub>;
|
|
|
|
def MSUBU : MArithR<5, "msubu", MipsMSubu>;
|
Several changes to Mips backend, experimental fp support being the most
important.
- Cleanup in the Subtarget info with addition of new features, not all support
yet, but they allow the future inclusion of features easier. Among new features,
we have : Arch family info (mips1, mips2, ...), ABI info (o32, eabi), 64-bit
integer
and float registers, allegrex vector FPU (VFPU), single float only support.
- TargetMachine now detects allegrex core.
- Added allegrex (Mips32r2) sext_inreg instructions.
- *Added Float Point Instructions*, handling single float only, and
aliased accesses for 32-bit FPUs.
- Some cleanup in FP instruction formats and FP register classes.
- Calling conventions improved to support mips 32-bit EABI.
- Added Asm Printer support for fp cond codes.
- Added support for sret copy to a return register.
- EABI support added into LowerCALL and FORMAL_ARGS.
- MipsFunctionInfo now keeps a virtual register per function to track the
sret on function entry until function ret.
- MipsInstrInfo FP support into methods (isMoveInstr, isLoadFromStackSlot, ...),
FP cond codes mapping and initial FP Branch Analysis.
- Two new Mips SDNode to handle fp branch and compare instructions : FPBrcond,
FPCmp
- MipsTargetLowering : handling different FP classes, Allegrex support, sret
return copy, no homing location within EABI, non 32-bit stack objects
arguments, and asm constraint for float.
llvm-svn: 53146
2008-07-06 03:05:21 +08:00
|
|
|
|
2008-07-31 00:58:59 +08:00
|
|
|
// MUL is a assembly macro in the current used ISAs. In recent ISA's
|
|
|
|
// it is a real instruction.
|
2011-10-12 07:05:46 +08:00
|
|
|
def MUL : ArithLogicR<0x1c, 0x02, "mul", mul, IIImul, CPURegs, 1>,
|
2012-12-07 11:06:09 +08:00
|
|
|
Requires<[HasStdEnc]>;
|
Several changes to Mips backend, experimental fp support being the most
important.
- Cleanup in the Subtarget info with addition of new features, not all support
yet, but they allow the future inclusion of features easier. Among new features,
we have : Arch family info (mips1, mips2, ...), ABI info (o32, eabi), 64-bit
integer
and float registers, allegrex vector FPU (VFPU), single float only support.
- TargetMachine now detects allegrex core.
- Added allegrex (Mips32r2) sext_inreg instructions.
- *Added Float Point Instructions*, handling single float only, and
aliased accesses for 32-bit FPUs.
- Some cleanup in FP instruction formats and FP register classes.
- Calling conventions improved to support mips 32-bit EABI.
- Added Asm Printer support for fp cond codes.
- Added support for sret copy to a return register.
- EABI support added into LowerCALL and FORMAL_ARGS.
- MipsFunctionInfo now keeps a virtual register per function to track the
sret on function entry until function ret.
- MipsInstrInfo FP support into methods (isMoveInstr, isLoadFromStackSlot, ...),
FP cond codes mapping and initial FP Branch Analysis.
- Two new Mips SDNode to handle fp branch and compare instructions : FPBrcond,
FPCmp
- MipsTargetLowering : handling different FP classes, Allegrex support, sret
return copy, no homing location within EABI, non 32-bit stack objects
arguments, and asm constraint for float.
llvm-svn: 53146
2008-07-06 03:05:21 +08:00
|
|
|
|
2011-12-08 07:31:26 +08:00
|
|
|
def RDHWR : ReadHardware<CPURegs, HWRegs>;
|
2011-05-31 10:53:58 +08:00
|
|
|
|
2011-12-06 05:14:28 +08:00
|
|
|
def EXT : ExtBase<0, "ext", CPURegs>;
|
|
|
|
def INS : InsBase<4, "ins", CPURegs>;
|
2011-08-17 10:05:42 +08:00
|
|
|
|
2012-09-07 09:42:38 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Instruction aliases
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
def : InstAlias<"move $dst,$src", (ADD CPURegs:$dst,CPURegs:$src,ZERO)>;
|
|
|
|
def : InstAlias<"bal $offset", (BGEZAL RA,brtarget:$offset)>;
|
|
|
|
def : InstAlias<"addu $rs,$rt,$imm",
|
|
|
|
(ADDiu CPURegs:$rs,CPURegs:$rt,simm16:$imm)>;
|
|
|
|
def : InstAlias<"add $rs,$rt,$imm",
|
|
|
|
(ADDi CPURegs:$rs,CPURegs:$rt,simm16:$imm)>;
|
|
|
|
def : InstAlias<"and $rs,$rt,$imm",
|
|
|
|
(ANDi CPURegs:$rs,CPURegs:$rt,simm16:$imm)>;
|
|
|
|
def : InstAlias<"j $rs", (JR CPURegs:$rs)>;
|
|
|
|
def : InstAlias<"not $rt,$rs", (NOR CPURegs:$rt,CPURegs:$rs,ZERO)>;
|
|
|
|
def : InstAlias<"neg $rt,$rs", (SUB CPURegs:$rt,ZERO,CPURegs:$rs)>;
|
|
|
|
def : InstAlias<"negu $rt,$rs", (SUBu CPURegs:$rt,ZERO,CPURegs:$rs)>;
|
|
|
|
def : InstAlias<"slt $rs,$rt,$imm",
|
|
|
|
(SLTi CPURegs:$rs,CPURegs:$rt,simm16:$imm)>;
|
|
|
|
def : InstAlias<"xor $rs,$rt,$imm",
|
|
|
|
(XORi CPURegs:$rs,CPURegs:$rt,simm16:$imm)>;
|
|
|
|
|
2011-04-16 05:51:11 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2007-06-06 15:42:06 +08:00
|
|
|
// Arbitrary patterns that map to one or more instructions
|
2011-04-16 05:51:11 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2007-06-06 15:42:06 +08:00
|
|
|
|
|
|
|
// Small immediates
|
2012-06-15 05:03:23 +08:00
|
|
|
def : MipsPat<(i32 immSExt16:$in),
|
|
|
|
(ADDiu ZERO, imm:$in)>;
|
|
|
|
def : MipsPat<(i32 immZExt16:$in),
|
|
|
|
(ORi ZERO, imm:$in)>;
|
|
|
|
def : MipsPat<(i32 immLow16Zero:$in),
|
|
|
|
(LUi (HI16 imm:$in))>;
|
2007-06-06 15:42:06 +08:00
|
|
|
|
|
|
|
// Arbitrary immediates
|
2012-06-15 05:03:23 +08:00
|
|
|
def : MipsPat<(i32 imm:$imm),
|
2007-06-06 15:42:06 +08:00
|
|
|
(ORi (LUi (HI16 imm:$imm)), (LO16 imm:$imm))>;
|
|
|
|
|
2012-06-15 05:03:23 +08:00
|
|
|
// Carry MipsPatterns
|
|
|
|
def : MipsPat<(subc CPURegs:$lhs, CPURegs:$rhs),
|
|
|
|
(SUBu CPURegs:$lhs, CPURegs:$rhs)>;
|
|
|
|
def : MipsPat<(addc CPURegs:$lhs, CPURegs:$rhs),
|
|
|
|
(ADDu CPURegs:$lhs, CPURegs:$rhs)>;
|
|
|
|
def : MipsPat<(addc CPURegs:$src, immSExt16:$imm),
|
|
|
|
(ADDiu CPURegs:$src, imm:$imm)>;
|
2008-06-06 08:58:26 +08:00
|
|
|
|
2007-06-06 15:42:06 +08:00
|
|
|
// Call
|
2012-06-15 05:03:23 +08:00
|
|
|
def : MipsPat<(MipsJmpLink (i32 tglobaladdr:$dst)),
|
|
|
|
(JAL tglobaladdr:$dst)>;
|
|
|
|
def : MipsPat<(MipsJmpLink (i32 texternalsym:$dst)),
|
|
|
|
(JAL texternalsym:$dst)>;
|
|
|
|
//def : MipsPat<(MipsJmpLink CPURegs:$dst),
|
|
|
|
// (JALR CPURegs:$dst)>;
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2012-10-20 05:30:15 +08:00
|
|
|
// Tail call
|
|
|
|
def : MipsPat<(MipsTailCall (iPTR tglobaladdr:$dst)),
|
|
|
|
(TAILCALL tglobaladdr:$dst)>;
|
|
|
|
def : MipsPat<(MipsTailCall (iPTR texternalsym:$dst)),
|
|
|
|
(TAILCALL texternalsym:$dst)>;
|
2008-07-24 00:01:50 +08:00
|
|
|
// hi/lo relocs
|
2012-06-15 05:03:23 +08:00
|
|
|
def : MipsPat<(MipsHi tglobaladdr:$in), (LUi tglobaladdr:$in)>;
|
|
|
|
def : MipsPat<(MipsHi tblockaddress:$in), (LUi tblockaddress:$in)>;
|
|
|
|
def : MipsPat<(MipsHi tjumptable:$in), (LUi tjumptable:$in)>;
|
|
|
|
def : MipsPat<(MipsHi tconstpool:$in), (LUi tconstpool:$in)>;
|
|
|
|
def : MipsPat<(MipsHi tglobaltlsaddr:$in), (LUi tglobaltlsaddr:$in)>;
|
2012-11-22 04:40:38 +08:00
|
|
|
def : MipsPat<(MipsHi texternalsym:$in), (LUi texternalsym:$in)>;
|
2012-06-15 05:03:23 +08:00
|
|
|
|
|
|
|
def : MipsPat<(MipsLo tglobaladdr:$in), (ADDiu ZERO, tglobaladdr:$in)>;
|
|
|
|
def : MipsPat<(MipsLo tblockaddress:$in), (ADDiu ZERO, tblockaddress:$in)>;
|
|
|
|
def : MipsPat<(MipsLo tjumptable:$in), (ADDiu ZERO, tjumptable:$in)>;
|
|
|
|
def : MipsPat<(MipsLo tconstpool:$in), (ADDiu ZERO, tconstpool:$in)>;
|
|
|
|
def : MipsPat<(MipsLo tglobaltlsaddr:$in), (ADDiu ZERO, tglobaltlsaddr:$in)>;
|
2012-11-22 04:40:38 +08:00
|
|
|
def : MipsPat<(MipsLo texternalsym:$in), (ADDiu ZERO, texternalsym:$in)>;
|
2012-06-15 05:03:23 +08:00
|
|
|
|
|
|
|
def : MipsPat<(add CPURegs:$hi, (MipsLo tglobaladdr:$lo)),
|
|
|
|
(ADDiu CPURegs:$hi, tglobaladdr:$lo)>;
|
|
|
|
def : MipsPat<(add CPURegs:$hi, (MipsLo tblockaddress:$lo)),
|
|
|
|
(ADDiu CPURegs:$hi, tblockaddress:$lo)>;
|
|
|
|
def : MipsPat<(add CPURegs:$hi, (MipsLo tjumptable:$lo)),
|
|
|
|
(ADDiu CPURegs:$hi, tjumptable:$lo)>;
|
|
|
|
def : MipsPat<(add CPURegs:$hi, (MipsLo tconstpool:$lo)),
|
|
|
|
(ADDiu CPURegs:$hi, tconstpool:$lo)>;
|
|
|
|
def : MipsPat<(add CPURegs:$hi, (MipsLo tglobaltlsaddr:$lo)),
|
|
|
|
(ADDiu CPURegs:$hi, tglobaltlsaddr:$lo)>;
|
2008-07-24 00:01:50 +08:00
|
|
|
|
|
|
|
// gp_rel relocs
|
2012-06-15 05:03:23 +08:00
|
|
|
def : MipsPat<(add CPURegs:$gp, (MipsGPRel tglobaladdr:$in)),
|
|
|
|
(ADDiu CPURegs:$gp, tglobaladdr:$in)>;
|
|
|
|
def : MipsPat<(add CPURegs:$gp, (MipsGPRel tconstpool:$in)),
|
|
|
|
(ADDiu CPURegs:$gp, tconstpool:$in)>;
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2011-05-28 09:07:07 +08:00
|
|
|
// wrapper_pic
|
2012-02-25 06:34:47 +08:00
|
|
|
class WrapperPat<SDNode node, Instruction ADDiuOp, RegisterClass RC>:
|
2012-06-15 05:03:23 +08:00
|
|
|
MipsPat<(MipsWrapper RC:$gp, node:$in),
|
|
|
|
(ADDiuOp RC:$gp, node:$in)>;
|
2012-02-25 06:34:47 +08:00
|
|
|
|
|
|
|
def : WrapperPat<tglobaladdr, ADDiu, CPURegs>;
|
|
|
|
def : WrapperPat<tconstpool, ADDiu, CPURegs>;
|
|
|
|
def : WrapperPat<texternalsym, ADDiu, CPURegs>;
|
|
|
|
def : WrapperPat<tblockaddress, ADDiu, CPURegs>;
|
|
|
|
def : WrapperPat<tjumptable, ADDiu, CPURegs>;
|
|
|
|
def : WrapperPat<tglobaltlsaddr, ADDiu, CPURegs>;
|
2011-05-28 09:07:07 +08:00
|
|
|
|
Several changes to Mips backend, experimental fp support being the most
important.
- Cleanup in the Subtarget info with addition of new features, not all support
yet, but they allow the future inclusion of features easier. Among new features,
we have : Arch family info (mips1, mips2, ...), ABI info (o32, eabi), 64-bit
integer
and float registers, allegrex vector FPU (VFPU), single float only support.
- TargetMachine now detects allegrex core.
- Added allegrex (Mips32r2) sext_inreg instructions.
- *Added Float Point Instructions*, handling single float only, and
aliased accesses for 32-bit FPUs.
- Some cleanup in FP instruction formats and FP register classes.
- Calling conventions improved to support mips 32-bit EABI.
- Added Asm Printer support for fp cond codes.
- Added support for sret copy to a return register.
- EABI support added into LowerCALL and FORMAL_ARGS.
- MipsFunctionInfo now keeps a virtual register per function to track the
sret on function entry until function ret.
- MipsInstrInfo FP support into methods (isMoveInstr, isLoadFromStackSlot, ...),
FP cond codes mapping and initial FP Branch Analysis.
- Two new Mips SDNode to handle fp branch and compare instructions : FPBrcond,
FPCmp
- MipsTargetLowering : handling different FP classes, Allegrex support, sret
return copy, no homing location within EABI, non 32-bit stack objects
arguments, and asm constraint for float.
llvm-svn: 53146
2008-07-06 03:05:21 +08:00
|
|
|
// Mips does not have "not", so we expand our way
|
2012-06-15 05:03:23 +08:00
|
|
|
def : MipsPat<(not CPURegs:$in),
|
|
|
|
(NOR CPURegs:$in, ZERO)>;
|
2007-06-06 15:42:06 +08:00
|
|
|
|
2011-12-21 06:33:53 +08:00
|
|
|
// extended loads
|
2012-12-07 11:06:09 +08:00
|
|
|
let Predicates = [NotN64, HasStdEnc] in {
|
2012-06-15 05:03:23 +08:00
|
|
|
def : MipsPat<(i32 (extloadi1 addr:$src)), (LBu addr:$src)>;
|
|
|
|
def : MipsPat<(i32 (extloadi8 addr:$src)), (LBu addr:$src)>;
|
2012-09-15 09:52:08 +08:00
|
|
|
def : MipsPat<(i32 (extloadi16 addr:$src)), (LHu addr:$src)>;
|
2011-12-21 06:33:53 +08:00
|
|
|
}
|
2012-12-07 11:06:09 +08:00
|
|
|
let Predicates = [IsN64, HasStdEnc] in {
|
2012-06-15 05:03:23 +08:00
|
|
|
def : MipsPat<(i32 (extloadi1 addr:$src)), (LBu_P8 addr:$src)>;
|
|
|
|
def : MipsPat<(i32 (extloadi8 addr:$src)), (LBu_P8 addr:$src)>;
|
2012-09-15 09:52:08 +08:00
|
|
|
def : MipsPat<(i32 (extloadi16 addr:$src)), (LHu_P8 addr:$src)>;
|
2011-12-21 06:33:53 +08:00
|
|
|
}
|
2007-08-18 10:37:46 +08:00
|
|
|
|
2008-06-06 08:58:26 +08:00
|
|
|
// peepholes
|
2012-12-07 11:06:09 +08:00
|
|
|
let Predicates = [NotN64, HasStdEnc] in {
|
2012-09-15 09:52:08 +08:00
|
|
|
def : MipsPat<(store (i32 0), addr:$dst), (SW ZERO, addr:$dst)>;
|
2011-12-21 08:31:10 +08:00
|
|
|
}
|
2012-12-07 11:06:09 +08:00
|
|
|
let Predicates = [IsN64, HasStdEnc] in {
|
2012-09-15 09:52:08 +08:00
|
|
|
def : MipsPat<(store (i32 0), addr:$dst), (SW_P8 ZERO, addr:$dst)>;
|
2011-12-21 08:31:10 +08:00
|
|
|
}
|
2007-11-05 11:02:32 +08:00
|
|
|
|
Several changes to Mips backend, experimental fp support being the most
important.
- Cleanup in the Subtarget info with addition of new features, not all support
yet, but they allow the future inclusion of features easier. Among new features,
we have : Arch family info (mips1, mips2, ...), ABI info (o32, eabi), 64-bit
integer
and float registers, allegrex vector FPU (VFPU), single float only support.
- TargetMachine now detects allegrex core.
- Added allegrex (Mips32r2) sext_inreg instructions.
- *Added Float Point Instructions*, handling single float only, and
aliased accesses for 32-bit FPUs.
- Some cleanup in FP instruction formats and FP register classes.
- Calling conventions improved to support mips 32-bit EABI.
- Added Asm Printer support for fp cond codes.
- Added support for sret copy to a return register.
- EABI support added into LowerCALL and FORMAL_ARGS.
- MipsFunctionInfo now keeps a virtual register per function to track the
sret on function entry until function ret.
- MipsInstrInfo FP support into methods (isMoveInstr, isLoadFromStackSlot, ...),
FP cond codes mapping and initial FP Branch Analysis.
- Two new Mips SDNode to handle fp branch and compare instructions : FPBrcond,
FPCmp
- MipsTargetLowering : handling different FP classes, Allegrex support, sret
return copy, no homing location within EABI, non 32-bit stack objects
arguments, and asm constraint for float.
llvm-svn: 53146
2008-07-06 03:05:21 +08:00
|
|
|
// brcond patterns
|
2011-10-12 03:09:09 +08:00
|
|
|
multiclass BrcondPats<RegisterClass RC, Instruction BEQOp, Instruction BNEOp,
|
|
|
|
Instruction SLTOp, Instruction SLTuOp, Instruction SLTiOp,
|
|
|
|
Instruction SLTiuOp, Register ZEROReg> {
|
2012-06-15 05:03:23 +08:00
|
|
|
def : MipsPat<(brcond (i32 (setne RC:$lhs, 0)), bb:$dst),
|
|
|
|
(BNEOp RC:$lhs, ZEROReg, bb:$dst)>;
|
|
|
|
def : MipsPat<(brcond (i32 (seteq RC:$lhs, 0)), bb:$dst),
|
|
|
|
(BEQOp RC:$lhs, ZEROReg, bb:$dst)>;
|
2011-10-12 03:09:09 +08:00
|
|
|
|
2012-06-15 05:03:23 +08:00
|
|
|
def : MipsPat<(brcond (i32 (setge RC:$lhs, RC:$rhs)), bb:$dst),
|
|
|
|
(BEQ (SLTOp RC:$lhs, RC:$rhs), ZERO, bb:$dst)>;
|
|
|
|
def : MipsPat<(brcond (i32 (setuge RC:$lhs, RC:$rhs)), bb:$dst),
|
|
|
|
(BEQ (SLTuOp RC:$lhs, RC:$rhs), ZERO, bb:$dst)>;
|
|
|
|
def : MipsPat<(brcond (i32 (setge RC:$lhs, immSExt16:$rhs)), bb:$dst),
|
|
|
|
(BEQ (SLTiOp RC:$lhs, immSExt16:$rhs), ZERO, bb:$dst)>;
|
|
|
|
def : MipsPat<(brcond (i32 (setuge RC:$lhs, immSExt16:$rhs)), bb:$dst),
|
|
|
|
(BEQ (SLTiuOp RC:$lhs, immSExt16:$rhs), ZERO, bb:$dst)>;
|
2011-10-12 03:09:09 +08:00
|
|
|
|
2012-06-15 05:03:23 +08:00
|
|
|
def : MipsPat<(brcond (i32 (setle RC:$lhs, RC:$rhs)), bb:$dst),
|
|
|
|
(BEQ (SLTOp RC:$rhs, RC:$lhs), ZERO, bb:$dst)>;
|
|
|
|
def : MipsPat<(brcond (i32 (setule RC:$lhs, RC:$rhs)), bb:$dst),
|
|
|
|
(BEQ (SLTuOp RC:$rhs, RC:$lhs), ZERO, bb:$dst)>;
|
2011-10-12 03:09:09 +08:00
|
|
|
|
2012-06-15 05:03:23 +08:00
|
|
|
def : MipsPat<(brcond RC:$cond, bb:$dst),
|
|
|
|
(BNEOp RC:$cond, ZEROReg, bb:$dst)>;
|
2011-10-12 03:09:09 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
defm : BrcondPats<CPURegs, BEQ, BNE, SLT, SLTu, SLTi, SLTiu, ZERO>;
|
2007-08-18 10:37:46 +08:00
|
|
|
|
2008-08-13 15:13:40 +08:00
|
|
|
// setcc patterns
|
2011-10-12 05:40:01 +08:00
|
|
|
multiclass SeteqPats<RegisterClass RC, Instruction SLTiuOp, Instruction XOROp,
|
|
|
|
Instruction SLTuOp, Register ZEROReg> {
|
2012-06-15 05:03:23 +08:00
|
|
|
def : MipsPat<(seteq RC:$lhs, RC:$rhs),
|
|
|
|
(SLTiuOp (XOROp RC:$lhs, RC:$rhs), 1)>;
|
|
|
|
def : MipsPat<(setne RC:$lhs, RC:$rhs),
|
|
|
|
(SLTuOp ZEROReg, (XOROp RC:$lhs, RC:$rhs))>;
|
2011-10-12 05:40:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
multiclass SetlePats<RegisterClass RC, Instruction SLTOp, Instruction SLTuOp> {
|
2012-06-15 05:03:23 +08:00
|
|
|
def : MipsPat<(setle RC:$lhs, RC:$rhs),
|
|
|
|
(XORi (SLTOp RC:$rhs, RC:$lhs), 1)>;
|
|
|
|
def : MipsPat<(setule RC:$lhs, RC:$rhs),
|
|
|
|
(XORi (SLTuOp RC:$rhs, RC:$lhs), 1)>;
|
2011-10-12 05:40:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
multiclass SetgtPats<RegisterClass RC, Instruction SLTOp, Instruction SLTuOp> {
|
2012-06-15 05:03:23 +08:00
|
|
|
def : MipsPat<(setgt RC:$lhs, RC:$rhs),
|
|
|
|
(SLTOp RC:$rhs, RC:$lhs)>;
|
|
|
|
def : MipsPat<(setugt RC:$lhs, RC:$rhs),
|
|
|
|
(SLTuOp RC:$rhs, RC:$lhs)>;
|
2011-10-12 05:40:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
multiclass SetgePats<RegisterClass RC, Instruction SLTOp, Instruction SLTuOp> {
|
2012-06-15 05:03:23 +08:00
|
|
|
def : MipsPat<(setge RC:$lhs, RC:$rhs),
|
|
|
|
(XORi (SLTOp RC:$lhs, RC:$rhs), 1)>;
|
|
|
|
def : MipsPat<(setuge RC:$lhs, RC:$rhs),
|
|
|
|
(XORi (SLTuOp RC:$lhs, RC:$rhs), 1)>;
|
2011-10-12 05:40:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
multiclass SetgeImmPats<RegisterClass RC, Instruction SLTiOp,
|
|
|
|
Instruction SLTiuOp> {
|
2012-06-15 05:03:23 +08:00
|
|
|
def : MipsPat<(setge RC:$lhs, immSExt16:$rhs),
|
|
|
|
(XORi (SLTiOp RC:$lhs, immSExt16:$rhs), 1)>;
|
|
|
|
def : MipsPat<(setuge RC:$lhs, immSExt16:$rhs),
|
|
|
|
(XORi (SLTiuOp RC:$lhs, immSExt16:$rhs), 1)>;
|
2011-10-12 05:40:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
defm : SeteqPats<CPURegs, SLTiu, XOR, SLTu, ZERO>;
|
|
|
|
defm : SetlePats<CPURegs, SLT, SLTu>;
|
|
|
|
defm : SetgtPats<CPURegs, SLT, SLTu>;
|
|
|
|
defm : SetgePats<CPURegs, SLT, SLTu>;
|
|
|
|
defm : SetgeImmPats<CPURegs, SLTi, SLTiu>;
|
Several changes to Mips backend, experimental fp support being the most
important.
- Cleanup in the Subtarget info with addition of new features, not all support
yet, but they allow the future inclusion of features easier. Among new features,
we have : Arch family info (mips1, mips2, ...), ABI info (o32, eabi), 64-bit
integer
and float registers, allegrex vector FPU (VFPU), single float only support.
- TargetMachine now detects allegrex core.
- Added allegrex (Mips32r2) sext_inreg instructions.
- *Added Float Point Instructions*, handling single float only, and
aliased accesses for 32-bit FPUs.
- Some cleanup in FP instruction formats and FP register classes.
- Calling conventions improved to support mips 32-bit EABI.
- Added Asm Printer support for fp cond codes.
- Added support for sret copy to a return register.
- EABI support added into LowerCALL and FORMAL_ARGS.
- MipsFunctionInfo now keeps a virtual register per function to track the
sret on function entry until function ret.
- MipsInstrInfo FP support into methods (isMoveInstr, isLoadFromStackSlot, ...),
FP cond codes mapping and initial FP Branch Analysis.
- Two new Mips SDNode to handle fp branch and compare instructions : FPBrcond,
FPCmp
- MipsTargetLowering : handling different FP classes, Allegrex support, sret
return copy, no homing location within EABI, non 32-bit stack objects
arguments, and asm constraint for float.
llvm-svn: 53146
2008-07-06 03:05:21 +08:00
|
|
|
|
2011-12-21 07:47:44 +08:00
|
|
|
// bswap pattern
|
2012-06-15 05:03:23 +08:00
|
|
|
def : MipsPat<(bswap CPURegs:$rt), (ROTR (WSBH CPURegs:$rt), 16)>;
|
2011-12-21 07:47:44 +08:00
|
|
|
|
2011-04-16 05:51:11 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
Several changes to Mips backend, experimental fp support being the most
important.
- Cleanup in the Subtarget info with addition of new features, not all support
yet, but they allow the future inclusion of features easier. Among new features,
we have : Arch family info (mips1, mips2, ...), ABI info (o32, eabi), 64-bit
integer
and float registers, allegrex vector FPU (VFPU), single float only support.
- TargetMachine now detects allegrex core.
- Added allegrex (Mips32r2) sext_inreg instructions.
- *Added Float Point Instructions*, handling single float only, and
aliased accesses for 32-bit FPUs.
- Some cleanup in FP instruction formats and FP register classes.
- Calling conventions improved to support mips 32-bit EABI.
- Added Asm Printer support for fp cond codes.
- Added support for sret copy to a return register.
- EABI support added into LowerCALL and FORMAL_ARGS.
- MipsFunctionInfo now keeps a virtual register per function to track the
sret on function entry until function ret.
- MipsInstrInfo FP support into methods (isMoveInstr, isLoadFromStackSlot, ...),
FP cond codes mapping and initial FP Branch Analysis.
- Two new Mips SDNode to handle fp branch and compare instructions : FPBrcond,
FPCmp
- MipsTargetLowering : handling different FP classes, Allegrex support, sret
return copy, no homing location within EABI, non 32-bit stack objects
arguments, and asm constraint for float.
llvm-svn: 53146
2008-07-06 03:05:21 +08:00
|
|
|
// Floating Point Support
|
2011-04-16 05:51:11 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
Several changes to Mips backend, experimental fp support being the most
important.
- Cleanup in the Subtarget info with addition of new features, not all support
yet, but they allow the future inclusion of features easier. Among new features,
we have : Arch family info (mips1, mips2, ...), ABI info (o32, eabi), 64-bit
integer
and float registers, allegrex vector FPU (VFPU), single float only support.
- TargetMachine now detects allegrex core.
- Added allegrex (Mips32r2) sext_inreg instructions.
- *Added Float Point Instructions*, handling single float only, and
aliased accesses for 32-bit FPUs.
- Some cleanup in FP instruction formats and FP register classes.
- Calling conventions improved to support mips 32-bit EABI.
- Added Asm Printer support for fp cond codes.
- Added support for sret copy to a return register.
- EABI support added into LowerCALL and FORMAL_ARGS.
- MipsFunctionInfo now keeps a virtual register per function to track the
sret on function entry until function ret.
- MipsInstrInfo FP support into methods (isMoveInstr, isLoadFromStackSlot, ...),
FP cond codes mapping and initial FP Branch Analysis.
- Two new Mips SDNode to handle fp branch and compare instructions : FPBrcond,
FPCmp
- MipsTargetLowering : handling different FP classes, Allegrex support, sret
return copy, no homing location within EABI, non 32-bit stack objects
arguments, and asm constraint for float.
llvm-svn: 53146
2008-07-06 03:05:21 +08:00
|
|
|
|
|
|
|
include "MipsInstrFPU.td"
|
2011-09-24 09:34:44 +08:00
|
|
|
include "Mips64InstrInfo.td"
|
2011-10-18 02:53:29 +08:00
|
|
|
include "MipsCondMov.td"
|
2011-04-16 05:51:11 +08:00
|
|
|
|
2012-05-09 03:08:58 +08:00
|
|
|
//
|
|
|
|
// Mips16
|
|
|
|
|
|
|
|
include "Mips16InstrFormats.td"
|
2012-05-25 02:32:33 +08:00
|
|
|
include "Mips16InstrInfo.td"
|
2012-09-27 09:50:59 +08:00
|
|
|
|
|
|
|
// DSP
|
|
|
|
include "MipsDSPInstrFormats.td"
|
|
|
|
include "MipsDSPInstrInfo.td"
|
|
|
|
|