LLVM 23.0.0git
SPIRVTargetMachine.cpp
Go to the documentation of this file.
1//===- SPIRVTargetMachine.cpp - Define TargetMachine for SPIR-V -*- C++ -*-===//
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// Implements the info about SPIR-V target spec.
10//
11//===----------------------------------------------------------------------===//
12
13#include "SPIRVTargetMachine.h"
15#include "SPIRV.h"
16#include "SPIRVCBufferAccess.h"
18#include "SPIRVEmitIntrinsics.h"
19#include "SPIRVGlobalRegistry.h"
23#include "SPIRVLegalizerInfo.h"
26#include "SPIRVRegularizer.h"
35#include "llvm/CodeGen/Passes.h"
39#include "llvm/Pass.h"
46#include <optional>
47
48using namespace llvm;
49
77
78static Reloc::Model getEffectiveRelocModel(std::optional<Reloc::Model> RM) {
79 if (!RM)
80 return Reloc::PIC_;
81 return *RM;
82}
83
84// Pin SPIRVTargetObjectFile's vtables to this file.
86
88 StringRef CPU, StringRef FS,
90 std::optional<Reloc::Model> RM,
91 std::optional<CodeModel::Model> CM,
92 CodeGenOptLevel OL, bool JIT)
93 : CodeGenTargetMachineImpl(T, TT.computeDataLayout(), TT, CPU, FS, Options,
95 getEffectiveCodeModel(CM, CodeModel::Small), OL),
96 TLOF(std::make_unique<SPIRVTargetObjectFile>()),
97 Subtarget(TT, CPU.str(), FS.str(), *this) {
99 setGlobalISel(true);
100 setFastISel(false);
101 setO0WantsFastISel(false);
103}
104
106#define GET_PASS_REGISTRY "SPIRVPassRegistry.def"
108}
109
110namespace {
111// SPIR-V Code Generator Pass Configuration Options.
112class SPIRVPassConfig : public TargetPassConfig {
113public:
114 SPIRVPassConfig(SPIRVTargetMachine &TM, PassManagerBase &PM)
115 : TargetPassConfig(TM, PM), TM(TM) {}
116
117 SPIRVTargetMachine &getSPIRVTargetMachine() const {
119 }
120 void addMachineSSAOptimization() override;
121 void addIRPasses() override;
122 void addISelPrepare() override;
123
124 bool addIRTranslator() override;
125 void addPreLegalizeMachineIR() override;
126 bool addLegalizeMachineIR() override;
127 bool addRegBankSelect() override;
128 bool addGlobalInstructionSelect() override;
129
130 FunctionPass *createTargetRegisterAllocator(bool) override;
131 void addFastRegAlloc() override {}
132 void addOptimizedRegAlloc() override {}
133
134 void addPostRegAlloc() override;
135
136private:
137 const SPIRVTargetMachine &TM;
138};
139} // namespace
140
141// We do not use physical registers, and maintain virtual registers throughout
142// the entire pipeline, so return nullptr to disable register allocation.
143FunctionPass *SPIRVPassConfig::createTargetRegisterAllocator(bool) {
144 return nullptr;
145}
146
147// A place to disable passes that may break CFG.
148void SPIRVPassConfig::addMachineSSAOptimization() {
150}
151
152// Disable passes that break from assuming no virtual registers exist.
153void SPIRVPassConfig::addPostRegAlloc() {
154 // Do not work with vregs instead of physical regs.
155 disablePass(&MachineCopyPropagationID);
156 disablePass(&PostRAMachineSinkingID);
157 disablePass(&PostRASchedulerID);
158 disablePass(&FuncletLayoutID);
159 disablePass(&StackMapLivenessID);
160 disablePass(&PatchableFunctionID);
161 disablePass(&ShrinkWrapID);
162 disablePass(&LiveDebugValuesID);
163 disablePass(&MachineLateInstrsCleanupID);
164 disablePass(&RemoveLoadsIntoFakeUsesID);
165
166 // Do not work with OpPhi.
167 disablePass(&BranchFolderPassID);
168 disablePass(&MachineBlockPlacementID);
169
171}
172
175 return TargetTransformInfo(std::make_unique<SPIRVTTIImpl>(this, F));
176}
177
179 return new SPIRVPassConfig(*this, PM);
180}
181
182void SPIRVPassConfig::addIRPasses() {
184
186
187 // Variadic function calls aren't supported in shader code.
188 // This needs to come before SPIRVPrepareFunctions because this
189 // may introduce intrinsic calls.
190 if (!TM.getSubtargetImpl()->isShader()) {
192 }
193
198}
199
200void SPIRVPassConfig::addISelPrepare() {
201 if (TM.getSubtargetImpl()->isShader()) {
202 // Vulkan does not allow address space casts. This pass is run to remove
203 // address space casts that can be removed.
204 // If an address space cast is not removed while targeting Vulkan, lowering
205 // will fail during MIR lowering.
207
208 // 1. Simplify loop for subsequent transformations. After this steps, loops
209 // have the following properties:
210 // - loops have a single entry edge (pre-header to loop header).
211 // - all loop exits are dominated by the loop pre-header.
212 // - loops have a single back-edge.
213 addPass(createLoopSimplifyPass());
214
215 // 2. Removes registers whose lifetime spans across basic blocks. Also
216 // removes phi nodes. This will greatly simplify the next steps.
217 addPass(createRegToMemWrapperPass());
218
219 // 3. Merge the convergence region exit nodes into one. After this step,
220 // regions are single-entry, single-exit. This will help determine the
221 // correct merge block.
223
224 // 4. Structurize.
226
227 // 5. Reduce the amount of variables required by pushing some operations
228 // back to virtual registers.
230 } else {
231 // Canonicalize loops so they have a single latch and preheader.
232 // This enables OpLoopMerge emission for non-shader targets.
233 addPass(createLoopSimplifyPass());
234 }
245}
246
247bool SPIRVPassConfig::addIRTranslator() {
248 addPass(new IRTranslator(getOptLevel()));
249 return false;
250}
251
252void SPIRVPassConfig::addPreLegalizeMachineIR() {
255}
256
257// Use the default legalizer.
258bool SPIRVPassConfig::addLegalizeMachineIR() {
259 addPass(new Legalizer());
261 return false;
262}
263
264// Do not add the RegBankSelect pass, as we only ever need virtual registers.
265bool SPIRVPassConfig::addRegBankSelect() {
266 disablePass(&RegBankSelect::ID);
267 return false;
268}
269
270// Deprecated flag kept for backward compatibility. NSDI emission is now handled
271// by SPIRVNonSemanticDebugHandler, registered in SPIRVAsmPrinter::
272// doInitialization() when the module contains debug info (llvm.dbg.cu).
273// TODO: Remove this option after a deprecation period. Callers that used
274// -spv-emit-nonsemantic-debug-info should switch to -g.
276 "spv-emit-nonsemantic-debug-info",
277 cl::desc("Deprecated. Use -g to emit SPIR-V NonSemantic.Shader.DebugInfo "
278 "instructions"),
279 cl::Optional, cl::init(false));
280
281namespace {
282// A custom subclass of InstructionSelect, which is mostly the same except from
283// not requiring RegBankSelect to occur previously.
284class SPIRVInstructionSelect : public InstructionSelect {
285 // We don't use register banks, so unset the requirement for them
286 MachineFunctionProperties getRequiredProperties() const override {
287 return InstructionSelect::getRequiredProperties().resetRegBankSelected();
288 }
289};
290} // namespace
291
292// Add the custom SPIRVInstructionSelect from above.
293bool SPIRVPassConfig::addGlobalInstructionSelect() {
294 addPass(new SPIRVInstructionSelect());
295 return false;
296}
static Reloc::Model getEffectiveRelocModel()
#define X(NUM, ENUM, NAME)
Definition ELF.h:851
#define LLVM_ABI
Definition Compiler.h:213
#define LLVM_EXTERNAL_VISIBILITY
Definition Compiler.h:132
DXIL Legalizer
This file declares the IRTranslator pass.
#define F(x, y, z)
Definition MD5.cpp:54
#define T
PassBuilder PB(Machine, PassOpts->PTO, std::nullopt, &PIC)
This file describes the interface of the MachineFunctionPass responsible for assigning the generic vi...
const GCNTargetMachine & getTM(const GCNSubtarget *STI)
LLVM_ABI LLVM_EXTERNAL_VISIBILITY void LLVMInitializeSPIRVTarget()
static cl::opt< bool > SPVEnableNonSemanticDI("spv-emit-nonsemantic-debug-info", cl::desc("Deprecated. Use -g to emit SPIR-V NonSemantic.Shader.DebugInfo " "instructions"), cl::Optional, cl::init(false))
static TableGen::Emitter::Opt Y("gen-skeleton-entry", EmitSkeleton, "Generate example skeleton entry")
Target-Independent Code Generator Pass Configuration Options pass.
CodeGenTargetMachineImpl(const Target &T, StringRef DataLayoutString, const Triple &TT, StringRef CPU, StringRef FS, const TargetOptions &Options, Reloc::Model RM, CodeModel::Model CM, CodeGenOptLevel OL)
FunctionPass class - This class is used to implement most global optimizations.
Definition Pass.h:314
This pass is responsible for selecting generic machine instructions to target-specific instructions.
MachineFunctionProperties getRequiredProperties() const override
Properties which a MachineFunction may have at a given point in time.
This class provides access to building LLVM's passes.
PassRegistry - This class manages the registration and intitialization of the pass subsystem as appli...
static LLVM_ABI PassRegistry * getPassRegistry()
getPassRegistry - Access the global registry object, which is automatically initialized at applicatio...
bool isLogicalSPIRV() const
void registerPassBuilderCallbacks(PassBuilder &PB) override
Allow the target to modify the pass pipeline.
SPIRVTargetMachine(const Target &T, const Triple &TT, StringRef CPU, StringRef FS, const TargetOptions &Options, std::optional< Reloc::Model > RM, std::optional< CodeModel::Model > CM, CodeGenOptLevel OL, bool JIT)
const SPIRVSubtarget * getSubtargetImpl() const
TargetTransformInfo getTargetTransformInfo(const Function &F) const override
Get a TargetTransformInfo implementation for the target.
TargetPassConfig * createPassConfig(PassManagerBase &PM) override
Create a pass configuration object to be used by addPassToEmitX methods for generating a pipeline of ...
StringRef - Represent a constant reference to a string, i.e.
Definition StringRef.h:55
void setFastISel(bool Enable)
void setRequiresStructuredCFG(bool Value)
void setGlobalISel(bool Enable)
TargetOptions Options
void setO0WantsFastISel(bool Enable)
Target-Independent Code Generator Pass Configuration Options.
virtual void addPostRegAlloc()
This method may be implemented by targets that want to run passes after register allocation pass pipe...
virtual void addIRPasses()
Add common target configurable passes that perform LLVM IR to IR transforms following machine indepen...
virtual void addMachineSSAOptimization()
addMachineSSAOptimization - Add standard passes that optimize machine instructions in SSA form.
virtual void addISelPrepare()
Add common passes that perform LLVM IR to IR transforms in preparation for instruction selection.
This pass provides access to the codegen interfaces that are needed for IR-level transformations.
Target - Wrapper for Target specific information.
Triple - Helper class for working with autoconf configuration names.
Definition Triple.h:47
PassManagerBase - An abstract interface to allow code to add passes to a pass manager without having ...
Interfaces for registering analysis passes, producing common pass manager configurations,...
initializer< Ty > init(const Ty &Val)
This is an optimization pass for GlobalISel generic memory operations.
ModulePass * createSPIRVPushConstantAccessLegacyPass(SPIRVTargetMachine *TM)
ModulePass * createSPIRVCtorDtorLoweringLegacyPass()
void initializeSPIRVEmitIntrinsicsPass(PassRegistry &)
FunctionPass * createSPIRVStructurizerPass()
ModulePass * createSPIRVEmitIntrinsicsPass(const SPIRVTargetMachine &TM)
LLVM_ABI FunctionPass * createPromoteMemoryToRegisterPass()
Definition Mem2Reg.cpp:114
ModulePass * createExpandVariadicsPass(ExpandVariadicsMode)
Target & getTheSPIRV32Target()
void initializeSPIRVPrepareFunctionsPass(PassRegistry &)
LLVM_ABI FunctionPass * createRegToMemWrapperPass()
Definition Reg2Mem.cpp:146
FunctionPass * createSPIRVPreLegalizerPass()
void initializeSPIRVPushConstantAccessLegacyPass(PassRegistry &)
LLVM_ABI char & PatchableFunctionID
This pass implements the "patchable-function" attribute.
LLVM_ABI char & PostRASchedulerID
PostRAScheduler - This pass performs post register allocation scheduling.
LLVM_ABI char & RemoveLoadsIntoFakeUsesID
RemoveLoadsIntoFakeUses pass.
void initializeSPIRVLegalizePointerCastLegacyPass(PassRegistry &)
void initializeSPIRVPreLegalizerCombinerPass(PassRegistry &)
LLVM_ABI char & LiveDebugValuesID
LiveDebugValues pass.
FunctionPass * createSPIRVPreLegalizerCombiner()
void initializeSPIRVModuleAnalysisPass(PassRegistry &)
static Reloc::Model getEffectiveRelocModel(std::optional< Reloc::Model > RM)
FunctionPass * createSPIRVPostLegalizerPass()
CodeModel::Model getEffectiveCodeModel(std::optional< CodeModel::Model > CM, CodeModel::Model Default)
Helper method for getting the code model, returning Default if CM does not have a value.
LLVM_ABI char & ShrinkWrapID
ShrinkWrap pass. Look for the best place to insert save and restore.
LLVM_ABI char & MachineLateInstrsCleanupID
MachineLateInstrsCleanup - This pass removes redundant identical instructions after register allocati...
void initializeSPIRVLegalizeZeroSizeArraysLegacyPass(PassRegistry &)
ModulePass * createSPIRVPrepareGlobalsPass()
LLVM_ABI char & StackMapLivenessID
StackMapLiveness - This pass analyses the register live-out set of stackmap/patchpoint intrinsics and...
LLVM_ABI char & FuncletLayoutID
This pass lays out funclets contiguously.
LLVM_ABI char & PostRAMachineSinkingID
This pass perform post-ra machine sink for COPY instructions.
CodeGenOptLevel
Code generation optimization level.
Definition CodeGen.h:82
Target & getTheSPIRV64Target()
ModulePass * createSPIRVLegalizeZeroSizeArraysPass(const SPIRVTargetMachine &TM)
LLVM_ABI FunctionPass * createStripConvergenceIntrinsicsPass()
void initializeSPIRVPostLegalizerPass(PassRegistry &)
void initializeSPIRVCBufferAccessLegacyPass(PassRegistry &)
ModulePass * createSPIRVCBufferAccessLegacyPass()
LLVM_ABI void initializeGlobalISel(PassRegistry &)
Initialize all passes linked into the GlobalISel library.
Target & getTheSPIRVLogicalTarget()
void initializeSPIRVAsmPrinterPass(PassRegistry &)
void initializeSPIRVRegularizerLegacyPass(PassRegistry &)
FunctionPass * createSPIRVRegularizerPass()
void initializeSPIRVStructurizerPass(PassRegistry &)
FunctionPass * createSPIRVMergeRegionExitTargetsPass()
LLVM_ABI FunctionPass * createInferAddressSpacesPass(unsigned AddressSpace=~0u)
void initializeSPIRVPreLegalizerPass(PassRegistry &)
void initializeSPIRVConvergenceRegionAnalysisWrapperPassPass(PassRegistry &)
LLVM_ABI char & MachineBlockPlacementID
MachineBlockPlacement - This pass places basic blocks based on branch probabilities.
LLVM_ABI FunctionPass * createAtomicExpandLegacyPass()
AtomicExpandPass - At IR level this pass replace atomic instructions with __atomic_* library calls,...
LLVM_ABI char & BranchFolderPassID
BranchFolding - This pass performs machine code CFG based optimizations to delete branches to branche...
ModulePass * createSPIRVPrepareFunctionsPass(const SPIRVTargetMachine &TM)
void initializeSPIRVPrepareGlobalsPass(PassRegistry &)
FunctionPass * createSPIRVLegalizePointerCastPass(SPIRVTargetMachine *TM)
LLVM_ABI Pass * createLoopSimplifyPass()
void initializeSPIRVLegalizeImplicitBindingLegacyPass(PassRegistry &)
void initializeSPIRVCtorDtorLoweringLegacyPass(PassRegistry &)
LLVM_ABI char & MachineCopyPropagationID
MachineCopyPropagation - This pass performs copy propagation on machine instructions.
void initializeSPIRVMergeRegionExitTargetsLegacyPass(PassRegistry &)
ModulePass * createSPIRVLegalizeImplicitBindingPass()
Implement std::hash so that hash_code can be used in STL containers.
Definition BitVector.h:870
RegisterTargetMachine - Helper template for registering a target machine implementation,...