dyld-750.5.tar.gz
[apple/dyld.git] / dyld3 / MachOLoaded.h
1 /*
2 * Copyright (c) 2017 Apple Inc. All rights reserved.
3 *
4 * @APPLE_LICENSE_HEADER_START@
5 *
6 * This file contains Original Code and/or Modifications of Original Code
7 * as defined in and that are subject to the Apple Public Source License
8 * Version 2.0 (the 'License'). You may not use this file except in
9 * compliance with the License. Please obtain a copy of the License at
10 * http://www.opensource.apple.com/apsl/ and read it before using this
11 * file.
12 *
13 * The Original Code and all software distributed under the License are
14 * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
15 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
16 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
17 * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
18 * Please see the License for the specific language governing rights and
19 * limitations under the License.
20 *
21 * @APPLE_LICENSE_HEADER_END@
22 */
23
24 #ifndef MachOLoaded_h
25 #define MachOLoaded_h
26
27 #include <stdint.h>
28
29 #include "Array.h"
30 #include "MachOFile.h"
31
32
33 class SharedCacheBuilder;
34
35 namespace dyld3 {
36
37
38 // A mach-o mapped into memory with zero-fill expansion
39 // Can be used in dyld at runtime or during closure building
40 struct VIS_HIDDEN MachOLoaded : public MachOFile
41 {
42 typedef const MachOLoaded* (^DependentToMachOLoaded)(const MachOLoaded* image, uint32_t depIndex);
43
44 // for dlsym()
45 bool hasExportedSymbol(const char* symbolName, DependentToMachOLoaded finder, void** result,
46 bool* resultPointsToInstructions) const;
47
48 // for DYLD_PRINT_SEGMENTS
49 const char* segmentName(uint32_t segIndex) const;
50
51 // used to see if main executable overlaps shared region
52 bool intersectsRange(uintptr_t start, uintptr_t length) const;
53
54 // for _dyld_get_image_slide()
55 intptr_t getSlide() const;
56
57 // for dladdr()
58 bool findClosestSymbol(uint64_t unSlidAddr, const char** symbolName, uint64_t* symbolUnslidAddr) const;
59
60 // for _dyld_find_unwind_sections()
61 const void* findSectionContent(const char* segName, const char* sectName, uint64_t& size) const;
62
63 // used at runtime to validate loaded image matches closure
64 void forEachCDHashOfCodeSignature(const void* codeSigStart, size_t codeSignLen,
65 void (^callback)(const uint8_t cdHash[20])) const;
66
67 // used by DyldSharedCache to find closure
68 static const uint8_t* trieWalk(Diagnostics& diag, const uint8_t* start, const uint8_t* end, const char* symbol);
69
70 // used by cache builder during error handling in chain bind processing
71 const char* dependentDylibLoadPath(uint32_t depIndex) const;
72
73 // used by closure builder to find the offset and size of the trie.
74 bool hasExportTrie(uint32_t& runtimeOffset, uint32_t& size) const;
75
76 // used by dyld/libdyld to apply fixups
77 #if BUILDING_DYLD || BUILDING_LIBDYLD
78 void fixupAllChainedFixups(Diagnostics& diag, const dyld_chained_starts_in_image* starts, uintptr_t slide,
79 Array<const void*> bindTargets, void (^fixupLogger)(void* loc, void* newValue)) const;
80 #endif
81
82
83
84 // For use with new rebase/bind scheme were each fixup location on disk contains info on what
85 // fix up it needs plus the offset to the next fixup.
86 union ChainedFixupPointerOnDisk
87 {
88 union Arm64e {
89 dyld_chained_ptr_arm64e_auth_rebase authRebase;
90 dyld_chained_ptr_arm64e_auth_bind authBind;
91 dyld_chained_ptr_arm64e_rebase rebase;
92 dyld_chained_ptr_arm64e_bind bind;
93
94 uint64_t signExtendedAddend() const;
95 uint64_t unpackTarget() const;
96 const char* keyName() const;
97 uint64_t signPointer(void* loc, uint64_t target) const;
98 };
99
100 union Generic64 {
101 dyld_chained_ptr_64_rebase rebase;
102 dyld_chained_ptr_64_bind bind;
103
104 uint64_t signExtendedAddend() const;
105 uint64_t unpackedTarget() const;
106 };
107
108 union Generic32 {
109 dyld_chained_ptr_32_rebase rebase;
110 dyld_chained_ptr_32_bind bind;
111
112 uint64_t signExtendedAddend() const;
113 };
114
115 typedef dyld_chained_ptr_32_cache_rebase Cache32;
116
117 uint64_t raw64;
118 Arm64e arm64e;
119 Generic64 generic64;
120
121 uint32_t raw32;
122 Generic32 generic32;
123 Cache32 cache32;
124
125
126 bool isRebase(uint16_t pointerFormat, uint64_t preferedLoadAddress, uint64_t& targetRuntimeOffset) const;
127 bool isBind(uint16_t pointerFormat, uint32_t& bindOrdinal) const;
128 };
129
130
131 struct LayoutInfo {
132 uintptr_t slide;
133 uintptr_t textUnslidVMAddr;
134 uintptr_t linkeditUnslidVMAddr;
135 uint32_t linkeditFileOffset;
136 uint32_t linkeditFileSize;
137 uint32_t linkeditSegIndex;
138 };
139
140 struct LinkEditInfo
141 {
142 const dyld_info_command* dyldInfo;
143 const linkedit_data_command* exportsTrie;
144 const linkedit_data_command* chainedFixups;
145 const symtab_command* symTab;
146 const dysymtab_command* dynSymTab;
147 const linkedit_data_command* splitSegInfo;
148 const linkedit_data_command* functionStarts;
149 const linkedit_data_command* dataInCode;
150 const linkedit_data_command* codeSig;
151 LayoutInfo layout;
152 };
153 void getLinkEditPointers(Diagnostics& diag, LinkEditInfo&) const;
154
155 // use by dyldinfo
156 void forEachFixupInAllChains(Diagnostics& diag, const dyld_chained_starts_in_image* starts, bool notifyNonPointers,
157 void (^callback)(ChainedFixupPointerOnDisk* fixupLocation, const dyld_chained_starts_in_segment* segInfo, bool& stop)) const;
158 protected:
159 friend SharedCacheBuilder;
160
161 struct FoundSymbol {
162 enum class Kind { headerOffset, absolute, resolverOffset };
163 Kind kind;
164 bool isThreadLocal;
165 bool isWeakDef;
166 const MachOLoaded* foundInDylib;
167 uint64_t value;
168 uint32_t resolverFuncOffset;
169 const char* foundSymbolName;
170 };
171
172 bool findExportedSymbol(Diagnostics& diag, const char* symbolName, bool weakImport, FoundSymbol& foundInfo, DependentToMachOLoaded finder) const;
173
174 void getLinkEditLoadCommands(Diagnostics& diag, LinkEditInfo& result) const;
175 void getLayoutInfo(LayoutInfo&) const;
176 const uint8_t* getLinkEditContent(const LayoutInfo& info, uint32_t fileOffset) const;
177 const uint8_t* getExportsTrie(const LinkEditInfo& info, uint64_t& trieSize) const;
178 void forEachGlobalSymbol(Diagnostics& diag, void (^callback)(const char* symbolName, uint64_t n_value, uint8_t n_type, uint8_t n_sect, uint16_t n_desc, bool& stop)) const;
179 void forEachLocalSymbol(Diagnostics& diag, void (^callback)(const char* symbolName, uint64_t n_value, uint8_t n_type, uint8_t n_sect, uint16_t n_desc, bool& stop)) const;
180 uint32_t dependentDylibCount() const;
181 bool findClosestFunctionStart(uint64_t address, uint64_t* functionStartAddress) const;
182
183 // This calls the callback for all code directories required for a given platform/binary combination.
184 // On watchOS main executables this is all cd hashes.
185 // On watchOS dylibs this is only the single cd hash we need (by rank defined by dyld, not the kernel).
186 // On all other platforms this always returns a single best cd hash (ranked to match the kernel).
187 // Note the callback parameter is really a CS_CodeDirectory.
188 void forEachCodeDirectoryBlob(const void* codeSigStart, size_t codeSignLen, void (^callback)(const void* cd)) const;
189 bool walkChain(Diagnostics& diag, const dyld_chained_starts_in_segment* segInfo, uint32_t pageIndex, uint16_t offsetInPage,
190 bool notifyNonPointers, void (^handler)(ChainedFixupPointerOnDisk* fixupLocation, const dyld_chained_starts_in_segment* segInfo, bool& stop)) const;
191
192
193 };
194
195 } // namespace dyld3
196
197 #endif /* MachOLoaded_h */