Brian Gaeke | 0af759d | 2003-10-16 04:43:15 +0000 | [diff] [blame] | 1 | //===-- llvm-nm.cpp - Symbol table dumping utility for llvm ---------------===// |
Misha Brukman | 650ba8e | 2005-04-22 00:00:37 +0000 | [diff] [blame] | 2 | // |
John Criswell | 09344dc | 2003-10-20 17:47:21 +0000 | [diff] [blame] | 3 | // The LLVM Compiler Infrastructure |
| 4 | // |
Chris Lattner | 345353d | 2007-12-29 20:44:31 +0000 | [diff] [blame] | 5 | // This file is distributed under the University of Illinois Open Source |
| 6 | // License. See LICENSE.TXT for details. |
Misha Brukman | 650ba8e | 2005-04-22 00:00:37 +0000 | [diff] [blame] | 7 | // |
John Criswell | 09344dc | 2003-10-20 17:47:21 +0000 | [diff] [blame] | 8 | //===----------------------------------------------------------------------===// |
Brian Gaeke | 0af759d | 2003-10-16 04:43:15 +0000 | [diff] [blame] | 9 | // |
Michael J. Spencer | 838e5ad | 2012-06-06 23:34:10 +0000 | [diff] [blame] | 10 | // This program is a utility that works like traditional Unix "nm", that is, it |
| 11 | // prints out the names of symbols in a bitcode or object file, along with some |
| 12 | // information about each symbol. |
Misha Brukman | 650ba8e | 2005-04-22 00:00:37 +0000 | [diff] [blame] | 13 | // |
Michael J. Spencer | 838e5ad | 2012-06-06 23:34:10 +0000 | [diff] [blame] | 14 | // This "nm" supports many of the features of GNU "nm", including its different |
| 15 | // output formats. |
Brian Gaeke | 0af759d | 2003-10-16 04:43:15 +0000 | [diff] [blame] | 16 | // |
| 17 | //===----------------------------------------------------------------------===// |
| 18 | |
Rafael Espindola | f12b828 | 2014-02-21 20:10:59 +0000 | [diff] [blame] | 19 | #include "llvm/IR/Function.h" |
| 20 | #include "llvm/IR/GlobalAlias.h" |
| 21 | #include "llvm/IR/GlobalVariable.h" |
Chandler Carruth | 9fb823b | 2013-01-02 11:36:10 +0000 | [diff] [blame] | 22 | #include "llvm/IR/LLVMContext.h" |
Michael J. Spencer | 2bc774a | 2011-09-27 19:37:18 +0000 | [diff] [blame] | 23 | #include "llvm/Object/Archive.h" |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 24 | #include "llvm/Object/COFF.h" |
| 25 | #include "llvm/Object/ELFObjectFile.h" |
Chandler Carruth | 442f784 | 2014-03-04 10:07:28 +0000 | [diff] [blame] | 26 | #include "llvm/Object/IRObjectFile.h" |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 27 | #include "llvm/Object/MachO.h" |
Alexey Samsonov | e6388e6 | 2013-06-18 15:03:28 +0000 | [diff] [blame] | 28 | #include "llvm/Object/MachOUniversal.h" |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 29 | #include "llvm/Object/ObjectFile.h" |
Rui Ueyama | f078eff | 2014-03-18 23:37:53 +0000 | [diff] [blame] | 30 | #include "llvm/Support/COFF.h" |
Reid Spencer | 7c16caa | 2004-09-01 22:55:40 +0000 | [diff] [blame] | 31 | #include "llvm/Support/CommandLine.h" |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 32 | #include "llvm/Support/FileSystem.h" |
Chandler Carruth | 4d88a1c | 2012-12-04 10:44:52 +0000 | [diff] [blame] | 33 | #include "llvm/Support/Format.h" |
Chris Lattner | 76d4632 | 2006-12-06 01:18:01 +0000 | [diff] [blame] | 34 | #include "llvm/Support/ManagedStatic.h" |
Chris Lattner | ef8f389 | 2007-05-06 05:36:18 +0000 | [diff] [blame] | 35 | #include "llvm/Support/MemoryBuffer.h" |
Chris Lattner | e3fc2d1 | 2009-03-06 05:34:10 +0000 | [diff] [blame] | 36 | #include "llvm/Support/PrettyStackTrace.h" |
Michael J. Spencer | bc96f37 | 2011-12-13 23:17:29 +0000 | [diff] [blame] | 37 | #include "llvm/Support/Program.h" |
Michael J. Spencer | 447762d | 2010-11-29 18:16:10 +0000 | [diff] [blame] | 38 | #include "llvm/Support/Signals.h" |
Chandler Carruth | 4d88a1c | 2012-12-04 10:44:52 +0000 | [diff] [blame] | 39 | #include "llvm/Support/raw_ostream.h" |
Michael J. Spencer | 7b6fef8 | 2010-12-09 17:36:48 +0000 | [diff] [blame] | 40 | #include "llvm/Support/system_error.h" |
Jeff Cohen | b622c11 | 2007-03-05 00:00:42 +0000 | [diff] [blame] | 41 | #include <algorithm> |
Brian Gaeke | 0af759d | 2003-10-16 04:43:15 +0000 | [diff] [blame] | 42 | #include <cctype> |
Alkis Evlogimenos | f68f40e | 2004-04-21 16:11:40 +0000 | [diff] [blame] | 43 | #include <cerrno> |
Brian Gaeke | 55447b4 | 2003-11-19 21:52:09 +0000 | [diff] [blame] | 44 | #include <cstring> |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 45 | #include <vector> |
Brian Gaeke | 960707c | 2003-11-11 22:41:34 +0000 | [diff] [blame] | 46 | using namespace llvm; |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 47 | using namespace object; |
Brian Gaeke | 960707c | 2003-11-11 22:41:34 +0000 | [diff] [blame] | 48 | |
Brian Gaeke | 0af759d | 2003-10-16 04:43:15 +0000 | [diff] [blame] | 49 | namespace { |
Rafael Espindola | ce82a07 | 2014-01-30 19:24:00 +0000 | [diff] [blame] | 50 | enum OutputFormatTy { bsd, sysv, posix }; |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 51 | cl::opt<OutputFormatTy> OutputFormat( |
| 52 | "format", cl::desc("Specify output format"), |
| 53 | cl::values(clEnumVal(bsd, "BSD format"), clEnumVal(sysv, "System V format"), |
| 54 | clEnumVal(posix, "POSIX.2 format"), clEnumValEnd), |
| 55 | cl::init(bsd)); |
| 56 | cl::alias OutputFormat2("f", cl::desc("Alias for --format"), |
| 57 | cl::aliasopt(OutputFormat)); |
Brian Gaeke | 0af759d | 2003-10-16 04:43:15 +0000 | [diff] [blame] | 58 | |
Rafael Espindola | f12b828 | 2014-02-21 20:10:59 +0000 | [diff] [blame] | 59 | cl::list<std::string> InputFilenames(cl::Positional, cl::desc("<input files>"), |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 60 | cl::ZeroOrMore); |
Brian Gaeke | 0af759d | 2003-10-16 04:43:15 +0000 | [diff] [blame] | 61 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 62 | cl::opt<bool> UndefinedOnly("undefined-only", |
| 63 | cl::desc("Show only undefined symbols")); |
| 64 | cl::alias UndefinedOnly2("u", cl::desc("Alias for --undefined-only"), |
| 65 | cl::aliasopt(UndefinedOnly)); |
Brian Gaeke | 0af759d | 2003-10-16 04:43:15 +0000 | [diff] [blame] | 66 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 67 | cl::opt<bool> DynamicSyms("dynamic", |
| 68 | cl::desc("Display the dynamic symbols instead " |
| 69 | "of normal symbols.")); |
| 70 | cl::alias DynamicSyms2("D", cl::desc("Alias for --dynamic"), |
| 71 | cl::aliasopt(DynamicSyms)); |
Michael J. Spencer | 8c4729f | 2012-02-28 00:40:37 +0000 | [diff] [blame] | 72 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 73 | cl::opt<bool> DefinedOnly("defined-only", |
| 74 | cl::desc("Show only defined symbols")); |
Brian Gaeke | 0af759d | 2003-10-16 04:43:15 +0000 | [diff] [blame] | 75 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 76 | cl::opt<bool> ExternalOnly("extern-only", |
| 77 | cl::desc("Show only external symbols")); |
| 78 | cl::alias ExternalOnly2("g", cl::desc("Alias for --extern-only"), |
| 79 | cl::aliasopt(ExternalOnly)); |
Brian Gaeke | 0af759d | 2003-10-16 04:43:15 +0000 | [diff] [blame] | 80 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 81 | cl::opt<bool> BSDFormat("B", cl::desc("Alias for --format=bsd")); |
| 82 | cl::opt<bool> POSIXFormat("P", cl::desc("Alias for --format=posix")); |
Brian Gaeke | 0af759d | 2003-10-16 04:43:15 +0000 | [diff] [blame] | 83 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 84 | cl::opt<bool> PrintFileName( |
| 85 | "print-file-name", |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 86 | cl::desc("Precede each symbol with the object file it came from")); |
| 87 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 88 | cl::alias PrintFileNameA("A", cl::desc("Alias for --print-file-name"), |
| 89 | cl::aliasopt(PrintFileName)); |
| 90 | cl::alias PrintFileNameo("o", cl::desc("Alias for --print-file-name"), |
| 91 | cl::aliasopt(PrintFileName)); |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 92 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 93 | cl::opt<bool> DebugSyms("debug-syms", |
| 94 | cl::desc("Show all symbols, even debugger only")); |
| 95 | cl::alias DebugSymsa("a", cl::desc("Alias for --debug-syms"), |
| 96 | cl::aliasopt(DebugSyms)); |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 97 | |
Rafael Espindola | b4865d6 | 2014-04-03 00:19:35 +0000 | [diff] [blame] | 98 | cl::opt<bool> NumericSort("numeric-sort", cl::desc("Sort symbols by address")); |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 99 | cl::alias NumericSortn("n", cl::desc("Alias for --numeric-sort"), |
| 100 | cl::aliasopt(NumericSort)); |
| 101 | cl::alias NumericSortv("v", cl::desc("Alias for --numeric-sort"), |
| 102 | cl::aliasopt(NumericSort)); |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 103 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 104 | cl::opt<bool> NoSort("no-sort", cl::desc("Show symbols in order encountered")); |
| 105 | cl::alias NoSortp("p", cl::desc("Alias for --no-sort"), cl::aliasopt(NoSort)); |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 106 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 107 | cl::opt<bool> PrintSize("print-size", |
Rafael Espindola | b4865d6 | 2014-04-03 00:19:35 +0000 | [diff] [blame] | 108 | cl::desc("Show symbol size instead of address")); |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 109 | cl::alias PrintSizeS("S", cl::desc("Alias for --print-size"), |
| 110 | cl::aliasopt(PrintSize)); |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 111 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 112 | cl::opt<bool> SizeSort("size-sort", cl::desc("Sort symbols by size")); |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 113 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 114 | cl::opt<bool> WithoutAliases("without-aliases", cl::Hidden, |
| 115 | cl::desc("Exclude aliases from output")); |
Jan Sjödin | 4d0c299 | 2012-09-18 18:47:58 +0000 | [diff] [blame] | 116 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 117 | cl::opt<bool> ArchiveMap("print-armap", cl::desc("Print the archive map")); |
| 118 | cl::alias ArchiveMaps("s", cl::desc("Alias for --print-armap"), |
| 119 | cl::aliasopt(ArchiveMap)); |
Rafael Espindola | b4865d6 | 2014-04-03 00:19:35 +0000 | [diff] [blame] | 120 | bool PrintAddress = true; |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 121 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 122 | bool MultipleFiles = false; |
Brian Gaeke | 0af759d | 2003-10-16 04:43:15 +0000 | [diff] [blame] | 123 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 124 | bool HadError = false; |
Rafael Espindola | 8b82a4d | 2013-07-03 15:46:03 +0000 | [diff] [blame] | 125 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 126 | std::string ToolName; |
Chris Lattner | aa237256 | 2006-05-24 17:04:05 +0000 | [diff] [blame] | 127 | } |
Brian Gaeke | 0af759d | 2003-10-16 04:43:15 +0000 | [diff] [blame] | 128 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 129 | static void error(Twine Message, Twine Path = Twine()) { |
| 130 | HadError = true; |
| 131 | errs() << ToolName << ": " << Path << ": " << Message << ".\n"; |
Michael J. Spencer | bc96f37 | 2011-12-13 23:17:29 +0000 | [diff] [blame] | 132 | } |
| 133 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 134 | static bool error(error_code EC, Twine Path = Twine()) { |
| 135 | if (EC) { |
| 136 | error(EC.message(), Path); |
Michael J. Spencer | bc96f37 | 2011-12-13 23:17:29 +0000 | [diff] [blame] | 137 | return true; |
| 138 | } |
| 139 | return false; |
| 140 | } |
| 141 | |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 142 | namespace { |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 143 | struct NMSymbol { |
Rafael Espindola | b4865d6 | 2014-04-03 00:19:35 +0000 | [diff] [blame] | 144 | uint64_t Address; |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 145 | uint64_t Size; |
| 146 | char TypeChar; |
| 147 | StringRef Name; |
| 148 | }; |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 149 | } |
| 150 | |
Rafael Espindola | b4865d6 | 2014-04-03 00:19:35 +0000 | [diff] [blame] | 151 | static bool compareSymbolAddress(const NMSymbol &A, const NMSymbol &B) { |
| 152 | if (A.Address < B.Address) |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 153 | return true; |
Rafael Espindola | b4865d6 | 2014-04-03 00:19:35 +0000 | [diff] [blame] | 154 | else if (A.Address == B.Address && A.Name < B.Name) |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 155 | return true; |
Rafael Espindola | b4865d6 | 2014-04-03 00:19:35 +0000 | [diff] [blame] | 156 | else if (A.Address == B.Address && A.Name == B.Name && A.Size < B.Size) |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 157 | return true; |
| 158 | else |
| 159 | return false; |
| 160 | } |
| 161 | |
| 162 | static bool compareSymbolSize(const NMSymbol &A, const NMSymbol &B) { |
| 163 | if (A.Size < B.Size) |
| 164 | return true; |
| 165 | else if (A.Size == B.Size && A.Name < B.Name) |
| 166 | return true; |
Rafael Espindola | b4865d6 | 2014-04-03 00:19:35 +0000 | [diff] [blame] | 167 | else if (A.Size == B.Size && A.Name == B.Name && A.Address < B.Address) |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 168 | return true; |
| 169 | else |
| 170 | return false; |
| 171 | } |
| 172 | |
| 173 | static bool compareSymbolName(const NMSymbol &A, const NMSymbol &B) { |
| 174 | if (A.Name < B.Name) |
| 175 | return true; |
| 176 | else if (A.Name == B.Name && A.Size < B.Size) |
| 177 | return true; |
Rafael Espindola | b4865d6 | 2014-04-03 00:19:35 +0000 | [diff] [blame] | 178 | else if (A.Name == B.Name && A.Size == B.Size && A.Address < B.Address) |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 179 | return true; |
| 180 | else |
| 181 | return false; |
| 182 | } |
| 183 | |
Kevin Enderby | 6abc2e5 | 2014-05-09 23:57:49 +0000 | [diff] [blame^] | 184 | static char isSymbolList64Bit(SymbolicFile *Obj) { |
| 185 | if (dyn_cast<IRObjectFile>(Obj)) |
| 186 | return false; |
| 187 | else if (dyn_cast<COFFObjectFile>(Obj)) |
| 188 | return false; |
| 189 | else if (MachOObjectFile *MachO = dyn_cast<MachOObjectFile>(Obj)) |
| 190 | return MachO->is64Bit(); |
| 191 | else if (dyn_cast<ELF32LEObjectFile>(Obj)) |
| 192 | return false; |
| 193 | else if (dyn_cast<ELF64LEObjectFile>(Obj)) |
| 194 | return true; |
| 195 | else if (dyn_cast<ELF32BEObjectFile>(Obj)) |
| 196 | return false; |
| 197 | else if(dyn_cast<ELF64BEObjectFile>(Obj)) |
| 198 | return true; |
| 199 | else |
| 200 | return false; |
| 201 | } |
| 202 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 203 | static StringRef CurrentFilename; |
| 204 | typedef std::vector<NMSymbol> SymbolListT; |
| 205 | static SymbolListT SymbolList; |
| 206 | |
Kevin Enderby | 6abc2e5 | 2014-05-09 23:57:49 +0000 | [diff] [blame^] | 207 | static void sortAndPrintSymbolList(SymbolicFile *Obj) { |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 208 | if (!NoSort) { |
| 209 | if (NumericSort) |
Rafael Espindola | b4865d6 | 2014-04-03 00:19:35 +0000 | [diff] [blame] | 210 | std::sort(SymbolList.begin(), SymbolList.end(), compareSymbolAddress); |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 211 | else if (SizeSort) |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 212 | std::sort(SymbolList.begin(), SymbolList.end(), compareSymbolSize); |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 213 | else |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 214 | std::sort(SymbolList.begin(), SymbolList.end(), compareSymbolName); |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 215 | } |
| 216 | |
| 217 | if (OutputFormat == posix && MultipleFiles) { |
| 218 | outs() << '\n' << CurrentFilename << ":\n"; |
| 219 | } else if (OutputFormat == bsd && MultipleFiles) { |
| 220 | outs() << "\n" << CurrentFilename << ":\n"; |
| 221 | } else if (OutputFormat == sysv) { |
| 222 | outs() << "\n\nSymbols from " << CurrentFilename << ":\n\n" |
| 223 | << "Name Value Class Type" |
| 224 | << " Size Line Section\n"; |
| 225 | } |
| 226 | |
Kevin Enderby | 6abc2e5 | 2014-05-09 23:57:49 +0000 | [diff] [blame^] | 227 | const char *printBlanks, *printFormat; |
| 228 | if (isSymbolList64Bit(Obj)) { |
| 229 | printBlanks = " "; |
| 230 | printFormat = "%016" PRIx64; |
| 231 | } else { |
| 232 | printBlanks = " "; |
| 233 | printFormat = "%08" PRIx64; |
| 234 | } |
| 235 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 236 | for (SymbolListT::iterator I = SymbolList.begin(), E = SymbolList.end(); |
| 237 | I != E; ++I) { |
| 238 | if ((I->TypeChar != 'U') && UndefinedOnly) |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 239 | continue; |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 240 | if ((I->TypeChar == 'U') && DefinedOnly) |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 241 | continue; |
Rafael Espindola | b4865d6 | 2014-04-03 00:19:35 +0000 | [diff] [blame] | 242 | if (SizeSort && !PrintAddress && I->Size == UnknownAddressOrSize) |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 243 | continue; |
| 244 | |
Kevin Enderby | 6abc2e5 | 2014-05-09 23:57:49 +0000 | [diff] [blame^] | 245 | char SymbolAddrStr[18] = ""; |
| 246 | char SymbolSizeStr[18] = ""; |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 247 | |
Rafael Espindola | b4865d6 | 2014-04-03 00:19:35 +0000 | [diff] [blame] | 248 | if (OutputFormat == sysv || I->Address == UnknownAddressOrSize) |
Kevin Enderby | 6abc2e5 | 2014-05-09 23:57:49 +0000 | [diff] [blame^] | 249 | strcpy(SymbolAddrStr, printBlanks); |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 250 | if (OutputFormat == sysv) |
Kevin Enderby | 6abc2e5 | 2014-05-09 23:57:49 +0000 | [diff] [blame^] | 251 | strcpy(SymbolSizeStr, printBlanks); |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 252 | |
Rafael Espindola | b4865d6 | 2014-04-03 00:19:35 +0000 | [diff] [blame] | 253 | if (I->Address != UnknownAddressOrSize) |
Kevin Enderby | 6abc2e5 | 2014-05-09 23:57:49 +0000 | [diff] [blame^] | 254 | format(printFormat, I->Address) |
Rafael Espindola | b4865d6 | 2014-04-03 00:19:35 +0000 | [diff] [blame] | 255 | .print(SymbolAddrStr, sizeof(SymbolAddrStr)); |
Rafael Espindola | 0115b08 | 2014-01-30 21:51:42 +0000 | [diff] [blame] | 256 | if (I->Size != UnknownAddressOrSize) |
Kevin Enderby | 6abc2e5 | 2014-05-09 23:57:49 +0000 | [diff] [blame^] | 257 | format(printFormat, I->Size).print(SymbolSizeStr, sizeof(SymbolSizeStr)); |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 258 | |
| 259 | if (OutputFormat == posix) { |
Rafael Espindola | b4865d6 | 2014-04-03 00:19:35 +0000 | [diff] [blame] | 260 | outs() << I->Name << " " << I->TypeChar << " " << SymbolAddrStr |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 261 | << SymbolSizeStr << "\n"; |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 262 | } else if (OutputFormat == bsd) { |
Rafael Espindola | b4865d6 | 2014-04-03 00:19:35 +0000 | [diff] [blame] | 263 | if (PrintAddress) |
| 264 | outs() << SymbolAddrStr << ' '; |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 265 | if (PrintSize) { |
| 266 | outs() << SymbolSizeStr; |
Rafael Espindola | 0115b08 | 2014-01-30 21:51:42 +0000 | [diff] [blame] | 267 | if (I->Size != UnknownAddressOrSize) |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 268 | outs() << ' '; |
| 269 | } |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 270 | outs() << I->TypeChar << " " << I->Name << "\n"; |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 271 | } else if (OutputFormat == sysv) { |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 272 | std::string PaddedName(I->Name); |
| 273 | while (PaddedName.length() < 20) |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 274 | PaddedName += " "; |
Rafael Espindola | b4865d6 | 2014-04-03 00:19:35 +0000 | [diff] [blame] | 275 | outs() << PaddedName << "|" << SymbolAddrStr << "| " << I->TypeChar |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 276 | << " | |" << SymbolSizeStr << "| |\n"; |
| 277 | } |
| 278 | } |
| 279 | |
| 280 | SymbolList.clear(); |
| 281 | } |
| 282 | |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 283 | template <class ELFT> |
Rafael Espindola | f12b828 | 2014-02-21 20:10:59 +0000 | [diff] [blame] | 284 | static char getSymbolNMTypeChar(ELFObjectFile<ELFT> &Obj, |
| 285 | basic_symbol_iterator I) { |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 286 | typedef typename ELFObjectFile<ELFT>::Elf_Sym Elf_Sym; |
| 287 | typedef typename ELFObjectFile<ELFT>::Elf_Shdr Elf_Shdr; |
| 288 | |
Rafael Espindola | f12b828 | 2014-02-21 20:10:59 +0000 | [diff] [blame] | 289 | // OK, this is ELF |
| 290 | symbol_iterator SymI(I); |
| 291 | |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 292 | DataRefImpl Symb = I->getRawDataRefImpl(); |
| 293 | const Elf_Sym *ESym = Obj.getSymbol(Symb); |
| 294 | const ELFFile<ELFT> &EF = *Obj.getELFFile(); |
| 295 | const Elf_Shdr *ESec = EF.getSection(ESym); |
| 296 | |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 297 | if (ESec) { |
| 298 | switch (ESec->sh_type) { |
| 299 | case ELF::SHT_PROGBITS: |
| 300 | case ELF::SHT_DYNAMIC: |
| 301 | switch (ESec->sh_flags) { |
| 302 | case(ELF::SHF_ALLOC | ELF::SHF_EXECINSTR) : |
Rafael Espindola | 22fe9c1 | 2014-02-05 05:19:19 +0000 | [diff] [blame] | 303 | return 't'; |
Rafael Espindola | 9fa5215 | 2014-01-30 00:42:30 +0000 | [diff] [blame] | 304 | case(ELF::SHF_TLS | ELF::SHF_ALLOC | ELF::SHF_WRITE) : |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 305 | case(ELF::SHF_ALLOC | ELF::SHF_WRITE) : |
Rafael Espindola | 22fe9c1 | 2014-02-05 05:19:19 +0000 | [diff] [blame] | 306 | return 'd'; |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 307 | case ELF::SHF_ALLOC: |
| 308 | case(ELF::SHF_ALLOC | ELF::SHF_MERGE) : |
| 309 | case(ELF::SHF_ALLOC | ELF::SHF_MERGE | ELF::SHF_STRINGS) : |
Rafael Espindola | 22fe9c1 | 2014-02-05 05:19:19 +0000 | [diff] [blame] | 310 | return 'r'; |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 311 | } |
| 312 | break; |
| 313 | case ELF::SHT_NOBITS: |
Rafael Espindola | 22fe9c1 | 2014-02-05 05:19:19 +0000 | [diff] [blame] | 314 | return 'b'; |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 315 | } |
| 316 | } |
| 317 | |
Rafael Espindola | 22fe9c1 | 2014-02-05 05:19:19 +0000 | [diff] [blame] | 318 | if (ESym->getType() == ELF::STT_SECTION) { |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 319 | StringRef Name; |
Rafael Espindola | f12b828 | 2014-02-21 20:10:59 +0000 | [diff] [blame] | 320 | if (error(SymI->getName(Name))) |
Rafael Espindola | 7437589 | 2014-02-04 00:21:18 +0000 | [diff] [blame] | 321 | return '?'; |
| 322 | return StringSwitch<char>(Name) |
| 323 | .StartsWith(".debug", 'N') |
| 324 | .StartsWith(".note", 'n') |
| 325 | .Default('?'); |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 326 | } |
| 327 | |
Rafael Espindola | 22fe9c1 | 2014-02-05 05:19:19 +0000 | [diff] [blame] | 328 | return '?'; |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 329 | } |
| 330 | |
Rafael Espindola | 7437589 | 2014-02-04 00:21:18 +0000 | [diff] [blame] | 331 | static char getSymbolNMTypeChar(COFFObjectFile &Obj, symbol_iterator I) { |
Alexey Samsonov | 27dc839 | 2014-03-18 06:53:02 +0000 | [diff] [blame] | 332 | const coff_symbol *Symb = Obj.getCOFFSymbol(*I); |
Rafael Espindola | f12b828 | 2014-02-21 20:10:59 +0000 | [diff] [blame] | 333 | // OK, this is COFF. |
| 334 | symbol_iterator SymI(I); |
| 335 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 336 | StringRef Name; |
Rafael Espindola | f12b828 | 2014-02-21 20:10:59 +0000 | [diff] [blame] | 337 | if (error(SymI->getName(Name))) |
Rafael Espindola | 7437589 | 2014-02-04 00:21:18 +0000 | [diff] [blame] | 338 | return '?'; |
Rafael Espindola | f12b828 | 2014-02-21 20:10:59 +0000 | [diff] [blame] | 339 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 340 | char Ret = StringSwitch<char>(Name) |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 341 | .StartsWith(".debug", 'N') |
| 342 | .StartsWith(".sxdata", 'N') |
| 343 | .Default('?'); |
| 344 | |
Rafael Espindola | 7437589 | 2014-02-04 00:21:18 +0000 | [diff] [blame] | 345 | if (Ret != '?') |
| 346 | return Ret; |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 347 | |
| 348 | uint32_t Characteristics = 0; |
Rui Ueyama | f078eff | 2014-03-18 23:37:53 +0000 | [diff] [blame] | 349 | if (!COFF::isReservedSectionNumber(Symb->SectionNumber)) { |
Rafael Espindola | b5155a5 | 2014-02-10 20:24:04 +0000 | [diff] [blame] | 350 | section_iterator SecI = Obj.section_end(); |
Rafael Espindola | f12b828 | 2014-02-21 20:10:59 +0000 | [diff] [blame] | 351 | if (error(SymI->getSection(SecI))) |
Rafael Espindola | 7437589 | 2014-02-04 00:21:18 +0000 | [diff] [blame] | 352 | return '?'; |
Alexey Samsonov | 27dc839 | 2014-03-18 06:53:02 +0000 | [diff] [blame] | 353 | const coff_section *Section = Obj.getCOFFSection(*SecI); |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 354 | Characteristics = Section->Characteristics; |
| 355 | } |
| 356 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 357 | switch (Symb->SectionNumber) { |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 358 | case COFF::IMAGE_SYM_DEBUG: |
Rafael Espindola | 22fe9c1 | 2014-02-05 05:19:19 +0000 | [diff] [blame] | 359 | return 'n'; |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 360 | default: |
| 361 | // Check section type. |
| 362 | if (Characteristics & COFF::IMAGE_SCN_CNT_CODE) |
Rafael Espindola | 22fe9c1 | 2014-02-05 05:19:19 +0000 | [diff] [blame] | 363 | return 't'; |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 364 | else if (Characteristics & COFF::IMAGE_SCN_MEM_READ && |
| 365 | ~Characteristics & COFF::IMAGE_SCN_MEM_WRITE) // Read only. |
Rafael Espindola | 22fe9c1 | 2014-02-05 05:19:19 +0000 | [diff] [blame] | 366 | return 'r'; |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 367 | else if (Characteristics & COFF::IMAGE_SCN_CNT_INITIALIZED_DATA) |
Rafael Espindola | 22fe9c1 | 2014-02-05 05:19:19 +0000 | [diff] [blame] | 368 | return 'd'; |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 369 | else if (Characteristics & COFF::IMAGE_SCN_CNT_UNINITIALIZED_DATA) |
Rafael Espindola | 22fe9c1 | 2014-02-05 05:19:19 +0000 | [diff] [blame] | 370 | return 'b'; |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 371 | else if (Characteristics & COFF::IMAGE_SCN_LNK_INFO) |
Rafael Espindola | 22fe9c1 | 2014-02-05 05:19:19 +0000 | [diff] [blame] | 372 | return 'i'; |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 373 | |
| 374 | // Check for section symbol. |
David Majnemer | ddf28f2 | 2014-03-19 04:47:47 +0000 | [diff] [blame] | 375 | else if (Symb->isSectionDefinition()) |
Rafael Espindola | 22fe9c1 | 2014-02-05 05:19:19 +0000 | [diff] [blame] | 376 | return 's'; |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 377 | } |
| 378 | |
Rafael Espindola | 22fe9c1 | 2014-02-05 05:19:19 +0000 | [diff] [blame] | 379 | return '?'; |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 380 | } |
| 381 | |
| 382 | static uint8_t getNType(MachOObjectFile &Obj, DataRefImpl Symb) { |
| 383 | if (Obj.is64Bit()) { |
| 384 | MachO::nlist_64 STE = Obj.getSymbol64TableEntry(Symb); |
| 385 | return STE.n_type; |
| 386 | } |
| 387 | MachO::nlist STE = Obj.getSymbolTableEntry(Symb); |
| 388 | return STE.n_type; |
| 389 | } |
| 390 | |
Rafael Espindola | f12b828 | 2014-02-21 20:10:59 +0000 | [diff] [blame] | 391 | static char getSymbolNMTypeChar(MachOObjectFile &Obj, basic_symbol_iterator I) { |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 392 | DataRefImpl Symb = I->getRawDataRefImpl(); |
| 393 | uint8_t NType = getNType(Obj, Symb); |
| 394 | |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 395 | switch (NType & MachO::N_TYPE) { |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 396 | case MachO::N_ABS: |
Rafael Espindola | 22fe9c1 | 2014-02-05 05:19:19 +0000 | [diff] [blame] | 397 | return 's'; |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 398 | case MachO::N_SECT: { |
Rafael Espindola | b5155a5 | 2014-02-10 20:24:04 +0000 | [diff] [blame] | 399 | section_iterator Sec = Obj.section_end(); |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 400 | Obj.getSymbolSection(Symb, Sec); |
| 401 | DataRefImpl Ref = Sec->getRawDataRefImpl(); |
| 402 | StringRef SectionName; |
| 403 | Obj.getSectionName(Ref, SectionName); |
| 404 | StringRef SegmentName = Obj.getSectionFinalSegmentName(Ref); |
| 405 | if (SegmentName == "__TEXT" && SectionName == "__text") |
Rafael Espindola | 22fe9c1 | 2014-02-05 05:19:19 +0000 | [diff] [blame] | 406 | return 't'; |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 407 | else |
Rafael Espindola | 22fe9c1 | 2014-02-05 05:19:19 +0000 | [diff] [blame] | 408 | return 's'; |
| 409 | } |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 410 | } |
| 411 | |
Rafael Espindola | 22fe9c1 | 2014-02-05 05:19:19 +0000 | [diff] [blame] | 412 | return '?'; |
| 413 | } |
| 414 | |
Rafael Espindola | f12b828 | 2014-02-21 20:10:59 +0000 | [diff] [blame] | 415 | static char getSymbolNMTypeChar(const GlobalValue &GV) { |
| 416 | if (isa<Function>(GV)) |
| 417 | return 't'; |
| 418 | // FIXME: should we print 'b'? At the IR level we cannot be sure if this |
| 419 | // will be in bss or not, but we could approximate. |
| 420 | if (isa<GlobalVariable>(GV)) |
| 421 | return 'd'; |
| 422 | const GlobalAlias *GA = cast<GlobalAlias>(&GV); |
| 423 | const GlobalValue *AliasedGV = GA->getAliasedGlobal(); |
| 424 | return getSymbolNMTypeChar(*AliasedGV); |
| 425 | } |
| 426 | |
| 427 | static char getSymbolNMTypeChar(IRObjectFile &Obj, basic_symbol_iterator I) { |
| 428 | const GlobalValue &GV = Obj.getSymbolGV(I->getRawDataRefImpl()); |
| 429 | return getSymbolNMTypeChar(GV); |
| 430 | } |
| 431 | |
Rafael Espindola | 22fe9c1 | 2014-02-05 05:19:19 +0000 | [diff] [blame] | 432 | template <class ELFT> |
| 433 | static bool isObject(ELFObjectFile<ELFT> &Obj, symbol_iterator I) { |
| 434 | typedef typename ELFObjectFile<ELFT>::Elf_Sym Elf_Sym; |
| 435 | |
| 436 | DataRefImpl Symb = I->getRawDataRefImpl(); |
| 437 | const Elf_Sym *ESym = Obj.getSymbol(Symb); |
| 438 | |
| 439 | return ESym->getType() == ELF::STT_OBJECT; |
| 440 | } |
| 441 | |
Rafael Espindola | f12b828 | 2014-02-21 20:10:59 +0000 | [diff] [blame] | 442 | static bool isObject(SymbolicFile *Obj, basic_symbol_iterator I) { |
Rafael Espindola | 22fe9c1 | 2014-02-05 05:19:19 +0000 | [diff] [blame] | 443 | if (ELF32LEObjectFile *ELF = dyn_cast<ELF32LEObjectFile>(Obj)) |
| 444 | return isObject(*ELF, I); |
| 445 | if (ELF64LEObjectFile *ELF = dyn_cast<ELF64LEObjectFile>(Obj)) |
| 446 | return isObject(*ELF, I); |
| 447 | if (ELF32BEObjectFile *ELF = dyn_cast<ELF32BEObjectFile>(Obj)) |
| 448 | return isObject(*ELF, I); |
| 449 | if (ELF64BEObjectFile *ELF = dyn_cast<ELF64BEObjectFile>(Obj)) |
| 450 | return isObject(*ELF, I); |
| 451 | return false; |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 452 | } |
| 453 | |
Rafael Espindola | f12b828 | 2014-02-21 20:10:59 +0000 | [diff] [blame] | 454 | static char getNMTypeChar(SymbolicFile *Obj, basic_symbol_iterator I) { |
Rafael Espindola | 22fe9c1 | 2014-02-05 05:19:19 +0000 | [diff] [blame] | 455 | uint32_t Symflags = I->getFlags(); |
| 456 | if ((Symflags & object::SymbolRef::SF_Weak) && !isa<MachOObjectFile>(Obj)) { |
| 457 | char Ret = isObject(Obj, I) ? 'v' : 'w'; |
| 458 | if (!(Symflags & object::SymbolRef::SF_Undefined)) |
| 459 | Ret = toupper(Ret); |
| 460 | return Ret; |
| 461 | } |
| 462 | |
| 463 | if (Symflags & object::SymbolRef::SF_Undefined) |
| 464 | return 'U'; |
| 465 | |
| 466 | if (Symflags & object::SymbolRef::SF_Common) |
| 467 | return 'C'; |
| 468 | |
| 469 | char Ret = '?'; |
| 470 | if (Symflags & object::SymbolRef::SF_Absolute) |
| 471 | Ret = 'a'; |
Rafael Espindola | f12b828 | 2014-02-21 20:10:59 +0000 | [diff] [blame] | 472 | else if (IRObjectFile *IR = dyn_cast<IRObjectFile>(Obj)) |
| 473 | Ret = getSymbolNMTypeChar(*IR, I); |
Rafael Espindola | 22fe9c1 | 2014-02-05 05:19:19 +0000 | [diff] [blame] | 474 | else if (COFFObjectFile *COFF = dyn_cast<COFFObjectFile>(Obj)) |
| 475 | Ret = getSymbolNMTypeChar(*COFF, I); |
| 476 | else if (MachOObjectFile *MachO = dyn_cast<MachOObjectFile>(Obj)) |
| 477 | Ret = getSymbolNMTypeChar(*MachO, I); |
| 478 | else if (ELF32LEObjectFile *ELF = dyn_cast<ELF32LEObjectFile>(Obj)) |
| 479 | Ret = getSymbolNMTypeChar(*ELF, I); |
| 480 | else if (ELF64LEObjectFile *ELF = dyn_cast<ELF64LEObjectFile>(Obj)) |
| 481 | Ret = getSymbolNMTypeChar(*ELF, I); |
| 482 | else if (ELF32BEObjectFile *ELF = dyn_cast<ELF32BEObjectFile>(Obj)) |
| 483 | Ret = getSymbolNMTypeChar(*ELF, I); |
| 484 | else |
| 485 | Ret = getSymbolNMTypeChar(*cast<ELF64BEObjectFile>(Obj), I); |
| 486 | |
| 487 | if (Symflags & object::SymbolRef::SF_Global) |
| 488 | Ret = toupper(Ret); |
| 489 | |
| 490 | return Ret; |
Rafael Espindola | 586af97a | 2013-11-02 21:16:09 +0000 | [diff] [blame] | 491 | } |
| 492 | |
Rafael Espindola | f12b828 | 2014-02-21 20:10:59 +0000 | [diff] [blame] | 493 | static void dumpSymbolNamesFromObject(SymbolicFile *Obj) { |
| 494 | basic_symbol_iterator IBegin = Obj->symbol_begin(); |
| 495 | basic_symbol_iterator IEnd = Obj->symbol_end(); |
Michael J. Spencer | 8c4729f | 2012-02-28 00:40:37 +0000 | [diff] [blame] | 496 | if (DynamicSyms) { |
Rafael Espindola | 196666c | 2014-01-30 20:45:33 +0000 | [diff] [blame] | 497 | if (!Obj->isELF()) { |
| 498 | error("File format has no dynamic symbol table", Obj->getFileName()); |
| 499 | return; |
| 500 | } |
Alexey Samsonov | 65056a3 | 2014-02-26 12:51:19 +0000 | [diff] [blame] | 501 | std::pair<symbol_iterator, symbol_iterator> IDyn = |
| 502 | getELFDynamicSymbolIterators(Obj); |
| 503 | IBegin = IDyn.first; |
| 504 | IEnd = IDyn.second; |
Michael J. Spencer | 8c4729f | 2012-02-28 00:40:37 +0000 | [diff] [blame] | 505 | } |
Rafael Espindola | f12b828 | 2014-02-21 20:10:59 +0000 | [diff] [blame] | 506 | std::string NameBuffer; |
| 507 | raw_string_ostream OS(NameBuffer); |
| 508 | for (basic_symbol_iterator I = IBegin; I != IEnd; ++I) { |
Rafael Espindola | 20122a4 | 2014-01-31 20:57:12 +0000 | [diff] [blame] | 509 | uint32_t SymFlags = I->getFlags(); |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 510 | if (!DebugSyms && (SymFlags & SymbolRef::SF_FormatSpecific)) |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 511 | continue; |
Rafael Espindola | f12b828 | 2014-02-21 20:10:59 +0000 | [diff] [blame] | 512 | if (WithoutAliases) { |
| 513 | if (IRObjectFile *IR = dyn_cast<IRObjectFile>(Obj)) { |
| 514 | const GlobalValue &GV = IR->getSymbolGV(I->getRawDataRefImpl()); |
| 515 | if(isa<GlobalAlias>(GV)) |
| 516 | continue; |
| 517 | } |
| 518 | } |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 519 | NMSymbol S; |
Rafael Espindola | 0115b08 | 2014-01-30 21:51:42 +0000 | [diff] [blame] | 520 | S.Size = UnknownAddressOrSize; |
Rafael Espindola | b4865d6 | 2014-04-03 00:19:35 +0000 | [diff] [blame] | 521 | S.Address = UnknownAddressOrSize; |
Rafael Espindola | f12b828 | 2014-02-21 20:10:59 +0000 | [diff] [blame] | 522 | if ((PrintSize || SizeSort) && isa<ObjectFile>(Obj)) { |
| 523 | symbol_iterator SymI = I; |
| 524 | if (error(SymI->getSize(S.Size))) |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 525 | break; |
Michael J. Spencer | 1d6167f | 2011-06-25 17:55:23 +0000 | [diff] [blame] | 526 | } |
Rafael Espindola | b4865d6 | 2014-04-03 00:19:35 +0000 | [diff] [blame] | 527 | if (PrintAddress && isa<ObjectFile>(Obj)) |
| 528 | if (error(symbol_iterator(I)->getAddress(S.Address))) |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 529 | break; |
| 530 | S.TypeChar = getNMTypeChar(Obj, I); |
Rafael Espindola | f12b828 | 2014-02-21 20:10:59 +0000 | [diff] [blame] | 531 | if (error(I->printName(OS))) |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 532 | break; |
Rafael Espindola | f12b828 | 2014-02-21 20:10:59 +0000 | [diff] [blame] | 533 | OS << '\0'; |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 534 | SymbolList.push_back(S); |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 535 | } |
| 536 | |
Rafael Espindola | f12b828 | 2014-02-21 20:10:59 +0000 | [diff] [blame] | 537 | OS.flush(); |
| 538 | const char *P = NameBuffer.c_str(); |
| 539 | for (unsigned I = 0; I < SymbolList.size(); ++I) { |
| 540 | SymbolList[I].Name = P; |
| 541 | P += strlen(P) + 1; |
| 542 | } |
| 543 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 544 | CurrentFilename = Obj->getFileName(); |
Kevin Enderby | 6abc2e5 | 2014-05-09 23:57:49 +0000 | [diff] [blame^] | 545 | sortAndPrintSymbolList(Obj); |
Brian Gaeke | 0af759d | 2003-10-16 04:43:15 +0000 | [diff] [blame] | 546 | } |
| 547 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 548 | static void dumpSymbolNamesFromFile(std::string &Filename) { |
Ahmed Charles | 56440fd | 2014-03-06 05:51:42 +0000 | [diff] [blame] | 549 | std::unique_ptr<MemoryBuffer> Buffer; |
Michael J. Spencer | bc96f37 | 2011-12-13 23:17:29 +0000 | [diff] [blame] | 550 | if (error(MemoryBuffer::getFileOrSTDIN(Filename, Buffer), Filename)) |
| 551 | return; |
| 552 | |
Owen Anderson | 19251ec | 2009-07-15 22:16:10 +0000 | [diff] [blame] | 553 | LLVMContext &Context = getGlobalContext(); |
Ahmed Charles | 96c9d95 | 2014-03-05 10:19:29 +0000 | [diff] [blame] | 554 | ErrorOr<Binary *> BinaryOrErr = createBinary(Buffer.release(), &Context); |
Rafael Espindola | ce82a07 | 2014-01-30 19:24:00 +0000 | [diff] [blame] | 555 | if (error(BinaryOrErr.getError(), Filename)) |
| 556 | return; |
Ahmed Charles | 56440fd | 2014-03-06 05:51:42 +0000 | [diff] [blame] | 557 | std::unique_ptr<Binary> Bin(BinaryOrErr.get()); |
Shankar Easwaran | 15b28be | 2012-11-13 18:38:42 +0000 | [diff] [blame] | 558 | |
Rafael Espindola | 0115b08 | 2014-01-30 21:51:42 +0000 | [diff] [blame] | 559 | if (Archive *A = dyn_cast<Archive>(Bin.get())) { |
Rafael Espindola | ce82a07 | 2014-01-30 19:24:00 +0000 | [diff] [blame] | 560 | if (ArchiveMap) { |
Rafael Espindola | 0115b08 | 2014-01-30 21:51:42 +0000 | [diff] [blame] | 561 | Archive::symbol_iterator I = A->symbol_begin(); |
| 562 | Archive::symbol_iterator E = A->symbol_end(); |
Rafael Espindola | ce82a07 | 2014-01-30 19:24:00 +0000 | [diff] [blame] | 563 | if (I != E) { |
Rafael Espindola | 0115b08 | 2014-01-30 21:51:42 +0000 | [diff] [blame] | 564 | outs() << "Archive map\n"; |
Rafael Espindola | ce82a07 | 2014-01-30 19:24:00 +0000 | [diff] [blame] | 565 | for (; I != E; ++I) { |
Rafael Espindola | 0115b08 | 2014-01-30 21:51:42 +0000 | [diff] [blame] | 566 | Archive::child_iterator C; |
Rafael Espindola | ce82a07 | 2014-01-30 19:24:00 +0000 | [diff] [blame] | 567 | StringRef SymName; |
| 568 | StringRef FileName; |
| 569 | if (error(I->getMember(C))) |
Michael J. Spencer | 9718f45 | 2013-02-03 10:48:50 +0000 | [diff] [blame] | 570 | return; |
Rafael Espindola | ce82a07 | 2014-01-30 19:24:00 +0000 | [diff] [blame] | 571 | if (error(I->getName(SymName))) |
| 572 | return; |
| 573 | if (error(C->getName(FileName))) |
| 574 | return; |
| 575 | outs() << SymName << " in " << FileName << "\n"; |
Michael J. Spencer | 2bc774a | 2011-09-27 19:37:18 +0000 | [diff] [blame] | 576 | } |
Rafael Espindola | ce82a07 | 2014-01-30 19:24:00 +0000 | [diff] [blame] | 577 | outs() << "\n"; |
Michael J. Spencer | 2bc774a | 2011-09-27 19:37:18 +0000 | [diff] [blame] | 578 | } |
| 579 | } |
Alexey Samsonov | e6388e6 | 2013-06-18 15:03:28 +0000 | [diff] [blame] | 580 | |
Rafael Espindola | 0115b08 | 2014-01-30 21:51:42 +0000 | [diff] [blame] | 581 | for (Archive::child_iterator I = A->child_begin(), E = A->child_end(); |
Rafael Espindola | ce82a07 | 2014-01-30 19:24:00 +0000 | [diff] [blame] | 582 | I != E; ++I) { |
Ahmed Charles | 56440fd | 2014-03-06 05:51:42 +0000 | [diff] [blame] | 583 | std::unique_ptr<Binary> Child; |
Rafael Espindola | f12b828 | 2014-02-21 20:10:59 +0000 | [diff] [blame] | 584 | if (I->getAsBinary(Child, &Context)) |
Rafael Espindola | ce82a07 | 2014-01-30 19:24:00 +0000 | [diff] [blame] | 585 | continue; |
Rafael Espindola | f12b828 | 2014-02-21 20:10:59 +0000 | [diff] [blame] | 586 | if (SymbolicFile *O = dyn_cast<SymbolicFile>(Child.get())) { |
Rafael Espindola | ce82a07 | 2014-01-30 19:24:00 +0000 | [diff] [blame] | 587 | outs() << O->getFileName() << ":\n"; |
| 588 | dumpSymbolNamesFromObject(O); |
| 589 | } |
| 590 | } |
| 591 | return; |
| 592 | } |
Rafael Espindola | f12b828 | 2014-02-21 20:10:59 +0000 | [diff] [blame] | 593 | if (MachOUniversalBinary *UB = dyn_cast<MachOUniversalBinary>(Bin.get())) { |
Rafael Espindola | 0115b08 | 2014-01-30 21:51:42 +0000 | [diff] [blame] | 594 | for (MachOUniversalBinary::object_iterator I = UB->begin_objects(), |
| 595 | E = UB->end_objects(); |
Alexey Samsonov | e6388e6 | 2013-06-18 15:03:28 +0000 | [diff] [blame] | 596 | I != E; ++I) { |
Ahmed Charles | 56440fd | 2014-03-06 05:51:42 +0000 | [diff] [blame] | 597 | std::unique_ptr<ObjectFile> Obj; |
Alexey Samsonov | e6388e6 | 2013-06-18 15:03:28 +0000 | [diff] [blame] | 598 | if (!I->getAsObjectFile(Obj)) { |
| 599 | outs() << Obj->getFileName() << ":\n"; |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 600 | dumpSymbolNamesFromObject(Obj.get()); |
Alexey Samsonov | e6388e6 | 2013-06-18 15:03:28 +0000 | [diff] [blame] | 601 | } |
| 602 | } |
Brian Gaeke | 618026a | 2003-11-19 21:57:30 +0000 | [diff] [blame] | 603 | return; |
Brian Gaeke | 0af759d | 2003-10-16 04:43:15 +0000 | [diff] [blame] | 604 | } |
Rafael Espindola | f12b828 | 2014-02-21 20:10:59 +0000 | [diff] [blame] | 605 | if (SymbolicFile *O = dyn_cast<SymbolicFile>(Bin.get())) { |
Rafael Espindola | ce82a07 | 2014-01-30 19:24:00 +0000 | [diff] [blame] | 606 | dumpSymbolNamesFromObject(O); |
| 607 | return; |
| 608 | } |
| 609 | error("unrecognizable file type", Filename); |
| 610 | return; |
Brian Gaeke | 0af759d | 2003-10-16 04:43:15 +0000 | [diff] [blame] | 611 | } |
| 612 | |
| 613 | int main(int argc, char **argv) { |
Chris Lattner | e3fc2d1 | 2009-03-06 05:34:10 +0000 | [diff] [blame] | 614 | // Print a stack trace if we signal out. |
Chris Lattner | ef8f389 | 2007-05-06 05:36:18 +0000 | [diff] [blame] | 615 | sys::PrintStackTraceOnErrorSignal(); |
Chris Lattner | e3fc2d1 | 2009-03-06 05:34:10 +0000 | [diff] [blame] | 616 | PrettyStackTraceProgram X(argc, argv); |
Michael J. Spencer | 618d219 | 2010-08-31 06:36:46 +0000 | [diff] [blame] | 617 | |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 618 | llvm_shutdown_obj Y; // Call llvm_shutdown() on exit. |
Chris Lattner | e3fc2d1 | 2009-03-06 05:34:10 +0000 | [diff] [blame] | 619 | cl::ParseCommandLineOptions(argc, argv, "llvm symbol table dumper\n"); |
Chris Lattner | 12439ff | 2004-02-19 20:32:12 +0000 | [diff] [blame] | 620 | |
Michael J. Spencer | bc96f37 | 2011-12-13 23:17:29 +0000 | [diff] [blame] | 621 | // llvm-nm only reads binary files. |
Rafael Espindola | cb2eca0 | 2013-06-12 20:58:35 +0000 | [diff] [blame] | 622 | if (error(sys::ChangeStdinToBinary())) |
Michael J. Spencer | bc96f37 | 2011-12-13 23:17:29 +0000 | [diff] [blame] | 623 | return 1; |
| 624 | |
Chris Lattner | ef8f389 | 2007-05-06 05:36:18 +0000 | [diff] [blame] | 625 | ToolName = argv[0]; |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 626 | if (BSDFormat) |
| 627 | OutputFormat = bsd; |
| 628 | if (POSIXFormat) |
| 629 | OutputFormat = posix; |
Chris Lattner | 4aae1f4 | 2003-10-16 18:45:23 +0000 | [diff] [blame] | 630 | |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 631 | // The relative order of these is important. If you pass --size-sort it should |
| 632 | // only print out the size. However, if you pass -S --size-sort, it should |
Rafael Espindola | b4865d6 | 2014-04-03 00:19:35 +0000 | [diff] [blame] | 633 | // print out both the size and address. |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 634 | if (SizeSort && !PrintSize) |
Rafael Espindola | b4865d6 | 2014-04-03 00:19:35 +0000 | [diff] [blame] | 635 | PrintAddress = false; |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 636 | if (OutputFormat == sysv || SizeSort) |
| 637 | PrintSize = true; |
Michael J. Spencer | b8672a5 | 2011-01-20 06:38:57 +0000 | [diff] [blame] | 638 | |
Chris Lattner | ef8f389 | 2007-05-06 05:36:18 +0000 | [diff] [blame] | 639 | switch (InputFilenames.size()) { |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 640 | case 0: |
| 641 | InputFilenames.push_back("-"); |
| 642 | case 1: |
| 643 | break; |
| 644 | default: |
| 645 | MultipleFiles = true; |
Chris Lattner | 4aae1f4 | 2003-10-16 18:45:23 +0000 | [diff] [blame] | 646 | } |
Chris Lattner | ef8f389 | 2007-05-06 05:36:18 +0000 | [diff] [blame] | 647 | |
| 648 | std::for_each(InputFilenames.begin(), InputFilenames.end(), |
Rafael Espindola | 619581c | 2014-01-29 04:56:19 +0000 | [diff] [blame] | 649 | dumpSymbolNamesFromFile); |
Rafael Espindola | 8b82a4d | 2013-07-03 15:46:03 +0000 | [diff] [blame] | 650 | |
| 651 | if (HadError) |
| 652 | return 1; |
| 653 | |
Chris Lattner | ef8f389 | 2007-05-06 05:36:18 +0000 | [diff] [blame] | 654 | return 0; |
Brian Gaeke | 0af759d | 2003-10-16 04:43:15 +0000 | [diff] [blame] | 655 | } |