]> git.saurik.com Git - apple/security.git/blobdiff - Security/libsecurity_utilities/lib/macho++.h
Security-57031.1.35.tar.gz
[apple/security.git] / Security / libsecurity_utilities / lib / macho++.h
diff --git a/Security/libsecurity_utilities/lib/macho++.h b/Security/libsecurity_utilities/lib/macho++.h
new file mode 100644 (file)
index 0000000..093cc5b
--- /dev/null
@@ -0,0 +1,239 @@
+/*
+ * Copyright (c) 2006,2011-2014 Apple Inc. All Rights Reserved.
+ * 
+ * @APPLE_LICENSE_HEADER_START@
+ * 
+ * This file contains Original Code and/or Modifications of Original Code
+ * as defined in and that are subject to the Apple Public Source License
+ * Version 2.0 (the 'License'). You may not use this file except in
+ * compliance with the License. Please obtain a copy of the License at
+ * http://www.opensource.apple.com/apsl/ and read it before using this
+ * file.
+ * 
+ * The Original Code and all software distributed under the License are
+ * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
+ * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
+ * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
+ * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
+ * Please see the License for the specific language governing rights and
+ * limitations under the License.
+ * 
+ * @APPLE_LICENSE_HEADER_END@
+ */
+
+//
+// macho++ - Mach-O object file helpers
+//
+#ifndef _H_MACHOPLUSPLUS
+#define _H_MACHOPLUSPLUS
+
+#include <mach-o/loader.h>
+#include <mach-o/fat.h>
+#include <mach-o/arch.h>
+#include <security_utilities/globalizer.h>
+#include <security_utilities/endian.h>
+#include <security_utilities/unix++.h>
+#include <security_utilities/cfutilities.h>
+#include <map>
+
+namespace Security {
+
+
+//
+// An architecture specification.
+// Simply a pair or (cpu type, cpu subtype), really.
+//
+class Architecture : public std::pair<cpu_type_t, cpu_subtype_t> {
+       typedef std::pair<cpu_type_t, cpu_subtype_t> _Pair;
+public:
+       Architecture() { }
+       explicit Architecture(cpu_type_t type, cpu_subtype_t sub = CPU_SUBTYPE_MULTIPLE)
+               : std::pair<cpu_type_t, cpu_subtype_t>(type, sub) { }
+       Architecture(const fat_arch &archInFile);
+       Architecture(const char *name);
+
+       cpu_type_t cpuType() const { return this->first; }
+       cpu_subtype_t cpuSubtype() const { return this->second; }
+       const char *name() const;                       // NULL if unknown
+       std::string displayName() const;        // always display-able
+       
+       static const cpu_type_t none = 0;
+       operator bool () const { return cpuType() != none; }
+       bool operator ! () const { return cpuType() == none; }
+       
+public:
+       friend bool operator == (const Architecture &a1, const Architecture &a2)
+       { return _Pair(a1) == _Pair(a2); }
+
+       friend bool operator < (const Architecture &a1, const Architecture &a2)
+       { return _Pair(a1) < _Pair(a2); }
+       
+       bool matches(const Architecture &templ) const;
+
+public:
+       static Architecture local();
+};
+
+
+//
+// Common features of Mach-O object images.
+// MachOBase does not define where we get this from.
+//
+class MachOBase {
+protected:
+       virtual ~MachOBase();
+
+public:
+       template <class T>
+       T flip(T value) const
+       { return mFlip ? Security::flip(value) : value; }
+       
+       bool isFlipped() const { return mFlip; }
+       bool is64() const { return m64; }
+       
+       const mach_header &header() const { return *mHeader; }
+       Architecture architecture() const;
+       uint32_t type() const;
+       uint32_t flags() const;
+       
+       const load_command *loadCommands() const { return mCommands; }
+       const load_command *nextCommand(const load_command *command) const;
+       size_t commandLength() const { return flip(mHeader->sizeofcmds); }
+       
+       const load_command *findCommand(uint32_t cmd) const;
+       const segment_command *findSegment(const char *segname) const;
+       const section *findSection(const char *segname, const char *sectname) const;
+       
+       const char *string(const load_command *cmd, const lc_str &str) const;
+
+       const linkedit_data_command *findCodeSignature() const;
+       const linkedit_data_command *findLibraryDependencies() const;
+       
+       size_t signingOffset() const;   // starting offset of CS section, or 0 if none
+       size_t signingLength() const;   // length of CS section, or 0 if none
+
+protected:
+       void initHeader(const mach_header *address);
+       void initCommands(const load_command *commands);
+       
+       size_t headerSize() const;              // size of header
+       size_t commandSize() const;             // size of commands area
+
+private:
+       const mach_header *mHeader;     // Mach-O header
+       const load_command *mCommands; // load commands
+       const load_command *mEndCommands; // end of load commands
+       
+       bool m64;                                       // is 64-bit
+       bool mFlip;                                     // wrong byte order (flip all integers)
+};
+
+
+//
+// A Mach-O object image that resides on disk.
+// We only read small parts of the contents into (discontinuous) memory.
+//
+class MachO : public MachOBase, public UnixPlusPlus::FileDesc {
+public:
+       MachO(FileDesc fd, size_t offset = 0, size_t length = 0);
+       ~MachO();
+       
+       size_t offset() const { return mOffset; }
+       size_t length() const { return mLength; }
+       size_t signingExtent() const;   // signingOffset, or file length if none
+
+       void seek(size_t offset);       // relative to start of image
+       CFDataRef dataAt(size_t offset, size_t size);
+       void validateStructure();       // is the structure of the mach-o sane
+
+       bool isSuspicious() const { return mSuspicious; }
+
+private:
+       size_t mOffset;                 // starting file offset
+       size_t mLength;                 // Mach-O file length
+       
+       mach_header mHeaderBuffer; // read-in Mach-O header
+       load_command *mCommandBuffer; // read-in (malloc'ed) Mach-O load commands
+
+       bool mSuspicious;               // strict validation failed
+};
+
+
+//
+// A Mach-O object image that was mapped into memory.
+// We expect the entire image to be contiguously mapped starting at the
+// address given. No particular alignment is required (beyond native
+// alignment constraints on member variables).
+//
+class MachOImage : public MachOBase {
+public:
+       MachOImage(const void *address);
+       
+       const void *address() { return &this->header(); }
+};
+
+class MainMachOImage : public MachOImage {
+public:
+       MainMachOImage();
+       
+       static const void *mainImageAddress();
+};
+
+
+//
+// A Universal object represents a Mach-O binary image (whole) file.
+// It can represent a true Universal (aka "Fat") file with multiple
+// architectures; but it will also represent a single Mach-O ("thin")
+// binary and make you believe it's a Universal with just one architecture.
+//
+class Universal : public UnixPlusPlus::FileDesc {
+public:
+       Universal(FileDesc fd, size_t offset = 0, size_t length = 0);
+       ~Universal();
+       
+       // return a genuine MachO object for the given architecture
+       MachO *architecture() const;            // native
+       MachO *architecture(const Architecture &arch) const; // given
+       MachO *architecture(size_t offset) const; // given by file offset
+       
+       // return (just) the starting offset of an architecture
+       size_t archOffset() const;                      // native
+       size_t archOffset(const Architecture &arch) const; // given
+       size_t archLength(const Architecture &arch) const; // given
+       bool narrowed() const { return mBase != 0; }    // part of a fat file
+       
+       // return a set of architectures contained
+       typedef std::set<Architecture> Architectures;
+       void architectures(Architectures &archs) const;
+
+       bool isUniversal() const { return mArchList != NULL; }
+       Architecture bestNativeArch() const;
+       const size_t lengthOfSlice(size_t offset) const;
+
+       size_t offset() const { return mBase; }
+       size_t length() const { return mLength; }
+
+       bool isSuspicious() const;
+       
+public:
+       static uint32_t typeOf(FileDesc fd);
+
+private:
+       const fat_arch *findArch(const Architecture &arch) const;
+       MachO *findImage(const Architecture &arch) const;
+
+private:
+       fat_arch *mArchList;            // architectures (NULL if thin file)
+       unsigned mArchCount;            // number of architectures (if fat)
+       Architecture mThinArch;         // single architecture (if thin)
+       size_t mBase;                           // overriding offset in file (all types)
+       size_t mLength;                         // length of the architecture if thin file
+       typedef std::map<size_t, size_t> OffsetsToLength;
+       OffsetsToLength mSizes; // the length for the slice at a given offset
+       bool mSuspicious;                       // strict validation failed
+};
+
+
+} // end namespace Security
+
+#endif // !_H_MACHOPLUSPLUS