| 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" | 
| Rafael Espindola | a6e9c3e | 2014-06-12 17:38:55 +0000 | [diff] [blame] | 23 | #include <system_error> | 
| 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; | 
| Rui Ueyama | bc654b1 | 2013-09-27 21:47:05 +0000 | [diff] [blame] | 99 | for (relocation_iterator I = Obj.section_rel_begin(Index), | 
| Rafael Espindola | 5e812af | 2014-01-30 02:49:50 +0000 | [diff] [blame] | 100 | E = Obj.section_rel_end(Index); | 
|  | 101 | I != E; ++I, ++RelNum) { | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 102 | MachO::any_relocation_info RE = Obj.getRelocation(I->getRawDataRefImpl()); | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 103 | outs() << "    # Relocation " << RelNum << "\n"; | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 104 | outs() << "    (('word-0', " << format("0x%x", RE.r_word0) << "),\n"; | 
|  | 105 | outs() << "     ('word-1', " << format("0x%x", RE.r_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) { | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 127 | MachO::segment_command SLC = Obj.getSegmentLoadCommand(LCI); | 
| Daniel Dunbar | a8070e0 | 2010-11-27 08:22:29 +0000 | [diff] [blame] | 128 |  | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 129 | DumpSegmentCommandData(StringRef(SLC.segname, 16), SLC.vmaddr, | 
|  | 130 | SLC.vmsize, SLC.fileoff, SLC.filesize, | 
|  | 131 | SLC.maxprot, SLC.initprot, SLC.nsects, SLC.flags); | 
| Daniel Dunbar | a8070e0 | 2010-11-27 08:22:29 +0000 | [diff] [blame] | 132 |  | 
| Daniel Dunbar | 5867690 | 2010-11-27 13:33:15 +0000 | [diff] [blame] | 133 | // Dump the sections. | 
| Daniel Dunbar | 5867690 | 2010-11-27 13:33:15 +0000 | [diff] [blame] | 134 | outs() << "  ('sections', [\n"; | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 135 | for (unsigned i = 0; i != SLC.nsects; ++i) { | 
|  | 136 | MachO::section Sect = Obj.getSection(LCI, i); | 
|  | 137 | DumpSectionData(Obj, i, StringRef(Sect.sectname, 16), | 
|  | 138 | StringRef(Sect.segname, 16), Sect.addr, | 
|  | 139 | Sect.size, Sect.offset, Sect.align, | 
|  | 140 | Sect.reloff, Sect.nreloc, Sect.flags, | 
|  | 141 | Sect.reserved1, Sect.reserved2); | 
| Daniel Dunbar | 5867690 | 2010-11-27 13:33:15 +0000 | [diff] [blame] | 142 | } | 
|  | 143 | outs() << "  ])\n"; | 
|  | 144 |  | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 145 | return 0; | 
| Daniel Dunbar | a8070e0 | 2010-11-27 08:22:29 +0000 | [diff] [blame] | 146 | } | 
| Daniel Dunbar | 5867690 | 2010-11-27 13:33:15 +0000 | [diff] [blame] | 147 |  | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 148 | static int DumpSegment64Command(const MachOObjectFile &Obj, | 
|  | 149 | const MachOObjectFile::LoadCommandInfo &LCI) { | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 150 | MachO::segment_command_64 SLC = Obj.getSegment64LoadCommand(LCI); | 
|  | 151 | DumpSegmentCommandData(StringRef(SLC.segname, 16), SLC.vmaddr, | 
|  | 152 | SLC.vmsize, SLC.fileoff, SLC.filesize, | 
|  | 153 | SLC.maxprot, SLC.initprot, SLC.nsects, SLC.flags); | 
| Daniel Dunbar | a8070e0 | 2010-11-27 08:22:29 +0000 | [diff] [blame] | 154 |  | 
| Daniel Dunbar | 5867690 | 2010-11-27 13:33:15 +0000 | [diff] [blame] | 155 | // Dump the sections. | 
| Daniel Dunbar | 5867690 | 2010-11-27 13:33:15 +0000 | [diff] [blame] | 156 | outs() << "  ('sections', [\n"; | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 157 | for (unsigned i = 0; i != SLC.nsects; ++i) { | 
|  | 158 | MachO::section_64 Sect = Obj.getSection64(LCI, i); | 
| Daniel Dunbar | 5867690 | 2010-11-27 13:33:15 +0000 | [diff] [blame] | 159 |  | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 160 | DumpSectionData(Obj, i, StringRef(Sect.sectname, 16), | 
|  | 161 | StringRef(Sect.segname, 16), Sect.addr, | 
|  | 162 | Sect.size, Sect.offset, Sect.align, | 
|  | 163 | Sect.reloff, Sect.nreloc, Sect.flags, | 
|  | 164 | Sect.reserved1, Sect.reserved2, | 
|  | 165 | Sect.reserved3); | 
| Daniel Dunbar | 5867690 | 2010-11-27 13:33:15 +0000 | [diff] [blame] | 166 | } | 
|  | 167 | outs() << "  ])\n"; | 
|  | 168 |  | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 169 | return 0; | 
| Daniel Dunbar | a8070e0 | 2010-11-27 08:22:29 +0000 | [diff] [blame] | 170 | } | 
|  | 171 |  | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 172 | static void DumpSymbolTableEntryData(const MachOObjectFile &Obj, | 
| Daniel Dunbar | 83224fc | 2010-11-27 13:52:53 +0000 | [diff] [blame] | 173 | unsigned Index, uint32_t StringIndex, | 
|  | 174 | uint8_t Type, uint8_t SectionIndex, | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 175 | uint16_t Flags, uint64_t Value, | 
|  | 176 | StringRef StringTable) { | 
|  | 177 | const char *Name = &StringTable.data()[StringIndex]; | 
| Daniel Dunbar | 83224fc | 2010-11-27 13:52:53 +0000 | [diff] [blame] | 178 | outs() << "    # Symbol " << Index << "\n"; | 
|  | 179 | outs() << "   (('n_strx', " << StringIndex << ")\n"; | 
| Daniel Dunbar | e308b8e | 2010-11-27 13:58:16 +0000 | [diff] [blame] | 180 | outs() << "    ('n_type', " << format("0x%x", Type) << ")\n"; | 
| Daniel Dunbar | 83224fc | 2010-11-27 13:52:53 +0000 | [diff] [blame] | 181 | outs() << "    ('n_sect', " << uint32_t(SectionIndex) << ")\n"; | 
|  | 182 | outs() << "    ('n_desc', " << Flags << ")\n"; | 
|  | 183 | outs() << "    ('n_value', " << Value << ")\n"; | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 184 | outs() << "    ('_string', '" << Name << "')\n"; | 
| Daniel Dunbar | 83224fc | 2010-11-27 13:52:53 +0000 | [diff] [blame] | 185 | outs() << "   ),\n"; | 
|  | 186 | } | 
|  | 187 |  | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 188 | static int DumpSymtabCommand(const MachOObjectFile &Obj) { | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 189 | MachO::symtab_command SLC = Obj.getSymtabLoadCommand(); | 
| Daniel Dunbar | 33dab2a | 2010-11-27 08:33:44 +0000 | [diff] [blame] | 190 |  | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 191 | outs() << "  ('symoff', " << SLC.symoff << ")\n"; | 
|  | 192 | outs() << "  ('nsyms', " << SLC.nsyms << ")\n"; | 
|  | 193 | outs() << "  ('stroff', " << SLC.stroff << ")\n"; | 
|  | 194 | outs() << "  ('strsize', " << SLC.strsize << ")\n"; | 
| Daniel Dunbar | 8680ce6 | 2010-11-27 13:46:11 +0000 | [diff] [blame] | 195 |  | 
|  | 196 | // Dump the string data. | 
|  | 197 | outs() << "  ('_string_data', '"; | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 198 | StringRef StringTable = Obj.getStringTableData(); | 
|  | 199 | outs().write_escaped(StringTable, | 
| Daniel Dunbar | 8680ce6 | 2010-11-27 13:46:11 +0000 | [diff] [blame] | 200 | /*UseHexEscapes=*/true) << "')\n"; | 
|  | 201 |  | 
| Daniel Dunbar | 83224fc | 2010-11-27 13:52:53 +0000 | [diff] [blame] | 202 | // Dump the symbol table. | 
| Daniel Dunbar | 83224fc | 2010-11-27 13:52:53 +0000 | [diff] [blame] | 203 | outs() << "  ('_symbols', [\n"; | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 204 | unsigned SymNum = 0; | 
| Alexey Samsonov | 464d2e4 | 2014-03-17 07:28:19 +0000 | [diff] [blame] | 205 | for (const SymbolRef &Symbol : Obj.symbols()) { | 
|  | 206 | DataRefImpl DRI = Symbol.getRawDataRefImpl(); | 
| Daniel Dunbar | 83224fc | 2010-11-27 13:52:53 +0000 | [diff] [blame] | 207 | if (Obj.is64Bit()) { | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 208 | MachO::nlist_64 STE = Obj.getSymbol64TableEntry(DRI); | 
|  | 209 | DumpSymbolTableEntryData(Obj, SymNum, STE.n_strx, STE.n_type, | 
|  | 210 | STE.n_sect, STE.n_desc, STE.n_value, | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 211 | StringTable); | 
| Daniel Dunbar | 83224fc | 2010-11-27 13:52:53 +0000 | [diff] [blame] | 212 | } else { | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 213 | MachO::nlist STE = Obj.getSymbolTableEntry(DRI); | 
|  | 214 | DumpSymbolTableEntryData(Obj, SymNum, STE.n_strx, STE.n_type, | 
|  | 215 | STE.n_sect, STE.n_desc, STE.n_value, | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 216 | StringTable); | 
| Daniel Dunbar | 83224fc | 2010-11-27 13:52:53 +0000 | [diff] [blame] | 217 | } | 
| Alexey Samsonov | 464d2e4 | 2014-03-17 07:28:19 +0000 | [diff] [blame] | 218 | SymNum++; | 
| Daniel Dunbar | 83224fc | 2010-11-27 13:52:53 +0000 | [diff] [blame] | 219 | } | 
|  | 220 | outs() << "  ])\n"; | 
|  | 221 |  | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 222 | return 0; | 
| Daniel Dunbar | 33dab2a | 2010-11-27 08:33:44 +0000 | [diff] [blame] | 223 | } | 
|  | 224 |  | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 225 | static int DumpDysymtabCommand(const MachOObjectFile &Obj) { | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 226 | MachO::dysymtab_command DLC = Obj.getDysymtabLoadCommand(); | 
| Daniel Dunbar | 33dab2a | 2010-11-27 08:33:44 +0000 | [diff] [blame] | 227 |  | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 228 | outs() << "  ('ilocalsym', " << DLC.ilocalsym << ")\n"; | 
|  | 229 | outs() << "  ('nlocalsym', " << DLC.nlocalsym << ")\n"; | 
|  | 230 | outs() << "  ('iextdefsym', " << DLC.iextdefsym << ")\n"; | 
|  | 231 | outs() << "  ('nextdefsym', " << DLC.nextdefsym << ")\n"; | 
|  | 232 | outs() << "  ('iundefsym', " << DLC.iundefsym << ")\n"; | 
|  | 233 | outs() << "  ('nundefsym', " << DLC.nundefsym << ")\n"; | 
|  | 234 | outs() << "  ('tocoff', " << DLC.tocoff << ")\n"; | 
|  | 235 | outs() << "  ('ntoc', " << DLC.ntoc << ")\n"; | 
|  | 236 | outs() << "  ('modtaboff', " << DLC.modtaboff << ")\n"; | 
|  | 237 | outs() << "  ('nmodtab', " << DLC.nmodtab << ")\n"; | 
|  | 238 | outs() << "  ('extrefsymoff', " << DLC.extrefsymoff << ")\n"; | 
|  | 239 | outs() << "  ('nextrefsyms', " << DLC.nextrefsyms << ")\n"; | 
|  | 240 | outs() << "  ('indirectsymoff', " << DLC.indirectsymoff << ")\n"; | 
|  | 241 | outs() << "  ('nindirectsyms', " << DLC.nindirectsyms << ")\n"; | 
|  | 242 | outs() << "  ('extreloff', " << DLC.extreloff << ")\n"; | 
|  | 243 | outs() << "  ('nextrel', " << DLC.nextrel << ")\n"; | 
|  | 244 | outs() << "  ('locreloff', " << DLC.locreloff << ")\n"; | 
|  | 245 | outs() << "  ('nlocrel', " << DLC.nlocrel << ")\n"; | 
| Daniel Dunbar | 33dab2a | 2010-11-27 08:33:44 +0000 | [diff] [blame] | 246 |  | 
| Daniel Dunbar | c983afc | 2010-11-27 13:26:12 +0000 | [diff] [blame] | 247 | // Dump the indirect symbol table. | 
| Daniel Dunbar | c983afc | 2010-11-27 13:26:12 +0000 | [diff] [blame] | 248 | outs() << "  ('_indirect_symbols', [\n"; | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 249 | for (unsigned i = 0; i != DLC.nindirectsyms; ++i) { | 
|  | 250 | uint32_t ISTE = Obj.getIndirectSymbolTableEntry(DLC, i); | 
| Daniel Dunbar | c983afc | 2010-11-27 13:26:12 +0000 | [diff] [blame] | 251 | outs() << "    # Indirect Symbol " << i << "\n"; | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 252 | outs() << "    (('symbol_index', " << format("0x%x", ISTE) << "),),\n"; | 
| Daniel Dunbar | c983afc | 2010-11-27 13:26:12 +0000 | [diff] [blame] | 253 | } | 
|  | 254 | outs() << "  ])\n"; | 
|  | 255 |  | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 256 | return 0; | 
| Daniel Dunbar | 33dab2a | 2010-11-27 08:33:44 +0000 | [diff] [blame] | 257 | } | 
|  | 258 |  | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 259 | static int | 
|  | 260 | DumpLinkeditDataCommand(const MachOObjectFile &Obj, | 
|  | 261 | const MachOObjectFile::LoadCommandInfo &LCI) { | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 262 | MachO::linkedit_data_command LLC = Obj.getLinkeditDataLoadCommand(LCI); | 
|  | 263 | outs() << "  ('dataoff', " << LLC.dataoff << ")\n" | 
|  | 264 | << "  ('datasize', " << LLC.datasize << ")\n" | 
| Benjamin Kramer | 58298f0 | 2011-08-30 22:10:58 +0000 | [diff] [blame] | 265 | << "  ('_addresses', [\n"; | 
|  | 266 |  | 
|  | 267 | SmallVector<uint64_t, 8> Addresses; | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 268 | Obj.ReadULEB128s(LLC.dataoff, Addresses); | 
| Benjamin Kramer | 58298f0 | 2011-08-30 22:10:58 +0000 | [diff] [blame] | 269 | for (unsigned i = 0, e = Addresses.size(); i != e; ++i) | 
|  | 270 | outs() << "    # Address " << i << '\n' | 
|  | 271 | << "    ('address', " << format("0x%x", Addresses[i]) << "),\n"; | 
|  | 272 |  | 
|  | 273 | outs() << "  ])\n"; | 
| Benjamin Kramer | 267a6d9 | 2011-08-30 18:33:37 +0000 | [diff] [blame] | 274 |  | 
|  | 275 | return 0; | 
|  | 276 | } | 
|  | 277 |  | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 278 | static int | 
|  | 279 | DumpDataInCodeDataCommand(const MachOObjectFile &Obj, | 
|  | 280 | const MachOObjectFile::LoadCommandInfo &LCI) { | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 281 | MachO::linkedit_data_command LLC = Obj.getLinkeditDataLoadCommand(LCI); | 
|  | 282 | outs() << "  ('dataoff', " << LLC.dataoff << ")\n" | 
|  | 283 | << "  ('datasize', " << LLC.datasize << ")\n" | 
| Jim Grosbach | 4b63d2a | 2012-05-18 19:12:01 +0000 | [diff] [blame] | 284 | << "  ('_data_regions', [\n"; | 
|  | 285 |  | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 286 | unsigned NumRegions = LLC.datasize / sizeof(MachO::data_in_code_entry); | 
| Jim Grosbach | 4b63d2a | 2012-05-18 19:12:01 +0000 | [diff] [blame] | 287 | for (unsigned i = 0; i < NumRegions; ++i) { | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 288 | MachO::data_in_code_entry DICE= Obj.getDataInCodeTableEntry(LLC.dataoff, i); | 
| Jim Grosbach | 4b63d2a | 2012-05-18 19:12:01 +0000 | [diff] [blame] | 289 | outs() << "    # DICE " << i << "\n" | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 290 | << "    ('offset', " << DICE.offset << ")\n" | 
|  | 291 | << "    ('length', " << DICE.length << ")\n" | 
|  | 292 | << "    ('kind', " << DICE.kind << ")\n"; | 
| Jim Grosbach | 4b63d2a | 2012-05-18 19:12:01 +0000 | [diff] [blame] | 293 | } | 
|  | 294 |  | 
|  | 295 | outs() <<"  ])\n"; | 
|  | 296 |  | 
|  | 297 | return 0; | 
|  | 298 | } | 
|  | 299 |  | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 300 | static int | 
|  | 301 | DumpLinkerOptionsCommand(const MachOObjectFile &Obj, | 
|  | 302 | const MachOObjectFile::LoadCommandInfo &LCI) { | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 303 | MachO::linker_options_command LOLC = Obj.getLinkerOptionsLoadCommand(LCI); | 
|  | 304 | outs() << "  ('count', " << LOLC.count << ")\n" | 
|  | 305 | << "  ('_strings', [\n"; | 
| Daniel Dunbar | eec0f32 | 2013-01-18 01:26:07 +0000 | [diff] [blame] | 306 |  | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 307 | uint64_t DataSize = LOLC.cmdsize - sizeof(MachO::linker_options_command); | 
|  | 308 | const char *P = LCI.Ptr + sizeof(MachO::linker_options_command); | 
|  | 309 | StringRef Data(P, DataSize); | 
|  | 310 | for (unsigned i = 0; i != LOLC.count; ++i) { | 
|  | 311 | std::pair<StringRef,StringRef> Split = Data.split('\0'); | 
|  | 312 | outs() << "\t\""; | 
|  | 313 | outs().write_escaped(Split.first); | 
|  | 314 | outs() << "\",\n"; | 
|  | 315 | Data = Split.second; | 
|  | 316 | } | 
|  | 317 | outs() <<"  ])\n"; | 
| Daniel Dunbar | eec0f32 | 2013-01-18 01:26:07 +0000 | [diff] [blame] | 318 |  | 
|  | 319 | return 0; | 
|  | 320 | } | 
|  | 321 |  | 
| Jim Grosbach | 448334a | 2014-03-18 22:09:05 +0000 | [diff] [blame] | 322 | static int | 
|  | 323 | DumpVersionMin(const MachOObjectFile &Obj, | 
|  | 324 | const MachOObjectFile::LoadCommandInfo &LCI) { | 
|  | 325 | MachO::version_min_command VMLC = Obj.getVersionMinLoadCommand(LCI); | 
|  | 326 | outs() << "  ('version, " << VMLC.version << ")\n" | 
|  | 327 | << "  ('reserved, " << VMLC.reserved << ")\n"; | 
|  | 328 | return 0; | 
|  | 329 | } | 
|  | 330 |  | 
| Tim Northover | 8f9590b | 2014-06-30 14:40:57 +0000 | [diff] [blame] | 331 | static int | 
|  | 332 | DumpDylibID(const MachOObjectFile &Obj, | 
|  | 333 | const MachOObjectFile::LoadCommandInfo &LCI) { | 
|  | 334 | MachO::dylib_command DLLC = Obj.getDylibIDLoadCommand(LCI); | 
|  | 335 | outs() << "  ('install_name', '" << LCI.Ptr + DLLC.dylib.name << "')\n" | 
|  | 336 | << "  ('timestamp, " << DLLC.dylib.timestamp << ")\n" | 
|  | 337 | << "  ('cur_version, " << DLLC.dylib.current_version << ")\n" | 
|  | 338 | << "  ('compat_version, " << DLLC.dylib.compatibility_version << ")\n"; | 
|  | 339 | return 0; | 
|  | 340 | } | 
|  | 341 |  | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 342 | static int DumpLoadCommand(const MachOObjectFile &Obj, | 
|  | 343 | MachOObjectFile::LoadCommandInfo &LCI) { | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 344 | switch (LCI.C.cmd) { | 
|  | 345 | case MachO::LC_SEGMENT: | 
| David Blaikie | ed80aa5 | 2013-08-27 05:16:07 +0000 | [diff] [blame] | 346 | return DumpSegmentCommand(Obj, LCI); | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 347 | case MachO::LC_SEGMENT_64: | 
| David Blaikie | ed80aa5 | 2013-08-27 05:16:07 +0000 | [diff] [blame] | 348 | return DumpSegment64Command(Obj, LCI); | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 349 | case MachO::LC_SYMTAB: | 
| David Blaikie | ed80aa5 | 2013-08-27 05:16:07 +0000 | [diff] [blame] | 350 | return DumpSymtabCommand(Obj); | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 351 | case MachO::LC_DYSYMTAB: | 
| David Blaikie | ed80aa5 | 2013-08-27 05:16:07 +0000 | [diff] [blame] | 352 | return DumpDysymtabCommand(Obj); | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 353 | case MachO::LC_CODE_SIGNATURE: | 
|  | 354 | case MachO::LC_SEGMENT_SPLIT_INFO: | 
|  | 355 | case MachO::LC_FUNCTION_STARTS: | 
| David Blaikie | ed80aa5 | 2013-08-27 05:16:07 +0000 | [diff] [blame] | 356 | return DumpLinkeditDataCommand(Obj, LCI); | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 357 | case MachO::LC_DATA_IN_CODE: | 
| David Blaikie | ed80aa5 | 2013-08-27 05:16:07 +0000 | [diff] [blame] | 358 | return DumpDataInCodeDataCommand(Obj, LCI); | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 359 | case MachO::LC_LINKER_OPTIONS: | 
| David Blaikie | ed80aa5 | 2013-08-27 05:16:07 +0000 | [diff] [blame] | 360 | return DumpLinkerOptionsCommand(Obj, LCI); | 
| Jim Grosbach | 448334a | 2014-03-18 22:09:05 +0000 | [diff] [blame] | 361 | case MachO::LC_VERSION_MIN_IPHONEOS: | 
|  | 362 | case MachO::LC_VERSION_MIN_MACOSX: | 
|  | 363 | return DumpVersionMin(Obj, LCI); | 
| Tim Northover | 8f9590b | 2014-06-30 14:40:57 +0000 | [diff] [blame] | 364 | case MachO::LC_ID_DYLIB: | 
|  | 365 | return DumpDylibID(Obj, LCI); | 
| Daniel Dunbar | 3977e7d | 2010-11-27 07:19:41 +0000 | [diff] [blame] | 366 | default: | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 367 | Warning("unknown load command: " + Twine(LCI.C.cmd)); | 
| David Blaikie | ed80aa5 | 2013-08-27 05:16:07 +0000 | [diff] [blame] | 368 | return 0; | 
| Daniel Dunbar | 3977e7d | 2010-11-27 07:19:41 +0000 | [diff] [blame] | 369 | } | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 370 | } | 
| Daniel Dunbar | 3977e7d | 2010-11-27 07:19:41 +0000 | [diff] [blame] | 371 |  | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 372 |  | 
|  | 373 | static int DumpLoadCommand(const MachOObjectFile &Obj, unsigned Index, | 
|  | 374 | MachOObjectFile::LoadCommandInfo &LCI) { | 
|  | 375 | outs() << "  # Load Command " << Index << "\n" | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 376 | << " (('command', " << LCI.C.cmd << ")\n" | 
|  | 377 | << "  ('size', " << LCI.C.cmdsize << ")\n"; | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 378 | int Res = DumpLoadCommand(Obj, LCI); | 
|  | 379 | outs() << " ),\n"; | 
| Daniel Dunbar | a8070e0 | 2010-11-27 08:22:29 +0000 | [diff] [blame] | 380 | return Res; | 
| Daniel Dunbar | 3977e7d | 2010-11-27 07:19:41 +0000 | [diff] [blame] | 381 | } | 
|  | 382 |  | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 383 | static void printHeader(const MachOObjectFile *Obj, | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 384 | const MachO::mach_header &Header) { | 
|  | 385 | outs() << "('cputype', " << Header.cputype << ")\n"; | 
|  | 386 | outs() << "('cpusubtype', " << Header.cpusubtype << ")\n"; | 
|  | 387 | outs() << "('filetype', " << Header.filetype << ")\n"; | 
|  | 388 | outs() << "('num_load_commands', " << Header.ncmds << ")\n"; | 
|  | 389 | outs() << "('load_commands_size', " << Header.sizeofcmds << ")\n"; | 
|  | 390 | outs() << "('flag', " << Header.flags << ")\n"; | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 391 |  | 
|  | 392 | // Print extended header if 64-bit. | 
|  | 393 | if (Obj->is64Bit()) { | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 394 | const MachO::mach_header_64 *Header64 = | 
|  | 395 | reinterpret_cast<const MachO::mach_header_64 *>(&Header); | 
|  | 396 | outs() << "('reserved', " << Header64->reserved << ")\n"; | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 397 | } | 
|  | 398 | } | 
|  | 399 |  | 
| Daniel Dunbar | fdfb635 | 2010-11-27 05:58:44 +0000 | [diff] [blame] | 400 | int main(int argc, char **argv) { | 
| Daniel Dunbar | 3977e7d | 2010-11-27 07:19:41 +0000 | [diff] [blame] | 401 | ProgramName = argv[0]; | 
| Daniel Dunbar | fdfb635 | 2010-11-27 05:58:44 +0000 | [diff] [blame] | 402 | llvm_shutdown_obj Y;  // Call llvm_shutdown() on exit. | 
|  | 403 |  | 
|  | 404 | cl::ParseCommandLineOptions(argc, argv, "llvm Mach-O dumping tool\n"); | 
|  | 405 |  | 
| Rafael Espindola | 63da295 | 2014-01-15 19:37:43 +0000 | [diff] [blame] | 406 | ErrorOr<Binary *> BinaryOrErr = createBinary(InputFile); | 
| Rafael Espindola | 4453e4294 | 2014-06-13 03:07:50 +0000 | [diff] [blame] | 407 | if (std::error_code EC = BinaryOrErr.getError()) | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 408 | return Error("unable to read input: '" + EC.message() + "'"); | 
| Ahmed Charles | 56440fd | 2014-03-06 05:51:42 +0000 | [diff] [blame] | 409 | std::unique_ptr<Binary> Binary(BinaryOrErr.get()); | 
| Daniel Dunbar | 9630fb3 | 2010-11-27 06:19:17 +0000 | [diff] [blame] | 410 |  | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 411 | const MachOObjectFile *InputObject = dyn_cast<MachOObjectFile>(Binary.get()); | 
| Daniel Dunbar | 3977e7d | 2010-11-27 07:19:41 +0000 | [diff] [blame] | 412 | if (!InputObject) | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 413 | return Error("Not a MachO object"); | 
| Daniel Dunbar | 9630fb3 | 2010-11-27 06:19:17 +0000 | [diff] [blame] | 414 |  | 
| Eric Christopher | 19ea5ae | 2011-04-03 23:51:47 +0000 | [diff] [blame] | 415 | // Print the header | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 416 | MachO::mach_header_64 Header64; | 
|  | 417 | MachO::mach_header *Header = reinterpret_cast<MachO::mach_header*>(&Header64); | 
|  | 418 | if (InputObject->is64Bit()) | 
|  | 419 | Header64 = InputObject->getHeader64(); | 
|  | 420 | else | 
|  | 421 | *Header = InputObject->getHeader(); | 
|  | 422 | printHeader(InputObject, *Header); | 
| Daniel Dunbar | 3977e7d | 2010-11-27 07:19:41 +0000 | [diff] [blame] | 423 |  | 
|  | 424 | // Print the load commands. | 
| Daniel Dunbar | a8070e0 | 2010-11-27 08:22:29 +0000 | [diff] [blame] | 425 | int Res = 0; | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 426 | MachOObjectFile::LoadCommandInfo Command = | 
|  | 427 | InputObject->getFirstLoadCommandInfo(); | 
| Daniel Dunbar | 3977e7d | 2010-11-27 07:19:41 +0000 | [diff] [blame] | 428 | outs() << "('load_commands', [\n"; | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 429 | for (unsigned i = 0; ; ++i) { | 
|  | 430 | if (DumpLoadCommand(*InputObject, i, Command)) | 
| Daniel Dunbar | a8070e0 | 2010-11-27 08:22:29 +0000 | [diff] [blame] | 431 | break; | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 432 |  | 
| Charles Davis | 8bdfafd | 2013-09-01 04:28:48 +0000 | [diff] [blame] | 433 | if (i == Header->ncmds - 1) | 
| Rafael Espindola | 6e040c0 | 2013-04-26 20:07:33 +0000 | [diff] [blame] | 434 | break; | 
|  | 435 | Command = InputObject->getNextLoadCommandInfo(Command); | 
|  | 436 | } | 
| Daniel Dunbar | 3977e7d | 2010-11-27 07:19:41 +0000 | [diff] [blame] | 437 | outs() << "])\n"; | 
|  | 438 |  | 
| Daniel Dunbar | a8070e0 | 2010-11-27 08:22:29 +0000 | [diff] [blame] | 439 | return Res; | 
| Daniel Dunbar | fdfb635 | 2010-11-27 05:58:44 +0000 | [diff] [blame] | 440 | } |