mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-05-14 09:38:40 +00:00
DebugInfo library: add support for fetching absolute paths to source files
(instead of basenames) from DWARF. Use this behavior in llvm-dwarfdump tool. Reviewed by Benjamin Kramer. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@160496 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
72ea0c9ffa
commit
71d94f8055
@ -15,9 +15,9 @@
|
|||||||
#ifndef LLVM_DEBUGINFO_DICONTEXT_H
|
#ifndef LLVM_DEBUGINFO_DICONTEXT_H
|
||||||
#define LLVM_DEBUGINFO_DICONTEXT_H
|
#define LLVM_DEBUGINFO_DICONTEXT_H
|
||||||
|
|
||||||
|
#include "llvm/ADT/SmallString.h"
|
||||||
#include "llvm/ADT/StringRef.h"
|
#include "llvm/ADT/StringRef.h"
|
||||||
#include "llvm/Support/DataTypes.h"
|
#include "llvm/Support/DataTypes.h"
|
||||||
#include <cstring>
|
|
||||||
|
|
||||||
namespace llvm {
|
namespace llvm {
|
||||||
|
|
||||||
@ -25,28 +25,29 @@ class raw_ostream;
|
|||||||
|
|
||||||
/// DILineInfo - a format-neutral container for source line information.
|
/// DILineInfo - a format-neutral container for source line information.
|
||||||
class DILineInfo {
|
class DILineInfo {
|
||||||
const char *FileName;
|
SmallString<16> FileName;
|
||||||
const char *FunctionName;
|
SmallString<16> FunctionName;
|
||||||
uint32_t Line;
|
uint32_t Line;
|
||||||
uint32_t Column;
|
uint32_t Column;
|
||||||
public:
|
public:
|
||||||
DILineInfo()
|
DILineInfo()
|
||||||
: FileName("<invalid>"), FunctionName("<invalid>"),
|
: FileName("<invalid>"), FunctionName("<invalid>"),
|
||||||
Line(0), Column(0) {}
|
Line(0), Column(0) {}
|
||||||
DILineInfo(const char *fileName, const char *functionName,
|
DILineInfo(const SmallString<16> &fileName,
|
||||||
|
const SmallString<16> &functionName,
|
||||||
uint32_t line, uint32_t column)
|
uint32_t line, uint32_t column)
|
||||||
: FileName(fileName), FunctionName(functionName),
|
: FileName(fileName), FunctionName(functionName),
|
||||||
Line(line), Column(column) {}
|
Line(line), Column(column) {}
|
||||||
|
|
||||||
const char *getFileName() const { return FileName; }
|
const char *getFileName() { return FileName.c_str(); }
|
||||||
const char *getFunctionName() const { return FunctionName; }
|
const char *getFunctionName() { return FunctionName.c_str(); }
|
||||||
uint32_t getLine() const { return Line; }
|
uint32_t getLine() const { return Line; }
|
||||||
uint32_t getColumn() const { return Column; }
|
uint32_t getColumn() const { return Column; }
|
||||||
|
|
||||||
bool operator==(const DILineInfo &RHS) const {
|
bool operator==(const DILineInfo &RHS) const {
|
||||||
return Line == RHS.Line && Column == RHS.Column &&
|
return Line == RHS.Line && Column == RHS.Column &&
|
||||||
std::strcmp(FileName, RHS.FileName) == 0 &&
|
FileName.equals(RHS.FileName) &&
|
||||||
std::strcmp(FunctionName, RHS.FunctionName) == 0;
|
FunctionName.equals(RHS.FunctionName);
|
||||||
}
|
}
|
||||||
bool operator!=(const DILineInfo &RHS) const {
|
bool operator!=(const DILineInfo &RHS) const {
|
||||||
return !(*this == RHS);
|
return !(*this == RHS);
|
||||||
@ -60,7 +61,8 @@ class DILineInfoSpecifier {
|
|||||||
public:
|
public:
|
||||||
enum Specification {
|
enum Specification {
|
||||||
FileLineInfo = 1 << 0,
|
FileLineInfo = 1 << 0,
|
||||||
FunctionName = 1 << 1
|
AbsoluteFilePath = 1 << 1,
|
||||||
|
FunctionName = 1 << 2
|
||||||
};
|
};
|
||||||
// Use file/line info by default.
|
// Use file/line info by default.
|
||||||
DILineInfoSpecifier(uint32_t flags = FileLineInfo) : Flags(flags) {}
|
DILineInfoSpecifier(uint32_t flags = FileLineInfo) : Flags(flags) {}
|
||||||
|
@ -97,6 +97,13 @@ void DWARFCompileUnit::dump(raw_ostream &OS) {
|
|||||||
getCompileUnitDIE(false)->dump(OS, this, -1U);
|
getCompileUnitDIE(false)->dump(OS, this, -1U);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const char *DWARFCompileUnit::getCompilationDir() {
|
||||||
|
extractDIEsIfNeeded(true);
|
||||||
|
if (DieArray.empty())
|
||||||
|
return 0;
|
||||||
|
return DieArray[0].getAttributeValueAsString(this, DW_AT_comp_dir, 0);
|
||||||
|
}
|
||||||
|
|
||||||
void DWARFCompileUnit::setDIERelations() {
|
void DWARFCompileUnit::setDIERelations() {
|
||||||
if (DieArray.empty())
|
if (DieArray.empty())
|
||||||
return;
|
return;
|
||||||
|
@ -78,6 +78,8 @@ public:
|
|||||||
return &DieArray[0];
|
return &DieArray[0];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const char *getCompilationDir();
|
||||||
|
|
||||||
/// setDIERelations - We read in all of the DIE entries into our flat list
|
/// setDIERelations - We read in all of the DIE entries into our flat list
|
||||||
/// of DIE entries and now we need to go back through all of them and set the
|
/// of DIE entries and now we need to go back through all of them and set the
|
||||||
/// parent, sibling and child pointers for quick DIE navigation.
|
/// parent, sibling and child pointers for quick DIE navigation.
|
||||||
|
@ -8,8 +8,10 @@
|
|||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
|
|
||||||
#include "DWARFContext.h"
|
#include "DWARFContext.h"
|
||||||
|
#include "llvm/ADT/SmallString.h"
|
||||||
#include "llvm/Support/Dwarf.h"
|
#include "llvm/Support/Dwarf.h"
|
||||||
#include "llvm/Support/Format.h"
|
#include "llvm/Support/Format.h"
|
||||||
|
#include "llvm/Support/Path.h"
|
||||||
#include "llvm/Support/raw_ostream.h"
|
#include "llvm/Support/raw_ostream.h"
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
using namespace llvm;
|
using namespace llvm;
|
||||||
@ -148,8 +150,8 @@ DILineInfo DWARFContext::getLineInfoForAddress(uint64_t address,
|
|||||||
DWARFCompileUnit *cu = getCompileUnitForOffset(cuOffset);
|
DWARFCompileUnit *cu = getCompileUnitForOffset(cuOffset);
|
||||||
if (!cu)
|
if (!cu)
|
||||||
return DILineInfo();
|
return DILineInfo();
|
||||||
const char *fileName = "<invalid>";
|
SmallString<16> fileName("<invalid>");
|
||||||
const char *functionName = "<invalid>";
|
SmallString<16> functionName("<invalid>");
|
||||||
uint32_t line = 0;
|
uint32_t line = 0;
|
||||||
uint32_t column = 0;
|
uint32_t column = 0;
|
||||||
if (specifier.needs(DILineInfoSpecifier::FunctionName)) {
|
if (specifier.needs(DILineInfoSpecifier::FunctionName)) {
|
||||||
@ -171,7 +173,29 @@ DILineInfo DWARFContext::getLineInfoForAddress(uint64_t address,
|
|||||||
if (rowIndex != -1U) {
|
if (rowIndex != -1U) {
|
||||||
const DWARFDebugLine::Row &row = lineTable->Rows[rowIndex];
|
const DWARFDebugLine::Row &row = lineTable->Rows[rowIndex];
|
||||||
// Take file/line info from the line table.
|
// Take file/line info from the line table.
|
||||||
fileName = lineTable->Prologue.FileNames[row.File - 1].Name.c_str();
|
const DWARFDebugLine::FileNameEntry &fileNameEntry =
|
||||||
|
lineTable->Prologue.FileNames[row.File - 1];
|
||||||
|
fileName = fileNameEntry.Name;
|
||||||
|
if (specifier.needs(DILineInfoSpecifier::AbsoluteFilePath) &&
|
||||||
|
sys::path::is_relative(fileName.str())) {
|
||||||
|
// Append include directory of file (if it is present in line table)
|
||||||
|
// and compilation directory of compile unit to make path absolute.
|
||||||
|
const char *includeDir = 0;
|
||||||
|
if (uint64_t includeDirIndex = fileNameEntry.DirIdx) {
|
||||||
|
includeDir = lineTable->Prologue
|
||||||
|
.IncludeDirectories[includeDirIndex - 1];
|
||||||
|
}
|
||||||
|
SmallString<16> absFileName;
|
||||||
|
if (includeDir == 0 || sys::path::is_relative(includeDir)) {
|
||||||
|
if (const char *compilationDir = cu->getCompilationDir())
|
||||||
|
sys::path::append(absFileName, compilationDir);
|
||||||
|
}
|
||||||
|
if (includeDir) {
|
||||||
|
sys::path::append(absFileName, includeDir);
|
||||||
|
}
|
||||||
|
sys::path::append(absFileName, fileName.str());
|
||||||
|
fileName = absFileName;
|
||||||
|
}
|
||||||
line = row.Line;
|
line = row.Line;
|
||||||
column = row.Column;
|
column = row.Column;
|
||||||
}
|
}
|
||||||
|
@ -12,7 +12,6 @@
|
|||||||
|
|
||||||
#include "llvm/Support/DataExtractor.h"
|
#include "llvm/Support/DataExtractor.h"
|
||||||
#include <map>
|
#include <map>
|
||||||
#include <string>
|
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
namespace llvm {
|
namespace llvm {
|
||||||
@ -22,9 +21,9 @@ class raw_ostream;
|
|||||||
class DWARFDebugLine {
|
class DWARFDebugLine {
|
||||||
public:
|
public:
|
||||||
struct FileNameEntry {
|
struct FileNameEntry {
|
||||||
FileNameEntry() : DirIdx(0), ModTime(0), Length(0) {}
|
FileNameEntry() : Name(0), DirIdx(0), ModTime(0), Length(0) {}
|
||||||
|
|
||||||
std::string Name;
|
const char *Name;
|
||||||
uint64_t DirIdx;
|
uint64_t DirIdx;
|
||||||
uint64_t ModTime;
|
uint64_t ModTime;
|
||||||
uint64_t Length;
|
uint64_t Length;
|
||||||
@ -56,7 +55,7 @@ public:
|
|||||||
// The number assigned to the first special opcode.
|
// The number assigned to the first special opcode.
|
||||||
uint8_t OpcodeBase;
|
uint8_t OpcodeBase;
|
||||||
std::vector<uint8_t> StandardOpcodeLengths;
|
std::vector<uint8_t> StandardOpcodeLengths;
|
||||||
std::vector<std::string> IncludeDirectories;
|
std::vector<const char*> IncludeDirectories;
|
||||||
std::vector<FileNameEntry> FileNames;
|
std::vector<FileNameEntry> FileNames;
|
||||||
|
|
||||||
// Length of the prologue in bytes.
|
// Length of the prologue in bytes.
|
||||||
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
@ -9,22 +9,32 @@ RUN: --address=0x4004b8 --functions | FileCheck %s -check-prefix MANY_CU_1
|
|||||||
RUN: llvm-dwarfdump %p/Inputs/dwarfdump-test2.elf-x86-64 \
|
RUN: llvm-dwarfdump %p/Inputs/dwarfdump-test2.elf-x86-64 \
|
||||||
RUN: --address=0x4004c4 --functions | FileCheck %s -check-prefix MANY_CU_2
|
RUN: --address=0x4004c4 --functions | FileCheck %s -check-prefix MANY_CU_2
|
||||||
RUN: llvm-dwarfdump %p/Inputs/dwarfdump-test3.elf-x86-64 \
|
RUN: llvm-dwarfdump %p/Inputs/dwarfdump-test3.elf-x86-64 \
|
||||||
RUN: --address=0x538 --functions | FileCheck %s -check-prefix ABS_ORIGIN_1
|
RUN: --address=0x580 --functions | FileCheck %s -check-prefix ABS_ORIGIN_1
|
||||||
|
RUN: llvm-dwarfdump %p/Inputs/dwarfdump-test3.elf-x86-64 \
|
||||||
|
RUN: --address=0x573 --functions | FileCheck %s -check-prefix INCLUDE_TEST_1
|
||||||
|
RUN: llvm-dwarfdump %p/Inputs/dwarfdump-test3.elf-x86-64 \
|
||||||
|
RUN: --address=0x56d --functions | FileCheck %s -check-prefix INCLUDE_TEST_2
|
||||||
|
|
||||||
MAIN: main
|
MAIN: main
|
||||||
MAIN-NEXT: dwarfdump-test.cc:16:10
|
MAIN-NEXT: /tmp/dbginfo/dwarfdump-test.cc:16:10
|
||||||
|
|
||||||
FUNCTION: _Z1fii
|
FUNCTION: _Z1fii
|
||||||
FUNCTION-NEXT: dwarfdump-test.cc:11:18
|
FUNCTION-NEXT: /tmp/dbginfo/dwarfdump-test.cc:11:18
|
||||||
|
|
||||||
CTOR_WITH_SPEC: _ZN10DummyClassC1Ei
|
CTOR_WITH_SPEC: _ZN10DummyClassC1Ei
|
||||||
CTOR_WITH_SPEC-NEXT: dwarfdump-test.cc:4:30
|
CTOR_WITH_SPEC-NEXT: /tmp/dbginfo/dwarfdump-test.cc:4:30
|
||||||
|
|
||||||
MANY_CU_1: a
|
MANY_CU_1: a
|
||||||
MANY_CU_1-NEXT: a.cc:2:0
|
MANY_CU_1-NEXT: /tmp/dbginfo/a.cc:2:0
|
||||||
|
|
||||||
MANY_CU_2: main
|
MANY_CU_2: main
|
||||||
MANY_CU_2-NEXT: main.cc:4:0
|
MANY_CU_2-NEXT: /tmp/dbginfo/main.cc:4:0
|
||||||
|
|
||||||
ABS_ORIGIN_1: C
|
ABS_ORIGIN_1: C
|
||||||
ABS_ORIGIN_1-NEXT: def.cc:3:0
|
ABS_ORIGIN_1-NEXT: /tmp/dbginfo/def2.cc:4:0
|
||||||
|
|
||||||
|
INCLUDE_TEST_1: _Z3do2v
|
||||||
|
INCLUDE_TEST_1-NEXT: /tmp/dbginfo/include/decl2.h:1:0
|
||||||
|
|
||||||
|
INCLUDE_TEST_2: _Z3do1v
|
||||||
|
INCLUDE_TEST_2-NEXT: /tmp/include/decl.h:5:0
|
||||||
|
@ -97,7 +97,8 @@ static void DumpInput(const StringRef &Filename) {
|
|||||||
dictx->dump(outs());
|
dictx->dump(outs());
|
||||||
} else {
|
} else {
|
||||||
// Print line info for the specified address.
|
// Print line info for the specified address.
|
||||||
int spec_flags = DILineInfoSpecifier::FileLineInfo;
|
int spec_flags = DILineInfoSpecifier::FileLineInfo |
|
||||||
|
DILineInfoSpecifier::AbsoluteFilePath;
|
||||||
if (PrintFunctions)
|
if (PrintFunctions)
|
||||||
spec_flags |= DILineInfoSpecifier::FunctionName;
|
spec_flags |= DILineInfoSpecifier::FunctionName;
|
||||||
DILineInfo dli = dictx->getLineInfoForAddress(Address, spec_flags);
|
DILineInfo dli = dictx->getLineInfoForAddress(Address, spec_flags);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user