xref: /aosp_15_r20/external/llvm/lib/Target/AMDGPU/GCNHazardRecognizer.cpp (revision 9880d6810fe72a1726cb53787c6711e909410d58)
1*9880d681SAndroid Build Coastguard Worker //===-- GCNHazardRecognizers.cpp - GCN Hazard Recognizer Impls ------------===//
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 implements hazard recognizers for scheduling on GCN processors.
11*9880d681SAndroid Build Coastguard Worker //
12*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
13*9880d681SAndroid Build Coastguard Worker 
14*9880d681SAndroid Build Coastguard Worker #include "GCNHazardRecognizer.h"
15*9880d681SAndroid Build Coastguard Worker #include "AMDGPUSubtarget.h"
16*9880d681SAndroid Build Coastguard Worker #include "SIInstrInfo.h"
17*9880d681SAndroid Build Coastguard Worker #include "llvm/CodeGen/ScheduleDAG.h"
18*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/Debug.h"
19*9880d681SAndroid Build Coastguard Worker 
20*9880d681SAndroid Build Coastguard Worker using namespace llvm;
21*9880d681SAndroid Build Coastguard Worker 
22*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
23*9880d681SAndroid Build Coastguard Worker // Hazard Recoginizer Implementation
24*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
25*9880d681SAndroid Build Coastguard Worker 
GCNHazardRecognizer(const MachineFunction & MF)26*9880d681SAndroid Build Coastguard Worker GCNHazardRecognizer::GCNHazardRecognizer(const MachineFunction &MF) :
27*9880d681SAndroid Build Coastguard Worker   CurrCycleInstr(nullptr),
28*9880d681SAndroid Build Coastguard Worker   MF(MF),
29*9880d681SAndroid Build Coastguard Worker   ST(MF.getSubtarget<SISubtarget>()) {
30*9880d681SAndroid Build Coastguard Worker   MaxLookAhead = 5;
31*9880d681SAndroid Build Coastguard Worker }
32*9880d681SAndroid Build Coastguard Worker 
EmitInstruction(SUnit * SU)33*9880d681SAndroid Build Coastguard Worker void GCNHazardRecognizer::EmitInstruction(SUnit *SU) {
34*9880d681SAndroid Build Coastguard Worker   EmitInstruction(SU->getInstr());
35*9880d681SAndroid Build Coastguard Worker }
36*9880d681SAndroid Build Coastguard Worker 
EmitInstruction(MachineInstr * MI)37*9880d681SAndroid Build Coastguard Worker void GCNHazardRecognizer::EmitInstruction(MachineInstr *MI) {
38*9880d681SAndroid Build Coastguard Worker   CurrCycleInstr = MI;
39*9880d681SAndroid Build Coastguard Worker }
40*9880d681SAndroid Build Coastguard Worker 
41*9880d681SAndroid Build Coastguard Worker ScheduleHazardRecognizer::HazardType
getHazardType(SUnit * SU,int Stalls)42*9880d681SAndroid Build Coastguard Worker GCNHazardRecognizer::getHazardType(SUnit *SU, int Stalls) {
43*9880d681SAndroid Build Coastguard Worker   MachineInstr *MI = SU->getInstr();
44*9880d681SAndroid Build Coastguard Worker 
45*9880d681SAndroid Build Coastguard Worker   if (SIInstrInfo::isSMRD(*MI) && checkSMRDHazards(MI) > 0)
46*9880d681SAndroid Build Coastguard Worker     return NoopHazard;
47*9880d681SAndroid Build Coastguard Worker 
48*9880d681SAndroid Build Coastguard Worker   if (SIInstrInfo::isVMEM(*MI) && checkVMEMHazards(MI) > 0)
49*9880d681SAndroid Build Coastguard Worker     return NoopHazard;
50*9880d681SAndroid Build Coastguard Worker 
51*9880d681SAndroid Build Coastguard Worker   if (SIInstrInfo::isDPP(*MI) && checkDPPHazards(MI) > 0)
52*9880d681SAndroid Build Coastguard Worker     return NoopHazard;
53*9880d681SAndroid Build Coastguard Worker 
54*9880d681SAndroid Build Coastguard Worker   return NoHazard;
55*9880d681SAndroid Build Coastguard Worker }
56*9880d681SAndroid Build Coastguard Worker 
PreEmitNoops(SUnit * SU)57*9880d681SAndroid Build Coastguard Worker unsigned GCNHazardRecognizer::PreEmitNoops(SUnit *SU) {
58*9880d681SAndroid Build Coastguard Worker   return PreEmitNoops(SU->getInstr());
59*9880d681SAndroid Build Coastguard Worker }
60*9880d681SAndroid Build Coastguard Worker 
PreEmitNoops(MachineInstr * MI)61*9880d681SAndroid Build Coastguard Worker unsigned GCNHazardRecognizer::PreEmitNoops(MachineInstr *MI) {
62*9880d681SAndroid Build Coastguard Worker   if (SIInstrInfo::isSMRD(*MI))
63*9880d681SAndroid Build Coastguard Worker     return std::max(0, checkSMRDHazards(MI));
64*9880d681SAndroid Build Coastguard Worker 
65*9880d681SAndroid Build Coastguard Worker   if (SIInstrInfo::isVMEM(*MI))
66*9880d681SAndroid Build Coastguard Worker     return std::max(0, checkVMEMHazards(MI));
67*9880d681SAndroid Build Coastguard Worker 
68*9880d681SAndroid Build Coastguard Worker   if (SIInstrInfo::isDPP(*MI))
69*9880d681SAndroid Build Coastguard Worker     return std::max(0, checkDPPHazards(MI));
70*9880d681SAndroid Build Coastguard Worker 
71*9880d681SAndroid Build Coastguard Worker   return 0;
72*9880d681SAndroid Build Coastguard Worker }
73*9880d681SAndroid Build Coastguard Worker 
EmitNoop()74*9880d681SAndroid Build Coastguard Worker void GCNHazardRecognizer::EmitNoop() {
75*9880d681SAndroid Build Coastguard Worker   EmittedInstrs.push_front(nullptr);
76*9880d681SAndroid Build Coastguard Worker }
77*9880d681SAndroid Build Coastguard Worker 
AdvanceCycle()78*9880d681SAndroid Build Coastguard Worker void GCNHazardRecognizer::AdvanceCycle() {
79*9880d681SAndroid Build Coastguard Worker 
80*9880d681SAndroid Build Coastguard Worker   // When the scheduler detects a stall, it will call AdvanceCycle() without
81*9880d681SAndroid Build Coastguard Worker   // emitting any instructions.
82*9880d681SAndroid Build Coastguard Worker   if (!CurrCycleInstr)
83*9880d681SAndroid Build Coastguard Worker     return;
84*9880d681SAndroid Build Coastguard Worker 
85*9880d681SAndroid Build Coastguard Worker   const SIInstrInfo *TII = ST.getInstrInfo();
86*9880d681SAndroid Build Coastguard Worker   unsigned NumWaitStates = TII->getNumWaitStates(*CurrCycleInstr);
87*9880d681SAndroid Build Coastguard Worker 
88*9880d681SAndroid Build Coastguard Worker   // Keep track of emitted instructions
89*9880d681SAndroid Build Coastguard Worker   EmittedInstrs.push_front(CurrCycleInstr);
90*9880d681SAndroid Build Coastguard Worker 
91*9880d681SAndroid Build Coastguard Worker   // Add a nullptr for each additional wait state after the first.  Make sure
92*9880d681SAndroid Build Coastguard Worker   // not to add more than getMaxLookAhead() items to the list, since we
93*9880d681SAndroid Build Coastguard Worker   // truncate the list to that size right after this loop.
94*9880d681SAndroid Build Coastguard Worker   for (unsigned i = 1, e = std::min(NumWaitStates, getMaxLookAhead());
95*9880d681SAndroid Build Coastguard Worker        i < e; ++i) {
96*9880d681SAndroid Build Coastguard Worker     EmittedInstrs.push_front(nullptr);
97*9880d681SAndroid Build Coastguard Worker   }
98*9880d681SAndroid Build Coastguard Worker 
99*9880d681SAndroid Build Coastguard Worker   // getMaxLookahead() is the largest number of wait states we will ever need
100*9880d681SAndroid Build Coastguard Worker   // to insert, so there is no point in keeping track of more than that many
101*9880d681SAndroid Build Coastguard Worker   // wait states.
102*9880d681SAndroid Build Coastguard Worker   EmittedInstrs.resize(getMaxLookAhead());
103*9880d681SAndroid Build Coastguard Worker 
104*9880d681SAndroid Build Coastguard Worker   CurrCycleInstr = nullptr;
105*9880d681SAndroid Build Coastguard Worker }
106*9880d681SAndroid Build Coastguard Worker 
RecedeCycle()107*9880d681SAndroid Build Coastguard Worker void GCNHazardRecognizer::RecedeCycle() {
108*9880d681SAndroid Build Coastguard Worker   llvm_unreachable("hazard recognizer does not support bottom-up scheduling.");
109*9880d681SAndroid Build Coastguard Worker }
110*9880d681SAndroid Build Coastguard Worker 
111*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
112*9880d681SAndroid Build Coastguard Worker // Helper Functions
113*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
114*9880d681SAndroid Build Coastguard Worker 
getWaitStatesSinceDef(unsigned Reg,function_ref<bool (MachineInstr *)> IsHazardDef)115*9880d681SAndroid Build Coastguard Worker int GCNHazardRecognizer::getWaitStatesSinceDef(
116*9880d681SAndroid Build Coastguard Worker     unsigned Reg, function_ref<bool(MachineInstr *)> IsHazardDef) {
117*9880d681SAndroid Build Coastguard Worker   const SIRegisterInfo *TRI = ST.getRegisterInfo();
118*9880d681SAndroid Build Coastguard Worker 
119*9880d681SAndroid Build Coastguard Worker   int WaitStates = -1;
120*9880d681SAndroid Build Coastguard Worker   for (MachineInstr *MI : EmittedInstrs) {
121*9880d681SAndroid Build Coastguard Worker     ++WaitStates;
122*9880d681SAndroid Build Coastguard Worker     if (!MI || !IsHazardDef(MI))
123*9880d681SAndroid Build Coastguard Worker       continue;
124*9880d681SAndroid Build Coastguard Worker     if (MI->modifiesRegister(Reg, TRI))
125*9880d681SAndroid Build Coastguard Worker       return WaitStates;
126*9880d681SAndroid Build Coastguard Worker   }
127*9880d681SAndroid Build Coastguard Worker   return std::numeric_limits<int>::max();
128*9880d681SAndroid Build Coastguard Worker }
129*9880d681SAndroid Build Coastguard Worker 
130*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
131*9880d681SAndroid Build Coastguard Worker // No-op Hazard Detection
132*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
133*9880d681SAndroid Build Coastguard Worker 
addRegsToSet(iterator_range<MachineInstr::const_mop_iterator> Ops,std::set<unsigned> & Set)134*9880d681SAndroid Build Coastguard Worker static void addRegsToSet(iterator_range<MachineInstr::const_mop_iterator> Ops,
135*9880d681SAndroid Build Coastguard Worker                          std::set<unsigned> &Set) {
136*9880d681SAndroid Build Coastguard Worker   for (const MachineOperand &Op : Ops) {
137*9880d681SAndroid Build Coastguard Worker     if (Op.isReg())
138*9880d681SAndroid Build Coastguard Worker       Set.insert(Op.getReg());
139*9880d681SAndroid Build Coastguard Worker   }
140*9880d681SAndroid Build Coastguard Worker }
141*9880d681SAndroid Build Coastguard Worker 
checkSMEMSoftClauseHazards(MachineInstr * SMEM)142*9880d681SAndroid Build Coastguard Worker int GCNHazardRecognizer::checkSMEMSoftClauseHazards(MachineInstr *SMEM) {
143*9880d681SAndroid Build Coastguard Worker   // SMEM soft clause are only present on VI+
144*9880d681SAndroid Build Coastguard Worker   if (ST.getGeneration() < SISubtarget::VOLCANIC_ISLANDS)
145*9880d681SAndroid Build Coastguard Worker     return 0;
146*9880d681SAndroid Build Coastguard Worker 
147*9880d681SAndroid Build Coastguard Worker   // A soft-clause is any group of consecutive SMEM instructions.  The
148*9880d681SAndroid Build Coastguard Worker   // instructions in this group may return out of order and/or may be
149*9880d681SAndroid Build Coastguard Worker   // replayed (i.e. the same instruction issued more than once).
150*9880d681SAndroid Build Coastguard Worker   //
151*9880d681SAndroid Build Coastguard Worker   // In order to handle these situations correctly we need to make sure
152*9880d681SAndroid Build Coastguard Worker   // that when a clause has more than one instruction, no instruction in the
153*9880d681SAndroid Build Coastguard Worker   // clause writes to a register that is read another instruction in the clause
154*9880d681SAndroid Build Coastguard Worker   // (including itself). If we encounter this situaion, we need to break the
155*9880d681SAndroid Build Coastguard Worker   // clause by inserting a non SMEM instruction.
156*9880d681SAndroid Build Coastguard Worker 
157*9880d681SAndroid Build Coastguard Worker   std::set<unsigned> ClauseDefs;
158*9880d681SAndroid Build Coastguard Worker   std::set<unsigned> ClauseUses;
159*9880d681SAndroid Build Coastguard Worker 
160*9880d681SAndroid Build Coastguard Worker   for (MachineInstr *MI : EmittedInstrs) {
161*9880d681SAndroid Build Coastguard Worker 
162*9880d681SAndroid Build Coastguard Worker     // When we hit a non-SMEM instruction then we have passed the start of the
163*9880d681SAndroid Build Coastguard Worker     // clause and we can stop.
164*9880d681SAndroid Build Coastguard Worker     if (!MI || !SIInstrInfo::isSMRD(*MI))
165*9880d681SAndroid Build Coastguard Worker       break;
166*9880d681SAndroid Build Coastguard Worker 
167*9880d681SAndroid Build Coastguard Worker     addRegsToSet(MI->defs(), ClauseDefs);
168*9880d681SAndroid Build Coastguard Worker     addRegsToSet(MI->uses(), ClauseUses);
169*9880d681SAndroid Build Coastguard Worker   }
170*9880d681SAndroid Build Coastguard Worker 
171*9880d681SAndroid Build Coastguard Worker   if (ClauseDefs.empty())
172*9880d681SAndroid Build Coastguard Worker     return 0;
173*9880d681SAndroid Build Coastguard Worker 
174*9880d681SAndroid Build Coastguard Worker   // FIXME: When we support stores, we need to make sure not to put loads and
175*9880d681SAndroid Build Coastguard Worker   // stores in the same clause if they use the same address.  For now, just
176*9880d681SAndroid Build Coastguard Worker   // start a new clause whenever we see a store.
177*9880d681SAndroid Build Coastguard Worker   if (SMEM->mayStore())
178*9880d681SAndroid Build Coastguard Worker     return 1;
179*9880d681SAndroid Build Coastguard Worker 
180*9880d681SAndroid Build Coastguard Worker   addRegsToSet(SMEM->defs(), ClauseDefs);
181*9880d681SAndroid Build Coastguard Worker   addRegsToSet(SMEM->uses(), ClauseUses);
182*9880d681SAndroid Build Coastguard Worker 
183*9880d681SAndroid Build Coastguard Worker   std::vector<unsigned> Result(std::max(ClauseDefs.size(), ClauseUses.size()));
184*9880d681SAndroid Build Coastguard Worker   std::vector<unsigned>::iterator End;
185*9880d681SAndroid Build Coastguard Worker 
186*9880d681SAndroid Build Coastguard Worker   End = std::set_intersection(ClauseDefs.begin(), ClauseDefs.end(),
187*9880d681SAndroid Build Coastguard Worker                               ClauseUses.begin(), ClauseUses.end(), Result.begin());
188*9880d681SAndroid Build Coastguard Worker 
189*9880d681SAndroid Build Coastguard Worker   // If the set of defs and uses intersect then we cannot add this instruction
190*9880d681SAndroid Build Coastguard Worker   // to the clause, so we have a hazard.
191*9880d681SAndroid Build Coastguard Worker   if (End != Result.begin())
192*9880d681SAndroid Build Coastguard Worker     return 1;
193*9880d681SAndroid Build Coastguard Worker 
194*9880d681SAndroid Build Coastguard Worker   return 0;
195*9880d681SAndroid Build Coastguard Worker }
196*9880d681SAndroid Build Coastguard Worker 
checkSMRDHazards(MachineInstr * SMRD)197*9880d681SAndroid Build Coastguard Worker int GCNHazardRecognizer::checkSMRDHazards(MachineInstr *SMRD) {
198*9880d681SAndroid Build Coastguard Worker   const SISubtarget &ST = MF.getSubtarget<SISubtarget>();
199*9880d681SAndroid Build Coastguard Worker   const SIInstrInfo *TII = ST.getInstrInfo();
200*9880d681SAndroid Build Coastguard Worker   int WaitStatesNeeded = 0;
201*9880d681SAndroid Build Coastguard Worker 
202*9880d681SAndroid Build Coastguard Worker   WaitStatesNeeded = checkSMEMSoftClauseHazards(SMRD);
203*9880d681SAndroid Build Coastguard Worker 
204*9880d681SAndroid Build Coastguard Worker   // This SMRD hazard only affects SI.
205*9880d681SAndroid Build Coastguard Worker   if (ST.getGeneration() != SISubtarget::SOUTHERN_ISLANDS)
206*9880d681SAndroid Build Coastguard Worker     return WaitStatesNeeded;
207*9880d681SAndroid Build Coastguard Worker 
208*9880d681SAndroid Build Coastguard Worker   // A read of an SGPR by SMRD instruction requires 4 wait states when the
209*9880d681SAndroid Build Coastguard Worker   // SGPR was written by a VALU instruction.
210*9880d681SAndroid Build Coastguard Worker   int SmrdSgprWaitStates = 4;
211*9880d681SAndroid Build Coastguard Worker   auto IsHazardDefFn = [TII] (MachineInstr *MI) { return TII->isVALU(*MI); };
212*9880d681SAndroid Build Coastguard Worker 
213*9880d681SAndroid Build Coastguard Worker   for (const MachineOperand &Use : SMRD->uses()) {
214*9880d681SAndroid Build Coastguard Worker     if (!Use.isReg())
215*9880d681SAndroid Build Coastguard Worker       continue;
216*9880d681SAndroid Build Coastguard Worker     int WaitStatesNeededForUse =
217*9880d681SAndroid Build Coastguard Worker         SmrdSgprWaitStates - getWaitStatesSinceDef(Use.getReg(), IsHazardDefFn);
218*9880d681SAndroid Build Coastguard Worker     WaitStatesNeeded = std::max(WaitStatesNeeded, WaitStatesNeededForUse);
219*9880d681SAndroid Build Coastguard Worker   }
220*9880d681SAndroid Build Coastguard Worker   return WaitStatesNeeded;
221*9880d681SAndroid Build Coastguard Worker }
222*9880d681SAndroid Build Coastguard Worker 
checkVMEMHazards(MachineInstr * VMEM)223*9880d681SAndroid Build Coastguard Worker int GCNHazardRecognizer::checkVMEMHazards(MachineInstr* VMEM) {
224*9880d681SAndroid Build Coastguard Worker   const SIInstrInfo *TII = ST.getInstrInfo();
225*9880d681SAndroid Build Coastguard Worker 
226*9880d681SAndroid Build Coastguard Worker   if (ST.getGeneration() < SISubtarget::VOLCANIC_ISLANDS)
227*9880d681SAndroid Build Coastguard Worker     return 0;
228*9880d681SAndroid Build Coastguard Worker 
229*9880d681SAndroid Build Coastguard Worker   const SIRegisterInfo &TRI = TII->getRegisterInfo();
230*9880d681SAndroid Build Coastguard Worker 
231*9880d681SAndroid Build Coastguard Worker   // A read of an SGPR by a VMEM instruction requires 5 wait states when the
232*9880d681SAndroid Build Coastguard Worker   // SGPR was written by a VALU Instruction.
233*9880d681SAndroid Build Coastguard Worker   int VmemSgprWaitStates = 5;
234*9880d681SAndroid Build Coastguard Worker   int WaitStatesNeeded = 0;
235*9880d681SAndroid Build Coastguard Worker   auto IsHazardDefFn = [TII] (MachineInstr *MI) { return TII->isVALU(*MI); };
236*9880d681SAndroid Build Coastguard Worker 
237*9880d681SAndroid Build Coastguard Worker   for (const MachineOperand &Use : VMEM->uses()) {
238*9880d681SAndroid Build Coastguard Worker     if (!Use.isReg() || TRI.isVGPR(MF.getRegInfo(), Use.getReg()))
239*9880d681SAndroid Build Coastguard Worker       continue;
240*9880d681SAndroid Build Coastguard Worker 
241*9880d681SAndroid Build Coastguard Worker     int WaitStatesNeededForUse =
242*9880d681SAndroid Build Coastguard Worker         VmemSgprWaitStates - getWaitStatesSinceDef(Use.getReg(), IsHazardDefFn);
243*9880d681SAndroid Build Coastguard Worker     WaitStatesNeeded = std::max(WaitStatesNeeded, WaitStatesNeededForUse);
244*9880d681SAndroid Build Coastguard Worker   }
245*9880d681SAndroid Build Coastguard Worker   return WaitStatesNeeded;
246*9880d681SAndroid Build Coastguard Worker }
247*9880d681SAndroid Build Coastguard Worker 
checkDPPHazards(MachineInstr * DPP)248*9880d681SAndroid Build Coastguard Worker int GCNHazardRecognizer::checkDPPHazards(MachineInstr *DPP) {
249*9880d681SAndroid Build Coastguard Worker   const SIRegisterInfo *TRI = ST.getRegisterInfo();
250*9880d681SAndroid Build Coastguard Worker 
251*9880d681SAndroid Build Coastguard Worker   // Check for DPP VGPR read after VALU VGPR write.
252*9880d681SAndroid Build Coastguard Worker   int DppVgprWaitStates = 2;
253*9880d681SAndroid Build Coastguard Worker   int WaitStatesNeeded = 0;
254*9880d681SAndroid Build Coastguard Worker 
255*9880d681SAndroid Build Coastguard Worker   for (const MachineOperand &Use : DPP->uses()) {
256*9880d681SAndroid Build Coastguard Worker     if (!Use.isReg() || !TRI->isVGPR(MF.getRegInfo(), Use.getReg()))
257*9880d681SAndroid Build Coastguard Worker       continue;
258*9880d681SAndroid Build Coastguard Worker     int WaitStatesNeededForUse =
259*9880d681SAndroid Build Coastguard Worker         DppVgprWaitStates - getWaitStatesSinceDef(Use.getReg());
260*9880d681SAndroid Build Coastguard Worker     WaitStatesNeeded = std::max(WaitStatesNeeded, WaitStatesNeededForUse);
261*9880d681SAndroid Build Coastguard Worker   }
262*9880d681SAndroid Build Coastguard Worker 
263*9880d681SAndroid Build Coastguard Worker   return WaitStatesNeeded;
264*9880d681SAndroid Build Coastguard Worker }
265