LLVM 22.0.0git
AMDGPUPrepareAGPRAlloc.cpp
Go to the documentation of this file.
1//===-- AMDGPUPrepareAGPRAlloc.cpp ----------------------------------------===//
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// Make simple transformations to relax register constraints for cases which can
10// allocate to AGPRs or VGPRs. Replace materialize of inline immediates into
11// AGPR or VGPR with a pseudo with an AV_* class register constraint. This
12// allows later passes to inflate the register class if necessary. The register
13// allocator does not know to replace instructions to relax constraints.
14//
15//===----------------------------------------------------------------------===//
16
18#include "AMDGPU.h"
19#include "GCNSubtarget.h"
21#include "SIRegisterInfo.h"
25
26using namespace llvm;
27
28#define DEBUG_TYPE "amdgpu-prepare-agpr-alloc"
29
30namespace {
31
32class AMDGPUPrepareAGPRAllocImpl {
33private:
34 const SIInstrInfo &TII;
36
37 bool isAV64Imm(const MachineOperand &MO) const;
38
39public:
40 AMDGPUPrepareAGPRAllocImpl(const GCNSubtarget &ST, MachineRegisterInfo &MRI)
41 : TII(*ST.getInstrInfo()), MRI(MRI) {}
42 bool run(MachineFunction &MF);
43};
44
45class AMDGPUPrepareAGPRAllocLegacy : public MachineFunctionPass {
46public:
47 static char ID;
48
49 AMDGPUPrepareAGPRAllocLegacy() : MachineFunctionPass(ID) {
52 }
53
54 bool runOnMachineFunction(MachineFunction &MF) override;
55
56 StringRef getPassName() const override { return "AMDGPU Prepare AGPR Alloc"; }
57
58 void getAnalysisUsage(AnalysisUsage &AU) const override {
59 AU.setPreservesAll();
61 }
62};
63} // End anonymous namespace.
64
65INITIALIZE_PASS_BEGIN(AMDGPUPrepareAGPRAllocLegacy, DEBUG_TYPE,
66 "AMDGPU Prepare AGPR Alloc", false, false)
67INITIALIZE_PASS_END(AMDGPUPrepareAGPRAllocLegacy, DEBUG_TYPE,
68 "AMDGPU Prepare AGPR Alloc", false, false)
69
70char AMDGPUPrepareAGPRAllocLegacy::ID = 0;
71
72char &llvm::AMDGPUPrepareAGPRAllocLegacyID = AMDGPUPrepareAGPRAllocLegacy::ID;
73
74bool AMDGPUPrepareAGPRAllocLegacy::runOnMachineFunction(MachineFunction &MF) {
75 if (skipFunction(MF.getFunction()))
76 return false;
77
78 const GCNSubtarget &ST = MF.getSubtarget<GCNSubtarget>();
79 return AMDGPUPrepareAGPRAllocImpl(ST, MF.getRegInfo()).run(MF);
80}
81
85 const GCNSubtarget &ST = MF.getSubtarget<GCNSubtarget>();
86 AMDGPUPrepareAGPRAllocImpl(ST, MF.getRegInfo()).run(MF);
88}
89
90bool AMDGPUPrepareAGPRAllocImpl::isAV64Imm(const MachineOperand &MO) const {
91 return MO.isImm() && TII.isLegalAV64PseudoImm(MO.getImm());
92}
93
94bool AMDGPUPrepareAGPRAllocImpl::run(MachineFunction &MF) {
95 if (MRI.isReserved(AMDGPU::AGPR0))
96 return false;
97
98 const MCInstrDesc &AVImmPseudo32 = TII.get(AMDGPU::AV_MOV_B32_IMM_PSEUDO);
99 const MCInstrDesc &AVImmPseudo64 = TII.get(AMDGPU::AV_MOV_B64_IMM_PSEUDO);
100
101 bool Changed = false;
102 for (MachineBasicBlock &MBB : MF) {
103 for (MachineInstr &MI : MBB) {
104 if ((MI.getOpcode() == AMDGPU::V_MOV_B32_e32 &&
105 TII.isInlineConstant(MI, 1)) ||
106 (MI.getOpcode() == AMDGPU::V_ACCVGPR_WRITE_B32_e64 &&
107 MI.getOperand(1).isImm())) {
108 MI.setDesc(AVImmPseudo32);
109 Changed = true;
110 continue;
111 }
112
113 // TODO: If only half of the value is rewritable, is it worth splitting it
114 // up?
115 if ((MI.getOpcode() == AMDGPU::V_MOV_B64_e64 ||
116 MI.getOpcode() == AMDGPU::V_MOV_B64_PSEUDO) &&
117 isAV64Imm(MI.getOperand(1))) {
118 MI.setDesc(AVImmPseudo64);
119 Changed = true;
120 continue;
121 }
122 }
123 }
124
125 return Changed;
126}
unsigned const MachineRegisterInfo * MRI
AMDGPU Prepare AGPR Alloc
#define DEBUG_TYPE
MachineBasicBlock & MBB
AMD GCN specific subclass of TargetSubtarget.
const HexagonInstrInfo * TII
IRTranslator LLVM IR MI
#define INITIALIZE_PASS_END(passName, arg, name, cfg, analysis)
Definition: PassSupport.h:44
#define INITIALIZE_PASS_BEGIN(passName, arg, name, cfg, analysis)
Definition: PassSupport.h:39
Interface definition for SIRegisterInfo.
PreservedAnalyses run(MachineFunction &MF, MachineFunctionAnalysisManager &MFAM)
A container for analyses that lazily runs them and caches their results.
Definition: PassManager.h:255
Represent the analysis usage information of a pass.
void setPreservesAll()
Set by analyses that do not transform their input at all.
Describe properties that are true of each instruction in the target description file.
Definition: MCInstrDesc.h:199
MachineFunctionPass - This class adapts the FunctionPass interface to allow convenient creation of pa...
void getAnalysisUsage(AnalysisUsage &AU) const override
getAnalysisUsage - Subclasses that override getAnalysisUsage must call this.
virtual bool runOnMachineFunction(MachineFunction &MF)=0
runOnMachineFunction - This method must be overloaded to perform the desired machine code transformat...
const TargetSubtargetInfo & getSubtarget() const
getSubtarget - Return the subtarget for which this machine code is being compiled.
MachineRegisterInfo & getRegInfo()
getRegInfo - Return information about the registers currently in use.
Representation of each machine instruction.
Definition: MachineInstr.h:72
MachineOperand class - Representation of each machine instruction operand.
int64_t getImm() const
bool isImm() const
isImm - Tests if this is a MO_Immediate operand.
MachineRegisterInfo - Keep track of information for virtual and physical registers,...
static LLVM_ABI PassRegistry * getPassRegistry()
getPassRegistry - Access the global registry object, which is automatically initialized at applicatio...
virtual StringRef getPassName() const
getPassName - Return a nice clean name for a pass.
Definition: Pass.cpp:85
A set of analyses that are preserved following a run of a transformation pass.
Definition: Analysis.h:112
static PreservedAnalyses all()
Construct a special preserved set that preserves all passes.
Definition: Analysis.h:118
StringRef - Represent a constant reference to a string, i.e.
Definition: StringRef.h:55
unsigned ID
LLVM IR allows to use arbitrary numbers as calling convention identifiers.
Definition: CallingConv.h:24
This is an optimization pass for GlobalISel generic memory operations.
Definition: AddressRanges.h:18
void initializeAMDGPUPrepareAGPRAllocLegacyPass(PassRegistry &)
char & AMDGPUPrepareAGPRAllocLegacyID