LLVM 23.0.0git
DXILWriterPass.cpp
Go to the documentation of this file.
1//===- DXILWriterPass.cpp - Bitcode writing pass --------------------------===//
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// DXILWriterPass implementation.
10//
11//===----------------------------------------------------------------------===//
12
13#include "DXILWriterPass.h"
14#include "DXILBitcodeWriter.h"
16#include "llvm/ADT/DenseMap.h"
17#include "llvm/ADT/STLExtras.h"
18#include "llvm/ADT/StringRef.h"
20#include "llvm/IR/Constants.h"
21#include "llvm/IR/DebugInfo.h"
25#include "llvm/IR/Intrinsics.h"
26#include "llvm/IR/LLVMContext.h"
27#include "llvm/IR/Module.h"
28#include "llvm/IR/PassManager.h"
30#include "llvm/Pass.h"
34
35using namespace llvm;
36using namespace llvm::dxil;
37
38namespace {
39class WriteDXILPass : public llvm::ModulePass {
40 raw_ostream &OS; // raw_ostream to print on
41
42public:
43 static char ID; // Pass identification, replacement for typeid
44 WriteDXILPass() : ModulePass(ID), OS(dbgs()) {
46 }
47
48 explicit WriteDXILPass(raw_ostream &o) : ModulePass(ID), OS(o) {
50 }
51
52 StringRef getPassName() const override { return "Bitcode Writer"; }
53
54 bool runOnModule(Module &M) override {
55 const auto DIMap = DXILDebugInfoPass::run(M);
56 WriteDXILToFile(M, OS, DIMap);
57 return false;
58 }
59 void getAnalysisUsage(AnalysisUsage &AU) const override {
60 AU.setPreservesAll();
61 }
62};
63
64static void legalizeLifetimeIntrinsics(Module &M) {
65 LLVMContext &Ctx = M.getContext();
66 Type *I64Ty = IntegerType::get(Ctx, 64);
67 Type *PtrTy = PointerType::get(Ctx, 0);
68 Intrinsic::ID LifetimeIIDs[2] = {Intrinsic::lifetime_start,
69 Intrinsic::lifetime_end};
70 for (Intrinsic::ID &IID : LifetimeIIDs) {
71 Function *F = M.getFunction(Intrinsic::getName(IID, {PtrTy}, &M));
72 if (!F)
73 continue;
74
75 // Get or insert an LLVM 3.7-compliant lifetime intrinsic function of the
76 // form `void @llvm.lifetime.[start/end](i64, ptr)` with the NoUnwind
77 // attribute
78 AttributeList Attr;
79 Attr = Attr.addFnAttribute(Ctx, Attribute::NoUnwind);
80 FunctionCallee LifetimeCallee = M.getOrInsertFunction(
81 Intrinsic::getBaseName(IID), Attr, Type::getVoidTy(Ctx), I64Ty, PtrTy);
82
83 // Replace all calls to lifetime intrinsics with calls to the
84 // LLVM 3.7-compliant version of the lifetime intrinsic
85 for (User *U : make_early_inc_range(F->users())) {
87 assert(CI &&
88 "Expected user of a lifetime intrinsic function to be a CallInst");
89
90 // LLVM 3.7 lifetime intrinics require an i8* operand, so we insert
91 // a bitcast to ensure that is the case
92 Value *PtrOperand = CI->getArgOperand(0);
93 PointerType *PtrOpPtrTy = cast<PointerType>(PtrOperand->getType());
94 Value *NoOpBitCast = CastInst::Create(Instruction::BitCast, PtrOperand,
95 PtrOpPtrTy, "", CI->getIterator());
96
97 // LLVM 3.7 lifetime intrinsics have an explicit size operand, whose value
98 // we can obtain from the pointer operand which must be an AllocaInst (as
99 // of https://github.com/llvm/llvm-project/pull/149310)
100 AllocaInst *AI = dyn_cast<AllocaInst>(PtrOperand);
101 assert(AI &&
102 "The pointer operand of a lifetime intrinsic call must be an "
103 "AllocaInst");
104 std::optional<TypeSize> AllocSize =
106 assert(AllocSize.has_value() &&
107 "Expected the allocation size of AllocaInst to be known");
108 CallInst *NewCI = CallInst::Create(
109 LifetimeCallee,
110 {ConstantInt::get(I64Ty, AllocSize.value().getFixedValue()),
111 NoOpBitCast},
112 "", CI->getIterator());
114 NewCI->addParamAttr(1, ParamAttr);
115
116 CI->eraseFromParent();
117 }
118
119 F->eraseFromParent();
120 }
121}
122
123static void removeLifetimeIntrinsics(Module &M) {
124 Intrinsic::ID LifetimeIIDs[2] = {Intrinsic::lifetime_start,
125 Intrinsic::lifetime_end};
126 for (Intrinsic::ID &IID : LifetimeIIDs) {
127 Function *F = M.getFunction(Intrinsic::getBaseName(IID));
128 if (!F)
129 continue;
130
131 for (User *U : make_early_inc_range(F->users())) {
133 assert(CI && "Expected user of lifetime function to be a CallInst");
135 assert(BCI && "Expected pointer operand of CallInst to be a BitCastInst");
136 CI->eraseFromParent();
137 BCI->eraseFromParent();
138 }
139 F->eraseFromParent();
140 }
141}
142
143static void replaceNamedMetadataArray(Module &M, StringRef Name,
144 ArrayRef<Metadata *> NewOps) {
145 NamedMDNode *NMD = M.getNamedMetadata(Name);
146 if (!NMD)
147 return;
148 NMD->eraseFromParent();
149 M.getOrInsertNamedMetadata(Name)->addOperand(
150 MDTuple::get(M.getContext(), NewOps));
151}
152
153class EmbedDXILPass : public llvm::ModulePass {
154 std::string writeModule(Module &M, bool HasDebugInfo, bool WriteDebug) {
155 std::string Data;
156 llvm::raw_string_ostream OS(Data);
157
158 if (HasDebugInfo) {
159 if (WriteDebug) {
160 // Replace dx.source metadata nodes with stubs.
161 // TODO: Add /Qsource_in_debug_module flag to enable/disable this.
162 LLVMContext &Ctx = M.getContext();
163 MDString *EmptyString = MDString::get(Ctx, "");
164 replaceNamedMetadataArray(M, "dx.source.contents",
165 {EmptyString, EmptyString});
166 replaceNamedMetadataArray(M, "dx.source.defines", {});
167 replaceNamedMetadataArray(M, "dx.source.mainFileName", {EmptyString});
168 replaceNamedMetadataArray(M, "dx.source.args", {});
169 } else {
170 // If we have an ILDB part, strip DXIL from all debug info.
172
173 // Also, manually remove debug version flags and dx.source nodes.
174 if (NamedMDNode *Flags = M.getModuleFlagsMetadata()) {
176 M.getModuleFlagsMetadata(FlagEntries);
177 Flags->eraseFromParent();
178 for (llvm::Module::ModuleFlagEntry &Entry : FlagEntries) {
179 if (Entry.Key->getString() == "Dwarf Version" ||
180 Entry.Key->getString() == "Debug Info Version") {
181 continue;
182 }
183 M.addModuleFlag(Entry.Behavior, Entry.Key->getString(), Entry.Val);
184 }
185 }
186 for (NamedMDNode &NMD : llvm::make_early_inc_range(M.named_metadata()))
187 if (NMD.getName().starts_with("dx.source"))
188 NMD.eraseFromParent();
189 }
190 } else {
191#ifdef EXPENSIVE_CHECKS
192 assert(
193 StripDebugInfo(M) == false &&
194 "The module must not contain any debug info here."
195 "Shader modules with debug info must have !DICompileUnit metadata.");
196#endif
197 }
198 const auto DIMap = DXILDebugInfoPass::run(M);
199 WriteDXILToFile(M, OS, DIMap);
200 return Data;
201 }
202
203 GlobalVariable *createSectionGlobal(Module &M, StringRef Data,
204 StringRef GlobalName,
205 StringRef SectionName) {
206 Constant *ModuleConstant =
208 auto *GV = new llvm::GlobalVariable(M, ModuleConstant->getType(), true,
210 ModuleConstant, GlobalName);
211 GV->setSection(SectionName);
212 GV->setAlignment(Align(4));
213 return GV;
214 }
215
216public:
217 static char ID; // Pass identification, replacement for typeid
218 EmbedDXILPass() : ModulePass(ID) {
220 }
221
222 StringRef getPassName() const override { return "DXIL Embedder"; }
223
224 bool runOnModule(Module &M) override {
225 // Perform late legalization of lifetime intrinsics that would otherwise
226 // fail the Module Verifier if performed in an earlier pass
227 legalizeLifetimeIntrinsics(M);
228
229 bool HasDebugInfo = !M.debug_compile_units().empty();
230 std::string ILDBData;
231 if (HasDebugInfo) {
232 // Write DXIL with debug info to ILDB part.
233 // Clone the module to avoid alternating it with DebugInfoPass
234 // before stripping the debug info later.
235 ILDBData =
236 writeModule(*llvm::CloneModule(M), HasDebugInfo, /*WriteDebug=*/true);
237 }
238
239 // Clone the module to save dx.source metadata nodes from stripping, as they
240 // are needed for DXILMetadataAnalysisWrapperPass.
241 std::string DXILData =
242 writeModule(*llvm::CloneModule(M), HasDebugInfo, /*WriteDebug=*/false);
243
244 // We no longer need lifetime intrinsics after bitcode serialization, so we
245 // simply remove them to keep the Module Verifier happy after our
246 // not-so-legal legalizations
247 removeLifetimeIntrinsics(M);
248
250 if (HasDebugInfo) {
251 // Create a GV after both parts are written, otherwise it gets
252 // added to DXIL when `writeModule` is called the second time.
253 Globals.emplace_back(createSectionGlobal(M, ILDBData, "dx.ildb", "ILDB"));
254 }
255 Globals.emplace_back(createSectionGlobal(M, DXILData, "dx.dxil", "DXIL"));
256 appendToCompilerUsed(M, Globals);
257 return true;
258 }
259
260 void getAnalysisUsage(AnalysisUsage &AU) const override {
261 AU.setPreservesAll();
262 }
263};
264} // namespace
265
266char WriteDXILPass::ID = 0;
267INITIALIZE_PASS_BEGIN(WriteDXILPass, "dxil-write-bitcode", "Write Bitcode",
268 false, true)
270INITIALIZE_PASS_END(WriteDXILPass, "dxil-write-bitcode", "Write Bitcode", false,
271 true)
272
274 return new WriteDXILPass(Str);
275}
276
277char EmbedDXILPass::ID = 0;
278INITIALIZE_PASS(EmbedDXILPass, "dxil-embed", "Embed DXIL", false, true)
279
280ModulePass *llvm::createDXILEmbedderPass() { return new EmbedDXILPass(); }
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
@ ParamAttr
This file contains the declarations for the subclasses of Constant, which represent the different fla...
This file provides a bitcode writing pass.
This file defines the DenseMap class.
Module.h This file contains the declarations for the Module class.
This header defines various interfaces for pass management in LLVM.
#define F(x, y, z)
Definition MD5.cpp:54
Machine Check Debug Module
This is the interface to build a ModuleSummaryIndex for a module.
#define INITIALIZE_PASS_DEPENDENCY(depName)
Definition PassSupport.h:42
#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
#define INITIALIZE_PASS(passName, arg, name, cfg, analysis)
Definition PassSupport.h:56
This file contains some templates that are useful if you are working with the STL at all.
an instruction to allocate memory on the stack
LLVM_ABI std::optional< TypeSize > getAllocationSize(const DataLayout &DL) const
Get allocation size in bytes.
void setPreservesAll()
Set by analyses that do not transform their input at all.
Represent a constant reference to an array (0 or more elements consecutively in memory),...
Definition ArrayRef.h:40
Functions, function parameters, and return types can have attributes to indicate how they should be t...
Definition Attributes.h:105
This class represents a no-op cast from one type to another.
AttributeSet getParamAttributes(unsigned ArgNo) const
Return the param attributes for this call.
Value * getArgOperand(unsigned i) const
void addParamAttr(unsigned ArgNo, Attribute::AttrKind Kind)
Adds the attribute to the indicated argument.
This class represents a function call, abstracting a target machine's calling convention.
static CallInst * Create(FunctionType *Ty, Value *F, const Twine &NameStr="", InsertPosition InsertBefore=nullptr)
static LLVM_ABI CastInst * Create(Instruction::CastOps, Value *S, Type *Ty, const Twine &Name="", InsertPosition InsertBefore=nullptr)
Provides a way to construct any of the CastInst subclasses using an opcode instead of the subclass's ...
static Constant * get(LLVMContext &Context, ArrayRef< ElementTy > Elts)
get() constructor - Return a constant with array type with an element count and element type matching...
Definition Constants.h:872
A handy container for a FunctionType+Callee-pointer pair, which can be passed around as a single enti...
@ PrivateLinkage
Like Internal, but omit from symbol table.
Definition GlobalValue.h:61
LLVM_ABI InstListType::iterator eraseFromParent()
This method unlinks 'this' from the containing basic block and deletes it.
LLVM_ABI const DataLayout & getDataLayout() const
Get the data layout of the module this instruction belongs to.
static LLVM_ABI IntegerType * get(LLVMContext &C, unsigned NumBits)
This static method is the primary way of constructing an IntegerType.
Definition Type.cpp:350
This is an important class for using LLVM in a threaded context.
Definition LLVMContext.h:68
static LLVM_ABI MDString * get(LLVMContext &Context, StringRef Str)
Definition Metadata.cpp:614
static MDTuple * get(LLVMContext &Context, ArrayRef< Metadata * > MDs)
Definition Metadata.h:1518
ModulePass class - This class is used to implement unstructured interprocedural optimizations and ana...
Definition Pass.h:255
Legacy wrapper pass to provide the ModuleSummaryIndex object.
A Module instance is used to store all the information related to an LLVM module.
Definition Module.h:67
A tuple of MDNodes.
Definition Metadata.h:1749
LLVM_ABI StringRef getName() const
LLVM_ABI void eraseFromParent()
Drop all references and remove the node from parent module.
static LLVM_ABI PassRegistry * getPassRegistry()
getPassRegistry - Access the global registry object, which is automatically initialized at applicatio...
static LLVM_ABI PointerType * get(Type *ElementType, unsigned AddressSpace)
This constructs a pointer to an object of the specified type in a numbered address space.
reference emplace_back(ArgTypes &&... Args)
Represent a constant reference to a string, i.e.
Definition StringRef.h:56
bool starts_with(StringRef Prefix) const
Check if this string starts with the given Prefix.
Definition StringRef.h:258
The instances of the Type class are immutable: once they are created, they are never changed.
Definition Type.h:46
static LLVM_ABI Type * getVoidTy(LLVMContext &C)
Definition Type.cpp:282
LLVM Value Representation.
Definition Value.h:75
Type * getType() const
All values are typed, get the type of this value.
Definition Value.h:255
self_iterator getIterator()
Definition ilist_node.h:123
This class implements an extremely fast bulk output stream that can only output to a stream.
Definition raw_ostream.h:53
constexpr char Align[]
Key for Kernel::Arg::Metadata::mAlign.
@ Entry
Definition COFF.h:862
LLVM_ABI StringRef getName(ID id)
Return the LLVM name for an intrinsic, such as "llvm.ppc.altivec.lvx".
LLVM_ABI StringRef getBaseName(ID id)
Return the LLVM name for an intrinsic, without encoded types for overloading, such as "llvm....
DXILDebugInfoMap run(Module &M)
void WriteDXILToFile(const Module &M, raw_ostream &Out, const DXILDebugInfoMap &DebugInfo)
Write the specified module to the specified raw output stream.
This is an optimization pass for GlobalISel generic memory operations.
ArrayRef< CharT > arrayRefFromStringRef(StringRef Input)
Construct an array ref of bytes from a string ref.
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
Definition Casting.h:643
ModulePass * createDXILWriterPass(raw_ostream &Str)
Create and return a pass that writes the module to the specified ostream.
iterator_range< early_inc_iterator_impl< detail::IterOfRange< RangeT > > > make_early_inc_range(RangeT &&Range)
Make a range that does early increment to allow mutation of the underlying range without disrupting i...
Definition STLExtras.h:633
LLVM_ABI size_t writeModule(const Module &M, uint8_t *Dest, size_t MaxSize)
Fuzzer friendly interface for the llvm bitcode printer.
LLVM_ABI raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
Definition Debug.cpp:209
void initializeEmbedDXILPassPass(PassRegistry &)
Initializer for dxil embedder pass.
class LLVM_GSL_OWNER SmallVector
Forward declaration of SmallVector so that calculateSmallVectorDefaultInlinedElements can reference s...
LLVM_ABI bool StripDebugInfo(Module &M)
Strip debug info in the module if it exists.
FunctionAddr VTableAddr uintptr_t uintptr_t Data
Definition InstrProf.h:221
LLVM_ABI void appendToCompilerUsed(Module &M, ArrayRef< GlobalValue * > Values)
Adds global values to the llvm.compiler.used list.
ModulePass * createDXILEmbedderPass()
Create and return a pass that writes the module to a global variable in the module for later emission...
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
Definition Casting.h:559
void initializeWriteDXILPassPass(PassRegistry &)
Initializer for dxil writer pass.
LLVM_ABI std::unique_ptr< Module > CloneModule(const Module &M)
Return an exact copy of the specified module.