• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 //===-- R600EmitClauseMarkers.cpp - Emit CF_ALU ---------------------------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 //
9 /// \file
10 /// Add CF_ALU. R600 Alu instructions are grouped in clause which can hold
11 /// 128 Alu instructions ; these instructions can access up to 4 prefetched
12 /// 4 lines of 16 registers from constant buffers. Such ALU clauses are
13 /// initiated by CF_ALU instructions.
14 //===----------------------------------------------------------------------===//
15 
16 #include "AMDGPU.h"
17 #include "AMDGPUSubtarget.h"
18 #include "R600Defines.h"
19 #include "R600InstrInfo.h"
20 #include "R600RegisterInfo.h"
21 #include "MCTargetDesc/AMDGPUMCTargetDesc.h"
22 #include "llvm/ADT/SmallVector.h"
23 #include "llvm/ADT/StringRef.h"
24 #include "llvm/CodeGen/MachineBasicBlock.h"
25 #include "llvm/CodeGen/MachineFunction.h"
26 #include "llvm/CodeGen/MachineFunctionPass.h"
27 #include "llvm/CodeGen/MachineInstr.h"
28 #include "llvm/CodeGen/MachineInstrBuilder.h"
29 #include "llvm/CodeGen/MachineOperand.h"
30 #include "llvm/Pass.h"
31 #include "llvm/Support/ErrorHandling.h"
32 #include <cassert>
33 #include <cstdint>
34 #include <utility>
35 #include <vector>
36 
37 using namespace llvm;
38 
39 namespace llvm {
40 
41   void initializeR600EmitClauseMarkersPass(PassRegistry&);
42 
43 } // end namespace llvm
44 
45 namespace {
46 
47 class R600EmitClauseMarkers : public MachineFunctionPass {
48 private:
49   const R600InstrInfo *TII = nullptr;
50   int Address = 0;
51 
OccupiedDwords(MachineInstr & MI) const52   unsigned OccupiedDwords(MachineInstr &MI) const {
53     switch (MI.getOpcode()) {
54     case R600::INTERP_PAIR_XY:
55     case R600::INTERP_PAIR_ZW:
56     case R600::INTERP_VEC_LOAD:
57     case R600::DOT_4:
58       return 4;
59     case R600::KILL:
60       return 0;
61     default:
62       break;
63     }
64 
65     // These will be expanded to two ALU instructions in the
66     // ExpandSpecialInstructions pass.
67     if (TII->isLDSRetInstr(MI.getOpcode()))
68       return 2;
69 
70     if (TII->isVector(MI) || TII->isCubeOp(MI.getOpcode()) ||
71         TII->isReductionOp(MI.getOpcode()))
72       return 4;
73 
74     unsigned NumLiteral = 0;
75     for (MachineInstr::mop_iterator It = MI.operands_begin(),
76                                     E = MI.operands_end();
77          It != E; ++It) {
78       MachineOperand &MO = *It;
79       if (MO.isReg() && MO.getReg() == R600::ALU_LITERAL_X)
80         ++NumLiteral;
81     }
82     return 1 + NumLiteral;
83   }
84 
isALU(const MachineInstr & MI) const85   bool isALU(const MachineInstr &MI) const {
86     if (TII->isALUInstr(MI.getOpcode()))
87       return true;
88     if (TII->isVector(MI) || TII->isCubeOp(MI.getOpcode()))
89       return true;
90     switch (MI.getOpcode()) {
91     case R600::PRED_X:
92     case R600::INTERP_PAIR_XY:
93     case R600::INTERP_PAIR_ZW:
94     case R600::INTERP_VEC_LOAD:
95     case R600::COPY:
96     case R600::DOT_4:
97       return true;
98     default:
99       return false;
100     }
101   }
102 
IsTrivialInst(MachineInstr & MI) const103   bool IsTrivialInst(MachineInstr &MI) const {
104     switch (MI.getOpcode()) {
105     case R600::KILL:
106     case R600::RETURN:
107     case R600::IMPLICIT_DEF:
108       return true;
109     default:
110       return false;
111     }
112   }
113 
getAccessedBankLine(unsigned Sel) const114   std::pair<unsigned, unsigned> getAccessedBankLine(unsigned Sel) const {
115     // Sel is (512 + (kc_bank << 12) + ConstIndex) << 2
116     // (See also R600ISelLowering.cpp)
117     // ConstIndex value is in [0, 4095];
118     return std::pair<unsigned, unsigned>(
119         ((Sel >> 2) - 512) >> 12, // KC_BANK
120         // Line Number of ConstIndex
121         // A line contains 16 constant registers however KCX bank can lock
122         // two line at the same time ; thus we want to get an even line number.
123         // Line number can be retrieved with (>>4), using (>>5) <<1 generates
124         // an even number.
125         ((((Sel >> 2) - 512) & 4095) >> 5) << 1);
126   }
127 
128   bool
SubstituteKCacheBank(MachineInstr & MI,std::vector<std::pair<unsigned,unsigned>> & CachedConsts,bool UpdateInstr=true) const129   SubstituteKCacheBank(MachineInstr &MI,
130                        std::vector<std::pair<unsigned, unsigned>> &CachedConsts,
131                        bool UpdateInstr = true) const {
132     std::vector<std::pair<unsigned, unsigned>> UsedKCache;
133 
134     if (!TII->isALUInstr(MI.getOpcode()) && MI.getOpcode() != R600::DOT_4)
135       return true;
136 
137     const SmallVectorImpl<std::pair<MachineOperand *, int64_t>> &Consts =
138         TII->getSrcs(MI);
139     assert(
140         (TII->isALUInstr(MI.getOpcode()) || MI.getOpcode() == R600::DOT_4) &&
141         "Can't assign Const");
142     for (unsigned i = 0, n = Consts.size(); i < n; ++i) {
143       if (Consts[i].first->getReg() != R600::ALU_CONST)
144         continue;
145       unsigned Sel = Consts[i].second;
146       unsigned Chan = Sel & 3, Index = ((Sel >> 2) - 512) & 31;
147       unsigned KCacheIndex = Index * 4 + Chan;
148       const std::pair<unsigned, unsigned> &BankLine = getAccessedBankLine(Sel);
149       if (CachedConsts.empty()) {
150         CachedConsts.push_back(BankLine);
151         UsedKCache.push_back(std::pair<unsigned, unsigned>(0, KCacheIndex));
152         continue;
153       }
154       if (CachedConsts[0] == BankLine) {
155         UsedKCache.push_back(std::pair<unsigned, unsigned>(0, KCacheIndex));
156         continue;
157       }
158       if (CachedConsts.size() == 1) {
159         CachedConsts.push_back(BankLine);
160         UsedKCache.push_back(std::pair<unsigned, unsigned>(1, KCacheIndex));
161         continue;
162       }
163       if (CachedConsts[1] == BankLine) {
164         UsedKCache.push_back(std::pair<unsigned, unsigned>(1, KCacheIndex));
165         continue;
166       }
167       return false;
168     }
169 
170     if (!UpdateInstr)
171       return true;
172 
173     for (unsigned i = 0, j = 0, n = Consts.size(); i < n; ++i) {
174       if (Consts[i].first->getReg() != R600::ALU_CONST)
175         continue;
176       switch(UsedKCache[j].first) {
177       case 0:
178         Consts[i].first->setReg(
179             R600::R600_KC0RegClass.getRegister(UsedKCache[j].second));
180         break;
181       case 1:
182         Consts[i].first->setReg(
183             R600::R600_KC1RegClass.getRegister(UsedKCache[j].second));
184         break;
185       default:
186         llvm_unreachable("Wrong Cache Line");
187       }
188       j++;
189     }
190     return true;
191   }
192 
canClauseLocalKillFitInClause(unsigned AluInstCount,std::vector<std::pair<unsigned,unsigned>> KCacheBanks,MachineBasicBlock::iterator Def,MachineBasicBlock::iterator BBEnd)193   bool canClauseLocalKillFitInClause(
194                         unsigned AluInstCount,
195                         std::vector<std::pair<unsigned, unsigned>> KCacheBanks,
196                         MachineBasicBlock::iterator Def,
197                         MachineBasicBlock::iterator BBEnd) {
198     const R600RegisterInfo &TRI = TII->getRegisterInfo();
199     //TODO: change this to defs?
200     for (MachineInstr::const_mop_iterator
201            MOI = Def->operands_begin(),
202            MOE = Def->operands_end(); MOI != MOE; ++MOI) {
203       if (!MOI->isReg() || !MOI->isDef() ||
204           TRI.isPhysRegLiveAcrossClauses(MOI->getReg()))
205         continue;
206 
207       // Def defines a clause local register, so check that its use will fit
208       // in the clause.
209       unsigned LastUseCount = 0;
210       for (MachineBasicBlock::iterator UseI = Def; UseI != BBEnd; ++UseI) {
211         AluInstCount += OccupiedDwords(*UseI);
212         // Make sure we won't need to end the clause due to KCache limitations.
213         if (!SubstituteKCacheBank(*UseI, KCacheBanks, false))
214           return false;
215 
216         // We have reached the maximum instruction limit before finding the
217         // use that kills this register, so we cannot use this def in the
218         // current clause.
219         if (AluInstCount >= TII->getMaxAlusPerClause())
220           return false;
221 
222         // TODO: Is this true? kill flag appears to work OK below
223         // Register kill flags have been cleared by the time we get to this
224         // pass, but it is safe to assume that all uses of this register
225         // occur in the same basic block as its definition, because
226         // it is illegal for the scheduler to schedule them in
227         // different blocks.
228         if (UseI->readsRegister(MOI->getReg(), &TRI))
229           LastUseCount = AluInstCount;
230 
231         // Exit early if the current use kills the register
232         if (UseI != Def && UseI->killsRegister(MOI->getReg(), &TRI))
233           break;
234       }
235       if (LastUseCount)
236         return LastUseCount <= TII->getMaxAlusPerClause();
237       llvm_unreachable("Clause local register live at end of clause.");
238     }
239     return true;
240   }
241 
242   MachineBasicBlock::iterator
MakeALUClause(MachineBasicBlock & MBB,MachineBasicBlock::iterator I)243   MakeALUClause(MachineBasicBlock &MBB, MachineBasicBlock::iterator I) {
244     MachineBasicBlock::iterator ClauseHead = I;
245     std::vector<std::pair<unsigned, unsigned>> KCacheBanks;
246     bool PushBeforeModifier = false;
247     unsigned AluInstCount = 0;
248     for (MachineBasicBlock::iterator E = MBB.end(); I != E; ++I) {
249       if (IsTrivialInst(*I))
250         continue;
251       if (!isALU(*I))
252         break;
253       if (AluInstCount > TII->getMaxAlusPerClause())
254         break;
255       if (I->getOpcode() == R600::PRED_X) {
256         // We put PRED_X in its own clause to ensure that ifcvt won't create
257         // clauses with more than 128 insts.
258         // IfCvt is indeed checking that "then" and "else" branches of an if
259         // statement have less than ~60 insts thus converted clauses can't be
260         // bigger than ~121 insts (predicate setter needs to be in the same
261         // clause as predicated alus).
262         if (AluInstCount > 0)
263           break;
264         if (TII->getFlagOp(*I).getImm() & MO_FLAG_PUSH)
265           PushBeforeModifier = true;
266         AluInstCount ++;
267         continue;
268       }
269       // XXX: GROUP_BARRIER instructions cannot be in the same ALU clause as:
270       //
271       // * KILL or INTERP instructions
272       // * Any instruction that sets UPDATE_EXEC_MASK or UPDATE_PRED bits
273       // * Uses waterfalling (i.e. INDEX_MODE = AR.X)
274       //
275       // XXX: These checks have not been implemented yet.
276       if (TII->mustBeLastInClause(I->getOpcode())) {
277         I++;
278         break;
279       }
280 
281       // If this instruction defines a clause local register, make sure
282       // its use can fit in this clause.
283       if (!canClauseLocalKillFitInClause(AluInstCount, KCacheBanks, I, E))
284         break;
285 
286       if (!SubstituteKCacheBank(*I, KCacheBanks))
287         break;
288       AluInstCount += OccupiedDwords(*I);
289     }
290     unsigned Opcode = PushBeforeModifier ?
291         R600::CF_ALU_PUSH_BEFORE : R600::CF_ALU;
292     BuildMI(MBB, ClauseHead, MBB.findDebugLoc(ClauseHead), TII->get(Opcode))
293     // We don't use the ADDR field until R600ControlFlowFinalizer pass, where
294     // it is safe to assume it is 0. However if we always put 0 here, the ifcvt
295     // pass may assume that identical ALU clause starter at the beginning of a
296     // true and false branch can be factorized which is not the case.
297         .addImm(Address++) // ADDR
298         .addImm(KCacheBanks.empty()?0:KCacheBanks[0].first) // KB0
299         .addImm((KCacheBanks.size() < 2)?0:KCacheBanks[1].first) // KB1
300         .addImm(KCacheBanks.empty()?0:2) // KM0
301         .addImm((KCacheBanks.size() < 2)?0:2) // KM1
302         .addImm(KCacheBanks.empty()?0:KCacheBanks[0].second) // KLINE0
303         .addImm((KCacheBanks.size() < 2)?0:KCacheBanks[1].second) // KLINE1
304         .addImm(AluInstCount) // COUNT
305         .addImm(1); // Enabled
306     return I;
307   }
308 
309 public:
310   static char ID;
311 
R600EmitClauseMarkers()312   R600EmitClauseMarkers() : MachineFunctionPass(ID) {
313     initializeR600EmitClauseMarkersPass(*PassRegistry::getPassRegistry());
314   }
315 
runOnMachineFunction(MachineFunction & MF)316   bool runOnMachineFunction(MachineFunction &MF) override {
317     const R600Subtarget &ST = MF.getSubtarget<R600Subtarget>();
318     TII = ST.getInstrInfo();
319 
320     for (MachineFunction::iterator BB = MF.begin(), BB_E = MF.end();
321                                                     BB != BB_E; ++BB) {
322       MachineBasicBlock &MBB = *BB;
323       MachineBasicBlock::iterator I = MBB.begin();
324       if (I != MBB.end() && I->getOpcode() == R600::CF_ALU)
325         continue; // BB was already parsed
326       for (MachineBasicBlock::iterator E = MBB.end(); I != E;) {
327         if (isALU(*I)) {
328           auto next = MakeALUClause(MBB, I);
329           assert(next != I);
330           I = next;
331         } else
332           ++I;
333       }
334     }
335     return false;
336   }
337 
getPassName() const338   StringRef getPassName() const override {
339     return "R600 Emit Clause Markers Pass";
340   }
341 };
342 
343 char R600EmitClauseMarkers::ID = 0;
344 
345 } // end anonymous namespace
346 
347 INITIALIZE_PASS_BEGIN(R600EmitClauseMarkers, "emitclausemarkers",
348                       "R600 Emit Clause Markters", false, false)
349 INITIALIZE_PASS_END(R600EmitClauseMarkers, "emitclausemarkers",
350                       "R600 Emit Clause Markters", false, false)
351 
createR600EmitClauseMarkers()352 FunctionPass *llvm::createR600EmitClauseMarkers() {
353   return new R600EmitClauseMarkers();
354 }
355