blob: 2229a3aa987193597b506f669a3e1da5d5db5ad4 [file] [log] [blame]
Eric Christopherd7169e92012-10-16 23:46:21 +00001//===-- llvm-dwarfdump.cpp - Debug info dumping utility for llvm ----------===//
Benjamin Kramer72c0d7f2011-09-13 19:42:23 +00002//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// This program is a utility that works like "dwarfdump".
11//
12//===----------------------------------------------------------------------===//
13
14#include "llvm/ADT/OwningPtr.h"
15#include "llvm/ADT/Triple.h"
16#include "llvm/ADT/STLExtras.h"
17#include "llvm/Object/ObjectFile.h"
Eric Christopher806e03d2012-11-07 23:22:07 +000018#include "llvm/Object/RelocVisitor.h"
Benjamin Kramer72c0d7f2011-09-13 19:42:23 +000019#include "llvm/DebugInfo/DIContext.h"
20#include "llvm/Support/CommandLine.h"
21#include "llvm/Support/Debug.h"
22#include "llvm/Support/Format.h"
23#include "llvm/Support/ManagedStatic.h"
24#include "llvm/Support/MemoryBuffer.h"
25#include "llvm/Support/MemoryObject.h"
26#include "llvm/Support/PrettyStackTrace.h"
27#include "llvm/Support/Signals.h"
28#include "llvm/Support/raw_ostream.h"
29#include "llvm/Support/system_error.h"
30#include <algorithm>
31#include <cstring>
Eric Christopher806e03d2012-11-07 23:22:07 +000032#include <list>
33#include <string>
34
Benjamin Kramer72c0d7f2011-09-13 19:42:23 +000035using namespace llvm;
36using namespace object;
37
38static cl::list<std::string>
39InputFilenames(cl::Positional, cl::desc("<input object files>"),
40 cl::ZeroOrMore);
41
Benjamin Kramer6b3ae462011-09-15 21:17:40 +000042static cl::opt<unsigned long long>
43Address("address", cl::init(-1ULL),
44 cl::desc("Print line information for a given address"));
45
Alexey Samsonov3e25c4a2012-07-02 05:54:45 +000046static cl::opt<bool>
47PrintFunctions("functions", cl::init(false),
48 cl::desc("Print function names as well as line information "
49 "for a given address"));
50
Alexey Samsonov5eae90d2012-09-04 08:12:33 +000051static cl::opt<bool>
52PrintInlining("inlining", cl::init(false),
53 cl::desc("Print all inlined frames for a given address"));
54
55static void PrintDILineInfo(DILineInfo dli) {
56 if (PrintFunctions)
57 outs() << (dli.getFunctionName() ? dli.getFunctionName() : "<unknown>")
58 << "\n";
59 outs() << (dli.getFileName() ? dli.getFileName() : "<unknown>") << ':'
60 << dli.getLine() << ':' << dli.getColumn() << '\n';
61}
62
Benjamin Kramer72c0d7f2011-09-13 19:42:23 +000063static void DumpInput(const StringRef &Filename) {
64 OwningPtr<MemoryBuffer> Buff;
65
66 if (error_code ec = MemoryBuffer::getFileOrSTDIN(Filename, Buff)) {
67 errs() << Filename << ": " << ec.message() << "\n";
68 return;
69 }
70
71 OwningPtr<ObjectFile> Obj(ObjectFile::createObjectFile(Buff.take()));
Eric Christopherd1726a42012-11-12 21:40:38 +000072 OwningPtr<DIContext> dictx(DIContext::getDWARFContext(Obj.get()));
Benjamin Kramer72c0d7f2011-09-13 19:42:23 +000073
Benjamin Kramer6b3ae462011-09-15 21:17:40 +000074 if (Address == -1ULL) {
75 outs() << Filename
76 << ":\tfile format " << Obj->getFileFormatName() << "\n\n";
77 // Dump the complete DWARF structure.
78 dictx->dump(outs());
79 } else {
80 // Print line info for the specified address.
Alexey Samsonov5eae90d2012-09-04 08:12:33 +000081 int SpecFlags = DILineInfoSpecifier::FileLineInfo |
82 DILineInfoSpecifier::AbsoluteFilePath;
Alexey Samsonov3e25c4a2012-07-02 05:54:45 +000083 if (PrintFunctions)
Alexey Samsonov5eae90d2012-09-04 08:12:33 +000084 SpecFlags |= DILineInfoSpecifier::FunctionName;
85 if (PrintInlining) {
Eric Christopherd7169e92012-10-16 23:46:21 +000086 DIInliningInfo InliningInfo =
87 dictx->getInliningInfoForAddress(Address, SpecFlags);
Alexey Samsonov5eae90d2012-09-04 08:12:33 +000088 uint32_t n = InliningInfo.getNumberOfFrames();
89 if (n == 0) {
90 // Print one empty debug line info in any case.
91 PrintDILineInfo(DILineInfo());
92 } else {
93 for (uint32_t i = 0; i < n; i++) {
94 DILineInfo dli = InliningInfo.getFrame(i);
95 PrintDILineInfo(dli);
96 }
97 }
98 } else {
99 DILineInfo dli = dictx->getLineInfoForAddress(Address, SpecFlags);
100 PrintDILineInfo(dli);
101 }
Benjamin Kramer6b3ae462011-09-15 21:17:40 +0000102 }
Benjamin Kramer72c0d7f2011-09-13 19:42:23 +0000103}
104
105int main(int argc, char **argv) {
106 // Print a stack trace if we signal out.
107 sys::PrintStackTraceOnErrorSignal();
108 PrettyStackTraceProgram X(argc, argv);
109 llvm_shutdown_obj Y; // Call llvm_shutdown() on exit.
110
111 cl::ParseCommandLineOptions(argc, argv, "llvm dwarf dumper\n");
112
113 // Defaults to a.out if no filenames specified.
114 if (InputFilenames.size() == 0)
115 InputFilenames.push_back("a.out");
116
117 std::for_each(InputFilenames.begin(), InputFilenames.end(), DumpInput);
118
119 return 0;
120}