/*
* Copyright 2015, The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "Assert.h"
#include "Log.h"
#include "RSTransforms.h"
#include "bcinfo/MetadataExtractor.h"
#include <string>
#include <llvm/Pass.h>
#include <llvm/IR/DIBuilder.h>
#include <llvm/IR/Function.h>
#include <llvm/IR/InstIterator.h>
#include <llvm/IR/Instructions.h>
#include <llvm/IR/IRBuilder.h>
#include <llvm/IR/Module.h>
#include <llvm/ADT/SetVector.h>
namespace {
const char DEBUG_SOURCE_PATH[] = "/opt/renderscriptdebugger/1";
const char DEBUG_GENERATED_FILE[] = "generated.rs";
const char DEBUG_PROTOTYPE_VAR_NAME[] = "rsDebugOuterForeachT";
const char DEBUG_COMPILE_UNIT_MDNAME[] = "llvm.dbg.cu";
/*
* LLVM pass to attach debug information to the bits of code
* generated by the compiler.
*/
class RSAddDebugInfoPass : public llvm::ModulePass {
public:
// Pass ID
static char ID;
RSAddDebugInfoPass() : ModulePass(ID), kernelTypeMD(nullptr),
sourceFileName(nullptr), emptyExpr(nullptr), abiMetaCU(nullptr),
indexVarType(nullptr) {
}
virtual bool runOnModule(llvm::Module &Module) {
// Gather information about this bcc module.
bcinfo::MetadataExtractor me(&Module);
if (!me.extract()) {
ALOGE("Could not extract metadata from module!");
return false;
}
const size_t nForEachKernels = me.getExportForEachSignatureCount();
const char **forEachKernels = me.getExportForEachNameList();
const bcinfo::MetadataExtractor::Reduce *reductions =
me.getExportReduceList();
const size_t nReductions = me.getExportReduceCount();
llvm::SmallSetVector<llvm::Function *, 16> expandFuncs{};
auto pushExpanded = [&](const char *const name) -> void {
bccAssert(name && *name && (::strcmp(name, ".") != 0));
const std::string expandName = std::string(name) + ".expand";
if (llvm::Function *const func = Module.getFunction(expandName))
expandFuncs.insert(func);
};
for (size_t i = 0; i < nForEachKernels; ++i)
pushExpanded(forEachKernels[i]);
for (size_t i = 0; i < nReductions; ++i) {
const bcinfo::MetadataExtractor::Reduce &reduction = reductions[i];
pushExpanded(reduction.mAccumulatorName);
}
// Set up the debug info builder.
llvm::DIBuilder DebugInfo(Module);
initializeDebugInfo(DebugInfo, Module);
for (const auto &expandFunc : expandFuncs) {
// Attach DI metadata to each generated function.
// No inlining has occurred at this point so it's safe to name match
// without worrying about inlined function bodies.
attachDebugInfo(DebugInfo, *expandFunc);
}
DebugInfo.finalize();
cleanupDebugInfo(Module);
return true;
}
private:
// @brief Initialize the debug info generation.
//
// This method does a couple of things:
// * Look up debug metadata for kernel ABI and store it if present.
// * Store a couple of useful pieces of debug metadata in member
// variables so they do not have to be created multiple times.
void initializeDebugInfo(llvm::DIBuilder &DebugInfo,
const llvm::Module &Module) {
llvm::LLVMContext &ctx = Module.getContext();
// Start generating debug information for bcc-generated code.
DebugInfo.createCompileUnit(llvm::dwarf::DW_LANG_GOOGLE_RenderScript,
DEBUG_GENERATED_FILE, DEBUG_SOURCE_PATH,
"RS", false, "", 0);
// Pre-generate and save useful pieces of debug metadata.
sourceFileName = DebugInfo.createFile(DEBUG_GENERATED_FILE, DEBUG_SOURCE_PATH);
emptyExpr = DebugInfo.createExpression();
// Lookup compile unit with kernel ABI debug metadata.
llvm::NamedMDNode *mdCompileUnitList =
Module.getNamedMetadata(DEBUG_COMPILE_UNIT_MDNAME);
bccAssert(mdCompileUnitList != nullptr &&
"DebugInfo pass could not find any existing compile units.");
llvm::DIGlobalVariable *kernelPrototypeVarMD = nullptr;
for (llvm::MDNode* CUNode : mdCompileUnitList->operands()) {
if (auto *CU = llvm::dyn_cast<llvm::DICompileUnit>(CUNode)) {
for (llvm::DIGlobalVariable* GV : CU->getGlobalVariables()) {
if (GV->getDisplayName() == DEBUG_PROTOTYPE_VAR_NAME) {
kernelPrototypeVarMD = GV;
abiMetaCU = CU;
break;
}
}
if (kernelPrototypeVarMD != nullptr)
break;
}
}
// Lookup the expanded function interface type metadata.
llvm::MDTuple *kernelPrototypeMD = nullptr;
if (kernelPrototypeVarMD != nullptr) {
// Dig into the metadata to look for function prototype.
llvm::DIDerivedType *DT = nullptr;
DT = llvm::cast<llvm::DIDerivedType>(kernelPrototypeVarMD->getType());
DT = llvm::cast<llvm::DIDerivedType>(DT->getBaseType());
llvm::DISubroutineType *ST = llvm::cast<llvm::DISubroutineType>(DT->getBaseType());
kernelPrototypeMD = llvm::cast<llvm::MDTuple>(ST->getRawTypeArray());
indexVarType = llvm::dyn_cast_or_null<llvm::DIType>(
kernelPrototypeMD->getOperand(2));
}
// Fall back to the function type of void() if there is no proper debug info.
if (kernelPrototypeMD == nullptr)
kernelPrototypeMD = llvm::MDTuple::get(ctx, {nullptr});
// Fall back to unspecified type if we don't have a proper index type.
if (indexVarType == nullptr)
indexVarType = DebugInfo.createBasicType("uint32_t", 32, 32,
llvm::dwarf::DW_ATE_unsigned);
// Capture the expanded kernel type debug info.
kernelTypeMD = DebugInfo.createSubroutineType(kernelPrototypeMD);
}
/// @brief Add debug information to a generated function.
///
/// This procedure adds the following pieces of debug information
/// to the function specified by Func:
/// * Entry for the function to the current compile unit.
/// * Adds debug info entries for each function argument.
/// * Adds debug info entry for the rsIndex local variable.
/// * File/line information to each instruction set to generates.rs:1.
void attachDebugInfo(llvm::DIBuilder &DebugInfo, llvm::Function &Func) {
// Lookup the current thread coordinate variable.
llvm::AllocaInst* indexVar = nullptr;
for (llvm::Instruction &inst : llvm::instructions(Func)) {
if (auto *allocaInst = llvm::dyn_cast<llvm::AllocaInst>(&inst)) {
if (allocaInst->getName() == bcc::BCC_INDEX_VAR_NAME) {
indexVar = allocaInst;
break;
}
}
}
// Create function-level debug metadata.
llvm::DISubprogram *ExpandedFunc = DebugInfo.createFunction(
sourceFileName, // scope
Func.getName(), Func.getName(),
sourceFileName, 1, kernelTypeMD,
false, true, 1, 0, false
);
Func.setSubprogram(ExpandedFunc);
// IRBuilder for allocating variables for arguments.
llvm::IRBuilder<> ir(&*Func.getEntryBlock().begin());
// Walk through the argument list and expanded function prototype
// debuginfo in lockstep to create debug entries for
// the expanded function arguments.
unsigned argIdx = 1;
llvm::MDTuple *argTypes = kernelTypeMD->getTypeArray().get();
for (llvm::Argument &arg : Func.getArgumentList()) {
// Stop processing arguments if we run out of debug info.
if (argIdx >= argTypes->getNumOperands())
break;
// Create debuginfo entry for the argument and advance.
llvm::DILocalVariable *argVarDI = DebugInfo.createParameterVariable(
ExpandedFunc, arg.getName(), argIdx, sourceFileName, 1,
llvm::cast<llvm::DIType>(argTypes->getOperand(argIdx).get()),
true, 0
);
// Annotate the argument variable in the IR.
llvm::AllocaInst *argVar =
ir.CreateAlloca(arg.getType(), nullptr, arg.getName() + ".var");
llvm::StoreInst *argStore = ir.CreateStore(&arg, argVar);
llvm::LoadInst *loadedVar = ir.CreateLoad(argVar, arg.getName() + ".l");
DebugInfo.insertDeclare(argVar, argVarDI, emptyExpr,
llvm::DebugLoc::get(1, 1, ExpandedFunc), loadedVar);
for (llvm::Use &u : arg.uses())
if (u.getUser() != argStore)
u.set(loadedVar);
argIdx++;
}
// Annotate the index variable with metadata.
if (indexVar) {
// Debug information for loop index variable.
llvm::DILocalVariable *indexVarDI = DebugInfo.createAutoVariable(
ExpandedFunc, bcc::BCC_INDEX_VAR_NAME, sourceFileName, 1,
indexVarType, true
);
// Insert declaration annotation in the instruction stream.
llvm::Instruction *decl = DebugInfo.insertDeclare(
indexVar, indexVarDI, emptyExpr,
llvm::DebugLoc::get(1, 1, ExpandedFunc), indexVar);
indexVar->moveBefore(decl);
}
// Attach location information to each instruction in the function.
for (llvm::Instruction &inst : llvm::instructions(Func)) {
inst.setDebugLoc(llvm::DebugLoc::get(1, 1, ExpandedFunc));
}
}
// @brief Clean up the debug info.
//
// At the moment, it only finds the compile unit for the expanded function
// metadata generated by clang and removes it.
void cleanupDebugInfo(llvm::Module& Module) {
if (abiMetaCU == nullptr)
return;
// Remove the compile unit with the runtime interface DI.
llvm::SmallVector<llvm::MDNode*, 4> unitsTmp;
llvm::NamedMDNode *debugMD =
Module.getNamedMetadata(DEBUG_COMPILE_UNIT_MDNAME);
for (llvm::MDNode *cu : debugMD->operands())
if (cu != abiMetaCU)
unitsTmp.push_back(cu);
debugMD->eraseFromParent();
debugMD = Module.getOrInsertNamedMetadata(DEBUG_COMPILE_UNIT_MDNAME);
for (llvm::MDNode *cu : unitsTmp)
debugMD->addOperand(cu);
}
private:
// private attributes
llvm::DISubroutineType* kernelTypeMD;
llvm::DIFile *sourceFileName;
llvm::DIExpression *emptyExpr;
llvm::DICompileUnit *abiMetaCU;
llvm::DIType *indexVarType;
}; // end class RSAddDebugInfoPass
char RSAddDebugInfoPass::ID = 0;
static llvm::RegisterPass<RSAddDebugInfoPass> X("addrsdi", "Add RS DebugInfo Pass");
} // end anonymous namespace
namespace bcc {
llvm::ModulePass * createRSAddDebugInfoPass() {
return new RSAddDebugInfoPass();
}
} // end namespace bcc