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