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");
2844 *McpuDefault =
"cortex-m23";
2846 *ArchFlag =
"armv8m.base";
2847 return Triple(
"thumbv8m-apple-darwin");
2850 *McpuDefault =
"cortex-m33";
2852 *ArchFlag =
"armv8m.main";
2853 return Triple(
"thumbv8m-apple-darwin");
2856 *McpuDefault =
"cortex-m52";
2858 *ArchFlag =
"armv8.1m.main";
2859 return Triple(
"thumbv8m-apple-darwin");
2867 *McpuDefault =
"cyclone";
2869 *ArchFlag =
"arm64";
2870 return Triple(
"arm64-apple-darwin");
2873 *McpuDefault =
"apple-a12";
2875 *ArchFlag =
"arm64e";
2876 return Triple(
"arm64e-apple-darwin");
2884 *McpuDefault =
"cyclone";
2886 *ArchFlag =
"arm64_32";
2887 return Triple(
"arm64_32-apple-darwin");
2896 return Triple(
"ppc-apple-darwin");
2904 *ArchFlag =
"ppc64";
2905 return Triple(
"ppc64-apple-darwin");
2913 *ArchFlag =
"riscv32";
2914 return Triple(
"riscv32-apple-macho");
2933 static const std::array<StringRef, 21> ValidArchs = {{
2934 "i386",
"x86_64",
"x86_64h",
"armv4t",
"arm",
2935 "armv5e",
"armv6",
"armv6m",
"armv7",
"armv7em",
2936 "armv7k",
"armv7m",
"armv7s",
"armv8m.base",
"armv8m.main",
2937 "armv8.1m.main",
"arm64",
"arm64e",
"arm64_32",
"ppc",
2966 if (!DataInCodeLoadCmd)
2970 DRI.
p =
reinterpret_cast<uintptr_t
>(
getPtr(*
this, DicLC.
dataoff));
2976 if (!DataInCodeLoadCmd)
2988void ExportEntry::moveToFirst() {
2993 pushDownUntilBottom();
2996void ExportEntry::moveToEnd() {
3003 if (Done ||
Other.Done)
3004 return (Done ==
Other.Done);
3006 if (Stack.size() !=
Other.Stack.size())
3009 if (!CumulativeString.equals(
Other.CumulativeString))
3012 for (
unsigned i=0; i < Stack.size(); ++i) {
3013 if (Stack[i].Start !=
Other.Stack[i].Start)
3023 if (Ptr > Trie.
end())
3029 return CumulativeString;
3033 return Stack.back().Flags;
3037 return Stack.back().Address;
3041 return Stack.back().Other;
3045 const char* ImportName = Stack.
back().ImportName;
3052 return Stack.back().Start - Trie.begin();
3055ExportEntry::NodeState::NodeState(
const uint8_t *Ptr)
3056 : Start(Ptr), Current(Ptr) {}
3058void ExportEntry::pushNode(
uint64_t offset) {
3061 NodeState State(Ptr);
3062 const char *
error =
nullptr;
3066 " in export trie data at node: 0x" +
3071 State.IsExportNode = (ExportInfoSize != 0);
3072 const uint8_t* Children = State.Current + ExportInfoSize;
3073 if (Children > Trie.
end()) {
3077 " too big and extends past end of trie data");
3081 if (State.IsExportNode) {
3082 const uint8_t *ExportStart = State.Current;
3083 State.Flags = readULEB128(State.Current, &
error);
3086 " in export trie data at node: 0x" +
3092 if (State.Flags != 0 &&
3097 "unsupported exported symbol kind: " + Twine((
int)
Kind) +
3105 State.Other = readULEB128(State.Current, &
error);
3108 " in export trie data at node: 0x" +
3116 if ((int64_t)State.Other > 0 && State.Other > O->getLibraryCount()) {
3118 "bad library ordinal: " + Twine((
int)State.Other) +
" (max " +
3119 Twine((
int)O->getLibraryCount()) +
3125 State.ImportName =
reinterpret_cast<const char*
>(State.Current);
3126 if (*State.ImportName ==
'\0') {
3129 const uint8_t *End = State.Current + 1;
3130 if (End >= Trie.end()) {
3131 *E =
malformedError(
"import name of re-export in export trie data at "
3134 " starts past end of trie data");
3138 while(*End !=
'\0' && End < Trie.end())
3141 *E =
malformedError(
"import name of re-export in export trie data at "
3144 " extends past end of trie data");
3148 State.Current = End + 1;
3151 State.Address = readULEB128(State.Current, &
error);
3154 " in export trie data at node: 0x" +
3160 State.Other = readULEB128(State.Current, &
error);
3163 " in export trie data at node: 0x" +
3170 if (ExportStart + ExportInfoSize < State.Current) {
3172 "inconsistent export info size: 0x" +
3181 if (State.ChildCount != 0 && Children + 1 >= Trie.end()) {
3182 *E =
malformedError(
"byte for count of children in export trie data at "
3185 " extends past end of trie data");
3190 State.NextChildIndex = 0;
3191 State.ParentStringLength = CumulativeString.size();
3192 Stack.push_back(State);
3195void ExportEntry::pushDownUntilBottom() {
3196 ErrorAsOutParameter ErrAsOutParam(E);
3197 const char *
error =
nullptr;
3198 while (Stack.back().NextChildIndex < Stack.back().ChildCount) {
3199 NodeState &Top = Stack.back();
3200 CumulativeString.resize(Top.ParentStringLength);
3201 for (;*Top.Current != 0 && Top.Current < Trie.end(); Top.Current++) {
3202 char C = *Top.Current;
3203 CumulativeString.push_back(
C);
3205 if (Top.Current >= Trie.end()) {
3206 *E =
malformedError(
"edge sub-string in export trie data at node: 0x" +
3208 " for child #" + Twine((
int)Top.NextChildIndex) +
3209 " extends past end of trie data");
3214 uint64_t childNodeIndex = readULEB128(Top.Current, &
error);
3217 " in export trie data at node: 0x" +
3222 for (
const NodeState &node :
nodes()) {
3223 if (node.Start == Trie.begin() + childNodeIndex){
3224 *E =
malformedError(
"loop in children in export trie data at node: 0x" +
3226 " back to node: 0x" +
3232 Top.NextChildIndex += 1;
3233 pushNode(childNodeIndex);
3237 if (!Stack.back().IsExportNode) {
3238 *E =
malformedError(
"node is not an export node in export trie data at "
3262 assert(!Stack.empty() &&
"ExportEntry::moveNext() with empty node stack");
3263 if (!Stack.back().IsExportNode) {
3264 *E =
malformedError(
"node is not an export node in export trie data at "
3272 while (!Stack.empty()) {
3273 NodeState &Top = Stack.back();
3274 if (Top.NextChildIndex < Top.ChildCount) {
3275 pushDownUntilBottom();
3279 if (Top.IsExportNode) {
3281 CumulativeString.resize(Top.ParentStringLength);
3292 const MachOObjectFile *O) {
3297 Start.moveToFirst();
3307 if (DyldInfoLoadCmd)
3309 else if (DyldExportsTrieLoadCmd)
3312 return exports(Err, Trie,
this);
3319 for (
const auto &Command :
O->load_commands()) {
3320 if (Command.C.cmd == MachO::LC_SEGMENT) {
3321 MachO::segment_command SLC = O->getSegmentLoadCommand(Command);
3322 if (StringRef(SLC.segname) ==
"__TEXT") {
3323 TextAddress = SLC.vmaddr;
3326 }
else if (Command.C.cmd == MachO::LC_SEGMENT_64) {
3327 MachO::segment_command_64 SLC_64 = O->getSegment64LoadCommand(Command);
3328 if (StringRef(SLC_64.segname) ==
"__TEXT") {
3329 TextAddress = SLC_64.vmaddr;
3389 if (
auto FixupTargetsOrErr =
O->getDyldChainedFixupTargets()) {
3390 FixupTargets = *FixupTargetsOrErr;
3392 *
E = FixupTargetsOrErr.takeError();
3396 if (
auto SegmentsOrErr =
O->getChainedFixupsSegments()) {
3397 Segments = std::move(SegmentsOrErr->second);
3399 *
E = SegmentsOrErr.takeError();
3404void MachOChainedFixupEntry::findNextPageWithFixups() {
3405 auto FindInSegment = [
this]() {
3407 while (PageIndex < SegInfo.
PageStarts.size() &&
3410 return PageIndex < SegInfo.
PageStarts.size();
3413 while (InfoSegIndex < Segments.size()) {
3414 if (FindInSegment()) {
3415 PageOffset = Segments[InfoSegIndex].PageStarts[PageIndex];
3427 if (Segments.empty()) {
3435 findNextPageWithFixups();
3446 if (InfoSegIndex == Segments.size()) {
3460 " has unsupported chained fixup pointer_format " +
3461 Twine(PointerFormat));
3475 " extends past segment's end");
3486 assert(
O->isLittleEndian() &&
"big-endian object should have been rejected "
3487 "by getDyldChainedFixupTargets()");
3494 bool IsBind =
Field(63, 1);
3501 if (ImportOrdinal >= FixupTargets.size()) {
3504 " has out-of range import ordinal " +
3505 Twine(ImportOrdinal));
3512 Addend = InlineAddend ? InlineAddend :
Target.addend();
3526 PageOffset += 4 *
Next;
3529 findNextPageWithFixups();
3539 return InfoSegIndex ==
Other.InfoSegIndex && PageIndex ==
Other.PageIndex &&
3540 PageOffset ==
Other.PageOffset;
3545 : E(E), O(O), Opcodes(Bytes), Ptr(Bytes.begin()),
3546 PointerSize(
is64Bit ? 8 : 4) {}
3548void MachORebaseEntry::moveToFirst() {
3549 Ptr = Opcodes.
begin();
3553void MachORebaseEntry::moveToEnd() {
3554 Ptr = Opcodes.
end();
3555 RemainingLoopCount = 0;
3562 SegmentOffset += AdvanceAmount;
3563 if (RemainingLoopCount) {
3564 --RemainingLoopCount;
3573 if (Ptr == Opcodes.end()) {
3579 const uint8_t *OpcodeStart = Ptr;
3584 const char *
error =
nullptr;
3593 RebaseType = ImmValue;
3595 *E =
malformedError(
"for REBASE_OPCODE_SET_TYPE_IMM bad bind type: " +
3596 Twine((
int)RebaseType) +
" for opcode at: 0x" +
3603 dbgs() <<
"REBASE_OPCODE_SET_TYPE_IMM: "
3604 <<
"RebaseType=" << (
int) RebaseType <<
"\n");
3607 SegmentIndex = ImmValue;
3608 SegmentOffset = readULEB128(&
error);
3610 *E =
malformedError(
"for REBASE_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB " +
3616 error = O->RebaseEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset,
3619 *E =
malformedError(
"for REBASE_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB " +
3627 dbgs() <<
"REBASE_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB: "
3628 <<
"SegmentIndex=" << SegmentIndex <<
", "
3629 <<
format(
"SegmentOffset=0x%06X", SegmentOffset)
3633 SegmentOffset += readULEB128(&
error);
3636 " for opcode at: 0x" +
3641 error = O->RebaseEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset,
3645 " for opcode at: 0x" +
3651 dbgs() <<
"REBASE_OPCODE_ADD_ADDR_ULEB: "
3652 <<
format(
"SegmentOffset=0x%06X",
3653 SegmentOffset) <<
"\n");
3656 SegmentOffset += ImmValue * PointerSize;
3657 error = O->RebaseEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset,
3667 dbgs() <<
"REBASE_OPCODE_ADD_ADDR_IMM_SCALED: "
3668 <<
format(
"SegmentOffset=0x%06X",
3669 SegmentOffset) <<
"\n");
3672 AdvanceAmount = PointerSize;
3676 RemainingLoopCount = ImmValue - 1;
3678 RemainingLoopCount = 0;
3679 error = O->RebaseEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset,
3680 PointerSize,
Count, Skip);
3690 dbgs() <<
"REBASE_OPCODE_DO_REBASE_IMM_TIMES: "
3691 <<
format(
"SegmentOffset=0x%06X", SegmentOffset)
3692 <<
", AdvanceAmount=" << AdvanceAmount
3693 <<
", RemainingLoopCount=" << RemainingLoopCount
3697 AdvanceAmount = PointerSize;
3708 RemainingLoopCount =
Count - 1;
3710 RemainingLoopCount = 0;
3711 error = O->RebaseEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset,
3712 PointerSize,
Count, Skip);
3722 dbgs() <<
"REBASE_OPCODE_DO_REBASE_ULEB_TIMES: "
3723 <<
format(
"SegmentOffset=0x%06X", SegmentOffset)
3724 <<
", AdvanceAmount=" << AdvanceAmount
3725 <<
", RemainingLoopCount=" << RemainingLoopCount
3729 Skip = readULEB128(&
error);
3731 *E =
malformedError(
"for REBASE_OPCODE_DO_REBASE_ADD_ADDR_ULEB " +
3737 AdvanceAmount = Skip + PointerSize;
3739 RemainingLoopCount = 0;
3740 error = O->RebaseEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset,
3741 PointerSize,
Count, Skip);
3743 *E =
malformedError(
"for REBASE_OPCODE_DO_REBASE_ADD_ADDR_ULEB " +
3751 dbgs() <<
"REBASE_OPCODE_DO_REBASE_ADD_ADDR_ULEB: "
3752 <<
format(
"SegmentOffset=0x%06X", SegmentOffset)
3753 <<
", AdvanceAmount=" << AdvanceAmount
3754 <<
", RemainingLoopCount=" << RemainingLoopCount
3760 *E =
malformedError(
"for REBASE_OPCODE_DO_REBASE_ULEB_TIMES_SKIPPING_"
3768 RemainingLoopCount =
Count - 1;
3770 RemainingLoopCount = 0;
3771 Skip = readULEB128(&
error);
3773 *E =
malformedError(
"for REBASE_OPCODE_DO_REBASE_ULEB_TIMES_SKIPPING_"
3780 AdvanceAmount = Skip + PointerSize;
3782 error = O->RebaseEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset,
3783 PointerSize,
Count, Skip);
3785 *E =
malformedError(
"for REBASE_OPCODE_DO_REBASE_ULEB_TIMES_SKIPPING_"
3794 dbgs() <<
"REBASE_OPCODE_DO_REBASE_ULEB_TIMES_SKIPPING_ULEB: "
3795 <<
format(
"SegmentOffset=0x%06X", SegmentOffset)
3796 <<
", AdvanceAmount=" << AdvanceAmount
3797 <<
", RemainingLoopCount=" << RemainingLoopCount
3814 if (Ptr > Opcodes.
end())
3815 Ptr = Opcodes.
end();
3824 switch (RebaseType) {
3828 return "text abs32";
3830 return "text rel32";
3838 return O->BindRebaseSegmentName(SegmentIndex);
3844 return O->BindRebaseSectionName(SegmentIndex, SegmentOffset);
3850 return O->BindRebaseAddress(SegmentIndex, SegmentOffset);
3854#ifdef EXPENSIVE_CHECKS
3855 assert(Opcodes ==
Other.Opcodes &&
"compare iterators of different files");
3857 assert(Opcodes.data() ==
Other.Opcodes.data() &&
"compare iterators of different files");
3859 return (Ptr ==
Other.Ptr) &&
3860 (RemainingLoopCount ==
Other.RemainingLoopCount) &&
3861 (Done ==
Other.Done);
3867 if (O->BindRebaseSectionTable ==
nullptr)
3868 O->BindRebaseSectionTable = std::make_unique<BindRebaseSegInfo>(O);
3870 Start.moveToFirst();
3884 : E(E), O(O), Opcodes(Bytes), Ptr(Bytes.begin()),
3885 PointerSize(
is64Bit ? 8 : 4), TableKind(BK) {}
3887void MachOBindEntry::moveToFirst() {
3888 Ptr = Opcodes.
begin();
3892void MachOBindEntry::moveToEnd() {
3893 Ptr = Opcodes.
end();
3894 RemainingLoopCount = 0;
3901 SegmentOffset += AdvanceAmount;
3902 if (RemainingLoopCount) {
3903 --RemainingLoopCount;
3912 if (Ptr == Opcodes.end()) {
3918 const uint8_t *OpcodeStart = Ptr;
3922 int8_t SignExtended;
3925 const char *
error =
nullptr;
3931 bool NotLastEntry =
false;
3932 for (
const uint8_t *
P = Ptr;
P < Opcodes.end(); ++
P) {
3934 NotLastEntry =
true;
3946 *E =
malformedError(
"BIND_OPCODE_SET_DYLIB_ORDINAL_IMM not allowed in "
3947 "weak bind table for opcode at: 0x" +
3953 LibraryOrdinalSet =
true;
3954 if (ImmValue > O->getLibraryCount()) {
3955 *E =
malformedError(
"for BIND_OPCODE_SET_DYLIB_ORDINAL_ULEB bad "
3956 "library ordinal: " +
3957 Twine((
int)ImmValue) +
" (max " +
3958 Twine((
int)O->getLibraryCount()) +
3959 ") for opcode at: 0x" +
3966 dbgs() <<
"BIND_OPCODE_SET_DYLIB_ORDINAL_IMM: "
3967 <<
"Ordinal=" << Ordinal <<
"\n");
3971 *E =
malformedError(
"BIND_OPCODE_SET_DYLIB_ORDINAL_ULEB not allowed in "
3972 "weak bind table for opcode at: 0x" +
3977 Ordinal = readULEB128(&
error);
3978 LibraryOrdinalSet =
true;
3986 if (Ordinal > (
int)O->getLibraryCount()) {
3987 *E =
malformedError(
"for BIND_OPCODE_SET_DYLIB_ORDINAL_ULEB bad "
3988 "library ordinal: " +
3989 Twine((
int)Ordinal) +
" (max " +
3990 Twine((
int)O->getLibraryCount()) +
3991 ") for opcode at: 0x" +
3998 dbgs() <<
"BIND_OPCODE_SET_DYLIB_ORDINAL_ULEB: "
3999 <<
"Ordinal=" << Ordinal <<
"\n");
4003 *E =
malformedError(
"BIND_OPCODE_SET_DYLIB_SPECIAL_IMM not allowed in "
4004 "weak bind table for opcode at: 0x" +
4011 Ordinal = SignExtended;
4013 *E =
malformedError(
"for BIND_OPCODE_SET_DYLIB_SPECIAL_IMM unknown "
4014 "special ordinal: " +
4015 Twine((
int)Ordinal) +
" for opcode at: 0x" +
4022 LibraryOrdinalSet =
true;
4025 dbgs() <<
"BIND_OPCODE_SET_DYLIB_SPECIAL_IMM: "
4026 <<
"Ordinal=" << Ordinal <<
"\n");
4031 while (*Ptr && (Ptr < Opcodes.end())) {
4034 if (Ptr == Opcodes.end()) {
4036 "for BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM "
4037 "symbol name extends past opcodes for opcode at: 0x" +
4042 SymbolName =
StringRef(
reinterpret_cast<const char*
>(SymStart),
4047 dbgs() <<
"BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM: "
4048 <<
"SymbolName=" << SymbolName <<
"\n");
4055 BindType = ImmValue;
4057 *E =
malformedError(
"for BIND_OPCODE_SET_TYPE_IMM bad bind type: " +
4058 Twine((
int)ImmValue) +
" for opcode at: 0x" +
4065 dbgs() <<
"BIND_OPCODE_SET_TYPE_IMM: "
4066 <<
"BindType=" << (
int)BindType <<
"\n");
4069 Addend = readSLEB128(&
error);
4072 " for opcode at: 0x" +
4079 dbgs() <<
"BIND_OPCODE_SET_ADDEND_SLEB: "
4080 <<
"Addend=" << Addend <<
"\n");
4083 SegmentIndex = ImmValue;
4084 SegmentOffset = readULEB128(&
error);
4086 *E =
malformedError(
"for BIND_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB " +
4092 error = O->BindEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset,
4095 *E =
malformedError(
"for BIND_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB " +
4103 dbgs() <<
"BIND_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB: "
4104 <<
"SegmentIndex=" << SegmentIndex <<
", "
4105 <<
format(
"SegmentOffset=0x%06X", SegmentOffset)
4109 SegmentOffset += readULEB128(&
error);
4112 " for opcode at: 0x" +
4117 error = O->BindEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset,
4121 " for opcode at: 0x" +
4127 dbgs() <<
"BIND_OPCODE_ADD_ADDR_ULEB: "
4128 <<
format(
"SegmentOffset=0x%06X",
4129 SegmentOffset) <<
"\n");
4132 AdvanceAmount = PointerSize;
4133 RemainingLoopCount = 0;
4134 error = O->BindEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset,
4138 " for opcode at: 0x" +
4145 "for BIND_OPCODE_DO_BIND missing preceding "
4146 "BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM for opcode at: 0x" +
4151 if (!LibraryOrdinalSet && TableKind !=
Kind::Weak) {
4154 "BIND_OPCODE_SET_DYLIB_ORDINAL_* for opcode at: 0x" +
4160 dbgs() <<
"BIND_OPCODE_DO_BIND: "
4161 <<
format(
"SegmentOffset=0x%06X",
4162 SegmentOffset) <<
"\n");
4166 *E =
malformedError(
"BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB not allowed in "
4167 "lazy bind table for opcode at: 0x" +
4172 error = O->BindEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset,
4183 "for BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB missing "
4184 "preceding BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM for opcode "
4190 if (!LibraryOrdinalSet && TableKind !=
Kind::Weak) {
4192 "for BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB missing "
4193 "preceding BIND_OPCODE_SET_DYLIB_ORDINAL_* for opcode at: 0x" +
4198 AdvanceAmount = readULEB128(&
error) + PointerSize;
4209 error = O->BindEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset +
4210 AdvanceAmount, PointerSize);
4212 *E =
malformedError(
"for BIND_OPCODE_ADD_ADDR_ULEB (after adding "
4219 RemainingLoopCount = 0;
4222 dbgs() <<
"BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB: "
4223 <<
format(
"SegmentOffset=0x%06X", SegmentOffset)
4224 <<
", AdvanceAmount=" << AdvanceAmount
4225 <<
", RemainingLoopCount=" << RemainingLoopCount
4230 *E =
malformedError(
"BIND_OPCODE_DO_BIND_ADD_ADDR_IMM_SCALED not "
4231 "allowed in lazy bind table for opcode at: 0x" +
4238 "for BIND_OPCODE_DO_BIND_ADD_ADDR_IMM_SCALED "
4239 "missing preceding BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM for "
4245 if (!LibraryOrdinalSet && TableKind !=
Kind::Weak) {
4247 "for BIND_OPCODE_DO_BIND_ADD_ADDR_IMM_SCALED "
4248 "missing preceding BIND_OPCODE_SET_DYLIB_ORDINAL_* for opcode "
4254 AdvanceAmount = ImmValue * PointerSize + PointerSize;
4255 RemainingLoopCount = 0;
4256 error = O->BindEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset +
4257 AdvanceAmount, PointerSize);
4259 *E =
malformedError(
"for BIND_OPCODE_DO_BIND_ADD_ADDR_IMM_SCALED " +
4267 <<
"BIND_OPCODE_DO_BIND_ADD_ADDR_IMM_SCALED: "
4268 <<
format(
"SegmentOffset=0x%06X", SegmentOffset) <<
"\n");
4272 *E =
malformedError(
"BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB not "
4273 "allowed in lazy bind table for opcode at: 0x" +
4280 RemainingLoopCount =
Count - 1;
4282 RemainingLoopCount = 0;
4284 *E =
malformedError(
"for BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB "
4291 Skip = readULEB128(&
error);
4292 AdvanceAmount = Skip + PointerSize;
4294 *E =
malformedError(
"for BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB "
4303 "for BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB "
4304 "missing preceding BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM for "
4310 if (!LibraryOrdinalSet && TableKind !=
Kind::Weak) {
4312 "for BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB "
4313 "missing preceding BIND_OPCODE_SET_DYLIB_ORDINAL_* for opcode "
4319 error = O->BindEntryCheckSegAndOffsets(SegmentIndex, SegmentOffset,
4320 PointerSize,
Count, Skip);
4323 malformedError(
"for BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB " +
4331 dbgs() <<
"BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB: "
4332 <<
format(
"SegmentOffset=0x%06X", SegmentOffset)
4333 <<
", AdvanceAmount=" << AdvanceAmount
4334 <<
", RemainingLoopCount=" << RemainingLoopCount
4351 if (Ptr > Opcodes.
end())
4352 Ptr = Opcodes.
end();
4356int64_t MachOBindEntry::readSLEB128(
const char **
error) {
4360 if (Ptr > Opcodes.
end())
4361 Ptr = Opcodes.
end();
4374 return "text abs32";
4376 return "text rel32";
4392 return O->BindRebaseSegmentName(SegmentIndex);
4398 return O->BindRebaseSectionName(SegmentIndex, SegmentOffset);
4404 return O->BindRebaseAddress(SegmentIndex, SegmentOffset);
4408#ifdef EXPENSIVE_CHECKS
4409 assert(Opcodes ==
Other.Opcodes &&
"compare iterators of different files");
4411 assert(Opcodes.data() ==
Other.Opcodes.data() &&
"compare iterators of different files");
4413 return (Ptr ==
Other.Ptr) &&
4414 (RemainingLoopCount ==
Other.RemainingLoopCount) &&
4415 (Done ==
Other.Done);
4420 uint32_t CurSegIndex = Obj->hasPageZeroSegment() ? 1 : 0;
4423 for (
const SectionRef &Section : Obj->sections()) {
4429 Info.SectionName = *NameOrErr;
4430 Info.Address = Section.getAddress();
4431 Info.Size = Section.getSize();
4433 Obj->getSectionFinalSegmentName(Section.getRawDataRefImpl());
4434 if (Info.SegmentName != CurSegName) {
4436 CurSegName = Info.SegmentName;
4437 CurSegAddress = Info.Address;
4439 Info.SegmentIndex = CurSegIndex - 1;
4440 Info.OffsetInSegment = Info.Address - CurSegAddress;
4441 Info.SegmentStartAddress = CurSegAddress;
4442 Sections.push_back(Info);
4444 MaxSegIndex = CurSegIndex;
4460 return "missing preceding *_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB";
4461 if (SegIndex >= MaxSegIndex)
4462 return "bad segIndex (too large)";
4464 uint64_t Start = SegOffset + i * (PointerSize + Skip);
4465 uint64_t End = Start + PointerSize;
4467 for (
const SectionInfo &
SI : Sections) {
4468 if (
SI.SegmentIndex != SegIndex)
4470 if ((
SI.OffsetInSegment<=Start) && (Start<(
SI.OffsetInSegment+
SI.Size))) {
4471 if (End <=
SI.OffsetInSegment +
SI.Size) {
4476 return "bad offset, extends beyond section boundary";
4480 return "bad offset, not in section";
4488 for (
const SectionInfo &
SI : Sections) {
4489 if (
SI.SegmentIndex == SegIndex)
4490 return SI.SegmentName;
4497const BindRebaseSegInfo::SectionInfo &BindRebaseSegInfo::findSection(
4498 int32_t SegIndex,
uint64_t SegOffset) {
4499 for (
const SectionInfo &
SI : Sections) {
4500 if (
SI.SegmentIndex != SegIndex)
4502 if (
SI.OffsetInSegment > SegOffset)
4504 if (SegOffset >= (
SI.OffsetInSegment +
SI.Size))
4515 return findSection(SegIndex, SegOffset).SectionName;
4521 const SectionInfo &
SI = findSection(SegIndex, OffsetInSeg);
4522 return SI.SegmentStartAddress + OffsetInSeg;
4529 if (O->BindRebaseSectionTable ==
nullptr)
4530 O->BindRebaseSectionTable = std::make_unique<BindRebaseSegInfo>(O);
4532 Start.moveToFirst();
4556 if (BindRebaseSectionTable ==
nullptr)
4557 BindRebaseSectionTable = std::make_unique<BindRebaseSegInfo>(
this);
4560 Start.moveToFirst();
4570 return LoadCommands.begin();
4575 return LoadCommands.end();
4591 assert(Sec.
d.
a < Sections.size() &&
"Should have detected this earlier");
4592 const section_base *
Base =
4593 reinterpret_cast<const section_base *
>(Sections[Sec.
d.
a]);
4599 assert(Sec.
d.
a < Sections.size() &&
"Should have detected this earlier");
4600 const section_base *
Base =
4601 reinterpret_cast<const section_base *
>(Sections[Sec.
d.
a]);
4623 return (RE.
r_word1 >> 27) & 1;
4639 return (RE.
r_word0 >> 24) & 0xf;
4677 if (SecNum ==
MachO::R_ABS || SecNum > Sections.size())
4680 DRI.
d.
a = SecNum - 1;
4685 assert(DRI.
d.
a < Sections.size() &&
"Should have detected this earlier");
4690 assert(DRI.
d.
a < Sections.size() &&
"Should have detected this earlier");
4695 unsigned Index)
const {
4701 unsigned Index)
const {
4708 const char *
P =
reinterpret_cast<const char *
>(DRI.
p);
4714 const char *
P =
reinterpret_cast<const char *
>(DRI.
p);
4859 Offset = DysymtabLoadCmd.extreloff;
4861 Offset = DysymtabLoadCmd.locreloff;
4867 *
this,
reinterpret_cast<const char *
>(
P));
4872 const char *
P =
reinterpret_cast<const char *
>(Rel.
p);
4887 unsigned Index)
const {
4894 unsigned Index)
const {
4905 Cmd.
cmd = MachO::LC_SYMTAB;
4915 if (DysymtabLoadCmd)
4920 Cmd.
cmd = MachO::LC_DYSYMTAB;
4945 if (DataInCodeLoadCmd)
4950 Cmd.
cmd = MachO::LC_DATA_IN_CODE;
4959 if (LinkOptHintsLoadCmd)
4965 Cmd.
cmd = MachO::LC_LINKER_OPTIMIZATION_HINT;
4973 if (!DyldInfoLoadCmd)
4976 auto DyldInfoOrErr =
4987 if (!DyldInfoLoadCmd)
4990 auto DyldInfoOrErr =
5001 if (!DyldInfoLoadCmd)
5004 auto DyldInfoOrErr =
5015 if (!DyldInfoLoadCmd)
5018 auto DyldInfoOrErr =
5029 if (!DyldInfoLoadCmd)
5032 auto DyldInfoOrErr =
5045 if (!DyldChainedFixupsLoadCmd)
5046 return std::nullopt;
5048 *
this, DyldChainedFixupsLoadCmd);
5049 if (!DyldChainedFixupsOrErr)
5050 return DyldChainedFixupsOrErr.takeError();
5052 *DyldChainedFixupsOrErr;
5056 if (!DyldChainedFixups.
dataoff)
5057 return std::nullopt;
5058 return DyldChainedFixups;
5065 return CFOrErr.takeError();
5066 if (!CFOrErr->has_value())
5067 return std::nullopt;
5075 const char *CFHeaderPtr =
getPtr(*
this, CFHeaderOffset);
5076 auto CFHeaderOrErr =
5079 return CFHeaderOrErr.takeError();
5088 Twine(
"bad chained fixups: unknown imports format: ") +
5098 " overlaps with chained fixups header");
5100 uint32_t EndOffset = CFHeaderOffset + CFSize;
5104 Twine(CFImageStartsOffset +
5106 " extends past end " +
Twine(EndOffset));
5116 return CFOrErr.takeError();
5118 std::vector<ChainedFixupsSegment> Segments;
5119 if (!CFOrErr->has_value())
5120 return std::make_pair(0, Segments);
5126 return HeaderOrErr.takeError();
5127 if (!HeaderOrErr->has_value())
5128 return std::make_pair(0, Segments);
5131 const char *Contents =
getPtr(*
this, DyldChainedFixups.
dataoff);
5134 *
this, Contents +
Header.starts_offset);
5135 if (!ImageStartsOrErr)
5136 return ImageStartsOrErr.takeError();
5139 const char *SegOffsPtr =
5140 Contents +
Header.starts_offset +
5142 const char *SegOffsEnd =
5144 if (SegOffsEnd > Contents + DyldChainedFixups.
datasize)
5146 "bad chained fixups: seg_info_offset extends past end");
5148 const char *LastSegEnd =
nullptr;
5153 return OffOrErr.takeError();
5161 " at offset " +
Twine(*OffOrErr) + Message);
5164 const char *SegPtr = Contents +
Header.starts_offset + *OffOrErr;
5165 if (LastSegEnd && SegPtr < LastSegEnd)
5166 return Fail(
" overlaps with previous segment info");
5171 return SegOrErr.takeError();
5174 LastSegEnd = SegPtr + Seg.
size;
5178 const char *PageStart =
5181 if (PageEnd > SegPtr + Seg.
size)
5182 return Fail(
" : page_starts extend past seg_info size");
5186 std::vector<uint16_t> PageStarts;
5187 for (
size_t PageIdx = 0; PageIdx < Seg.
page_count; ++PageIdx) {
5192 PageStarts.push_back(Start);
5195 Segments.emplace_back(
I, *OffOrErr, Seg, std::move(PageStarts));
5198 return std::make_pair(ImageStarts.
seg_count, Segments);
5211template <
typename T,
unsigned N>
5213 std::array<T, N> RawValue;
5214 memcpy(RawValue.data(), Ptr,
N *
sizeof(
T));
5216 for (
auto &Element : RawValue)
5221Expected<std::vector<ChainedFixupTarget>>
5225 return CFOrErr.takeError();
5227 std::vector<ChainedFixupTarget> Targets;
5228 if (!CFOrErr->has_value())
5235 return CFHeaderOrErr.takeError();
5236 if (!(*CFHeaderOrErr))
5240 size_t ImportSize = 0;
5248 return malformedError(
"bad chained fixups: unknown imports format: " +
5251 const char *Contents =
getPtr(*
this, DyldChainedFixups.
dataoff);
5252 const char *Imports = Contents +
Header.imports_offset;
5253 size_t ImportsEndOffset =
5254 Header.imports_offset + ImportSize *
Header.imports_count;
5255 const char *ImportsEnd = Contents + ImportsEndOffset;
5256 const char *Symbols = Contents +
Header.symbols_offset;
5257 const char *SymbolsEnd = Contents + DyldChainedFixups.
datasize;
5259 if (ImportsEnd > Symbols)
5261 Twine(ImportsEndOffset) +
" overlaps with symbols");
5266 return createError(
"parsing big-endian chained fixups is not implemented");
5267 for (
const char *ImportPtr = Imports; ImportPtr < ImportsEnd;
5268 ImportPtr += ImportSize) {
5278 WeakImport = (RawValue[0] >> 8) & 1;
5279 NameOffset = RawValue[0] >> 9;
5287 WeakImport = (RawValue[0] >> 8) & 1;
5288 NameOffset = RawValue[0] >> 9;
5291 static_assert(2 *
sizeof(
uint64_t) ==
5296 NameOffset = (RawValue[0] >> 16) & 1;
5297 WeakImport = RawValue[0] >> 17;
5298 Addend = RawValue[1];
5303 const char *Str = Symbols + NameOffset;
5304 if (Str >= SymbolsEnd)
5306 Twine(NameOffset) +
" extends past end " +
5308 Targets.emplace_back(LibOrdinal, NameOffset, Str, Addend, WeakImport);
5311 return std::move(Targets);
5315 if (!DyldExportsTrieLoadCmd)
5319 *
this, DyldExportsTrieLoadCmd);
5320 if (!DyldExportsTrieOrError)
5329 if (!FuncStartsLoadCmd)
5340 return std::move(FunctionStarts);
5386 size_t MachOFilesetEntryOffset) {
5388 if (Magic ==
"\xFE\xED\xFA\xCE")
5390 UniversalIndex, MachOFilesetEntryOffset);
5391 if (Magic ==
"\xCE\xFA\xED\xFE")
5393 UniversalIndex, MachOFilesetEntryOffset);
5394 if (Magic ==
"\xFE\xED\xFA\xCF")
5396 UniversalIndex, MachOFilesetEntryOffset);
5397 if (Magic ==
"\xCF\xFA\xED\xFE")
5399 UniversalIndex, MachOFilesetEntryOffset);
5406 .
Case(
"debug_str_offs",
"debug_str_offsets")
5417 return std::vector<std::string>();
5423 EC,
"%s: expected directory 'Contents/Resources/DWARF' in dSYM bundle",
5424 Path.str().c_str());
5428 std::vector<std::string> ObjectPaths;
5430 Dir != DirEnd && !EC; Dir.increment(EC)) {
5439 ObjectPaths.push_back(ObjectPath.
str());
5446 if (ObjectPaths.empty())
5448 "%s: no objects found in dSYM bundle",
5449 Path.str().c_str());
5456#define HANDLE_SWIFT_SECTION(KIND, MACHO, ELF, COFF) \
5457 .Case(MACHO, llvm::binaryformat::Swift5ReflectionSectionKind::KIND)
5460#include "llvm/BinaryFormat/Swift.def"
5462#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
@ CPU_SUBTYPE_ARM_V8M_BASE
@ CPU_SUBTYPE_ARM_V8M_MAIN
@ CPU_SUBTYPE_ARM_V8_1M_MAIN
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)
Remove '.
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