]>
git.saurik.com Git - apple/libsecurity_codesigning.git/blob - lib/macho++.cpp
11eb33b669f3a8c502edde2bc9fa857f164caf97
2 * Copyright (c) 2006 Apple Computer, Inc. All Rights Reserved.
4 * @APPLE_LICENSE_HEADER_START@
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
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.
21 * @APPLE_LICENSE_HEADER_END@
25 // macho++ - Mach-O object file helpers
28 #include <security_utilities/memutils.h>
34 // Architecture values
36 Architecture::Architecture(const fat_arch
&arch
)
37 : pair
<cpu_type_t
, cpu_subtype_t
>(ntohl(arch
.cputype
), ntohl(arch
.cpusubtype
))
43 // The local architecture (on demand; cached)
46 const NXArchInfo
*arch
;
47 LocalArch() { arch
= NXGetLocalArchInfo(); }
49 static ModuleNexus
<LocalArch
> localArch
;
51 Architecture
Architecture::local()
53 const NXArchInfo
&local
= *localArch().arch
;
54 return Architecture(local
.cputype
, local
.cpusubtype
);
59 // Translate between names and numbers
61 const char *Architecture::name() const
63 if (const NXArchInfo
*info
= NXGetArchInfoFromCpuType(cpuType(), cpuSubtype()))
65 else if (cpuType() == CPU_TYPE_ARM
) { // work-around for non-ARM Leopard systems
66 if (cpuSubtype() == CPU_SUBTYPE_ARM_V6
)
76 // Create a MachO object from an open file and a starting offset.
77 // We load (only) the header and load commands into memory at that time.
79 MachO::MachO(FileDesc fd
, size_t offset
, size_t length
)
80 : FileDesc(fd
), mOffset(offset
), mLength(length
? length
: (fd
.fileSize() - offset
))
82 size_t size
= fd
.read(&mHeader
, sizeof(mHeader
), mOffset
);
83 if (size
!= sizeof(mHeader
))
84 UnixError::throwMe(ENOEXEC
);
85 switch (mHeader
.magic
) {
103 UnixError::throwMe(ENOEXEC
);
106 size_t cmdSize
= flip(mHeader
.sizeofcmds
);
107 size_t cmdStart
= m64
? sizeof(mach_header_64
) : sizeof(mach_header
);
108 mCommands
= (load_command
*)malloc(cmdSize
);
110 UnixError::throwMe();
111 if (fd
.read(mCommands
, cmdSize
, cmdStart
+ mOffset
) != cmdSize
)
112 UnixError::throwMe(ENOEXEC
);
113 mEndCommands
= LowLevelMemoryUtilities::increment
<load_command
>(mCommands
, cmdSize
);
114 secdebug("macho", "%p created fd=%d offset=0x%zx size=0x%zx %s%s %d command(s)",
115 this, this->fd(), mOffset
, mLength
, mFlip
? " flipped" : "", m64
? " 64-bit" : "",
116 flip(mHeader
.ncmds
));
122 // Note that we don't close the file descriptor.
126 secdebug("macho", "%p destroyed", this);
132 // Return various header fields
134 Architecture
MachO::architecture() const
136 return Architecture(flip(mHeader
.cputype
), flip(mHeader
.cpusubtype
));
139 uint32_t MachO::type() const
141 return flip(mHeader
.filetype
);
144 uint32_t MachO::flags() const
146 return flip(mHeader
.flags
);
151 // Iterate through load commands
153 const load_command
*MachO::nextCommand(const load_command
*command
) const
155 using LowLevelMemoryUtilities::increment
;
156 command
= increment
<const load_command
>(command
, flip(command
->cmdsize
));
157 return (command
< mEndCommands
) ? command
: NULL
;
162 // Locate a segment command, by name
164 const segment_command
*MachO::findSegment(const char *segname
) const
166 for (const load_command
*command
= loadCommands(); command
; command
= nextCommand(command
)) {
167 if (flip(command
->cmd
) == LC_SEGMENT
) {
168 const segment_command
*seg
= reinterpret_cast<const segment_command
*>(command
);
169 if (!strcmp(seg
->segname
, segname
))
176 const section
*MachO::findSection(const char *segname
, const char *sectname
) const
178 using LowLevelMemoryUtilities::increment
;
179 if (const segment_command
*seg
= findSegment(segname
)) {
181 const segment_command_64
*seg64
= reinterpret_cast<const segment_command_64
*>(seg
);
182 const section_64
*sect
= increment
<const section_64
>(seg64
+ 1, 0);
183 for (unsigned n
= flip(seg64
->nsects
); n
> 0; n
--, sect
++) {
184 if (!strcmp(sect
->sectname
, sectname
))
185 return reinterpret_cast<const section
*>(sect
);
188 const section
*sect
= increment
<const section
>(seg
+ 1, 0);
189 for (unsigned n
= flip(seg
->nsects
); n
> 0; n
--, sect
++) {
190 if (!strcmp(sect
->sectname
, sectname
))
200 // Figure out where the Code Signing information starts in the Mach-O binary image.
201 // The code signature is at the end of the file, and identified
202 // by a specially-named section. So its starting offset is also the end
203 // of the signable part.
204 // Note that the offset returned is relative to the start of the Mach-O image.
205 // Returns zero if not found (usually indicating that the binary was not signed).
207 const linkedit_data_command
*MachO::findCodeSignature() const
209 for (const load_command
*cmd
= loadCommands(); cmd
; cmd
= nextCommand(cmd
))
210 if (flip(cmd
->cmd
) == LC_CODE_SIGNATURE
)
211 return reinterpret_cast<const linkedit_data_command
*>(cmd
);
212 return NULL
; // not found
215 size_t MachO::signingOffset() const
217 if (const linkedit_data_command
*lec
= findCodeSignature())
218 return flip(lec
->dataoff
);
223 size_t MachO::signingLength() const
225 if (const linkedit_data_command
*lec
= findCodeSignature())
226 return flip(lec
->datasize
);
233 // Return the signing-limit length for this Mach-O binary image.
234 // This is the signingOffset if present, or the full length if not.
236 size_t MachO::signingExtent() const
238 if (size_t offset
= signingOffset())
248 void MachO::seek(size_t offset
)
250 FileDesc::seek(mOffset
+ offset
);
253 CFDataRef
MachO::dataAt(size_t offset
, size_t size
)
255 CFMallocData
buffer(size
);
256 if (this->read(buffer
, size
, mOffset
+ offset
) != size
)
257 UnixError::throwMe();
263 // Fat (aka universal) file wrappers
265 Universal::Universal(FileDesc fd
)
268 //@@@ we could save a read by using a heuristically sized combined buffer here
269 fat_header header
; // note how in the thin-file case, we'll reinterpret the header below
270 if (fd
.read(&header
, sizeof(header
), 0) != sizeof(header
))
271 UnixError::throwMe(ENOEXEC
);
272 switch (header
.magic
) {
276 mArchCount
= ntohl(header
.nfat_arch
);
277 size_t archSize
= sizeof(fat_arch
) * mArchCount
;
278 mArchList
= (fat_arch
*)malloc(archSize
);
280 UnixError::throwMe();
281 if (fd
.read(mArchList
, archSize
, sizeof(header
)) != archSize
) {
283 UnixError::throwMe(ENOEXEC
);
285 secdebug("macho", "%p is a fat file with %d architectures",
293 mThinArch
= Architecture(reinterpret_cast<mach_header
&>(header
).cputype
);
294 secdebug("macho", "%p is a thin file (%s)", this, mThinArch
.name());
300 mThinArch
= Architecture(ntohl(reinterpret_cast<mach_header
&>(header
).cputype
));
301 secdebug("macho", "%p is a thin file (%s)", this, mThinArch
.name());
304 UnixError::throwMe(ENOEXEC
);
308 Universal::~Universal()
315 // Get the "local" architecture from the fat file
316 // Throws ENOEXEC if not found.
318 MachO
*Universal::architecture() const
321 return findImage(bestNativeArch());
323 return new MachO(*this);
326 size_t Universal::archOffset() const
329 return ntohl(findArch(bestNativeArch())->offset
);
336 // Get the specified architecture from the fat file
337 // Throws ENOEXEC if not found.
339 MachO
*Universal::architecture(const Architecture
&arch
) const
342 return findImage(arch
);
343 else if (arch
== mThinArch
)
344 return new MachO(*this);
346 UnixError::throwMe(ENOEXEC
);
349 size_t Universal::archOffset(const Architecture
&arch
) const
352 return ntohl(findArch(arch
)->offset
);
353 else if (arch
== mThinArch
)
356 UnixError::throwMe(ENOEXEC
);
361 // Locate an architecture from the fat file's list.
362 // Throws ENOEXEC if not found.
364 const fat_arch
*Universal::findArch(const Architecture
&target
) const
366 assert(isUniversal());
367 const fat_arch
*end
= mArchList
+ mArchCount
;
368 for (const fat_arch
*arch
= mArchList
; arch
< end
; ++arch
)
369 if (cpu_type_t(ntohl(arch
->cputype
)) == target
.cpuType()) //@@@ ignoring subarch
371 UnixError::throwMe(ENOEXEC
); // not found
374 MachO
*Universal::findImage(const Architecture
&target
) const
376 const fat_arch
*arch
= findArch(target
);
377 return new MachO(*this, ntohl(arch
->offset
), ntohl(arch
->size
));
382 // Find the best-matching architecture for this fat file.
383 // We pick the native architecture if it's available.
384 // If it contains exactly one architecture, we take that.
385 // Otherwise, we throw.
387 Architecture
Universal::bestNativeArch() const
390 const cpu_type_t native
= Architecture::local().cpuType();
391 const fat_arch
*end
= mArchList
+ mArchCount
;
392 for (const fat_arch
*arch
= mArchList
; arch
< end
; ++arch
)
393 if (cpu_type_t(ntohl(arch
->cputype
)) == native
) // ignoring subarch
394 return Architecture(native
);
396 return Architecture(ntohl(mArchList
->cputype
));
397 UnixError::throwMe(ENOEXEC
);
404 // List all architectures from the fat file's list.
406 void Universal::architectures(Architectures
&archs
)
409 for (unsigned n
= 0; n
< mArchCount
; n
++)
410 archs
.insert(mArchList
[n
]);
412 auto_ptr
<MachO
> macho(architecture());
413 archs
.insert(macho
->architecture());
419 // Quickly guess the Mach-O type of a file.
420 // Returns type zero if the file isn't Mach-O or Universal.
421 // Does not reposition the file.
423 uint32_t Universal::typeOf(FileDesc fd
)
426 if (fd
.read(&header
, sizeof(header
), 0) != sizeof(header
))
429 switch (header
.magic
) {
432 return header
.filetype
;
436 return flip(header
.filetype
);
441 const fat_arch
*arch1
=
442 LowLevelMemoryUtilities::increment
<fat_arch
>(&header
, sizeof(fat_header
));
443 if (fd
.read(&header
, sizeof(header
), ntohl(arch1
->offset
)) != sizeof(header
))