xref: /freebsd-src/contrib/llvm-project/llvm/lib/Target/RISCV/RISCVGISel.td (revision 5f757f3ff9144b609b3c433dfd370cc6bdc191ad)
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