12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736 |
- //== LoongArchInstrInfo.td - Target Description for LoongArch -*- tablegen -*-//
- //
- // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
- // See https://llvm.org/LICENSE.txt for license information.
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- //
- //===----------------------------------------------------------------------===//
- //
- // This file describes the LoongArch instructions in TableGen format.
- //
- //===----------------------------------------------------------------------===//
- //===----------------------------------------------------------------------===//
- // LoongArch specific DAG Nodes.
- //===----------------------------------------------------------------------===//
- // Target-independent type requirements, but with target-specific formats.
- def SDT_CallSeqStart : SDCallSeqStart<[SDTCisVT<0, i32>,
- SDTCisVT<1, i32>]>;
- def SDT_CallSeqEnd : SDCallSeqEnd<[SDTCisVT<0, i32>,
- SDTCisVT<1, i32>]>;
- // Target-dependent type requirements.
- def SDT_LoongArchCall : SDTypeProfile<0, -1, [SDTCisVT<0, GRLenVT>]>;
- def SDT_LoongArchIntBinOpW : SDTypeProfile<1, 2, [
- SDTCisSameAs<0, 1>, SDTCisSameAs<0, 2>, SDTCisVT<0, i64>
- ]>;
- def SDT_LoongArchBStrIns: SDTypeProfile<1, 4, [
- SDTCisInt<0>, SDTCisSameAs<0, 1>, SDTCisSameAs<0, 2>, SDTCisInt<3>,
- SDTCisSameAs<3, 4>
- ]>;
- def SDT_LoongArchBStrPick: SDTypeProfile<1, 3, [
- SDTCisInt<0>, SDTCisSameAs<0, 1>, SDTCisInt<2>, SDTCisSameAs<2, 3>
- ]>;
- // "VI" means no output and an integer input.
- def SDT_LoongArchVI : SDTypeProfile<0, 1, [SDTCisVT<0, GRLenVT>]>;
- def SDT_LoongArchCsrrd : SDTypeProfile<1, 1, [SDTCisInt<0>,
- SDTCisVT<1, GRLenVT>]>;
- def SDT_LoongArchCsrwr : SDTypeProfile<1, 2, [SDTCisInt<0>, SDTCisSameAs<0, 1>,
- SDTCisVT<2, GRLenVT>]>;
- def SDT_LoongArchCsrxchg : SDTypeProfile<1, 3, [SDTCisInt<0>,
- SDTCisSameAs<0, 1>,
- SDTCisSameAs<0, 2>,
- SDTCisVT<3, GRLenVT>]>;
- def SDT_LoongArchIocsrwr : SDTypeProfile<0, 2, [SDTCisInt<0>,
- SDTCisSameAs<0, 1>]>;
- def SDT_LoongArchMovgr2fcsr : SDTypeProfile<0, 2, [SDTCisVT<0, GRLenVT>,
- SDTCisSameAs<0, 1>]>;
- def SDT_LoongArchMovfcsr2gr : SDTypeProfile<1, 1, [SDTCisVT<0, GRLenVT>,
- SDTCisSameAs<0, 1>]>;
- // TODO: Add LoongArch specific DAG Nodes
- // Target-independent nodes, but with target-specific formats.
- def callseq_start : SDNode<"ISD::CALLSEQ_START", SDT_CallSeqStart,
- [SDNPHasChain, SDNPOutGlue]>;
- def callseq_end : SDNode<"ISD::CALLSEQ_END", SDT_CallSeqEnd,
- [SDNPHasChain, SDNPOptInGlue, SDNPOutGlue]>;
- // Target-dependent nodes.
- def loongarch_call : SDNode<"LoongArchISD::CALL", SDT_LoongArchCall,
- [SDNPHasChain, SDNPOptInGlue, SDNPOutGlue,
- SDNPVariadic]>;
- def loongarch_ret : SDNode<"LoongArchISD::RET", SDTNone,
- [SDNPHasChain, SDNPOptInGlue, SDNPVariadic]>;
- def loongarch_tail : SDNode<"LoongArchISD::TAIL", SDT_LoongArchCall,
- [SDNPHasChain, SDNPOptInGlue, SDNPOutGlue,
- SDNPVariadic]>;
- def loongarch_sll_w : SDNode<"LoongArchISD::SLL_W", SDT_LoongArchIntBinOpW>;
- def loongarch_sra_w : SDNode<"LoongArchISD::SRA_W", SDT_LoongArchIntBinOpW>;
- def loongarch_srl_w : SDNode<"LoongArchISD::SRL_W", SDT_LoongArchIntBinOpW>;
- def loongarch_rotr_w : SDNode<"LoongArchISD::ROTR_W", SDT_LoongArchIntBinOpW>;
- def loongarch_rotl_w : SDNode<"LoongArchISD::ROTL_W", SDT_LoongArchIntBinOpW>;
- def loongarch_crc_w_b_w
- : SDNode<"LoongArchISD::CRC_W_B_W", SDT_LoongArchIntBinOpW>;
- def loongarch_crc_w_h_w
- : SDNode<"LoongArchISD::CRC_W_H_W", SDT_LoongArchIntBinOpW>;
- def loongarch_crc_w_w_w
- : SDNode<"LoongArchISD::CRC_W_W_W", SDT_LoongArchIntBinOpW>;
- def loongarch_crc_w_d_w
- : SDNode<"LoongArchISD::CRC_W_D_W", SDT_LoongArchIntBinOpW>;
- def loongarch_crcc_w_b_w
- : SDNode<"LoongArchISD::CRCC_W_B_W", SDT_LoongArchIntBinOpW>;
- def loongarch_crcc_w_h_w
- : SDNode<"LoongArchISD::CRCC_W_H_W", SDT_LoongArchIntBinOpW>;
- def loongarch_crcc_w_w_w
- : SDNode<"LoongArchISD::CRCC_W_W_W", SDT_LoongArchIntBinOpW>;
- def loongarch_crcc_w_d_w
- : SDNode<"LoongArchISD::CRCC_W_D_W", SDT_LoongArchIntBinOpW>;
- def loongarch_bstrins
- : SDNode<"LoongArchISD::BSTRINS", SDT_LoongArchBStrIns>;
- def loongarch_bstrpick
- : SDNode<"LoongArchISD::BSTRPICK", SDT_LoongArchBStrPick>;
- def loongarch_revb_2h : SDNode<"LoongArchISD::REVB_2H", SDTUnaryOp>;
- def loongarch_revb_2w : SDNode<"LoongArchISD::REVB_2W", SDTUnaryOp>;
- def loongarch_bitrev_4b : SDNode<"LoongArchISD::BITREV_4B", SDTUnaryOp>;
- def loongarch_bitrev_w : SDNode<"LoongArchISD::BITREV_W", SDTUnaryOp>;
- def loongarch_clzw : SDNode<"LoongArchISD::CLZ_W", SDTIntBitCountUnaryOp>;
- def loongarch_ctzw : SDNode<"LoongArchISD::CTZ_W", SDTIntBitCountUnaryOp>;
- def loongarch_dbar : SDNode<"LoongArchISD::DBAR", SDT_LoongArchVI,
- [SDNPHasChain, SDNPSideEffect]>;
- def loongarch_ibar : SDNode<"LoongArchISD::IBAR", SDT_LoongArchVI,
- [SDNPHasChain, SDNPSideEffect]>;
- def loongarch_break : SDNode<"LoongArchISD::BREAK", SDT_LoongArchVI,
- [SDNPHasChain, SDNPSideEffect]>;
- def loongarch_movfcsr2gr : SDNode<"LoongArchISD::MOVFCSR2GR", SDT_LoongArchMovfcsr2gr>;
- def loongarch_movgr2fcsr : SDNode<"LoongArchISD::MOVGR2FCSR", SDT_LoongArchMovgr2fcsr,
- [SDNPHasChain, SDNPSideEffect]>;
- def loongarch_syscall : SDNode<"LoongArchISD::SYSCALL", SDT_LoongArchVI,
- [SDNPHasChain, SDNPSideEffect]>;
- def loongarch_csrrd : SDNode<"LoongArchISD::CSRRD", SDT_LoongArchCsrrd,
- [SDNPHasChain, SDNPSideEffect]>;
- def loongarch_csrwr : SDNode<"LoongArchISD::CSRWR", SDT_LoongArchCsrwr,
- [SDNPHasChain, SDNPSideEffect]>;
- def loongarch_csrxchg : SDNode<"LoongArchISD::CSRXCHG",
- SDT_LoongArchCsrxchg,
- [SDNPHasChain, SDNPSideEffect]>;
- def loongarch_iocsrrd_b : SDNode<"LoongArchISD::IOCSRRD_B", SDTUnaryOp,
- [SDNPHasChain, SDNPSideEffect]>;
- def loongarch_iocsrrd_h : SDNode<"LoongArchISD::IOCSRRD_H", SDTUnaryOp,
- [SDNPHasChain, SDNPSideEffect]>;
- def loongarch_iocsrrd_w : SDNode<"LoongArchISD::IOCSRRD_W", SDTUnaryOp,
- [SDNPHasChain, SDNPSideEffect]>;
- def loongarch_iocsrrd_d : SDNode<"LoongArchISD::IOCSRRD_D", SDTUnaryOp,
- [SDNPHasChain, SDNPSideEffect]>;
- def loongarch_iocsrwr_b : SDNode<"LoongArchISD::IOCSRWR_B",
- SDT_LoongArchIocsrwr,
- [SDNPHasChain, SDNPSideEffect]>;
- def loongarch_iocsrwr_h : SDNode<"LoongArchISD::IOCSRWR_H",
- SDT_LoongArchIocsrwr,
- [SDNPHasChain, SDNPSideEffect]>;
- def loongarch_iocsrwr_w : SDNode<"LoongArchISD::IOCSRWR_W",
- SDT_LoongArchIocsrwr,
- [SDNPHasChain, SDNPSideEffect]>;
- def loongarch_iocsrwr_d : SDNode<"LoongArchISD::IOCSRWR_D",
- SDT_LoongArchIocsrwr,
- [SDNPHasChain, SDNPSideEffect]>;
- def loongarch_cpucfg : SDNode<"LoongArchISD::CPUCFG", SDTUnaryOp,
- [SDNPHasChain, SDNPSideEffect]>;
- //===----------------------------------------------------------------------===//
- // Operand and SDNode transformation definitions.
- //===----------------------------------------------------------------------===//
- class ImmAsmOperand<string prefix, int width, string suffix>
- : AsmOperandClass {
- let Name = prefix # "Imm" # width # suffix;
- let DiagnosticType = !strconcat("Invalid", Name);
- let RenderMethod = "addImmOperands";
- }
- class SImmAsmOperand<int width, string suffix = "">
- : ImmAsmOperand<"S", width, suffix> {
- }
- class UImmAsmOperand<int width, string suffix = "">
- : ImmAsmOperand<"U", width, suffix> {
- }
- // A parse method for "$r*" or "$r*, 0", where the 0 is be silently ignored.
- // Only used for "AM*" instructions, in order to be compatible with GAS.
- def AtomicMemAsmOperand : AsmOperandClass {
- let Name = "AtomicMemAsmOperand";
- let RenderMethod = "addRegOperands";
- let PredicateMethod = "isGPR";
- let ParserMethod = "parseAtomicMemOp";
- }
- def GPRMemAtomic : RegisterOperand<GPR> {
- let ParserMatchClass = AtomicMemAsmOperand;
- let PrintMethod = "printAtomicMemOp";
- }
- // A parameterized register class alternative to i32imm/i64imm from Target.td.
- def grlenimm : Operand<GRLenVT>;
- def imm32 : Operand<GRLenVT> {
- let ParserMatchClass = ImmAsmOperand<"", 32, "">;
- }
- def uimm2 : Operand<GRLenVT>, ImmLeaf<GRLenVT, [{return isUInt<2>(Imm);}]> {
- let ParserMatchClass = UImmAsmOperand<2>;
- }
- def uimm2_plus1 : Operand<GRLenVT>,
- ImmLeaf<GRLenVT, [{return isUInt<2>(Imm - 1);}]> {
- let ParserMatchClass = UImmAsmOperand<2, "plus1">;
- let EncoderMethod = "getImmOpValueSub1";
- let DecoderMethod = "decodeUImmOperand<2, 1>";
- }
- def uimm3 : Operand<GRLenVT> {
- let ParserMatchClass = UImmAsmOperand<3>;
- }
- def uimm5 : Operand<GRLenVT>, ImmLeaf<GRLenVT, [{return isUInt<5>(Imm);}]> {
- let ParserMatchClass = UImmAsmOperand<5>;
- }
- def uimm6 : Operand<GRLenVT>, ImmLeaf<GRLenVT, [{return isUInt<6>(Imm);}]> {
- let ParserMatchClass = UImmAsmOperand<6>;
- }
- def uimm8 : Operand<GRLenVT>, ImmLeaf<GRLenVT, [{return isUInt<8>(Imm);}]> {
- let ParserMatchClass = UImmAsmOperand<8>;
- }
- class UImm12Operand : Operand<GRLenVT>,
- ImmLeaf <GRLenVT, [{return isUInt<12>(Imm);}]> {
- let DecoderMethod = "decodeUImmOperand<12>";
- }
- def uimm12 : UImm12Operand {
- let ParserMatchClass = UImmAsmOperand<12>;
- }
- def uimm12_ori : UImm12Operand {
- let ParserMatchClass = UImmAsmOperand<12, "ori">;
- }
- def uimm14 : Operand<GRLenVT>,
- ImmLeaf <GRLenVT, [{return isUInt<14>(Imm);}]> {
- let ParserMatchClass = UImmAsmOperand<14>;
- }
- def uimm15 : Operand<GRLenVT>,
- ImmLeaf <GRLenVT, [{return isUInt<15>(Imm);}]> {
- let ParserMatchClass = UImmAsmOperand<15>;
- }
- class SImm12Operand : Operand<GRLenVT>,
- ImmLeaf <GRLenVT, [{return isInt<12>(Imm);}]> {
- let DecoderMethod = "decodeSImmOperand<12>";
- }
- def simm12 : SImm12Operand {
- let ParserMatchClass = SImmAsmOperand<12>;
- }
- def simm12_addlike : SImm12Operand {
- let ParserMatchClass = SImmAsmOperand<12, "addlike">;
- }
- def simm12_lu52id : SImm12Operand {
- let ParserMatchClass = SImmAsmOperand<12, "lu52id">;
- }
- def simm14_lsl2 : Operand<GRLenVT>,
- ImmLeaf<GRLenVT, [{return isShiftedInt<14,2>(Imm);}]> {
- let ParserMatchClass = SImmAsmOperand<14, "lsl2">;
- let EncoderMethod = "getImmOpValueAsr2";
- let DecoderMethod = "decodeSImmOperand<14, 2>";
- }
- def simm16 : Operand<GRLenVT> {
- let ParserMatchClass = SImmAsmOperand<16>;
- let DecoderMethod = "decodeSImmOperand<16>";
- }
- def simm16_lsl2 : Operand<GRLenVT>,
- ImmLeaf<GRLenVT, [{return isInt<16>(Imm>>2);}]> {
- let ParserMatchClass = SImmAsmOperand<16, "lsl2">;
- let EncoderMethod = "getImmOpValueAsr2";
- let DecoderMethod = "decodeSImmOperand<16, 2>";
- }
- def simm16_lsl2_br : Operand<OtherVT> {
- let ParserMatchClass = SImmAsmOperand<16, "lsl2">;
- let EncoderMethod = "getImmOpValueAsr2";
- let DecoderMethod = "decodeSImmOperand<16, 2>";
- }
- class SImm20Operand : Operand<GRLenVT> {
- let DecoderMethod = "decodeSImmOperand<20>";
- }
- def simm20 : SImm20Operand {
- let ParserMatchClass = SImmAsmOperand<20>;
- }
- def simm20_pcalau12i : SImm20Operand {
- let ParserMatchClass = SImmAsmOperand<20, "pcalau12i">;
- }
- def simm20_lu12iw : SImm20Operand {
- let ParserMatchClass = SImmAsmOperand<20, "lu12iw">;
- }
- def simm20_lu32id : SImm20Operand {
- let ParserMatchClass = SImmAsmOperand<20, "lu32id">;
- }
- def simm21_lsl2 : Operand<OtherVT> {
- let ParserMatchClass = SImmAsmOperand<21, "lsl2">;
- let EncoderMethod = "getImmOpValueAsr2";
- let DecoderMethod = "decodeSImmOperand<21, 2>";
- }
- def SImm26OperandB: AsmOperandClass {
- let Name = "SImm26OperandB";
- let PredicateMethod = "isSImm26Operand";
- let RenderMethod = "addImmOperands";
- let DiagnosticType = "InvalidSImm26Operand";
- let ParserMethod = "parseImmediate";
- }
- // A symbol or an imm used in B/PseudoBR.
- def simm26_b : Operand<OtherVT> {
- let ParserMatchClass = SImm26OperandB;
- let EncoderMethod = "getImmOpValueAsr2";
- let DecoderMethod = "decodeSImmOperand<26, 2>";
- }
- def SImm26OperandBL: AsmOperandClass {
- let Name = "SImm26OperandBL";
- let PredicateMethod = "isSImm26Operand";
- let RenderMethod = "addImmOperands";
- let DiagnosticType = "InvalidSImm26Operand";
- let ParserMethod = "parseSImm26Operand";
- }
- // A symbol or an imm used in BL/PseudoCALL/PseudoTAIL.
- def simm26_symbol : Operand<GRLenVT> {
- let ParserMatchClass = SImm26OperandBL;
- let EncoderMethod = "getImmOpValueAsr2";
- let DecoderMethod = "decodeSImmOperand<26, 2>";
- }
- def BareSymbol : AsmOperandClass {
- let Name = "BareSymbol";
- let RenderMethod = "addImmOperands";
- let DiagnosticType = "InvalidBareSymbol";
- let ParserMethod = "parseImmediate";
- }
- // A bare symbol used in "PseudoLA_*" instructions.
- def bare_symbol : Operand<GRLenVT> {
- let ParserMatchClass = BareSymbol;
- }
- // Standalone (codegen-only) immleaf patterns.
- // A 12-bit signed immediate plus one where the imm range will be [-2047, 2048].
- def simm12_plus1 : ImmLeaf<GRLenVT,
- [{return (isInt<12>(Imm) && Imm != -2048) || Imm == 2048;}]>;
- // Return the negation of an immediate value.
- def NegImm : SDNodeXForm<imm, [{
- return CurDAG->getTargetConstant(-N->getSExtValue(), SDLoc(N),
- N->getValueType(0));
- }]>;
- // FP immediate patterns.
- def fpimm0 : PatLeaf<(fpimm), [{return N->isExactlyValue(+0.0);}]>;
- def fpimm0neg : PatLeaf<(fpimm), [{return N->isExactlyValue(-0.0);}]>;
- def fpimm1 : PatLeaf<(fpimm), [{return N->isExactlyValue(+1.0);}]>;
- // Return an immediate subtracted from 32.
- def ImmSubFrom32 : SDNodeXForm<imm, [{
- return CurDAG->getTargetConstant(32 - N->getZExtValue(), SDLoc(N),
- N->getValueType(0));
- }]>;
- def BaseAddr : ComplexPattern<iPTR, 1, "SelectBaseAddr">;
- def NonFIBaseAddr : ComplexPattern<iPTR, 1, "selectNonFIBaseAddr">;
- def fma_nsz : PatFrag<(ops node:$fj, node:$fk, node:$fa),
- (fma node:$fj, node:$fk, node:$fa), [{
- return N->getFlags().hasNoSignedZeros();
- }]>;
- //===----------------------------------------------------------------------===//
- // Instruction Formats
- //===----------------------------------------------------------------------===//
- include "LoongArchInstrFormats.td"
- include "LoongArchFloatInstrFormats.td"
- //===----------------------------------------------------------------------===//
- // Instruction Class Templates
- //===----------------------------------------------------------------------===//
- class ALU_3R<bits<17> op, string opstr>
- : Fmt3R<op, (outs GPR:$rd), (ins GPR:$rj, GPR:$rk), opstr, "$rd, $rj, $rk">;
- class ALU_2R<bits<22> op, string opstr>
- : Fmt2R<op, (outs GPR:$rd), (ins GPR:$rj), opstr, "$rd, $rj">;
- class ALU_3RI2<bits<15> op, string opstr, Operand ImmOpnd>
- : Fmt3RI2<op, (outs GPR:$rd), (ins GPR:$rj, GPR:$rk, ImmOpnd:$imm2), opstr,
- "$rd, $rj, $rk, $imm2">;
- class ALU_3RI3<bits<14> op, string opstr, Operand ImmOpnd>
- : Fmt3RI3<op, (outs GPR:$rd), (ins GPR:$rj, GPR:$rk, ImmOpnd:$imm3), opstr,
- "$rd, $rj, $rk, $imm3">;
- class ALU_2RI5<bits<17> op, string opstr, Operand ImmOpnd>
- : Fmt2RI5<op, (outs GPR:$rd), (ins GPR:$rj, ImmOpnd:$imm5), opstr,
- "$rd, $rj, $imm5">;
- class ALU_2RI6<bits<16> op, string opstr, Operand ImmOpnd>
- : Fmt2RI6<op, (outs GPR:$rd), (ins GPR:$rj, ImmOpnd:$imm6), opstr,
- "$rd, $rj, $imm6">;
- class ALU_2RI12<bits<10> op, string opstr, Operand ImmOpnd>
- : Fmt2RI12<op, (outs GPR:$rd), (ins GPR:$rj, ImmOpnd:$imm12), opstr,
- "$rd, $rj, $imm12">;
- class ALU_2RI16<bits<6> op, string opstr, Operand ImmOpnd>
- : Fmt2RI16<op, (outs GPR:$rd), (ins GPR:$rj, ImmOpnd:$imm16), opstr,
- "$rd, $rj, $imm16">;
- class ALU_1RI20<bits<7> op, string opstr, Operand ImmOpnd>
- : Fmt1RI20<op, (outs GPR:$rd), (ins ImmOpnd:$imm20), opstr, "$rd, $imm20">;
- class MISC_I15<bits<17> op, string opstr>
- : FmtI15<op, (outs), (ins uimm15:$imm15), opstr, "$imm15">;
- class RDTIME_2R<bits<22> op, string opstr>
- : Fmt2R<op, (outs GPR:$rd, GPR:$rj), (ins), opstr, "$rd, $rj">;
- class BrCC_2RI16<bits<6> op, string opstr>
- : Fmt2RI16<op, (outs), (ins GPR:$rj, GPR:$rd, simm16_lsl2_br:$imm16), opstr,
- "$rj, $rd, $imm16"> {
- let isBranch = 1;
- let isTerminator = 1;
- }
- class BrCCZ_1RI21<bits<6> op, string opstr>
- : Fmt1RI21<op, (outs), (ins GPR:$rj, simm21_lsl2:$imm21), opstr,
- "$rj, $imm21"> {
- let isBranch = 1;
- let isTerminator = 1;
- }
- class Br_I26<bits<6> op, string opstr>
- : FmtI26<op, (outs), (ins simm26_b:$imm26), opstr, "$imm26"> {
- let isBranch = 1;
- let isTerminator = 1;
- }
- let mayLoad = 1 in {
- class LOAD_3R<bits<17> op, string opstr>
- : Fmt3R<op, (outs GPR:$rd), (ins GPR:$rj, GPR:$rk), opstr, "$rd, $rj, $rk">;
- class LOAD_2RI12<bits<10> op, string opstr>
- : Fmt2RI12<op, (outs GPR:$rd), (ins GPR:$rj, simm12_addlike:$imm12), opstr,
- "$rd, $rj, $imm12">;
- class LOAD_2RI14<bits<8> op, string opstr>
- : Fmt2RI14<op, (outs GPR:$rd), (ins GPR:$rj, simm14_lsl2:$imm14), opstr,
- "$rd, $rj, $imm14">;
- } // mayLoad = 1
- let mayStore = 1 in {
- class STORE_3R<bits<17> op, string opstr>
- : Fmt3R<op, (outs), (ins GPR:$rd, GPR:$rj, GPR:$rk), opstr,
- "$rd, $rj, $rk">;
- class STORE_2RI12<bits<10> op, string opstr>
- : Fmt2RI12<op, (outs), (ins GPR:$rd, GPR:$rj, simm12_addlike:$imm12), opstr,
- "$rd, $rj, $imm12">;
- class STORE_2RI14<bits<8> op, string opstr>
- : Fmt2RI14<op, (outs), (ins GPR:$rd, GPR:$rj, simm14_lsl2:$imm14), opstr,
- "$rd, $rj, $imm14">;
- } // mayStore = 1
- let mayLoad = 1, mayStore = 1, Constraints = "@earlyclobber $rd" in
- class AM_3R<bits<17> op, string opstr>
- : Fmt3R<op, (outs GPR:$rd), (ins GPR:$rk, GPRMemAtomic:$rj), opstr,
- "$rd, $rk, $rj">;
- let mayLoad = 1 in
- class LLBase<bits<8> op, string opstr>
- : Fmt2RI14<op, (outs GPR:$rd), (ins GPR:$rj, simm14_lsl2:$imm14), opstr,
- "$rd, $rj, $imm14">;
- let mayStore = 1, Constraints = "$rd = $dst" in
- class SCBase<bits<8> op, string opstr>
- : Fmt2RI14<op, (outs GPR:$dst), (ins GPR:$rd, GPR:$rj, simm14_lsl2:$imm14),
- opstr, "$rd, $rj, $imm14">;
- class IOCSRRD<bits<22> op, string opstr>
- : Fmt2R<op, (outs GPR:$rd), (ins GPR:$rj), opstr, "$rd, $rj">;
- class IOCSRWR<bits<22> op, string opstr>
- : Fmt2R<op, (outs), (ins GPR:$rd, GPR:$rj), opstr, "$rd, $rj">;
- //===----------------------------------------------------------------------===//
- // Basic Integer Instructions
- //===----------------------------------------------------------------------===//
- // Arithmetic Operation Instructions
- def ADD_W : ALU_3R<0b00000000000100000, "add.w">;
- def SUB_W : ALU_3R<0b00000000000100010, "sub.w">;
- def ADDI_W : ALU_2RI12<0b0000001010, "addi.w", simm12_addlike>;
- def ALSL_W : ALU_3RI2<0b000000000000010, "alsl.w", uimm2_plus1>;
- def LU12I_W : ALU_1RI20<0b0001010, "lu12i.w", simm20_lu12iw>;
- def SLT : ALU_3R<0b00000000000100100, "slt">;
- def SLTU : ALU_3R<0b00000000000100101, "sltu">;
- def SLTI : ALU_2RI12<0b0000001000, "slti", simm12>;
- def SLTUI : ALU_2RI12<0b0000001001, "sltui", simm12>;
- def PCADDI : ALU_1RI20<0b0001100, "pcaddi", simm20>;
- def PCADDU12I : ALU_1RI20<0b0001110, "pcaddu12i", simm20>;
- def PCALAU12I : ALU_1RI20<0b0001101, "pcalau12i", simm20_pcalau12i>;
- def AND : ALU_3R<0b00000000000101001, "and">;
- def OR : ALU_3R<0b00000000000101010, "or">;
- def NOR : ALU_3R<0b00000000000101000, "nor">;
- def XOR : ALU_3R<0b00000000000101011, "xor">;
- def ANDN : ALU_3R<0b00000000000101101, "andn">;
- def ORN : ALU_3R<0b00000000000101100, "orn">;
- def ANDI : ALU_2RI12<0b0000001101, "andi", uimm12>;
- def ORI : ALU_2RI12<0b0000001110, "ori", uimm12_ori>;
- def XORI : ALU_2RI12<0b0000001111, "xori", uimm12>;
- def MUL_W : ALU_3R<0b00000000000111000, "mul.w">;
- def MULH_W : ALU_3R<0b00000000000111001, "mulh.w">;
- def MULH_WU : ALU_3R<0b00000000000111010, "mulh.wu">;
- let usesCustomInserter = true in {
- def DIV_W : ALU_3R<0b00000000001000000, "div.w">;
- def MOD_W : ALU_3R<0b00000000001000001, "mod.w">;
- def DIV_WU : ALU_3R<0b00000000001000010, "div.wu">;
- def MOD_WU : ALU_3R<0b00000000001000011, "mod.wu">;
- } // usesCustomInserter = true
- // Bit-shift Instructions
- def SLL_W : ALU_3R<0b00000000000101110, "sll.w">;
- def SRL_W : ALU_3R<0b00000000000101111, "srl.w">;
- def SRA_W : ALU_3R<0b00000000000110000, "sra.w">;
- def ROTR_W : ALU_3R<0b00000000000110110, "rotr.w">;
- def SLLI_W : ALU_2RI5<0b00000000010000001, "slli.w", uimm5>;
- def SRLI_W : ALU_2RI5<0b00000000010001001, "srli.w", uimm5>;
- def SRAI_W : ALU_2RI5<0b00000000010010001, "srai.w", uimm5>;
- def ROTRI_W : ALU_2RI5<0b00000000010011001, "rotri.w", uimm5>;
- // Bit-manipulation Instructions
- def EXT_W_B : ALU_2R<0b0000000000000000010111, "ext.w.b">;
- def EXT_W_H : ALU_2R<0b0000000000000000010110, "ext.w.h">;
- def CLO_W : ALU_2R<0b0000000000000000000100, "clo.w">;
- def CLZ_W : ALU_2R<0b0000000000000000000101, "clz.w">;
- def CTO_W : ALU_2R<0b0000000000000000000110, "cto.w">;
- def CTZ_W : ALU_2R<0b0000000000000000000111, "ctz.w">;
- def BYTEPICK_W : ALU_3RI2<0b000000000000100, "bytepick.w", uimm2>;
- def REVB_2H : ALU_2R<0b0000000000000000001100, "revb.2h">;
- def BITREV_4B : ALU_2R<0b0000000000000000010010, "bitrev.4b">;
- def BITREV_W : ALU_2R<0b0000000000000000010100, "bitrev.w">;
- let Constraints = "$rd = $dst" in {
- def BSTRINS_W : FmtBSTR_W<0b000000000110, (outs GPR:$dst),
- (ins GPR:$rd, GPR:$rj, uimm5:$msbw, uimm5:$lsbw),
- "bstrins.w", "$rd, $rj, $msbw, $lsbw">;
- }
- def BSTRPICK_W : FmtBSTR_W<0b000000000111, (outs GPR:$rd),
- (ins GPR:$rj, uimm5:$msbw, uimm5:$lsbw),
- "bstrpick.w", "$rd, $rj, $msbw, $lsbw">;
- def MASKEQZ : ALU_3R<0b00000000000100110, "maskeqz">;
- def MASKNEZ : ALU_3R<0b00000000000100111, "masknez">;
- // Branch Instructions
- def BEQ : BrCC_2RI16<0b010110, "beq">;
- def BNE : BrCC_2RI16<0b010111, "bne">;
- def BLT : BrCC_2RI16<0b011000, "blt">;
- def BGE : BrCC_2RI16<0b011001, "bge">;
- def BLTU : BrCC_2RI16<0b011010, "bltu">;
- def BGEU : BrCC_2RI16<0b011011, "bgeu">;
- def BEQZ : BrCCZ_1RI21<0b010000, "beqz">;
- def BNEZ : BrCCZ_1RI21<0b010001, "bnez">;
- def B : Br_I26<0b010100, "b">;
- let isCall = 1, Defs=[R1] in
- def BL : FmtI26<0b010101, (outs), (ins simm26_symbol:$imm26), "bl", "$imm26">;
- def JIRL : Fmt2RI16<0b010011, (outs GPR:$rd),
- (ins GPR:$rj, simm16_lsl2:$imm16), "jirl",
- "$rd, $rj, $imm16">;
- // Common Memory Access Instructions
- def LD_B : LOAD_2RI12<0b0010100000, "ld.b">;
- def LD_H : LOAD_2RI12<0b0010100001, "ld.h">;
- def LD_W : LOAD_2RI12<0b0010100010, "ld.w">;
- def LD_BU : LOAD_2RI12<0b0010101000, "ld.bu">;
- def LD_HU : LOAD_2RI12<0b0010101001, "ld.hu">;
- def ST_B : STORE_2RI12<0b0010100100, "st.b">;
- def ST_H : STORE_2RI12<0b0010100101, "st.h">;
- def ST_W : STORE_2RI12<0b0010100110, "st.w">;
- def PRELD : FmtPRELD<(outs), (ins uimm5:$imm5, GPR:$rj, simm12:$imm12), "preld",
- "$imm5, $rj, $imm12">;
- // Atomic Memory Access Instructions
- def LL_W : LLBase<0b00100000, "ll.w">;
- def SC_W : SCBase<0b00100001, "sc.w">;
- // Barrier Instructions
- def DBAR : MISC_I15<0b00111000011100100, "dbar">;
- def IBAR : MISC_I15<0b00111000011100101, "ibar">;
- // Other Miscellaneous Instructions
- def SYSCALL : MISC_I15<0b00000000001010110, "syscall">;
- def BREAK : MISC_I15<0b00000000001010100, "break">;
- def RDTIMEL_W : RDTIME_2R<0b0000000000000000011000, "rdtimel.w">;
- def RDTIMEH_W : RDTIME_2R<0b0000000000000000011001, "rdtimeh.w">;
- def CPUCFG : ALU_2R<0b0000000000000000011011, "cpucfg">;
- // Cache Maintenance Instructions
- def CACOP : FmtCACOP<(outs), (ins uimm5:$op, GPR:$rj, simm12:$imm12), "cacop",
- "$op, $rj, $imm12">;
- /// LA64 instructions
- let Predicates = [IsLA64] in {
- // Arithmetic Operation Instructions for 64-bits
- def ADD_D : ALU_3R<0b00000000000100001, "add.d">;
- def SUB_D : ALU_3R<0b00000000000100011, "sub.d">;
- def ADDI_D : ALU_2RI12<0b0000001011, "addi.d", simm12_addlike>;
- def ADDU16I_D : ALU_2RI16<0b000100, "addu16i.d", simm16>;
- def ALSL_WU : ALU_3RI2<0b000000000000011, "alsl.wu", uimm2_plus1>;
- def ALSL_D : ALU_3RI2<0b000000000010110, "alsl.d", uimm2_plus1>;
- let Constraints = "$rd = $dst" in {
- def LU32I_D : Fmt1RI20<0b0001011, (outs GPR:$dst),
- (ins GPR:$rd, simm20_lu32id:$imm20), "lu32i.d",
- "$rd, $imm20">;
- }
- def LU52I_D : ALU_2RI12<0b0000001100, "lu52i.d", simm12_lu52id>;
- def PCADDU18I : ALU_1RI20<0b0001111, "pcaddu18i", simm20>;
- def MUL_D : ALU_3R<0b00000000000111011, "mul.d">;
- def MULH_D : ALU_3R<0b00000000000111100, "mulh.d">;
- def MULH_DU : ALU_3R<0b00000000000111101, "mulh.du">;
- def MULW_D_W : ALU_3R<0b00000000000111110, "mulw.d.w">;
- def MULW_D_WU : ALU_3R<0b00000000000111111, "mulw.d.wu">;
- let usesCustomInserter = true in {
- def DIV_D : ALU_3R<0b00000000001000100, "div.d">;
- def MOD_D : ALU_3R<0b00000000001000101, "mod.d">;
- def DIV_DU : ALU_3R<0b00000000001000110, "div.du">;
- def MOD_DU : ALU_3R<0b00000000001000111, "mod.du">;
- } // usesCustomInserter = true
- // Bit-shift Instructions for 64-bits
- def SLL_D : ALU_3R<0b00000000000110001, "sll.d">;
- def SRL_D : ALU_3R<0b00000000000110010, "srl.d">;
- def SRA_D : ALU_3R<0b00000000000110011, "sra.d">;
- def ROTR_D : ALU_3R<0b00000000000110111, "rotr.d">;
- def SLLI_D : ALU_2RI6<0b0000000001000001, "slli.d", uimm6>;
- def SRLI_D : ALU_2RI6<0b0000000001000101, "srli.d", uimm6>;
- def SRAI_D : ALU_2RI6<0b0000000001001001, "srai.d", uimm6>;
- def ROTRI_D : ALU_2RI6<0b0000000001001101, "rotri.d", uimm6>;
- // Bit-manipulation Instructions for 64-bits
- def CLO_D : ALU_2R<0b0000000000000000001000, "clo.d">;
- def CLZ_D : ALU_2R<0b0000000000000000001001, "clz.d">;
- def CTO_D : ALU_2R<0b0000000000000000001010, "cto.d">;
- def CTZ_D : ALU_2R<0b0000000000000000001011, "ctz.d">;
- def BYTEPICK_D : ALU_3RI3<0b00000000000011, "bytepick.d", uimm3>;
- def REVB_4H : ALU_2R<0b0000000000000000001101, "revb.4h">;
- def REVB_2W : ALU_2R<0b0000000000000000001110, "revb.2w">;
- def REVB_D : ALU_2R<0b0000000000000000001111, "revb.d">;
- def REVH_2W : ALU_2R<0b0000000000000000010000, "revh.2w">;
- def REVH_D : ALU_2R<0b0000000000000000010001, "revh.d">;
- def BITREV_8B : ALU_2R<0b0000000000000000010011, "bitrev.8b">;
- def BITREV_D : ALU_2R<0b0000000000000000010101, "bitrev.d">;
- let Constraints = "$rd = $dst" in {
- def BSTRINS_D : FmtBSTR_D<0b0000000010, (outs GPR:$dst),
- (ins GPR:$rd, GPR:$rj, uimm6:$msbd, uimm6:$lsbd),
- "bstrins.d", "$rd, $rj, $msbd, $lsbd">;
- }
- def BSTRPICK_D : FmtBSTR_D<0b0000000011, (outs GPR:$rd),
- (ins GPR:$rj, uimm6:$msbd, uimm6:$lsbd),
- "bstrpick.d", "$rd, $rj, $msbd, $lsbd">;
- // Common Memory Access Instructions for 64-bits
- def LD_WU : LOAD_2RI12<0b0010101010, "ld.wu">;
- def LD_D : LOAD_2RI12<0b0010100011, "ld.d">;
- def ST_D : STORE_2RI12<0b0010100111, "st.d">;
- def LDX_B : LOAD_3R<0b00111000000000000, "ldx.b">;
- def LDX_H : LOAD_3R<0b00111000000001000, "ldx.h">;
- def LDX_W : LOAD_3R<0b00111000000010000, "ldx.w">;
- def LDX_D : LOAD_3R<0b00111000000011000, "ldx.d">;
- def LDX_BU : LOAD_3R<0b00111000001000000, "ldx.bu">;
- def LDX_HU : LOAD_3R<0b00111000001001000, "ldx.hu">;
- def LDX_WU : LOAD_3R<0b00111000001010000, "ldx.wu">;
- def STX_B : STORE_3R<0b00111000000100000, "stx.b">;
- def STX_H : STORE_3R<0b00111000000101000, "stx.h">;
- def STX_W : STORE_3R<0b00111000000110000, "stx.w">;
- def STX_D : STORE_3R<0b00111000000111000, "stx.d">;
- def LDPTR_W : LOAD_2RI14<0b00100100, "ldptr.w">;
- def LDPTR_D : LOAD_2RI14<0b00100110, "ldptr.d">;
- def STPTR_W : STORE_2RI14<0b00100101, "stptr.w">;
- def STPTR_D : STORE_2RI14<0b00100111, "stptr.d">;
- def PRELDX : FmtPRELDX<(outs), (ins uimm5:$imm5, GPR:$rj, GPR:$rk), "preldx",
- "$imm5, $rj, $rk">;
- // Bound Check Memory Access Instructions
- def LDGT_B : LOAD_3R<0b00111000011110000, "ldgt.b">;
- def LDGT_H : LOAD_3R<0b00111000011110001, "ldgt.h">;
- def LDGT_W : LOAD_3R<0b00111000011110010, "ldgt.w">;
- def LDGT_D : LOAD_3R<0b00111000011110011, "ldgt.d">;
- def LDLE_B : LOAD_3R<0b00111000011110100, "ldle.b">;
- def LDLE_H : LOAD_3R<0b00111000011110101, "ldle.h">;
- def LDLE_W : LOAD_3R<0b00111000011110110, "ldle.w">;
- def LDLE_D : LOAD_3R<0b00111000011110111, "ldle.d">;
- def STGT_B : STORE_3R<0b00111000011111000, "stgt.b">;
- def STGT_H : STORE_3R<0b00111000011111001, "stgt.h">;
- def STGT_W : STORE_3R<0b00111000011111010, "stgt.w">;
- def STGT_D : STORE_3R<0b00111000011111011, "stgt.d">;
- def STLE_B : STORE_3R<0b00111000011111100, "stle.b">;
- def STLE_H : STORE_3R<0b00111000011111101, "stle.h">;
- def STLE_W : STORE_3R<0b00111000011111110, "stle.w">;
- def STLE_D : STORE_3R<0b00111000011111111, "stle.d">;
- // Atomic Memory Access Instructions for 64-bits
- def AMSWAP_W : AM_3R<0b00111000011000000, "amswap.w">;
- def AMSWAP_D : AM_3R<0b00111000011000001, "amswap.d">;
- def AMADD_W : AM_3R<0b00111000011000010, "amadd.w">;
- def AMADD_D : AM_3R<0b00111000011000011, "amadd.d">;
- def AMAND_W : AM_3R<0b00111000011000100, "amand.w">;
- def AMAND_D : AM_3R<0b00111000011000101, "amand.d">;
- def AMOR_W : AM_3R<0b00111000011000110, "amor.w">;
- def AMOR_D : AM_3R<0b00111000011000111, "amor.d">;
- def AMXOR_W : AM_3R<0b00111000011001000, "amxor.w">;
- def AMXOR_D : AM_3R<0b00111000011001001, "amxor.d">;
- def AMMAX_W : AM_3R<0b00111000011001010, "ammax.w">;
- def AMMAX_D : AM_3R<0b00111000011001011, "ammax.d">;
- def AMMIN_W : AM_3R<0b00111000011001100, "ammin.w">;
- def AMMIN_D : AM_3R<0b00111000011001101, "ammin.d">;
- def AMMAX_WU : AM_3R<0b00111000011001110, "ammax.wu">;
- def AMMAX_DU : AM_3R<0b00111000011001111, "ammax.du">;
- def AMMIN_WU : AM_3R<0b00111000011010000, "ammin.wu">;
- def AMMIN_DU : AM_3R<0b00111000011010001, "ammin.du">;
- def AMSWAP_DB_W : AM_3R<0b00111000011010010, "amswap_db.w">;
- def AMSWAP_DB_D : AM_3R<0b00111000011010011, "amswap_db.d">;
- def AMADD_DB_W : AM_3R<0b00111000011010100, "amadd_db.w">;
- def AMADD_DB_D : AM_3R<0b00111000011010101, "amadd_db.d">;
- def AMAND_DB_W : AM_3R<0b00111000011010110, "amand_db.w">;
- def AMAND_DB_D : AM_3R<0b00111000011010111, "amand_db.d">;
- def AMOR_DB_W : AM_3R<0b00111000011011000, "amor_db.w">;
- def AMOR_DB_D : AM_3R<0b00111000011011001, "amor_db.d">;
- def AMXOR_DB_W : AM_3R<0b00111000011011010, "amxor_db.w">;
- def AMXOR_DB_D : AM_3R<0b00111000011011011, "amxor_db.d">;
- def AMMAX_DB_W : AM_3R<0b00111000011011100, "ammax_db.w">;
- def AMMAX_DB_D : AM_3R<0b00111000011011101, "ammax_db.d">;
- def AMMIN_DB_W : AM_3R<0b00111000011011110, "ammin_db.w">;
- def AMMIN_DB_D : AM_3R<0b00111000011011111, "ammin_db.d">;
- def AMMAX_DB_WU : AM_3R<0b00111000011100000, "ammax_db.wu">;
- def AMMAX_DB_DU : AM_3R<0b00111000011100001, "ammax_db.du">;
- def AMMIN_DB_WU : AM_3R<0b00111000011100010, "ammin_db.wu">;
- def AMMIN_DB_DU : AM_3R<0b00111000011100011, "ammin_db.du">;
- def LL_D : LLBase<0b00100010, "ll.d">;
- def SC_D : SCBase<0b00100011, "sc.d">;
- // CRC Check Instructions
- def CRC_W_B_W : ALU_3R<0b00000000001001000, "crc.w.b.w">;
- def CRC_W_H_W : ALU_3R<0b00000000001001001, "crc.w.h.w">;
- def CRC_W_W_W : ALU_3R<0b00000000001001010, "crc.w.w.w">;
- def CRC_W_D_W : ALU_3R<0b00000000001001011, "crc.w.d.w">;
- def CRCC_W_B_W : ALU_3R<0b00000000001001100, "crcc.w.b.w">;
- def CRCC_W_H_W : ALU_3R<0b00000000001001101, "crcc.w.h.w">;
- def CRCC_W_W_W : ALU_3R<0b00000000001001110, "crcc.w.w.w">;
- def CRCC_W_D_W : ALU_3R<0b00000000001001111, "crcc.w.d.w">;
- // Other Miscellaneous Instructions for 64-bits
- def ASRTLE_D : FmtASRT<0b00000000000000010, (outs), (ins GPR:$rj, GPR:$rk),
- "asrtle.d", "$rj, $rk">;
- def ASRTGT_D : FmtASRT<0b00000000000000011, (outs), (ins GPR:$rj, GPR:$rk),
- "asrtgt.d", "$rj, $rk">;
- def RDTIME_D : RDTIME_2R<0b0000000000000000011010, "rdtime.d">;
- } // Predicates = [IsLA64]
- //===----------------------------------------------------------------------===//
- // Pseudo-instructions and codegen patterns
- //
- // Naming convention: For 'generic' pattern classes, we use the naming
- // convention PatTy1Ty2.
- //===----------------------------------------------------------------------===//
- /// Generic pattern classes
- class PatGprGpr<SDPatternOperator OpNode, LAInst Inst>
- : Pat<(OpNode GPR:$rj, GPR:$rk), (Inst GPR:$rj, GPR:$rk)>;
- class PatGprGpr_32<SDPatternOperator OpNode, LAInst Inst>
- : Pat<(sext_inreg (OpNode GPR:$rj, GPR:$rk), i32), (Inst GPR:$rj, GPR:$rk)>;
- class PatGpr<SDPatternOperator OpNode, LAInst Inst>
- : Pat<(OpNode GPR:$rj), (Inst GPR:$rj)>;
- class PatGprImm<SDPatternOperator OpNode, LAInst Inst, Operand ImmOpnd>
- : Pat<(OpNode GPR:$rj, ImmOpnd:$imm),
- (Inst GPR:$rj, ImmOpnd:$imm)>;
- class PatGprImm_32<SDPatternOperator OpNode, LAInst Inst, Operand ImmOpnd>
- : Pat<(sext_inreg (OpNode GPR:$rj, ImmOpnd:$imm), i32),
- (Inst GPR:$rj, ImmOpnd:$imm)>;
- /// Predicates
- def AddLike: PatFrags<(ops node:$A, node:$B),
- [(add node:$A, node:$B), (or node:$A, node:$B)], [{
- return N->getOpcode() == ISD::ADD || isOrEquivalentToAdd(N);
- }]>;
- /// Simple arithmetic operations
- // Match both a plain shift and one where the shift amount is masked (this is
- // typically introduced when the legalizer promotes the shift amount and
- // zero-extends it). For LoongArch, the mask is unnecessary as shifts in the
- // base ISA only read the least significant 5 bits (LA32) or 6 bits (LA64).
- def shiftMaskGRLen
- : ComplexPattern<GRLenVT, 1, "selectShiftMaskGRLen", [], [], 0>;
- def shiftMask32 : ComplexPattern<i64, 1, "selectShiftMask32", [], [], 0>;
- def sexti32 : ComplexPattern<i64, 1, "selectSExti32">;
- def zexti32 : ComplexPattern<i64, 1, "selectZExti32">;
- class shiftop<SDPatternOperator operator>
- : PatFrag<(ops node:$val, node:$count),
- (operator node:$val, (GRLenVT (shiftMaskGRLen node:$count)))>;
- class shiftopw<SDPatternOperator operator>
- : PatFrag<(ops node:$val, node:$count),
- (operator node:$val, (i64 (shiftMask32 node:$count)))>;
- let Predicates = [IsLA32] in {
- def : PatGprGpr<add, ADD_W>;
- def : PatGprImm<add, ADDI_W, simm12>;
- def : PatGprGpr<sub, SUB_W>;
- def : PatGprGpr<sdiv, DIV_W>;
- def : PatGprGpr<udiv, DIV_WU>;
- def : PatGprGpr<srem, MOD_W>;
- def : PatGprGpr<urem, MOD_WU>;
- def : PatGprGpr<mul, MUL_W>;
- def : PatGprGpr<mulhs, MULH_W>;
- def : PatGprGpr<mulhu, MULH_WU>;
- def : PatGprGpr<rotr, ROTR_W>;
- def : PatGprImm<rotr, ROTRI_W, uimm5>;
- } // Predicates = [IsLA32]
- let Predicates = [IsLA64] in {
- def : PatGprGpr<add, ADD_D>;
- def : PatGprGpr_32<add, ADD_W>;
- def : PatGprImm<add, ADDI_D, simm12>;
- def : PatGprImm_32<add, ADDI_W, simm12>;
- def : PatGprGpr<sub, SUB_D>;
- def : PatGprGpr_32<sub, SUB_W>;
- def : PatGprGpr<sdiv, DIV_D>;
- def : PatGprGpr<udiv, DIV_DU>;
- def : PatGprGpr<srem, MOD_D>;
- def : PatGprGpr<urem, MOD_DU>;
- def : PatGprGpr<rotr, ROTR_D>;
- def : PatGprGpr<loongarch_rotr_w, ROTR_W>;
- def : PatGprImm<rotr, ROTRI_D, uimm6>;
- def : PatGprImm_32<rotr, ROTRI_W, uimm5>;
- def : Pat<(loongarch_rotl_w GPR:$rj, uimm5:$imm),
- (ROTRI_W GPR:$rj, (ImmSubFrom32 uimm5:$imm))>;
- def : Pat<(sext_inreg (loongarch_rotl_w GPR:$rj, uimm5:$imm), i32),
- (ROTRI_W GPR:$rj, (ImmSubFrom32 uimm5:$imm))>;
- // TODO: Select "_W[U]" instructions for i32xi32 if only lower 32 bits of the
- // product are used.
- def : PatGprGpr<mul, MUL_D>;
- def : PatGprGpr<mulhs, MULH_D>;
- def : PatGprGpr<mulhu, MULH_DU>;
- // Select MULW_D_W for calculating the full 64 bits product of i32xi32 signed
- // multiplication.
- def : Pat<(i64 (mul (sext_inreg GPR:$rj, i32), (sext_inreg GPR:$rk, i32))),
- (MULW_D_W GPR:$rj, GPR:$rk)>;
- // Select MULW_D_WU for calculating the full 64 bits product of i32xi32
- // unsigned multiplication.
- def : Pat<(i64 (mul (loongarch_bstrpick GPR:$rj, (i64 31), (i64 0)),
- (loongarch_bstrpick GPR:$rk, (i64 31), (i64 0)))),
- (MULW_D_WU GPR:$rj, GPR:$rk)>;
- } // Predicates = [IsLA64]
- def : PatGprGpr<and, AND>;
- def : PatGprImm<and, ANDI, uimm12>;
- def : PatGprGpr<or, OR>;
- def : PatGprImm<or, ORI, uimm12>;
- def : PatGprGpr<xor, XOR>;
- def : PatGprImm<xor, XORI, uimm12>;
- def : Pat<(not GPR:$rj), (NOR GPR:$rj, R0)>;
- def : Pat<(not (or GPR:$rj, GPR:$rk)), (NOR GPR:$rj, GPR:$rk)>;
- def : Pat<(or GPR:$rj, (not GPR:$rk)), (ORN GPR:$rj, GPR:$rk)>;
- def : Pat<(and GPR:$rj, (not GPR:$rk)), (ANDN GPR:$rj, GPR:$rk)>;
- /// Traps
- // We lower `trap` to `amswap.w rd:$r0, rk:$r1, rj:$r0`, as this is guaranteed
- // to trap with an INE (non-existent on LA32, explicitly documented to INE on
- // LA64). And the resulting signal is different from `debugtrap` like on some
- // other existing ports so programs/porters might have an easier time.
- def PseudoUNIMP : Pseudo<(outs), (ins), [(trap)]>,
- PseudoInstExpansion<(AMSWAP_W R0, R1, R0)>;
- // We lower `debugtrap` to `break 0`, as this is guaranteed to exist and work,
- // even for LA32 Primary. Also, because so far the ISA does not provide a
- // specific trap instruction/kind exclusively for alerting the debugger,
- // every other project uses the generic immediate of 0 for this.
- def : Pat<(debugtrap), (BREAK 0)>;
- /// Bit counting operations
- let Predicates = [IsLA64] in {
- def : PatGpr<ctlz, CLZ_D>;
- def : PatGpr<cttz, CTZ_D>;
- def : Pat<(ctlz (not GPR:$rj)), (CLO_D GPR:$rj)>;
- def : Pat<(cttz (not GPR:$rj)), (CTO_D GPR:$rj)>;
- def : PatGpr<loongarch_clzw, CLZ_W>;
- def : PatGpr<loongarch_ctzw, CTZ_W>;
- def : Pat<(loongarch_clzw (not GPR:$rj)), (CLO_W GPR:$rj)>;
- def : Pat<(loongarch_ctzw (not GPR:$rj)), (CTO_W GPR:$rj)>;
- } // Predicates = [IsLA64]
- let Predicates = [IsLA32] in {
- def : PatGpr<ctlz, CLZ_W>;
- def : PatGpr<cttz, CTZ_W>;
- def : Pat<(ctlz (not GPR:$rj)), (CLO_W GPR:$rj)>;
- def : Pat<(cttz (not GPR:$rj)), (CTO_W GPR:$rj)>;
- } // Predicates = [IsLA32]
- /// FrameIndex calculations
- let Predicates = [IsLA32] in {
- def : Pat<(AddLike (i32 BaseAddr:$rj), simm12:$imm12),
- (ADDI_W (i32 BaseAddr:$rj), simm12:$imm12)>;
- } // Predicates = [IsLA32]
- let Predicates = [IsLA64] in {
- def : Pat<(AddLike (i64 BaseAddr:$rj), simm12:$imm12),
- (ADDI_D (i64 BaseAddr:$rj), simm12:$imm12)>;
- } // Predicates = [IsLA64]
- /// Shifted addition
- let Predicates = [IsLA32] in {
- def : Pat<(add GPR:$rk, (shl GPR:$rj, uimm2_plus1:$imm2)),
- (ALSL_W GPR:$rj, GPR:$rk, uimm2_plus1:$imm2)>;
- } // Predicates = [IsLA32]
- let Predicates = [IsLA64] in {
- def : Pat<(add GPR:$rk, (shl GPR:$rj, uimm2_plus1:$imm2)),
- (ALSL_D GPR:$rj, GPR:$rk, uimm2_plus1:$imm2)>;
- def : Pat<(loongarch_bstrpick (add GPR:$rk, (shl GPR:$rj, uimm2_plus1:$imm2)),
- (i64 31), (i64 0)),
- (ALSL_WU GPR:$rj, GPR:$rk, uimm2_plus1:$imm2)>;
- } // Predicates = [IsLA64]
- /// Shift
- let Predicates = [IsLA32] in {
- def : PatGprGpr<shiftop<shl>, SLL_W>;
- def : PatGprGpr<shiftop<sra>, SRA_W>;
- def : PatGprGpr<shiftop<srl>, SRL_W>;
- def : PatGprImm<shl, SLLI_W, uimm5>;
- def : PatGprImm<sra, SRAI_W, uimm5>;
- def : PatGprImm<srl, SRLI_W, uimm5>;
- } // Predicates = [IsLA32]
- let Predicates = [IsLA64] in {
- def : PatGprGpr<shiftopw<loongarch_sll_w>, SLL_W>;
- def : PatGprGpr<shiftopw<loongarch_sra_w>, SRA_W>;
- def : PatGprGpr<shiftopw<loongarch_srl_w>, SRL_W>;
- def : PatGprGpr<shiftop<shl>, SLL_D>;
- def : PatGprGpr<shiftop<sra>, SRA_D>;
- def : PatGprGpr<shiftop<srl>, SRL_D>;
- def : PatGprImm<shl, SLLI_D, uimm6>;
- def : PatGprImm<sra, SRAI_D, uimm6>;
- def : PatGprImm<srl, SRLI_D, uimm6>;
- } // Predicates = [IsLA64]
- /// sext and zext
- def : Pat<(sext_inreg GPR:$rj, i8), (EXT_W_B GPR:$rj)>;
- def : Pat<(sext_inreg GPR:$rj, i16), (EXT_W_H GPR:$rj)>;
- let Predicates = [IsLA64] in {
- def : Pat<(sext_inreg GPR:$rj, i32), (ADDI_W GPR:$rj, 0)>;
- } // Predicates = [IsLA64]
- /// Setcc
- def : PatGprGpr<setlt, SLT>;
- def : PatGprImm<setlt, SLTI, simm12>;
- def : PatGprGpr<setult, SLTU>;
- def : PatGprImm<setult, SLTUI, simm12>;
- // Define pattern expansions for setcc operations that aren't directly
- // handled by a LoongArch instruction.
- def : Pat<(seteq GPR:$rj, 0), (SLTUI GPR:$rj, 1)>;
- def : Pat<(seteq GPR:$rj, GPR:$rk), (SLTUI (XOR GPR:$rj, GPR:$rk), 1)>;
- let Predicates = [IsLA32] in {
- def : Pat<(seteq GPR:$rj, simm12_plus1:$imm12),
- (SLTUI (ADDI_W GPR:$rj, (NegImm simm12_plus1:$imm12)), 1)>;
- } // Predicates = [IsLA32]
- let Predicates = [IsLA64] in {
- def : Pat<(seteq GPR:$rj, simm12_plus1:$imm12),
- (SLTUI (ADDI_D GPR:$rj, (NegImm simm12_plus1:$imm12)), 1)>;
- } // Predicates = [IsLA64]
- def : Pat<(setne GPR:$rj, 0), (SLTU R0, GPR:$rj)>;
- def : Pat<(setne GPR:$rj, GPR:$rk), (SLTU R0, (XOR GPR:$rj, GPR:$rk))>;
- let Predicates = [IsLA32] in {
- def : Pat<(setne GPR:$rj, simm12_plus1:$imm12),
- (SLTU R0, (ADDI_W GPR:$rj, (NegImm simm12_plus1:$imm12)))>;
- } // Predicates = [IsLA32]
- let Predicates = [IsLA64] in {
- def : Pat<(setne GPR:$rj, simm12_plus1:$imm12),
- (SLTU R0, (ADDI_D GPR:$rj, (NegImm simm12_plus1:$imm12)))>;
- } // Predicates = [IsLA64]
- def : Pat<(setugt GPR:$rj, GPR:$rk), (SLTU GPR:$rk, GPR:$rj)>;
- def : Pat<(setuge GPR:$rj, GPR:$rk), (XORI (SLTU GPR:$rj, GPR:$rk), 1)>;
- def : Pat<(setule GPR:$rj, GPR:$rk), (XORI (SLTU GPR:$rk, GPR:$rj), 1)>;
- def : Pat<(setgt GPR:$rj, GPR:$rk), (SLT GPR:$rk, GPR:$rj)>;
- def : Pat<(setge GPR:$rj, GPR:$rk), (XORI (SLT GPR:$rj, GPR:$rk), 1)>;
- def : Pat<(setle GPR:$rj, GPR:$rk), (XORI (SLT GPR:$rk, GPR:$rj), 1)>;
- /// Select
- def : Pat<(select GPR:$cond, GPR:$t, GPR:$f),
- (OR (MASKEQZ GPR:$t, GPR:$cond), (MASKNEZ GPR:$f, GPR:$cond))>;
- /// Branches and jumps
- class BccPat<PatFrag CondOp, LAInst Inst>
- : Pat<(brcond (GRLenVT (CondOp GPR:$rj, GPR:$rd)), bb:$imm16),
- (Inst GPR:$rj, GPR:$rd, bb:$imm16)>;
- def : BccPat<seteq, BEQ>;
- def : BccPat<setne, BNE>;
- def : BccPat<setlt, BLT>;
- def : BccPat<setge, BGE>;
- def : BccPat<setult, BLTU>;
- def : BccPat<setuge, BGEU>;
- class BccSwapPat<PatFrag CondOp, LAInst InstBcc>
- : Pat<(brcond (GRLenVT (CondOp GPR:$rd, GPR:$rj)), bb:$imm16),
- (InstBcc GPR:$rj, GPR:$rd, bb:$imm16)>;
- // Condition codes that don't have matching LoongArch branch instructions, but
- // are trivially supported by swapping the two input operands.
- def : BccSwapPat<setgt, BLT>;
- def : BccSwapPat<setle, BGE>;
- def : BccSwapPat<setugt, BLTU>;
- def : BccSwapPat<setule, BGEU>;
- // An extra pattern is needed for a brcond without a setcc (i.e. where the
- // condition was calculated elsewhere).
- def : Pat<(brcond GPR:$rj, bb:$imm21), (BNEZ GPR:$rj, bb:$imm21)>;
- def : Pat<(brcond (GRLenVT (seteq GPR:$rj, 0)), bb:$imm21),
- (BEQZ GPR:$rj, bb:$imm21)>;
- def : Pat<(brcond (GRLenVT (setne GPR:$rj, 0)), bb:$imm21),
- (BNEZ GPR:$rj, bb:$imm21)>;
- let isBarrier = 1, isBranch = 1, isTerminator = 1 in
- def PseudoBR : Pseudo<(outs), (ins simm26_b:$imm26), [(br bb:$imm26)]>,
- PseudoInstExpansion<(B simm26_b:$imm26)>;
- let isBarrier = 1, isBranch = 1, isIndirectBranch = 1, isTerminator = 1 in
- def PseudoBRIND : Pseudo<(outs), (ins GPR:$rj, simm16_lsl2:$imm16)>,
- PseudoInstExpansion<(JIRL R0, GPR:$rj, simm16_lsl2:$imm16)>;
- def : Pat<(brind GPR:$rj), (PseudoBRIND GPR:$rj, 0)>;
- def : Pat<(brind (add GPR:$rj, simm16_lsl2:$imm16)),
- (PseudoBRIND GPR:$rj, simm16_lsl2:$imm16)>;
- let isCall = 1, Defs = [R1] in
- def PseudoCALL : Pseudo<(outs), (ins simm26_symbol:$func)>;
- def : Pat<(loongarch_call tglobaladdr:$func), (PseudoCALL tglobaladdr:$func)>;
- def : Pat<(loongarch_call texternalsym:$func), (PseudoCALL texternalsym:$func)>;
- let isCall = 1, Defs = [R1] in
- def PseudoCALLIndirect : Pseudo<(outs), (ins GPR:$rj),
- [(loongarch_call GPR:$rj)]>,
- PseudoInstExpansion<(JIRL R1, GPR:$rj, 0)>;
- let isCall = 1, Defs = [R1] in
- def PseudoJIRL_CALL : Pseudo<(outs), (ins GPR:$rj, simm16_lsl2:$imm16)>,
- PseudoInstExpansion<(JIRL R1, GPR:$rj,
- simm16_lsl2:$imm16)>;
- let isBarrier = 1, isReturn = 1, isTerminator = 1 in
- def PseudoRET : Pseudo<(outs), (ins), [(loongarch_ret)]>,
- PseudoInstExpansion<(JIRL R0, R1, 0)>;
- let isCall = 1, isTerminator = 1, isReturn = 1, isBarrier = 1, Uses = [R3] in
- def PseudoTAIL : Pseudo<(outs), (ins simm26_symbol:$dst)>;
- def : Pat<(loongarch_tail (iPTR tglobaladdr:$dst)),
- (PseudoTAIL tglobaladdr:$dst)>;
- def : Pat<(loongarch_tail (iPTR texternalsym:$dst)),
- (PseudoTAIL texternalsym:$dst)>;
- let isCall = 1, isTerminator = 1, isReturn = 1, isBarrier = 1, Uses = [R3] in
- def PseudoTAILIndirect : Pseudo<(outs), (ins GPRT:$rj),
- [(loongarch_tail GPRT:$rj)]>,
- PseudoInstExpansion<(JIRL R0, GPR:$rj, 0)>;
- let isCall = 1, isTerminator = 1, isReturn = 1, isBarrier = 1, Uses = [R3] in
- def PseudoB_TAIL : Pseudo<(outs), (ins simm26_b:$imm26)>,
- PseudoInstExpansion<(B simm26_b:$imm26)>;
- let isCall = 1, isTerminator = 1, isReturn = 1, isBarrier = 1, Uses = [R3] in
- def PseudoJIRL_TAIL : Pseudo<(outs), (ins GPR:$rj, simm16_lsl2:$imm16)>,
- PseudoInstExpansion<(JIRL R0, GPR:$rj,
- simm16_lsl2:$imm16)>;
- /// Load address (la*) macro instructions.
- // Define isCodeGenOnly = 0 to expose them to tablegened assembly parser.
- let hasSideEffects = 0, mayLoad = 0, mayStore = 0, isCodeGenOnly = 0,
- isAsmParserOnly = 1 in {
- def PseudoLA_ABS : Pseudo<(outs GPR:$dst), (ins bare_symbol:$src), [],
- "la.abs", "$dst, $src">;
- def PseudoLA_ABS_LARGE : Pseudo<(outs GPR:$dst),
- (ins GPR:$tmp, bare_symbol:$src), [],
- "la.abs", "$dst, $src">;
- def PseudoLA_PCREL : Pseudo<(outs GPR:$dst), (ins bare_symbol:$src), [],
- "la.pcrel", "$dst, $src">;
- def PseudoLA_PCREL_LARGE : Pseudo<(outs GPR:$dst),
- (ins GPR:$tmp, bare_symbol:$src), [],
- "la.pcrel", "$dst, $tmp, $src">,
- Requires<[IsLA64]>;
- def PseudoLA_TLS_LE : Pseudo<(outs GPR:$dst), (ins bare_symbol:$src), [],
- "la.tls.le", "$dst, $src">;
- }
- let hasSideEffects = 0, mayLoad = 1, mayStore = 0, isCodeGenOnly = 0,
- isAsmParserOnly = 1 in {
- def PseudoLA_GOT : Pseudo<(outs GPR:$dst), (ins bare_symbol:$src), [],
- "la.got", "$dst, $src">;
- def PseudoLA_GOT_LARGE : Pseudo<(outs GPR:$dst),
- (ins GPR:$tmp, bare_symbol:$src), [],
- "la.got", "$dst, $tmp, $src">,
- Requires<[IsLA64]>;
- def PseudoLA_TLS_IE : Pseudo<(outs GPR:$dst), (ins bare_symbol:$src), [],
- "la.tls.ie", "$dst, $src">;
- def PseudoLA_TLS_IE_LARGE : Pseudo<(outs GPR:$dst),
- (ins GPR:$tmp, bare_symbol:$src), [],
- "la.tls.ie", "$dst, $tmp, $src">,
- Requires<[IsLA64]>;
- def PseudoLA_TLS_LD : Pseudo<(outs GPR:$dst), (ins bare_symbol:$src), [],
- "la.tls.ld", "$dst, $src">;
- def PseudoLA_TLS_LD_LARGE : Pseudo<(outs GPR:$dst),
- (ins GPR:$tmp, bare_symbol:$src), [],
- "la.tls.ld", "$dst, $tmp, $src">,
- Requires<[IsLA64]>;
- def PseudoLA_TLS_GD : Pseudo<(outs GPR:$dst), (ins bare_symbol:$src), [],
- "la.tls.gd", "$dst, $src">;
- def PseudoLA_TLS_GD_LARGE : Pseudo<(outs GPR:$dst),
- (ins GPR:$tmp, bare_symbol:$src), [],
- "la.tls.gd", "$dst, $tmp, $src">,
- Requires<[IsLA64]>;
- }
- // Load address inst alias: "la", "la.global" and "la.local".
- // Default:
- // la = la.global = la.got
- // la.local = la.pcrel
- // With feature "+la-global-with-pcrel":
- // la = la.global = la.pcrel
- // With feature "+la-global-with-abs":
- // la = la.global = la.abs
- // With feature "+la-local-with-abs":
- // la.local = la.abs
- // With features "+la-global-with-pcrel,+la-global-with-abs"(disorder):
- // la = la.global = la.pcrel
- // Note: To keep consistent with gnu-as behavior, the "la" can only have one
- // register operand.
- def : InstAlias<"la $dst, $src", (PseudoLA_GOT GPR:$dst, bare_symbol:$src)>;
- def : InstAlias<"la.global $dst, $src",
- (PseudoLA_GOT GPR:$dst, bare_symbol:$src)>;
- def : InstAlias<"la.global $dst, $tmp, $src",
- (PseudoLA_GOT_LARGE GPR:$dst, GPR:$tmp, bare_symbol:$src)>;
- def : InstAlias<"la.local $dst, $src",
- (PseudoLA_PCREL GPR:$dst, bare_symbol:$src)>;
- def : InstAlias<"la.local $dst, $tmp, $src",
- (PseudoLA_PCREL_LARGE GPR:$dst, GPR:$tmp, bare_symbol:$src)>;
- // Note: Keep HasLaGlobalWithPcrel before HasLaGlobalWithAbs to ensure
- // "la-global-with-pcrel" takes effect when bose "la-global-with-pcrel" and
- // "la-global-with-abs" are enabled.
- let Predicates = [HasLaGlobalWithPcrel] in {
- def : InstAlias<"la $dst, $src", (PseudoLA_PCREL GPR:$dst, bare_symbol:$src)>;
- def : InstAlias<"la.global $dst, $src",
- (PseudoLA_PCREL GPR:$dst, bare_symbol:$src)>;
- def : InstAlias<"la.global $dst, $tmp, $src",
- (PseudoLA_PCREL_LARGE GPR:$dst, GPR:$tmp, bare_symbol:$src)>;
- } // Predicates = [HasLaGlobalWithPcrel]
- let Predicates = [HasLaGlobalWithAbs] in {
- def : InstAlias<"la $dst, $src", (PseudoLA_ABS GPR:$dst, bare_symbol:$src)>;
- def : InstAlias<"la.global $dst, $src",
- (PseudoLA_ABS GPR:$dst, bare_symbol:$src)>;
- def : InstAlias<"la.global $dst, $tmp, $src",
- (PseudoLA_ABS_LARGE GPR:$dst, GPR:$tmp, bare_symbol:$src)>;
- } // Predicates = [HasLaGlobalWithAbs]
- let Predicates = [HasLaLocalWithAbs] in {
- def : InstAlias<"la.local $dst, $src",
- (PseudoLA_ABS GPR:$dst, bare_symbol:$src)>;
- def : InstAlias<"la.local $dst, $tmp, $src",
- (PseudoLA_ABS_LARGE GPR:$dst, GPR:$tmp, bare_symbol:$src)>;
- } // Predicates = [HasLaLocalWithAbs]
- /// BSTRINS and BSTRPICK
- let Predicates = [IsLA32] in {
- def : Pat<(loongarch_bstrins GPR:$rd, GPR:$rj, uimm5:$msbd, uimm5:$lsbd),
- (BSTRINS_W GPR:$rd, GPR:$rj, uimm5:$msbd, uimm5:$lsbd)>;
- def : Pat<(loongarch_bstrpick GPR:$rj, uimm5:$msbd, uimm5:$lsbd),
- (BSTRPICK_W GPR:$rj, uimm5:$msbd, uimm5:$lsbd)>;
- } // Predicates = [IsLA32]
- let Predicates = [IsLA64] in {
- def : Pat<(loongarch_bstrins GPR:$rd, GPR:$rj, uimm6:$msbd, uimm6:$lsbd),
- (BSTRINS_D GPR:$rd, GPR:$rj, uimm6:$msbd, uimm6:$lsbd)>;
- def : Pat<(loongarch_bstrpick GPR:$rj, uimm6:$msbd, uimm6:$lsbd),
- (BSTRPICK_D GPR:$rj, uimm6:$msbd, uimm6:$lsbd)>;
- } // Predicates = [IsLA64]
- /// Byte-swapping and bit-reversal
- def : Pat<(loongarch_revb_2h GPR:$rj), (REVB_2H GPR:$rj)>;
- def : Pat<(loongarch_bitrev_4b GPR:$rj), (BITREV_4B GPR:$rj)>;
- let Predicates = [IsLA32] in {
- def : Pat<(bswap GPR:$rj), (ROTRI_W (REVB_2H GPR:$rj), 16)>;
- def : Pat<(bitreverse GPR:$rj), (BITREV_W GPR:$rj)>;
- def : Pat<(bswap (bitreverse GPR:$rj)), (BITREV_4B GPR:$rj)>;
- def : Pat<(bitreverse (bswap GPR:$rj)), (BITREV_4B GPR:$rj)>;
- } // Predicates = [IsLA32]
- let Predicates = [IsLA64] in {
- def : Pat<(loongarch_revb_2w GPR:$rj), (REVB_2W GPR:$rj)>;
- def : Pat<(bswap GPR:$rj), (REVB_D GPR:$rj)>;
- def : Pat<(loongarch_bitrev_w GPR:$rj), (BITREV_W GPR:$rj)>;
- def : Pat<(bitreverse GPR:$rj), (BITREV_D GPR:$rj)>;
- def : Pat<(bswap (bitreverse GPR:$rj)), (BITREV_8B GPR:$rj)>;
- def : Pat<(bitreverse (bswap GPR:$rj)), (BITREV_8B GPR:$rj)>;
- } // Predicates = [IsLA64]
- /// Loads
- multiclass LdPat<PatFrag LoadOp, LAInst Inst, ValueType vt = GRLenVT> {
- def : Pat<(vt (LoadOp BaseAddr:$rj)), (Inst BaseAddr:$rj, 0)>;
- def : Pat<(vt (LoadOp (AddLike BaseAddr:$rj, simm12:$imm12))),
- (Inst BaseAddr:$rj, simm12:$imm12)>;
- }
- defm : LdPat<sextloadi8, LD_B>;
- defm : LdPat<extloadi8, LD_B>;
- defm : LdPat<sextloadi16, LD_H>;
- defm : LdPat<extloadi16, LD_H>;
- defm : LdPat<load, LD_W>, Requires<[IsLA32]>;
- defm : LdPat<zextloadi8, LD_BU>;
- defm : LdPat<zextloadi16, LD_HU>;
- let Predicates = [IsLA64] in {
- defm : LdPat<sextloadi32, LD_W, i64>;
- defm : LdPat<extloadi32, LD_W, i64>;
- defm : LdPat<zextloadi32, LD_WU, i64>;
- defm : LdPat<load, LD_D, i64>;
- } // Predicates = [IsLA64]
- // LA64 register-register-addressed loads
- let Predicates = [IsLA64] in {
- class RegRegLdPat<PatFrag LoadOp, LAInst Inst, ValueType vt>
- : Pat<(vt (LoadOp (add NonFIBaseAddr:$rj, GPR:$rk))),
- (Inst NonFIBaseAddr:$rj, GPR:$rk)>;
- def : RegRegLdPat<extloadi8, LDX_B, i64>;
- def : RegRegLdPat<sextloadi8, LDX_B, i64>;
- def : RegRegLdPat<zextloadi8, LDX_BU, i64>;
- def : RegRegLdPat<extloadi16, LDX_H, i64>;
- def : RegRegLdPat<sextloadi16, LDX_H, i64>;
- def : RegRegLdPat<zextloadi16, LDX_HU, i64>;
- def : RegRegLdPat<extloadi32, LDX_W, i64>;
- def : RegRegLdPat<sextloadi32, LDX_W, i64>;
- def : RegRegLdPat<zextloadi32, LDX_WU, i64>;
- def : RegRegLdPat<load, LDX_D, i64>;
- } // Predicates = [IsLA64]
- /// Stores
- multiclass StPat<PatFrag StoreOp, LAInst Inst, RegisterClass StTy,
- ValueType vt> {
- def : Pat<(StoreOp (vt StTy:$rd), BaseAddr:$rj),
- (Inst StTy:$rd, BaseAddr:$rj, 0)>;
- def : Pat<(StoreOp (vt StTy:$rd), (AddLike BaseAddr:$rj, simm12:$imm12)),
- (Inst StTy:$rd, BaseAddr:$rj, simm12:$imm12)>;
- }
- defm : StPat<truncstorei8, ST_B, GPR, GRLenVT>;
- defm : StPat<truncstorei16, ST_H, GPR, GRLenVT>;
- defm : StPat<store, ST_W, GPR, i32>, Requires<[IsLA32]>;
- let Predicates = [IsLA64] in {
- defm : StPat<truncstorei32, ST_W, GPR, i64>;
- defm : StPat<store, ST_D, GPR, i64>;
- } // Predicates = [IsLA64]
- let Predicates = [IsLA64] in {
- def : Pat<(i64 (sextloadi32 (AddLike BaseAddr:$rj, simm14_lsl2:$imm14))),
- (LDPTR_W BaseAddr:$rj, simm14_lsl2:$imm14)>;
- def : Pat<(i64 (load (AddLike BaseAddr:$rj, simm14_lsl2:$imm14))),
- (LDPTR_D BaseAddr:$rj, simm14_lsl2:$imm14)>;
- def : Pat<(truncstorei32 (i64 GPR:$rd),
- (AddLike BaseAddr:$rj, simm14_lsl2:$imm14)),
- (STPTR_W GPR:$rd, BaseAddr:$rj, simm14_lsl2:$imm14)>;
- def : Pat<(store (i64 GPR:$rd), (AddLike BaseAddr:$rj, simm14_lsl2:$imm14)),
- (STPTR_D GPR:$rd, BaseAddr:$rj, simm14_lsl2:$imm14)>;
- } // Predicates = [IsLA64]
- // LA64 register-register-addressed stores
- let Predicates = [IsLA64] in {
- class RegRegStPat<PatFrag StoreOp, LAInst Inst, RegisterClass StTy,
- ValueType vt>
- : Pat<(StoreOp (vt StTy:$rd), (add NonFIBaseAddr:$rj, GPR:$rk)),
- (Inst StTy:$rd, NonFIBaseAddr:$rj, GPR:$rk)>;
- def : RegRegStPat<truncstorei8, STX_B, GPR, i64>;
- def : RegRegStPat<truncstorei16, STX_H, GPR, i64>;
- def : RegRegStPat<truncstorei32, STX_W, GPR, i64>;
- def : RegRegStPat<store, STX_D, GPR, i64>;
- } // Predicates = [IsLA64]
- /// Atomic loads and stores
- def : Pat<(atomic_fence timm, timm), (DBAR 0)>;
- defm : LdPat<atomic_load_8, LD_B>;
- defm : LdPat<atomic_load_16, LD_H>;
- defm : LdPat<atomic_load_32, LD_W>;
- class release_seqcst_store<PatFrag base>
- : PatFrag<(ops node:$ptr, node:$val), (base node:$ptr, node:$val), [{
- AtomicOrdering Ordering = cast<AtomicSDNode>(N)->getSuccessOrdering();
- return isReleaseOrStronger(Ordering);
- }]>;
- class unordered_monotonic_store<PatFrag base>
- : PatFrag<(ops node:$ptr, node:$val), (base node:$ptr, node:$val), [{
- AtomicOrdering Ordering = cast<AtomicSDNode>(N)->getSuccessOrdering();
- return !isReleaseOrStronger(Ordering);
- }]>;
- def atomic_store_release_seqcst_32 : release_seqcst_store<atomic_store_32>;
- def atomic_store_release_seqcst_64 : release_seqcst_store<atomic_store_64>;
- def atomic_store_unordered_monotonic_32
- : unordered_monotonic_store<atomic_store_32>;
- def atomic_store_unordered_monotonic_64
- : unordered_monotonic_store<atomic_store_64>;
- /// AtomicStores
- multiclass AtomicStPat<PatFrag StoreOp, LAInst Inst, RegisterClass StTy,
- ValueType vt> {
- def : Pat<(StoreOp BaseAddr:$ptr, (vt StTy:$val)),
- (Inst StTy:$val, BaseAddr:$ptr, 0)>;
- def : Pat<(StoreOp (AddLike BaseAddr:$ptr, simm12:$imm12), (vt StTy:$val)),
- (Inst StTy:$val, BaseAddr:$ptr, simm12:$imm12)>;
- }
- defm : AtomicStPat<atomic_store_8, ST_B, GPR, GRLenVT>;
- defm : AtomicStPat<atomic_store_16, ST_H, GPR, GRLenVT>;
- defm : AtomicStPat<atomic_store_unordered_monotonic_32, ST_W, GPR, i32>,
- Requires<[IsLA32]>;
- def PseudoAtomicStoreW
- : Pseudo<(outs GPR:$dst), (ins GPR:$rj, GPR:$rk)>,
- PseudoInstExpansion<(AMSWAP_DB_W R0, GPR:$rk, GPRMemAtomic:$rj)>;
- def : Pat<(atomic_store_release_seqcst_32 GPR:$rj, GPR:$rk),
- (PseudoAtomicStoreW GPR:$rj, GPR:$rk)>;
- let Predicates = [IsLA64] in {
- def PseudoAtomicStoreD
- : Pseudo<(outs GPR:$dst), (ins GPR:$rj, GPR:$rk)>,
- PseudoInstExpansion<(AMSWAP_DB_D R0, GPR:$rk, GPRMemAtomic:$rj)>;
- def : Pat<(atomic_store_release_seqcst_64 GPR:$rj, GPR:$rk),
- (PseudoAtomicStoreD GPR:$rj, GPR:$rk)>;
- defm : LdPat<atomic_load_64, LD_D>;
- defm : AtomicStPat<atomic_store_unordered_monotonic_32, ST_W, GPR, i64>;
- defm : AtomicStPat<atomic_store_unordered_monotonic_64, ST_D, GPR, i64>;
- } // Predicates = [IsLA64]
- /// Atomic Ops
- class PseudoMaskedAM
- : Pseudo<(outs GPR:$res, GPR:$scratch),
- (ins GPR:$addr, GPR:$incr, GPR:$mask, grlenimm:$ordering)> {
- let Constraints = "@earlyclobber $res,@earlyclobber $scratch";
- let mayLoad = 1;
- let mayStore = 1;
- let hasSideEffects = 0;
- let Size = 36;
- }
- def PseudoMaskedAtomicSwap32 : PseudoMaskedAM;
- def PseudoMaskedAtomicLoadAdd32 : PseudoMaskedAM;
- def PseudoMaskedAtomicLoadSub32 : PseudoMaskedAM;
- def PseudoMaskedAtomicLoadNand32 : PseudoMaskedAM;
- class PseudoAM : Pseudo<(outs GPR:$res, GPR:$scratch),
- (ins GPR:$addr, GPR:$incr, grlenimm:$ordering)> {
- let Constraints = "@earlyclobber $res,@earlyclobber $scratch";
- let mayLoad = 1;
- let mayStore = 1;
- let hasSideEffects = 0;
- let Size = 24;
- }
- def PseudoAtomicSwap32 : PseudoAM;
- def PseudoAtomicLoadNand32 : PseudoAM;
- def PseudoAtomicLoadNand64 : PseudoAM;
- def PseudoAtomicLoadAdd32 : PseudoAM;
- def PseudoAtomicLoadSub32 : PseudoAM;
- def PseudoAtomicLoadAnd32 : PseudoAM;
- def PseudoAtomicLoadOr32 : PseudoAM;
- def PseudoAtomicLoadXor32 : PseudoAM;
- multiclass PseudoBinPat<string Op, Pseudo BinInst> {
- def : Pat<(!cast<PatFrag>(Op#"_monotonic") GPR:$addr, GPR:$incr),
- (BinInst GPR:$addr, GPR:$incr, 2)>;
- def : Pat<(!cast<PatFrag>(Op#"_acquire") GPR:$addr, GPR:$incr),
- (BinInst GPR:$addr, GPR:$incr, 4)>;
- def : Pat<(!cast<PatFrag>(Op#"_release") GPR:$addr, GPR:$incr),
- (BinInst GPR:$addr, GPR:$incr, 5)>;
- def : Pat<(!cast<PatFrag>(Op#"_acq_rel") GPR:$addr, GPR:$incr),
- (BinInst GPR:$addr, GPR:$incr, 6)>;
- def : Pat<(!cast<PatFrag>(Op#"_seq_cst") GPR:$addr, GPR:$incr),
- (BinInst GPR:$addr, GPR:$incr, 7)>;
- }
- class PseudoMaskedAMUMinUMax
- : Pseudo<(outs GPR:$res, GPR:$scratch1, GPR:$scratch2),
- (ins GPR:$addr, GPR:$incr, GPR:$mask, grlenimm:$ordering)> {
- let Constraints = "@earlyclobber $res,@earlyclobber $scratch1,"
- "@earlyclobber $scratch2";
- let mayLoad = 1;
- let mayStore = 1;
- let hasSideEffects = 0;
- let Size = 48;
- }
- def PseudoMaskedAtomicLoadUMax32 : PseudoMaskedAMUMinUMax;
- def PseudoMaskedAtomicLoadUMin32 : PseudoMaskedAMUMinUMax;
- class PseudoMaskedAMMinMax
- : Pseudo<(outs GPR:$res, GPR:$scratch1, GPR:$scratch2),
- (ins GPR:$addr, GPR:$incr, GPR:$mask, grlenimm:$sextshamt,
- grlenimm:$ordering)> {
- let Constraints = "@earlyclobber $res,@earlyclobber $scratch1,"
- "@earlyclobber $scratch2";
- let mayLoad = 1;
- let mayStore = 1;
- let hasSideEffects = 0;
- let Size = 56;
- }
- def PseudoMaskedAtomicLoadMax32 : PseudoMaskedAMMinMax;
- def PseudoMaskedAtomicLoadMin32 : PseudoMaskedAMMinMax;
- /// Compare and exchange
- class PseudoCmpXchg
- : Pseudo<(outs GPR:$res, GPR:$scratch),
- (ins GPR:$addr, GPR:$cmpval, GPR:$newval)> {
- let Constraints = "@earlyclobber $res,@earlyclobber $scratch";
- let mayLoad = 1;
- let mayStore = 1;
- let hasSideEffects = 0;
- let Size = 36;
- }
- def PseudoCmpXchg32 : PseudoCmpXchg;
- def PseudoCmpXchg64 : PseudoCmpXchg;
- def PseudoMaskedCmpXchg32
- : Pseudo<(outs GPR:$res, GPR:$scratch),
- (ins GPR:$addr, GPR:$cmpval, GPR:$newval, GPR:$mask,
- grlenimm:$ordering)> {
- let Constraints = "@earlyclobber $res,@earlyclobber $scratch";
- let mayLoad = 1;
- let mayStore = 1;
- let hasSideEffects = 0;
- let Size = 44;
- }
- class PseudoMaskedAMMinMaxPat<Intrinsic intrin, Pseudo AMInst>
- : Pat<(intrin GPR:$addr, GPR:$incr, GPR:$mask, GPR:$shiftamt,
- timm:$ordering),
- (AMInst GPR:$addr, GPR:$incr, GPR:$mask, GPR:$shiftamt,
- timm:$ordering)>;
- class AtomicPat<Intrinsic intrin, Pseudo AMInst>
- : Pat<(intrin GPR:$addr, GPR:$incr, GPR:$mask, timm:$ordering),
- (AMInst GPR:$addr, GPR:$incr, GPR:$mask, timm:$ordering)>;
- let Predicates = [IsLA64] in {
- def : AtomicPat<int_loongarch_masked_atomicrmw_xchg_i64,
- PseudoMaskedAtomicSwap32>;
- def : Pat<(atomic_swap_32 GPR:$addr, GPR:$incr),
- (AMSWAP_DB_W GPR:$incr, GPR:$addr)>;
- def : Pat<(atomic_swap_64 GPR:$addr, GPR:$incr),
- (AMSWAP_DB_D GPR:$incr, GPR:$addr)>;
- def : Pat<(atomic_load_add_64 GPR:$rj, GPR:$rk),
- (AMADD_DB_D GPR:$rk, GPR:$rj)>;
- def : AtomicPat<int_loongarch_masked_atomicrmw_add_i64,
- PseudoMaskedAtomicLoadAdd32>;
- def : Pat<(atomic_load_sub_32 GPR:$rj, GPR:$rk),
- (AMADD_DB_W (SUB_W R0, GPR:$rk), GPR:$rj)>;
- def : Pat<(atomic_load_sub_64 GPR:$rj, GPR:$rk),
- (AMADD_DB_D (SUB_D R0, GPR:$rk), GPR:$rj)>;
- def : AtomicPat<int_loongarch_masked_atomicrmw_sub_i64,
- PseudoMaskedAtomicLoadSub32>;
- defm : PseudoBinPat<"atomic_load_nand_64", PseudoAtomicLoadNand64>;
- def : AtomicPat<int_loongarch_masked_atomicrmw_nand_i64,
- PseudoMaskedAtomicLoadNand32>;
- def : Pat<(atomic_load_add_32 GPR:$rj, GPR:$rk),
- (AMADD_DB_W GPR:$rk, GPR:$rj)>;
- def : Pat<(atomic_load_and_32 GPR:$rj, GPR:$rk),
- (AMAND_DB_W GPR:$rk, GPR:$rj)>;
- def : Pat<(atomic_load_and_64 GPR:$rj, GPR:$rk),
- (AMAND_DB_D GPR:$rk, GPR:$rj)>;
- def : Pat<(atomic_load_or_32 GPR:$rj, GPR:$rk),
- (AMOR_DB_W GPR:$rk, GPR:$rj)>;
- def : Pat<(atomic_load_or_64 GPR:$rj, GPR:$rk),
- (AMOR_DB_D GPR:$rk, GPR:$rj)>;
- def : Pat<(atomic_load_xor_32 GPR:$rj, GPR:$rk),
- (AMXOR_DB_W GPR:$rk, GPR:$rj)>;
- def : Pat<(atomic_load_xor_64 GPR:$rj, GPR:$rk),
- (AMXOR_DB_D GPR:$rk, GPR:$rj)>;
- def : Pat<(atomic_load_umin_32 GPR:$rj, GPR:$rk),
- (AMMIN_DB_WU GPR:$rk, GPR:$rj)>;
- def : Pat<(atomic_load_umin_64 GPR:$rj, GPR:$rk),
- (AMMIN_DB_DU GPR:$rk, GPR:$rj)>;
- def : Pat<(atomic_load_umax_32 GPR:$rj, GPR:$rk),
- (AMMAX_DB_WU GPR:$rk, GPR:$rj)>;
- def : Pat<(atomic_load_umax_64 GPR:$rj, GPR:$rk),
- (AMMAX_DB_DU GPR:$rk, GPR:$rj)>;
- def : Pat<(atomic_load_min_32 GPR:$rj, GPR:$rk),
- (AMMIN_DB_W GPR:$rk, GPR:$rj)>;
- def : Pat<(atomic_load_min_64 GPR:$rj, GPR:$rk),
- (AMMIN_DB_D GPR:$rk, GPR:$rj)>;
- def : Pat<(atomic_load_max_32 GPR:$rj, GPR:$rk),
- (AMMAX_DB_W GPR:$rk, GPR:$rj)>;
- def : Pat<(atomic_load_max_64 GPR:$rj, GPR:$rk),
- (AMMAX_DB_D GPR:$rk, GPR:$rj)>;
- def : AtomicPat<int_loongarch_masked_atomicrmw_umax_i64,
- PseudoMaskedAtomicLoadUMax32>;
- def : AtomicPat<int_loongarch_masked_atomicrmw_umin_i64,
- PseudoMaskedAtomicLoadUMin32>;
- def : Pat<(atomic_cmp_swap_64 GPR:$addr, GPR:$cmp, GPR:$new),
- (PseudoCmpXchg64 GPR:$addr, GPR:$cmp, GPR:$new)>;
- def : Pat<(int_loongarch_masked_cmpxchg_i64
- GPR:$addr, GPR:$cmpval, GPR:$newval, GPR:$mask, timm:$ordering),
- (PseudoMaskedCmpXchg32
- GPR:$addr, GPR:$cmpval, GPR:$newval, GPR:$mask, timm:$ordering)>;
- def : Pat<(atomic_cmp_swap_32 GPR:$addr, GPR:$cmp, GPR:$new),
- (PseudoCmpXchg32 GPR:$addr, GPR:$cmp, GPR:$new)>;
- def : PseudoMaskedAMMinMaxPat<int_loongarch_masked_atomicrmw_max_i64,
- PseudoMaskedAtomicLoadMax32>;
- def : PseudoMaskedAMMinMaxPat<int_loongarch_masked_atomicrmw_min_i64,
- PseudoMaskedAtomicLoadMin32>;
- } // Predicates = [IsLA64]
- defm : PseudoBinPat<"atomic_load_nand_32", PseudoAtomicLoadNand32>;
- let Predicates = [IsLA32] in {
- def : AtomicPat<int_loongarch_masked_atomicrmw_xchg_i32,
- PseudoMaskedAtomicSwap32>;
- defm : PseudoBinPat<"atomic_swap_32", PseudoAtomicSwap32>;
- def : AtomicPat<int_loongarch_masked_atomicrmw_add_i32,
- PseudoMaskedAtomicLoadAdd32>;
- def : AtomicPat<int_loongarch_masked_atomicrmw_sub_i32,
- PseudoMaskedAtomicLoadSub32>;
- def : AtomicPat<int_loongarch_masked_atomicrmw_nand_i32,
- PseudoMaskedAtomicLoadNand32>;
- defm : PseudoBinPat<"atomic_load_add_32", PseudoAtomicLoadAdd32>;
- defm : PseudoBinPat<"atomic_load_sub_32", PseudoAtomicLoadSub32>;
- defm : PseudoBinPat<"atomic_load_and_32", PseudoAtomicLoadAnd32>;
- defm : PseudoBinPat<"atomic_load_or_32", PseudoAtomicLoadOr32>;
- defm : PseudoBinPat<"atomic_load_xor_32", PseudoAtomicLoadXor32>;
- } // Predicates = [IsLA32]
- /// Intrinsics
- def : Pat<(int_loongarch_cacop_d timm:$op, i64:$rj, timm:$imm12),
- (CACOP uimm5:$op, GPR:$rj, simm12:$imm12)>;
- def : Pat<(int_loongarch_cacop_w i32:$op, i32:$rj, i32:$imm12),
- (CACOP uimm5:$op, GPR:$rj, simm12:$imm12)>;
- def : Pat<(loongarch_dbar uimm15:$imm15), (DBAR uimm15:$imm15)>;
- def : Pat<(loongarch_ibar uimm15:$imm15), (IBAR uimm15:$imm15)>;
- def : Pat<(loongarch_break uimm15:$imm15), (BREAK uimm15:$imm15)>;
- def : Pat<(loongarch_syscall uimm15:$imm15), (SYSCALL uimm15:$imm15)>;
- let Predicates = [IsLA64] in {
- // CRC Check Instructions
- def : PatGprGpr<loongarch_crc_w_b_w, CRC_W_B_W>;
- def : PatGprGpr<loongarch_crc_w_h_w, CRC_W_H_W>;
- def : PatGprGpr<loongarch_crc_w_w_w, CRC_W_W_W>;
- def : PatGprGpr<loongarch_crc_w_d_w, CRC_W_D_W>;
- def : PatGprGpr<loongarch_crcc_w_b_w, CRCC_W_B_W>;
- def : PatGprGpr<loongarch_crcc_w_h_w, CRCC_W_H_W>;
- def : PatGprGpr<loongarch_crcc_w_w_w, CRCC_W_W_W>;
- def : PatGprGpr<loongarch_crcc_w_d_w, CRCC_W_D_W>;
- } // Predicates = [IsLA64]
- /// Other pseudo-instructions
- // Pessimistically assume the stack pointer will be clobbered
- let Defs = [R3], Uses = [R3] in {
- def ADJCALLSTACKDOWN : Pseudo<(outs), (ins i32imm:$amt1, i32imm:$amt2),
- [(callseq_start timm:$amt1, timm:$amt2)]>;
- def ADJCALLSTACKUP : Pseudo<(outs), (ins i32imm:$amt1, i32imm:$amt2),
- [(callseq_end timm:$amt1, timm:$amt2)]>;
- } // Defs = [R3], Uses = [R3]
- //===----------------------------------------------------------------------===//
- // Assembler Pseudo Instructions
- //===----------------------------------------------------------------------===//
- def : InstAlias<"nop", (ANDI R0, R0, 0)>;
- def : InstAlias<"move $dst, $src", (OR GPR:$dst, GPR:$src, R0)>;
- // `ret` is supported since binutils commit 20f2e2686c79a5ac (version 2.40 and
- // later).
- def : InstAlias<"ret", (JIRL R0, R1, 0)>;
- def : InstAlias<"jr $rj", (JIRL R0, GPR:$rj, 0)>;
- // Branches implemented with alias.
- // Always output the canonical mnemonic for the pseudo branch instructions.
- // The GNU tools emit the canonical mnemonic for the branch pseudo instructions
- // as well (e.g. "bgt" will be recognised by the assembler but never printed by
- // objdump). Match this behaviour by setting a zero weight.
- def : InstAlias<"bgt $rj, $rd, $imm16",
- (BLT GPR:$rd, GPR:$rj, simm16_lsl2_br:$imm16), 0>;
- def : InstAlias<"bgtu $rj, $rd, $imm16",
- (BLTU GPR:$rd, GPR:$rj, simm16_lsl2_br:$imm16), 0>;
- def : InstAlias<"ble $rj, $rd, $imm16",
- (BGE GPR:$rd, GPR:$rj, simm16_lsl2_br:$imm16), 0>;
- def : InstAlias<"bleu $rj, $rd, $imm16",
- (BGEU GPR:$rd, GPR:$rj, simm16_lsl2_br:$imm16), 0>;
- def : InstAlias<"bltz $rd, $imm16",
- (BLT GPR:$rd, R0, simm16_lsl2_br:$imm16), 0>;
- def : InstAlias<"bgtz $rj, $imm16",
- (BLT R0, GPR:$rj, simm16_lsl2_br:$imm16), 0>;
- def : InstAlias<"blez $rj, $imm16",
- (BGE R0, GPR:$rj, simm16_lsl2_br:$imm16), 0>;
- def : InstAlias<"bgez $rd, $imm16",
- (BGE GPR:$rd, R0, simm16_lsl2_br:$imm16), 0>;
- // Load immediate.
- let hasSideEffects = 0, mayLoad = 0, mayStore = 0, isCodeGenOnly = 0,
- isAsmParserOnly = 1 in {
- def PseudoLI_W : Pseudo<(outs GPR:$rd), (ins imm32:$imm), [],
- "li.w", "$rd, $imm">;
- def PseudoLI_D : Pseudo<(outs GPR:$rd), (ins grlenimm:$imm), [],
- "li.d", "$rd, $imm">, Requires<[IsLA64]>;
- }
- //===----------------------------------------------------------------------===//
- // Basic Floating-Point Instructions
- //===----------------------------------------------------------------------===//
- include "LoongArchFloat32InstrInfo.td"
- include "LoongArchFloat64InstrInfo.td"
- let Predicates = [HasBasicF], usesCustomInserter = 1 in {
- def WRFCSR : Pseudo<(outs), (ins uimm2:$fcsr, GPR:$src),
- [(loongarch_movgr2fcsr uimm2:$fcsr, GRLenVT:$src)]>;
- def RDFCSR : Pseudo<(outs GPR:$rd), (ins uimm2:$fcsr),
- [(set GPR:$rd, (loongarch_movfcsr2gr uimm2:$fcsr))]>;
- }
- //===----------------------------------------------------------------------===//
- // Privilege Instructions
- //===----------------------------------------------------------------------===//
- // CSR Access Instructions
- def CSRRD : FmtCSR<0b0000010000000, (outs GPR:$rd), (ins uimm14:$csr_num),
- "csrrd", "$rd, $csr_num">;
- let Constraints = "$rd = $dst" in {
- def CSRWR : FmtCSR<0b0000010000001, (outs GPR:$dst),
- (ins GPR:$rd, uimm14:$csr_num), "csrwr", "$rd, $csr_num">;
- def CSRXCHG : FmtCSRXCHG<0b00000100, (outs GPR:$dst),
- (ins GPR:$rd, GPR:$rj, uimm14:$csr_num),
- "csrxchg", "$rd, $rj, $csr_num">;
- } // Constraints = "$rd = $dst"
- // IOCSR Access Instructions
- def IOCSRRD_B : IOCSRRD<0b0000011001001000000000, "iocsrrd.b">;
- def IOCSRRD_H : IOCSRRD<0b0000011001001000000001, "iocsrrd.h">;
- def IOCSRRD_W : IOCSRRD<0b0000011001001000000010, "iocsrrd.w">;
- def IOCSRWR_B : IOCSRWR<0b0000011001001000000100, "iocsrwr.b">;
- def IOCSRWR_H : IOCSRWR<0b0000011001001000000101, "iocsrwr.h">;
- def IOCSRWR_W : IOCSRWR<0b0000011001001000000110, "iocsrwr.w">;
- let Predicates = [IsLA64] in {
- def IOCSRRD_D : IOCSRRD<0b0000011001001000000011, "iocsrrd.d">;
- def IOCSRWR_D : IOCSRWR<0b0000011001001000000111, "iocsrwr.d">;
- } // Predicates = [IsLA64]
- // TLB Maintenance Instructions
- def TLBSRCH : FmtI32<0b00000110010010000010100000000000, "tlbsrch">;
- def TLBRD : FmtI32<0b00000110010010000010110000000000, "tlbrd">;
- def TLBWR : FmtI32<0b00000110010010000011000000000000, "tlbwr">;
- def TLBFILL : FmtI32<0b00000110010010000011010000000000, "tlbfill">;
- def TLBCLR : FmtI32<0b00000110010010000010000000000000, "tlbclr">;
- def TLBFLUSH : FmtI32<0b00000110010010000010010000000000, "tlbflush">;
- def INVTLB : FmtINVTLB<(outs), (ins GPR:$rk, GPR:$rj, uimm5:$op), "invtlb",
- "$op, $rj, $rk">;
- // Software Page Walking Instructions
- def LDDIR : Fmt2RI8<0b00000110010000, (outs GPR:$rd),
- (ins GPR:$rj, uimm8:$imm8), "lddir", "$rd, $rj, $imm8">;
- def LDPTE : FmtLDPTE<(outs), (ins GPR:$rj, uimm8:$seq), "ldpte", "$rj, $seq">;
- // Other Miscellaneous Instructions
- def ERTN : FmtI32<0b00000110010010000011100000000000, "ertn">;
- def DBCL : MISC_I15<0b00000000001010101, "dbcl">;
- def IDLE : MISC_I15<0b00000110010010001, "idle">;
- //===----------------------------------------------------------------------===//
- // Privilege Intrinsics
- //===----------------------------------------------------------------------===//
- def : Pat<(loongarch_csrrd uimm14:$imm14), (CSRRD uimm14:$imm14)>;
- def : Pat<(loongarch_csrwr GPR:$rd, uimm14:$imm14),
- (CSRWR GPR:$rd, uimm14:$imm14)>;
- def : Pat<(loongarch_csrxchg GPR:$rd, GPR:$rj, uimm14:$imm14),
- (CSRXCHG GPR:$rd, GPR:$rj, uimm14:$imm14)>;
- def : Pat<(loongarch_iocsrrd_b GPR:$rj), (IOCSRRD_B GPR:$rj)>;
- def : Pat<(loongarch_iocsrrd_h GPR:$rj), (IOCSRRD_H GPR:$rj)>;
- def : Pat<(loongarch_iocsrrd_w GPR:$rj), (IOCSRRD_W GPR:$rj)>;
- def : Pat<(loongarch_iocsrwr_b GPR:$rd, GPR:$rj), (IOCSRWR_B GPR:$rd, GPR:$rj)>;
- def : Pat<(loongarch_iocsrwr_h GPR:$rd, GPR:$rj), (IOCSRWR_H GPR:$rd, GPR:$rj)>;
- def : Pat<(loongarch_iocsrwr_w GPR:$rd, GPR:$rj), (IOCSRWR_W GPR:$rd, GPR:$rj)>;
- def : Pat<(loongarch_cpucfg GPR:$rj), (CPUCFG GPR:$rj)>;
- let Predicates = [IsLA64] in {
- def : Pat<(loongarch_iocsrrd_d GPR:$rj), (IOCSRRD_D GPR:$rj)>;
- def : Pat<(loongarch_iocsrwr_d GPR:$rd, GPR:$rj), (IOCSRWR_D GPR:$rd, GPR:$rj)>;
- def : Pat<(int_loongarch_asrtle_d GPR:$rj, GPR:$rk),
- (ASRTLE_D GPR:$rj, GPR:$rk)>;
- def : Pat<(int_loongarch_asrtgt_d GPR:$rj, GPR:$rk),
- (ASRTGT_D GPR:$rj, GPR:$rk)>;
- def : Pat<(int_loongarch_lddir_d GPR:$rj, timm:$imm8),
- (LDDIR GPR:$rj, uimm8:$imm8)>;
- def : Pat<(int_loongarch_ldpte_d GPR:$rj, timm:$imm8),
- (LDPTE GPR:$rj, uimm8:$imm8)>;
- } // Predicates = [IsLA64]
|