LLVM 22.0.0git
SPIRVCBufferAccess.cpp
Go to the documentation of this file.
1//===- SPIRVCBufferAccess.cpp - Translate CBuffer Loads ---------*- 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// This pass replaces all accesses to constant buffer global variables with
10// accesses to the proper SPIR-V resource.
11//
12// The pass operates as follows:
13// 1. It finds all constant buffers by looking for the `!hlsl.cbs` metadata.
14// 2. For each cbuffer, it finds the global variable holding the resource handle
15// and the global variables for each of the cbuffer's members.
16// 3. For each member variable, it creates a call to the
17// `llvm.spv.resource.getpointer` intrinsic. This intrinsic takes the
18// resource handle and the member's index within the cbuffer as arguments.
19// The result is a pointer to that member within the SPIR-V resource.
20// 4. It then replaces all uses of the original member global variable with the
21// pointer returned by the `getpointer` intrinsic. This effectively retargets
22// all loads and GEPs to the new resource pointer.
23// 5. Finally, it cleans up by deleting the original global variables and the
24// `!hlsl.cbs` metadata.
25//
26// This approach allows subsequent passes, like SPIRVEmitIntrinsics, to
27// correctly handle GEPs that operate on the result of the `getpointer` call,
28// folding them into a single OpAccessChain instruction.
29//
30//===----------------------------------------------------------------------===//
31
32#include "SPIRVCBufferAccess.h"
33#include "SPIRV.h"
35#include "llvm/IR/IRBuilder.h"
36#include "llvm/IR/IntrinsicsSPIRV.h"
37#include "llvm/IR/Module.h"
39
40#define DEBUG_TYPE "spirv-cbuffer-access"
41using namespace llvm;
42
43// Finds the single instruction that defines the resource handle. This is
44// typically a call to `llvm.spv.resource.handlefrombinding`.
46 for (User *U : HandleVar->users()) {
47 if (auto *SI = dyn_cast<StoreInst>(U)) {
48 if (auto *I = dyn_cast<Instruction>(SI->getValueOperand())) {
49 return I;
50 }
51 }
52 }
53 return nullptr;
54}
55
57 std::optional<hlsl::CBufferMetadata> CBufMD =
59 if (auto *TET = dyn_cast<TargetExtType>(Ty))
60 return TET->getName() == "spirv.Padding";
61 return false;
62 });
63 if (!CBufMD)
64 return false;
65
66 SmallVector<Constant *> CBufferGlobals;
67 for (const hlsl::CBufferMapping &Mapping : *CBufMD)
68 for (const hlsl::CBufferMember &Member : Mapping.Members)
69 CBufferGlobals.push_back(Member.GV);
71
72 for (const hlsl::CBufferMapping &Mapping : *CBufMD) {
73 Instruction *HandleDef = findHandleDef(Mapping.Handle);
74 if (!HandleDef) {
75 report_fatal_error("Could not find handle definition for cbuffer: " +
76 Mapping.Handle->getName());
77 }
78
79 // The handle definition should dominate all uses of the cbuffer members.
80 // We'll insert our getpointer calls right after it.
81 IRBuilder<> Builder(HandleDef->getNextNode());
82
83 for (uint32_t Index = 0; Index < Mapping.Members.size(); ++Index) {
84 GlobalVariable *MemberGV = Mapping.Members[Index].GV;
85 if (MemberGV->use_empty()) {
86 continue;
87 }
88
89 // Create the getpointer intrinsic call.
90 Value *IndexVal = Builder.getInt32(Index);
91 Type *PtrType = MemberGV->getType();
92 Value *GetPointerCall = Builder.CreateIntrinsic(
93 PtrType, Intrinsic::spv_resource_getpointer, {HandleDef, IndexVal});
94
95 MemberGV->replaceAllUsesWith(GetPointerCall);
96 }
97 }
98
99 // Now that all uses are replaced, clean up the globals and metadata.
100 for (const hlsl::CBufferMapping &Mapping : *CBufMD) {
101 for (const auto &Member : Mapping.Members) {
102 Member.GV->eraseFromParent();
103 }
104 // Erase the stores to the handle variable before erasing the handle itself.
106 for (User *U : Mapping.Handle->users()) {
107 if (auto *SI = dyn_cast<StoreInst>(U)) {
108 HandleStores.push_back(SI);
109 }
110 }
111 for (Instruction *I : HandleStores) {
112 I->eraseFromParent();
113 }
114 Mapping.Handle->eraseFromParent();
115 }
116
117 CBufMD->eraseFromModule();
118 return true;
119}
120
128
129namespace {
130class SPIRVCBufferAccessLegacy : public ModulePass {
131public:
132 bool runOnModule(Module &M) override { return replaceCBufferAccesses(M); }
133 StringRef getPassName() const override { return "SPIRV CBuffer Access"; }
134 SPIRVCBufferAccessLegacy() : ModulePass(ID) {}
135
136 static char ID; // Pass identification.
137};
138char SPIRVCBufferAccessLegacy::ID = 0;
139} // end anonymous namespace
140
141INITIALIZE_PASS(SPIRVCBufferAccessLegacy, DEBUG_TYPE, "SPIRV CBuffer Access",
142 false, false)
143
145 return new SPIRVCBufferAccessLegacy();
146}
static bool replaceCBufferAccesses(Module &M)
#define DEBUG_TYPE
Module.h This file contains the declarations for the Module class.
#define I(x, y, z)
Definition MD5.cpp:57
#define INITIALIZE_PASS(passName, arg, name, cfg, analysis)
Definition PassSupport.h:56
static bool replaceCBufferAccesses(Module &M)
static Instruction * findHandleDef(GlobalVariable *HandleVar)
PointerType * getType() const
Global values are always pointers.
This provides a uniform API for creating instructions and inserting them into a basic block: either a...
Definition IRBuilder.h:2788
ModulePass class - This class is used to implement unstructured interprocedural optimizations and ana...
Definition Pass.h:255
A Module instance is used to store all the information related to an LLVM module.
Definition Module.h:67
A set of analyses that are preserved following a run of a transformation pass.
Definition Analysis.h:112
static PreservedAnalyses none()
Convenience factory function for the empty preserved set.
Definition Analysis.h:115
static PreservedAnalyses all()
Construct a special preserved set that preserves all passes.
Definition Analysis.h:118
PreservedAnalyses run(Module &M, ModuleAnalysisManager &AM)
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
StringRef - Represent a constant reference to a string, i.e.
Definition StringRef.h:55
The instances of the Type class are immutable: once they are created, they are never changed.
Definition Type.h:45
LLVM Value Representation.
Definition Value.h:75
LLVM_ABI void replaceAllUsesWith(Value *V)
Change all uses of this to point to a new Value.
Definition Value.cpp:546
iterator_range< user_iterator > users()
Definition Value.h:426
bool use_empty() const
Definition Value.h:346
static std::optional< CBufferMetadata > get(Module &M, llvm::function_ref< bool(Type *)> IsPadding)
Definition CBuffer.cpp:39
NodeTy * getNextNode()
Get the next node, or nullptr for the list tail.
Definition ilist_node.h:348
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.
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
Definition Casting.h:643
LLVM_ABI bool convertUsersOfConstantsToInstructions(ArrayRef< Constant * > Consts, Function *RestrictToFunc=nullptr, bool RemoveDeadConstants=true, bool IncludeSelf=false)
Replace constant expressions users of the given constants with instructions.
LLVM_ABI void report_fatal_error(Error Err, bool gen_crash_diag=true)
Definition Error.cpp:167
ModulePass * createSPIRVCBufferAccessLegacyPass()
AnalysisManager< Module > ModuleAnalysisManager
Convenience typedef for the Module analysis manager.
Definition MIRParser.h:39