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" |
Kevin Enderby | 8ae63c1 | 2014-09-04 16:54:47 +0000 | [diff] [blame] | 327 | << " ('sdk, " << VMLC.sdk << ")\n"; |
Jim Grosbach | 448334a | 2014-03-18 22:09:05 +0000 | [diff] [blame] | 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 | 48af1c2 | 2014-08-19 18:44:46 +0000 | [diff] [blame] | 406 | ErrorOr<OwningBinary<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() + "'"); |
Rafael Espindola | 48af1c2 | 2014-08-19 18:44:46 +0000 | [diff] [blame] | 409 | Binary &Binary = *BinaryOrErr.get().getBinary(); |
Daniel Dunbar | 9630fb3 | 2010-11-27 06:19:17 +0000 | [diff] [blame] | 410 | |
Rafael Espindola | 3f6481d | 2014-08-01 14:31:55 +0000 | [diff] [blame] | 411 | const MachOObjectFile *InputObject = dyn_cast<MachOObjectFile>(&Binary); |
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 | } |