diff options
Diffstat (limited to 'contrib/llvm/lib/Object/MachOObjectFile.cpp')
| -rw-r--r-- | contrib/llvm/lib/Object/MachOObjectFile.cpp | 518 | 
1 files changed, 155 insertions, 363 deletions
| diff --git a/contrib/llvm/lib/Object/MachOObjectFile.cpp b/contrib/llvm/lib/Object/MachOObjectFile.cpp index 79f810064988..d02ca48a7d19 100644 --- a/contrib/llvm/lib/Object/MachOObjectFile.cpp +++ b/contrib/llvm/lib/Object/MachOObjectFile.cpp @@ -38,6 +38,7 @@ namespace {    };  } +// FIXME: Replace all uses of this function with getStructOrErr.  template <typename T>  static T getStruct(const MachOObjectFile *O, const char *P) {    // Don't read before the beginning or past the end of the file @@ -51,39 +52,19 @@ static T getStruct(const MachOObjectFile *O, const char *P) {    return Cmd;  } -template <typename SegmentCmd> -static uint32_t getSegmentLoadCommandNumSections(const SegmentCmd &S, -                                                 uint32_t Cmdsize) { -  const unsigned SectionSize = sizeof(SegmentCmd); -  if (S.nsects > std::numeric_limits<uint32_t>::max() / SectionSize || -      S.nsects * SectionSize > Cmdsize - sizeof(S)) -    report_fatal_error( -        "Number of sections too large for size of load command."); -  return S.nsects; -} - -static uint32_t -getSegmentLoadCommandNumSections(const MachOObjectFile *O, -                                 const MachOObjectFile::LoadCommandInfo &L) { -  if (O->is64Bit()) -    return getSegmentLoadCommandNumSections(O->getSegment64LoadCommand(L), -                                            L.C.cmdsize); - -  return getSegmentLoadCommandNumSections(O->getSegmentLoadCommand(L), -                                          L.C.cmdsize); -} +template <typename T> +static ErrorOr<T> getStructOrErr(const MachOObjectFile *O, const char *P) { +  // Don't read before the beginning or past the end of the file +  if (P < O->getData().begin() || P + sizeof(T) > O->getData().end()) +    return object_error::parse_failed; -static bool isPageZeroSegment(const MachOObjectFile *O, -                              const MachOObjectFile::LoadCommandInfo &L) { -  if (O->is64Bit()) { -    MachO::segment_command_64 S = O->getSegment64LoadCommand(L); -    return StringRef("__PAGEZERO").equals(S.segname); -  } -  MachO::segment_command S = O->getSegmentLoadCommand(L); -  return StringRef("__PAGEZERO").equals(S.segname); +  T Cmd; +  memcpy(&Cmd, P, sizeof(T)); +  if (O->isLittleEndian() != sys::IsLittleEndianHost) +    MachO::swapStruct(Cmd); +  return Cmd;  } -  static const char *  getSectionPtr(const MachOObjectFile *O, MachOObjectFile::LoadCommandInfo L,                unsigned Sec) { @@ -128,70 +109,6 @@ static unsigned getCPUType(const MachOObjectFile *O) {    return O->getHeader().cputype;  } -static void printRelocationTargetName(const MachOObjectFile *O, -                                      const MachO::any_relocation_info &RE, -                                      raw_string_ostream &fmt) { -  bool IsScattered = O->isRelocationScattered(RE); - -  // Target of a scattered relocation is an address.  In the interest of -  // generating pretty output, scan through the symbol table looking for a -  // symbol that aligns with that address.  If we find one, print it. -  // Otherwise, we just print the hex address of the target. -  if (IsScattered) { -    uint32_t Val = O->getPlainRelocationSymbolNum(RE); - -    for (const SymbolRef &Symbol : O->symbols()) { -      std::error_code ec; -      uint64_t Addr; -      StringRef Name; - -      if ((ec = Symbol.getAddress(Addr))) -        report_fatal_error(ec.message()); -      if (Addr != Val) -        continue; -      if ((ec = Symbol.getName(Name))) -        report_fatal_error(ec.message()); -      fmt << Name; -      return; -    } - -    // If we couldn't find a symbol that this relocation refers to, try -    // to find a section beginning instead. -    for (const SectionRef &Section : O->sections()) { -      std::error_code ec; - -      StringRef Name; -      uint64_t Addr = Section.getAddress(); -      if (Addr != Val) -        continue; -      if ((ec = Section.getName(Name))) -        report_fatal_error(ec.message()); -      fmt << Name; -      return; -    } - -    fmt << format("0x%x", Val); -    return; -  } - -  StringRef S; -  bool isExtern = O->getPlainRelocationExternal(RE); -  uint64_t Val = O->getPlainRelocationSymbolNum(RE); - -  if (isExtern) { -    symbol_iterator SI = O->symbol_begin(); -    advance(SI, Val); -    SI->getName(S); -  } else { -    section_iterator SI = O->section_begin(); -    // Adjust for the fact that sections are 1-indexed. -    advance(SI, Val - 1); -    SI->getName(S); -  } - -  fmt << S; -} -  static uint32_t  getPlainRelocationAddress(const MachO::any_relocation_info &RE) {    return RE.r_word0; @@ -244,6 +161,69 @@ static uint32_t getSectionFlags(const MachOObjectFile *O,    return Sect.flags;  } +static ErrorOr<MachOObjectFile::LoadCommandInfo> +getLoadCommandInfo(const MachOObjectFile *Obj, const char *Ptr) { +  auto CmdOrErr = getStructOrErr<MachO::load_command>(Obj, Ptr); +  if (!CmdOrErr) +    return CmdOrErr.getError(); +  if (CmdOrErr->cmdsize < 8) +    return object_error::macho_small_load_command; +  MachOObjectFile::LoadCommandInfo Load; +  Load.Ptr = Ptr; +  Load.C = CmdOrErr.get(); +  return Load; +} + +static ErrorOr<MachOObjectFile::LoadCommandInfo> +getFirstLoadCommandInfo(const MachOObjectFile *Obj) { +  unsigned HeaderSize = Obj->is64Bit() ? sizeof(MachO::mach_header_64) +                                       : sizeof(MachO::mach_header); +  return getLoadCommandInfo(Obj, getPtr(Obj, HeaderSize)); +} + +static ErrorOr<MachOObjectFile::LoadCommandInfo> +getNextLoadCommandInfo(const MachOObjectFile *Obj, +                       const MachOObjectFile::LoadCommandInfo &L) { +  return getLoadCommandInfo(Obj, L.Ptr + L.C.cmdsize); +} + +template <typename T> +static void parseHeader(const MachOObjectFile *Obj, T &Header, +                        std::error_code &EC) { +  auto HeaderOrErr = getStructOrErr<T>(Obj, getPtr(Obj, 0)); +  if (HeaderOrErr) +    Header = HeaderOrErr.get(); +  else +    EC = HeaderOrErr.getError(); +} + +// Parses LC_SEGMENT or LC_SEGMENT_64 load command, adds addresses of all +// sections to \param Sections, and optionally sets +// \param IsPageZeroSegment to true. +template <typename SegmentCmd> +static std::error_code parseSegmentLoadCommand( +    const MachOObjectFile *Obj, const MachOObjectFile::LoadCommandInfo &Load, +    SmallVectorImpl<const char *> &Sections, bool &IsPageZeroSegment) { +  const unsigned SegmentLoadSize = sizeof(SegmentCmd); +  if (Load.C.cmdsize < SegmentLoadSize) +    return object_error::macho_load_segment_too_small; +  auto SegOrErr = getStructOrErr<SegmentCmd>(Obj, Load.Ptr); +  if (!SegOrErr) +    return SegOrErr.getError(); +  SegmentCmd S = SegOrErr.get(); +  const unsigned SectionSize = +      Obj->is64Bit() ? sizeof(MachO::section_64) : sizeof(MachO::section); +  if (S.nsects > std::numeric_limits<uint32_t>::max() / SectionSize || +      S.nsects * SectionSize > Load.C.cmdsize - SegmentLoadSize) +    return object_error::macho_load_segment_too_many_sections; +  for (unsigned J = 0; J < S.nsects; ++J) { +    const char *Sec = getSectionPtr(Obj, Load, J); +    Sections.push_back(Sec); +  } +  IsPageZeroSegment |= StringRef("__PAGEZERO").equals(S.segname); +  return std::error_code(); +} +  MachOObjectFile::MachOObjectFile(MemoryBufferRef Object, bool IsLittleEndian,                                   bool Is64bits, std::error_code &EC)      : ObjectFile(getMachOType(IsLittleEndian, Is64bits), Object), @@ -251,15 +231,25 @@ MachOObjectFile::MachOObjectFile(MemoryBufferRef Object, bool IsLittleEndian,        DataInCodeLoadCmd(nullptr), LinkOptHintsLoadCmd(nullptr),        DyldInfoLoadCmd(nullptr), UuidLoadCmd(nullptr),        HasPageZeroSegment(false) { -  uint32_t LoadCommandCount = this->getHeader().ncmds; -  if (LoadCommandCount == 0) +  if (is64Bit()) +    parseHeader(this, Header64, EC); +  else +    parseHeader(this, Header, EC); +  if (EC)      return; -  MachO::LoadCommandType SegmentLoadType = is64Bit() ? -    MachO::LC_SEGMENT_64 : MachO::LC_SEGMENT; +  uint32_t LoadCommandCount = getHeader().ncmds; +  if (LoadCommandCount == 0) +    return; -  MachOObjectFile::LoadCommandInfo Load = getFirstLoadCommandInfo(); -  for (unsigned I = 0; ; ++I) { +  auto LoadOrErr = getFirstLoadCommandInfo(this); +  if (!LoadOrErr) { +    EC = LoadOrErr.getError(); +    return; +  } +  LoadCommandInfo Load = LoadOrErr.get(); +  for (unsigned I = 0; I < LoadCommandCount; ++I) { +    LoadCommands.push_back(Load);      if (Load.C.cmd == MachO::LC_SYMTAB) {        // Multiple symbol tables        if (SymtabLoadCmd) { @@ -303,20 +293,14 @@ MachOObjectFile::MachOObjectFile(MemoryBufferRef Object, bool IsLittleEndian,          return;        }        UuidLoadCmd = Load.Ptr; -    } else if (Load.C.cmd == SegmentLoadType) { -      const unsigned SegmentLoadSize = this->is64Bit() -                                           ? sizeof(MachO::segment_command_64) -                                           : sizeof(MachO::segment_command); -      if (Load.C.cmdsize < SegmentLoadSize) -        report_fatal_error("Segment load command size is too small."); - -      uint32_t NumSections = getSegmentLoadCommandNumSections(this, Load); -      for (unsigned J = 0; J < NumSections; ++J) { -        const char *Sec = getSectionPtr(this, Load, J); -        Sections.push_back(Sec); -      } -      if (isPageZeroSegment(this, Load)) -        HasPageZeroSegment = true; +    } else if (Load.C.cmd == MachO::LC_SEGMENT_64) { +      if ((EC = parseSegmentLoadCommand<MachO::segment_command_64>( +               this, Load, Sections, HasPageZeroSegment))) +        return; +    } else if (Load.C.cmd == MachO::LC_SEGMENT) { +      if ((EC = parseSegmentLoadCommand<MachO::segment_command>( +               this, Load, Sections, HasPageZeroSegment))) +        return;      } else if (Load.C.cmd == MachO::LC_LOAD_DYLIB ||                 Load.C.cmd == MachO::LC_LOAD_WEAK_DYLIB ||                 Load.C.cmd == MachO::LC_LAZY_LOAD_DYLIB || @@ -324,12 +308,16 @@ MachOObjectFile::MachOObjectFile(MemoryBufferRef Object, bool IsLittleEndian,                 Load.C.cmd == MachO::LC_LOAD_UPWARD_DYLIB) {        Libraries.push_back(Load.Ptr);      } - -    if (I == LoadCommandCount - 1) -      break; -    else -      Load = getNextLoadCommandInfo(Load); +    if (I < LoadCommandCount - 1) { +      auto LoadOrErr = getNextLoadCommandInfo(this, Load); +      if (!LoadOrErr) { +        EC = LoadOrErr.getError(); +        return; +      } +      Load = LoadOrErr.get(); +    }    } +  assert(LoadCommands.size() == LoadCommandCount);  }  void MachOObjectFile::moveSymbolNext(DataRefImpl &Symb) const { @@ -348,7 +336,7 @@ std::error_code MachOObjectFile::getSymbolName(DataRefImpl Symb,      report_fatal_error(          "Symbol name entry points before beginning or past end of file.");    Res = StringRef(Start); -  return object_error::success; +  return std::error_code();  }  unsigned MachOObjectFile::getSectionType(SectionRef Sec) const { @@ -378,7 +366,7 @@ std::error_code MachOObjectFile::getIndirectName(DataRefImpl Symb,      return object_error::parse_failed;    const char *Start = &StringTable.data()[NValue];    Res = StringRef(Start); -  return object_error::success; +  return std::error_code();  }  std::error_code MachOObjectFile::getSymbolAddress(DataRefImpl Symb, @@ -398,31 +386,25 @@ std::error_code MachOObjectFile::getSymbolAddress(DataRefImpl Symb,      else        Res = Entry.n_value;    } -  return object_error::success; +  return std::error_code();  } -std::error_code MachOObjectFile::getSymbolAlignment(DataRefImpl DRI, -                                                    uint32_t &Result) const { +uint32_t MachOObjectFile::getSymbolAlignment(DataRefImpl DRI) const {    uint32_t flags = getSymbolFlags(DRI);    if (flags & SymbolRef::SF_Common) {      MachO::nlist_base Entry = getSymbolTableEntryBase(this, DRI); -    Result = 1 << MachO::GET_COMM_ALIGN(Entry.n_desc); -  } else { -    Result = 0; +    return 1 << MachO::GET_COMM_ALIGN(Entry.n_desc);    } -  return object_error::success; +  return 0;  } -std::error_code MachOObjectFile::getSymbolSize(DataRefImpl DRI, -                                               uint64_t &Result) const { +uint64_t MachOObjectFile::getSymbolSize(DataRefImpl DRI) const {    uint64_t Value;    getSymbolAddress(DRI, Value);    uint32_t flags = getSymbolFlags(DRI);    if (flags & SymbolRef::SF_Common) -    Result = Value; -  else -    Result = UnknownAddressOrSize; -  return object_error::success; +    return Value; +  return UnknownAddressOrSize;  }  std::error_code MachOObjectFile::getSymbolType(DataRefImpl Symb, @@ -435,7 +417,7 @@ std::error_code MachOObjectFile::getSymbolType(DataRefImpl Symb,    // If this is a STAB debugging symbol, we can do nothing more.    if (n_type & MachO::N_STAB) {      Res = SymbolRef::ST_Debug; -    return object_error::success; +    return std::error_code();    }    switch (n_type & MachO::N_TYPE) { @@ -446,7 +428,7 @@ std::error_code MachOObjectFile::getSymbolType(DataRefImpl Symb,        Res = SymbolRef::ST_Function;        break;    } -  return object_error::success; +  return std::error_code();  }  uint32_t MachOObjectFile::getSymbolFlags(DataRefImpl DRI) const { @@ -506,7 +488,7 @@ std::error_code MachOObjectFile::getSymbolSection(DataRefImpl Symb,      Res = section_iterator(SectionRef(DRI, this));    } -  return object_error::success; +  return std::error_code();  }  void MachOObjectFile::moveSectionNext(DataRefImpl &Sec) const { @@ -517,7 +499,7 @@ std::error_code MachOObjectFile::getSectionName(DataRefImpl Sec,                                                  StringRef &Result) const {    ArrayRef<char> Raw = getSectionRawName(Sec);    Result = parseSegmentOrSectionName(Raw.data()); -  return object_error::success; +  return std::error_code();  }  uint64_t MachOObjectFile::getSectionAddress(DataRefImpl Sec) const { @@ -548,7 +530,7 @@ std::error_code MachOObjectFile::getSectionContents(DataRefImpl Sec,    }    Res = this->getData().substr(Offset, Size); -  return object_error::success; +  return std::error_code();  }  uint64_t MachOObjectFile::getSectionAlignment(DataRefImpl Sec) const { @@ -643,7 +625,7 @@ std::error_code MachOObjectFile::getRelocationAddress(DataRefImpl Rel,    Sec.d.a = Rel.d.a;    uint64_t SecAddress = getSectionAddress(Sec);    Res = SecAddress + Offset; -  return object_error::success; +  return std::error_code();  }  std::error_code MachOObjectFile::getRelocationOffset(DataRefImpl Rel, @@ -652,7 +634,7 @@ std::error_code MachOObjectFile::getRelocationOffset(DataRefImpl Rel,           "Only implemented for MH_OBJECT");    MachO::any_relocation_info RE = getRelocation(Rel);    Res = getAnyRelocationAddress(RE); -  return object_error::success; +  return std::error_code();  }  symbol_iterator @@ -685,7 +667,7 @@ std::error_code MachOObjectFile::getRelocationType(DataRefImpl Rel,                                                     uint64_t &Res) const {    MachO::any_relocation_info RE = getRelocation(Rel);    Res = getAnyRelocationType(RE); -  return object_error::success; +  return std::error_code();  }  std::error_code @@ -797,183 +779,7 @@ MachOObjectFile::getRelocationTypeName(DataRefImpl Rel,        break;    }    Result.append(res.begin(), res.end()); -  return object_error::success; -} - -std::error_code -MachOObjectFile::getRelocationValueString(DataRefImpl Rel, -                                          SmallVectorImpl<char> &Result) const { -  MachO::any_relocation_info RE = getRelocation(Rel); - -  unsigned Arch = this->getArch(); - -  std::string fmtbuf; -  raw_string_ostream fmt(fmtbuf); -  unsigned Type = this->getAnyRelocationType(RE); -  bool IsPCRel = this->getAnyRelocationPCRel(RE); - -  // Determine any addends that should be displayed with the relocation. -  // These require decoding the relocation type, which is triple-specific. - -  // X86_64 has entirely custom relocation types. -  if (Arch == Triple::x86_64) { -    bool isPCRel = getAnyRelocationPCRel(RE); - -    switch (Type) { -      case MachO::X86_64_RELOC_GOT_LOAD: -      case MachO::X86_64_RELOC_GOT: { -        printRelocationTargetName(this, RE, fmt); -        fmt << "@GOT"; -        if (isPCRel) fmt << "PCREL"; -        break; -      } -      case MachO::X86_64_RELOC_SUBTRACTOR: { -        DataRefImpl RelNext = Rel; -        moveRelocationNext(RelNext); -        MachO::any_relocation_info RENext = getRelocation(RelNext); - -        // X86_64_RELOC_SUBTRACTOR must be followed by a relocation of type -        // X86_64_RELOC_UNSIGNED. -        // NOTE: Scattered relocations don't exist on x86_64. -        unsigned RType = getAnyRelocationType(RENext); -        if (RType != MachO::X86_64_RELOC_UNSIGNED) -          report_fatal_error("Expected X86_64_RELOC_UNSIGNED after " -                             "X86_64_RELOC_SUBTRACTOR."); - -        // The X86_64_RELOC_UNSIGNED contains the minuend symbol; -        // X86_64_RELOC_SUBTRACTOR contains the subtrahend. -        printRelocationTargetName(this, RENext, fmt); -        fmt << "-"; -        printRelocationTargetName(this, RE, fmt); -        break; -      } -      case MachO::X86_64_RELOC_TLV: -        printRelocationTargetName(this, RE, fmt); -        fmt << "@TLV"; -        if (isPCRel) fmt << "P"; -        break; -      case MachO::X86_64_RELOC_SIGNED_1: -        printRelocationTargetName(this, RE, fmt); -        fmt << "-1"; -        break; -      case MachO::X86_64_RELOC_SIGNED_2: -        printRelocationTargetName(this, RE, fmt); -        fmt << "-2"; -        break; -      case MachO::X86_64_RELOC_SIGNED_4: -        printRelocationTargetName(this, RE, fmt); -        fmt << "-4"; -        break; -      default: -        printRelocationTargetName(this, RE, fmt); -        break; -    } -  // X86 and ARM share some relocation types in common. -  } else if (Arch == Triple::x86 || Arch == Triple::arm || -             Arch == Triple::ppc) { -    // Generic relocation types... -    switch (Type) { -      case MachO::GENERIC_RELOC_PAIR: // prints no info -        return object_error::success; -      case MachO::GENERIC_RELOC_SECTDIFF: { -        DataRefImpl RelNext = Rel; -        moveRelocationNext(RelNext); -        MachO::any_relocation_info RENext = getRelocation(RelNext); - -        // X86 sect diff's must be followed by a relocation of type -        // GENERIC_RELOC_PAIR. -        unsigned RType = getAnyRelocationType(RENext); - -        if (RType != MachO::GENERIC_RELOC_PAIR) -          report_fatal_error("Expected GENERIC_RELOC_PAIR after " -                             "GENERIC_RELOC_SECTDIFF."); - -        printRelocationTargetName(this, RE, fmt); -        fmt << "-"; -        printRelocationTargetName(this, RENext, fmt); -        break; -      } -    } - -    if (Arch == Triple::x86 || Arch == Triple::ppc) { -      switch (Type) { -        case MachO::GENERIC_RELOC_LOCAL_SECTDIFF: { -          DataRefImpl RelNext = Rel; -          moveRelocationNext(RelNext); -          MachO::any_relocation_info RENext = getRelocation(RelNext); - -          // X86 sect diff's must be followed by a relocation of type -          // GENERIC_RELOC_PAIR. -          unsigned RType = getAnyRelocationType(RENext); -          if (RType != MachO::GENERIC_RELOC_PAIR) -            report_fatal_error("Expected GENERIC_RELOC_PAIR after " -                               "GENERIC_RELOC_LOCAL_SECTDIFF."); - -          printRelocationTargetName(this, RE, fmt); -          fmt << "-"; -          printRelocationTargetName(this, RENext, fmt); -          break; -        } -        case MachO::GENERIC_RELOC_TLV: { -          printRelocationTargetName(this, RE, fmt); -          fmt << "@TLV"; -          if (IsPCRel) fmt << "P"; -          break; -        } -        default: -          printRelocationTargetName(this, RE, fmt); -      } -    } else { // ARM-specific relocations -      switch (Type) { -        case MachO::ARM_RELOC_HALF: -        case MachO::ARM_RELOC_HALF_SECTDIFF: { -          // Half relocations steal a bit from the length field to encode -          // whether this is an upper16 or a lower16 relocation. -          bool isUpper = getAnyRelocationLength(RE) >> 1; - -          if (isUpper) -            fmt << ":upper16:("; -          else -            fmt << ":lower16:("; -          printRelocationTargetName(this, RE, fmt); - -          DataRefImpl RelNext = Rel; -          moveRelocationNext(RelNext); -          MachO::any_relocation_info RENext = getRelocation(RelNext); - -          // ARM half relocs must be followed by a relocation of type -          // ARM_RELOC_PAIR. -          unsigned RType = getAnyRelocationType(RENext); -          if (RType != MachO::ARM_RELOC_PAIR) -            report_fatal_error("Expected ARM_RELOC_PAIR after " -                               "ARM_RELOC_HALF"); - -          // NOTE: The half of the target virtual address is stashed in the -          // address field of the secondary relocation, but we can't reverse -          // engineer the constant offset from it without decoding the movw/movt -          // instruction to find the other half in its immediate field. - -          // ARM_RELOC_HALF_SECTDIFF encodes the second section in the -          // symbol/section pointer of the follow-on relocation. -          if (Type == MachO::ARM_RELOC_HALF_SECTDIFF) { -            fmt << "-"; -            printRelocationTargetName(this, RENext, fmt); -          } - -          fmt << ")"; -          break; -        } -        default: { -          printRelocationTargetName(this, RE, fmt); -        } -      } -    } -  } else -    printRelocationTargetName(this, RE, fmt); - -  fmt.flush(); -  Result.append(fmtbuf.begin(), fmtbuf.end()); -  return object_error::success; +  return std::error_code();  }  std::error_code MachOObjectFile::getRelocationHidden(DataRefImpl Rel, @@ -1001,7 +807,12 @@ std::error_code MachOObjectFile::getRelocationHidden(DataRefImpl Rel,      }    } -  return object_error::success; +  return std::error_code(); +} + +uint8_t MachOObjectFile::getRelocationLength(DataRefImpl Rel) const { +  MachO::any_relocation_info RE = getRelocation(Rel); +  return getAnyRelocationLength(RE);  }  // @@ -1179,7 +990,7 @@ std::error_code MachOObjectFile::getLibraryShortNameByIndex(unsigned Index,    }    Res = LibrariesShortNames[Index]; -  return object_error::success; +  return std::error_code();  }  basic_symbol_iterator MachOObjectFile::symbol_begin_impl() const { @@ -1437,21 +1248,8 @@ unsigned MachOObjectFile::getArch() const {  Triple MachOObjectFile::getArch(const char **McpuDefault,                                  Triple *ThumbTriple) const { -  Triple T; -  if (is64Bit()) { -    MachO::mach_header_64 H_64; -    H_64 = getHeader64(); -    T = MachOObjectFile::getArch(H_64.cputype, H_64.cpusubtype, McpuDefault); -    *ThumbTriple = MachOObjectFile::getThumbArch(H_64.cputype, H_64.cpusubtype, -                                                 McpuDefault); -  } else { -    MachO::mach_header H; -    H = getHeader(); -    T = MachOObjectFile::getArch(H.cputype, H.cpusubtype, McpuDefault); -    *ThumbTriple = MachOObjectFile::getThumbArch(H.cputype, H.cpusubtype, -                                                 McpuDefault); -  } -  return T; +  *ThumbTriple = getThumbArch(Header.cputype, Header.cpusubtype, McpuDefault); +  return getArch(Header.cputype, Header.cpusubtype, McpuDefault);  }  relocation_iterator MachOObjectFile::section_rel_begin(unsigned Index) const { @@ -2104,6 +1902,22 @@ iterator_range<bind_iterator> MachOObjectFile::weakBindTable() const {                     MachOBindEntry::Kind::Weak);  } +MachOObjectFile::load_command_iterator +MachOObjectFile::begin_load_commands() const { +  return LoadCommands.begin(); +} + +MachOObjectFile::load_command_iterator +MachOObjectFile::end_load_commands() const { +  return LoadCommands.end(); +} + +iterator_range<MachOObjectFile::load_command_iterator> +MachOObjectFile::load_commands() const { +  return iterator_range<load_command_iterator>(begin_load_commands(), +                                               end_load_commands()); +} +  StringRef  MachOObjectFile::getSectionFinalSegmentName(DataRefImpl Sec) const {    ArrayRef<char> Raw = getSectionRawFinalSegmentName(Sec); @@ -2203,29 +2017,6 @@ MachOObjectFile::getAnyRelocationSection(    return SectionRef(DRI, this);  } -MachOObjectFile::LoadCommandInfo -MachOObjectFile::getFirstLoadCommandInfo() const { -  MachOObjectFile::LoadCommandInfo Load; - -  unsigned HeaderSize = is64Bit() ? sizeof(MachO::mach_header_64) : -                                    sizeof(MachO::mach_header); -  Load.Ptr = getPtr(this, HeaderSize); -  Load.C = getStruct<MachO::load_command>(this, Load.Ptr); -  if (Load.C.cmdsize < 8) -    report_fatal_error("Load command with size < 8 bytes."); -  return Load; -} - -MachOObjectFile::LoadCommandInfo -MachOObjectFile::getNextLoadCommandInfo(const LoadCommandInfo &L) const { -  MachOObjectFile::LoadCommandInfo Next; -  Next.Ptr = L.Ptr + L.C.cmdsize; -  Next.C = getStruct<MachO::load_command>(this, Next.Ptr); -  if (Next.C.cmdsize < 8) -    report_fatal_error("Load command with size < 8 bytes."); -  return Next; -} -  MachO::section MachOObjectFile::getSection(DataRefImpl DRI) const {    assert(DRI.d.a < Sections.size() && "Should have detected this earlier");    return getStruct<MachO::section>(this, Sections[DRI.d.a]); @@ -2390,12 +2181,13 @@ MachOObjectFile::getDice(DataRefImpl Rel) const {    return getStruct<MachO::data_in_code_entry>(this, P);  } -MachO::mach_header MachOObjectFile::getHeader() const { -  return getStruct<MachO::mach_header>(this, getPtr(this, 0)); +const MachO::mach_header &MachOObjectFile::getHeader() const { +  return Header;  } -MachO::mach_header_64 MachOObjectFile::getHeader64() const { -  return getStruct<MachO::mach_header_64>(this, getPtr(this, 0)); +const MachO::mach_header_64 &MachOObjectFile::getHeader64() const { +  assert(is64Bit()); +  return Header64;  }  uint32_t MachOObjectFile::getIndirectSymbolTableEntry( | 
