to reflect the new license. We understand that people may be surprised that we're moving the header entirely to discuss the new license. We checked this carefully with the Foundation's lawyer and we believe this is the correct approach. Essentially, all code in the project is now made available by the LLVM project under our new license, so you will see that the license headers include that license only. Some of our contributors have contributed code under our old license, and accordingly, we have retained a copy of our old license notice in the top-level files in each project and repository. llvm-svn: 351636
290 lines
9.7 KiB
C++
290 lines
9.7 KiB
C++
//===--- RuntimeDebugBuilder.cpp - Helper to insert prints into LLVM-IR ---===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "polly/CodeGen/RuntimeDebugBuilder.h"
|
|
#include "llvm/IR/Intrinsics.h"
|
|
#include "llvm/IR/Module.h"
|
|
#include "llvm/Support/Debug.h"
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
using namespace llvm;
|
|
using namespace polly;
|
|
|
|
Function *RuntimeDebugBuilder::getVPrintF(PollyIRBuilder &Builder) {
|
|
Module *M = Builder.GetInsertBlock()->getParent()->getParent();
|
|
const char *Name = "vprintf";
|
|
Function *F = M->getFunction(Name);
|
|
|
|
if (!F) {
|
|
GlobalValue::LinkageTypes Linkage = Function::ExternalLinkage;
|
|
FunctionType *Ty = FunctionType::get(
|
|
Builder.getInt32Ty(), {Builder.getInt8PtrTy(), Builder.getInt8PtrTy()},
|
|
false);
|
|
F = Function::Create(Ty, Linkage, Name, M);
|
|
}
|
|
|
|
return F;
|
|
}
|
|
|
|
Function *RuntimeDebugBuilder::getAddressSpaceCast(PollyIRBuilder &Builder,
|
|
unsigned Src, unsigned Dst,
|
|
unsigned SrcBits,
|
|
unsigned DstBits) {
|
|
Module *M = Builder.GetInsertBlock()->getParent()->getParent();
|
|
auto Name = std::string("llvm.nvvm.ptr.constant.to.gen.p") +
|
|
std::to_string(Dst) + "i" + std::to_string(DstBits) + ".p" +
|
|
std::to_string(Src) + "i" + std::to_string(SrcBits);
|
|
Function *F = M->getFunction(Name);
|
|
|
|
if (!F) {
|
|
GlobalValue::LinkageTypes Linkage = Function::ExternalLinkage;
|
|
FunctionType *Ty = FunctionType::get(
|
|
PointerType::get(Builder.getIntNTy(DstBits), Dst),
|
|
PointerType::get(Builder.getIntNTy(SrcBits), Src), false);
|
|
F = Function::Create(Ty, Linkage, Name, M);
|
|
}
|
|
|
|
return F;
|
|
}
|
|
|
|
std::vector<Value *>
|
|
RuntimeDebugBuilder::getGPUThreadIdentifiers(PollyIRBuilder &Builder) {
|
|
std::vector<Value *> Identifiers;
|
|
|
|
auto M = Builder.GetInsertBlock()->getParent()->getParent();
|
|
|
|
std::vector<Function *> BlockIDs = {
|
|
Intrinsic::getDeclaration(M, Intrinsic::nvvm_read_ptx_sreg_ctaid_x),
|
|
Intrinsic::getDeclaration(M, Intrinsic::nvvm_read_ptx_sreg_ctaid_y),
|
|
Intrinsic::getDeclaration(M, Intrinsic::nvvm_read_ptx_sreg_ctaid_z),
|
|
};
|
|
|
|
Identifiers.push_back(Builder.CreateGlobalStringPtr("> block-id: ", "", 4));
|
|
for (auto GetID : BlockIDs) {
|
|
Value *Id = Builder.CreateCall(GetID, {});
|
|
Id = Builder.CreateIntCast(Id, Builder.getInt64Ty(), false);
|
|
Identifiers.push_back(Id);
|
|
Identifiers.push_back(Builder.CreateGlobalStringPtr(" ", "", 4));
|
|
}
|
|
|
|
Identifiers.push_back(Builder.CreateGlobalStringPtr("| ", "", 4));
|
|
|
|
std::vector<Function *> ThreadIDs = {
|
|
Intrinsic::getDeclaration(M, Intrinsic::nvvm_read_ptx_sreg_tid_x),
|
|
Intrinsic::getDeclaration(M, Intrinsic::nvvm_read_ptx_sreg_tid_y),
|
|
Intrinsic::getDeclaration(M, Intrinsic::nvvm_read_ptx_sreg_tid_z),
|
|
};
|
|
|
|
Identifiers.push_back(Builder.CreateGlobalStringPtr("thread-id: ", "", 4));
|
|
for (auto GetId : ThreadIDs) {
|
|
Value *Id = Builder.CreateCall(GetId, {});
|
|
Id = Builder.CreateIntCast(Id, Builder.getInt64Ty(), false);
|
|
Identifiers.push_back(Id);
|
|
Identifiers.push_back(Builder.CreateGlobalStringPtr(" ", "", 4));
|
|
}
|
|
|
|
return Identifiers;
|
|
}
|
|
|
|
void RuntimeDebugBuilder::createPrinter(PollyIRBuilder &Builder, bool IsGPU,
|
|
ArrayRef<Value *> Values) {
|
|
if (IsGPU)
|
|
createGPUPrinterT(Builder, Values);
|
|
else
|
|
createCPUPrinterT(Builder, Values);
|
|
}
|
|
|
|
bool RuntimeDebugBuilder::isPrintable(Type *Ty) {
|
|
if (Ty->isFloatingPointTy())
|
|
return true;
|
|
|
|
if (Ty->isIntegerTy())
|
|
return Ty->getIntegerBitWidth() <= 64;
|
|
|
|
if (isa<PointerType>(Ty))
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
static std::tuple<std::string, std::vector<Value *>>
|
|
prepareValuesForPrinting(PollyIRBuilder &Builder, ArrayRef<Value *> Values) {
|
|
std::string FormatString;
|
|
std::vector<Value *> ValuesToPrint;
|
|
|
|
for (auto Val : Values) {
|
|
Type *Ty = Val->getType();
|
|
|
|
if (Ty->isFloatingPointTy()) {
|
|
if (!Ty->isDoubleTy())
|
|
Val = Builder.CreateFPExt(Val, Builder.getDoubleTy());
|
|
} else if (Ty->isIntegerTy()) {
|
|
if (Ty->getIntegerBitWidth() < 64)
|
|
Val = Builder.CreateSExt(Val, Builder.getInt64Ty());
|
|
else
|
|
assert(Ty->getIntegerBitWidth() &&
|
|
"Integer types larger 64 bit not supported");
|
|
} else if (isa<PointerType>(Ty)) {
|
|
if (Ty->getPointerElementType() == Builder.getInt8Ty() &&
|
|
Ty->getPointerAddressSpace() == 4) {
|
|
Val = Builder.CreateGEP(Val, Builder.getInt64(0));
|
|
} else {
|
|
Val = Builder.CreatePtrToInt(Val, Builder.getInt64Ty());
|
|
}
|
|
} else {
|
|
llvm_unreachable("Unknown type");
|
|
}
|
|
|
|
Ty = Val->getType();
|
|
|
|
if (Ty->isFloatingPointTy())
|
|
FormatString += "%f";
|
|
else if (Ty->isIntegerTy())
|
|
FormatString += "%ld";
|
|
else
|
|
FormatString += "%s";
|
|
|
|
ValuesToPrint.push_back(Val);
|
|
}
|
|
|
|
return std::make_tuple(FormatString, ValuesToPrint);
|
|
}
|
|
|
|
void RuntimeDebugBuilder::createCPUPrinterT(PollyIRBuilder &Builder,
|
|
ArrayRef<Value *> Values) {
|
|
|
|
std::string FormatString;
|
|
std::vector<Value *> ValuesToPrint;
|
|
|
|
std::tie(FormatString, ValuesToPrint) =
|
|
prepareValuesForPrinting(Builder, Values);
|
|
|
|
createPrintF(Builder, FormatString, ValuesToPrint);
|
|
createFlush(Builder);
|
|
}
|
|
|
|
void RuntimeDebugBuilder::createGPUPrinterT(PollyIRBuilder &Builder,
|
|
ArrayRef<Value *> Values) {
|
|
std::string str;
|
|
|
|
auto *Zero = Builder.getInt64(0);
|
|
|
|
auto ToPrint = getGPUThreadIdentifiers(Builder);
|
|
|
|
ToPrint.push_back(Builder.CreateGlobalStringPtr("\n ", "", 4));
|
|
ToPrint.insert(ToPrint.end(), Values.begin(), Values.end());
|
|
|
|
const DataLayout &DL = Builder.GetInsertBlock()->getModule()->getDataLayout();
|
|
|
|
// Allocate print buffer (assuming 2*32 bit per element)
|
|
auto T = ArrayType::get(Builder.getInt32Ty(), ToPrint.size() * 2);
|
|
Value *Data = new AllocaInst(
|
|
T, DL.getAllocaAddrSpace(), "polly.vprint.buffer",
|
|
&Builder.GetInsertBlock()->getParent()->getEntryBlock().front());
|
|
auto *DataPtr = Builder.CreateGEP(Data, {Zero, Zero});
|
|
|
|
int Offset = 0;
|
|
for (auto Val : ToPrint) {
|
|
auto Ptr = Builder.CreateGEP(DataPtr, Builder.getInt64(Offset));
|
|
Type *Ty = Val->getType();
|
|
|
|
if (Ty->isFloatingPointTy()) {
|
|
if (!Ty->isDoubleTy())
|
|
Val = Builder.CreateFPExt(Val, Builder.getDoubleTy());
|
|
} else if (Ty->isIntegerTy()) {
|
|
if (Ty->getIntegerBitWidth() < 64) {
|
|
Val = Builder.CreateSExt(Val, Builder.getInt64Ty());
|
|
} else {
|
|
assert(Ty->getIntegerBitWidth() == 64 &&
|
|
"Integer types larger 64 bit not supported");
|
|
// fallthrough
|
|
}
|
|
} else if (auto PtTy = dyn_cast<PointerType>(Ty)) {
|
|
if (PtTy->getAddressSpace() == 4) {
|
|
// Pointers in constant address space are printed as strings
|
|
Val = Builder.CreateGEP(Val, Builder.getInt64(0));
|
|
auto F = RuntimeDebugBuilder::getAddressSpaceCast(Builder, 4, 0);
|
|
Val = Builder.CreateCall(F, Val);
|
|
} else {
|
|
Val = Builder.CreatePtrToInt(Val, Builder.getInt64Ty());
|
|
}
|
|
} else {
|
|
llvm_unreachable("Unknown type");
|
|
}
|
|
|
|
Ty = Val->getType();
|
|
Ptr = Builder.CreatePointerBitCastOrAddrSpaceCast(Ptr, Ty->getPointerTo(5));
|
|
Builder.CreateAlignedStore(Val, Ptr, 4);
|
|
|
|
if (Ty->isFloatingPointTy())
|
|
str += "%f";
|
|
else if (Ty->isIntegerTy())
|
|
str += "%ld";
|
|
else
|
|
str += "%s";
|
|
|
|
Offset += 2;
|
|
}
|
|
|
|
Value *Format = Builder.CreateGlobalStringPtr(str, "polly.vprintf.buffer", 4);
|
|
Format = Builder.CreateCall(getAddressSpaceCast(Builder, 4, 0), Format);
|
|
|
|
Data = Builder.CreateBitCast(Data, Builder.getInt8PtrTy());
|
|
|
|
Builder.CreateCall(getVPrintF(Builder), {Format, Data});
|
|
}
|
|
|
|
Function *RuntimeDebugBuilder::getPrintF(PollyIRBuilder &Builder) {
|
|
Module *M = Builder.GetInsertBlock()->getParent()->getParent();
|
|
const char *Name = "printf";
|
|
Function *F = M->getFunction(Name);
|
|
|
|
if (!F) {
|
|
GlobalValue::LinkageTypes Linkage = Function::ExternalLinkage;
|
|
FunctionType *Ty = FunctionType::get(Builder.getInt32Ty(), true);
|
|
F = Function::Create(Ty, Linkage, Name, M);
|
|
}
|
|
|
|
return F;
|
|
}
|
|
|
|
void RuntimeDebugBuilder::createPrintF(PollyIRBuilder &Builder,
|
|
std::string Format,
|
|
ArrayRef<Value *> Values) {
|
|
Value *FormatString = Builder.CreateGlobalStringPtr(Format);
|
|
std::vector<Value *> Arguments;
|
|
|
|
Arguments.push_back(FormatString);
|
|
Arguments.insert(Arguments.end(), Values.begin(), Values.end());
|
|
Builder.CreateCall(getPrintF(Builder), Arguments);
|
|
}
|
|
|
|
void RuntimeDebugBuilder::createFlush(PollyIRBuilder &Builder) {
|
|
Module *M = Builder.GetInsertBlock()->getParent()->getParent();
|
|
const char *Name = "fflush";
|
|
Function *F = M->getFunction(Name);
|
|
|
|
if (!F) {
|
|
GlobalValue::LinkageTypes Linkage = Function::ExternalLinkage;
|
|
FunctionType *Ty =
|
|
FunctionType::get(Builder.getInt32Ty(), Builder.getInt8PtrTy(), false);
|
|
F = Function::Create(Ty, Linkage, Name, M);
|
|
}
|
|
|
|
// fflush(NULL) flushes _all_ open output streams.
|
|
//
|
|
// fflush is declared as 'int fflush(FILE *stream)'. As we only pass on a NULL
|
|
// pointer, the type we point to does conceptually not matter. However, if
|
|
// fflush is already declared in this translation unit, we use the very same
|
|
// type to ensure that LLVM does not complain about mismatching types.
|
|
Builder.CreateCall(F, Constant::getNullValue(F->arg_begin()->getType()));
|
|
}
|