mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-02-06 06:33:24 +00:00
1545953510
`DIDescriptor`'s subclasses allow construction from incompatible pointers, and `DIDescriptor` defines a series of `isa<>`-like functions (e.g., `isCompileUnit()` instead of `isa<MDCompileUnit>()`) that clients tend to use like this: if (DICompileUnit(N).isCompileUnit()) foo(DICompileUnit(N)); These construction patterns work together to make `DIDescriptor` behave differently from normal pointers. Instead, use built-in `isa<>`, `dyn_cast<>`, etc., and only build `DIDescriptor`s from pointers that are valid for their type. I've split this into a few commits for different parts of LLVM and clang (to decrease the patch size and increase the chance of review). Generally the changes I made were NFC, but in a few places I made things stricter if it made sense from the surrounded code. Eventually a follow-up commit will remove the API for the "old" way. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@234255 91177308-0d34-0410-b5e6-96231b3b80d8
79 lines
2.3 KiB
C++
79 lines
2.3 KiB
C++
//===- BreakpointPrinter.cpp - Breakpoint location printer ----------------===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
///
|
|
/// \file
|
|
/// \brief Breakpoint location printer.
|
|
///
|
|
//===----------------------------------------------------------------------===//
|
|
#include "BreakpointPrinter.h"
|
|
#include "llvm/ADT/StringSet.h"
|
|
#include "llvm/IR/DebugInfo.h"
|
|
#include "llvm/IR/Module.h"
|
|
#include "llvm/Pass.h"
|
|
#include "llvm/Support/raw_ostream.h"
|
|
|
|
using namespace llvm;
|
|
|
|
namespace {
|
|
|
|
struct BreakpointPrinter : public ModulePass {
|
|
raw_ostream &Out;
|
|
static char ID;
|
|
DITypeIdentifierMap TypeIdentifierMap;
|
|
|
|
BreakpointPrinter(raw_ostream &out) : ModulePass(ID), Out(out) {}
|
|
|
|
void getContextName(DIDescriptor Context, std::string &N) {
|
|
if (DINameSpace NS = dyn_cast<MDNamespace>(Context)) {
|
|
if (!NS.getName().empty()) {
|
|
getContextName(NS.getContext(), N);
|
|
N = N + NS.getName().str() + "::";
|
|
}
|
|
} else if (DIType TY = dyn_cast<MDType>(Context)) {
|
|
if (!TY.getName().empty()) {
|
|
getContextName(TY.getContext().resolve(TypeIdentifierMap), N);
|
|
N = N + TY.getName().str() + "::";
|
|
}
|
|
}
|
|
}
|
|
|
|
bool runOnModule(Module &M) override {
|
|
TypeIdentifierMap.clear();
|
|
NamedMDNode *CU_Nodes = M.getNamedMetadata("llvm.dbg.cu");
|
|
if (CU_Nodes)
|
|
TypeIdentifierMap = generateDITypeIdentifierMap(CU_Nodes);
|
|
|
|
StringSet<> Processed;
|
|
if (NamedMDNode *NMD = M.getNamedMetadata("llvm.dbg.sp"))
|
|
for (unsigned i = 0, e = NMD->getNumOperands(); i != e; ++i) {
|
|
std::string Name;
|
|
DISubprogram SP = cast_or_null<MDSubprogram>(NMD->getOperand(i));
|
|
if (!SP)
|
|
continue;
|
|
getContextName(SP.getContext().resolve(TypeIdentifierMap), Name);
|
|
Name = Name + SP.getDisplayName().str();
|
|
if (!Name.empty() && Processed.insert(Name).second) {
|
|
Out << Name << "\n";
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void getAnalysisUsage(AnalysisUsage &AU) const override {
|
|
AU.setPreservesAll();
|
|
}
|
|
};
|
|
|
|
char BreakpointPrinter::ID = 0;
|
|
}
|
|
|
|
ModulePass *llvm::createBreakpointPrinter(raw_ostream &out) {
|
|
return new BreakpointPrinter(out);
|
|
}
|