49#include <system_error>
66 object_error::parse_failed);
73 if (
P < O.getData().begin() ||
P +
sizeof(
T) > O.getData().end())
77 memcpy(&Cmd,
P,
sizeof(
T));
86 if (
P < O.getData().begin() ||
P +
sizeof(
T) > O.getData().end())
90 memcpy(&Cmd,
P,
sizeof(
T));
99 uintptr_t CommandAddr =
reinterpret_cast<uintptr_t
>(L.Ptr);
101 bool Is64 = O.is64Bit();
107 uintptr_t SectionAddr = CommandAddr + SegmentLoadSize + Sec * SectionSize;
108 return reinterpret_cast<const char*
>(SectionAddr);
112 size_t MachOFilesetEntryOffset = 0) {
114 MachOFilesetEntryOffset <= O.getData().size());
115 return O.getData().data() +
Offset + MachOFilesetEntryOffset;
120 const char *
P =
reinterpret_cast<const char *
>(DRI.
p);
133 return O.getHeader().cputype;
137 return O.getHeader().cpusubtype &
~MachO::CPU_SUBTYPE_MASK;
152 if (O.isLittleEndian())
164 if (O.isLittleEndian())
176 if (O.isLittleEndian())
195 assert(Ptr <= Obj.getData().end() &&
"Start must be before end");
196 if (CmdOrErr->cmdsize > (uintptr_t)(Obj.getData().end() - Ptr))
198 " extends past end of file");
199 if (CmdOrErr->cmdsize < 8)
201 " with size less than 8 bytes");
212 return malformedError(
"load command 0 extends past the end all load "
213 "commands in the file");
215 Obj,
getPtr(Obj, HeaderSize, Obj.getMachOFilesetEntryOffset()), 0);
224 Obj.getData().data() + Obj.getMachOFilesetEntryOffset() + HeaderSize +
225 Obj.getHeader().sizeofcmds)
227 " extends past the end all load commands in the file");
234 if (
sizeof(
T) > Obj.getData().size()) {
235 Err =
malformedError(
"the mach header extends past the end of the "
240 Obj,
getPtr(Obj, 0, Obj.getMachOFilesetEntryOffset())))
241 Header = *HeaderOrErr;
243 Err = HeaderOrErr.takeError();
259 for (
auto it = Elements.begin(); it != Elements.end(); ++it) {
265 " with a size of " +
Twine(
Size) +
", overlaps " +
266 E.Name +
" at offset " +
Twine(
E.Offset) +
" with "
267 "a size of " +
Twine(
E.Size));
270 if (nt != Elements.end()) {
285template <
typename Segment,
typename Section>
290 std::list<MachOElement> &Elements) {
291 const unsigned SegmentLoadSize =
sizeof(Segment);
292 if (Load.C.cmdsize < SegmentLoadSize)
294 " " + CmdName +
" cmdsize too small");
296 Segment S = SegOrErr.get();
297 const unsigned SectionSize =
sizeof(Section);
298 uint64_t FileSize = Obj.getData().size();
299 if (S.nsects > std::numeric_limits<uint32_t>::max() / SectionSize ||
300 S.nsects * SectionSize > Load.C.cmdsize - SegmentLoadSize)
302 " inconsistent cmdsize in " + CmdName +
303 " for the number of sections");
304 for (
unsigned J = 0; J < S.nsects; ++J) {
309 return SectionOrErr.takeError();
310 Section s = SectionOrErr.get();
317 CmdName +
" command " +
Twine(LoadCommandIndex) +
318 " extends past the end of the file");
323 s.offset < SizeOfHeaders && s.size != 0)
325 CmdName +
" command " +
Twine(LoadCommandIndex) +
326 " not past the headers of the file");
334 return malformedError(
"offset field plus size field of section " +
335 Twine(J) +
" in " + CmdName +
" command " +
336 Twine(LoadCommandIndex) +
337 " extends past the end of the file");
344 Twine(J) +
" in " + CmdName +
" command " +
345 Twine(LoadCommandIndex) +
346 " greater than the segment");
351 CmdName +
" command " +
Twine(LoadCommandIndex) +
352 " less than the segment's vmaddr");
357 if (S.vmsize != 0 && s.size != 0 && BigSize > BigEnd)
359 " in " + CmdName +
" command " +
360 Twine(LoadCommandIndex) +
361 " greater than than "
362 "the segment's vmaddr plus vmsize");
370 if (s.reloff > FileSize)
372 CmdName +
" command " +
Twine(LoadCommandIndex) +
373 " extends past the end of the file");
377 if (BigSize > FileSize)
378 return malformedError(
"reloff field plus nreloc field times sizeof("
379 "struct relocation_info) of section " +
380 Twine(J) +
" in " + CmdName +
" command " +
381 Twine(LoadCommandIndex) +
382 " extends past the end of the file");
386 "section relocation entries"))
389 if (S.fileoff > FileSize)
391 " fileoff field in " + CmdName +
392 " extends past the end of the file");
394 BigSize += S.filesize;
395 if (BigSize > FileSize)
397 " fileoff field plus filesize field in " +
398 CmdName +
" extends past the end of the file");
399 if (S.vmsize != 0 && S.filesize > S.vmsize)
401 " filesize field in " + CmdName +
402 " greater than vmsize field");
403 IsPageZeroSegment |=
StringRef(
"__PAGEZERO") == S.segname;
405 return SegOrErr.takeError();
413 const char **SymtabLoadCmd,
414 std::list<MachOElement> &Elements) {
417 " LC_SYMTAB cmdsize too small");
418 if (*SymtabLoadCmd !=
nullptr)
422 return SymtabOrErr.takeError();
426 " has incorrect cmdsize");
427 uint64_t FileSize = Obj.getData().size();
428 if (Symtab.
symoff > FileSize)
430 Twine(LoadCommandIndex) +
" extends past the end "
433 const char *struct_nlist_name;
436 struct_nlist_name =
"struct nlist_64";
439 struct_nlist_name =
"struct nlist";
443 if (BigSize > FileSize)
444 return malformedError(
"symoff field plus nsyms field times sizeof(" +
445 Twine(struct_nlist_name) +
") of LC_SYMTAB command " +
446 Twine(LoadCommandIndex) +
" extends past the end "
451 if (Symtab.
stroff > FileSize)
453 Twine(LoadCommandIndex) +
" extends past the end "
457 if (BigSize > FileSize)
458 return malformedError(
"stroff field plus strsize field of LC_SYMTAB "
459 "command " +
Twine(LoadCommandIndex) +
" extends "
460 "past the end of the file");
462 Symtab.
strsize,
"string table"))
464 *SymtabLoadCmd = Load.Ptr;
471 const char **DysymtabLoadCmd,
472 std::list<MachOElement> &Elements) {
475 " LC_DYSYMTAB cmdsize too small");
476 if (*DysymtabLoadCmd !=
nullptr)
481 return DysymtabOrErr.takeError();
485 " has incorrect cmdsize");
486 uint64_t FileSize = Obj.getData().size();
487 if (Dysymtab.
tocoff > FileSize)
489 Twine(LoadCommandIndex) +
" extends past the end of "
493 BigSize += Dysymtab.
tocoff;
494 if (BigSize > FileSize)
495 return malformedError(
"tocoff field plus ntoc field times sizeof(struct "
496 "dylib_table_of_contents) of LC_DYSYMTAB command " +
497 Twine(LoadCommandIndex) +
" extends past the end of "
500 Dysymtab.
ntoc *
sizeof(
struct
502 "table of contents"))
506 Twine(LoadCommandIndex) +
" extends past the end of "
509 const char *struct_dylib_module_name;
513 struct_dylib_module_name =
"struct dylib_module_64";
516 struct_dylib_module_name =
"struct dylib_module";
518 BigSize *= sizeof_modtab;
520 if (BigSize > FileSize)
521 return malformedError(
"modtaboff field plus nmodtab field times sizeof(" +
522 Twine(struct_dylib_module_name) +
") of LC_DYSYMTAB "
523 "command " +
Twine(LoadCommandIndex) +
" extends "
524 "past the end of the file");
526 Dysymtab.
nmodtab * sizeof_modtab,
530 return malformedError(
"extrefsymoff field of LC_DYSYMTAB command " +
531 Twine(LoadCommandIndex) +
" extends past the end of "
536 if (BigSize > FileSize)
537 return malformedError(
"extrefsymoff field plus nextrefsyms field times "
538 "sizeof(struct dylib_reference) of LC_DYSYMTAB "
539 "command " +
Twine(LoadCommandIndex) +
" extends "
540 "past the end of the file");
547 return malformedError(
"indirectsymoff field of LC_DYSYMTAB command " +
548 Twine(LoadCommandIndex) +
" extends past the end of "
553 if (BigSize > FileSize)
554 return malformedError(
"indirectsymoff field plus nindirectsyms field times "
555 "sizeof(uint32_t) of LC_DYSYMTAB command " +
556 Twine(LoadCommandIndex) +
" extends past the end of "
565 Twine(LoadCommandIndex) +
" extends past the end of "
570 if (BigSize > FileSize)
571 return malformedError(
"extreloff field plus nextrel field times sizeof"
572 "(struct relocation_info) of LC_DYSYMTAB command " +
573 Twine(LoadCommandIndex) +
" extends past the end of "
578 "external relocation table"))
582 Twine(LoadCommandIndex) +
" extends past the end of "
587 if (BigSize > FileSize)
588 return malformedError(
"locreloff field plus nlocrel field times sizeof"
589 "(struct relocation_info) of LC_DYSYMTAB command " +
590 Twine(LoadCommandIndex) +
" extends past the end of "
595 "local relocation table"))
597 *DysymtabLoadCmd = Load.Ptr;
604 const char **LoadCmd,
const char *CmdName,
605 std::list<MachOElement> &Elements,
606 const char *ElementName) {
609 CmdName +
" cmdsize too small");
610 if (*LoadCmd !=
nullptr)
612 auto LinkDataOrError =
614 if (!LinkDataOrError)
615 return LinkDataOrError.takeError();
619 Twine(LoadCommandIndex) +
" has incorrect cmdsize");
620 uint64_t FileSize = Obj.getData().size();
621 if (LinkData.
dataoff > FileSize)
623 Twine(LoadCommandIndex) +
" extends past the end of "
627 if (BigSize > FileSize)
629 Twine(CmdName) +
" command " +
630 Twine(LoadCommandIndex) +
" extends past the end of "
642 const char **LoadCmd,
const char *CmdName,
643 std::list<MachOElement> &Elements) {
646 CmdName +
" cmdsize too small");
647 if (*LoadCmd !=
nullptr)
648 return malformedError(
"more than one LC_DYLD_INFO and or LC_DYLD_INFO_ONLY "
653 return DyldInfoOrErr.takeError();
657 Twine(LoadCommandIndex) +
" has incorrect cmdsize");
658 uint64_t FileSize = Obj.getData().size();
661 " command " +
Twine(LoadCommandIndex) +
" extends "
662 "past the end of the file");
665 if (BigSize > FileSize)
666 return malformedError(
"rebase_off field plus rebase_size field of " +
667 Twine(CmdName) +
" command " +
668 Twine(LoadCommandIndex) +
" extends past the end of "
676 " command " +
Twine(LoadCommandIndex) +
" extends "
677 "past the end of the file");
680 if (BigSize > FileSize)
682 Twine(CmdName) +
" command " +
683 Twine(LoadCommandIndex) +
" extends past the end of "
691 " command " +
Twine(LoadCommandIndex) +
" extends "
692 "past the end of the file");
695 if (BigSize > FileSize)
696 return malformedError(
"weak_bind_off field plus weak_bind_size field of " +
697 Twine(CmdName) +
" command " +
698 Twine(LoadCommandIndex) +
" extends past the end of "
702 "dyld weak bind info"))
706 " command " +
Twine(LoadCommandIndex) +
" extends "
707 "past the end of the file");
710 if (BigSize > FileSize)
711 return malformedError(
"lazy_bind_off field plus lazy_bind_size field of " +
712 Twine(CmdName) +
" command " +
713 Twine(LoadCommandIndex) +
" extends past the end of "
717 "dyld lazy bind info"))
721 " command " +
Twine(LoadCommandIndex) +
" extends "
722 "past the end of the file");
725 if (BigSize > FileSize)
726 return malformedError(
"export_off field plus export_size field of " +
727 Twine(CmdName) +
" command " +
728 Twine(LoadCommandIndex) +
" extends past the end of "
740 uint32_t LoadCommandIndex,
const char *CmdName) {
743 CmdName +
" cmdsize too small");
746 return CommandOrErr.takeError();
750 CmdName +
" name.offset field too small, not past "
751 "the end of the dylib_command struct");
752 if (
D.dylib.name >=
D.cmdsize)
754 CmdName +
" name.offset field extends past the end "
755 "of the load command");
759 const char *
P = (
const char *)Load.Ptr;
760 for (i =
D.dylib.name; i <
D.cmdsize; i++)
765 CmdName +
" library name extends past the end of the "
773 const char **LoadCmd) {
777 if (*LoadCmd !=
nullptr)
781 return malformedError(
"LC_ID_DYLIB load command in non-dynamic library "
789 uint32_t LoadCommandIndex,
const char *CmdName) {
792 CmdName +
" cmdsize too small");
795 return CommandOrErr.takeError();
799 CmdName +
" name.offset field too small, not past "
800 "the end of the dylinker_command struct");
801 if (
D.name >=
D.cmdsize)
803 CmdName +
" name.offset field extends past the end "
804 "of the load command");
808 const char *
P = (
const char *)Load.Ptr;
809 for (i =
D.name; i <
D.cmdsize; i++)
814 CmdName +
" dyld name extends past the end of the "
822 const char **LoadCmd,
const char *CmdName) {
825 CmdName +
" has incorrect cmdsize");
826 if (*LoadCmd !=
nullptr)
828 "LC_VERSION_MIN_IPHONEOS, LC_VERSION_MIN_TVOS or "
829 "LC_VERSION_MIN_WATCHOS command");
837 std::list<MachOElement> &Elements) {
840 " LC_NOTE has incorrect cmdsize");
843 return NoteCmdOrErr.takeError();
845 uint64_t FileSize = Obj.getData().size();
848 Twine(LoadCommandIndex) +
" extends "
849 "past the end of the file");
852 if (BigSize > FileSize)
853 return malformedError(
"size field plus offset field of LC_NOTE command " +
854 Twine(LoadCommandIndex) +
" extends past the end of "
870 return BVCOrErr.takeError();
872 if (Load.C.cmdsize !=
876 " LC_BUILD_VERSION_COMMAND has incorrect cmdsize");
880 for (
unsigned i = 0; i < BVC.
ntools; ++i)
891 " LC_RPATH cmdsize too small");
894 return ROrErr.takeError();
898 " LC_RPATH path.offset field too small, not past "
899 "the end of the rpath_command struct");
900 if (R.path >= R.cmdsize)
902 " LC_RPATH path.offset field extends past the end "
903 "of the load command");
907 const char *
P = (
const char *)Load.Ptr;
908 for (i = R.path; i < R.cmdsize; i++)
913 " LC_RPATH library name extends past the end of the "
922 const char **LoadCmd,
const char *CmdName) {
923 if (*LoadCmd !=
nullptr)
925 "LC_ENCRYPTION_INFO_64 command");
926 uint64_t FileSize = Obj.getData().size();
927 if (cryptoff > FileSize)
929 " command " +
Twine(LoadCommandIndex) +
" extends "
930 "past the end of the file");
932 BigSize += cryptsize;
933 if (BigSize > FileSize)
935 Twine(CmdName) +
" command " +
936 Twine(LoadCommandIndex) +
" extends past the end of "
947 " LC_LINKER_OPTION cmdsize too small");
948 auto LinkOptionOrErr =
950 if (!LinkOptionOrErr)
951 return LinkOptionOrErr.takeError();
954 const char *
string = (
const char *)Load.Ptr +
959 while (*
string ==
'\0' && left > 0) {
966 if (0xffffffff == NullPos)
968 " LC_LINKER_OPTION string #" +
Twine(i) +
969 " is not NULL terminated");
970 uint32_t len = std::min(NullPos, left) + 1;
977 " LC_LINKER_OPTION string count " +
Twine(L.count) +
978 " does not match number of strings");
984 uint32_t LoadCommandIndex,
const char *CmdName,
985 size_t SizeOfCmd,
const char *CmdStructName,
986 uint32_t PathOffset,
const char *PathFieldName) {
987 if (PathOffset < SizeOfCmd)
989 CmdName +
" " + PathFieldName +
".offset field too "
990 "small, not past the end of the " + CmdStructName);
991 if (PathOffset >= Load.C.cmdsize)
993 CmdName +
" " + PathFieldName +
".offset field "
994 "extends past the end of the load command");
998 const char *
P = (
const char *)Load.Ptr;
999 for (i = PathOffset; i < Load.C.cmdsize; i++)
1002 if (i >= Load.C.cmdsize)
1004 CmdName +
" " + PathFieldName +
" name extends past "
1005 "the end of the load command");
1012 const char *CmdName) {
1015 CmdName +
" cmdsize too small");
1016 auto ThreadCommandOrErr =
1018 if (!ThreadCommandOrErr)
1019 return ThreadCommandOrErr.takeError();
1022 const char *end = Load.Ptr +
T.cmdsize;
1025 while (state < end) {
1028 "flavor in " + CmdName +
" extends past end of "
1031 memcpy(&flavor, state,
sizeof(
uint32_t));
1038 " count in " + CmdName +
" extends past end of "
1050 " count not x86_THREAD_STATE32_COUNT for "
1051 "flavor number " +
Twine(nflavor) +
" which is "
1052 "a x86_THREAD_STATE32 flavor in " + CmdName +
1056 " x86_THREAD_STATE32 extends past end of "
1057 "command in " + CmdName +
" command");
1061 " unknown flavor (" +
Twine(flavor) +
") for "
1062 "flavor number " +
Twine(nflavor) +
" in " +
1063 CmdName +
" command");
1069 " count not x86_THREAD_STATE_COUNT for "
1070 "flavor number " +
Twine(nflavor) +
" which is "
1071 "a x86_THREAD_STATE flavor in " + CmdName +
1075 " x86_THREAD_STATE extends past end of "
1076 "command in " + CmdName +
" command");
1081 " count not x86_FLOAT_STATE_COUNT for "
1082 "flavor number " +
Twine(nflavor) +
" which is "
1083 "a x86_FLOAT_STATE flavor in " + CmdName +
1087 " x86_FLOAT_STATE extends past end of "
1088 "command in " + CmdName +
" command");
1093 " count not x86_EXCEPTION_STATE_COUNT for "
1094 "flavor number " +
Twine(nflavor) +
" which is "
1095 "a x86_EXCEPTION_STATE flavor in " + CmdName +
1099 " x86_EXCEPTION_STATE extends past end of "
1100 "command in " + CmdName +
" command");
1105 " count not x86_THREAD_STATE64_COUNT for "
1106 "flavor number " +
Twine(nflavor) +
" which is "
1107 "a x86_THREAD_STATE64 flavor in " + CmdName +
1111 " x86_THREAD_STATE64 extends past end of "
1112 "command in " + CmdName +
" command");
1117 " count not x86_EXCEPTION_STATE64_COUNT for "
1118 "flavor number " +
Twine(nflavor) +
" which is "
1119 "a x86_EXCEPTION_STATE64 flavor in " + CmdName +
1123 " x86_EXCEPTION_STATE64 extends past end of "
1124 "command in " + CmdName +
" command");
1128 " unknown flavor (" +
Twine(flavor) +
") for "
1129 "flavor number " +
Twine(nflavor) +
" in " +
1130 CmdName +
" command");
1136 " count not ARM_THREAD_STATE_COUNT for "
1137 "flavor number " +
Twine(nflavor) +
" which is "
1138 "a ARM_THREAD_STATE flavor in " + CmdName +
1142 " ARM_THREAD_STATE extends past end of "
1143 "command in " + CmdName +
" command");
1147 " unknown flavor (" +
Twine(flavor) +
") for "
1148 "flavor number " +
Twine(nflavor) +
" in " +
1149 CmdName +
" command");
1156 " count not ARM_THREAD_STATE64_COUNT for "
1157 "flavor number " +
Twine(nflavor) +
" which is "
1158 "a ARM_THREAD_STATE64 flavor in " + CmdName +
1162 " ARM_THREAD_STATE64 extends past end of "
1163 "command in " + CmdName +
" command");
1167 " unknown flavor (" +
Twine(flavor) +
") for "
1168 "flavor number " +
Twine(nflavor) +
" in " +
1169 CmdName +
" command");
1175 " count not PPC_THREAD_STATE_COUNT for "
1176 "flavor number " +
Twine(nflavor) +
" which is "
1177 "a PPC_THREAD_STATE flavor in " + CmdName +
1181 " PPC_THREAD_STATE extends past end of "
1182 "command in " + CmdName +
" command");
1186 " unknown flavor (" +
Twine(flavor) +
") for "
1187 "flavor number " +
Twine(nflavor) +
" in " +
1188 CmdName +
" command");
1192 "command " +
Twine(LoadCommandIndex) +
" for " +
1193 CmdName +
" command can't be checked");
1204 const char **LoadCmd,
1205 std::list<MachOElement> &Elements) {
1208 " LC_TWOLEVEL_HINTS has incorrect cmdsize");
1209 if (*LoadCmd !=
nullptr)
1210 return malformedError(
"more than one LC_TWOLEVEL_HINTS command");
1213 return HintsOrErr.takeError();
1215 uint64_t FileSize = Obj.getData().size();
1216 if (Hints.
offset > FileSize)
1217 return malformedError(
"offset field of LC_TWOLEVEL_HINTS command " +
1218 Twine(LoadCommandIndex) +
" extends past the end of "
1223 if (BigSize > FileSize)
1224 return malformedError(
"offset field plus nhints times sizeof(struct "
1225 "twolevel_hint) field of LC_TWOLEVEL_HINTS command " +
1226 Twine(LoadCommandIndex) +
" extends past the end of "
1232 *LoadCmd = Load.Ptr;
1240 if (
cmd == MachO::LC_SYMSEG ||
1241 cmd == MachO::LC_LOADFVMLIB ||
1242 cmd == MachO::LC_IDFVMLIB ||
1243 cmd == MachO::LC_IDENT ||
1244 cmd == MachO::LC_FVMFILE ||
1245 cmd == MachO::LC_PREPAGE ||
1246 cmd == MachO::LC_PREBOUND_DYLIB ||
1247 cmd == MachO::LC_TWOLEVEL_HINTS ||
1248 cmd == MachO::LC_PREBIND_CKSUM)
1255 bool Is64Bits,
uint32_t UniversalCputype,
1257 size_t MachOFilesetEntryOffset) {
1259 std::unique_ptr<MachOObjectFile> Obj(
new MachOObjectFile(
1260 std::move(Object), IsLittleEndian, Is64Bits, Err, UniversalCputype,
1261 UniversalIndex, MachOFilesetEntryOffset));
1263 return std::move(Err);
1264 return std::move(Obj);
1267MachOObjectFile::MachOObjectFile(
MemoryBufferRef Object,
bool IsLittleEndian,
1268 bool Is64bits,
Error &Err,
1271 size_t MachOFilesetEntryOffset)
1272 :
ObjectFile(getMachOType(IsLittleEndian, Is64bits), Object),
1273 MachOFilesetEntryOffset(MachOFilesetEntryOffset) {
1284 cputype = Header.cputype;
1290 Err =
malformedError(
"load commands extend past the end of the file");
1293 if (UniversalCputype != 0 && cputype != UniversalCputype) {
1295 Twine(UniversalIndex) +
"'s cputype does not match "
1296 "object file's mach header");
1300 Elements.push_back({0, SizeOfHeaders,
"Mach-O headers"});
1304 if (LoadCommandCount != 0) {
1308 Err = LoadOrErr.takeError();
1313 const char *DyldIdLoadCmd =
nullptr;
1314 const char *SplitInfoLoadCmd =
nullptr;
1315 const char *CodeSignDrsLoadCmd =
nullptr;
1316 const char *CodeSignLoadCmd =
nullptr;
1317 const char *VersLoadCmd =
nullptr;
1318 const char *SourceLoadCmd =
nullptr;
1319 const char *EntryPointLoadCmd =
nullptr;
1320 const char *EncryptLoadCmd =
nullptr;
1321 const char *RoutinesLoadCmd =
nullptr;
1322 const char *UnixThreadLoadCmd =
nullptr;
1323 const char *TwoLevelHintsLoadCmd =
nullptr;
1324 for (
unsigned I = 0;
I < LoadCommandCount; ++
I) {
1326 if (
Load.C.cmdsize % 8 != 0) {
1331 Load.C.cmd != MachO::LC_THREAD ||
Load.C.cmdsize % 4) {
1338 if (
Load.C.cmdsize % 4 != 0) {
1344 LoadCommands.push_back(Load);
1345 if (
Load.C.cmd == MachO::LC_SYMTAB) {
1348 }
else if (
Load.C.cmd == MachO::LC_DYSYMTAB) {
1352 }
else if (
Load.C.cmd == MachO::LC_DATA_IN_CODE) {
1354 "LC_DATA_IN_CODE", Elements,
1355 "data in code info")))
1357 }
else if (
Load.C.cmd == MachO::LC_LINKER_OPTIMIZATION_HINT) {
1359 "LC_LINKER_OPTIMIZATION_HINT",
1360 Elements,
"linker optimization "
1363 }
else if (
Load.C.cmd == MachO::LC_FUNCTION_STARTS) {
1365 "LC_FUNCTION_STARTS", Elements,
1366 "function starts data")))
1368 }
else if (
Load.C.cmd == MachO::LC_SEGMENT_SPLIT_INFO) {
1370 "LC_SEGMENT_SPLIT_INFO", Elements,
1371 "split info data")))
1373 }
else if (
Load.C.cmd == MachO::LC_DYLIB_CODE_SIGN_DRS) {
1375 "LC_DYLIB_CODE_SIGN_DRS", Elements,
1376 "code signing RDs data")))
1378 }
else if (
Load.C.cmd == MachO::LC_CODE_SIGNATURE) {
1380 "LC_CODE_SIGNATURE", Elements,
1381 "code signature data")))
1383 }
else if (
Load.C.cmd == MachO::LC_DYLD_INFO) {
1385 "LC_DYLD_INFO", Elements)))
1387 }
else if (
Load.C.cmd == MachO::LC_DYLD_INFO_ONLY) {
1389 "LC_DYLD_INFO_ONLY", Elements)))
1391 }
else if (
Load.C.cmd == MachO::LC_DYLD_CHAINED_FIXUPS) {
1393 *
this, Load,
I, &DyldChainedFixupsLoadCmd,
1394 "LC_DYLD_CHAINED_FIXUPS", Elements,
"chained fixups")))
1396 }
else if (
Load.C.cmd == MachO::LC_DYLD_EXPORTS_TRIE) {
1398 *
this, Load,
I, &DyldExportsTrieLoadCmd,
"LC_DYLD_EXPORTS_TRIE",
1399 Elements,
"exports trie")))
1401 }
else if (
Load.C.cmd == MachO::LC_UUID) {
1402 if (
Load.C.cmdsize !=
sizeof(MachO::uuid_command)) {
1403 Err =
malformedError(
"LC_UUID command " + Twine(
I) +
" has incorrect "
1411 UuidLoadCmd =
Load.Ptr;
1412 }
else if (
Load.C.cmd == MachO::LC_SEGMENT_64) {
1415 *
this, Load, Sections, HasPageZeroSegment,
I,
1416 "LC_SEGMENT_64", SizeOfHeaders, Elements)))
1418 }
else if (
Load.C.cmd == MachO::LC_SEGMENT) {
1421 *
this, Load, Sections, HasPageZeroSegment,
I,
1422 "LC_SEGMENT", SizeOfHeaders, Elements)))
1424 }
else if (
Load.C.cmd == MachO::LC_ID_DYLIB) {
1427 }
else if (
Load.C.cmd == MachO::LC_LOAD_DYLIB) {
1430 Libraries.push_back(
Load.Ptr);
1431 }
else if (
Load.C.cmd == MachO::LC_LOAD_WEAK_DYLIB) {
1434 Libraries.push_back(
Load.Ptr);
1435 }
else if (
Load.C.cmd == MachO::LC_LAZY_LOAD_DYLIB) {
1438 Libraries.push_back(
Load.Ptr);
1439 }
else if (
Load.C.cmd == MachO::LC_REEXPORT_DYLIB) {
1442 Libraries.push_back(
Load.Ptr);
1443 }
else if (
Load.C.cmd == MachO::LC_LOAD_UPWARD_DYLIB) {
1446 Libraries.push_back(
Load.Ptr);
1447 }
else if (
Load.C.cmd == MachO::LC_ID_DYLINKER) {
1450 }
else if (
Load.C.cmd == MachO::LC_LOAD_DYLINKER) {
1453 }
else if (
Load.C.cmd == MachO::LC_DYLD_ENVIRONMENT) {
1456 }
else if (
Load.C.cmd == MachO::LC_VERSION_MIN_MACOSX) {
1458 "LC_VERSION_MIN_MACOSX")))
1460 }
else if (
Load.C.cmd == MachO::LC_VERSION_MIN_IPHONEOS) {
1462 "LC_VERSION_MIN_IPHONEOS")))
1464 }
else if (
Load.C.cmd == MachO::LC_VERSION_MIN_TVOS) {
1466 "LC_VERSION_MIN_TVOS")))
1468 }
else if (
Load.C.cmd == MachO::LC_VERSION_MIN_WATCHOS) {
1470 "LC_VERSION_MIN_WATCHOS")))
1472 }
else if (
Load.C.cmd == MachO::LC_NOTE) {
1475 }
else if (
Load.C.cmd == MachO::LC_BUILD_VERSION) {
1478 }
else if (
Load.C.cmd == MachO::LC_RPATH) {
1481 }
else if (
Load.C.cmd == MachO::LC_SOURCE_VERSION) {
1482 if (
Load.C.cmdsize !=
sizeof(MachO::source_version_command)) {
1484 " has incorrect cmdsize");
1487 if (SourceLoadCmd) {
1491 SourceLoadCmd =
Load.Ptr;
1492 }
else if (
Load.C.cmd == MachO::LC_MAIN) {
1493 if (
Load.C.cmdsize !=
sizeof(MachO::entry_point_command)) {
1495 " has incorrect cmdsize");
1498 if (EntryPointLoadCmd) {
1502 EntryPointLoadCmd =
Load.Ptr;
1503 }
else if (
Load.C.cmd == MachO::LC_ENCRYPTION_INFO) {
1504 if (
Load.C.cmdsize !=
sizeof(MachO::encryption_info_command)) {
1506 " has incorrect cmdsize");
1509 MachO::encryption_info_command
E =
1512 &EncryptLoadCmd,
"LC_ENCRYPTION_INFO")))
1514 }
else if (
Load.C.cmd == MachO::LC_ENCRYPTION_INFO_64) {
1515 if (
Load.C.cmdsize !=
sizeof(MachO::encryption_info_command_64)) {
1517 " has incorrect cmdsize");
1520 MachO::encryption_info_command_64
E =
1523 &EncryptLoadCmd,
"LC_ENCRYPTION_INFO_64")))
1525 }
else if (
Load.C.cmd == MachO::LC_LINKER_OPTION) {
1528 }
else if (
Load.C.cmd == MachO::LC_SUB_FRAMEWORK) {
1529 if (
Load.C.cmdsize <
sizeof(MachO::sub_framework_command)) {
1531 " LC_SUB_FRAMEWORK cmdsize too small");
1534 MachO::sub_framework_command S =
1537 sizeof(MachO::sub_framework_command),
1538 "sub_framework_command", S.
umbrella,
1541 }
else if (
Load.C.cmd == MachO::LC_SUB_UMBRELLA) {
1542 if (
Load.C.cmdsize <
sizeof(MachO::sub_umbrella_command)) {
1544 " LC_SUB_UMBRELLA cmdsize too small");
1547 MachO::sub_umbrella_command S =
1550 sizeof(MachO::sub_umbrella_command),
1554 }
else if (
Load.C.cmd == MachO::LC_SUB_LIBRARY) {
1555 if (
Load.C.cmdsize <
sizeof(MachO::sub_library_command)) {
1557 " LC_SUB_LIBRARY cmdsize too small");
1560 MachO::sub_library_command S =
1563 sizeof(MachO::sub_library_command),
1567 }
else if (
Load.C.cmd == MachO::LC_SUB_CLIENT) {
1568 if (
Load.C.cmdsize <
sizeof(MachO::sub_client_command)) {
1570 " LC_SUB_CLIENT cmdsize too small");
1573 MachO::sub_client_command S =
1576 sizeof(MachO::sub_client_command),
1577 "sub_client_command", S.
client,
"client")))
1579 }
else if (
Load.C.cmd == MachO::LC_ROUTINES) {
1580 if (
Load.C.cmdsize !=
sizeof(MachO::routines_command)) {
1582 " has incorrect cmdsize");
1585 if (RoutinesLoadCmd) {
1586 Err =
malformedError(
"more than one LC_ROUTINES and or LC_ROUTINES_64 "
1590 RoutinesLoadCmd =
Load.Ptr;
1591 }
else if (
Load.C.cmd == MachO::LC_ROUTINES_64) {
1592 if (
Load.C.cmdsize !=
sizeof(MachO::routines_command_64)) {
1594 " has incorrect cmdsize");
1597 if (RoutinesLoadCmd) {
1598 Err =
malformedError(
"more than one LC_ROUTINES_64 and or LC_ROUTINES "
1602 RoutinesLoadCmd =
Load.Ptr;
1603 }
else if (
Load.C.cmd == MachO::LC_UNIXTHREAD) {
1606 if (UnixThreadLoadCmd) {
1610 UnixThreadLoadCmd =
Load.Ptr;
1611 }
else if (
Load.C.cmd == MachO::LC_THREAD) {
1615 }
else if (
Load.C.cmd == MachO::LC_TWOLEVEL_HINTS) {
1617 &TwoLevelHintsLoadCmd, Elements)))
1619 }
else if (
Load.C.cmd == MachO::LC_IDENT) {
1623 Err =
malformedError(
"load command " + Twine(
I) +
" for cmd value of: " +
1624 Twine(
Load.C.cmd) +
" is obsolete and not "
1631 if (
I < LoadCommandCount - 1) {
1635 Err = LoadOrErr.takeError();
1640 if (!SymtabLoadCmd) {
1641 if (DysymtabLoadCmd) {
1642 Err =
malformedError(
"contains LC_DYSYMTAB load command without a "
1643 "LC_SYMTAB load command");
1646 }
else if (DysymtabLoadCmd) {
1647 MachO::symtab_command Symtab =
1649 MachO::dysymtab_command Dysymtab =
1653 "extends past the end of the symbol table");
1659 Err =
malformedError(
"ilocalsym plus nlocalsym in LC_DYSYMTAB load "
1660 "command extends past the end of the symbol table");
1665 "extends past the end of the symbol table");
1671 Err =
malformedError(
"iextdefsym plus nextdefsym in LC_DYSYMTAB "
1672 "load command extends past the end of the symbol "
1678 "extends past the end of the symbol table");
1684 Err =
malformedError(
"iundefsym plus nundefsym in LC_DYSYMTAB load "
1685 " command extends past the end of the symbol table");
1691 DyldIdLoadCmd ==
nullptr) {
1692 Err =
malformedError(
"no LC_ID_DYLIB load command in dynamic library "
1696 assert(LoadCommands.size() == LoadCommandCount);
1736 if (NSect == 0 || NSect > Sections.size())
1738 " for symbol at index " +
Twine(SymbolIndex));
1743 "the end of string table, for N_INDR symbol at "
1744 "index " +
Twine(SymbolIndex));
1750 if (LibraryOrdinal != 0 &&
1753 LibraryOrdinal - 1 >= Libraries.size() ) {
1755 " for symbol at index " +
Twine(SymbolIndex));
1761 " past the end of string table, for symbol at "
1762 "index " +
Twine(SymbolIndex));
1778 if (Entry.n_strx == 0)
1782 const char *Start = &
StringTable.data()[Entry.n_strx];
1799 return Entry.n_value;
1802 return Entry.n_value;
1818 return std::error_code();
1845 uint8_t n_type = Entry.n_type;
1871 uint8_t MachOType = Entry.n_type;
1872 uint16_t MachOFlags = Entry.n_desc;
1919 DRI.
d.
a = index - 1;
1920 if (DRI.
d.
a >= Sections.size()){
1930 return Entry.n_sect - 1;
1962 SectOffset = Sect.
offset;
1963 SectSize = Sect.
size;
1967 SectOffset = Sect.
offset;
1968 SectSize = Sect.
size;
1974 if (SectOffset > FileSize)
1976 if (FileSize - SectOffset < SectSize)
1977 return FileSize - SectOffset;
2003 for (
uint32_t SectIdx = 0; SectIdx < Sec.
d.
a; ++SectIdx) {
2007 if ((SectOffsetAdjust > 0) && (PrevTrueOffset > CurrTrueOffset))
2008 return malformedError(
"section data exceeds 4GB and section file "
2009 "offsets are not ordered");
2012 if (EndSectFileOffset > UINT32_MAX)
2013 SectOffsetAdjust += EndSectFileOffset & 0xFFFFFFFF00000000ull;
2014 PrevTrueOffset = CurrTrueOffset;
2016 Offset += SectOffsetAdjust;
2040 if (SectionIndex < 1 || SectionIndex > Sections.size())
2044 DRI.
d.
a = SectionIndex - 1;
2050 auto NameOrErr = Section.getName();
2052 return NameOrErr.takeError();
2086 if (!SectionNameOrErr) {
2099template <
typename LoadCommandType>
2104 if (!SegmentOrErr) {
2108 auto &Segment = SegmentOrErr.get();
2111 Segment.fileoff, Segment.fileoff + Segment.filesize));
2115template <
typename LoadCommandType>
2116ArrayRef<uint8_t> getSegmentContents(
const MachOObjectFile &Obj,
2117 MachOObjectFile::LoadCommandInfo LoadCmd) {
2119 if (!SegmentOrErr) {
2123 auto &Segment = SegmentOrErr.get();
2125 Obj.getData().substr(Segment.fileoff, Segment.filesize));
2133 switch (LoadCmd.
C.
cmd) {
2134 case MachO::LC_SEGMENT:
2138 case MachO::LC_SEGMENT_64:
2145 if (!Contents.
empty())
2155 switch (LoadCmd.
C.
cmd) {
2156 case MachO::LC_SEGMENT:
2157 if (Idx == SegmentIndex)
2158 return ::getSegmentContents<MachO::segment_command>(*
this, LoadCmd);
2161 case MachO::LC_SEGMENT_64:
2162 if (Idx == SegmentIndex)
2163 return ::getSegmentContents<MachO::segment_command_64>(*
this, LoadCmd);
2187 return (SegmentName ==
"__LLVM" && *NameOrErr ==
"__bitcode");
2234 Ret.
d.
b = DysymtabLoadCmd.nextrel;
2251 Ret.
d.
b = DysymtabLoadCmd.nlocrel;
2262 "Only implemented for MH_OBJECT && MH_KEXT_BUNDLE");
2303 unsigned Arch = this->
getArch();
2307 static const char *
const Table[] = {
2308 "GENERIC_RELOC_VANILLA",
2309 "GENERIC_RELOC_PAIR",
2310 "GENERIC_RELOC_SECTDIFF",
2311 "GENERIC_RELOC_PB_LA_PTR",
2312 "GENERIC_RELOC_LOCAL_SECTDIFF",
2313 "GENERIC_RELOC_TLV" };
2322 static const char *
const Table[] = {
2323 "X86_64_RELOC_UNSIGNED",
2324 "X86_64_RELOC_SIGNED",
2325 "X86_64_RELOC_BRANCH",
2326 "X86_64_RELOC_GOT_LOAD",
2328 "X86_64_RELOC_SUBTRACTOR",
2329 "X86_64_RELOC_SIGNED_1",
2330 "X86_64_RELOC_SIGNED_2",
2331 "X86_64_RELOC_SIGNED_4",
2332 "X86_64_RELOC_TLV" };
2341 static const char *
const Table[] = {
2342 "ARM_RELOC_VANILLA",
2344 "ARM_RELOC_SECTDIFF",
2345 "ARM_RELOC_LOCAL_SECTDIFF",
2346 "ARM_RELOC_PB_LA_PTR",
2348 "ARM_THUMB_RELOC_BR22",
2349 "ARM_THUMB_32BIT_BRANCH",
2351 "ARM_RELOC_HALF_SECTDIFF" };
2361 static const char *
const Table[] = {
2362 "ARM64_RELOC_UNSIGNED",
"ARM64_RELOC_SUBTRACTOR",
2363 "ARM64_RELOC_BRANCH26",
"ARM64_RELOC_PAGE21",
2364 "ARM64_RELOC_PAGEOFF12",
"ARM64_RELOC_GOT_LOAD_PAGE21",
2365 "ARM64_RELOC_GOT_LOAD_PAGEOFF12",
"ARM64_RELOC_POINTER_TO_GOT",
2366 "ARM64_RELOC_TLVP_LOAD_PAGE21",
"ARM64_RELOC_TLVP_LOAD_PAGEOFF12",
2367 "ARM64_RELOC_ADDEND",
"ARM64_RELOC_AUTHENTICATED_POINTER"
2370 if (RType >= std::size(Table))
2377 static const char *
const Table[] = {
2378 "PPC_RELOC_VANILLA",
2386 "PPC_RELOC_SECTDIFF",
2387 "PPC_RELOC_PB_LA_PTR",
2388 "PPC_RELOC_HI16_SECTDIFF",
2389 "PPC_RELOC_LO16_SECTDIFF",
2390 "PPC_RELOC_HA16_SECTDIFF",
2392 "PPC_RELOC_LO14_SECTDIFF",
2393 "PPC_RELOC_LOCAL_SECTDIFF" };
2402 static const char *
const Table[] = {
2403 "RISCV_RELOC_UNSIGNED",
"RISCV_RELOC_SUBTRACTOR",
2404 "RISCV_RELOC_BRANCH21",
"RISCV_RELOC_HI20",
2405 "RISCV_RELOC_LO12",
"RISCV_RELOC_GOT_HI20",
2406 "RISCV_RELOC_GOT_LO12",
"RISCV_RELOC_POINTER_TO_GOT",
2407 "RISCV_RELOC_ADDEND",
2410 if (RType >= std::size(Table))
2414 Result.append(res.
begin(), res.
end());
2421 Result.append(PCRel.
begin(), PCRel.
end());
2429 Result.append(res.
begin(), res.
end());
2477 size_t a, b, c, d, Idx;
2479 isFramework =
false;
2483 a = Name.rfind(
'/');
2484 if (a == Name.npos || a == 0)
2486 Foo = Name.substr(a + 1);
2489 Idx = Foo.
rfind(
'_');
2490 if (Idx != Foo.
npos && Foo.
size() >= 2) {
2491 Suffix = Foo.
substr(Idx);
2492 if (Suffix !=
"_debug" && Suffix !=
"_profile")
2495 Foo = Foo.
slice(0, Idx);
2499 b = Name.rfind(
'/', a);
2504 F = Name.substr(Idx, Foo.
size());
2505 DotFramework = Name.substr(Idx + Foo.
size(),
sizeof(
".framework/") - 1);
2506 if (
F == Foo && DotFramework ==
".framework/") {
2514 c = Name.rfind(
'/', b);
2515 if (c == Name.npos || c == 0)
2517 V = Name.substr(c + 1);
2518 if (!V.starts_with(
"Versions/"))
2520 d = Name.rfind(
'/', c);
2525 F = Name.substr(Idx, Foo.
size());
2526 DotFramework = Name.substr(Idx + Foo.
size(),
sizeof(
".framework/") - 1);
2527 if (
F == Foo && DotFramework ==
".framework/") {
2534 a = Name.rfind(
'.');
2535 if (a == Name.npos || a == 0)
2537 Dylib = Name.substr(a);
2538 if (Dylib !=
".dylib")
2543 Dot = Name.substr(a - 2, 1);
2548 b = Name.rfind(
'/', a);
2554 Idx = Name.rfind(
'_');
2555 if (Idx != Name.npos && Idx != b) {
2556 Lib = Name.slice(b, Idx);
2557 Suffix = Name.slice(Idx, a);
2558 if (Suffix !=
"_debug" && Suffix !=
"_profile") {
2560 Lib = Name.slice(b, a);
2564 Lib = Name.slice(b, a);
2567 if (
Lib.size() >= 3) {
2568 Dot =
Lib.substr(
Lib.size() - 2, 1);
2575 Qtx = Name.substr(a);
2578 b = Name.rfind(
'/', a);
2580 Lib = Name.slice(0, a);
2582 Lib = Name.slice(b+1, a);
2584 if (
Lib.size() >= 3) {
2585 Dot =
Lib.substr(
Lib.size() - 2, 1);
2599 if (Index >= Libraries.size())
2604 if (LibrariesShortNames.size() == 0) {
2605 for (
unsigned i = 0; i < Libraries.size(); i++) {
2611 if (
D.dylib.name >=
D.cmdsize)
2613 const char *
P = (
const char *)(Libraries[i]) +
D.dylib.name;
2615 if (
D.dylib.name+Name.size() >=
D.cmdsize)
2620 if (shortName.
empty())
2621 LibrariesShortNames.push_back(Name);
2623 LibrariesShortNames.push_back(shortName);
2627 Res = LibrariesShortNames[Index];
2628 return std::error_code();
2632 return Libraries.size();
2645 if (!SymtabLoadCmd || Symtab.
nsyms == 0)
2654 if (!SymtabLoadCmd || Symtab.
nsyms == 0)
2668 if (!SymtabLoadCmd || Index >= Symtab.
nsyms)
2673 DRI.
p =
reinterpret_cast<uintptr_t
>(
getPtr(*
this, Symtab.
symoff));
2685 DRIstart.
p =
reinterpret_cast<uintptr_t
>(
getPtr(*
this, Symtab.
symoff));
2697 DRI.
d.
a = Sections.size();
2710 return "Mach-O 32-bit i386";
2712 return "Mach-O arm";
2714 return "Mach-O arm64 (ILP32)";
2716 return "Mach-O 32-bit ppc";
2718 return "Mach-O 32-bit RISC-V";
2720 return "Mach-O 32-bit unknown";
2726 return "Mach-O 64-bit x86-64";
2728 return "Mach-O arm64";
2730 return "Mach-O 64-bit ppc64";
2732 return "Mach-O 64-bit unknown";
2760 const char **McpuDefault,
2761 const char **ArchFlag) {
2763 *McpuDefault =
nullptr;
2765 *ArchFlag =
nullptr;
2773 return Triple(
"i386-apple-darwin");
2781 *ArchFlag =
"x86_64";
2782 return Triple(
"x86_64-apple-darwin");
2785 *ArchFlag =
"x86_64h";
2786 return Triple(
"x86_64h-apple-darwin");
2794 *ArchFlag =
"armv4t";
2795 return Triple(
"armv4t-apple-darwin");
2798 *ArchFlag =
"armv5e";
2799 return Triple(
"armv5e-apple-darwin");
2802 *ArchFlag =
"xscale";
2803 return Triple(
"xscale-apple-darwin");
2806 *ArchFlag =
"armv6";
2807 return Triple(
"armv6-apple-darwin");
2810 *McpuDefault =
"cortex-m0";
2812 *ArchFlag =
"armv6m";
2813 return Triple(
"armv6m-apple-darwin");
2816 *ArchFlag =
"armv7";
2817 return Triple(
"armv7-apple-darwin");
2820 *McpuDefault =
"cortex-m4";
2822 *ArchFlag =
"armv7em";
2823 return Triple(
"thumbv7em-apple-darwin");
2826 *McpuDefault =
"cortex-a7";
2828 *ArchFlag =
"armv7k";
2829 return Triple(
"armv7k-apple-darwin");
2832 *McpuDefault =
"cortex-m3";
2834 *ArchFlag =
"armv7m";
2835 return Triple(
"thumbv7m-apple-darwin");
2838 *McpuDefault =
"cortex-a7";
2840 *ArchFlag =
"armv7s";
2841 return Triple(
"armv7s-apple-darwin");
2849 *McpuDefault =
"cyclone";
2851 *ArchFlag =
"arm64";
2852 return Triple(
"arm64-apple-darwin");
2855 *McpuDefault =
"apple-a12";
2857 *ArchFlag =
"arm64e";
2858 return Triple(
"arm64e-apple-darwin");
2866 *McpuDefault =
"cyclone";
2868 *ArchFlag =
"arm64_32";
2869 return Triple(
"arm64_32-apple-darwin");
2878 return Triple(
"ppc-apple-darwin");
2886 *ArchFlag =
"ppc64";
2887 return Triple(
"ppc64-apple-darwin");
2895 *ArchFlag =
"riscv32";
2896 return Triple(
"riscv32-apple-macho");
2915 static const std::array<StringRef, 18> ValidArchs = {{
2961 if (!DataInCodeLoadCmd)
2965 DRI.
p =
reinterpret_cast<uintptr_t
>(
getPtr(*
this, DicLC.
dataoff));
2971 if (!DataInCodeLoadCmd)
2983void ExportEntry::moveToFirst() {
2988 pushDownUntilBottom();
2991void ExportEntry::moveToEnd() {
2998 if (Done ||
Other.Done)
2999 return (Done ==
Other.Done);
3001 if (Stack.size() !=
Other.Stack.size())
3004 if (!CumulativeString.equals(
Other.CumulativeString))
3007 for (
unsigned i=0; i < Stack.size(); ++i) {
3008 if (Stack[i].Start !=
Other.Stack[i].Start)
3018 if (Ptr > Trie.
end())
3024 return CumulativeString;
3028 return Stack.back().Flags;
3032 return Stack.back().Address;
3036 return Stack.back().Other;
3040 const char* ImportName = Stack.
back().ImportName;
3047 return Stack.back().Start - Trie.begin();
3050ExportEntry::NodeState::NodeState(
const uint8_t *Ptr)
3051 : Start(Ptr), Current(Ptr) {}
3053void ExportEntry::pushNode(
uint64_t offset) {
3056 NodeState State(Ptr);
3057 const char *
error =
nullptr;
3061 " in export trie data at node: 0x" +
3066 State.IsExportNode = (ExportInfoSize != 0);
3067 const uint8_t* Children = State.Current + ExportInfoSize;
3068 if (Children > Trie.
end()) {
3072 " too big and extends past end of trie data");
3076 if (State.IsExportNode) {
3077 const uint8_t *ExportStart = State.Current;
3078 State.Flags = readULEB128(State.Current, &
error);
3081 " in export trie data at node: 0x" +
3087 if (State.Flags != 0 &&
3092 "unsupported exported symbol kind: " + Twine((
int)
Kind) +
3100 State.Other = readULEB128(State.Current, &
error);
3103 " in export trie data at node: 0x" +
3111 if ((int64_t)State.Other > 0 && State.Other > O->getLibraryCount()) {
3113 "bad library ordinal: " + Twine((
int)State.Other) +
" (max " +
3114 Twine((
int)O->getLibraryCount()) +
3120 State.ImportName =
reinterpret_cast<const char*
>(State.Current);
3121 if (*State.ImportName ==
'\0') {
3124 const uint8_t *End = State.Current + 1;
3125 if (End >= Trie.end()) {
3126 *E =
malformedError(
"import name of re-export in export trie data at "
3129 " starts past end of trie data");
3133 while(*End !=
'\0' && End < Trie.end())
3136 *E =
malformedError(
"import name of re-export in export trie data at "
3139 " extends past end of trie data");
3143 State.Current = End + 1;
3146 State.Address = readULEB128(State.Current, &
error);
3149 " in export trie data at node: 0x" +
3155 State.Other = readULEB128(State.Current, &
error);
3158 " in export trie data at node: 0x" +
3165 if (ExportStart + ExportInfoSize < State.Current) {
3167 "inconsistent export info size: 0x" +
3176 if (State.ChildCount != 0 && Children + 1 >= Trie.end()) {
3177 *E =
malformedError(
"byte for count of children in export trie data at "
3180 " extends past end of trie data");
3185 State.NextChildIndex = 0;
3186 State.ParentStringLength = CumulativeString.size();
3187 Stack.push_back(State);
3190void ExportEntry::pushDownUntilBottom() {
3191 ErrorAsOutParameter ErrAsOutParam(E);
3192 const char *
error =
nullptr;
3193 while (Stack.back().NextChildIndex < Stack.back().ChildCount) {
3194 NodeState &Top = Stack.back();
3195 CumulativeString.resize(Top.ParentStringLength);
3196 for (;*Top.Current != 0 && Top.Current < Trie.end(); Top.Current++) {
3197 char C = *Top.Current;
3198 CumulativeString.push_back(
C);
3200 if (Top.Current >= Trie.end()) {
3201 *E =
malformedError(
"edge sub-string in export trie data at node: 0x" +
3203 " for child #" + Twine((
int)Top.NextChildIndex) +
3204 " extends past end of trie data");
3209 uint64_t childNodeIndex = readULEB128(Top.Current, &
error);
3212 " in export trie data at node: 0x" +
3217 for (
const NodeState &node :
nodes()) {
3218 if (node.Start == Trie.begin() + childNodeIndex){
3219 *E =
malformedError(
"loop in children in export trie data at node: 0x" +
3221 " back to node: 0x" +
3227 Top.NextChildIndex += 1;
3228 pushNode(childNodeIndex);
3232 if (!Stack.back().IsExportNode) {
3233 *E =
malformedError(
"node is not an export node in export trie data at "
3257 assert(!Stack.empty() &&
"ExportEntry::moveNext() with empty node stack");
3258 if (!Stack.back().IsExportNode) {
3259 *E =
malformedError(
"node is not an export node in export trie data at "
3267 while (!Stack.empty()) {
3268 NodeState &Top = Stack.back();
3269 if (Top.NextChildIndex < Top.ChildCount) {
3270 pushDownUntilBottom();
3274 if (Top.IsExportNode) {
3276 CumulativeString.resize(Top.ParentStringLength);
3287 const MachOObjectFile *O) {
3292 Start.moveToFirst();
3302 if (DyldInfoLoadCmd)
3304 else if (DyldExportsTrieLoadCmd)
3307 return exports(Err, Trie,
this);
3314 for (
const auto &Command :
O->load_commands()) {
3315 if (Command.C.cmd == MachO::LC_SEGMENT) {
3316 MachO::segment_command SLC = O->getSegmentLoadCommand(Command);
3317 if (StringRef(SLC.segname) ==
"__TEXT") {
3318 TextAddress = SLC.vmaddr;
3321 }
else if (Command.C.cmd == MachO::LC_SEGMENT_64) {
3322 MachO::segment_command_64 SLC_64 = O->getSegment64LoadCommand(Command);
3323 if (StringRef(SLC_64.segname) ==
"__TEXT") {
3324 TextAddress = SLC_64.vmaddr;
3384 if (
auto FixupTargetsOrErr =
O->getDyldChainedFixupTargets()) {
3385 FixupTargets = *FixupTargetsOrErr;
3387 *
E = FixupTargetsOrErr.takeError();
3391 if (
auto SegmentsOrErr =
O->getChainedFixupsSegments()) {
3392 Segments = std::move(SegmentsOrErr->second);
3394 *
E = SegmentsOrErr.takeError();
3399void MachOChainedFixupEntry::findNextPageWithFixups() {
3400 auto FindInSegment = [
this]() {
3402 while (PageIndex < SegInfo.
PageStarts.size() &&
3405 return PageIndex < SegInfo.
PageStarts.size();
3408 while (InfoSegIndex < Segments.size()) {
3409 if (FindInSegment()) {
3410 PageOffset = Segments[InfoSegIndex].PageStarts[PageIndex];
3422 if (Segments.empty()) {
3430 findNextPageWithFixups();
3441 if (InfoSegIndex == Segments.size()) {
3455 " has unsupported chained fixup pointer_format " +
3456 Twine(PointerFormat));
3470 " extends past segment's end");
3481 assert(
O->isLittleEndian() &&
"big-endian object should have been rejected "
3482 "by getDyldChainedFixupTargets()");
3489 bool IsBind =
Field(63, 1);
3496 if (ImportOrdinal >= FixupTargets.size()) {
3499 " has out-of range import ordinal " +
3500 Twine(ImportOrdinal));
3507 Addend = InlineAddend ? InlineAddend :
Target.addend();
3521 PageOffset += 4 *
Next;
3524 findNextPageWithFixups();
3534 return InfoSegIndex ==
Other.InfoSegIndex && PageIndex ==
Other.PageIndex &&
3535 PageOffset ==
Other.PageOffset;
3540 : E(E), O(O), Opcodes(Bytes), Ptr(Bytes.begin()),
3541 PointerSize(
is64Bit ? 8 : 4) {}
3543void MachORebaseEntry::moveToFirst() {
3544 Ptr = Opcodes.
begin();
3548void MachORebaseEntry::moveToEnd() {
3549 Ptr = Opcodes.
end();
3550 RemainingLoopCount = 0;
3557 SegmentOffset += AdvanceAmount;
3558 if (RemainingLoopCount) {
3559 --RemainingLoopCount;
3568 if (Ptr == Opcodes.end()) {
3574 const uint8_t *OpcodeStart = Ptr;
3579 const char *
error =
nullptr;
3588 RebaseType = ImmValue;
3590 *E =
malformedError(
"for REBASE_OPCODE_SET_TYPE_IMM bad bind type: " +
3591 Twine((
int)RebaseType) +
" for opcode at: 0x" +
3598 dbgs() <<
"REBASE_OPCODE_SET_TYPE_IMM: "
3599 <<
"RebaseType=" << (
int) RebaseType <<
"\n");
3602 SegmentIndex = ImmValue;
3603 SegmentOffset = readULEB128(&
error);
3605 *E =
malformedError(
"for REBASE_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB " +
3611 error = O->RebaseEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset,
3614 *E =
malformedError(
"for REBASE_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB " +
3622 dbgs() <<
"REBASE_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB: "
3623 <<
"SegmentIndex=" << SegmentIndex <<
", "
3624 <<
format(
"SegmentOffset=0x%06X", SegmentOffset)
3628 SegmentOffset += readULEB128(&
error);
3631 " for opcode at: 0x" +
3636 error = O->RebaseEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset,
3640 " for opcode at: 0x" +
3646 dbgs() <<
"REBASE_OPCODE_ADD_ADDR_ULEB: "
3647 <<
format(
"SegmentOffset=0x%06X",
3648 SegmentOffset) <<
"\n");
3651 SegmentOffset += ImmValue * PointerSize;
3652 error = O->RebaseEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset,
3662 dbgs() <<
"REBASE_OPCODE_ADD_ADDR_IMM_SCALED: "
3663 <<
format(
"SegmentOffset=0x%06X",
3664 SegmentOffset) <<
"\n");
3667 AdvanceAmount = PointerSize;
3671 RemainingLoopCount = ImmValue - 1;
3673 RemainingLoopCount = 0;
3674 error = O->RebaseEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset,
3675 PointerSize,
Count, Skip);
3685 dbgs() <<
"REBASE_OPCODE_DO_REBASE_IMM_TIMES: "
3686 <<
format(
"SegmentOffset=0x%06X", SegmentOffset)
3687 <<
", AdvanceAmount=" << AdvanceAmount
3688 <<
", RemainingLoopCount=" << RemainingLoopCount
3692 AdvanceAmount = PointerSize;
3703 RemainingLoopCount =
Count - 1;
3705 RemainingLoopCount = 0;
3706 error = O->RebaseEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset,
3707 PointerSize,
Count, Skip);
3717 dbgs() <<
"REBASE_OPCODE_DO_REBASE_ULEB_TIMES: "
3718 <<
format(
"SegmentOffset=0x%06X", SegmentOffset)
3719 <<
", AdvanceAmount=" << AdvanceAmount
3720 <<
", RemainingLoopCount=" << RemainingLoopCount
3724 Skip = readULEB128(&
error);
3726 *E =
malformedError(
"for REBASE_OPCODE_DO_REBASE_ADD_ADDR_ULEB " +
3732 AdvanceAmount = Skip + PointerSize;
3734 RemainingLoopCount = 0;
3735 error = O->RebaseEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset,
3736 PointerSize,
Count, Skip);
3738 *E =
malformedError(
"for REBASE_OPCODE_DO_REBASE_ADD_ADDR_ULEB " +
3746 dbgs() <<
"REBASE_OPCODE_DO_REBASE_ADD_ADDR_ULEB: "
3747 <<
format(
"SegmentOffset=0x%06X", SegmentOffset)
3748 <<
", AdvanceAmount=" << AdvanceAmount
3749 <<
", RemainingLoopCount=" << RemainingLoopCount
3755 *E =
malformedError(
"for REBASE_OPCODE_DO_REBASE_ULEB_TIMES_SKIPPING_"
3763 RemainingLoopCount =
Count - 1;
3765 RemainingLoopCount = 0;
3766 Skip = readULEB128(&
error);
3768 *E =
malformedError(
"for REBASE_OPCODE_DO_REBASE_ULEB_TIMES_SKIPPING_"
3775 AdvanceAmount = Skip + PointerSize;
3777 error = O->RebaseEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset,
3778 PointerSize,
Count, Skip);
3780 *E =
malformedError(
"for REBASE_OPCODE_DO_REBASE_ULEB_TIMES_SKIPPING_"
3789 dbgs() <<
"REBASE_OPCODE_DO_REBASE_ULEB_TIMES_SKIPPING_ULEB: "
3790 <<
format(
"SegmentOffset=0x%06X", SegmentOffset)
3791 <<
", AdvanceAmount=" << AdvanceAmount
3792 <<
", RemainingLoopCount=" << RemainingLoopCount
3809 if (Ptr > Opcodes.
end())
3810 Ptr = Opcodes.
end();
3819 switch (RebaseType) {
3823 return "text abs32";
3825 return "text rel32";
3833 return O->BindRebaseSegmentName(SegmentIndex);
3839 return O->BindRebaseSectionName(SegmentIndex, SegmentOffset);
3845 return O->BindRebaseAddress(SegmentIndex, SegmentOffset);
3849#ifdef EXPENSIVE_CHECKS
3850 assert(Opcodes ==
Other.Opcodes &&
"compare iterators of different files");
3852 assert(Opcodes.data() ==
Other.Opcodes.data() &&
"compare iterators of different files");
3854 return (Ptr ==
Other.Ptr) &&
3855 (RemainingLoopCount ==
Other.RemainingLoopCount) &&
3856 (Done ==
Other.Done);
3862 if (O->BindRebaseSectionTable ==
nullptr)
3863 O->BindRebaseSectionTable = std::make_unique<BindRebaseSegInfo>(O);
3865 Start.moveToFirst();
3879 : E(E), O(O), Opcodes(Bytes), Ptr(Bytes.begin()),
3880 PointerSize(
is64Bit ? 8 : 4), TableKind(BK) {}
3882void MachOBindEntry::moveToFirst() {
3883 Ptr = Opcodes.
begin();
3887void MachOBindEntry::moveToEnd() {
3888 Ptr = Opcodes.
end();
3889 RemainingLoopCount = 0;
3896 SegmentOffset += AdvanceAmount;
3897 if (RemainingLoopCount) {
3898 --RemainingLoopCount;
3907 if (Ptr == Opcodes.end()) {
3913 const uint8_t *OpcodeStart = Ptr;
3917 int8_t SignExtended;
3920 const char *
error =
nullptr;
3926 bool NotLastEntry =
false;
3927 for (
const uint8_t *
P = Ptr;
P < Opcodes.end(); ++
P) {
3929 NotLastEntry =
true;
3941 *E =
malformedError(
"BIND_OPCODE_SET_DYLIB_ORDINAL_IMM not allowed in "
3942 "weak bind table for opcode at: 0x" +
3948 LibraryOrdinalSet =
true;
3949 if (ImmValue > O->getLibraryCount()) {
3950 *E =
malformedError(
"for BIND_OPCODE_SET_DYLIB_ORDINAL_ULEB bad "
3951 "library ordinal: " +
3952 Twine((
int)ImmValue) +
" (max " +
3953 Twine((
int)O->getLibraryCount()) +
3954 ") for opcode at: 0x" +
3961 dbgs() <<
"BIND_OPCODE_SET_DYLIB_ORDINAL_IMM: "
3962 <<
"Ordinal=" << Ordinal <<
"\n");
3966 *E =
malformedError(
"BIND_OPCODE_SET_DYLIB_ORDINAL_ULEB not allowed in "
3967 "weak bind table for opcode at: 0x" +
3972 Ordinal = readULEB128(&
error);
3973 LibraryOrdinalSet =
true;
3981 if (Ordinal > (
int)O->getLibraryCount()) {
3982 *E =
malformedError(
"for BIND_OPCODE_SET_DYLIB_ORDINAL_ULEB bad "
3983 "library ordinal: " +
3984 Twine((
int)Ordinal) +
" (max " +
3985 Twine((
int)O->getLibraryCount()) +
3986 ") for opcode at: 0x" +
3993 dbgs() <<
"BIND_OPCODE_SET_DYLIB_ORDINAL_ULEB: "
3994 <<
"Ordinal=" << Ordinal <<
"\n");
3998 *E =
malformedError(
"BIND_OPCODE_SET_DYLIB_SPECIAL_IMM not allowed in "
3999 "weak bind table for opcode at: 0x" +
4006 Ordinal = SignExtended;
4008 *E =
malformedError(
"for BIND_OPCODE_SET_DYLIB_SPECIAL_IMM unknown "
4009 "special ordinal: " +
4010 Twine((
int)Ordinal) +
" for opcode at: 0x" +
4017 LibraryOrdinalSet =
true;
4020 dbgs() <<
"BIND_OPCODE_SET_DYLIB_SPECIAL_IMM: "
4021 <<
"Ordinal=" << Ordinal <<
"\n");
4026 while (*Ptr && (Ptr < Opcodes.end())) {
4029 if (Ptr == Opcodes.end()) {
4031 "for BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM "
4032 "symbol name extends past opcodes for opcode at: 0x" +
4037 SymbolName =
StringRef(
reinterpret_cast<const char*
>(SymStart),
4042 dbgs() <<
"BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM: "
4043 <<
"SymbolName=" << SymbolName <<
"\n");
4050 BindType = ImmValue;
4052 *E =
malformedError(
"for BIND_OPCODE_SET_TYPE_IMM bad bind type: " +
4053 Twine((
int)ImmValue) +
" for opcode at: 0x" +
4060 dbgs() <<
"BIND_OPCODE_SET_TYPE_IMM: "
4061 <<
"BindType=" << (
int)BindType <<
"\n");
4064 Addend = readSLEB128(&
error);
4067 " for opcode at: 0x" +
4074 dbgs() <<
"BIND_OPCODE_SET_ADDEND_SLEB: "
4075 <<
"Addend=" << Addend <<
"\n");
4078 SegmentIndex = ImmValue;
4079 SegmentOffset = readULEB128(&
error);
4081 *E =
malformedError(
"for BIND_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB " +
4087 error = O->BindEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset,
4090 *E =
malformedError(
"for BIND_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB " +
4098 dbgs() <<
"BIND_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB: "
4099 <<
"SegmentIndex=" << SegmentIndex <<
", "
4100 <<
format(
"SegmentOffset=0x%06X", SegmentOffset)
4104 SegmentOffset += readULEB128(&
error);
4107 " for opcode at: 0x" +
4112 error = O->BindEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset,
4116 " for opcode at: 0x" +
4122 dbgs() <<
"BIND_OPCODE_ADD_ADDR_ULEB: "
4123 <<
format(
"SegmentOffset=0x%06X",
4124 SegmentOffset) <<
"\n");
4127 AdvanceAmount = PointerSize;
4128 RemainingLoopCount = 0;
4129 error = O->BindEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset,
4133 " for opcode at: 0x" +
4140 "for BIND_OPCODE_DO_BIND missing preceding "
4141 "BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM for opcode at: 0x" +
4146 if (!LibraryOrdinalSet && TableKind !=
Kind::Weak) {
4149 "BIND_OPCODE_SET_DYLIB_ORDINAL_* for opcode at: 0x" +
4155 dbgs() <<
"BIND_OPCODE_DO_BIND: "
4156 <<
format(
"SegmentOffset=0x%06X",
4157 SegmentOffset) <<
"\n");
4161 *E =
malformedError(
"BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB not allowed in "
4162 "lazy bind table for opcode at: 0x" +
4167 error = O->BindEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset,
4178 "for BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB missing "
4179 "preceding BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM for opcode "
4185 if (!LibraryOrdinalSet && TableKind !=
Kind::Weak) {
4187 "for BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB missing "
4188 "preceding BIND_OPCODE_SET_DYLIB_ORDINAL_* for opcode at: 0x" +
4193 AdvanceAmount = readULEB128(&
error) + PointerSize;
4204 error = O->BindEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset +
4205 AdvanceAmount, PointerSize);
4207 *E =
malformedError(
"for BIND_OPCODE_ADD_ADDR_ULEB (after adding "
4214 RemainingLoopCount = 0;
4217 dbgs() <<
"BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB: "
4218 <<
format(
"SegmentOffset=0x%06X", SegmentOffset)
4219 <<
", AdvanceAmount=" << AdvanceAmount
4220 <<
", RemainingLoopCount=" << RemainingLoopCount
4225 *E =
malformedError(
"BIND_OPCODE_DO_BIND_ADD_ADDR_IMM_SCALED not "
4226 "allowed in lazy bind table for opcode at: 0x" +
4233 "for BIND_OPCODE_DO_BIND_ADD_ADDR_IMM_SCALED "
4234 "missing preceding BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM for "
4240 if (!LibraryOrdinalSet && TableKind !=
Kind::Weak) {
4242 "for BIND_OPCODE_DO_BIND_ADD_ADDR_IMM_SCALED "
4243 "missing preceding BIND_OPCODE_SET_DYLIB_ORDINAL_* for opcode "
4249 AdvanceAmount = ImmValue * PointerSize + PointerSize;
4250 RemainingLoopCount = 0;
4251 error = O->BindEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset +
4252 AdvanceAmount, PointerSize);
4254 *E =
malformedError(
"for BIND_OPCODE_DO_BIND_ADD_ADDR_IMM_SCALED " +
4262 <<
"BIND_OPCODE_DO_BIND_ADD_ADDR_IMM_SCALED: "
4263 <<
format(
"SegmentOffset=0x%06X", SegmentOffset) <<
"\n");
4267 *E =
malformedError(
"BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB not "
4268 "allowed in lazy bind table for opcode at: 0x" +
4275 RemainingLoopCount =
Count - 1;
4277 RemainingLoopCount = 0;
4279 *E =
malformedError(
"for BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB "
4286 Skip = readULEB128(&
error);
4287 AdvanceAmount = Skip + PointerSize;
4289 *E =
malformedError(
"for BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB "
4298 "for BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB "
4299 "missing preceding BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM for "
4305 if (!LibraryOrdinalSet && TableKind !=
Kind::Weak) {
4307 "for BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB "
4308 "missing preceding BIND_OPCODE_SET_DYLIB_ORDINAL_* for opcode "
4314 error = O->BindEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset,
4315 PointerSize,
Count, Skip);
4318 malformedError(
"for BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB " +
4326 dbgs() <<
"BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB: "
4327 <<
format(
"SegmentOffset=0x%06X", SegmentOffset)
4328 <<
", AdvanceAmount=" << AdvanceAmount
4329 <<
", RemainingLoopCount=" << RemainingLoopCount
4346 if (Ptr > Opcodes.
end())
4347 Ptr = Opcodes.
end();
4351int64_t MachOBindEntry::readSLEB128(
const char **
error) {
4355 if (Ptr > Opcodes.
end())
4356 Ptr = Opcodes.
end();
4369 return "text abs32";
4371 return "text rel32";
4387 return O->BindRebaseSegmentName(SegmentIndex);
4393 return O->BindRebaseSectionName(SegmentIndex, SegmentOffset);
4399 return O->BindRebaseAddress(SegmentIndex, SegmentOffset);
4403#ifdef EXPENSIVE_CHECKS
4404 assert(Opcodes ==
Other.Opcodes &&
"compare iterators of different files");
4406 assert(Opcodes.data() ==
Other.Opcodes.data() &&
"compare iterators of different files");
4408 return (Ptr ==
Other.Ptr) &&
4409 (RemainingLoopCount ==
Other.RemainingLoopCount) &&
4410 (Done ==
Other.Done);
4415 uint32_t CurSegIndex = Obj->hasPageZeroSegment() ? 1 : 0;
4418 for (
const SectionRef &Section : Obj->sections()) {
4424 Info.SectionName = *NameOrErr;
4425 Info.Address = Section.getAddress();
4426 Info.Size = Section.getSize();
4428 Obj->getSectionFinalSegmentName(Section.getRawDataRefImpl());
4429 if (Info.SegmentName != CurSegName) {
4431 CurSegName = Info.SegmentName;
4432 CurSegAddress = Info.Address;
4434 Info.SegmentIndex = CurSegIndex - 1;
4435 Info.OffsetInSegment = Info.Address - CurSegAddress;
4436 Info.SegmentStartAddress = CurSegAddress;
4437 Sections.push_back(Info);
4439 MaxSegIndex = CurSegIndex;
4455 return "missing preceding *_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB";
4456 if (SegIndex >= MaxSegIndex)
4457 return "bad segIndex (too large)";
4459 uint64_t Start = SegOffset + i * (PointerSize + Skip);
4460 uint64_t End = Start + PointerSize;
4462 for (
const SectionInfo &
SI : Sections) {
4463 if (
SI.SegmentIndex != SegIndex)
4465 if ((
SI.OffsetInSegment<=Start) && (Start<(
SI.OffsetInSegment+
SI.Size))) {
4466 if (End <=
SI.OffsetInSegment +
SI.Size) {
4471 return "bad offset, extends beyond section boundary";
4475 return "bad offset, not in section";
4483 for (
const SectionInfo &
SI : Sections) {
4484 if (
SI.SegmentIndex == SegIndex)
4485 return SI.SegmentName;
4492const BindRebaseSegInfo::SectionInfo &BindRebaseSegInfo::findSection(
4493 int32_t SegIndex,
uint64_t SegOffset) {
4494 for (
const SectionInfo &
SI : Sections) {
4495 if (
SI.SegmentIndex != SegIndex)
4497 if (
SI.OffsetInSegment > SegOffset)
4499 if (SegOffset >= (
SI.OffsetInSegment +
SI.Size))
4510 return findSection(SegIndex, SegOffset).SectionName;
4516 const SectionInfo &
SI = findSection(SegIndex, OffsetInSeg);
4517 return SI.SegmentStartAddress + OffsetInSeg;
4524 if (O->BindRebaseSectionTable ==
nullptr)
4525 O->BindRebaseSectionTable = std::make_unique<BindRebaseSegInfo>(O);
4527 Start.moveToFirst();
4551 if (BindRebaseSectionTable ==
nullptr)
4552 BindRebaseSectionTable = std::make_unique<BindRebaseSegInfo>(
this);
4555 Start.moveToFirst();
4565 return LoadCommands.begin();
4570 return LoadCommands.end();
4586 assert(Sec.
d.
a < Sections.size() &&
"Should have detected this earlier");
4587 const section_base *
Base =
4588 reinterpret_cast<const section_base *
>(Sections[Sec.
d.
a]);
4594 assert(Sec.
d.
a < Sections.size() &&
"Should have detected this earlier");
4595 const section_base *
Base =
4596 reinterpret_cast<const section_base *
>(Sections[Sec.
d.
a]);
4618 return (RE.
r_word1 >> 27) & 1;
4634 return (RE.
r_word0 >> 24) & 0xf;
4672 if (SecNum ==
MachO::R_ABS || SecNum > Sections.size())
4675 DRI.
d.
a = SecNum - 1;
4680 assert(DRI.
d.
a < Sections.size() &&
"Should have detected this earlier");
4685 assert(DRI.
d.
a < Sections.size() &&
"Should have detected this earlier");
4690 unsigned Index)
const {
4696 unsigned Index)
const {
4703 const char *
P =
reinterpret_cast<const char *
>(DRI.
p);
4709 const char *
P =
reinterpret_cast<const char *
>(DRI.
p);
4854 Offset = DysymtabLoadCmd.extreloff;
4856 Offset = DysymtabLoadCmd.locreloff;
4862 *
this,
reinterpret_cast<const char *
>(
P));
4867 const char *
P =
reinterpret_cast<const char *
>(Rel.
p);
4882 unsigned Index)
const {
4889 unsigned Index)
const {
4900 Cmd.
cmd = MachO::LC_SYMTAB;
4910 if (DysymtabLoadCmd)
4915 Cmd.
cmd = MachO::LC_DYSYMTAB;
4940 if (DataInCodeLoadCmd)
4945 Cmd.
cmd = MachO::LC_DATA_IN_CODE;
4954 if (LinkOptHintsLoadCmd)
4960 Cmd.
cmd = MachO::LC_LINKER_OPTIMIZATION_HINT;
4968 if (!DyldInfoLoadCmd)
4971 auto DyldInfoOrErr =
4982 if (!DyldInfoLoadCmd)
4985 auto DyldInfoOrErr =
4996 if (!DyldInfoLoadCmd)
4999 auto DyldInfoOrErr =
5010 if (!DyldInfoLoadCmd)
5013 auto DyldInfoOrErr =
5024 if (!DyldInfoLoadCmd)
5027 auto DyldInfoOrErr =
5040 if (!DyldChainedFixupsLoadCmd)
5041 return std::nullopt;
5043 *
this, DyldChainedFixupsLoadCmd);
5044 if (!DyldChainedFixupsOrErr)
5045 return DyldChainedFixupsOrErr.takeError();
5047 *DyldChainedFixupsOrErr;
5051 if (!DyldChainedFixups.
dataoff)
5052 return std::nullopt;
5053 return DyldChainedFixups;
5060 return CFOrErr.takeError();
5061 if (!CFOrErr->has_value())
5062 return std::nullopt;
5070 const char *CFHeaderPtr =
getPtr(*
this, CFHeaderOffset);
5071 auto CFHeaderOrErr =
5074 return CFHeaderOrErr.takeError();
5083 Twine(
"bad chained fixups: unknown imports format: ") +
5093 " overlaps with chained fixups header");
5095 uint32_t EndOffset = CFHeaderOffset + CFSize;
5099 Twine(CFImageStartsOffset +
5101 " extends past end " +
Twine(EndOffset));
5111 return CFOrErr.takeError();
5113 std::vector<ChainedFixupsSegment> Segments;
5114 if (!CFOrErr->has_value())
5115 return std::make_pair(0, Segments);
5121 return HeaderOrErr.takeError();
5122 if (!HeaderOrErr->has_value())
5123 return std::make_pair(0, Segments);
5126 const char *Contents =
getPtr(*
this, DyldChainedFixups.
dataoff);
5129 *
this, Contents +
Header.starts_offset);
5130 if (!ImageStartsOrErr)
5131 return ImageStartsOrErr.takeError();
5134 const char *SegOffsPtr =
5135 Contents +
Header.starts_offset +
5137 const char *SegOffsEnd =
5139 if (SegOffsEnd > Contents + DyldChainedFixups.
datasize)
5141 "bad chained fixups: seg_info_offset extends past end");
5143 const char *LastSegEnd =
nullptr;
5148 return OffOrErr.takeError();
5156 " at offset " +
Twine(*OffOrErr) + Message);
5159 const char *SegPtr = Contents +
Header.starts_offset + *OffOrErr;
5160 if (LastSegEnd && SegPtr < LastSegEnd)
5161 return Fail(
" overlaps with previous segment info");
5166 return SegOrErr.takeError();
5169 LastSegEnd = SegPtr + Seg.
size;
5173 const char *PageStart =
5176 if (PageEnd > SegPtr + Seg.
size)
5177 return Fail(
" : page_starts extend past seg_info size");
5181 std::vector<uint16_t> PageStarts;
5182 for (
size_t PageIdx = 0; PageIdx < Seg.
page_count; ++PageIdx) {
5187 PageStarts.push_back(Start);
5190 Segments.emplace_back(
I, *OffOrErr, Seg, std::move(PageStarts));
5193 return std::make_pair(ImageStarts.
seg_count, Segments);
5206template <
typename T,
unsigned N>
5208 std::array<T, N> RawValue;
5209 memcpy(RawValue.data(), Ptr,
N *
sizeof(
T));
5211 for (
auto &Element : RawValue)
5216Expected<std::vector<ChainedFixupTarget>>
5220 return CFOrErr.takeError();
5222 std::vector<ChainedFixupTarget> Targets;
5223 if (!CFOrErr->has_value())
5230 return CFHeaderOrErr.takeError();
5231 if (!(*CFHeaderOrErr))
5235 size_t ImportSize = 0;
5243 return malformedError(
"bad chained fixups: unknown imports format: " +
5246 const char *Contents =
getPtr(*
this, DyldChainedFixups.
dataoff);
5247 const char *Imports = Contents +
Header.imports_offset;
5248 size_t ImportsEndOffset =
5249 Header.imports_offset + ImportSize *
Header.imports_count;
5250 const char *ImportsEnd = Contents + ImportsEndOffset;
5251 const char *Symbols = Contents +
Header.symbols_offset;
5252 const char *SymbolsEnd = Contents + DyldChainedFixups.
datasize;
5254 if (ImportsEnd > Symbols)
5256 Twine(ImportsEndOffset) +
" overlaps with symbols");
5261 return createError(
"parsing big-endian chained fixups is not implemented");
5262 for (
const char *ImportPtr = Imports; ImportPtr < ImportsEnd;
5263 ImportPtr += ImportSize) {
5273 WeakImport = (RawValue[0] >> 8) & 1;
5274 NameOffset = RawValue[0] >> 9;
5282 WeakImport = (RawValue[0] >> 8) & 1;
5283 NameOffset = RawValue[0] >> 9;
5286 static_assert(2 *
sizeof(
uint64_t) ==
5291 NameOffset = (RawValue[0] >> 16) & 1;
5292 WeakImport = RawValue[0] >> 17;
5293 Addend = RawValue[1];
5298 const char *Str = Symbols + NameOffset;
5299 if (Str >= SymbolsEnd)
5301 Twine(NameOffset) +
" extends past end " +
5303 Targets.emplace_back(LibOrdinal, NameOffset, Str, Addend, WeakImport);
5306 return std::move(Targets);
5310 if (!DyldExportsTrieLoadCmd)
5314 *
this, DyldExportsTrieLoadCmd);
5315 if (!DyldExportsTrieOrError)
5324 if (!FuncStartsLoadCmd)
5335 return std::move(FunctionStarts);
5381 size_t MachOFilesetEntryOffset) {
5383 if (Magic ==
"\xFE\xED\xFA\xCE")
5385 UniversalIndex, MachOFilesetEntryOffset);
5386 if (Magic ==
"\xCE\xFA\xED\xFE")
5388 UniversalIndex, MachOFilesetEntryOffset);
5389 if (Magic ==
"\xFE\xED\xFA\xCF")
5391 UniversalIndex, MachOFilesetEntryOffset);
5392 if (Magic ==
"\xCF\xFA\xED\xFE")
5394 UniversalIndex, MachOFilesetEntryOffset);
5401 .
Case(
"debug_str_offs",
"debug_str_offsets")
5412 return std::vector<std::string>();
5418 EC,
"%s: expected directory 'Contents/Resources/DWARF' in dSYM bundle",
5419 Path.str().c_str());
5423 std::vector<std::string> ObjectPaths;
5425 Dir != DirEnd && !EC; Dir.increment(EC)) {
5434 ObjectPaths.push_back(ObjectPath.
str());
5441 if (ObjectPaths.empty())
5443 "%s: no objects found in dSYM bundle",
5444 Path.str().c_str());
5451#define HANDLE_SWIFT_SECTION(KIND, MACHO, ELF, COFF) \
5452 .Case(MACHO, llvm::binaryformat::Swift5ReflectionSectionKind::KIND)
5455#include "llvm/BinaryFormat/Swift.def"
5457#undef HANDLE_SWIFT_SECTION
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
Unify divergent function exit nodes
static GCRegistry::Add< StatepointGC > D("statepoint-example", "an example strategy for statepoint")
static GCRegistry::Add< CoreCLRGC > E("coreclr", "CoreCLR-compatible GC")
#define offsetof(TYPE, MEMBER)
static MachO::nlist_base getSymbolTableEntryBase(const MachOObjectFile &O, DataRefImpl DRI)
static Error checkVersCommand(const MachOObjectFile &Obj, const MachOObjectFile::LoadCommandInfo &Load, uint32_t LoadCommandIndex, const char **LoadCmd, const char *CmdName)
static Error checkSymtabCommand(const MachOObjectFile &Obj, const MachOObjectFile::LoadCommandInfo &Load, uint32_t LoadCommandIndex, const char **SymtabLoadCmd, std::list< MachOElement > &Elements)
static Error checkTwoLevelHintsCommand(const MachOObjectFile &Obj, const MachOObjectFile::LoadCommandInfo &Load, uint32_t LoadCommandIndex, const char **LoadCmd, std::list< MachOElement > &Elements)
static Error parseBuildVersionCommand(const MachOObjectFile &Obj, const MachOObjectFile::LoadCommandInfo &Load, SmallVectorImpl< const char * > &BuildTools, uint32_t LoadCommandIndex)
static unsigned getPlainRelocationType(const MachOObjectFile &O, const MachO::any_relocation_info &RE)
static Error checkDysymtabCommand(const MachOObjectFile &Obj, const MachOObjectFile::LoadCommandInfo &Load, uint32_t LoadCommandIndex, const char **DysymtabLoadCmd, std::list< MachOElement > &Elements)
static Error checkDylibCommand(const MachOObjectFile &Obj, const MachOObjectFile::LoadCommandInfo &Load, uint32_t LoadCommandIndex, const char *CmdName)
static Expected< T > getStructOrErr(const MachOObjectFile &O, const char *P)
static Expected< MachOObjectFile::LoadCommandInfo > getFirstLoadCommandInfo(const MachOObjectFile &Obj)
static const char * getPtr(const MachOObjectFile &O, size_t Offset, size_t MachOFilesetEntryOffset=0)
static Error parseSegmentLoadCommand(const MachOObjectFile &Obj, const MachOObjectFile::LoadCommandInfo &Load, SmallVectorImpl< const char * > &Sections, bool &IsPageZeroSegment, uint32_t LoadCommandIndex, const char *CmdName, uint64_t SizeOfHeaders, std::list< MachOElement > &Elements)
static Error checkDyldInfoCommand(const MachOObjectFile &Obj, const MachOObjectFile::LoadCommandInfo &Load, uint32_t LoadCommandIndex, const char **LoadCmd, const char *CmdName, std::list< MachOElement > &Elements)
static unsigned getScatteredRelocationLength(const MachO::any_relocation_info &RE)
static unsigned getPlainRelocationLength(const MachOObjectFile &O, const MachO::any_relocation_info &RE)
static Error checkSubCommand(const MachOObjectFile &Obj, const MachOObjectFile::LoadCommandInfo &Load, uint32_t LoadCommandIndex, const char *CmdName, size_t SizeOfCmd, const char *CmdStructName, uint32_t PathOffset, const char *PathFieldName)
static Error checkRpathCommand(const MachOObjectFile &Obj, const MachOObjectFile::LoadCommandInfo &Load, uint32_t LoadCommandIndex)
static T getStruct(const MachOObjectFile &O, const char *P)
static uint32_t getPlainRelocationAddress(const MachO::any_relocation_info &RE)
static const char * getSectionPtr(const MachOObjectFile &O, MachOObjectFile::LoadCommandInfo L, unsigned Sec)
static Error checkLinkerOptCommand(const MachOObjectFile &Obj, const MachOObjectFile::LoadCommandInfo &Load, uint32_t LoadCommandIndex)
static bool getPlainRelocationPCRel(const MachOObjectFile &O, const MachO::any_relocation_info &RE)
static std::array< T, N > getArray(const MachOObjectFile &O, const void *Ptr)
static unsigned getScatteredRelocationAddress(const MachO::any_relocation_info &RE)
static Error checkThreadCommand(const MachOObjectFile &Obj, const MachOObjectFile::LoadCommandInfo &Load, uint32_t LoadCommandIndex, const char *CmdName)
static Error checkLinkeditDataCommand(const MachOObjectFile &Obj, const MachOObjectFile::LoadCommandInfo &Load, uint32_t LoadCommandIndex, const char **LoadCmd, const char *CmdName, std::list< MachOElement > &Elements, const char *ElementName)
static Error malformedError(const Twine &Msg)
static bool isLoadCommandObsolete(uint32_t cmd)
static uint32_t getSectionFlags(const MachOObjectFile &O, DataRefImpl Sec)
static int getEncodedOrdinal(T Value)
static bool getScatteredRelocationPCRel(const MachO::any_relocation_info &RE)
static Error checkOverlappingElement(std::list< MachOElement > &Elements, uint64_t Offset, uint64_t Size, const char *Name)
static StringRef parseSegmentOrSectionName(const char *P)
static Expected< MachOObjectFile::LoadCommandInfo > getLoadCommandInfo(const MachOObjectFile &Obj, const char *Ptr, uint32_t LoadCommandIndex)
static void parseHeader(const MachOObjectFile &Obj, T &Header, Error &Err)
static unsigned getCPUType(const MachOObjectFile &O)
static Error checkDyldCommand(const MachOObjectFile &Obj, const MachOObjectFile::LoadCommandInfo &Load, uint32_t LoadCommandIndex, const char *CmdName)
static Error checkNoteCommand(const MachOObjectFile &Obj, const MachOObjectFile::LoadCommandInfo &Load, uint32_t LoadCommandIndex, std::list< MachOElement > &Elements)
static Error checkDylibIdCommand(const MachOObjectFile &Obj, const MachOObjectFile::LoadCommandInfo &Load, uint32_t LoadCommandIndex, const char **LoadCmd)
static unsigned getCPUSubType(const MachOObjectFile &O)
static Error checkEncryptCommand(const MachOObjectFile &Obj, const MachOObjectFile::LoadCommandInfo &Load, uint32_t LoadCommandIndex, uint64_t cryptoff, uint64_t cryptsize, const char **LoadCmd, const char *CmdName)
static Expected< MachOObjectFile::LoadCommandInfo > getNextLoadCommandInfo(const MachOObjectFile &Obj, uint32_t LoadCommandIndex, const MachOObjectFile::LoadCommandInfo &L)
static Error malformedError(Twine Msg)
OptimizedStructLayoutField Field
static StringRef substr(StringRef Str, uint64_t Len)
This file defines the SmallVector class.
This file implements the StringSwitch template, which mimics a switch() statement whose cases are str...
#define DEBUG_WITH_TYPE(TYPE,...)
DEBUG_WITH_TYPE macro - This macro should be used by passes to emit debug information.
static uint64_t readULEB128(WasmObjectFile::ReadContext &Ctx)
static bool is64Bit(const char *name)
This file implements the C++20 <bit> header.
ArrayRef - Represent a constant reference to an array (0 or more elements consecutively in memory),...
bool empty() const
empty - Check if the array is empty.
Helper for Errors used as out-parameters.
Lightweight error class with error context and mandatory checking.
static ErrorSuccess success()
Create a success value.
Tagged union holding either a T or a Error.
Error takeError()
Take ownership of the stored error.
reference get()
Returns a reference to the stored T value.
StringRef getBuffer() const
SmallString - A SmallString is just a SmallVector with methods and accessors that make it work better...
This class consists of common code factored out of the SmallVector class to reduce code duplication b...
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
StringRef - Represent a constant reference to a string, i.e.
static constexpr size_t npos
std::string str() const
str - Get the contents as an std::string.
constexpr StringRef substr(size_t Start, size_t N=npos) const
Return a reference to the substring from [Start, Start + N).
bool starts_with(StringRef Prefix) const
Check if this string starts with the given Prefix.
constexpr bool empty() const
empty - Check if the string is empty.
char back() const
back - Get the last character in the string.
StringRef slice(size_t Start, size_t End) const
Return a reference to the substring from [Start, End).
constexpr size_t size() const
size - Get the string size.
size_t rfind(char C, size_t From=npos) const
Search for the last character C in the string.
size_t find(char C, size_t From=0) const
Search for the first character C in the string.
A switch()-like statement whose cases are string literals.
StringSwitch & Case(StringLiteral S, T Value)
A table of densely packed, null-terminated strings indexed by offset.
constexpr size_t size() const
Returns the byte size of the table.
Target - Wrapper for Target specific information.
Triple - Helper class for working with autoconf configuration names.
Twine - A lightweight data structure for efficiently representing the concatenation of temporary valu...
static Twine utohexstr(uint64_t Val)
LLVM Value Representation.
A range adaptor for a pair of iterators.
DataRefImpl getRawDataRefImpl() const
StringRef getData() const
unsigned int getType() const
bool isLittleEndian() const
static unsigned int getMachOType(bool isLE, bool is64Bits)
LLVM_ABI StringRef segmentName(int32_t SegIndex)
LLVM_ABI StringRef sectionName(int32_t SegIndex, uint64_t SegOffset)
LLVM_ABI BindRebaseSegInfo(const MachOObjectFile *Obj)
LLVM_ABI const char * checkSegAndOffsets(int32_t SegIndex, uint64_t SegOffset, uint8_t PointerSize, uint64_t Count=1, uint64_t Skip=0)
LLVM_ABI uint64_t address(uint32_t SegIndex, uint64_t SegOffset)
DiceRef - This is a value type class that represents a single data in code entry in the table in a Ma...
ExportEntry encapsulates the current-state-of-the-walk used when doing a non-recursive walk of the tr...
LLVM_ABI StringRef name() const
LLVM_ABI ExportEntry(Error *Err, const MachOObjectFile *O, ArrayRef< uint8_t > Trie)
LLVM_ABI bool operator==(const ExportEntry &) const
LLVM_ABI StringRef otherName() const
LLVM_ABI uint64_t address() const
LLVM_ABI uint64_t flags() const
LLVM_ABI uint32_t nodeOffset() const
friend class MachOObjectFile
LLVM_ABI uint64_t other() const
LLVM_ABI StringRef sectionName() const
LLVM_ABI uint64_t segmentAddress() const
LLVM_ABI int32_t segmentIndex() const
LLVM_ABI uint64_t address() const
LLVM_ABI StringRef typeName() const
LLVM_ABI int64_t addend() const
LLVM_ABI MachOAbstractFixupEntry(Error *Err, const MachOObjectFile *O)
uint64_t textAddress() const
LLVM_ABI uint32_t flags() const
LLVM_ABI StringRef symbolName() const
LLVM_ABI void moveToFirst()
LLVM_ABI int ordinal() const
LLVM_ABI StringRef segmentName() const
LLVM_ABI void moveToEnd()
const MachOObjectFile * O
LLVM_ABI uint64_t segmentOffset() const
MachOBindEntry encapsulates the current state in the decompression of binding opcodes.
LLVM_ABI uint32_t flags() const
LLVM_ABI bool operator==(const MachOBindEntry &) const
LLVM_ABI StringRef symbolName() const
LLVM_ABI int ordinal() const
LLVM_ABI StringRef sectionName() const
LLVM_ABI MachOBindEntry(Error *Err, const MachOObjectFile *O, ArrayRef< uint8_t > Opcodes, bool is64Bit, MachOBindEntry::Kind)
LLVM_ABI StringRef segmentName() const
LLVM_ABI uint64_t segmentOffset() const
LLVM_ABI int64_t addend() const
LLVM_ABI uint64_t address() const
friend class MachOObjectFile
LLVM_ABI int32_t segmentIndex() const
LLVM_ABI StringRef typeName() const
LLVM_ABI bool operator==(const MachOChainedFixupEntry &) const
LLVM_ABI MachOChainedFixupEntry(Error *Err, const MachOObjectFile *O, bool Parse)
LLVM_ABI void moveToEnd()
LLVM_ABI void moveToFirst()
MachO::sub_client_command getSubClientCommand(const LoadCommandInfo &L) const
void moveSectionNext(DataRefImpl &Sec) const override
ArrayRef< char > getSectionRawFinalSegmentName(DataRefImpl Sec) const
uint8_t getBytesInAddress() const override
The number of bytes used to represent an address in this object file format.
Triple::ArchType getArch() const override
MachO::mach_header_64 Header64
bool isSectionData(DataRefImpl Sec) const override
const MachO::mach_header_64 & getHeader64() const
Expected< std::vector< ChainedFixupTarget > > getDyldChainedFixupTargets() const
uint64_t getSectionAlignment(DataRefImpl Sec) const override
uint32_t getScatteredRelocationType(const MachO::any_relocation_info &RE) const
symbol_iterator getRelocationSymbol(DataRefImpl Rel) const override
Expected< SectionRef > getSection(unsigned SectionIndex) const
iterator_range< rebase_iterator > rebaseTable(Error &Err)
For use iterating over all rebase table entries.
std::error_code getIndirectName(DataRefImpl Symb, StringRef &Res) const
load_command_iterator begin_load_commands() const
MachO::encryption_info_command_64 getEncryptionInfoCommand64(const LoadCommandInfo &L) const
StringRef getFileFormatName() const override
dice_iterator begin_dices() const
basic_symbol_iterator symbol_begin() const override
Expected< std::optional< MachO::linkedit_data_command > > getChainedFixupsLoadCommand() const
iterator_range< export_iterator > exports(Error &Err) const
For use iterating over all exported symbols.
uint64_t getSymbolIndex(DataRefImpl Symb) const
MachO::build_version_command getBuildVersionLoadCommand(const LoadCommandInfo &L) const
section_iterator section_end() const override
MachO::build_tool_version getBuildToolVersion(unsigned index) const
MachO::linkedit_data_command getDataInCodeLoadCommand() const
MachO::routines_command getRoutinesCommand(const LoadCommandInfo &L) const
MachO::nlist getSymbolTableEntry(DataRefImpl DRI) const
unsigned getSymbolSectionID(SymbolRef Symb) const
static Expected< std::vector< std::string > > findDsymObjectMembers(StringRef Path)
If the input path is a .dSYM bundle (as created by the dsymutil tool), return the paths to the object...
uint32_t getScatteredRelocationValue(const MachO::any_relocation_info &RE) const
MachO::linker_option_command getLinkerOptionLoadCommand(const LoadCommandInfo &L) const
uint32_t getLibraryCount() const
MachO::entry_point_command getEntryPointCommand(const LoadCommandInfo &L) const
Expected< section_iterator > getSymbolSection(DataRefImpl Symb) const override
uint64_t getRelocationOffset(DataRefImpl Rel) const override
ArrayRef< uint8_t > getDyldInfoLazyBindOpcodes() const
void moveSymbolNext(DataRefImpl &Symb) const override
SectionRef getAnyRelocationSection(const MachO::any_relocation_info &RE) const
MachO::dysymtab_command getDysymtabLoadCommand() const
iterator_range< bind_iterator > bindTable(Error &Err)
For use iterating over all bind table entries.
MachO::mach_header Header
uint64_t getCommonSymbolSizeImpl(DataRefImpl Symb) const override
relocation_iterator section_rel_begin(DataRefImpl Sec) const override
MachO::section_64 getSection64(DataRefImpl DRI) const
MachO::fileset_entry_command getFilesetEntryLoadCommand(const LoadCommandInfo &L) const
MachO::note_command getNoteLoadCommand(const LoadCommandInfo &L) const
MachO::thread_command getThreadCommand(const LoadCommandInfo &L) const
section_iterator section_begin() const override
Error checkSymbolTable() const
bool isRelocatableObject() const override
True if this is a relocatable object (.o/.obj).
MachO::segment_command_64 getSegment64LoadCommand(const LoadCommandInfo &L) const
relocation_iterator section_rel_end(DataRefImpl Sec) const override
ArrayRef< uint8_t > getDyldInfoExportsTrie() const
bool isDebugSection(DataRefImpl Sec) const override
MachO::nlist_64 getSymbol64TableEntry(DataRefImpl DRI) const
unsigned getSectionType(SectionRef Sec) const
MachO::segment_command getSegmentLoadCommand(const LoadCommandInfo &L) const
static Expected< std::unique_ptr< MachOObjectFile > > create(MemoryBufferRef Object, bool IsLittleEndian, bool Is64Bits, uint32_t UniversalCputype=0, uint32_t UniversalIndex=0, size_t MachOFilesetEntryOffset=0)
StringRef getSectionFinalSegmentName(DataRefImpl Sec) const
MachO::linkedit_data_command getLinkOptHintsLoadCommand() const
unsigned getAnyRelocationType(const MachO::any_relocation_info &RE) const
MachO::rpath_command getRpathCommand(const LoadCommandInfo &L) const
dice_iterator end_dices() const
MachO::routines_command_64 getRoutinesCommand64(const LoadCommandInfo &L) const
MachO::sub_framework_command getSubFrameworkCommand(const LoadCommandInfo &L) const
SmallVector< uint64_t > getFunctionStarts() const
MachO::sub_library_command getSubLibraryCommand(const LoadCommandInfo &L) const
MachO::dyld_info_command getDyldInfoLoadCommand(const LoadCommandInfo &L) const
MachO::sub_umbrella_command getSubUmbrellaCommand(const LoadCommandInfo &L) const
ArrayRef< uint8_t > getDyldExportsTrie() const
Expected< uint32_t > getSymbolFlags(DataRefImpl Symb) const override
section_iterator getRelocationRelocatedSection(relocation_iterator Rel) const
bool isSectionBSS(DataRefImpl Sec) const override
Expected< std::pair< size_t, std::vector< ChainedFixupsSegment > > > getChainedFixupsSegments() const
bool isSectionVirtual(DataRefImpl Sec) const override
bool getScatteredRelocationScattered(const MachO::any_relocation_info &RE) const
Expected< StringRef > getSymbolName(DataRefImpl Symb) const override
bool getPlainRelocationExternal(const MachO::any_relocation_info &RE) const
LoadCommandList::const_iterator load_command_iterator
symbol_iterator getSymbolByIndex(unsigned Index) const
static Triple getHostArch()
MachO::encryption_info_command getEncryptionInfoCommand(const LoadCommandInfo &L) const
const MachO::mach_header & getHeader() const
unsigned getAnyRelocationPCRel(const MachO::any_relocation_info &RE) const
iterator_range< bind_iterator > weakBindTable(Error &Err)
For use iterating over all weak bind table entries.
static bool isMachOPairedReloc(uint64_t RelocType, uint64_t Arch)
ArrayRef< uint8_t > getDyldInfoRebaseOpcodes() const
iterator_range< load_command_iterator > load_commands() const
unsigned getAnyRelocationLength(const MachO::any_relocation_info &RE) const
MachO::symtab_command getSymtabLoadCommand() const
Triple getArchTriple(const char **McpuDefault=nullptr) const
MachO::uuid_command getUuidCommand(const LoadCommandInfo &L) const
unsigned getPlainRelocationSymbolNum(const MachO::any_relocation_info &RE) const
ArrayRef< uint8_t > getUuid() const
bool is64Bit() const override
MachO::version_min_command getVersionMinLoadCommand(const LoadCommandInfo &L) const
StringRef mapDebugSectionName(StringRef Name) const override
Maps a debug section name to a standard DWARF section name.
MachO::dylinker_command getDylinkerCommand(const LoadCommandInfo &L) const
uint64_t getRelocationType(DataRefImpl Rel) const override
relocation_iterator extrel_begin() const
void moveRelocationNext(DataRefImpl &Rel) const override
MachO::any_relocation_info getRelocation(DataRefImpl Rel) const
basic_symbol_iterator symbol_end() const override
MachO::data_in_code_entry getDataInCodeTableEntry(uint32_t DataOffset, unsigned Index) const
MachO::data_in_code_entry getDice(DataRefImpl Rel) const
bool isSectionStripped(DataRefImpl Sec) const override
When dsymutil generates the companion file, it strips all unnecessary sections (e....
uint64_t getSectionIndex(DataRefImpl Sec) const override
iterator_range< fixup_iterator > fixupTable(Error &Err)
For iterating over all chained fixups.
void ReadULEB128s(uint64_t Index, SmallVectorImpl< uint64_t > &Out) const
iterator_range< bind_iterator > lazyBindTable(Error &Err)
For use iterating over all lazy bind table entries.
load_command_iterator end_load_commands() const
ArrayRef< uint8_t > getDyldInfoBindOpcodes() const
Expected< SymbolRef::Type > getSymbolType(DataRefImpl Symb) const override
uint64_t getSectionAddress(DataRefImpl Sec) const override
Expected< StringRef > getSectionName(DataRefImpl Sec) const override
uint8_t getRelocationLength(DataRefImpl Rel) const
llvm::binaryformat::Swift5ReflectionSectionKind mapReflectionSectionNameToEnumValue(StringRef SectionName) const override
ArrayRef< uint8_t > getDyldInfoWeakBindOpcodes() const
static bool isValidArch(StringRef ArchFlag)
bool isSectionText(DataRefImpl Sec) const override
bool isSectionCompressed(DataRefImpl Sec) const override
static ArrayRef< StringRef > getValidArchs()
bool isSectionBitcode(DataRefImpl Sec) const override
bool isRelocationScattered(const MachO::any_relocation_info &RE) const
relocation_iterator locrel_begin() const
Expected< std::optional< MachO::dyld_chained_fixups_header > > getChainedFixupsHeader() const
If the optional is std::nullopt, no header was found, but the object was well-formed.
uint32_t getSymbolAlignment(DataRefImpl Symb) const override
MachO::source_version_command getSourceVersionCommand(const LoadCommandInfo &L) const
unsigned getAnyRelocationAddress(const MachO::any_relocation_info &RE) const
StringRef getStringTableData() const
void getRelocationTypeName(DataRefImpl Rel, SmallVectorImpl< char > &Result) const override
ArrayRef< char > getSectionRawName(DataRefImpl Sec) const
uint64_t getNValue(DataRefImpl Sym) const
ArrayRef< uint8_t > getSegmentContents(StringRef SegmentName) const
Return the raw contents of an entire segment.
section_iterator getRelocationSection(DataRefImpl Rel) const
unsigned getSectionID(SectionRef Sec) const
MachO::linkedit_data_command getLinkeditDataLoadCommand(const LoadCommandInfo &L) const
ArrayRef< uint8_t > getSectionContents(uint64_t Offset, uint64_t Size) const
Expected< uint64_t > getSymbolAddress(DataRefImpl Symb) const override
MachO::dylib_command getDylibIDLoadCommand(const LoadCommandInfo &L) const
uint32_t getIndirectSymbolTableEntry(const MachO::dysymtab_command &DLC, unsigned Index) const
uint64_t getSectionSize(DataRefImpl Sec) const override
relocation_iterator extrel_end() const
static StringRef guessLibraryShortName(StringRef Name, bool &isFramework, StringRef &Suffix)
relocation_iterator locrel_end() const
std::error_code getLibraryShortNameByIndex(unsigned Index, StringRef &) const
MachORebaseEntry encapsulates the current state in the decompression of rebasing opcodes.
LLVM_ABI int32_t segmentIndex() const
LLVM_ABI StringRef segmentName() const
LLVM_ABI MachORebaseEntry(Error *Err, const MachOObjectFile *O, ArrayRef< uint8_t > opcodes, bool is64Bit)
LLVM_ABI bool operator==(const MachORebaseEntry &) const
LLVM_ABI uint64_t address() const
LLVM_ABI StringRef sectionName() const
friend class MachOObjectFile
LLVM_ABI uint64_t segmentOffset() const
LLVM_ABI StringRef typeName() const
This class is the base class for all object file types.
friend class RelocationRef
static Expected< std::unique_ptr< MachOObjectFile > > createMachOObjectFile(MemoryBufferRef Object, uint32_t UniversalCputype=0, uint32_t UniversalIndex=0, size_t MachOFilesetEntryOffset=0)
Create a MachOObjectFile instance from a given buffer.
section_iterator_range sections() const
symbol_iterator_range symbols() const
Expected< uint64_t > getSymbolValue(DataRefImpl Symb) const
DataRefImpl getRawDataRefImpl() const
This is a value type class that represents a single section in the list of sections in the object fil...
DataRefImpl getRawDataRefImpl() const
bool isData() const
Whether this section contains data, not instructions.
bool isBSS() const
Whether this section contains BSS uninitialized data.
directory_iterator - Iterates through the entries in path.
Represents the result of a call to sys::fs::status().
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
@ C
The default llvm calling convention, compatible with C.
const uint32_t x86_FLOAT_STATE_COUNT
@ DYLD_CHAINED_IMPORT_ADDEND
@ DYLD_CHAINED_IMPORT_ADDEND64
const uint32_t ARM_THREAD_STATE64_COUNT
@ EXPORT_SYMBOL_FLAGS_KIND_ABSOLUTE
@ EXPORT_SYMBOL_FLAGS_KIND_THREAD_LOCAL
@ EXPORT_SYMBOL_FLAGS_KIND_REGULAR
@ BIND_TYPE_TEXT_ABSOLUTE32
const uint32_t x86_EXCEPTION_STATE_COUNT
@ EXPORT_SYMBOL_FLAGS_REEXPORT
@ EXPORT_SYMBOL_FLAGS_KIND_MASK
@ EXPORT_SYMBOL_FLAGS_STUB_AND_RESOLVER
@ REBASE_TYPE_TEXT_ABSOLUTE32
@ REBASE_TYPE_TEXT_PCREL32
@ S_GB_ZEROFILL
S_GB_ZEROFILL - Zero fill on demand section (that can be larger than 4 gigabytes).
@ S_THREAD_LOCAL_ZEROFILL
S_THREAD_LOCAL_ZEROFILL - Thread local zerofill section.
@ S_ZEROFILL
S_ZEROFILL - Zero fill on demand section.
@ BIND_SPECIAL_DYLIB_WEAK_LOOKUP
@ BIND_SPECIAL_DYLIB_MAIN_EXECUTABLE
@ BIND_SPECIAL_DYLIB_FLAT_LOOKUP
@ DYLD_CHAINED_PTR_START_NONE
uint8_t GET_COMM_ALIGN(uint16_t n_desc)
void swapStruct(fat_header &mh)
const uint32_t x86_THREAD_STATE32_COUNT
@ BIND_SYMBOL_FLAGS_WEAK_IMPORT
@ BIND_SYMBOL_FLAGS_NON_WEAK_DEFINITION
@ CPU_SUBTYPE_POWERPC_ALL
@ BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB
@ BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB
@ BIND_OPCODE_SET_ADDEND_SLEB
@ BIND_OPCODE_SET_DYLIB_ORDINAL_ULEB
@ BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM
@ BIND_OPCODE_ADD_ADDR_ULEB
@ BIND_OPCODE_DO_BIND_ADD_ADDR_IMM_SCALED
@ BIND_OPCODE_SET_DYLIB_SPECIAL_IMM
@ BIND_OPCODE_SET_TYPE_IMM
@ BIND_OPCODE_SET_DYLIB_ORDINAL_IMM
@ BIND_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB
@ DYLD_CHAINED_PTR_64_OFFSET
const uint32_t PPC_THREAD_STATE_COUNT
const uint32_t ARM_THREAD_STATE_COUNT
@ REBASE_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB
@ REBASE_OPCODE_DO_REBASE_IMM_TIMES
@ REBASE_OPCODE_DO_REBASE_ADD_ADDR_ULEB
@ REBASE_OPCODE_DO_REBASE_ULEB_TIMES_SKIPPING_ULEB
@ REBASE_OPCODE_DO_REBASE_ULEB_TIMES
@ REBASE_OPCODE_ADD_ADDR_ULEB
@ REBASE_OPCODE_SET_TYPE_IMM
@ REBASE_OPCODE_ADD_ADDR_IMM_SCALED
const uint32_t x86_THREAD_STATE_COUNT
@ CPU_SUBTYPE_ARM64_32_V8
@ GENERIC_RELOC_LOCAL_SECTDIFF
@ ARM_RELOC_LOCAL_SECTDIFF
@ ARM_RELOC_HALF_SECTDIFF
@ X86_64_RELOC_SUBTRACTOR
uint16_t GET_LIBRARY_ORDINAL(uint16_t n_desc)
@ S_ATTR_PURE_INSTRUCTIONS
S_ATTR_PURE_INSTRUCTIONS - Section contains only true machine instructions.
const uint32_t x86_EXCEPTION_STATE64_COUNT
const uint32_t x86_THREAD_STATE64_COUNT
constexpr size_t SymbolTableEntrySize
content_iterator< SectionRef > section_iterator
Error createError(const Twine &Err)
content_iterator< MachOBindEntry > bind_iterator
content_iterator< MachOChainedFixupEntry > fixup_iterator
content_iterator< MachORebaseEntry > rebase_iterator
content_iterator< BasicSymbolRef > basic_symbol_iterator
content_iterator< ExportEntry > export_iterator
content_iterator< RelocationRef > relocation_iterator
content_iterator< DiceRef > dice_iterator
LLVM_ABI std::error_code status(const Twine &path, file_status &result, bool follow=true)
Get file status as if by POSIX stat().
LLVM_ABI bool is_directory(const basic_file_status &status)
Does status represent a directory?
LLVM_ABI bool remove_dots(SmallVectorImpl< char > &path, bool remove_dot_dot=false, Style style=Style::native)
In-place remove any '.
LLVM_ABI void append(SmallVectorImpl< char > &path, const Twine &a, const Twine &b="", const Twine &c="", const Twine &d="")
Append to path.
LLVM_ABI StringRef extension(StringRef path LLVM_LIFETIME_BOUND, Style style=Style::native)
Get extension.
constexpr bool IsLittleEndianHost
LLVM_ABI std::string getDefaultTargetTriple()
getDefaultTargetTriple() - Return the default target triple the compiler has been configured to produ...
void swapByteOrder(T &Value)
This is an optimization pass for GlobalISel generic memory operations.
Error createFileError(const Twine &F, Error E)
Concatenate a source file path and/or name with an Error.
ArrayRef< CharT > arrayRefFromStringRef(StringRef Input)
Construct a string ref from an array ref of unsigned chars.
iterator_range< T > make_range(T x, T y)
Convenience function for iterating over sub-ranges.
uint64_t decodeULEB128(const uint8_t *p, unsigned *n=nullptr, const uint8_t *end=nullptr, const char **error=nullptr)
Utility function to decode a ULEB128 value.
int64_t decodeSLEB128(const uint8_t *p, unsigned *n=nullptr, const uint8_t *end=nullptr, const char **error=nullptr)
Utility function to decode a SLEB128 value.
Error createStringError(std::error_code EC, char const *Fmt, const Ts &... Vals)
Create formatted StringError object.
@ no_such_file_or_directory
LLVM_ABI raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
LLVM_ABI void report_fatal_error(Error Err, bool gen_crash_diag=true)
FunctionAddr VTableAddr Count
format_object< Ts... > format(const char *Fmt, const Ts &... Vals)
These are helper functions used to produce formatted output.
Error make_error(ArgTs &&... Args)
Make a Error instance representing failure using the given error info type.
void cantFail(Error Err, const char *Msg=nullptr)
Report a fatal error if Err is a failure value.
To bit_cast(const From &from) noexcept
FunctionAddr VTableAddr Next
auto count(R &&Range, const E &Element)
Wrapper function around std::count to count the number of times an element Element occurs in the give...
ArrayRef(const T &OneElt) -> ArrayRef< T >
constexpr int32_t SignExtend32(uint32_t X)
Sign-extend the number in the bottom B bits of X to a 32-bit integer.
bool is_contained(R &&Range, const E &Element)
Returns true if Element is found in Range.
LLVM_ABI Error errorCodeToError(std::error_code EC)
Helper for converting an std::error_code to a Error.
void consumeError(Error Err)
Consume a Error without doing anything.
This struct is a compact representation of a valid (non-zero power of two) alignment.
dyld_chained_starts_in_image is embedded in LC_DYLD_CHAINED_FIXUPS payload.
uint16_t page_count
Length of the page_start array.
uint16_t page_size
Page size in bytes (0x1000 or 0x4000)
uint16_t pointer_format
DYLD_CHAINED_PTR*.
uint32_t size
Size of this, including chain_starts entries.
ChainedFixupTarget holds all the information about an external symbol necessary to bind this binary t...
MachO::dyld_chained_starts_in_segment Header
std::vector< uint16_t > PageStarts
struct llvm::object::DataRefImpl::@005117267142344013370254144343227032034000327225 d