181ad6265SDimitry Andric //===- DXILWriterPass.cpp - Bitcode writing pass --------------------------===//
281ad6265SDimitry Andric //
381ad6265SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
481ad6265SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
581ad6265SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
681ad6265SDimitry Andric //
781ad6265SDimitry Andric //===----------------------------------------------------------------------===//
881ad6265SDimitry Andric //
981ad6265SDimitry Andric // DXILWriterPass implementation.
1081ad6265SDimitry Andric //
1181ad6265SDimitry Andric //===----------------------------------------------------------------------===//
1281ad6265SDimitry Andric
1381ad6265SDimitry Andric #include "DXILWriterPass.h"
1481ad6265SDimitry Andric #include "DXILBitcodeWriter.h"
1581ad6265SDimitry Andric #include "llvm/ADT/DenseMap.h"
1681ad6265SDimitry Andric #include "llvm/ADT/StringRef.h"
1781ad6265SDimitry Andric #include "llvm/Analysis/ModuleSummaryAnalysis.h"
1881ad6265SDimitry Andric #include "llvm/IR/Constants.h"
1981ad6265SDimitry Andric #include "llvm/IR/GlobalVariable.h"
2081ad6265SDimitry Andric #include "llvm/IR/Module.h"
2181ad6265SDimitry Andric #include "llvm/IR/PassManager.h"
2281ad6265SDimitry Andric #include "llvm/InitializePasses.h"
2381ad6265SDimitry Andric #include "llvm/Pass.h"
2481ad6265SDimitry Andric #include "llvm/Support/Alignment.h"
2581ad6265SDimitry Andric #include "llvm/Transforms/Utils/ModuleUtils.h"
2681ad6265SDimitry Andric
2781ad6265SDimitry Andric using namespace llvm;
2881ad6265SDimitry Andric using namespace llvm::dxil;
2981ad6265SDimitry Andric
3081ad6265SDimitry Andric namespace {
3181ad6265SDimitry Andric class WriteDXILPass : public llvm::ModulePass {
3281ad6265SDimitry Andric raw_ostream &OS; // raw_ostream to print on
3381ad6265SDimitry Andric
3481ad6265SDimitry Andric public:
3581ad6265SDimitry Andric static char ID; // Pass identification, replacement for typeid
WriteDXILPass()3681ad6265SDimitry Andric WriteDXILPass() : ModulePass(ID), OS(dbgs()) {
3781ad6265SDimitry Andric initializeWriteDXILPassPass(*PassRegistry::getPassRegistry());
3881ad6265SDimitry Andric }
3981ad6265SDimitry Andric
WriteDXILPass(raw_ostream & o)4081ad6265SDimitry Andric explicit WriteDXILPass(raw_ostream &o) : ModulePass(ID), OS(o) {
4181ad6265SDimitry Andric initializeWriteDXILPassPass(*PassRegistry::getPassRegistry());
4281ad6265SDimitry Andric }
4381ad6265SDimitry Andric
getPassName() const4481ad6265SDimitry Andric StringRef getPassName() const override { return "Bitcode Writer"; }
4581ad6265SDimitry Andric
runOnModule(Module & M)4681ad6265SDimitry Andric bool runOnModule(Module &M) override {
4781ad6265SDimitry Andric WriteDXILToFile(M, OS);
4881ad6265SDimitry Andric return false;
4981ad6265SDimitry Andric }
getAnalysisUsage(AnalysisUsage & AU) const5081ad6265SDimitry Andric void getAnalysisUsage(AnalysisUsage &AU) const override {
5181ad6265SDimitry Andric AU.setPreservesAll();
5281ad6265SDimitry Andric }
5381ad6265SDimitry Andric };
5481ad6265SDimitry Andric
5581ad6265SDimitry Andric class EmbedDXILPass : public llvm::ModulePass {
5681ad6265SDimitry Andric public:
5781ad6265SDimitry Andric static char ID; // Pass identification, replacement for typeid
EmbedDXILPass()5881ad6265SDimitry Andric EmbedDXILPass() : ModulePass(ID) {
5981ad6265SDimitry Andric initializeEmbedDXILPassPass(*PassRegistry::getPassRegistry());
6081ad6265SDimitry Andric }
6181ad6265SDimitry Andric
getPassName() const6281ad6265SDimitry Andric StringRef getPassName() const override { return "DXIL Embedder"; }
6381ad6265SDimitry Andric
runOnModule(Module & M)6481ad6265SDimitry Andric bool runOnModule(Module &M) override {
6581ad6265SDimitry Andric std::string Data;
6681ad6265SDimitry Andric llvm::raw_string_ostream OS(Data);
67*bdd1243dSDimitry Andric
68*bdd1243dSDimitry Andric const std::string OriginalTriple = M.getTargetTriple();
69*bdd1243dSDimitry Andric // Set to DXIL triple when write to bitcode.
70*bdd1243dSDimitry Andric // Only the output bitcode need to be DXIL triple.
71*bdd1243dSDimitry Andric M.setTargetTriple("dxil-ms-dx");
72*bdd1243dSDimitry Andric
7381ad6265SDimitry Andric WriteDXILToFile(M, OS);
7481ad6265SDimitry Andric
75*bdd1243dSDimitry Andric // Recover triple.
76*bdd1243dSDimitry Andric M.setTargetTriple(OriginalTriple);
77*bdd1243dSDimitry Andric
7881ad6265SDimitry Andric Constant *ModuleConstant =
7981ad6265SDimitry Andric ConstantDataArray::get(M.getContext(), arrayRefFromStringRef(Data));
8081ad6265SDimitry Andric auto *GV = new llvm::GlobalVariable(M, ModuleConstant->getType(), true,
8181ad6265SDimitry Andric GlobalValue::PrivateLinkage,
8281ad6265SDimitry Andric ModuleConstant, "dx.dxil");
8381ad6265SDimitry Andric GV->setSection("DXIL");
8481ad6265SDimitry Andric GV->setAlignment(Align(4));
8581ad6265SDimitry Andric appendToCompilerUsed(M, {GV});
8681ad6265SDimitry Andric return true;
8781ad6265SDimitry Andric }
8881ad6265SDimitry Andric
getAnalysisUsage(AnalysisUsage & AU) const8981ad6265SDimitry Andric void getAnalysisUsage(AnalysisUsage &AU) const override {
9081ad6265SDimitry Andric AU.setPreservesAll();
9181ad6265SDimitry Andric }
9281ad6265SDimitry Andric };
9381ad6265SDimitry Andric } // namespace
9481ad6265SDimitry Andric
9581ad6265SDimitry Andric char WriteDXILPass::ID = 0;
96*bdd1243dSDimitry Andric INITIALIZE_PASS_BEGIN(WriteDXILPass, "dxil-write-bitcode", "Write Bitcode",
97*bdd1243dSDimitry Andric false, true)
INITIALIZE_PASS_DEPENDENCY(ModuleSummaryIndexWrapperPass)9881ad6265SDimitry Andric INITIALIZE_PASS_DEPENDENCY(ModuleSummaryIndexWrapperPass)
99*bdd1243dSDimitry Andric INITIALIZE_PASS_END(WriteDXILPass, "dxil-write-bitcode", "Write Bitcode", false,
10081ad6265SDimitry Andric true)
10181ad6265SDimitry Andric
10281ad6265SDimitry Andric ModulePass *llvm::createDXILWriterPass(raw_ostream &Str) {
10381ad6265SDimitry Andric return new WriteDXILPass(Str);
10481ad6265SDimitry Andric }
10581ad6265SDimitry Andric
10681ad6265SDimitry Andric char EmbedDXILPass::ID = 0;
10781ad6265SDimitry Andric INITIALIZE_PASS(EmbedDXILPass, "dxil-embed", "Embed DXIL", false, true)
10881ad6265SDimitry Andric
createDXILEmbedderPass()10981ad6265SDimitry Andric ModulePass *llvm::createDXILEmbedderPass() { return new EmbedDXILPass(); }
110