From 1d4ec71798cf58591abc191ceeaf7a5c8b2714ae Mon Sep 17 00:00:00 2001 From: Alexey Samsonov Date: Thu, 4 Jun 2015 19:45:22 +0000 Subject: [PATCH] [Object, MachO] Don't crash on parsing invalid MachO header. Summary: Instead, properly report this error from MachOObjectFile constructor. Test Plan: regression test suite Reviewers: rafael Subscribers: llvm-commits git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@239078 91177308-0d34-0410-b5e6-96231b3b80d8 --- lib/Object/MachOObjectFile.cpp | 33 +++++++++++++++++++++--- test/Object/Inputs/macho-invalid-header | Bin 0 -> 24 bytes test/Object/macho-invalid.test | 3 +++ 3 files changed, 32 insertions(+), 4 deletions(-) create mode 100644 test/Object/Inputs/macho-invalid-header diff --git a/lib/Object/MachOObjectFile.cpp b/lib/Object/MachOObjectFile.cpp index 1620c1a6a91..c1d13838444 100644 --- a/lib/Object/MachOObjectFile.cpp +++ b/lib/Object/MachOObjectFile.cpp @@ -38,6 +38,7 @@ namespace { }; } +// FIXME: Replace all uses of this function with getStructOrErr. template static T getStruct(const MachOObjectFile *O, const char *P) { // Don't read before the beginning or past the end of the file @@ -51,6 +52,19 @@ static T getStruct(const MachOObjectFile *O, const char *P) { return Cmd; } +template +static ErrorOr 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; + + T Cmd; + memcpy(&Cmd, P, sizeof(T)); + if (O->isLittleEndian() != sys::IsLittleEndianHost) + MachO::swapStruct(Cmd); + return Cmd; +} + template static uint32_t getSegmentLoadCommandNumSections(const SegmentCmd &S, uint32_t Cmdsize) { @@ -203,6 +217,16 @@ getNextLoadCommandInfo(const MachOObjectFile *Obj, return getLoadCommandInfo(Obj, L.Ptr + L.C.cmdsize); } +template +static void parseHeader(const MachOObjectFile *Obj, T &Header, + std::error_code &EC) { + auto HeaderOrErr = getStructOrErr(Obj, getPtr(Obj, 0)); + if (HeaderOrErr) + Header = HeaderOrErr.get(); + else + EC = HeaderOrErr.getError(); +} + MachOObjectFile::MachOObjectFile(MemoryBufferRef Object, bool IsLittleEndian, bool Is64bits, std::error_code &EC) : ObjectFile(getMachOType(IsLittleEndian, Is64bits), Object), @@ -210,14 +234,15 @@ MachOObjectFile::MachOObjectFile(MemoryBufferRef Object, bool IsLittleEndian, DataInCodeLoadCmd(nullptr), LinkOptHintsLoadCmd(nullptr), DyldInfoLoadCmd(nullptr), UuidLoadCmd(nullptr), HasPageZeroSegment(false) { - // Parse header. if (is64Bit()) - Header64 = getStruct(this, getPtr(this, 0)); + parseHeader(this, Header64, EC); else // First fields of MachO::mach_header_64 are the same as // in MachO::mach_header. - *reinterpret_cast(&this->Header64) = - getStruct(this, getPtr(this, 0)); + parseHeader(this, *reinterpret_cast(&this->Header64), + EC); + if (EC) + return; uint32_t LoadCommandCount = getHeader().ncmds; if (LoadCommandCount == 0) diff --git a/test/Object/Inputs/macho-invalid-header b/test/Object/Inputs/macho-invalid-header new file mode 100644 index 0000000000000000000000000000000000000000..da52d43a03a493457d458ba0c87332dadbc25d9a GIT binary patch literal 24 dcmX^A>+L^w1_nlE1|Y$pz`(!)WSsx?764kU2Aco? literal 0 HcmV?d00001 diff --git a/test/Object/macho-invalid.test b/test/Object/macho-invalid.test index 8af30b6c844..fd09abf338b 100644 --- a/test/Object/macho-invalid.test +++ b/test/Object/macho-invalid.test @@ -34,3 +34,6 @@ NAME-PAST-EOF: Symbol name entry points before beginning or past end of file RUN: not llvm-nm %p/Inputs/macho-invalid-section-index-getSectionRawName 2>&1 \ RUN: | FileCheck -check-prefix INVALID-SECTION-IDX-SYMBOL-SEC %s INVALID-SECTION-IDX-SYMBOL-SEC: getSymbolSection: Invalid section index + +RUN: not llvm-objdump -private-headers %p/Inputs/macho-invalid-header 2>&1 | FileCheck -check-prefix INVALID-HEADER %s +INVALID-HEADER: Invalid data was encountered while parsing the file -- 2.34.1