1*5f757f3fSDimitry Andric//===-- RISCVGIsel.td - RISC-V GlobalISel Patterns ---------*- tablegen -*-===// 2*5f757f3fSDimitry Andric// 3*5f757f3fSDimitry Andric// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4*5f757f3fSDimitry Andric// See https://llvm.org/LICENSE.txt for license information. 5*5f757f3fSDimitry Andric// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6*5f757f3fSDimitry Andric// 7*5f757f3fSDimitry Andric//===----------------------------------------------------------------------===// 8*5f757f3fSDimitry Andric// 9*5f757f3fSDimitry Andric/// \file 10*5f757f3fSDimitry Andric/// This file contains patterns that are relevant to GlobalISel, including 11*5f757f3fSDimitry Andric/// GIComplexOperandMatcher definitions for equivalent SelectionDAG 12*5f757f3fSDimitry Andric/// ComplexPatterns. 13*5f757f3fSDimitry Andric// 14*5f757f3fSDimitry Andric//===----------------------------------------------------------------------===// 15*5f757f3fSDimitry Andric 16*5f757f3fSDimitry Andricinclude "RISCV.td" 17*5f757f3fSDimitry Andricinclude "RISCVCombine.td" 18*5f757f3fSDimitry Andric 19*5f757f3fSDimitry Andricdef simm12Plus1 : ImmLeaf<XLenVT, [{ 20*5f757f3fSDimitry Andric return (isInt<12>(Imm) && Imm != -2048) || Imm == 2048;}]>; 21*5f757f3fSDimitry Andricdef simm12Plus1i32 : ImmLeaf<i32, [{ 22*5f757f3fSDimitry Andric return (isInt<12>(Imm) && Imm != -2048) || Imm == 2048;}]>; 23*5f757f3fSDimitry Andric 24*5f757f3fSDimitry Andric// FIXME: This doesn't check that the G_CONSTANT we're deriving the immediate 25*5f757f3fSDimitry Andric// from is only used once 26*5f757f3fSDimitry Andricdef simm12Minus1Nonzero : ImmLeaf<XLenVT, [{ 27*5f757f3fSDimitry Andric return (Imm >= -2049 && Imm < 0) || (Imm > 0 && Imm <= 2046);}]>; 28*5f757f3fSDimitry Andric 29*5f757f3fSDimitry Andricdef simm12Minus1NonzeroNonNeg1 : ImmLeaf<XLenVT, [{ 30*5f757f3fSDimitry Andric return (Imm >= -2049 && Imm < -1) || (Imm > 0 && Imm <= 2046);}]>; 31*5f757f3fSDimitry Andric 32*5f757f3fSDimitry Andric// Return an immediate value plus 1. 33*5f757f3fSDimitry Andricdef ImmPlus1 : SDNodeXForm<imm, [{ 34*5f757f3fSDimitry Andric return CurDAG->getTargetConstant(N->getSExtValue() + 1, SDLoc(N), 35*5f757f3fSDimitry Andric N->getValuePtrVTpe(0));}]>; 36*5f757f3fSDimitry Andric 37*5f757f3fSDimitry Andricdef GINegImm : GICustomOperandRenderer<"renderNegImm">, 38*5f757f3fSDimitry Andric GISDNodeXFormEquiv<NegImm>; 39*5f757f3fSDimitry Andric 40*5f757f3fSDimitry Andricdef GIImmSubFromXLen : GICustomOperandRenderer<"renderImmSubFromXLen">, 41*5f757f3fSDimitry Andric GISDNodeXFormEquiv<ImmSubFromXLen>; 42*5f757f3fSDimitry Andricdef GIImmSubFrom32 : GICustomOperandRenderer<"renderImmSubFrom32">, 43*5f757f3fSDimitry Andric GISDNodeXFormEquiv<ImmSubFrom32>; 44*5f757f3fSDimitry Andric 45*5f757f3fSDimitry Andricdef GIImmPlus1 : 46*5f757f3fSDimitry Andric GICustomOperandRenderer<"renderImmPlus1">, 47*5f757f3fSDimitry Andric GISDNodeXFormEquiv<ImmPlus1>; 48*5f757f3fSDimitry Andric 49*5f757f3fSDimitry Andricdef GIAddrRegImm : 50*5f757f3fSDimitry Andric GIComplexOperandMatcher<s32, "selectAddrRegImm">, 51*5f757f3fSDimitry Andric GIComplexPatternEquiv<AddrRegImm>; 52*5f757f3fSDimitry Andric 53*5f757f3fSDimitry Andricdef gi_as_i64imm : GICustomOperandRenderer<"renderImm">, 54*5f757f3fSDimitry Andric GISDNodeXFormEquiv<as_i64imm>; 55*5f757f3fSDimitry Andric 56*5f757f3fSDimitry Andricdef gi_trailing_zero : GICustomOperandRenderer<"renderTrailingZeros">, 57*5f757f3fSDimitry Andric GISDNodeXFormEquiv<TrailingZeros>; 58*5f757f3fSDimitry Andric 59*5f757f3fSDimitry Andric// FIXME: This is labelled as handling 's32', however the ComplexPattern it 60*5f757f3fSDimitry Andric// refers to handles both i32 and i64 based on the HwMode. Currently this LLT 61*5f757f3fSDimitry Andric// parameter appears to be ignored so this pattern works for both, however we 62*5f757f3fSDimitry Andric// should add a LowLevelTypeByHwMode, and use that to define our XLenLLT instead 63*5f757f3fSDimitry Andric// here. 64*5f757f3fSDimitry Andricdef GIShiftMaskXLen : 65*5f757f3fSDimitry Andric GIComplexOperandMatcher<s32, "selectShiftMask">, 66*5f757f3fSDimitry Andric GIComplexPatternEquiv<shiftMaskXLen>; 67*5f757f3fSDimitry Andricdef GIShiftMask32 : 68*5f757f3fSDimitry Andric GIComplexOperandMatcher<s32, "selectShiftMask">, 69*5f757f3fSDimitry Andric GIComplexPatternEquiv<shiftMask32>; 70*5f757f3fSDimitry Andric 71*5f757f3fSDimitry Andricdef gi_sh1add_op : GIComplexOperandMatcher<s32, "selectSHXADDOp<1>">, 72*5f757f3fSDimitry Andric GIComplexPatternEquiv<sh1add_op>; 73*5f757f3fSDimitry Andricdef gi_sh2add_op : GIComplexOperandMatcher<s32, "selectSHXADDOp<2>">, 74*5f757f3fSDimitry Andric GIComplexPatternEquiv<sh2add_op>; 75*5f757f3fSDimitry Andricdef gi_sh3add_op : GIComplexOperandMatcher<s32, "selectSHXADDOp<3>">, 76*5f757f3fSDimitry Andric GIComplexPatternEquiv<sh3add_op>; 77*5f757f3fSDimitry Andric 78*5f757f3fSDimitry Andricdef gi_sh1add_uw_op : GIComplexOperandMatcher<s32, "selectSHXADD_UWOp<1>">, 79*5f757f3fSDimitry Andric GIComplexPatternEquiv<sh1add_uw_op>; 80*5f757f3fSDimitry Andricdef gi_sh2add_uw_op : GIComplexOperandMatcher<s32, "selectSHXADD_UWOp<2>">, 81*5f757f3fSDimitry Andric GIComplexPatternEquiv<sh2add_uw_op>; 82*5f757f3fSDimitry Andricdef gi_sh3add_uw_op : GIComplexOperandMatcher<s32, "selectSHXADD_UWOp<3>">, 83*5f757f3fSDimitry Andric GIComplexPatternEquiv<sh3add_uw_op>; 84*5f757f3fSDimitry Andric 85*5f757f3fSDimitry Andric// FIXME: Canonicalize (sub X, C) -> (add X, -C) earlier. 86*5f757f3fSDimitry Andricdef : Pat<(XLenVT (sub GPR:$rs1, simm12Plus1:$imm)), 87*5f757f3fSDimitry Andric (ADDI GPR:$rs1, (NegImm simm12Plus1:$imm))>; 88*5f757f3fSDimitry Andric 89*5f757f3fSDimitry Andriclet Predicates = [IsRV64] in { 90*5f757f3fSDimitry Andricdef : Pat<(i32 (sub GPR:$rs1, simm12Plus1i32:$imm)), 91*5f757f3fSDimitry Andric (ADDIW GPR:$rs1, (i64 (NegImm $imm)))>; 92*5f757f3fSDimitry Andric 93*5f757f3fSDimitry Andricdef : Pat<(i32 (shl GPR:$rs1, (i32 GPR:$rs2))), (SLLW GPR:$rs1, GPR:$rs2)>; 94*5f757f3fSDimitry Andricdef : Pat<(i32 (sra GPR:$rs1, (i32 GPR:$rs2))), (SRAW GPR:$rs1, GPR:$rs2)>; 95*5f757f3fSDimitry Andricdef : Pat<(i32 (srl GPR:$rs1, (i32 GPR:$rs2))), (SRLW GPR:$rs1, GPR:$rs2)>; 96*5f757f3fSDimitry Andric} 97*5f757f3fSDimitry Andric 98*5f757f3fSDimitry Andric// Ptr type used in patterns with GlobalISelEmitter 99*5f757f3fSDimitry Andricdef PtrVT : PtrValueTypeByHwMode<XLenVT, 0>; 100*5f757f3fSDimitry Andric 101*5f757f3fSDimitry Andric// Define pattern expansions for pointer ult/slt conditional codes 102*5f757f3fSDimitry Andricdef : Pat<(XLenVT (setult (PtrVT GPR:$rs1), simm12:$imm12)), 103*5f757f3fSDimitry Andric (SLTIU GPR:$rs1, simm12:$imm12)>; 104*5f757f3fSDimitry Andricdef : Pat<(XLenVT (setult (PtrVT GPR:$rs1), (PtrVT GPR:$rs2))), 105*5f757f3fSDimitry Andric (SLTU GPR:$rs1, GPR:$rs2)>; 106*5f757f3fSDimitry Andricdef : Pat<(XLenVT (setlt (PtrVT GPR:$rs1), simm12:$imm12)), 107*5f757f3fSDimitry Andric (SLTI GPR:$rs1, simm12:$imm12)>; 108*5f757f3fSDimitry Andricdef : Pat<(XLenVT (setlt (PtrVT GPR:$rs1), (PtrVT GPR:$rs2))), 109*5f757f3fSDimitry Andric (SLT GPR:$rs1, GPR:$rs2)>; 110*5f757f3fSDimitry Andric 111*5f757f3fSDimitry Andric// Define pattern expansions for setcc operations that aren't directly 112*5f757f3fSDimitry Andric// handled by a RISC-V instruction. 113*5f757f3fSDimitry Andricforeach Ty = [PtrVT, XLenVT] in { 114*5f757f3fSDimitry Andricdef : Pat<(XLenVT (seteq (Ty GPR:$rs1), (Ty 0))), (SLTIU GPR:$rs1, 1)>; 115*5f757f3fSDimitry Andricdef : Pat<(XLenVT (seteq (Ty GPR:$rs1), (Ty simm12Plus1:$imm12))), 116*5f757f3fSDimitry Andric (SLTIU (ADDI GPR:$rs1, (NegImm simm12Plus1:$imm12)), 1)>; 117*5f757f3fSDimitry Andricdef : Pat<(XLenVT (seteq (Ty GPR:$rs1), (Ty GPR:$rs2))), 118*5f757f3fSDimitry Andric (SLTIU (XOR GPR:$rs1, GPR:$rs2), 1)>; 119*5f757f3fSDimitry Andricdef : Pat<(XLenVT (setne (Ty GPR:$rs1), (Ty 0))), (SLTU (XLenVT X0), GPR:$rs1)>; 120*5f757f3fSDimitry Andricdef : Pat<(XLenVT (setne (Ty GPR:$rs1), (Ty simm12Plus1:$imm12))), 121*5f757f3fSDimitry Andric (SLTU (XLenVT X0), (ADDI GPR:$rs1, (NegImm simm12Plus1:$imm12)))>; 122*5f757f3fSDimitry Andricdef : Pat<(XLenVT (setne (Ty GPR:$rs1), (Ty GPR:$rs2))), 123*5f757f3fSDimitry Andric (SLTU (XLenVT X0), (XOR GPR:$rs1, GPR:$rs2))>; 124*5f757f3fSDimitry Andricdef : Pat<(XLenVT (setugt (Ty GPR:$rs1), (Ty simm12Minus1NonzeroNonNeg1:$imm))), 125*5f757f3fSDimitry Andric (XORI (SLTIU GPR:$rs1, 126*5f757f3fSDimitry Andric (ImmPlus1 simm12Minus1NonzeroNonNeg1:$imm)), 1)>; 127*5f757f3fSDimitry Andricdef : Pat<(XLenVT (setugt (Ty GPR:$rs1), (Ty GPR:$rs2))), 128*5f757f3fSDimitry Andric (SLTU GPR:$rs2, GPR:$rs1)>; 129*5f757f3fSDimitry Andricdef : Pat<(XLenVT (setgt (Ty GPR:$rs1), (Ty simm12Minus1Nonzero:$imm))), 130*5f757f3fSDimitry Andric (XORI (SLTI GPR:$rs1, (ImmPlus1 simm12Minus1Nonzero:$imm)), 1)>; 131*5f757f3fSDimitry Andricdef : Pat<(XLenVT (setgt (Ty GPR:$rs1), (Ty GPR:$rs2))), 132*5f757f3fSDimitry Andric (SLT GPR:$rs2, GPR:$rs1)>; 133*5f757f3fSDimitry Andricdef : Pat<(XLenVT (setuge (XLenVT GPR:$rs1), (Ty simm12:$imm))), 134*5f757f3fSDimitry Andric (XORI (SLTIU GPR:$rs1, simm12:$imm), 1)>; 135*5f757f3fSDimitry Andricdef : Pat<(XLenVT (setuge (Ty GPR:$rs1), (Ty GPR:$rs2))), 136*5f757f3fSDimitry Andric (XORI (SLTU GPR:$rs1, GPR:$rs2), 1)>; 137*5f757f3fSDimitry Andricdef : Pat<(XLenVT (setge (Ty GPR:$rs1), (Ty simm12:$imm))), 138*5f757f3fSDimitry Andric (XORI (SLTI GPR:$rs1, simm12:$imm), 1)>; 139*5f757f3fSDimitry Andricdef : Pat<(XLenVT (setge (Ty GPR:$rs1), (Ty GPR:$rs2))), 140*5f757f3fSDimitry Andric (XORI (SLT GPR:$rs1, GPR:$rs2), 1)>; 141*5f757f3fSDimitry Andricdef : Pat<(XLenVT (setule (Ty GPR:$rs1), (Ty simm12Minus1NonzeroNonNeg1:$imm))), 142*5f757f3fSDimitry Andric (SLTIU GPR:$rs1, (ImmPlus1 simm12Minus1NonzeroNonNeg1:$imm))>; 143*5f757f3fSDimitry Andricdef : Pat<(XLenVT (setule (Ty GPR:$rs1), (Ty GPR:$rs2))), 144*5f757f3fSDimitry Andric (XORI (SLTU GPR:$rs2, GPR:$rs1), 1)>; 145*5f757f3fSDimitry Andricdef : Pat<(XLenVT (setle (Ty GPR:$rs1), (Ty simm12Minus1Nonzero:$imm))), 146*5f757f3fSDimitry Andric (SLTI GPR:$rs1, (ImmPlus1 simm12Minus1Nonzero:$imm))>; 147*5f757f3fSDimitry Andricdef : Pat<(XLenVT (setle (Ty GPR:$rs1), (Ty GPR:$rs2))), 148*5f757f3fSDimitry Andric (XORI (SLT GPR:$rs2, GPR:$rs1), 1)>; 149*5f757f3fSDimitry Andric} 150*5f757f3fSDimitry Andric 151*5f757f3fSDimitry Andriclet Predicates = [IsRV32] in { 152*5f757f3fSDimitry Andricdef : LdPat<load, LW, PtrVT>; 153*5f757f3fSDimitry Andricdef : StPat<store, SW, GPR, PtrVT>; 154*5f757f3fSDimitry Andric} 155*5f757f3fSDimitry Andric 156*5f757f3fSDimitry Andriclet Predicates = [IsRV64] in { 157*5f757f3fSDimitry Andricdef : LdPat<load, LD, PtrVT>; 158*5f757f3fSDimitry Andricdef : StPat<store, SD, GPR, PtrVT>; 159*5f757f3fSDimitry Andric} 160