Daniel Dunbar | fdfb635 | 2010-11-27 05:58:44 +0000 | [diff] [blame] | 1 | //===-- macho-dump.cpp - Mach Object Dumping Tool -------------------------===// |
| 2 | // |
| 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 is a testing tool for use with the MC/Mach-O LLVM components. |
| 11 | // |
| 12 | //===----------------------------------------------------------------------===// |
| 13 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 14 | #include "llvm/Object/MachO.h" |
Daniel Dunbar | e308b8e | 2010-11-27 13:58:16 +0000 | [diff] [blame] | 15 | #include "llvm/ADT/StringExtras.h" |
Daniel Dunbar | 3977e7d | 2010-11-27 07:19:41 +0000 | [diff] [blame] | 16 | #include "llvm/ADT/Twine.h" |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 17 | #include "llvm/Support/Casting.h" |
Daniel Dunbar | fdfb635 | 2010-11-27 05:58:44 +0000 | [diff] [blame] | 18 | #include "llvm/Support/CommandLine.h" |
Daniel Dunbar | c983afc | 2010-11-27 13:26:12 +0000 | [diff] [blame] | 19 | #include "llvm/Support/Format.h" |
Daniel Dunbar | fdfb635 | 2010-11-27 05:58:44 +0000 | [diff] [blame] | 20 | #include "llvm/Support/ManagedStatic.h" |
Daniel Dunbar | 9630fb3 | 2010-11-27 06:19:17 +0000 | [diff] [blame] | 21 | #include "llvm/Support/MemoryBuffer.h" |
Daniel Dunbar | fdfb635 | 2010-11-27 05:58:44 +0000 | [diff] [blame] | 22 | #include "llvm/Support/raw_ostream.h" |
Michael J. Spencer | 7b6fef8 | 2010-12-09 17:36:48 +0000 | [diff] [blame] | 23 | #include "llvm/Support/system_error.h" |
Daniel Dunbar | fdfb635 | 2010-11-27 05:58:44 +0000 | [diff] [blame] | 24 | using namespace llvm; |
Daniel Dunbar | 9630fb3 | 2010-11-27 06:19:17 +0000 | [diff] [blame] | 25 | using namespace llvm::object; |
Daniel Dunbar | fdfb635 | 2010-11-27 05:58:44 +0000 | [diff] [blame] | 26 | |
| 27 | static cl::opt<std::string> |
| 28 | InputFile(cl::Positional, cl::desc("<input file>"), cl::init("-")); |
| 29 | |
Daniel Dunbar | 9630fb3 | 2010-11-27 06:19:17 +0000 | [diff] [blame] | 30 | static cl::opt<bool> |
Daniel Dunbar | 5867690 | 2010-11-27 13:33:15 +0000 | [diff] [blame] | 31 | ShowSectionData("dump-section-data", cl::desc("Dump the contents of sections"), |
Daniel Dunbar | 9630fb3 | 2010-11-27 06:19:17 +0000 | [diff] [blame] | 32 | cl::init(false)); |
| 33 | |
Daniel Dunbar | 3977e7d | 2010-11-27 07:19:41 +0000 | [diff] [blame] | 34 | /// |
| 35 | |
| 36 | static const char *ProgramName; |
| 37 | |
| 38 | static void Message(const char *Type, const Twine &Msg) { |
| 39 | errs() << ProgramName << ": " << Type << ": " << Msg << "\n"; |
| 40 | } |
| 41 | |
| 42 | static int Error(const Twine &Msg) { |
| 43 | Message("error", Msg); |
| 44 | return 1; |
| 45 | } |
| 46 | |
| 47 | static void Warning(const Twine &Msg) { |
| 48 | Message("warning", Msg); |
| 49 | } |
| 50 | |
| 51 | /// |
| 52 | |
Daniel Dunbar | a8070e0 | 2010-11-27 08:22:29 +0000 | [diff] [blame] | 53 | static void DumpSegmentCommandData(StringRef Name, |
| 54 | uint64_t VMAddr, uint64_t VMSize, |
| 55 | uint64_t FileOffset, uint64_t FileSize, |
| 56 | uint32_t MaxProt, uint32_t InitProt, |
| 57 | uint32_t NumSections, uint32_t Flags) { |
| 58 | outs() << " ('segment_name', '"; |
| 59 | outs().write_escaped(Name, /*UseHexEscapes=*/true) << "')\n"; |
| 60 | outs() << " ('vm_addr', " << VMAddr << ")\n"; |
| 61 | outs() << " ('vm_size', " << VMSize << ")\n"; |
| 62 | outs() << " ('file_offset', " << FileOffset << ")\n"; |
| 63 | outs() << " ('file_size', " << FileSize << ")\n"; |
| 64 | outs() << " ('maxprot', " << MaxProt << ")\n"; |
| 65 | outs() << " ('initprot', " << InitProt << ")\n"; |
| 66 | outs() << " ('num_sections', " << NumSections << ")\n"; |
| 67 | outs() << " ('flags', " << Flags << ")\n"; |
| 68 | } |
| 69 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 70 | static int DumpSectionData(const MachOObjectFile &Obj, unsigned Index, |
| 71 | StringRef Name, |
Daniel Dunbar | 0ac77d5 | 2010-11-27 13:39:48 +0000 | [diff] [blame] | 72 | StringRef SegmentName, uint64_t Address, |
| 73 | uint64_t Size, uint32_t Offset, |
| 74 | uint32_t Align, uint32_t RelocationTableOffset, |
| 75 | uint32_t NumRelocationTableEntries, |
| 76 | uint32_t Flags, uint32_t Reserved1, |
| 77 | uint32_t Reserved2, uint64_t Reserved3 = ~0ULL) { |
Daniel Dunbar | 5867690 | 2010-11-27 13:33:15 +0000 | [diff] [blame] | 78 | outs() << " # Section " << Index << "\n"; |
| 79 | outs() << " (('section_name', '"; |
| 80 | outs().write_escaped(Name, /*UseHexEscapes=*/true) << "')\n"; |
| 81 | outs() << " ('segment_name', '"; |
| 82 | outs().write_escaped(SegmentName, /*UseHexEscapes=*/true) << "')\n"; |
| 83 | outs() << " ('address', " << Address << ")\n"; |
| 84 | outs() << " ('size', " << Size << ")\n"; |
| 85 | outs() << " ('offset', " << Offset << ")\n"; |
| 86 | outs() << " ('alignment', " << Align << ")\n"; |
| 87 | outs() << " ('reloc_offset', " << RelocationTableOffset << ")\n"; |
| 88 | outs() << " ('num_reloc', " << NumRelocationTableEntries << ")\n"; |
Daniel Dunbar | e308b8e | 2010-11-27 13:58:16 +0000 | [diff] [blame] | 89 | outs() << " ('flags', " << format("0x%x", Flags) << ")\n"; |
Daniel Dunbar | 5867690 | 2010-11-27 13:33:15 +0000 | [diff] [blame] | 90 | outs() << " ('reserved1', " << Reserved1 << ")\n"; |
| 91 | outs() << " ('reserved2', " << Reserved2 << ")\n"; |
| 92 | if (Reserved3 != ~0ULL) |
| 93 | outs() << " ('reserved3', " << Reserved3 << ")\n"; |
Daniel Dunbar | 0ac77d5 | 2010-11-27 13:39:48 +0000 | [diff] [blame] | 94 | outs() << " ),\n"; |
| 95 | |
| 96 | // Dump the relocation entries. |
Daniel Dunbar | 0ac77d5 | 2010-11-27 13:39:48 +0000 | [diff] [blame] | 97 | outs() << " ('_relocations', [\n"; |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 98 | unsigned RelNum = 0; |
| 99 | error_code EC; |
| 100 | for (relocation_iterator I = Obj.getSectionRelBegin(Index), |
| 101 | E = Obj.getSectionRelEnd(Index); I != E; I.increment(EC), ++RelNum) { |
| 102 | macho::RelocationEntry RE = Obj.getRelocation(I->getRawDataRefImpl()); |
| 103 | outs() << " # Relocation " << RelNum << "\n"; |
| 104 | outs() << " (('word-0', " << format("0x%x", RE.Word0) << "),\n"; |
| 105 | outs() << " ('word-1', " << format("0x%x", RE.Word1) << ")),\n"; |
Daniel Dunbar | 0ac77d5 | 2010-11-27 13:39:48 +0000 | [diff] [blame] | 106 | } |
| 107 | outs() << " ])\n"; |
| 108 | |
Daniel Dunbar | e308b8e | 2010-11-27 13:58:16 +0000 | [diff] [blame] | 109 | // Dump the section data, if requested. |
| 110 | if (ShowSectionData) { |
| 111 | outs() << " ('_section_data', '"; |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 112 | StringRef Data = Obj.getData().substr(Offset, Size); |
Daniel Dunbar | e308b8e | 2010-11-27 13:58:16 +0000 | [diff] [blame] | 113 | for (unsigned i = 0; i != Data.size(); ++i) { |
| 114 | if (i && (i % 4) == 0) |
| 115 | outs() << ' '; |
| 116 | outs() << hexdigit((Data[i] >> 4) & 0xF, /*LowerCase=*/true); |
| 117 | outs() << hexdigit((Data[i] >> 0) & 0xF, /*LowerCase=*/true); |
| 118 | } |
| 119 | outs() << "')\n"; |
| 120 | } |
| 121 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 122 | return 0; |
Daniel Dunbar | 5867690 | 2010-11-27 13:33:15 +0000 | [diff] [blame] | 123 | } |
| 124 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 125 | static int DumpSegmentCommand(const MachOObjectFile &Obj, |
| 126 | const MachOObjectFile::LoadCommandInfo &LCI) { |
| 127 | macho::SegmentLoadCommand SLC = Obj.getSegmentLoadCommand(LCI); |
Daniel Dunbar | a8070e0 | 2010-11-27 08:22:29 +0000 | [diff] [blame] | 128 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 129 | DumpSegmentCommandData(StringRef(SLC.Name, 16), SLC.VMAddress, |
| 130 | SLC.VMSize, SLC.FileOffset, SLC.FileSize, |
| 131 | SLC.MaxVMProtection, SLC.InitialVMProtection, |
| 132 | SLC.NumSections, SLC.Flags); |
Daniel Dunbar | a8070e0 | 2010-11-27 08:22:29 +0000 | [diff] [blame] | 133 | |
Daniel Dunbar | 5867690 | 2010-11-27 13:33:15 +0000 | [diff] [blame] | 134 | // Dump the sections. |
Daniel Dunbar | 5867690 | 2010-11-27 13:33:15 +0000 | [diff] [blame] | 135 | outs() << " ('sections', [\n"; |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 136 | for (unsigned i = 0; i != SLC.NumSections; ++i) { |
| 137 | macho::Section Sect = Obj.getSection(LCI, i); |
| 138 | DumpSectionData(Obj, i, StringRef(Sect.Name, 16), |
| 139 | StringRef(Sect.SegmentName, 16), Sect.Address, |
| 140 | Sect.Size, Sect.Offset, Sect.Align, |
| 141 | Sect.RelocationTableOffset, |
| 142 | Sect.NumRelocationTableEntries, Sect.Flags, |
| 143 | Sect.Reserved1, Sect.Reserved2); |
Daniel Dunbar | 5867690 | 2010-11-27 13:33:15 +0000 | [diff] [blame] | 144 | } |
| 145 | outs() << " ])\n"; |
| 146 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 147 | return 0; |
Daniel Dunbar | a8070e0 | 2010-11-27 08:22:29 +0000 | [diff] [blame] | 148 | } |
Daniel Dunbar | 5867690 | 2010-11-27 13:33:15 +0000 | [diff] [blame] | 149 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 150 | static int DumpSegment64Command(const MachOObjectFile &Obj, |
| 151 | const MachOObjectFile::LoadCommandInfo &LCI) { |
| 152 | macho::Segment64LoadCommand SLC = Obj.getSegment64LoadCommand(LCI); |
| 153 | DumpSegmentCommandData(StringRef(SLC.Name, 16), SLC.VMAddress, |
| 154 | SLC.VMSize, SLC.FileOffset, SLC.FileSize, |
| 155 | SLC.MaxVMProtection, SLC.InitialVMProtection, |
| 156 | SLC.NumSections, SLC.Flags); |
Daniel Dunbar | a8070e0 | 2010-11-27 08:22:29 +0000 | [diff] [blame] | 157 | |
Daniel Dunbar | 5867690 | 2010-11-27 13:33:15 +0000 | [diff] [blame] | 158 | // Dump the sections. |
Daniel Dunbar | 5867690 | 2010-11-27 13:33:15 +0000 | [diff] [blame] | 159 | outs() << " ('sections', [\n"; |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 160 | for (unsigned i = 0; i != SLC.NumSections; ++i) { |
| 161 | macho::Section64 Sect = Obj.getSection64(LCI, i); |
Daniel Dunbar | 5867690 | 2010-11-27 13:33:15 +0000 | [diff] [blame] | 162 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 163 | DumpSectionData(Obj, i, StringRef(Sect.Name, 16), |
| 164 | StringRef(Sect.SegmentName, 16), Sect.Address, |
| 165 | Sect.Size, Sect.Offset, Sect.Align, |
| 166 | Sect.RelocationTableOffset, |
| 167 | Sect.NumRelocationTableEntries, Sect.Flags, |
| 168 | Sect.Reserved1, Sect.Reserved2, |
| 169 | Sect.Reserved3); |
Daniel Dunbar | 5867690 | 2010-11-27 13:33:15 +0000 | [diff] [blame] | 170 | } |
| 171 | outs() << " ])\n"; |
| 172 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 173 | return 0; |
Daniel Dunbar | a8070e0 | 2010-11-27 08:22:29 +0000 | [diff] [blame] | 174 | } |
| 175 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 176 | static void DumpSymbolTableEntryData(const MachOObjectFile &Obj, |
Daniel Dunbar | 83224fc | 2010-11-27 13:52:53 +0000 | [diff] [blame] | 177 | unsigned Index, uint32_t StringIndex, |
| 178 | uint8_t Type, uint8_t SectionIndex, |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 179 | uint16_t Flags, uint64_t Value, |
| 180 | StringRef StringTable) { |
| 181 | const char *Name = &StringTable.data()[StringIndex]; |
Daniel Dunbar | 83224fc | 2010-11-27 13:52:53 +0000 | [diff] [blame] | 182 | outs() << " # Symbol " << Index << "\n"; |
| 183 | outs() << " (('n_strx', " << StringIndex << ")\n"; |
Daniel Dunbar | e308b8e | 2010-11-27 13:58:16 +0000 | [diff] [blame] | 184 | outs() << " ('n_type', " << format("0x%x", Type) << ")\n"; |
Daniel Dunbar | 83224fc | 2010-11-27 13:52:53 +0000 | [diff] [blame] | 185 | outs() << " ('n_sect', " << uint32_t(SectionIndex) << ")\n"; |
| 186 | outs() << " ('n_desc', " << Flags << ")\n"; |
| 187 | outs() << " ('n_value', " << Value << ")\n"; |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 188 | outs() << " ('_string', '" << Name << "')\n"; |
Daniel Dunbar | 83224fc | 2010-11-27 13:52:53 +0000 | [diff] [blame] | 189 | outs() << " ),\n"; |
| 190 | } |
| 191 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 192 | static int DumpSymtabCommand(const MachOObjectFile &Obj) { |
| 193 | macho::SymtabLoadCommand SLC = Obj.getSymtabLoadCommand(); |
Daniel Dunbar | 33dab2a | 2010-11-27 08:33:44 +0000 | [diff] [blame] | 194 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 195 | outs() << " ('symoff', " << SLC.SymbolTableOffset << ")\n"; |
| 196 | outs() << " ('nsyms', " << SLC.NumSymbolTableEntries << ")\n"; |
| 197 | outs() << " ('stroff', " << SLC.StringTableOffset << ")\n"; |
| 198 | outs() << " ('strsize', " << SLC.StringTableSize << ")\n"; |
Daniel Dunbar | 8680ce6 | 2010-11-27 13:46:11 +0000 | [diff] [blame] | 199 | |
| 200 | // Dump the string data. |
| 201 | outs() << " ('_string_data', '"; |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 202 | StringRef StringTable = Obj.getStringTableData(); |
| 203 | outs().write_escaped(StringTable, |
Daniel Dunbar | 8680ce6 | 2010-11-27 13:46:11 +0000 | [diff] [blame] | 204 | /*UseHexEscapes=*/true) << "')\n"; |
| 205 | |
Daniel Dunbar | 83224fc | 2010-11-27 13:52:53 +0000 | [diff] [blame] | 206 | // Dump the symbol table. |
Daniel Dunbar | 83224fc | 2010-11-27 13:52:53 +0000 | [diff] [blame] | 207 | outs() << " ('_symbols', [\n"; |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 208 | error_code EC; |
| 209 | unsigned SymNum = 0; |
| 210 | for (symbol_iterator I = Obj.begin_symbols(), E = Obj.end_symbols(); I != E; |
| 211 | I.increment(EC), ++SymNum) { |
| 212 | DataRefImpl DRI = I->getRawDataRefImpl(); |
Daniel Dunbar | 83224fc | 2010-11-27 13:52:53 +0000 | [diff] [blame] | 213 | if (Obj.is64Bit()) { |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 214 | macho::Symbol64TableEntry STE = Obj.getSymbol64TableEntry(DRI); |
| 215 | DumpSymbolTableEntryData(Obj, SymNum, STE.StringIndex, STE.Type, |
| 216 | STE.SectionIndex, STE.Flags, STE.Value, |
| 217 | StringTable); |
Daniel Dunbar | 83224fc | 2010-11-27 13:52:53 +0000 | [diff] [blame] | 218 | } else { |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 219 | macho::SymbolTableEntry STE = Obj.getSymbolTableEntry(DRI); |
| 220 | DumpSymbolTableEntryData(Obj, SymNum, STE.StringIndex, STE.Type, |
| 221 | STE.SectionIndex, STE.Flags, STE.Value, |
| 222 | StringTable); |
Daniel Dunbar | 83224fc | 2010-11-27 13:52:53 +0000 | [diff] [blame] | 223 | } |
| 224 | } |
| 225 | outs() << " ])\n"; |
| 226 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 227 | return 0; |
Daniel Dunbar | 33dab2a | 2010-11-27 08:33:44 +0000 | [diff] [blame] | 228 | } |
| 229 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 230 | static int DumpDysymtabCommand(const MachOObjectFile &Obj) { |
| 231 | macho::DysymtabLoadCommand DLC = Obj.getDysymtabLoadCommand(); |
Daniel Dunbar | 33dab2a | 2010-11-27 08:33:44 +0000 | [diff] [blame] | 232 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 233 | outs() << " ('ilocalsym', " << DLC.LocalSymbolsIndex << ")\n"; |
| 234 | outs() << " ('nlocalsym', " << DLC.NumLocalSymbols << ")\n"; |
| 235 | outs() << " ('iextdefsym', " << DLC.ExternalSymbolsIndex << ")\n"; |
| 236 | outs() << " ('nextdefsym', " << DLC.NumExternalSymbols << ")\n"; |
| 237 | outs() << " ('iundefsym', " << DLC.UndefinedSymbolsIndex << ")\n"; |
| 238 | outs() << " ('nundefsym', " << DLC.NumUndefinedSymbols << ")\n"; |
| 239 | outs() << " ('tocoff', " << DLC.TOCOffset << ")\n"; |
| 240 | outs() << " ('ntoc', " << DLC.NumTOCEntries << ")\n"; |
| 241 | outs() << " ('modtaboff', " << DLC.ModuleTableOffset << ")\n"; |
| 242 | outs() << " ('nmodtab', " << DLC.NumModuleTableEntries << ")\n"; |
| 243 | outs() << " ('extrefsymoff', " << DLC.ReferenceSymbolTableOffset << ")\n"; |
Daniel Dunbar | 33dab2a | 2010-11-27 08:33:44 +0000 | [diff] [blame] | 244 | outs() << " ('nextrefsyms', " |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 245 | << DLC.NumReferencedSymbolTableEntries << ")\n"; |
| 246 | outs() << " ('indirectsymoff', " << DLC.IndirectSymbolTableOffset << ")\n"; |
Daniel Dunbar | 33dab2a | 2010-11-27 08:33:44 +0000 | [diff] [blame] | 247 | outs() << " ('nindirectsyms', " |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 248 | << DLC.NumIndirectSymbolTableEntries << ")\n"; |
| 249 | outs() << " ('extreloff', " << DLC.ExternalRelocationTableOffset << ")\n"; |
| 250 | outs() << " ('nextrel', " << DLC.NumExternalRelocationTableEntries << ")\n"; |
| 251 | outs() << " ('locreloff', " << DLC.LocalRelocationTableOffset << ")\n"; |
| 252 | outs() << " ('nlocrel', " << DLC.NumLocalRelocationTableEntries << ")\n"; |
Daniel Dunbar | 33dab2a | 2010-11-27 08:33:44 +0000 | [diff] [blame] | 253 | |
Daniel Dunbar | c983afc | 2010-11-27 13:26:12 +0000 | [diff] [blame] | 254 | // Dump the indirect symbol table. |
Daniel Dunbar | c983afc | 2010-11-27 13:26:12 +0000 | [diff] [blame] | 255 | outs() << " ('_indirect_symbols', [\n"; |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 256 | for (unsigned i = 0; i != DLC.NumIndirectSymbolTableEntries; ++i) { |
| 257 | macho::IndirectSymbolTableEntry ISTE = |
| 258 | Obj.getIndirectSymbolTableEntry(DLC, i); |
Daniel Dunbar | c983afc | 2010-11-27 13:26:12 +0000 | [diff] [blame] | 259 | outs() << " # Indirect Symbol " << i << "\n"; |
| 260 | outs() << " (('symbol_index', " |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 261 | << format("0x%x", ISTE.Index) << "),),\n"; |
Daniel Dunbar | c983afc | 2010-11-27 13:26:12 +0000 | [diff] [blame] | 262 | } |
| 263 | outs() << " ])\n"; |
| 264 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 265 | return 0; |
Daniel Dunbar | 33dab2a | 2010-11-27 08:33:44 +0000 | [diff] [blame] | 266 | } |
| 267 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 268 | static int |
| 269 | DumpLinkeditDataCommand(const MachOObjectFile &Obj, |
| 270 | const MachOObjectFile::LoadCommandInfo &LCI) { |
| 271 | macho::LinkeditDataLoadCommand LLC = Obj.getLinkeditDataLoadCommand(LCI); |
| 272 | outs() << " ('dataoff', " << LLC.DataOffset << ")\n" |
| 273 | << " ('datasize', " << LLC.DataSize << ")\n" |
Benjamin Kramer | 58298f0 | 2011-08-30 22:10:58 +0000 | [diff] [blame] | 274 | << " ('_addresses', [\n"; |
| 275 | |
| 276 | SmallVector<uint64_t, 8> Addresses; |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 277 | Obj.ReadULEB128s(LLC.DataOffset, Addresses); |
Benjamin Kramer | 58298f0 | 2011-08-30 22:10:58 +0000 | [diff] [blame] | 278 | for (unsigned i = 0, e = Addresses.size(); i != e; ++i) |
| 279 | outs() << " # Address " << i << '\n' |
| 280 | << " ('address', " << format("0x%x", Addresses[i]) << "),\n"; |
| 281 | |
| 282 | outs() << " ])\n"; |
Benjamin Kramer | 267a6d9 | 2011-08-30 18:33:37 +0000 | [diff] [blame] | 283 | |
| 284 | return 0; |
| 285 | } |
| 286 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 287 | static int |
| 288 | DumpDataInCodeDataCommand(const MachOObjectFile &Obj, |
| 289 | const MachOObjectFile::LoadCommandInfo &LCI) { |
| 290 | macho::LinkeditDataLoadCommand LLC = Obj.getLinkeditDataLoadCommand(LCI); |
| 291 | outs() << " ('dataoff', " << LLC.DataOffset << ")\n" |
| 292 | << " ('datasize', " << LLC.DataSize << ")\n" |
Jim Grosbach | 4b63d2a | 2012-05-18 19:12:01 +0000 | [diff] [blame] | 293 | << " ('_data_regions', [\n"; |
| 294 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 295 | unsigned NumRegions = LLC.DataSize / 8; |
Jim Grosbach | 4b63d2a | 2012-05-18 19:12:01 +0000 | [diff] [blame] | 296 | for (unsigned i = 0; i < NumRegions; ++i) { |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 297 | macho::DataInCodeTableEntry DICE = |
| 298 | Obj.getDataInCodeTableEntry(LLC.DataOffset, i); |
Jim Grosbach | 4b63d2a | 2012-05-18 19:12:01 +0000 | [diff] [blame] | 299 | outs() << " # DICE " << i << "\n" |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 300 | << " ('offset', " << DICE.Offset << ")\n" |
| 301 | << " ('length', " << DICE.Length << ")\n" |
| 302 | << " ('kind', " << DICE.Kind << ")\n"; |
Jim Grosbach | 4b63d2a | 2012-05-18 19:12:01 +0000 | [diff] [blame] | 303 | } |
| 304 | |
| 305 | outs() <<" ])\n"; |
| 306 | |
| 307 | return 0; |
| 308 | } |
| 309 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 310 | static int |
| 311 | DumpLinkerOptionsCommand(const MachOObjectFile &Obj, |
| 312 | const MachOObjectFile::LoadCommandInfo &LCI) { |
| 313 | macho::LinkerOptionsLoadCommand LOLC = Obj.getLinkerOptionsLoadCommand(LCI); |
| 314 | outs() << " ('count', " << LOLC.Count << ")\n" |
| 315 | << " ('_strings', [\n"; |
Daniel Dunbar | eec0f32 | 2013-01-18 01:26:07 +0000 | [diff] [blame] | 316 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 317 | uint64_t DataSize = LOLC.Size - sizeof(macho::LinkerOptionsLoadCommand); |
| 318 | const char *P = LCI.Ptr + sizeof(macho::LinkerOptionsLoadCommand); |
| 319 | StringRef Data(P, DataSize); |
| 320 | for (unsigned i = 0; i != LOLC.Count; ++i) { |
| 321 | std::pair<StringRef,StringRef> Split = Data.split('\0'); |
| 322 | outs() << "\t\""; |
| 323 | outs().write_escaped(Split.first); |
| 324 | outs() << "\",\n"; |
| 325 | Data = Split.second; |
| 326 | } |
| 327 | outs() <<" ])\n"; |
Daniel Dunbar | eec0f32 | 2013-01-18 01:26:07 +0000 | [diff] [blame] | 328 | |
| 329 | return 0; |
| 330 | } |
| 331 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 332 | static int DumpLoadCommand(const MachOObjectFile &Obj, |
| 333 | MachOObjectFile::LoadCommandInfo &LCI) { |
| 334 | switch (LCI.C.Type) { |
Daniel Dunbar | a8070e0 | 2010-11-27 08:22:29 +0000 | [diff] [blame] | 335 | case macho::LCT_Segment: |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 336 | return DumpSegmentCommand(Obj, LCI); |
Daniel Dunbar | a8070e0 | 2010-11-27 08:22:29 +0000 | [diff] [blame] | 337 | case macho::LCT_Segment64: |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 338 | return DumpSegment64Command(Obj, LCI); |
Daniel Dunbar | 33dab2a | 2010-11-27 08:33:44 +0000 | [diff] [blame] | 339 | case macho::LCT_Symtab: |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 340 | return DumpSymtabCommand(Obj); |
Daniel Dunbar | 33dab2a | 2010-11-27 08:33:44 +0000 | [diff] [blame] | 341 | case macho::LCT_Dysymtab: |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 342 | return DumpDysymtabCommand(Obj); |
Benjamin Kramer | 267a6d9 | 2011-08-30 18:33:37 +0000 | [diff] [blame] | 343 | case macho::LCT_CodeSignature: |
| 344 | case macho::LCT_SegmentSplitInfo: |
| 345 | case macho::LCT_FunctionStarts: |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 346 | return DumpLinkeditDataCommand(Obj, LCI); |
Jim Grosbach | 4b63d2a | 2012-05-18 19:12:01 +0000 | [diff] [blame] | 347 | case macho::LCT_DataInCode: |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 348 | return DumpDataInCodeDataCommand(Obj, LCI); |
Daniel Dunbar | eec0f32 | 2013-01-18 01:26:07 +0000 | [diff] [blame] | 349 | case macho::LCT_LinkerOptions: |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 350 | return DumpLinkerOptionsCommand(Obj, LCI); |
Daniel Dunbar | 3977e7d | 2010-11-27 07:19:41 +0000 | [diff] [blame] | 351 | default: |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 352 | Warning("unknown load command: " + Twine(LCI.C.Type)); |
| 353 | return 0; |
Daniel Dunbar | 3977e7d | 2010-11-27 07:19:41 +0000 | [diff] [blame] | 354 | } |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 355 | } |
Daniel Dunbar | 3977e7d | 2010-11-27 07:19:41 +0000 | [diff] [blame] | 356 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 357 | |
| 358 | static int DumpLoadCommand(const MachOObjectFile &Obj, unsigned Index, |
| 359 | MachOObjectFile::LoadCommandInfo &LCI) { |
| 360 | outs() << " # Load Command " << Index << "\n" |
| 361 | << " (('command', " << LCI.C.Type << ")\n" |
| 362 | << " ('size', " << LCI.C.Size << ")\n"; |
| 363 | int Res = DumpLoadCommand(Obj, LCI); |
| 364 | outs() << " ),\n"; |
Daniel Dunbar | a8070e0 | 2010-11-27 08:22:29 +0000 | [diff] [blame] | 365 | return Res; |
Daniel Dunbar | 3977e7d | 2010-11-27 07:19:41 +0000 | [diff] [blame] | 366 | } |
| 367 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 368 | static void printHeader(const MachOObjectFile *Obj, |
| 369 | const macho::Header &Header) { |
| 370 | outs() << "('cputype', " << Header.CPUType << ")\n"; |
| 371 | outs() << "('cpusubtype', " << Header.CPUSubtype << ")\n"; |
| 372 | outs() << "('filetype', " << Header.FileType << ")\n"; |
| 373 | outs() << "('num_load_commands', " << Header.NumLoadCommands << ")\n"; |
| 374 | outs() << "('load_commands_size', " << Header.SizeOfLoadCommands << ")\n"; |
| 375 | outs() << "('flag', " << Header.Flags << ")\n"; |
| 376 | |
| 377 | // Print extended header if 64-bit. |
| 378 | if (Obj->is64Bit()) { |
| 379 | macho::Header64Ext Header64Ext = Obj->getHeader64Ext(); |
| 380 | outs() << "('reserved', " << Header64Ext.Reserved << ")\n"; |
| 381 | } |
| 382 | } |
| 383 | |
Daniel Dunbar | fdfb635 | 2010-11-27 05:58:44 +0000 | [diff] [blame] | 384 | int main(int argc, char **argv) { |
Daniel Dunbar | 3977e7d | 2010-11-27 07:19:41 +0000 | [diff] [blame] | 385 | ProgramName = argv[0]; |
Daniel Dunbar | fdfb635 | 2010-11-27 05:58:44 +0000 | [diff] [blame] | 386 | llvm_shutdown_obj Y; // Call llvm_shutdown() on exit. |
| 387 | |
| 388 | cl::ParseCommandLineOptions(argc, argv, "llvm Mach-O dumping tool\n"); |
| 389 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 390 | OwningPtr<Binary> Binary; |
| 391 | if (error_code EC = createBinary(InputFile, Binary)) |
| 392 | return Error("unable to read input: '" + EC.message() + "'"); |
Daniel Dunbar | 9630fb3 | 2010-11-27 06:19:17 +0000 | [diff] [blame] | 393 | |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 394 | const MachOObjectFile *InputObject = dyn_cast<MachOObjectFile>(Binary.get()); |
Daniel Dunbar | 3977e7d | 2010-11-27 07:19:41 +0000 | [diff] [blame] | 395 | if (!InputObject) |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 396 | return Error("Not a MachO object"); |
Daniel Dunbar | 9630fb3 | 2010-11-27 06:19:17 +0000 | [diff] [blame] | 397 | |
Eric Christopher | 19ea5ae | 2011-04-03 23:51:47 +0000 | [diff] [blame] | 398 | // Print the header |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 399 | macho::Header Header = InputObject->getHeader(); |
| 400 | printHeader(InputObject, Header); |
Daniel Dunbar | 3977e7d | 2010-11-27 07:19:41 +0000 | [diff] [blame] | 401 | |
| 402 | // Print the load commands. |
Daniel Dunbar | a8070e0 | 2010-11-27 08:22:29 +0000 | [diff] [blame] | 403 | int Res = 0; |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 404 | MachOObjectFile::LoadCommandInfo Command = |
| 405 | InputObject->getFirstLoadCommandInfo(); |
Daniel Dunbar | 3977e7d | 2010-11-27 07:19:41 +0000 | [diff] [blame] | 406 | outs() << "('load_commands', [\n"; |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 407 | for (unsigned i = 0; ; ++i) { |
| 408 | if (DumpLoadCommand(*InputObject, i, Command)) |
Daniel Dunbar | a8070e0 | 2010-11-27 08:22:29 +0000 | [diff] [blame] | 409 | break; |
Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 410 | |
| 411 | if (i == Header.NumLoadCommands - 1) |
| 412 | break; |
| 413 | Command = InputObject->getNextLoadCommandInfo(Command); |
| 414 | } |
Daniel Dunbar | 3977e7d | 2010-11-27 07:19:41 +0000 | [diff] [blame] | 415 | outs() << "])\n"; |
| 416 | |
Daniel Dunbar | a8070e0 | 2010-11-27 08:22:29 +0000 | [diff] [blame] | 417 | return Res; |
Daniel Dunbar | fdfb635 | 2010-11-27 05:58:44 +0000 | [diff] [blame] | 418 | } |