[x86] fix assert with horizontal math + broadcast of vector (PR43402)
[llvm-core.git] / lib / Target / AMDGPU / SIInsertSkips.cpp
blob87e63fcc4a04fa65bd0c9f18c8d43b4605b3f37a
1 //===-- SIInsertSkips.cpp - Use predicates for control flow ---------------===//
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 /// This pass inserts branches on the 0 exec mask over divergent branches
11 /// branches when it's expected that jumping over the untaken control flow will
12 /// be cheaper than having every workitem no-op through it.
14 //===----------------------------------------------------------------------===//
16 #include "AMDGPU.h"
17 #include "AMDGPUSubtarget.h"
18 #include "SIInstrInfo.h"
19 #include "SIMachineFunctionInfo.h"
20 #include "MCTargetDesc/AMDGPUMCTargetDesc.h"
21 #include "llvm/ADT/SmallVector.h"
22 #include "llvm/ADT/StringRef.h"
23 #include "llvm/CodeGen/MachineBasicBlock.h"
24 #include "llvm/CodeGen/MachineFunction.h"
25 #include "llvm/CodeGen/MachineFunctionPass.h"
26 #include "llvm/CodeGen/MachineInstr.h"
27 #include "llvm/CodeGen/MachineInstrBuilder.h"
28 #include "llvm/CodeGen/MachineOperand.h"
29 #include "llvm/IR/CallingConv.h"
30 #include "llvm/IR/DebugLoc.h"
31 #include "llvm/MC/MCAsmInfo.h"
32 #include "llvm/Pass.h"
33 #include "llvm/Support/CommandLine.h"
34 #include "llvm/Target/TargetMachine.h"
35 #include <cassert>
36 #include <cstdint>
37 #include <iterator>
39 using namespace llvm;
41 #define DEBUG_TYPE "si-insert-skips"
43 static cl::opt<unsigned> SkipThresholdFlag(
44 "amdgpu-skip-threshold",
45 cl::desc("Number of instructions before jumping over divergent control flow"),
46 cl::init(12), cl::Hidden);
48 namespace {
50 class SIInsertSkips : public MachineFunctionPass {
51 private:
52 const SIRegisterInfo *TRI = nullptr;
53 const SIInstrInfo *TII = nullptr;
54 unsigned SkipThreshold = 0;
56 bool shouldSkip(const MachineBasicBlock &From,
57 const MachineBasicBlock &To) const;
59 bool skipIfDead(MachineInstr &MI, MachineBasicBlock &NextBB);
61 void kill(MachineInstr &MI);
63 MachineBasicBlock *insertSkipBlock(MachineBasicBlock &MBB,
64 MachineBasicBlock::iterator I) const;
66 bool skipMaskBranch(MachineInstr &MI, MachineBasicBlock &MBB);
68 bool optimizeVccBranch(MachineInstr &MI) const;
70 public:
71 static char ID;
73 SIInsertSkips() : MachineFunctionPass(ID) {}
75 bool runOnMachineFunction(MachineFunction &MF) override;
77 StringRef getPassName() const override {
78 return "SI insert s_cbranch_execz instructions";
81 void getAnalysisUsage(AnalysisUsage &AU) const override {
82 MachineFunctionPass::getAnalysisUsage(AU);
86 } // end anonymous namespace
88 char SIInsertSkips::ID = 0;
90 INITIALIZE_PASS(SIInsertSkips, DEBUG_TYPE,
91 "SI insert s_cbranch_execz instructions", false, false)
93 char &llvm::SIInsertSkipsPassID = SIInsertSkips::ID;
95 static bool opcodeEmitsNoInsts(const MachineInstr &MI) {
96 if (MI.isMetaInstruction())
97 return true;
99 // Handle target specific opcodes.
100 switch (MI.getOpcode()) {
101 case AMDGPU::SI_MASK_BRANCH:
102 return true;
103 default:
104 return false;
108 bool SIInsertSkips::shouldSkip(const MachineBasicBlock &From,
109 const MachineBasicBlock &To) const {
110 unsigned NumInstr = 0;
111 const MachineFunction *MF = From.getParent();
113 for (MachineFunction::const_iterator MBBI(&From), ToI(&To), End = MF->end();
114 MBBI != End && MBBI != ToI; ++MBBI) {
115 const MachineBasicBlock &MBB = *MBBI;
117 for (MachineBasicBlock::const_iterator I = MBB.begin(), E = MBB.end();
118 NumInstr < SkipThreshold && I != E; ++I) {
119 if (opcodeEmitsNoInsts(*I))
120 continue;
122 // FIXME: Since this is required for correctness, this should be inserted
123 // during SILowerControlFlow.
125 // When a uniform loop is inside non-uniform control flow, the branch
126 // leaving the loop might be an S_CBRANCH_VCCNZ, which is never taken
127 // when EXEC = 0. We should skip the loop lest it becomes infinite.
128 if (I->getOpcode() == AMDGPU::S_CBRANCH_VCCNZ ||
129 I->getOpcode() == AMDGPU::S_CBRANCH_VCCZ)
130 return true;
132 if (TII->hasUnwantedEffectsWhenEXECEmpty(*I))
133 return true;
135 // These instructions are potentially expensive even if EXEC = 0.
136 if (TII->isSMRD(*I) || TII->isVMEM(*I) || TII->isFLAT(*I) ||
137 I->getOpcode() == AMDGPU::S_WAITCNT)
138 return true;
140 ++NumInstr;
141 if (NumInstr >= SkipThreshold)
142 return true;
146 return false;
149 bool SIInsertSkips::skipIfDead(MachineInstr &MI, MachineBasicBlock &NextBB) {
150 MachineBasicBlock &MBB = *MI.getParent();
151 MachineFunction *MF = MBB.getParent();
153 if (MF->getFunction().getCallingConv() != CallingConv::AMDGPU_PS ||
154 !shouldSkip(MBB, MBB.getParent()->back()))
155 return false;
157 MachineBasicBlock *SkipBB = insertSkipBlock(MBB, MI.getIterator());
159 const DebugLoc &DL = MI.getDebugLoc();
161 // If the exec mask is non-zero, skip the next two instructions
162 BuildMI(&MBB, DL, TII->get(AMDGPU::S_CBRANCH_EXECNZ))
163 .addMBB(&NextBB);
165 MachineBasicBlock::iterator Insert = SkipBB->begin();
167 // Exec mask is zero: Export to NULL target...
168 BuildMI(*SkipBB, Insert, DL, TII->get(AMDGPU::EXP_DONE))
169 .addImm(0x09) // V_008DFC_SQ_EXP_NULL
170 .addReg(AMDGPU::VGPR0, RegState::Undef)
171 .addReg(AMDGPU::VGPR0, RegState::Undef)
172 .addReg(AMDGPU::VGPR0, RegState::Undef)
173 .addReg(AMDGPU::VGPR0, RegState::Undef)
174 .addImm(1) // vm
175 .addImm(0) // compr
176 .addImm(0); // en
178 // ... and terminate wavefront.
179 BuildMI(*SkipBB, Insert, DL, TII->get(AMDGPU::S_ENDPGM)).addImm(0);
181 return true;
184 void SIInsertSkips::kill(MachineInstr &MI) {
185 MachineBasicBlock &MBB = *MI.getParent();
186 DebugLoc DL = MI.getDebugLoc();
188 switch (MI.getOpcode()) {
189 case AMDGPU::SI_KILL_F32_COND_IMM_TERMINATOR: {
190 unsigned Opcode = 0;
192 // The opcodes are inverted because the inline immediate has to be
193 // the first operand, e.g. from "x < imm" to "imm > x"
194 switch (MI.getOperand(2).getImm()) {
195 case ISD::SETOEQ:
196 case ISD::SETEQ:
197 Opcode = AMDGPU::V_CMPX_EQ_F32_e64;
198 break;
199 case ISD::SETOGT:
200 case ISD::SETGT:
201 Opcode = AMDGPU::V_CMPX_LT_F32_e64;
202 break;
203 case ISD::SETOGE:
204 case ISD::SETGE:
205 Opcode = AMDGPU::V_CMPX_LE_F32_e64;
206 break;
207 case ISD::SETOLT:
208 case ISD::SETLT:
209 Opcode = AMDGPU::V_CMPX_GT_F32_e64;
210 break;
211 case ISD::SETOLE:
212 case ISD::SETLE:
213 Opcode = AMDGPU::V_CMPX_GE_F32_e64;
214 break;
215 case ISD::SETONE:
216 case ISD::SETNE:
217 Opcode = AMDGPU::V_CMPX_LG_F32_e64;
218 break;
219 case ISD::SETO:
220 Opcode = AMDGPU::V_CMPX_O_F32_e64;
221 break;
222 case ISD::SETUO:
223 Opcode = AMDGPU::V_CMPX_U_F32_e64;
224 break;
225 case ISD::SETUEQ:
226 Opcode = AMDGPU::V_CMPX_NLG_F32_e64;
227 break;
228 case ISD::SETUGT:
229 Opcode = AMDGPU::V_CMPX_NGE_F32_e64;
230 break;
231 case ISD::SETUGE:
232 Opcode = AMDGPU::V_CMPX_NGT_F32_e64;
233 break;
234 case ISD::SETULT:
235 Opcode = AMDGPU::V_CMPX_NLE_F32_e64;
236 break;
237 case ISD::SETULE:
238 Opcode = AMDGPU::V_CMPX_NLT_F32_e64;
239 break;
240 case ISD::SETUNE:
241 Opcode = AMDGPU::V_CMPX_NEQ_F32_e64;
242 break;
243 default:
244 llvm_unreachable("invalid ISD:SET cond code");
247 const GCNSubtarget &ST = MBB.getParent()->getSubtarget<GCNSubtarget>();
248 if (ST.hasNoSdstCMPX())
249 Opcode = AMDGPU::getVCMPXNoSDstOp(Opcode);
251 assert(MI.getOperand(0).isReg());
253 if (TRI->isVGPR(MBB.getParent()->getRegInfo(),
254 MI.getOperand(0).getReg())) {
255 Opcode = AMDGPU::getVOPe32(Opcode);
256 BuildMI(MBB, &MI, DL, TII->get(Opcode))
257 .add(MI.getOperand(1))
258 .add(MI.getOperand(0));
259 } else {
260 auto I = BuildMI(MBB, &MI, DL, TII->get(Opcode));
261 if (!ST.hasNoSdstCMPX())
262 I.addReg(AMDGPU::VCC, RegState::Define);
264 I.addImm(0) // src0 modifiers
265 .add(MI.getOperand(1))
266 .addImm(0) // src1 modifiers
267 .add(MI.getOperand(0));
269 I.addImm(0); // omod
271 break;
273 case AMDGPU::SI_KILL_I1_TERMINATOR: {
274 const MachineFunction *MF = MI.getParent()->getParent();
275 const GCNSubtarget &ST = MF->getSubtarget<GCNSubtarget>();
276 unsigned Exec = ST.isWave32() ? AMDGPU::EXEC_LO : AMDGPU::EXEC;
277 const MachineOperand &Op = MI.getOperand(0);
278 int64_t KillVal = MI.getOperand(1).getImm();
279 assert(KillVal == 0 || KillVal == -1);
281 // Kill all threads if Op0 is an immediate and equal to the Kill value.
282 if (Op.isImm()) {
283 int64_t Imm = Op.getImm();
284 assert(Imm == 0 || Imm == -1);
286 if (Imm == KillVal)
287 BuildMI(MBB, &MI, DL, TII->get(ST.isWave32() ? AMDGPU::S_MOV_B32
288 : AMDGPU::S_MOV_B64), Exec)
289 .addImm(0);
290 break;
293 unsigned Opcode = KillVal ? AMDGPU::S_ANDN2_B64 : AMDGPU::S_AND_B64;
294 if (ST.isWave32())
295 Opcode = KillVal ? AMDGPU::S_ANDN2_B32 : AMDGPU::S_AND_B32;
296 BuildMI(MBB, &MI, DL, TII->get(Opcode), Exec)
297 .addReg(Exec)
298 .add(Op);
299 break;
301 default:
302 llvm_unreachable("invalid opcode, expected SI_KILL_*_TERMINATOR");
306 MachineBasicBlock *SIInsertSkips::insertSkipBlock(
307 MachineBasicBlock &MBB, MachineBasicBlock::iterator I) const {
308 MachineFunction *MF = MBB.getParent();
310 MachineBasicBlock *SkipBB = MF->CreateMachineBasicBlock();
311 MachineFunction::iterator MBBI(MBB);
312 ++MBBI;
314 MF->insert(MBBI, SkipBB);
315 MBB.addSuccessor(SkipBB);
317 return SkipBB;
320 // Returns true if a branch over the block was inserted.
321 bool SIInsertSkips::skipMaskBranch(MachineInstr &MI,
322 MachineBasicBlock &SrcMBB) {
323 MachineBasicBlock *DestBB = MI.getOperand(0).getMBB();
325 if (!shouldSkip(**SrcMBB.succ_begin(), *DestBB))
326 return false;
328 const DebugLoc &DL = MI.getDebugLoc();
329 MachineBasicBlock::iterator InsPt = std::next(MI.getIterator());
331 BuildMI(SrcMBB, InsPt, DL, TII->get(AMDGPU::S_CBRANCH_EXECZ))
332 .addMBB(DestBB);
334 return true;
337 bool SIInsertSkips::optimizeVccBranch(MachineInstr &MI) const {
338 // Match:
339 // sreg = -1
340 // vcc = S_AND_B64 exec, sreg
341 // S_CBRANCH_VCC[N]Z
342 // =>
343 // S_CBRANCH_EXEC[N]Z
344 bool Changed = false;
345 MachineBasicBlock &MBB = *MI.getParent();
346 const GCNSubtarget &ST = MBB.getParent()->getSubtarget<GCNSubtarget>();
347 const bool IsWave32 = ST.isWave32();
348 const unsigned CondReg = TRI->getVCC();
349 const unsigned ExecReg = IsWave32 ? AMDGPU::EXEC_LO : AMDGPU::EXEC;
350 const unsigned And = IsWave32 ? AMDGPU::S_AND_B32 : AMDGPU::S_AND_B64;
352 MachineBasicBlock::reverse_iterator A = MI.getReverseIterator(),
353 E = MBB.rend();
354 bool ReadsCond = false;
355 unsigned Threshold = 5;
356 for (++A ; A != E ; ++A) {
357 if (!--Threshold)
358 return false;
359 if (A->modifiesRegister(ExecReg, TRI))
360 return false;
361 if (A->modifiesRegister(CondReg, TRI)) {
362 if (!A->definesRegister(CondReg, TRI) || A->getOpcode() != And)
363 return false;
364 break;
366 ReadsCond |= A->readsRegister(CondReg, TRI);
368 if (A == E)
369 return false;
371 MachineOperand &Op1 = A->getOperand(1);
372 MachineOperand &Op2 = A->getOperand(2);
373 if (Op1.getReg() != ExecReg && Op2.isReg() && Op2.getReg() == ExecReg) {
374 TII->commuteInstruction(*A);
375 Changed = true;
377 if (Op1.getReg() != ExecReg)
378 return Changed;
379 if (Op2.isImm() && Op2.getImm() != -1)
380 return Changed;
382 unsigned SReg = AMDGPU::NoRegister;
383 if (Op2.isReg()) {
384 SReg = Op2.getReg();
385 auto M = std::next(A);
386 bool ReadsSreg = false;
387 for ( ; M != E ; ++M) {
388 if (M->definesRegister(SReg, TRI))
389 break;
390 if (M->modifiesRegister(SReg, TRI))
391 return Changed;
392 ReadsSreg |= M->readsRegister(SReg, TRI);
394 if (M == E ||
395 !M->isMoveImmediate() ||
396 !M->getOperand(1).isImm() ||
397 M->getOperand(1).getImm() != -1)
398 return Changed;
399 // First if sreg is only used in and instruction fold the immediate
400 // into that and.
401 if (!ReadsSreg && Op2.isKill()) {
402 A->getOperand(2).ChangeToImmediate(-1);
403 M->eraseFromParent();
407 if (!ReadsCond && A->registerDefIsDead(AMDGPU::SCC) &&
408 MI.killsRegister(CondReg, TRI))
409 A->eraseFromParent();
411 bool IsVCCZ = MI.getOpcode() == AMDGPU::S_CBRANCH_VCCZ;
412 if (SReg == ExecReg) {
413 if (IsVCCZ) {
414 MI.eraseFromParent();
415 return true;
417 MI.setDesc(TII->get(AMDGPU::S_BRANCH));
418 } else {
419 MI.setDesc(TII->get(IsVCCZ ? AMDGPU::S_CBRANCH_EXECZ
420 : AMDGPU::S_CBRANCH_EXECNZ));
423 MI.RemoveOperand(MI.findRegisterUseOperandIdx(CondReg, false /*Kill*/, TRI));
424 MI.addImplicitDefUseOperands(*MBB.getParent());
426 return true;
429 bool SIInsertSkips::runOnMachineFunction(MachineFunction &MF) {
430 const GCNSubtarget &ST = MF.getSubtarget<GCNSubtarget>();
431 TII = ST.getInstrInfo();
432 TRI = &TII->getRegisterInfo();
433 SkipThreshold = SkipThresholdFlag;
435 bool HaveKill = false;
436 bool MadeChange = false;
438 // Track depth of exec mask, divergent branches.
439 SmallVector<MachineBasicBlock *, 16> ExecBranchStack;
441 MachineFunction::iterator NextBB;
443 MachineBasicBlock *EmptyMBBAtEnd = nullptr;
445 for (MachineFunction::iterator BI = MF.begin(), BE = MF.end();
446 BI != BE; BI = NextBB) {
447 NextBB = std::next(BI);
448 MachineBasicBlock &MBB = *BI;
449 bool HaveSkipBlock = false;
451 if (!ExecBranchStack.empty() && ExecBranchStack.back() == &MBB) {
452 // Reached convergence point for last divergent branch.
453 ExecBranchStack.pop_back();
456 if (HaveKill && ExecBranchStack.empty()) {
457 HaveKill = false;
459 // TODO: Insert skip if exec is 0?
462 MachineBasicBlock::iterator I, Next;
463 for (I = MBB.begin(); I != MBB.end(); I = Next) {
464 Next = std::next(I);
466 MachineInstr &MI = *I;
468 switch (MI.getOpcode()) {
469 case AMDGPU::SI_MASK_BRANCH:
470 ExecBranchStack.push_back(MI.getOperand(0).getMBB());
471 MadeChange |= skipMaskBranch(MI, MBB);
472 break;
474 case AMDGPU::S_BRANCH:
475 // Optimize out branches to the next block.
476 // FIXME: Shouldn't this be handled by BranchFolding?
477 if (MBB.isLayoutSuccessor(MI.getOperand(0).getMBB())) {
478 MI.eraseFromParent();
479 } else if (HaveSkipBlock) {
480 // Remove the given unconditional branch when a skip block has been
481 // inserted after the current one and let skip the two instructions
482 // performing the kill if the exec mask is non-zero.
483 MI.eraseFromParent();
485 break;
487 case AMDGPU::SI_KILL_F32_COND_IMM_TERMINATOR:
488 case AMDGPU::SI_KILL_I1_TERMINATOR:
489 MadeChange = true;
490 kill(MI);
492 if (ExecBranchStack.empty()) {
493 if (NextBB != BE && skipIfDead(MI, *NextBB)) {
494 HaveSkipBlock = true;
495 NextBB = std::next(BI);
496 BE = MF.end();
498 } else {
499 HaveKill = true;
502 MI.eraseFromParent();
503 break;
505 case AMDGPU::SI_RETURN_TO_EPILOG:
506 // FIXME: Should move somewhere else
507 assert(!MF.getInfo<SIMachineFunctionInfo>()->returnsVoid());
509 // Graphics shaders returning non-void shouldn't contain S_ENDPGM,
510 // because external bytecode will be appended at the end.
511 if (BI != --MF.end() || I != MBB.getFirstTerminator()) {
512 // SI_RETURN_TO_EPILOG is not the last instruction. Add an empty block at
513 // the end and jump there.
514 if (!EmptyMBBAtEnd) {
515 EmptyMBBAtEnd = MF.CreateMachineBasicBlock();
516 MF.insert(MF.end(), EmptyMBBAtEnd);
519 MBB.addSuccessor(EmptyMBBAtEnd);
520 BuildMI(*BI, I, MI.getDebugLoc(), TII->get(AMDGPU::S_BRANCH))
521 .addMBB(EmptyMBBAtEnd);
522 I->eraseFromParent();
524 break;
526 case AMDGPU::S_CBRANCH_VCCZ:
527 case AMDGPU::S_CBRANCH_VCCNZ:
528 MadeChange |= optimizeVccBranch(MI);
529 break;
531 default:
532 break;
537 return MadeChange;