1*c8fbf6ffSEugene Zelenko //===- AMDGPUUnifyMetadata.cpp - Unify OpenCL metadata --------------------===// 250ea93a2SStanislav Mekhanoshin // 350ea93a2SStanislav Mekhanoshin // The LLVM Compiler Infrastructure 450ea93a2SStanislav Mekhanoshin // 550ea93a2SStanislav Mekhanoshin // This file is distributed under the University of Illinois Open Source 650ea93a2SStanislav Mekhanoshin // License. See LICENSE.TXT for details. 750ea93a2SStanislav Mekhanoshin // 850ea93a2SStanislav Mekhanoshin //===----------------------------------------------------------------------===// 950ea93a2SStanislav Mekhanoshin // 1050ea93a2SStanislav Mekhanoshin // \file 1150ea93a2SStanislav Mekhanoshin // \brief This pass that unifies multiple OpenCL metadata due to linking. 1250ea93a2SStanislav Mekhanoshin // 1350ea93a2SStanislav Mekhanoshin //===----------------------------------------------------------------------===// 1450ea93a2SStanislav Mekhanoshin 1550ea93a2SStanislav Mekhanoshin #include "AMDGPU.h" 16734bb7bbSEugene Zelenko #include "llvm/ADT/SmallVector.h" 17734bb7bbSEugene Zelenko #include "llvm/ADT/StringRef.h" 1850ea93a2SStanislav Mekhanoshin #include "llvm/IR/Constants.h" 19*c8fbf6ffSEugene Zelenko #include "llvm/IR/Metadata.h" 2050ea93a2SStanislav Mekhanoshin #include "llvm/IR/Module.h" 2150ea93a2SStanislav Mekhanoshin #include "llvm/Pass.h" 22734bb7bbSEugene Zelenko #include <algorithm> 23734bb7bbSEugene Zelenko #include <cassert> 2450ea93a2SStanislav Mekhanoshin 2550ea93a2SStanislav Mekhanoshin using namespace llvm; 2650ea93a2SStanislav Mekhanoshin 2750ea93a2SStanislav Mekhanoshin namespace { 28734bb7bbSEugene Zelenko 2950ea93a2SStanislav Mekhanoshin namespace kOCLMD { 30734bb7bbSEugene Zelenko 3150ea93a2SStanislav Mekhanoshin const char SpirVer[] = "opencl.spir.version"; 3250ea93a2SStanislav Mekhanoshin const char OCLVer[] = "opencl.ocl.version"; 3350ea93a2SStanislav Mekhanoshin const char UsedExt[] = "opencl.used.extensions"; 3450ea93a2SStanislav Mekhanoshin const char UsedOptCoreFeat[] = "opencl.used.optional.core.features"; 3550ea93a2SStanislav Mekhanoshin const char CompilerOptions[] = "opencl.compiler.options"; 3650ea93a2SStanislav Mekhanoshin const char LLVMIdent[] = "llvm.ident"; 37734bb7bbSEugene Zelenko 38734bb7bbSEugene Zelenko } // end namespace kOCLMD 3950ea93a2SStanislav Mekhanoshin 4050ea93a2SStanislav Mekhanoshin /// \brief Unify multiple OpenCL metadata due to linking. 41f6c1feb8SStanislav Mekhanoshin class AMDGPUUnifyMetadata : public ModulePass { 4250ea93a2SStanislav Mekhanoshin public: 4350ea93a2SStanislav Mekhanoshin static char ID; 44*c8fbf6ffSEugene Zelenko 45*c8fbf6ffSEugene Zelenko explicit AMDGPUUnifyMetadata() : ModulePass(ID) {} 4650ea93a2SStanislav Mekhanoshin 4750ea93a2SStanislav Mekhanoshin private: 48*c8fbf6ffSEugene Zelenko bool runOnModule(Module &M) override; 4950ea93a2SStanislav Mekhanoshin 5050ea93a2SStanislav Mekhanoshin /// \brief Unify version metadata. 5150ea93a2SStanislav Mekhanoshin /// \return true if changes are made. 5250ea93a2SStanislav Mekhanoshin /// Assume the named metadata has operands each of which is a pair of 5350ea93a2SStanislav Mekhanoshin /// integer constant, e.g. 5450ea93a2SStanislav Mekhanoshin /// !Name = {!n1, !n2} 5550ea93a2SStanislav Mekhanoshin /// !n1 = {i32 1, i32 2} 5650ea93a2SStanislav Mekhanoshin /// !n2 = {i32 2, i32 0} 5750ea93a2SStanislav Mekhanoshin /// Keep the largest version as the sole operand if PickFirst is false. 5850ea93a2SStanislav Mekhanoshin /// Otherwise pick it from the first value, representing kernel module. 5950ea93a2SStanislav Mekhanoshin bool unifyVersionMD(Module &M, StringRef Name, bool PickFirst) { 6050ea93a2SStanislav Mekhanoshin auto NamedMD = M.getNamedMetadata(Name); 6150ea93a2SStanislav Mekhanoshin if (!NamedMD || NamedMD->getNumOperands() <= 1) 6250ea93a2SStanislav Mekhanoshin return false; 6350ea93a2SStanislav Mekhanoshin MDNode *MaxMD = nullptr; 6450ea93a2SStanislav Mekhanoshin auto MaxVer = 0U; 6550ea93a2SStanislav Mekhanoshin for (const auto &VersionMD : NamedMD->operands()) { 6650ea93a2SStanislav Mekhanoshin assert(VersionMD->getNumOperands() == 2); 6750ea93a2SStanislav Mekhanoshin auto CMajor = mdconst::extract<ConstantInt>(VersionMD->getOperand(0)); 6850ea93a2SStanislav Mekhanoshin auto VersionMajor = CMajor->getZExtValue(); 6950ea93a2SStanislav Mekhanoshin auto CMinor = mdconst::extract<ConstantInt>(VersionMD->getOperand(1)); 7050ea93a2SStanislav Mekhanoshin auto VersionMinor = CMinor->getZExtValue(); 7150ea93a2SStanislav Mekhanoshin auto Ver = (VersionMajor * 100) + (VersionMinor * 10); 7250ea93a2SStanislav Mekhanoshin if (Ver > MaxVer) { 7350ea93a2SStanislav Mekhanoshin MaxVer = Ver; 7450ea93a2SStanislav Mekhanoshin MaxMD = VersionMD; 7550ea93a2SStanislav Mekhanoshin } 7650ea93a2SStanislav Mekhanoshin if (PickFirst) 7750ea93a2SStanislav Mekhanoshin break; 7850ea93a2SStanislav Mekhanoshin } 7950ea93a2SStanislav Mekhanoshin NamedMD->eraseFromParent(); 8050ea93a2SStanislav Mekhanoshin NamedMD = M.getOrInsertNamedMetadata(Name); 8150ea93a2SStanislav Mekhanoshin NamedMD->addOperand(MaxMD); 8250ea93a2SStanislav Mekhanoshin return true; 8350ea93a2SStanislav Mekhanoshin } 8450ea93a2SStanislav Mekhanoshin 8550ea93a2SStanislav Mekhanoshin /// \brief Unify version metadata. 8650ea93a2SStanislav Mekhanoshin /// \return true if changes are made. 8750ea93a2SStanislav Mekhanoshin /// Assume the named metadata has operands each of which is a list e.g. 8850ea93a2SStanislav Mekhanoshin /// !Name = {!n1, !n2} 8950ea93a2SStanislav Mekhanoshin /// !n1 = !{!"cl_khr_fp16", {!"cl_khr_fp64"}} 9050ea93a2SStanislav Mekhanoshin /// !n2 = !{!"cl_khr_image"} 9150ea93a2SStanislav Mekhanoshin /// Combine it into a single list with unique operands. 9250ea93a2SStanislav Mekhanoshin bool unifyExtensionMD(Module &M, StringRef Name) { 9350ea93a2SStanislav Mekhanoshin auto NamedMD = M.getNamedMetadata(Name); 9450ea93a2SStanislav Mekhanoshin if (!NamedMD || NamedMD->getNumOperands() == 1) 9550ea93a2SStanislav Mekhanoshin return false; 9650ea93a2SStanislav Mekhanoshin 9750ea93a2SStanislav Mekhanoshin SmallVector<Metadata *, 4> All; 9850ea93a2SStanislav Mekhanoshin for (const auto &MD : NamedMD->operands()) 9950ea93a2SStanislav Mekhanoshin for (const auto &Op : MD->operands()) 10050ea93a2SStanislav Mekhanoshin if (std::find(All.begin(), All.end(), Op.get()) == All.end()) 10150ea93a2SStanislav Mekhanoshin All.push_back(Op.get()); 10250ea93a2SStanislav Mekhanoshin 10350ea93a2SStanislav Mekhanoshin NamedMD->eraseFromParent(); 10450ea93a2SStanislav Mekhanoshin NamedMD = M.getOrInsertNamedMetadata(Name); 105980688cdSKonstantin Zhuravlyov for (const auto &MD : All) 106980688cdSKonstantin Zhuravlyov NamedMD->addOperand(MDNode::get(M.getContext(), MD)); 107980688cdSKonstantin Zhuravlyov 10850ea93a2SStanislav Mekhanoshin return true; 10950ea93a2SStanislav Mekhanoshin } 11050ea93a2SStanislav Mekhanoshin }; 11150ea93a2SStanislav Mekhanoshin 11250ea93a2SStanislav Mekhanoshin } // end anonymous namespace 11350ea93a2SStanislav Mekhanoshin 11450ea93a2SStanislav Mekhanoshin char AMDGPUUnifyMetadata::ID = 0; 11550ea93a2SStanislav Mekhanoshin 11650ea93a2SStanislav Mekhanoshin char &llvm::AMDGPUUnifyMetadataID = AMDGPUUnifyMetadata::ID; 11750ea93a2SStanislav Mekhanoshin 11850ea93a2SStanislav Mekhanoshin INITIALIZE_PASS(AMDGPUUnifyMetadata, "amdgpu-unify-metadata", 11950ea93a2SStanislav Mekhanoshin "Unify multiple OpenCL metadata due to linking", 12050ea93a2SStanislav Mekhanoshin false, false) 12150ea93a2SStanislav Mekhanoshin 122f6c1feb8SStanislav Mekhanoshin ModulePass* llvm::createAMDGPUUnifyMetadataPass() { 12350ea93a2SStanislav Mekhanoshin return new AMDGPUUnifyMetadata(); 12450ea93a2SStanislav Mekhanoshin } 12550ea93a2SStanislav Mekhanoshin 12650ea93a2SStanislav Mekhanoshin bool AMDGPUUnifyMetadata::runOnModule(Module &M) { 12750ea93a2SStanislav Mekhanoshin const char* Vers[] = { 12850ea93a2SStanislav Mekhanoshin kOCLMD::SpirVer, 12950ea93a2SStanislav Mekhanoshin kOCLMD::OCLVer 13050ea93a2SStanislav Mekhanoshin }; 13150ea93a2SStanislav Mekhanoshin const char* Exts[] = { 13250ea93a2SStanislav Mekhanoshin kOCLMD::UsedExt, 13350ea93a2SStanislav Mekhanoshin kOCLMD::UsedOptCoreFeat, 13450ea93a2SStanislav Mekhanoshin kOCLMD::CompilerOptions, 13550ea93a2SStanislav Mekhanoshin kOCLMD::LLVMIdent 13650ea93a2SStanislav Mekhanoshin }; 13750ea93a2SStanislav Mekhanoshin 13850ea93a2SStanislav Mekhanoshin bool Changed = false; 13950ea93a2SStanislav Mekhanoshin 14050ea93a2SStanislav Mekhanoshin for (auto &I : Vers) 14150ea93a2SStanislav Mekhanoshin Changed |= unifyVersionMD(M, I, true); 14250ea93a2SStanislav Mekhanoshin 14350ea93a2SStanislav Mekhanoshin for (auto &I : Exts) 14450ea93a2SStanislav Mekhanoshin Changed |= unifyExtensionMD(M, I); 14550ea93a2SStanislav Mekhanoshin 14650ea93a2SStanislav Mekhanoshin return Changed; 14750ea93a2SStanislav Mekhanoshin } 148