xref: /aosp_15_r20/external/llvm/lib/Target/AMDGPU/AMDGPUInstructions.td (revision 9880d6810fe72a1726cb53787c6711e909410d58)
1*9880d681SAndroid Build Coastguard Worker//===-- AMDGPUInstructions.td - Common instruction defs ---*- tablegen -*-===//
2*9880d681SAndroid Build Coastguard Worker//
3*9880d681SAndroid Build Coastguard Worker//                     The LLVM Compiler Infrastructure
4*9880d681SAndroid Build Coastguard Worker//
5*9880d681SAndroid Build Coastguard Worker// This file is distributed under the University of Illinois Open Source
6*9880d681SAndroid Build Coastguard Worker// License. See LICENSE.TXT for details.
7*9880d681SAndroid Build Coastguard Worker//
8*9880d681SAndroid Build Coastguard Worker//===----------------------------------------------------------------------===//
9*9880d681SAndroid Build Coastguard Worker//
10*9880d681SAndroid Build Coastguard Worker// This file contains instruction defs that are common to all hw codegen
11*9880d681SAndroid Build Coastguard Worker// targets.
12*9880d681SAndroid Build Coastguard Worker//
13*9880d681SAndroid Build Coastguard Worker//===----------------------------------------------------------------------===//
14*9880d681SAndroid Build Coastguard Worker
15*9880d681SAndroid Build Coastguard Workerclass AMDGPUInst <dag outs, dag ins, string asm = "",
16*9880d681SAndroid Build Coastguard Worker  list<dag> pattern = []> : Instruction {
17*9880d681SAndroid Build Coastguard Worker  field bit isRegisterLoad = 0;
18*9880d681SAndroid Build Coastguard Worker  field bit isRegisterStore = 0;
19*9880d681SAndroid Build Coastguard Worker
20*9880d681SAndroid Build Coastguard Worker  let Namespace = "AMDGPU";
21*9880d681SAndroid Build Coastguard Worker  let OutOperandList = outs;
22*9880d681SAndroid Build Coastguard Worker  let InOperandList = ins;
23*9880d681SAndroid Build Coastguard Worker  let AsmString = asm;
24*9880d681SAndroid Build Coastguard Worker  let Pattern = pattern;
25*9880d681SAndroid Build Coastguard Worker  let Itinerary = NullALU;
26*9880d681SAndroid Build Coastguard Worker
27*9880d681SAndroid Build Coastguard Worker  // SoftFail is a field the disassembler can use to provide a way for
28*9880d681SAndroid Build Coastguard Worker  // instructions to not match without killing the whole decode process. It is
29*9880d681SAndroid Build Coastguard Worker  // mainly used for ARM, but Tablegen expects this field to exist or it fails
30*9880d681SAndroid Build Coastguard Worker  // to build the decode table.
31*9880d681SAndroid Build Coastguard Worker  field bits<64> SoftFail = 0;
32*9880d681SAndroid Build Coastguard Worker
33*9880d681SAndroid Build Coastguard Worker  let DecoderNamespace = Namespace;
34*9880d681SAndroid Build Coastguard Worker
35*9880d681SAndroid Build Coastguard Worker  let TSFlags{63} = isRegisterLoad;
36*9880d681SAndroid Build Coastguard Worker  let TSFlags{62} = isRegisterStore;
37*9880d681SAndroid Build Coastguard Worker}
38*9880d681SAndroid Build Coastguard Worker
39*9880d681SAndroid Build Coastguard Workerclass AMDGPUShaderInst <dag outs, dag ins, string asm = "",
40*9880d681SAndroid Build Coastguard Worker  list<dag> pattern = []> : AMDGPUInst<outs, ins, asm, pattern> {
41*9880d681SAndroid Build Coastguard Worker
42*9880d681SAndroid Build Coastguard Worker  field bits<32> Inst = 0xffffffff;
43*9880d681SAndroid Build Coastguard Worker}
44*9880d681SAndroid Build Coastguard Worker
45*9880d681SAndroid Build Coastguard Workerdef FP32Denormals : Predicate<"Subtarget.hasFP32Denormals()">;
46*9880d681SAndroid Build Coastguard Workerdef FP64Denormals : Predicate<"Subtarget.hasFP64Denormals()">;
47*9880d681SAndroid Build Coastguard Workerdef UnsafeFPMath : Predicate<"TM.Options.UnsafeFPMath">;
48*9880d681SAndroid Build Coastguard Worker
49*9880d681SAndroid Build Coastguard Workerdef InstFlag : OperandWithDefaultOps <i32, (ops (i32 0))>;
50*9880d681SAndroid Build Coastguard Workerdef ADDRIndirect : ComplexPattern<iPTR, 2, "SelectADDRIndirect", [], []>;
51*9880d681SAndroid Build Coastguard Worker
52*9880d681SAndroid Build Coastguard Worker// 32-bit VALU immediate operand that uses the constant bus.
53*9880d681SAndroid Build Coastguard Workerdef u32kimm : Operand<i32> {
54*9880d681SAndroid Build Coastguard Worker  let OperandNamespace = "AMDGPU";
55*9880d681SAndroid Build Coastguard Worker  let OperandType = "OPERAND_KIMM32";
56*9880d681SAndroid Build Coastguard Worker  let PrintMethod = "printU32ImmOperand";
57*9880d681SAndroid Build Coastguard Worker}
58*9880d681SAndroid Build Coastguard Worker
59*9880d681SAndroid Build Coastguard Workerlet OperandType = "OPERAND_IMMEDIATE" in {
60*9880d681SAndroid Build Coastguard Worker
61*9880d681SAndroid Build Coastguard Workerdef u32imm : Operand<i32> {
62*9880d681SAndroid Build Coastguard Worker  let PrintMethod = "printU32ImmOperand";
63*9880d681SAndroid Build Coastguard Worker}
64*9880d681SAndroid Build Coastguard Worker
65*9880d681SAndroid Build Coastguard Workerdef u16imm : Operand<i16> {
66*9880d681SAndroid Build Coastguard Worker  let PrintMethod = "printU16ImmOperand";
67*9880d681SAndroid Build Coastguard Worker}
68*9880d681SAndroid Build Coastguard Worker
69*9880d681SAndroid Build Coastguard Workerdef u8imm : Operand<i8> {
70*9880d681SAndroid Build Coastguard Worker  let PrintMethod = "printU8ImmOperand";
71*9880d681SAndroid Build Coastguard Worker}
72*9880d681SAndroid Build Coastguard Worker
73*9880d681SAndroid Build Coastguard Worker} // End OperandType = "OPERAND_IMMEDIATE"
74*9880d681SAndroid Build Coastguard Worker
75*9880d681SAndroid Build Coastguard Worker//===--------------------------------------------------------------------===//
76*9880d681SAndroid Build Coastguard Worker// Custom Operands
77*9880d681SAndroid Build Coastguard Worker//===--------------------------------------------------------------------===//
78*9880d681SAndroid Build Coastguard Workerdef brtarget   : Operand<OtherVT>;
79*9880d681SAndroid Build Coastguard Worker
80*9880d681SAndroid Build Coastguard Worker//===----------------------------------------------------------------------===//
81*9880d681SAndroid Build Coastguard Worker// PatLeafs for floating-point comparisons
82*9880d681SAndroid Build Coastguard Worker//===----------------------------------------------------------------------===//
83*9880d681SAndroid Build Coastguard Worker
84*9880d681SAndroid Build Coastguard Workerdef COND_OEQ : PatLeaf <
85*9880d681SAndroid Build Coastguard Worker  (cond),
86*9880d681SAndroid Build Coastguard Worker  [{return N->get() == ISD::SETOEQ || N->get() == ISD::SETEQ;}]
87*9880d681SAndroid Build Coastguard Worker>;
88*9880d681SAndroid Build Coastguard Worker
89*9880d681SAndroid Build Coastguard Workerdef COND_ONE : PatLeaf <
90*9880d681SAndroid Build Coastguard Worker  (cond),
91*9880d681SAndroid Build Coastguard Worker  [{return N->get() == ISD::SETONE || N->get() == ISD::SETNE;}]
92*9880d681SAndroid Build Coastguard Worker>;
93*9880d681SAndroid Build Coastguard Worker
94*9880d681SAndroid Build Coastguard Workerdef COND_OGT : PatLeaf <
95*9880d681SAndroid Build Coastguard Worker  (cond),
96*9880d681SAndroid Build Coastguard Worker  [{return N->get() == ISD::SETOGT || N->get() == ISD::SETGT;}]
97*9880d681SAndroid Build Coastguard Worker>;
98*9880d681SAndroid Build Coastguard Worker
99*9880d681SAndroid Build Coastguard Workerdef COND_OGE : PatLeaf <
100*9880d681SAndroid Build Coastguard Worker  (cond),
101*9880d681SAndroid Build Coastguard Worker  [{return N->get() == ISD::SETOGE || N->get() == ISD::SETGE;}]
102*9880d681SAndroid Build Coastguard Worker>;
103*9880d681SAndroid Build Coastguard Worker
104*9880d681SAndroid Build Coastguard Workerdef COND_OLT : PatLeaf <
105*9880d681SAndroid Build Coastguard Worker  (cond),
106*9880d681SAndroid Build Coastguard Worker  [{return N->get() == ISD::SETOLT || N->get() == ISD::SETLT;}]
107*9880d681SAndroid Build Coastguard Worker>;
108*9880d681SAndroid Build Coastguard Worker
109*9880d681SAndroid Build Coastguard Workerdef COND_OLE : PatLeaf <
110*9880d681SAndroid Build Coastguard Worker  (cond),
111*9880d681SAndroid Build Coastguard Worker  [{return N->get() == ISD::SETOLE || N->get() == ISD::SETLE;}]
112*9880d681SAndroid Build Coastguard Worker>;
113*9880d681SAndroid Build Coastguard Worker
114*9880d681SAndroid Build Coastguard Worker
115*9880d681SAndroid Build Coastguard Workerdef COND_O : PatLeaf <(cond), [{return N->get() == ISD::SETO;}]>;
116*9880d681SAndroid Build Coastguard Workerdef COND_UO : PatLeaf <(cond), [{return N->get() == ISD::SETUO;}]>;
117*9880d681SAndroid Build Coastguard Worker
118*9880d681SAndroid Build Coastguard Worker//===----------------------------------------------------------------------===//
119*9880d681SAndroid Build Coastguard Worker// PatLeafs for unsigned / unordered comparisons
120*9880d681SAndroid Build Coastguard Worker//===----------------------------------------------------------------------===//
121*9880d681SAndroid Build Coastguard Worker
122*9880d681SAndroid Build Coastguard Workerdef COND_UEQ : PatLeaf <(cond), [{return N->get() == ISD::SETUEQ;}]>;
123*9880d681SAndroid Build Coastguard Workerdef COND_UNE : PatLeaf <(cond), [{return N->get() == ISD::SETUNE;}]>;
124*9880d681SAndroid Build Coastguard Workerdef COND_UGT : PatLeaf <(cond), [{return N->get() == ISD::SETUGT;}]>;
125*9880d681SAndroid Build Coastguard Workerdef COND_UGE : PatLeaf <(cond), [{return N->get() == ISD::SETUGE;}]>;
126*9880d681SAndroid Build Coastguard Workerdef COND_ULT : PatLeaf <(cond), [{return N->get() == ISD::SETULT;}]>;
127*9880d681SAndroid Build Coastguard Workerdef COND_ULE : PatLeaf <(cond), [{return N->get() == ISD::SETULE;}]>;
128*9880d681SAndroid Build Coastguard Worker
129*9880d681SAndroid Build Coastguard Worker// XXX - For some reason R600 version is preferring to use unordered
130*9880d681SAndroid Build Coastguard Worker// for setne?
131*9880d681SAndroid Build Coastguard Workerdef COND_UNE_NE : PatLeaf <
132*9880d681SAndroid Build Coastguard Worker  (cond),
133*9880d681SAndroid Build Coastguard Worker  [{return N->get() == ISD::SETUNE || N->get() == ISD::SETNE;}]
134*9880d681SAndroid Build Coastguard Worker>;
135*9880d681SAndroid Build Coastguard Worker
136*9880d681SAndroid Build Coastguard Worker//===----------------------------------------------------------------------===//
137*9880d681SAndroid Build Coastguard Worker// PatLeafs for signed comparisons
138*9880d681SAndroid Build Coastguard Worker//===----------------------------------------------------------------------===//
139*9880d681SAndroid Build Coastguard Worker
140*9880d681SAndroid Build Coastguard Workerdef COND_SGT : PatLeaf <(cond), [{return N->get() == ISD::SETGT;}]>;
141*9880d681SAndroid Build Coastguard Workerdef COND_SGE : PatLeaf <(cond), [{return N->get() == ISD::SETGE;}]>;
142*9880d681SAndroid Build Coastguard Workerdef COND_SLT : PatLeaf <(cond), [{return N->get() == ISD::SETLT;}]>;
143*9880d681SAndroid Build Coastguard Workerdef COND_SLE : PatLeaf <(cond), [{return N->get() == ISD::SETLE;}]>;
144*9880d681SAndroid Build Coastguard Worker
145*9880d681SAndroid Build Coastguard Worker//===----------------------------------------------------------------------===//
146*9880d681SAndroid Build Coastguard Worker// PatLeafs for integer equality
147*9880d681SAndroid Build Coastguard Worker//===----------------------------------------------------------------------===//
148*9880d681SAndroid Build Coastguard Worker
149*9880d681SAndroid Build Coastguard Workerdef COND_EQ : PatLeaf <
150*9880d681SAndroid Build Coastguard Worker  (cond),
151*9880d681SAndroid Build Coastguard Worker  [{return N->get() == ISD::SETEQ || N->get() == ISD::SETUEQ;}]
152*9880d681SAndroid Build Coastguard Worker>;
153*9880d681SAndroid Build Coastguard Worker
154*9880d681SAndroid Build Coastguard Workerdef COND_NE : PatLeaf <
155*9880d681SAndroid Build Coastguard Worker  (cond),
156*9880d681SAndroid Build Coastguard Worker  [{return N->get() == ISD::SETNE || N->get() == ISD::SETUNE;}]
157*9880d681SAndroid Build Coastguard Worker>;
158*9880d681SAndroid Build Coastguard Worker
159*9880d681SAndroid Build Coastguard Workerdef COND_NULL : PatLeaf <
160*9880d681SAndroid Build Coastguard Worker  (cond),
161*9880d681SAndroid Build Coastguard Worker  [{(void)N; return false;}]
162*9880d681SAndroid Build Coastguard Worker>;
163*9880d681SAndroid Build Coastguard Worker
164*9880d681SAndroid Build Coastguard Worker
165*9880d681SAndroid Build Coastguard Worker//===----------------------------------------------------------------------===//
166*9880d681SAndroid Build Coastguard Worker// Misc. PatFrags
167*9880d681SAndroid Build Coastguard Worker//===----------------------------------------------------------------------===//
168*9880d681SAndroid Build Coastguard Worker
169*9880d681SAndroid Build Coastguard Workerclass HasOneUseBinOp<SDPatternOperator op> : PatFrag<
170*9880d681SAndroid Build Coastguard Worker  (ops node:$src0, node:$src1),
171*9880d681SAndroid Build Coastguard Worker  (op $src0, $src1),
172*9880d681SAndroid Build Coastguard Worker  [{ return N->hasOneUse(); }]
173*9880d681SAndroid Build Coastguard Worker>;
174*9880d681SAndroid Build Coastguard Worker
175*9880d681SAndroid Build Coastguard Worker//===----------------------------------------------------------------------===//
176*9880d681SAndroid Build Coastguard Worker// Load/Store Pattern Fragments
177*9880d681SAndroid Build Coastguard Worker//===----------------------------------------------------------------------===//
178*9880d681SAndroid Build Coastguard Worker
179*9880d681SAndroid Build Coastguard Workerclass PrivateMemOp <dag ops, dag frag> : PatFrag <ops, frag, [{
180*9880d681SAndroid Build Coastguard Worker  return cast<MemSDNode>(N)->getAddressSpace() == AMDGPUAS::PRIVATE_ADDRESS;
181*9880d681SAndroid Build Coastguard Worker}]>;
182*9880d681SAndroid Build Coastguard Worker
183*9880d681SAndroid Build Coastguard Workerclass PrivateLoad <SDPatternOperator op> : PrivateMemOp <
184*9880d681SAndroid Build Coastguard Worker  (ops node:$ptr), (op node:$ptr)
185*9880d681SAndroid Build Coastguard Worker>;
186*9880d681SAndroid Build Coastguard Worker
187*9880d681SAndroid Build Coastguard Workerclass PrivateStore <SDPatternOperator op> : PrivateMemOp <
188*9880d681SAndroid Build Coastguard Worker  (ops node:$value, node:$ptr), (op node:$value, node:$ptr)
189*9880d681SAndroid Build Coastguard Worker>;
190*9880d681SAndroid Build Coastguard Worker
191*9880d681SAndroid Build Coastguard Workerdef load_private : PrivateLoad <load>;
192*9880d681SAndroid Build Coastguard Worker
193*9880d681SAndroid Build Coastguard Workerdef truncstorei8_private : PrivateStore <truncstorei8>;
194*9880d681SAndroid Build Coastguard Workerdef truncstorei16_private : PrivateStore <truncstorei16>;
195*9880d681SAndroid Build Coastguard Workerdef store_private : PrivateStore <store>;
196*9880d681SAndroid Build Coastguard Worker
197*9880d681SAndroid Build Coastguard Workerclass GlobalMemOp <dag ops, dag frag> : PatFrag <ops, frag, [{
198*9880d681SAndroid Build Coastguard Worker  return cast<MemSDNode>(N)->getAddressSpace() == AMDGPUAS::GLOBAL_ADDRESS;
199*9880d681SAndroid Build Coastguard Worker}]>;
200*9880d681SAndroid Build Coastguard Worker
201*9880d681SAndroid Build Coastguard Worker// Global address space loads
202*9880d681SAndroid Build Coastguard Workerclass GlobalLoad <SDPatternOperator op> : GlobalMemOp <
203*9880d681SAndroid Build Coastguard Worker  (ops node:$ptr), (op node:$ptr)
204*9880d681SAndroid Build Coastguard Worker>;
205*9880d681SAndroid Build Coastguard Worker
206*9880d681SAndroid Build Coastguard Workerdef global_load : GlobalLoad <load>;
207*9880d681SAndroid Build Coastguard Worker
208*9880d681SAndroid Build Coastguard Worker// Global address space stores
209*9880d681SAndroid Build Coastguard Workerclass GlobalStore <SDPatternOperator op> : GlobalMemOp <
210*9880d681SAndroid Build Coastguard Worker  (ops node:$value, node:$ptr), (op node:$value, node:$ptr)
211*9880d681SAndroid Build Coastguard Worker>;
212*9880d681SAndroid Build Coastguard Worker
213*9880d681SAndroid Build Coastguard Workerdef global_store : GlobalStore <store>;
214*9880d681SAndroid Build Coastguard Workerdef global_store_atomic : GlobalStore<atomic_store>;
215*9880d681SAndroid Build Coastguard Worker
216*9880d681SAndroid Build Coastguard Worker
217*9880d681SAndroid Build Coastguard Workerclass ConstantMemOp <dag ops, dag frag> : PatFrag <ops, frag, [{
218*9880d681SAndroid Build Coastguard Worker  return cast<MemSDNode>(N)->getAddressSpace() == AMDGPUAS::CONSTANT_ADDRESS;
219*9880d681SAndroid Build Coastguard Worker}]>;
220*9880d681SAndroid Build Coastguard Worker
221*9880d681SAndroid Build Coastguard Worker// Constant address space loads
222*9880d681SAndroid Build Coastguard Workerclass ConstantLoad <SDPatternOperator op> : ConstantMemOp <
223*9880d681SAndroid Build Coastguard Worker  (ops node:$ptr), (op node:$ptr)
224*9880d681SAndroid Build Coastguard Worker>;
225*9880d681SAndroid Build Coastguard Worker
226*9880d681SAndroid Build Coastguard Workerdef constant_load : ConstantLoad<load>;
227*9880d681SAndroid Build Coastguard Worker
228*9880d681SAndroid Build Coastguard Workerclass LocalMemOp <dag ops, dag frag> : PatFrag <ops, frag, [{
229*9880d681SAndroid Build Coastguard Worker  return cast<MemSDNode>(N)->getAddressSpace() == AMDGPUAS::LOCAL_ADDRESS;
230*9880d681SAndroid Build Coastguard Worker}]>;
231*9880d681SAndroid Build Coastguard Worker
232*9880d681SAndroid Build Coastguard Worker// Local address space loads
233*9880d681SAndroid Build Coastguard Workerclass LocalLoad <SDPatternOperator op> : LocalMemOp <
234*9880d681SAndroid Build Coastguard Worker  (ops node:$ptr), (op node:$ptr)
235*9880d681SAndroid Build Coastguard Worker>;
236*9880d681SAndroid Build Coastguard Worker
237*9880d681SAndroid Build Coastguard Workerclass LocalStore <SDPatternOperator op> : LocalMemOp <
238*9880d681SAndroid Build Coastguard Worker  (ops node:$value, node:$ptr), (op node:$value, node:$ptr)
239*9880d681SAndroid Build Coastguard Worker>;
240*9880d681SAndroid Build Coastguard Worker
241*9880d681SAndroid Build Coastguard Workerclass FlatMemOp <dag ops, dag frag> : PatFrag <ops, frag, [{
242*9880d681SAndroid Build Coastguard Worker  return cast<MemSDNode>(N)->getAddressSPace() == AMDGPUAS::FLAT_ADDRESS;
243*9880d681SAndroid Build Coastguard Worker}]>;
244*9880d681SAndroid Build Coastguard Worker
245*9880d681SAndroid Build Coastguard Workerclass FlatLoad <SDPatternOperator op> : FlatMemOp <
246*9880d681SAndroid Build Coastguard Worker  (ops node:$ptr), (op node:$ptr)
247*9880d681SAndroid Build Coastguard Worker>;
248*9880d681SAndroid Build Coastguard Worker
249*9880d681SAndroid Build Coastguard Workerclass AZExtLoadBase <SDPatternOperator ld_node>: PatFrag<(ops node:$ptr),
250*9880d681SAndroid Build Coastguard Worker                                              (ld_node node:$ptr), [{
251*9880d681SAndroid Build Coastguard Worker  LoadSDNode *L = cast<LoadSDNode>(N);
252*9880d681SAndroid Build Coastguard Worker  return L->getExtensionType() == ISD::ZEXTLOAD ||
253*9880d681SAndroid Build Coastguard Worker         L->getExtensionType() == ISD::EXTLOAD;
254*9880d681SAndroid Build Coastguard Worker}]>;
255*9880d681SAndroid Build Coastguard Worker
256*9880d681SAndroid Build Coastguard Workerdef az_extload : AZExtLoadBase <unindexedload>;
257*9880d681SAndroid Build Coastguard Worker
258*9880d681SAndroid Build Coastguard Workerdef az_extloadi8 : PatFrag<(ops node:$ptr), (az_extload node:$ptr), [{
259*9880d681SAndroid Build Coastguard Worker  return cast<LoadSDNode>(N)->getMemoryVT() == MVT::i8;
260*9880d681SAndroid Build Coastguard Worker}]>;
261*9880d681SAndroid Build Coastguard Worker
262*9880d681SAndroid Build Coastguard Workerdef az_extloadi8_global : GlobalLoad <az_extloadi8>;
263*9880d681SAndroid Build Coastguard Workerdef sextloadi8_global : GlobalLoad <sextloadi8>;
264*9880d681SAndroid Build Coastguard Worker
265*9880d681SAndroid Build Coastguard Workerdef az_extloadi8_constant : ConstantLoad <az_extloadi8>;
266*9880d681SAndroid Build Coastguard Workerdef sextloadi8_constant : ConstantLoad <sextloadi8>;
267*9880d681SAndroid Build Coastguard Worker
268*9880d681SAndroid Build Coastguard Workerdef az_extloadi8_local : LocalLoad <az_extloadi8>;
269*9880d681SAndroid Build Coastguard Workerdef sextloadi8_local : LocalLoad <sextloadi8>;
270*9880d681SAndroid Build Coastguard Worker
271*9880d681SAndroid Build Coastguard Workerdef extloadi8_private : PrivateLoad <az_extloadi8>;
272*9880d681SAndroid Build Coastguard Workerdef sextloadi8_private : PrivateLoad <sextloadi8>;
273*9880d681SAndroid Build Coastguard Worker
274*9880d681SAndroid Build Coastguard Workerdef az_extloadi16 : PatFrag<(ops node:$ptr), (az_extload node:$ptr), [{
275*9880d681SAndroid Build Coastguard Worker  return cast<LoadSDNode>(N)->getMemoryVT() == MVT::i16;
276*9880d681SAndroid Build Coastguard Worker}]>;
277*9880d681SAndroid Build Coastguard Worker
278*9880d681SAndroid Build Coastguard Workerdef az_extloadi16_global : GlobalLoad <az_extloadi16>;
279*9880d681SAndroid Build Coastguard Workerdef sextloadi16_global : GlobalLoad <sextloadi16>;
280*9880d681SAndroid Build Coastguard Worker
281*9880d681SAndroid Build Coastguard Workerdef az_extloadi16_constant : ConstantLoad <az_extloadi16>;
282*9880d681SAndroid Build Coastguard Workerdef sextloadi16_constant : ConstantLoad <sextloadi16>;
283*9880d681SAndroid Build Coastguard Worker
284*9880d681SAndroid Build Coastguard Workerdef az_extloadi16_local : LocalLoad <az_extloadi16>;
285*9880d681SAndroid Build Coastguard Workerdef sextloadi16_local : LocalLoad <sextloadi16>;
286*9880d681SAndroid Build Coastguard Worker
287*9880d681SAndroid Build Coastguard Workerdef extloadi16_private : PrivateLoad <az_extloadi16>;
288*9880d681SAndroid Build Coastguard Workerdef sextloadi16_private : PrivateLoad <sextloadi16>;
289*9880d681SAndroid Build Coastguard Worker
290*9880d681SAndroid Build Coastguard Workerdef az_extloadi32 : PatFrag<(ops node:$ptr), (az_extload node:$ptr), [{
291*9880d681SAndroid Build Coastguard Worker  return cast<LoadSDNode>(N)->getMemoryVT() == MVT::i32;
292*9880d681SAndroid Build Coastguard Worker}]>;
293*9880d681SAndroid Build Coastguard Worker
294*9880d681SAndroid Build Coastguard Workerdef az_extloadi32_global : GlobalLoad <az_extloadi32>;
295*9880d681SAndroid Build Coastguard Worker
296*9880d681SAndroid Build Coastguard Workerdef az_extloadi32_flat : FlatLoad <az_extloadi32>;
297*9880d681SAndroid Build Coastguard Worker
298*9880d681SAndroid Build Coastguard Workerdef az_extloadi32_constant : ConstantLoad <az_extloadi32>;
299*9880d681SAndroid Build Coastguard Worker
300*9880d681SAndroid Build Coastguard Workerdef truncstorei8_global : GlobalStore <truncstorei8>;
301*9880d681SAndroid Build Coastguard Workerdef truncstorei16_global : GlobalStore <truncstorei16>;
302*9880d681SAndroid Build Coastguard Worker
303*9880d681SAndroid Build Coastguard Workerdef local_store : LocalStore <store>;
304*9880d681SAndroid Build Coastguard Workerdef truncstorei8_local : LocalStore <truncstorei8>;
305*9880d681SAndroid Build Coastguard Workerdef truncstorei16_local : LocalStore <truncstorei16>;
306*9880d681SAndroid Build Coastguard Worker
307*9880d681SAndroid Build Coastguard Workerdef local_load : LocalLoad <load>;
308*9880d681SAndroid Build Coastguard Worker
309*9880d681SAndroid Build Coastguard Workerclass Aligned8Bytes <dag ops, dag frag> : PatFrag <ops, frag, [{
310*9880d681SAndroid Build Coastguard Worker    return cast<MemSDNode>(N)->getAlignment() % 8 == 0;
311*9880d681SAndroid Build Coastguard Worker}]>;
312*9880d681SAndroid Build Coastguard Worker
313*9880d681SAndroid Build Coastguard Workerdef local_load_aligned8bytes : Aligned8Bytes <
314*9880d681SAndroid Build Coastguard Worker  (ops node:$ptr), (local_load node:$ptr)
315*9880d681SAndroid Build Coastguard Worker>;
316*9880d681SAndroid Build Coastguard Worker
317*9880d681SAndroid Build Coastguard Workerdef local_store_aligned8bytes : Aligned8Bytes <
318*9880d681SAndroid Build Coastguard Worker  (ops node:$val, node:$ptr), (local_store node:$val, node:$ptr)
319*9880d681SAndroid Build Coastguard Worker>;
320*9880d681SAndroid Build Coastguard Worker
321*9880d681SAndroid Build Coastguard Workerclass local_binary_atomic_op<SDNode atomic_op> :
322*9880d681SAndroid Build Coastguard Worker  PatFrag<(ops node:$ptr, node:$value),
323*9880d681SAndroid Build Coastguard Worker    (atomic_op node:$ptr, node:$value), [{
324*9880d681SAndroid Build Coastguard Worker  return cast<MemSDNode>(N)->getAddressSpace() == AMDGPUAS::LOCAL_ADDRESS;
325*9880d681SAndroid Build Coastguard Worker}]>;
326*9880d681SAndroid Build Coastguard Worker
327*9880d681SAndroid Build Coastguard Worker
328*9880d681SAndroid Build Coastguard Workerdef atomic_swap_local : local_binary_atomic_op<atomic_swap>;
329*9880d681SAndroid Build Coastguard Workerdef atomic_load_add_local : local_binary_atomic_op<atomic_load_add>;
330*9880d681SAndroid Build Coastguard Workerdef atomic_load_sub_local : local_binary_atomic_op<atomic_load_sub>;
331*9880d681SAndroid Build Coastguard Workerdef atomic_load_and_local : local_binary_atomic_op<atomic_load_and>;
332*9880d681SAndroid Build Coastguard Workerdef atomic_load_or_local : local_binary_atomic_op<atomic_load_or>;
333*9880d681SAndroid Build Coastguard Workerdef atomic_load_xor_local : local_binary_atomic_op<atomic_load_xor>;
334*9880d681SAndroid Build Coastguard Workerdef atomic_load_nand_local : local_binary_atomic_op<atomic_load_nand>;
335*9880d681SAndroid Build Coastguard Workerdef atomic_load_min_local : local_binary_atomic_op<atomic_load_min>;
336*9880d681SAndroid Build Coastguard Workerdef atomic_load_max_local : local_binary_atomic_op<atomic_load_max>;
337*9880d681SAndroid Build Coastguard Workerdef atomic_load_umin_local : local_binary_atomic_op<atomic_load_umin>;
338*9880d681SAndroid Build Coastguard Workerdef atomic_load_umax_local : local_binary_atomic_op<atomic_load_umax>;
339*9880d681SAndroid Build Coastguard Worker
340*9880d681SAndroid Build Coastguard Workerdef mskor_global : PatFrag<(ops node:$val, node:$ptr),
341*9880d681SAndroid Build Coastguard Worker                            (AMDGPUstore_mskor node:$val, node:$ptr), [{
342*9880d681SAndroid Build Coastguard Worker  return cast<MemSDNode>(N)->getAddressSpace() == AMDGPUAS::GLOBAL_ADDRESS;
343*9880d681SAndroid Build Coastguard Worker}]>;
344*9880d681SAndroid Build Coastguard Worker
345*9880d681SAndroid Build Coastguard Workermulticlass AtomicCmpSwapLocal <SDNode cmp_swap_node> {
346*9880d681SAndroid Build Coastguard Worker
347*9880d681SAndroid Build Coastguard Worker  def _32_local : PatFrag <
348*9880d681SAndroid Build Coastguard Worker    (ops node:$ptr, node:$cmp, node:$swap),
349*9880d681SAndroid Build Coastguard Worker    (cmp_swap_node node:$ptr, node:$cmp, node:$swap), [{
350*9880d681SAndroid Build Coastguard Worker      AtomicSDNode *AN = cast<AtomicSDNode>(N);
351*9880d681SAndroid Build Coastguard Worker      return AN->getMemoryVT() == MVT::i32 &&
352*9880d681SAndroid Build Coastguard Worker             AN->getAddressSpace() == AMDGPUAS::LOCAL_ADDRESS;
353*9880d681SAndroid Build Coastguard Worker  }]>;
354*9880d681SAndroid Build Coastguard Worker
355*9880d681SAndroid Build Coastguard Worker  def _64_local : PatFrag<
356*9880d681SAndroid Build Coastguard Worker    (ops node:$ptr, node:$cmp, node:$swap),
357*9880d681SAndroid Build Coastguard Worker    (cmp_swap_node node:$ptr, node:$cmp, node:$swap), [{
358*9880d681SAndroid Build Coastguard Worker      AtomicSDNode *AN = cast<AtomicSDNode>(N);
359*9880d681SAndroid Build Coastguard Worker      return AN->getMemoryVT() == MVT::i64 &&
360*9880d681SAndroid Build Coastguard Worker             AN->getAddressSpace() == AMDGPUAS::LOCAL_ADDRESS;
361*9880d681SAndroid Build Coastguard Worker  }]>;
362*9880d681SAndroid Build Coastguard Worker}
363*9880d681SAndroid Build Coastguard Worker
364*9880d681SAndroid Build Coastguard Workerdefm atomic_cmp_swap : AtomicCmpSwapLocal <atomic_cmp_swap>;
365*9880d681SAndroid Build Coastguard Worker
366*9880d681SAndroid Build Coastguard Workerdef mskor_flat : PatFrag<(ops node:$val, node:$ptr),
367*9880d681SAndroid Build Coastguard Worker                            (AMDGPUstore_mskor node:$val, node:$ptr), [{
368*9880d681SAndroid Build Coastguard Worker  return cast<MemSDNode>(N)->getAddressSpace() == AMDGPUAS::FLAT_ADDRESS;
369*9880d681SAndroid Build Coastguard Worker}]>;
370*9880d681SAndroid Build Coastguard Worker
371*9880d681SAndroid Build Coastguard Workerclass global_binary_atomic_op<SDNode atomic_op> : PatFrag<
372*9880d681SAndroid Build Coastguard Worker  (ops node:$ptr, node:$value),
373*9880d681SAndroid Build Coastguard Worker  (atomic_op node:$ptr, node:$value),
374*9880d681SAndroid Build Coastguard Worker  [{return cast<MemSDNode>(N)->getAddressSpace() == AMDGPUAS::GLOBAL_ADDRESS;}]
375*9880d681SAndroid Build Coastguard Worker>;
376*9880d681SAndroid Build Coastguard Worker
377*9880d681SAndroid Build Coastguard Workerclass flat_binary_atomic_op<SDNode atomic_op> : PatFrag<
378*9880d681SAndroid Build Coastguard Worker  (ops node:$ptr, node:$value),
379*9880d681SAndroid Build Coastguard Worker  (atomic_op node:$ptr, node:$value),
380*9880d681SAndroid Build Coastguard Worker  [{return cast<MemSDNode>(N)->getAddressSpace() == AMDGPUAS::FLAT_ADDRESS;}]
381*9880d681SAndroid Build Coastguard Worker>;
382*9880d681SAndroid Build Coastguard Worker
383*9880d681SAndroid Build Coastguard Workerdef atomic_swap_global : global_binary_atomic_op<atomic_swap>;
384*9880d681SAndroid Build Coastguard Workerdef atomic_add_global : global_binary_atomic_op<atomic_load_add>;
385*9880d681SAndroid Build Coastguard Workerdef atomic_and_global : global_binary_atomic_op<atomic_load_and>;
386*9880d681SAndroid Build Coastguard Workerdef atomic_max_global : global_binary_atomic_op<atomic_load_max>;
387*9880d681SAndroid Build Coastguard Workerdef atomic_min_global : global_binary_atomic_op<atomic_load_min>;
388*9880d681SAndroid Build Coastguard Workerdef atomic_or_global : global_binary_atomic_op<atomic_load_or>;
389*9880d681SAndroid Build Coastguard Workerdef atomic_sub_global : global_binary_atomic_op<atomic_load_sub>;
390*9880d681SAndroid Build Coastguard Workerdef atomic_umax_global : global_binary_atomic_op<atomic_load_umax>;
391*9880d681SAndroid Build Coastguard Workerdef atomic_umin_global : global_binary_atomic_op<atomic_load_umin>;
392*9880d681SAndroid Build Coastguard Workerdef atomic_xor_global : global_binary_atomic_op<atomic_load_xor>;
393*9880d681SAndroid Build Coastguard Worker
394*9880d681SAndroid Build Coastguard Workerdef atomic_cmp_swap_global : global_binary_atomic_op<AMDGPUatomic_cmp_swap>;
395*9880d681SAndroid Build Coastguard Workerdef atomic_cmp_swap_global_nortn : PatFrag<
396*9880d681SAndroid Build Coastguard Worker  (ops node:$ptr, node:$value),
397*9880d681SAndroid Build Coastguard Worker  (atomic_cmp_swap_global node:$ptr, node:$value),
398*9880d681SAndroid Build Coastguard Worker  [{ return SDValue(N, 0).use_empty(); }]
399*9880d681SAndroid Build Coastguard Worker>;
400*9880d681SAndroid Build Coastguard Worker
401*9880d681SAndroid Build Coastguard Workerdef atomic_swap_flat : flat_binary_atomic_op<atomic_swap>;
402*9880d681SAndroid Build Coastguard Workerdef atomic_add_flat : flat_binary_atomic_op<atomic_load_add>;
403*9880d681SAndroid Build Coastguard Workerdef atomic_and_flat : flat_binary_atomic_op<atomic_load_and>;
404*9880d681SAndroid Build Coastguard Workerdef atomic_max_flat : flat_binary_atomic_op<atomic_load_max>;
405*9880d681SAndroid Build Coastguard Workerdef atomic_min_flat : flat_binary_atomic_op<atomic_load_min>;
406*9880d681SAndroid Build Coastguard Workerdef atomic_or_flat : flat_binary_atomic_op<atomic_load_or>;
407*9880d681SAndroid Build Coastguard Workerdef atomic_sub_flat : flat_binary_atomic_op<atomic_load_sub>;
408*9880d681SAndroid Build Coastguard Workerdef atomic_umax_flat : flat_binary_atomic_op<atomic_load_umax>;
409*9880d681SAndroid Build Coastguard Workerdef atomic_umin_flat : flat_binary_atomic_op<atomic_load_umin>;
410*9880d681SAndroid Build Coastguard Workerdef atomic_xor_flat : flat_binary_atomic_op<atomic_load_xor>;
411*9880d681SAndroid Build Coastguard Worker
412*9880d681SAndroid Build Coastguard Workerdef atomic_cmp_swap_flat : flat_binary_atomic_op<AMDGPUatomic_cmp_swap>;
413*9880d681SAndroid Build Coastguard Worker
414*9880d681SAndroid Build Coastguard Worker//===----------------------------------------------------------------------===//
415*9880d681SAndroid Build Coastguard Worker// Misc Pattern Fragments
416*9880d681SAndroid Build Coastguard Worker//===----------------------------------------------------------------------===//
417*9880d681SAndroid Build Coastguard Worker
418*9880d681SAndroid Build Coastguard Workerclass Constants {
419*9880d681SAndroid Build Coastguard Workerint TWO_PI = 0x40c90fdb;
420*9880d681SAndroid Build Coastguard Workerint PI = 0x40490fdb;
421*9880d681SAndroid Build Coastguard Workerint TWO_PI_INV = 0x3e22f983;
422*9880d681SAndroid Build Coastguard Workerint FP_UINT_MAX_PLUS_1 = 0x4f800000;    // 1 << 32 in floating point encoding
423*9880d681SAndroid Build Coastguard Workerint FP32_NEG_ONE = 0xbf800000;
424*9880d681SAndroid Build Coastguard Workerint FP32_ONE = 0x3f800000;
425*9880d681SAndroid Build Coastguard Workerint FP64_ONE = 0x3ff0000000000000;
426*9880d681SAndroid Build Coastguard Worker}
427*9880d681SAndroid Build Coastguard Workerdef CONST : Constants;
428*9880d681SAndroid Build Coastguard Worker
429*9880d681SAndroid Build Coastguard Workerdef FP_ZERO : PatLeaf <
430*9880d681SAndroid Build Coastguard Worker  (fpimm),
431*9880d681SAndroid Build Coastguard Worker  [{return N->getValueAPF().isZero();}]
432*9880d681SAndroid Build Coastguard Worker>;
433*9880d681SAndroid Build Coastguard Worker
434*9880d681SAndroid Build Coastguard Workerdef FP_ONE : PatLeaf <
435*9880d681SAndroid Build Coastguard Worker  (fpimm),
436*9880d681SAndroid Build Coastguard Worker  [{return N->isExactlyValue(1.0);}]
437*9880d681SAndroid Build Coastguard Worker>;
438*9880d681SAndroid Build Coastguard Worker
439*9880d681SAndroid Build Coastguard Workerdef FP_HALF : PatLeaf <
440*9880d681SAndroid Build Coastguard Worker  (fpimm),
441*9880d681SAndroid Build Coastguard Worker  [{return N->isExactlyValue(0.5);}]
442*9880d681SAndroid Build Coastguard Worker>;
443*9880d681SAndroid Build Coastguard Worker
444*9880d681SAndroid Build Coastguard Workerlet isCodeGenOnly = 1, isPseudo = 1 in {
445*9880d681SAndroid Build Coastguard Worker
446*9880d681SAndroid Build Coastguard Workerlet usesCustomInserter = 1  in {
447*9880d681SAndroid Build Coastguard Worker
448*9880d681SAndroid Build Coastguard Workerclass CLAMP <RegisterClass rc> : AMDGPUShaderInst <
449*9880d681SAndroid Build Coastguard Worker  (outs rc:$dst),
450*9880d681SAndroid Build Coastguard Worker  (ins rc:$src0),
451*9880d681SAndroid Build Coastguard Worker  "CLAMP $dst, $src0",
452*9880d681SAndroid Build Coastguard Worker  [(set f32:$dst, (AMDGPUclamp f32:$src0, (f32 FP_ZERO), (f32 FP_ONE)))]
453*9880d681SAndroid Build Coastguard Worker>;
454*9880d681SAndroid Build Coastguard Worker
455*9880d681SAndroid Build Coastguard Workerclass FABS <RegisterClass rc> : AMDGPUShaderInst <
456*9880d681SAndroid Build Coastguard Worker  (outs rc:$dst),
457*9880d681SAndroid Build Coastguard Worker  (ins rc:$src0),
458*9880d681SAndroid Build Coastguard Worker  "FABS $dst, $src0",
459*9880d681SAndroid Build Coastguard Worker  [(set f32:$dst, (fabs f32:$src0))]
460*9880d681SAndroid Build Coastguard Worker>;
461*9880d681SAndroid Build Coastguard Worker
462*9880d681SAndroid Build Coastguard Workerclass FNEG <RegisterClass rc> : AMDGPUShaderInst <
463*9880d681SAndroid Build Coastguard Worker  (outs rc:$dst),
464*9880d681SAndroid Build Coastguard Worker  (ins rc:$src0),
465*9880d681SAndroid Build Coastguard Worker  "FNEG $dst, $src0",
466*9880d681SAndroid Build Coastguard Worker  [(set f32:$dst, (fneg f32:$src0))]
467*9880d681SAndroid Build Coastguard Worker>;
468*9880d681SAndroid Build Coastguard Worker
469*9880d681SAndroid Build Coastguard Worker} // usesCustomInserter = 1
470*9880d681SAndroid Build Coastguard Worker
471*9880d681SAndroid Build Coastguard Workermulticlass RegisterLoadStore <RegisterClass dstClass, Operand addrClass,
472*9880d681SAndroid Build Coastguard Worker                    ComplexPattern addrPat> {
473*9880d681SAndroid Build Coastguard Workerlet UseNamedOperandTable = 1 in {
474*9880d681SAndroid Build Coastguard Worker
475*9880d681SAndroid Build Coastguard Worker  def RegisterLoad : AMDGPUShaderInst <
476*9880d681SAndroid Build Coastguard Worker    (outs dstClass:$dst),
477*9880d681SAndroid Build Coastguard Worker    (ins addrClass:$addr, i32imm:$chan),
478*9880d681SAndroid Build Coastguard Worker    "RegisterLoad $dst, $addr",
479*9880d681SAndroid Build Coastguard Worker    [(set i32:$dst, (AMDGPUregister_load addrPat:$addr, (i32 timm:$chan)))]
480*9880d681SAndroid Build Coastguard Worker  > {
481*9880d681SAndroid Build Coastguard Worker    let isRegisterLoad = 1;
482*9880d681SAndroid Build Coastguard Worker  }
483*9880d681SAndroid Build Coastguard Worker
484*9880d681SAndroid Build Coastguard Worker  def RegisterStore : AMDGPUShaderInst <
485*9880d681SAndroid Build Coastguard Worker    (outs),
486*9880d681SAndroid Build Coastguard Worker    (ins dstClass:$val, addrClass:$addr, i32imm:$chan),
487*9880d681SAndroid Build Coastguard Worker    "RegisterStore $val, $addr",
488*9880d681SAndroid Build Coastguard Worker    [(AMDGPUregister_store i32:$val, addrPat:$addr, (i32 timm:$chan))]
489*9880d681SAndroid Build Coastguard Worker  > {
490*9880d681SAndroid Build Coastguard Worker    let isRegisterStore = 1;
491*9880d681SAndroid Build Coastguard Worker  }
492*9880d681SAndroid Build Coastguard Worker}
493*9880d681SAndroid Build Coastguard Worker}
494*9880d681SAndroid Build Coastguard Worker
495*9880d681SAndroid Build Coastguard Worker} // End isCodeGenOnly = 1, isPseudo = 1
496*9880d681SAndroid Build Coastguard Worker
497*9880d681SAndroid Build Coastguard Worker/* Generic helper patterns for intrinsics */
498*9880d681SAndroid Build Coastguard Worker/* -------------------------------------- */
499*9880d681SAndroid Build Coastguard Worker
500*9880d681SAndroid Build Coastguard Workerclass POW_Common <AMDGPUInst log_ieee, AMDGPUInst exp_ieee, AMDGPUInst mul>
501*9880d681SAndroid Build Coastguard Worker  : Pat <
502*9880d681SAndroid Build Coastguard Worker  (fpow f32:$src0, f32:$src1),
503*9880d681SAndroid Build Coastguard Worker  (exp_ieee (mul f32:$src1, (log_ieee f32:$src0)))
504*9880d681SAndroid Build Coastguard Worker>;
505*9880d681SAndroid Build Coastguard Worker
506*9880d681SAndroid Build Coastguard Worker/* Other helper patterns */
507*9880d681SAndroid Build Coastguard Worker/* --------------------- */
508*9880d681SAndroid Build Coastguard Worker
509*9880d681SAndroid Build Coastguard Worker/* Extract element pattern */
510*9880d681SAndroid Build Coastguard Workerclass Extract_Element <ValueType sub_type, ValueType vec_type, int sub_idx,
511*9880d681SAndroid Build Coastguard Worker                       SubRegIndex sub_reg>
512*9880d681SAndroid Build Coastguard Worker  : Pat<
513*9880d681SAndroid Build Coastguard Worker  (sub_type (extractelt vec_type:$src, sub_idx)),
514*9880d681SAndroid Build Coastguard Worker  (EXTRACT_SUBREG $src, sub_reg)
515*9880d681SAndroid Build Coastguard Worker>;
516*9880d681SAndroid Build Coastguard Worker
517*9880d681SAndroid Build Coastguard Worker/* Insert element pattern */
518*9880d681SAndroid Build Coastguard Workerclass Insert_Element <ValueType elem_type, ValueType vec_type,
519*9880d681SAndroid Build Coastguard Worker                      int sub_idx, SubRegIndex sub_reg>
520*9880d681SAndroid Build Coastguard Worker  : Pat <
521*9880d681SAndroid Build Coastguard Worker  (insertelt vec_type:$vec, elem_type:$elem, sub_idx),
522*9880d681SAndroid Build Coastguard Worker  (INSERT_SUBREG $vec, $elem, sub_reg)
523*9880d681SAndroid Build Coastguard Worker>;
524*9880d681SAndroid Build Coastguard Worker
525*9880d681SAndroid Build Coastguard Worker// XXX: Convert to new syntax and use COPY_TO_REG, once the DFAPacketizer
526*9880d681SAndroid Build Coastguard Worker// can handle COPY instructions.
527*9880d681SAndroid Build Coastguard Worker// bitconvert pattern
528*9880d681SAndroid Build Coastguard Workerclass BitConvert <ValueType dt, ValueType st, RegisterClass rc> : Pat <
529*9880d681SAndroid Build Coastguard Worker  (dt (bitconvert (st rc:$src0))),
530*9880d681SAndroid Build Coastguard Worker  (dt rc:$src0)
531*9880d681SAndroid Build Coastguard Worker>;
532*9880d681SAndroid Build Coastguard Worker
533*9880d681SAndroid Build Coastguard Worker// XXX: Convert to new syntax and use COPY_TO_REG, once the DFAPacketizer
534*9880d681SAndroid Build Coastguard Worker// can handle COPY instructions.
535*9880d681SAndroid Build Coastguard Workerclass DwordAddrPat<ValueType vt, RegisterClass rc> : Pat <
536*9880d681SAndroid Build Coastguard Worker  (vt (AMDGPUdwordaddr (vt rc:$addr))),
537*9880d681SAndroid Build Coastguard Worker  (vt rc:$addr)
538*9880d681SAndroid Build Coastguard Worker>;
539*9880d681SAndroid Build Coastguard Worker
540*9880d681SAndroid Build Coastguard Worker// BFI_INT patterns
541*9880d681SAndroid Build Coastguard Worker
542*9880d681SAndroid Build Coastguard Workermulticlass BFIPatterns <Instruction BFI_INT,
543*9880d681SAndroid Build Coastguard Worker                        Instruction LoadImm32,
544*9880d681SAndroid Build Coastguard Worker                        RegisterClass RC64> {
545*9880d681SAndroid Build Coastguard Worker  // Definition from ISA doc:
546*9880d681SAndroid Build Coastguard Worker  // (y & x) | (z & ~x)
547*9880d681SAndroid Build Coastguard Worker  def : Pat <
548*9880d681SAndroid Build Coastguard Worker    (or (and i32:$y, i32:$x), (and i32:$z, (not i32:$x))),
549*9880d681SAndroid Build Coastguard Worker    (BFI_INT $x, $y, $z)
550*9880d681SAndroid Build Coastguard Worker  >;
551*9880d681SAndroid Build Coastguard Worker
552*9880d681SAndroid Build Coastguard Worker  // SHA-256 Ch function
553*9880d681SAndroid Build Coastguard Worker  // z ^ (x & (y ^ z))
554*9880d681SAndroid Build Coastguard Worker  def : Pat <
555*9880d681SAndroid Build Coastguard Worker    (xor i32:$z, (and i32:$x, (xor i32:$y, i32:$z))),
556*9880d681SAndroid Build Coastguard Worker    (BFI_INT $x, $y, $z)
557*9880d681SAndroid Build Coastguard Worker  >;
558*9880d681SAndroid Build Coastguard Worker
559*9880d681SAndroid Build Coastguard Worker  def : Pat <
560*9880d681SAndroid Build Coastguard Worker    (fcopysign f32:$src0, f32:$src1),
561*9880d681SAndroid Build Coastguard Worker    (BFI_INT (LoadImm32 0x7fffffff), $src0, $src1)
562*9880d681SAndroid Build Coastguard Worker  >;
563*9880d681SAndroid Build Coastguard Worker
564*9880d681SAndroid Build Coastguard Worker  def : Pat <
565*9880d681SAndroid Build Coastguard Worker    (f64 (fcopysign f64:$src0, f64:$src1)),
566*9880d681SAndroid Build Coastguard Worker    (REG_SEQUENCE RC64,
567*9880d681SAndroid Build Coastguard Worker      (i32 (EXTRACT_SUBREG $src0, sub0)), sub0,
568*9880d681SAndroid Build Coastguard Worker      (BFI_INT (LoadImm32 0x7fffffff),
569*9880d681SAndroid Build Coastguard Worker               (i32 (EXTRACT_SUBREG $src0, sub1)),
570*9880d681SAndroid Build Coastguard Worker               (i32 (EXTRACT_SUBREG $src1, sub1))), sub1)
571*9880d681SAndroid Build Coastguard Worker  >;
572*9880d681SAndroid Build Coastguard Worker}
573*9880d681SAndroid Build Coastguard Worker
574*9880d681SAndroid Build Coastguard Worker// SHA-256 Ma patterns
575*9880d681SAndroid Build Coastguard Worker
576*9880d681SAndroid Build Coastguard Worker// ((x & z) | (y & (x | z))) -> BFI_INT (XOR x, y), z, y
577*9880d681SAndroid Build Coastguard Workerclass SHA256MaPattern <Instruction BFI_INT, Instruction XOR> : Pat <
578*9880d681SAndroid Build Coastguard Worker  (or (and i32:$x, i32:$z), (and i32:$y, (or i32:$x, i32:$z))),
579*9880d681SAndroid Build Coastguard Worker  (BFI_INT (XOR i32:$x, i32:$y), i32:$z, i32:$y)
580*9880d681SAndroid Build Coastguard Worker>;
581*9880d681SAndroid Build Coastguard Worker
582*9880d681SAndroid Build Coastguard Worker// Bitfield extract patterns
583*9880d681SAndroid Build Coastguard Worker
584*9880d681SAndroid Build Coastguard Workerdef IMMZeroBasedBitfieldMask : PatLeaf <(imm), [{
585*9880d681SAndroid Build Coastguard Worker  return isMask_32(N->getZExtValue());
586*9880d681SAndroid Build Coastguard Worker}]>;
587*9880d681SAndroid Build Coastguard Worker
588*9880d681SAndroid Build Coastguard Workerdef IMMPopCount : SDNodeXForm<imm, [{
589*9880d681SAndroid Build Coastguard Worker  return CurDAG->getTargetConstant(countPopulation(N->getZExtValue()), SDLoc(N),
590*9880d681SAndroid Build Coastguard Worker                                   MVT::i32);
591*9880d681SAndroid Build Coastguard Worker}]>;
592*9880d681SAndroid Build Coastguard Worker
593*9880d681SAndroid Build Coastguard Workerclass BFEPattern <Instruction BFE, Instruction MOV> : Pat <
594*9880d681SAndroid Build Coastguard Worker  (i32 (and (i32 (srl i32:$src, i32:$rshift)), IMMZeroBasedBitfieldMask:$mask)),
595*9880d681SAndroid Build Coastguard Worker  (BFE $src, $rshift, (MOV (i32 (IMMPopCount $mask))))
596*9880d681SAndroid Build Coastguard Worker>;
597*9880d681SAndroid Build Coastguard Worker
598*9880d681SAndroid Build Coastguard Worker// rotr pattern
599*9880d681SAndroid Build Coastguard Workerclass ROTRPattern <Instruction BIT_ALIGN> : Pat <
600*9880d681SAndroid Build Coastguard Worker  (rotr i32:$src0, i32:$src1),
601*9880d681SAndroid Build Coastguard Worker  (BIT_ALIGN $src0, $src0, $src1)
602*9880d681SAndroid Build Coastguard Worker>;
603*9880d681SAndroid Build Coastguard Worker
604*9880d681SAndroid Build Coastguard Worker// This matches 16 permutations of
605*9880d681SAndroid Build Coastguard Worker// max(min(x, y), min(max(x, y), z))
606*9880d681SAndroid Build Coastguard Workerclass IntMed3Pat<Instruction med3Inst,
607*9880d681SAndroid Build Coastguard Worker                 SDPatternOperator max,
608*9880d681SAndroid Build Coastguard Worker                 SDPatternOperator max_oneuse,
609*9880d681SAndroid Build Coastguard Worker                 SDPatternOperator min_oneuse> : Pat<
610*9880d681SAndroid Build Coastguard Worker  (max (min_oneuse i32:$src0, i32:$src1),
611*9880d681SAndroid Build Coastguard Worker       (min_oneuse (max_oneuse i32:$src0, i32:$src1), i32:$src2)),
612*9880d681SAndroid Build Coastguard Worker  (med3Inst $src0, $src1, $src2)
613*9880d681SAndroid Build Coastguard Worker>;
614*9880d681SAndroid Build Coastguard Worker
615*9880d681SAndroid Build Coastguard Workerlet Properties = [SDNPCommutative, SDNPAssociative] in {
616*9880d681SAndroid Build Coastguard Workerdef smax_oneuse : HasOneUseBinOp<smax>;
617*9880d681SAndroid Build Coastguard Workerdef smin_oneuse : HasOneUseBinOp<smin>;
618*9880d681SAndroid Build Coastguard Workerdef umax_oneuse : HasOneUseBinOp<umax>;
619*9880d681SAndroid Build Coastguard Workerdef umin_oneuse : HasOneUseBinOp<umin>;
620*9880d681SAndroid Build Coastguard Worker} // Properties = [SDNPCommutative, SDNPAssociative]
621*9880d681SAndroid Build Coastguard Worker
622*9880d681SAndroid Build Coastguard Worker
623*9880d681SAndroid Build Coastguard Worker// 24-bit arithmetic patterns
624*9880d681SAndroid Build Coastguard Workerdef umul24 : PatFrag <(ops node:$x, node:$y), (mul node:$x, node:$y)>;
625*9880d681SAndroid Build Coastguard Worker
626*9880d681SAndroid Build Coastguard Worker// Special conversion patterns
627*9880d681SAndroid Build Coastguard Worker
628*9880d681SAndroid Build Coastguard Workerdef cvt_rpi_i32_f32 : PatFrag <
629*9880d681SAndroid Build Coastguard Worker  (ops node:$src),
630*9880d681SAndroid Build Coastguard Worker  (fp_to_sint (ffloor (fadd $src, FP_HALF))),
631*9880d681SAndroid Build Coastguard Worker  [{ (void) N; return TM.Options.NoNaNsFPMath; }]
632*9880d681SAndroid Build Coastguard Worker>;
633*9880d681SAndroid Build Coastguard Worker
634*9880d681SAndroid Build Coastguard Workerdef cvt_flr_i32_f32 : PatFrag <
635*9880d681SAndroid Build Coastguard Worker  (ops node:$src),
636*9880d681SAndroid Build Coastguard Worker  (fp_to_sint (ffloor $src)),
637*9880d681SAndroid Build Coastguard Worker  [{ (void)N; return TM.Options.NoNaNsFPMath; }]
638*9880d681SAndroid Build Coastguard Worker>;
639*9880d681SAndroid Build Coastguard Worker
640*9880d681SAndroid Build Coastguard Workerclass IMad24Pat<Instruction Inst> : Pat <
641*9880d681SAndroid Build Coastguard Worker  (add (AMDGPUmul_i24 i32:$src0, i32:$src1), i32:$src2),
642*9880d681SAndroid Build Coastguard Worker  (Inst $src0, $src1, $src2)
643*9880d681SAndroid Build Coastguard Worker>;
644*9880d681SAndroid Build Coastguard Worker
645*9880d681SAndroid Build Coastguard Workerclass UMad24Pat<Instruction Inst> : Pat <
646*9880d681SAndroid Build Coastguard Worker  (add (AMDGPUmul_u24 i32:$src0, i32:$src1), i32:$src2),
647*9880d681SAndroid Build Coastguard Worker  (Inst $src0, $src1, $src2)
648*9880d681SAndroid Build Coastguard Worker>;
649*9880d681SAndroid Build Coastguard Worker
650*9880d681SAndroid Build Coastguard Workerclass RcpPat<Instruction RcpInst, ValueType vt> : Pat <
651*9880d681SAndroid Build Coastguard Worker  (fdiv FP_ONE, vt:$src),
652*9880d681SAndroid Build Coastguard Worker  (RcpInst $src)
653*9880d681SAndroid Build Coastguard Worker>;
654*9880d681SAndroid Build Coastguard Worker
655*9880d681SAndroid Build Coastguard Workerclass RsqPat<Instruction RsqInst, ValueType vt> : Pat <
656*9880d681SAndroid Build Coastguard Worker  (AMDGPUrcp (fsqrt vt:$src)),
657*9880d681SAndroid Build Coastguard Worker  (RsqInst $src)
658*9880d681SAndroid Build Coastguard Worker>;
659*9880d681SAndroid Build Coastguard Worker
660*9880d681SAndroid Build Coastguard Workerinclude "R600Instructions.td"
661*9880d681SAndroid Build Coastguard Workerinclude "R700Instructions.td"
662*9880d681SAndroid Build Coastguard Workerinclude "EvergreenInstructions.td"
663*9880d681SAndroid Build Coastguard Workerinclude "CaymanInstructions.td"
664*9880d681SAndroid Build Coastguard Worker
665*9880d681SAndroid Build Coastguard Workerinclude "SIInstrInfo.td"
666*9880d681SAndroid Build Coastguard Worker
667