From 69e21b4d6e3fdc4944ac579413101183e6974a47 Mon Sep 17 00:00:00 2001 From: Alexander Timofeev Date: Tue, 30 Aug 2016 17:17:51 +0300 Subject: [PATCH] AMDGPU printf runtime binding --- lib/Target/AMDGPU/AMDGPU.h | 4 + .../AMDGPU/AMDGPUPrinfRuntimeBinding.cpp | 688 ++++++++++++++++++ lib/Target/AMDGPU/AMDGPUTargetMachine.cpp | 7 + lib/Target/AMDGPU/AMDGPUTargetMachine.h | 1 + lib/Target/AMDGPU/CMakeLists.txt | 1 + test/CodeGen/AMDGPU/opencl-printf.ll | 59 ++ 6 files changed, 760 insertions(+) create mode 100644 lib/Target/AMDGPU/AMDGPUPrinfRuntimeBinding.cpp create mode 100644 test/CodeGen/AMDGPU/opencl-printf.ll diff --git a/lib/Target/AMDGPU/AMDGPU.h b/lib/Target/AMDGPU/AMDGPU.h index af39076727c0..166d511588cc 100644 --- a/lib/Target/AMDGPU/AMDGPU.h +++ b/lib/Target/AMDGPU/AMDGPU.h @@ -86,6 +86,10 @@ ModulePass *createAMDGPUAlwaysInlinePass(); ModulePass *createAMDGPUOpenCLImageTypeLoweringPass(); FunctionPass *createAMDGPUAnnotateUniformValues(); +ModulePass *createAMDGPUPrintfRuntimeBinding(); +void initializeAMDGPUPrintfRuntimeBindingPass(PassRegistry&); +extern char &AMDGPUPrintfRuntimeBindingID; + void initializeSIFixControlFlowLiveIntervalsPass(PassRegistry&); extern char &SIFixControlFlowLiveIntervalsID; diff --git a/lib/Target/AMDGPU/AMDGPUPrinfRuntimeBinding.cpp b/lib/Target/AMDGPU/AMDGPUPrinfRuntimeBinding.cpp new file mode 100644 index 000000000000..cf1aed5dae45 --- /dev/null +++ b/lib/Target/AMDGPU/AMDGPUPrinfRuntimeBinding.cpp @@ -0,0 +1,688 @@ +//=== AMDGPUPrintfRuntimeBinding.cpp -- For openCL -- bind Printfs to a kernel arg +// pointer that will be bound to a buffer later by the runtime ===// +//===----------------------------------------------------------------------===// +// March 2014. +// This pass traverses the functions in the module and converts +// each call to printf to a sequence of operations that +// store the following into the printf buffer : +// - format string (passed as a module's metadata unique ID) +// - bitwise copies of printf arguments +// The backend passes will need to store metadata in the kernel +//===----------------------------------------------------------------------===// + +#define DEBUG_TYPE "printfToRuntime" +#include "llvm/Analysis/AssumptionCache.h" +#include "llvm/Analysis/InstructionSimplify.h" +#include "llvm/IR/Constants.h" +#include "llvm/IR/DataLayout.h" +#include "llvm/IR/Dominators.h" +#include "llvm/IR/Module.h" +#include "llvm/IR/GlobalVariable.h" +#include "llvm/IR/Instructions.h" +#include "llvm/IR/IRBuilder.h" +#include "llvm/IR/Type.h" +#include "llvm/ADT/SmallString.h" +#include "llvm/ADT/StringExtras.h" +#include "llvm/ADT/Triple.h" +#include "llvm/CodeGen/Passes.h" +#include "llvm/Support/CommandLine.h" +#include "llvm/Support/Debug.h" +#include "llvm/Support/raw_ostream.h" +#include "llvm/Analysis/TargetLibraryInfo.h" +#include "llvm/Transforms/Utils/BasicBlockUtils.h" +#include "AMDGPU.h" +#define DWORD_ALIGN 4 +using namespace llvm; + +namespace { +class LLVM_LIBRARY_VISIBILITY AMDGPUPrintfRuntimeBinding : public ModulePass { +public: + static char ID; + explicit AMDGPUPrintfRuntimeBinding(); + SmallVector Printfs; + const char* getPassName() const; + bool runOnModule(Module &M); + bool doInitialization(Module &M); + bool doFinalization(Module &M); + void getConversionSpecifiers( + SmallVectorImpl &OpConvSpecifiers, + StringRef fmt, + size_t num_ops) const; + + bool shouldPrintAsStr(char Specifier, Type* OpType) const; + bool confirmSpirModule(Module& M) const; + bool confirmOpenCLVersion200(Module& M) const; + bool lowerPrintfForGpu(Module &M); + void collectPrintfsFromModule(Module &M); +private: + void getAnalysisUsage(AnalysisUsage &AU) const override { + AU.addRequired(); + AU.addRequired(); + AU.addRequired(); + } + + void initAnalysis(Module &M) { + TD = &M.getDataLayout(); + auto DTWP = getAnalysisIfAvailable(); + DT = DTWP ? &DTWP->getDomTree() : nullptr; + TLI = &getAnalysis().getTLI(); + } + + /// Prepare transformation. + /// \returns true if printf is found. + bool prepare(Module &M) { + collectPrintfsFromModule(M); + if (Printfs.empty()) + return false; + initAnalysis(M); + return true; + } + + Value *simplify(Instruction *I) { + auto AC = &getAnalysis().getAssumptionCache( + *I->getParent()->getParent()); + return SimplifyInstruction(I, *TD, TLI, DT, AC); + } + + const DataLayout *TD; + const DominatorTree *DT; + const TargetLibraryInfo *TLI; + static const int GlobalAddrspace = 1; +}; +} + +char AMDGPUPrintfRuntimeBinding::ID = 0; + +INITIALIZE_PASS_BEGIN(AMDGPUPrintfRuntimeBinding, "amdgpu-printf-runtime-binding", + "AMDGPU Printf lowering", false, false) +INITIALIZE_PASS_DEPENDENCY(AssumptionCacheTracker) +INITIALIZE_PASS_DEPENDENCY(TargetLibraryInfoWrapperPass) +INITIALIZE_PASS_DEPENDENCY(DominatorTreeWrapperPass) +INITIALIZE_PASS_END(AMDGPUPrintfRuntimeBinding, "amdgpu-printf-runtime-binding", + "AMDGPU Printf lowering", false, false) + +char &llvm::AMDGPUPrintfRuntimeBindingID = AMDGPUPrintfRuntimeBinding::ID; + +namespace llvm { +ModulePass *createAMDGPUPrintfRuntimeBinding() { + return new AMDGPUPrintfRuntimeBinding(); +} +} + +AMDGPUPrintfRuntimeBinding::AMDGPUPrintfRuntimeBinding() + : ModulePass(ID), TD(nullptr), DT(nullptr), TLI(nullptr) { + initializeAMDGPUPrintfRuntimeBindingPass(*PassRegistry::getPassRegistry()); +} + +bool AMDGPUPrintfRuntimeBinding::confirmOpenCLVersion200(Module& M) const { + NamedMDNode *OCLVersion = M.getNamedMetadata("opencl.ocl.version"); + if (!OCLVersion) { + return false; + } + if (OCLVersion->getNumOperands() != 1) { + return false; + } + MDNode *Ver = OCLVersion->getOperand(0); + if (Ver->getNumOperands() != 2) { + return false; + } + ConstantInt *Major = mdconst::dyn_extract(Ver->getOperand(0)); + ConstantInt *Minor = mdconst::dyn_extract(Ver->getOperand(1)); + if (0 == Major || 0 == Minor) { + return false; + } + if (Major->getZExtValue() == 2) { + return true; + } else { + return false; + } +} + +void AMDGPUPrintfRuntimeBinding::getConversionSpecifiers ( + SmallVectorImpl &OpConvSpecifiers, + StringRef Fmt, size_t NumOps) const { + // not all format characters are collected. + // At this time the format characters of interest + // are %p and %s, which use to know if we + // are either storing a literal string or a + // pointer to the printf buffer. + static const char ConvSpecifiers[] = "cdieEfgGaosuxXp"; + size_t CurFmtSpecifierIdx = 0; + size_t PrevFmtSpecifierIdx = 0; + + while ((CurFmtSpecifierIdx + = Fmt.find_first_of(ConvSpecifiers, CurFmtSpecifierIdx)) + != StringRef::npos) { + bool ArgDump = false; + StringRef CurFmt = Fmt.substr(PrevFmtSpecifierIdx, + CurFmtSpecifierIdx - PrevFmtSpecifierIdx); + size_t pTag = CurFmt.find_last_of("%"); + if (pTag != StringRef::npos) { + ArgDump = true; + while (pTag && CurFmt[--pTag] == '%') { + ArgDump = !ArgDump; + } + } + + if (ArgDump) { + OpConvSpecifiers.push_back(Fmt[CurFmtSpecifierIdx]); + } + + PrevFmtSpecifierIdx = ++CurFmtSpecifierIdx; + } +} + +bool AMDGPUPrintfRuntimeBinding::shouldPrintAsStr(char Specifier, + Type* OpType) const { + if (Specifier != 's') { + return false; + } + const PointerType *PT = dyn_cast(OpType); + if (!PT) { + return false; + } + if (PT->getAddressSpace() != 2) { + return false; + } + Type* ElemType = PT->getContainedType(0); + if (ElemType->getTypeID() != Type::IntegerTyID) { + return false; + } + IntegerType* ElemIType = cast(ElemType); + if (ElemIType->getBitWidth() == 8) { + return true; + } else { + return false; + } +} + +bool AMDGPUPrintfRuntimeBinding::confirmSpirModule(Module& M) const { + NamedMDNode *SPIRVersion = M.getNamedMetadata("opencl.spir.version"); + if (!SPIRVersion) return false; + else return true; +} + +void AMDGPUPrintfRuntimeBinding::collectPrintfsFromModule(Module& M) { + for (Module::iterator MF = M.begin(), E = M.end(); MF != E; ++MF) { + if (MF->isDeclaration()) continue; + BasicBlock::iterator CurInstr; + for (Function::iterator BB = MF->begin(), + MFE = MF->end(); BB != MFE; ++BB) { + for (BasicBlock::iterator Instr + = BB->begin(), instr_end = BB->end(); + Instr != instr_end;) { + CallInst *CI = dyn_cast(Instr); + CurInstr = Instr; + Instr++; + if (CI && CI->getCalledFunction() + && CI->getCalledFunction()->getName() == "printf") { + Printfs.push_back(CI); + } + } + } + } +} + +bool AMDGPUPrintfRuntimeBinding::lowerPrintfForGpu(Module &M) { + LLVMContext &Ctx = M.getContext(); + IRBuilder<> Builder(Ctx); + Type *I32Ty = Type::getInt32Ty(Ctx); + unsigned UniqID = 0; + // NB: This is important for this string size to be divizable by 4 + const char NonLiteralStr[4] = "???"; + + for (auto P : Printfs) { + CallInst* CI = dyn_cast(P); + + unsigned NumOps = CI->getNumArgOperands(); + + SmallString<16> OpConvSpecifiers; + Value *Op = CI->getArgOperand(0); + if (auto I = dyn_cast(Op)) + Op = simplify(I); + + ConstantExpr *ConstExpr = dyn_cast(Op); + + if (ConstExpr) { + GlobalVariable *GVar = dyn_cast( + ConstExpr->getOperand(0)); + + StringRef Str("unknown"); + if (GVar && GVar->hasInitializer()) { + ConstantDataArray *CA = dyn_cast( + GVar->getInitializer()); + if (CA->isString()) { + Str = CA->getAsCString(); + } + // + // we need this call to ascertain + // that we are printing a string + // or a pointer. It takes out the + // specifiers and fills up the first + // arg + getConversionSpecifiers(OpConvSpecifiers, Str, NumOps - 1); + } + // Add metadata for the string + std::string AStreamHolder; + raw_string_ostream Sizes(AStreamHolder); + int Sum = DWORD_ALIGN; + Sizes << CI->getNumArgOperands() -1; + Sizes << ':'; + for (unsigned ArgCount = 1; + ArgCount < CI->getNumArgOperands() + && ArgCount <= OpConvSpecifiers.size(); + ArgCount++) { + Value *Arg = CI->getArgOperand(ArgCount); + Type *ArgType = Arg->getType(); + unsigned ArgSize = TD->getTypeAllocSizeInBits(ArgType); + ArgSize = ArgSize/8; + // + // ArgSize by design should be a multiple of DWORD_ALIGN, + // expand the arguments that do not follow this rule. + // + if (ArgSize % DWORD_ALIGN != 0) { + llvm::Type* ResType = llvm::Type::getInt32Ty(Ctx); + VectorType* LLVMVecType = llvm::dyn_cast(ArgType); + int NumElem = LLVMVecType ? LLVMVecType->getNumElements() : 1; + if (LLVMVecType && NumElem > 1) + ResType = llvm::VectorType::get(ResType, NumElem); + Builder.SetInsertPoint(CI); + Builder.SetCurrentDebugLocation(CI->getDebugLoc()); + if (OpConvSpecifiers[ArgCount - 1] == 'x' || + OpConvSpecifiers[ArgCount - 1] == 'X' || + OpConvSpecifiers[ArgCount - 1] == 'u' || + OpConvSpecifiers[ArgCount - 1] == 'o') + Arg = Builder.CreateZExt(Arg, ResType); + else + Arg = Builder.CreateSExt(Arg, ResType); + ArgType = Arg->getType(); + ArgSize = TD->getTypeAllocSizeInBits(ArgType); + ArgSize = ArgSize / 8; + CI->setOperand(ArgCount, Arg); + } + if (OpConvSpecifiers[ArgCount - 1] == 'f') { + ConstantFP *FpCons = dyn_cast(Arg); + if (FpCons) + ArgSize = 4; + else { + FPExtInst *FpExt = dyn_cast(Arg); + if (FpExt && FpExt->getType()->isDoubleTy() && + FpExt->getOperand(0)->getType()->isFloatTy()) + ArgSize = 4; + } + } + if (shouldPrintAsStr(OpConvSpecifiers[ArgCount - 1], ArgType)) { + if (ConstantExpr *ConstExpr = dyn_cast(Arg)) { + GlobalVariable *GV + = dyn_cast(ConstExpr->getOperand(0)); + if (GV && GV->hasInitializer()) { + Constant *Init = GV->getInitializer(); + ConstantDataArray *CA = dyn_cast(Init); + if (Init->isZeroValue() || CA->isString()) { + size_t SizeStr = Init->isZeroValue() ? 1 : + (strlen(CA->getAsCString().data()) + 1); + size_t Rem = SizeStr % DWORD_ALIGN; + size_t NSizeStr = 0; + DEBUG(dbgs() << "Printf string original size = " << SizeStr << '\n'); + if (Rem) { + NSizeStr = SizeStr + (DWORD_ALIGN - Rem); + } else { + NSizeStr = SizeStr; + } + ArgSize = NSizeStr; + } + } else { + ArgSize = sizeof(NonLiteralStr); + } + } else { + ArgSize = sizeof(NonLiteralStr); + } + } + DEBUG(dbgs() << "Printf ArgSize (in buffer) = " + << ArgSize << " for type: " << *ArgType << '\n'); + Sizes << ArgSize << ':'; + Sum += ArgSize; + } + DEBUG(dbgs() << "Printf format string in source = " + << Str.str() << '\n'); + for (size_t I = 0; I < Str.size(); ++I) { + // Rest of the C escape sequences (e.g. \') are handled correctly + // by the MDParser + switch (Str[I]) { + case '\a': + Sizes << "\\a"; + break; + case '\b': + Sizes << "\\b"; + break; + case '\f': + Sizes << "\\f"; + break; + case '\n': + Sizes << "\\n"; + break; + case '\r': + Sizes << "\\r"; + break; + case '\v': + Sizes << "\\v"; + break; + case ':': + // ':' cannot be scanned by Flex, as it is defined as a delimiter + // Replace it with it's octal representation \72 + Sizes << "\\72"; + break; + default: + Sizes << Str[I]; + break; + } + } + + // Insert the printf_alloc call + Builder.SetInsertPoint(CI); + Builder.SetCurrentDebugLocation(CI->getDebugLoc()); + + AttributeSet Attr = AttributeSet::get(Ctx, AttributeSet::FunctionIndex, + Attribute::NoUnwind); + + Type *SizetTy = Type::getInt32Ty(Ctx); + + Type *Tys_alloc[1] = { SizetTy }; + Type *I8Ptr = PointerType::get( Type::getInt8Ty(Ctx), 1); + FunctionType *FTy_alloc + = FunctionType::get( I8Ptr, Tys_alloc, false); + Constant *PrintfAllocFn + = M.getOrInsertFunction(StringRef("__printf_alloc"), FTy_alloc, Attr); + Function *Afn = dyn_cast(PrintfAllocFn); + Afn->setCallingConv(llvm::CallingConv::SPIR_FUNC); + DEBUG(dbgs() << "inserting printf_alloc decl, an extern @ pre-link:"); + DEBUG(dbgs() << *Afn); + + DEBUG(dbgs() << "Printf metadata = " << Sizes.str() << '\n'); + std::string fmtstr = itostr(++UniqID) + ":" + Sizes.str().c_str(); + MDString *fmtStrArray + = MDString::get( Ctx, fmtstr ); + + + // Instead of creating global variables, the + // printf format strings are extracted + // and passed as metadata. This avoids + // polluting llvm's symbol tables in this module. + // Metadata is going to be extracted + // by the backend passes and inserted + // into the OpenCL binary as appropriate. + StringRef amd("llvm.printf.fmts"); + NamedMDNode *metaD = M.getOrInsertNamedMetadata(amd); + MDNode *myMD = MDNode::get(Ctx,fmtStrArray); + metaD->addOperand(myMD); + Value *sumC = ConstantInt::get( SizetTy, Sum, false); + SmallVector alloc_args; + alloc_args.push_back(sumC); + CallInst *pcall = CallInst::Create( Afn, alloc_args, + "printf_alloc_fn", CI); + pcall->setCallingConv(llvm::CallingConv::SPIR_FUNC); + + // + // Insert code to split basicblock with a + // piece of hammock code. + // basicblock splits after buffer overflow check + // + ConstantPointerNull *zeroIntPtr + = ConstantPointerNull::get(PointerType::get(Type::getInt8Ty(Ctx), + 1)); + ICmpInst *cmp + = dyn_cast( + Builder.CreateICmpNE(pcall, zeroIntPtr, "")); + if (!CI->use_empty()) { + Value *result = Builder.CreateSExt(Builder.CreateNot(cmp), I32Ty, + "printf_res"); + CI->replaceAllUsesWith(result); + } + SplitBlock(CI->getParent(), cmp); + TerminatorInst *Brnch + = SplitBlockAndInsertIfThen(cmp, cmp->getNextNode(), false); + + Builder.SetInsertPoint(Brnch); + + // store unique printf id in the buffer + // + SmallVector ZeroIdxList; + ConstantInt* zeroInt + = ConstantInt::get(Ctx, APInt(32, StringRef("0"), 10)); + ZeroIdxList.push_back(zeroInt); + + GetElementPtrInst *BufferIdx + = dyn_cast( + GetElementPtrInst::Create(nullptr, + pcall, ZeroIdxList, "PrintBuffID", Brnch)); + + Type *idPointer + = PointerType::get(I32Ty, GlobalAddrspace); + Value *id_gep_cast + = new BitCastInst( BufferIdx, idPointer, + "PrintBuffIdCast", Brnch); + + StoreInst* stbuff + = new StoreInst( ConstantInt::get(I32Ty, UniqID), id_gep_cast); + stbuff->insertBefore(Brnch); // to Remove unused variable warning + + SmallVector FourthIdxList; + ConstantInt* fourInt + = ConstantInt::get(Ctx, APInt( + 32, StringRef("4"), 10)); + + FourthIdxList.push_back(fourInt); // 1st 4 bytes hold the printf_id + // the following GEP is the buffer pointer + BufferIdx + = cast(GetElementPtrInst::Create(nullptr, + pcall, FourthIdxList, "PrintBuffGep", Brnch)); + + Type* Int32Ty = Type::getInt32Ty(Ctx); + Type* Int64Ty = Type::getInt64Ty(Ctx); + for (unsigned ArgCount = 1; + ArgCount < CI->getNumArgOperands() + && ArgCount <= OpConvSpecifiers.size(); + ArgCount++) { + Value *Arg = CI->getArgOperand(ArgCount); + Type *ArgType = Arg->getType(); + SmallVector WhatToStore; + if (ArgType->isFPOrFPVectorTy() + && (ArgType->getTypeID() != Type::VectorTyID)) { + Type *IType = (ArgType->isFloatTy()) ? Int32Ty : Int64Ty; + if (OpConvSpecifiers[ArgCount - 1] == 'f') { + ConstantFP *fpCons = dyn_cast(Arg); + if (fpCons) { + APFloat Val(fpCons->getValueAPF()); + bool Lost = false; + Val.convert(APFloat::IEEEsingle, + APFloat::rmNearestTiesToEven, + &Lost); + Arg = ConstantFP::get(Ctx, Val); + IType = Int32Ty; + } else { + FPExtInst *FpExt = dyn_cast(Arg); + if (FpExt && FpExt->getType()->isDoubleTy() + && FpExt->getOperand(0)->getType()->isFloatTy()) { + Arg = FpExt->getOperand(0); + IType = Int32Ty; + } + } + } + Arg = new BitCastInst(Arg, IType, "PrintArgFP", Brnch); + WhatToStore.push_back(Arg); + } else if (ArgType->getTypeID() == Type::PointerTyID) { + if (shouldPrintAsStr(OpConvSpecifiers[ArgCount - 1], ArgType)) { + const char *S = NonLiteralStr; + if (ConstantExpr *ConstExpr = dyn_cast(Arg)) { + GlobalVariable *GV + = dyn_cast(ConstExpr->getOperand(0)); + if (GV && GV->hasInitializer()) { + Constant *Init = GV->getInitializer(); + ConstantDataArray *CA = dyn_cast(Init); + if (Init->isZeroValue() || CA->isString()) { + S = Init->isZeroValue() ? "" : CA->getAsCString().data(); + } + } + } + size_t SizeStr = strlen(S) + 1; + size_t Rem = SizeStr % DWORD_ALIGN; + size_t NSizeStr = 0; + if (Rem) { + NSizeStr = SizeStr + (DWORD_ALIGN - Rem); + } else { + NSizeStr = SizeStr; + } + if (S[0]) { + char *MyNewStr = new char[NSizeStr](); + strcpy(MyNewStr, S); + int NumInts = NSizeStr/4; + int CharC = 0; + while(NumInts) { + int ANum = *(int*)(MyNewStr+CharC); + CharC += 4; + NumInts--; + Value *ANumV = ConstantInt::get( Int32Ty, ANum, false); + WhatToStore.push_back(ANumV); + } + delete MyNewStr; + } else { + // Empty string, give a hint to RT it is no NULL + Value *ANumV = ConstantInt::get(Int32Ty, 0xFFFFFF00, false); + WhatToStore.push_back(ANumV); + } + } else { + uint64_t Size = TD->getTypeAllocSizeInBits(ArgType); + assert((Size == 32 || Size == 64) && "unsupported size"); + Type* DstType = (Size == 32) ? Int32Ty : Int64Ty; + Arg = new PtrToIntInst(Arg, DstType, + "PrintArgPtr", Brnch); + WhatToStore.push_back(Arg); + } + } else if (ArgType->getTypeID() == Type::VectorTyID) { + Type *IType = NULL; + uint32_t EleCount = cast(ArgType)->getNumElements(); + uint32_t EleSize = ArgType->getScalarSizeInBits(); + uint32_t TotalSize = EleCount * EleSize; + if (EleCount == 3) { + IntegerType *Int32Ty + = Type::getInt32Ty(ArgType->getContext()); + Constant* Indices[4] + = { ConstantInt::get(Int32Ty, 0), + ConstantInt::get(Int32Ty, 1), + ConstantInt::get(Int32Ty, 2), + ConstantInt::get(Int32Ty, 2) + }; + Constant* Mask = ConstantVector::get(Indices); + ShuffleVectorInst* Shuffle + = new ShuffleVectorInst(Arg, Arg, Mask); + Shuffle->insertBefore(Brnch); + Arg = Shuffle; + ArgType = Arg->getType(); + TotalSize += EleSize; + } + switch (EleSize) { + default: + EleCount = TotalSize / 64; + IType = dyn_cast( + Type::getInt64Ty( + ArgType->getContext())); + break; + case 8: + if (EleCount >= 8) { + EleCount = TotalSize / 64; + IType = dyn_cast( + Type::getInt64Ty( + ArgType->getContext())); + } else if (EleCount >= 3) { + EleCount = 1; + IType = dyn_cast( + Type::getInt32Ty( + ArgType->getContext())); + } else { + EleCount = 1; + IType = dyn_cast( + Type::getInt16Ty( + ArgType->getContext())); + } + break; + case 16: + if (EleCount >= 3) { + EleCount = TotalSize / 64; + IType = dyn_cast( + Type::getInt64Ty( + ArgType->getContext())); + } else { + EleCount = 1; + IType = dyn_cast( + Type::getInt32Ty( + ArgType->getContext())); + } + break; + } + if (EleCount > 1) { + IType = dyn_cast( + VectorType::get( + IType, EleCount)); + } + Arg = new BitCastInst(Arg, IType, "PrintArgVect", Brnch); + WhatToStore.push_back(Arg); + } else { + WhatToStore.push_back(Arg); + } + for( auto W : WhatToStore ) { + Value* TheBtCast = W; + unsigned ArgSize + = TD->getTypeAllocSizeInBits(TheBtCast->getType())/8; + SmallVector BuffOffset; + BuffOffset.push_back( + ConstantInt::get( I32Ty, ArgSize)); + + Type *ArgPointer + = PointerType::get( TheBtCast->getType(), 1); + Value *CastedGEP + = new BitCastInst( BufferIdx, ArgPointer, + "PrintBuffPtrCast", Brnch); + StoreInst* StBuff + = new StoreInst( + TheBtCast, CastedGEP, Brnch); + DEBUG(dbgs() << "inserting store to printf buffer:\n" + << *StBuff << '\n'); + ++W; + if (W == *WhatToStore.end() + && ArgCount+1 == CI->getNumArgOperands()) + break; + BufferIdx + = dyn_cast(GetElementPtrInst::Create( + nullptr, BufferIdx, BuffOffset, "PrintBuffNextPtr", Brnch)); + DEBUG(dbgs() << "inserting gep to the printf buffer:\n" + << *BufferIdx << '\n'); + } + } + } + } + //erase the printf calls + for (auto P: Printfs) { + CallInst* CI + = dyn_cast(P); + CI->eraseFromParent(); + } + return true; +} + +bool AMDGPUPrintfRuntimeBinding::runOnModule(Module &M) { + if (!prepare(M)) + return false; + return lowerPrintfForGpu(M); +} + +const char* AMDGPUPrintfRuntimeBinding::getPassName() const { + return "AMD Printf lowering part 1"; +} + +bool AMDGPUPrintfRuntimeBinding::doInitialization(Module &M) { + return false; +} + +bool AMDGPUPrintfRuntimeBinding::doFinalization(Module &M) { + return false; +} diff --git a/lib/Target/AMDGPU/AMDGPUTargetMachine.cpp b/lib/Target/AMDGPU/AMDGPUTargetMachine.cpp index f144ce2eb339..34d2c58985ab 100644 --- a/lib/Target/AMDGPU/AMDGPUTargetMachine.cpp +++ b/lib/Target/AMDGPU/AMDGPUTargetMachine.cpp @@ -34,6 +34,7 @@ #include "llvm/Transforms/Scalar.h" #include "llvm/Transforms/Scalar/GVN.h" #include "llvm/Transforms/Vectorize.h" +#include "llvm/IR/LegacyPassManager.h" using namespace llvm; @@ -83,6 +84,7 @@ extern "C" void LLVMInitializeAMDGPUTarget() { initializeSILowerControlFlowPass(*PR); initializeSIInsertSkipsPass(*PR); initializeSIDebuggerInsertNopsPass(*PR); + initializeAMDGPUPrintfRuntimeBindingPass(*PR); } static std::unique_ptr createTLOF(const Triple &TT) { @@ -180,6 +182,11 @@ StringRef AMDGPUTargetMachine::getFeatureString(const Function &F) const { FSAttr.getValueAsString(); } +void AMDGPUTargetMachine::addPreLinkPasses(PassManagerBase &PM) { + PM.add(llvm::createAMDGPUPrintfRuntimeBinding()); +} + + //===----------------------------------------------------------------------===// // R600 Target Machine (R600 -> Cayman) //===----------------------------------------------------------------------===// diff --git a/lib/Target/AMDGPU/AMDGPUTargetMachine.h b/lib/Target/AMDGPU/AMDGPUTargetMachine.h index d8a71b464cc6..2cc3bbe4d7c0 100644 --- a/lib/Target/AMDGPU/AMDGPUTargetMachine.h +++ b/lib/Target/AMDGPU/AMDGPUTargetMachine.h @@ -50,6 +50,7 @@ class AMDGPUTargetMachine : public LLVMTargetMachine { TargetLoweringObjectFile *getObjFileLowering() const override { return TLOF.get(); } + virtual void addPreLinkPasses(PassManagerBase &) override; }; //===----------------------------------------------------------------------===// diff --git a/lib/Target/AMDGPU/CMakeLists.txt b/lib/Target/AMDGPU/CMakeLists.txt index e58e5b2f92d4..9a1c1d544f85 100644 --- a/lib/Target/AMDGPU/CMakeLists.txt +++ b/lib/Target/AMDGPU/CMakeLists.txt @@ -48,6 +48,7 @@ add_llvm_target(AMDGPUCodeGen AMDGPUInstrInfo.cpp AMDGPUPromoteAlloca.cpp AMDGPURegisterInfo.cpp + AMDGPUPrinfRuntimeBinding.cpp GCNHazardRecognizer.cpp GCNSchedStrategy.cpp R600ClauseMergePass.cpp diff --git a/test/CodeGen/AMDGPU/opencl-printf.ll b/test/CodeGen/AMDGPU/opencl-printf.ll new file mode 100644 index 000000000000..806c70c5332a --- /dev/null +++ b/test/CodeGen/AMDGPU/opencl-printf.ll @@ -0,0 +1,59 @@ +; RUN: opt -mtriple=amdgcn--amdhsa -amdgpu-printf-runtime-binding -mcpu=fiji -S < %s | FileCheck %s +; CHECK-LABEL: entry +; CHECK: call spir_func i8 addrspace(1)* @__printf_alloc +; CHECK-LABEL: entry.split +; CHECK: icmp ne i8 addrspace(1)* %printf_alloc_fn, null +; CHECK: %PrintBuffID = getelementptr i8, i8 addrspace(1)* %printf_alloc_fn, i32 0 +; CHECK: %PrintBuffIdCast = bitcast i8 addrspace(1)* %PrintBuffID to i32 addrspace(1)* +; CHECK: store i32 1, i32 addrspace(1)* %PrintBuffIdCast +; CHECK: %PrintBuffGep = getelementptr i8, i8 addrspace(1)* %printf_alloc_fn, i32 4 +; CHECK: %PrintArgPtr = ptrtoint i8* %arraydecay to i64 +; CHECK: %PrintBuffPtrCast = bitcast i8 addrspace(1)* %PrintBuffGep to i64 addrspace(1)* +; CHECK: store i64 %PrintArgPtr, i64 addrspace(1)* %PrintBuffPtrCast +; CHECK: %PrintBuffNextPtr = getelementptr i8, i8 addrspace(1)* %PrintBuffGep, i32 8 +; CHECK: %PrintBuffPtrCast1 = bitcast i8 addrspace(1)* %PrintBuffNextPtr to i32 addrspace(1)* +; CHECK: store i32 %3, i32 addrspace(1)* %PrintBuffPtrCast1 + +@test_kernel.str = private unnamed_addr constant [9 x i8] c"globalid\00", align 1 +@.str = private unnamed_addr addrspace(2) constant [6 x i8] c"%s:%d\00", align 1 + +define amdgpu_kernel void @test_kernel(i32 addrspace(1)* %in, i32 addrspace(1)* %out) { +entry: + %in.addr = alloca i32 addrspace(1)*, align 4 + %out.addr = alloca i32 addrspace(1)*, align 4 + %n = alloca i32, align 4 + %str = alloca [9 x i8], align 1 + store i32 addrspace(1)* %in, i32 addrspace(1)** %in.addr, align 4 + store i32 addrspace(1)* %out, i32 addrspace(1)** %out.addr, align 4 + %0 = bitcast i32* %n to i8* + %call = call i64 @_Z13get_global_idj(i32 0) #5 + %conv = trunc i64 %call to i32 + store i32 %conv, i32* %n, align 4 + %1 = bitcast [9 x i8]* %str to i8* + %2 = bitcast [9 x i8]* %str to i8* + call void @llvm.memcpy.p0i8.p0i8.i64(i8* %2, i8* getelementptr inbounds ([9 x i8], [9 x i8]* @test_kernel.str, i32 0, i32 0), i64 9, i32 1, i1 false) + %arraydecay = getelementptr inbounds [9 x i8], [9 x i8]* %str, i32 0, i32 0 + %3 = load i32, i32* %n, align 4 + %call1 = call i32 (i8 addrspace(2)*, ...) @printf(i8 addrspace(2)* getelementptr inbounds ([6 x i8], [6 x i8] addrspace(2)* @.str, i32 0, i32 0), i8* %arraydecay, i32 %3) + %4 = load i32, i32* %n, align 4 + %idxprom = sext i32 %4 to i64 + %5 = load i32 addrspace(1)*, i32 addrspace(1)** %in.addr, align 4 + %arrayidx = getelementptr inbounds i32, i32 addrspace(1)* %5, i64 %idxprom + %6 = load i32, i32 addrspace(1)* %arrayidx, align 4 + %7 = load i32, i32* %n, align 4 + %idxprom2 = sext i32 %7 to i64 + %8 = load i32 addrspace(1)*, i32 addrspace(1)** %out.addr, align 4 + %arrayidx3 = getelementptr inbounds i32, i32 addrspace(1)* %8, i64 %idxprom2 + store i32 %6, i32 addrspace(1)* %arrayidx3, align 4 + %9 = bitcast [9 x i8]* %str to i8* + %10 = bitcast i32* %n to i8* + ret void +} + +; Function Attrs: nounwind readnone +declare i64 @_Z13get_global_idj(i32) #2 + +; Function Attrs: argmemonly nounwind +declare void @llvm.memcpy.p0i8.p0i8.i64(i8* nocapture writeonly, i8* nocapture readonly, i64, i32, i1) #1 + +declare i32 @printf(i8 addrspace(2)*, ...) #3