boinc/lib/mac/QMachOImageList.c

662 lines
25 KiB
C

// This file is part of BOINC.
// http://boinc.berkeley.edu
// Copyright (C) 2008 University of California
//
// BOINC is free software; you can redistribute it and/or modify it
// under the terms of the GNU Lesser General Public License
// as published by the Free Software Foundation,
// either version 3 of the License, or (at your option) any later version.
//
// BOINC is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
// See the GNU Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public License
// along with BOINC. If not, see <http://www.gnu.org/licenses/>.
/*
* QMachOImageList.c
*
*/
/* This is part of a backtrace generator for boinc project applications.
*
* Adapted from Apple Developer Technical Support Sample Code QCrashReport
*
* This code handles Mac OS X 10.3.x through 10.4.9. It may require some
* adjustment for future OS versions; see the discussion of _sigtramp and
* PowerPC Signal Stack Frames in file QBacktrace.c.
*
* For useful tips on using backtrace information, see Apple Tech Note 2123:
* http://developer.apple.com/technotes/tn2004/tn2123.html#SECNOSYMBOLS
*
* To convert addresses to correct symbols, use the atos command-line tool:
* atos -o path/to/executable/with/symbols address
* Note: if address 1a23 is hex, use 0x1a23.
*
* To demangle mangled C++ symbols, use the c++filt command-line tool.
* You may need to prefix C++ symbols with an additonal underscore before
* passing them to c++filt (so they begin with two underscore characters).
*
* A very useful shell script to add symbols to a crash dump can be found at:
* http://developer.apple.com/tools/xcode/symbolizingcrashdumps.html
* Pipe the output of the shell script through c++filt to demangle C++ symbols.
*/
/*
File: QMachOImageList.c
Contains: Code to get all Mach-O images within a task.
Written by: DTS
Copyright: Copyright (c) 2007 Apple Inc. All Rights Reserved.
Disclaimer: IMPORTANT: This Apple software is supplied to you by Apple Inc.
("Apple") in consideration of your agreement to the following
terms, and your use, installation, modification or
redistribution of this Apple software constitutes acceptance of
these terms. If you do not agree with these terms, please do
not use, install, modify or redistribute this Apple software.
In consideration of your agreement to abide by the following
terms, and subject to these terms, Apple grants you a personal,
non-exclusive license, under Apple's copyrights in this
original Apple software (the "Apple Software"), to use,
reproduce, modify and redistribute the Apple Software, with or
without modifications, in source and/or binary forms; provided
that if you redistribute the Apple Software in its entirety and
without modifications, you must retain this notice and the
following text and disclaimers in all such redistributions of
the Apple Software. Neither the name, trademarks, service marks
or logos of Apple Inc. may be used to endorse or promote
products derived from the Apple Software without specific prior
written permission from Apple. Except as expressly stated in
this notice, no other rights or licenses, express or implied,
are granted by Apple herein, including but not limited to any
patent rights that may be infringed by your derivative works or
by other works in which the Apple Software may be incorporated.
The Apple Software is provided by Apple on an "AS IS" basis.
APPLE MAKES NO WARRANTIES, EXPRESS OR IMPLIED, INCLUDING
WITHOUT LIMITATION THE IMPLIED WARRANTIES OF NON-INFRINGEMENT,
MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE, REGARDING
THE APPLE SOFTWARE OR ITS USE AND OPERATION ALONE OR IN
COMBINATION WITH YOUR PRODUCTS.
IN NO EVENT SHALL APPLE BE LIABLE FOR ANY SPECIAL, INDIRECT,
INCIDENTAL OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
DATA, OR PROFITS; OR BUSINESS INTERRUPTION) ARISING IN ANY WAY
OUT OF THE USE, REPRODUCTION, MODIFICATION AND/OR DISTRIBUTION
OF THE APPLE SOFTWARE, HOWEVER CAUSED AND WHETHER UNDER THEORY
OF CONTRACT, TORT (INCLUDING NEGLIGENCE), STRICT LIABILITY OR
OTHERWISE, EVEN IF APPLE HAS BEEN ADVISED OF THE POSSIBILITY OF
SUCH DAMAGE.
Change History (most recent first):
$Log: QMachOImageList.c,v $
Revision 1.1 2007/03/02 12:20:22
First checked in.
*/
/////////////////////////////////////////////////////////////////
// Our prototypes
#include "QMachOImageList.h"
// System interfaces
#include <TargetConditionals.h>
//#include <assert.h>
#include <sys/param.h>
#include <errno.h>
#include <limits.h>
#include <stdio.h>
#include <stdlib.h>
#undef assert
#undef __assert
#define assert(e) ((void)0)
#include <mach-o/dyld.h>
#include <mach-o/loader.h>
// Scary Darwin-derived header file for dyld.
#include "dyld_gdb.h"
/////////////////////////////////////////////////////////////////
#pragma mark ***** New Technique
static int ImageListForTaskNew(
task_t task,
QMOImageRef dyldImage,
QTMAddr allImageInfoAddr,
QMOImageRef * imageArray,
size_t imageArraySize,
size_t * imageCountPtr
)
// The new interface is fairly straightforward: there's a single symbol
// ("dyld_all_image_infos", which has already been looked up by our caller,
// and whose address (in the remote task) is passed to us in allImageInfoAddr)
// that contains a simple data structure that describes all of the images.
// That data structure is defined by (struct dyld_all_image_infos) (see
// "dyld_gdb.h"); it, in turn, contains a pointer to an array of structures
// (struct dyld_image_info) for each image.
//
// Given the task, a dyld image, and the address of the
// "dyld_all_image_infos" structure within that task, this routine returns
// an array of images within that task (in the buffer specified by imageArray,
// if it's not NULL, and imageArraySize), and a count of those images
// (in *imageCountPtr).
//
// On entry, task must not be MACH_PORT_NULL
// On entry, dyldImage must not be NULL
// On entry, allImageInfoAddr must be the address of "dyld_all_image_infos"
// within the task, which shouldn't be NULL
// On entry, imageCountPtr must not be NULL
// On entry, *imageCountPtr is ignored
// On entry, if imageArray is NULL, imageArraySize must be 0
// On entry, if imageArray is not NULL, imageArraySize must be the number of
// elements available in imageArray
// On success, if imageArray is not NULL, up to imageArraySize image objects
// are returned in the array
// On success, *imageCountPtr is the number of images in the task; this may
// be larger than imageArraySize, in which case the imageArray wasn't big
// enough and the returned value tells you how big it needs to be
{
int err;
struct dyld_all_image_infos allImageInfo;
uint32_t infoCount = 0;
uint32_t infoIndex;
QTMAddr infoArrayAddr;
size_t infoArraySize;
const void * infoArrayLocal;
assert(task != MACH_PORT_NULL);
assert(dyldImage != NULL);
assert(allImageInfoAddr != 0); // very unlikely for dyld_all_image_infos to be at 0
assert( (imageArray != NULL) || (imageArraySize == 0) );
assert(imageCountPtr != NULL);
infoArrayLocal = NULL;
infoArraySize = 0; // quieten a warning
// Read the value of the dyld_all_image_infos structure.
err = QTMRead(task, allImageInfoAddr, sizeof(allImageInfo), &allImageInfo);
// Bletch. The layout of (struct dyld_all_image_infos) varies with the pointer
// size of the architecture. As we can't guarantee that the remote architecture
// is compatible with our local pointer size, we have to be very careful about
// the fields that we access within dyld_all_image_infos. It turns out that we
// only care about the "version", "infoArrayCount", and "infoArray" fields, and
// only the last one varies by pointer size, so this all works; but it's
// hardly nice.
if ( (err == 0) && (QMOImageToLocalUInt32(dyldImage, allImageInfo.version) != 1) ) {
err = ESHLIBVERS;
}
if (err == 0) {
infoCount = QMOImageToLocalUInt32(dyldImage, allImageInfo.infoArrayCount);
if ( QMOImageIs64Bit(dyldImage) ) {
infoArrayAddr = QMOImageToLocalUInt64(dyldImage, *(uint64_t *) &allImageInfo.infoArray);
infoArraySize = infoCount * sizeof(uint64_t) * 3;
} else {
infoArrayAddr = QMOImageToLocalUInt32(dyldImage, *(uint32_t *) &allImageInfo.infoArray);
infoArraySize = infoCount * sizeof(uint32_t) * 3;
}
// Read in the array of per-image structures.
err = QTMReadAllocated(task, infoArrayAddr, infoArraySize, &infoArrayLocal);
}
// Process the array of per-image structures, using the relevant information
// to create our output. Again, the size of these fields vary per architecture.
if (err == 0) {
*imageCountPtr = infoCount;
if (imageArray != NULL) {
bool is64Bit;
is64Bit = QMOImageIs64Bit(dyldImage);
for (infoIndex = 0; infoIndex < infoCount; infoIndex++) {
if (infoIndex < imageArraySize) {
QTMAddr machHeader;
QTMAddr filePathAddr;
char filePath[PATH_MAX];
if (is64Bit) {
machHeader = QMOImageToLocalUInt64(dyldImage, ((uint64_t *) infoArrayLocal)[infoIndex * 3] );
filePathAddr = QMOImageToLocalUInt64(dyldImage, ((uint64_t *) infoArrayLocal)[infoIndex * 3 + 1] );
} else {
machHeader = QMOImageToLocalUInt32(dyldImage, ((uint32_t *) infoArrayLocal)[infoIndex * 3] );
filePathAddr = QMOImageToLocalUInt32(dyldImage, ((uint32_t *) infoArrayLocal)[infoIndex * 3 + 1] );
}
err = QTMRead(task, filePathAddr, sizeof(filePath), filePath);
if (err == 0) {
filePath[sizeof(filePath) - 1] = 0; // guarantee NULL termination
err = QMOImageCreateFromTask(task, machHeader, filePath, &imageArray[infoIndex]);
}
if (err != 0) {
break;
}
}
}
}
}
// Clean up.
QTMFree(infoArrayLocal, infoArraySize);
return err;
}
/////////////////////////////////////////////////////////////////
#pragma mark ***** Old Technique
static int ImageListForTaskOldWithNames(
task_t task,
QMOImageRef dyldImage,
const char * listHeadName,
const char * elemCountName,
const char * elemSizeName,
QMOImageRef * imageArray,
size_t imageArraySize,
size_t * imageCountPtr
)
// The old interface is quite strange. dyld declares a global variable
// (whose name is given by listHeadName, for example, "library_images")
// that contains elemCount elements, each of elemSize bytes. The values
// of elemCount and elemSize are exported in global variables, each
// of which is a uint32_t. The names of these variables are also supplied
// as parameters to this routine (elemCountName and elemSizeName, for example,
// "gdb_nlibrary_images" and "gdb_library_image_size", respectively).
// Finally, at the end of this block of elements is a count of the number
// of elements in the block ("nimages") followed by a pointer to the
// next block of elements ("next_images").
//
// Within an element, there are only three fields that interest us:
//
// o "valid" (offset 12, size 4) -- Non-zero if this element is valid.
// o "mh" (offset 8, size 4) -- A pointer to the mach_header for this
// image.
// o "physical_name" (offset 0, size 4) -- A pointer to the file path for the
// on disk storage, if any.
//
// On entry, task must not be MACH_PORT_NULL
// On entry, dyldImage must not be NULL
// On entry, listHeadName, elemCountName, and elemSizeName must all not be NULL
// On entry, imageCountPtr must not be NULL
// On entry, *imageCountPtr is NOT ignored; rather, this routine assumes that
// imageArray already has *imageCountPtr elements filled in, and starts putting
// new elements at *imageCountPtr
// On entry, if imageArray is NULL, imageArraySize must be 0
// On entry, if imageArray is not NULL, imageArraySize must be the number of
// elements available in imageArray
// On success, if imageArray is not NULL, up to imageArraySize image objects
// are returned in the array, starting at the position specified by the
// original value of *imageCountPtr
// On success, *imageCountPtr is incremented by the number of images that were
// found; this may end up making it larger than imageArraySize, in which case the
// imageArray wasn't big enough and the returned value tells you how big
// it needs to be
{
int err;
QTMAddr listChunkAddr;
QTMAddr elemSizeAddr;
QTMAddr elemCountAddr;
uint32_t elemCount = 0;
uint32_t elemSize = 0;
size_t listChunkSize;
enum { // constants related to the array elements
kValidOffset = 12, // offset to "valid" field
kMHOffset = 8, // offset to "mh" field
kPhysicalNameOffset = 0 // offset to "physical_name" field
};
enum { // constants related to data after array
kNImagesOffset = 0, // offset to "nimages" field
kNextImagesOffset = 4, // offset to next_images field
kBlockSuffixSize = 8, // total size of trailing block
};
// The old-style interface is not present in 64-bit tasks.
assert( ! QMOImageIs64Bit(dyldImage) );
assert(task != MACH_PORT_NULL);
assert(dyldImage != NULL);
assert(listHeadName != NULL);
assert(elemCountName != NULL);
assert(elemSizeName != NULL);
assert( (imageArray != NULL) || (imageArraySize == 0) );
assert(imageCountPtr != NULL);
// Look up each of the symbols.
err = QMOImageLookupSymbol(dyldImage, listHeadName, &listChunkAddr);
if (err == 0) {
err = QMOImageLookupSymbol(dyldImage, elemCountName, &elemCountAddr);
}
if (err == 0) {
err = QMOImageLookupSymbol(dyldImage, elemSizeName, &elemSizeAddr);
}
// Read the elemCount and elemSize values.
if (err == 0) {
err = QTMRead(task, elemCountAddr, sizeof(elemCount), &elemCount);
}
if (err == 0) {
err = QTMRead(task, elemSizeAddr, sizeof(elemSize), &elemSize);
}
// Read the image data, one chunk at a time.
if (err == 0) {
elemCount = QMOImageToLocalUInt32(dyldImage, elemCount);
elemSize = QMOImageToLocalUInt32(dyldImage, elemSize);
listChunkSize = elemCount * elemSize + kBlockSuffixSize;
do {
const char * listChunk;
uint32_t elemCountInThisBlock;
uint32_t elemIndex;
listChunk = NULL;
// Read a copy of this chunk of the list.
err = QTMReadAllocated(task, listChunkAddr, listChunkSize, (const void **) &listChunk);
if (err == 0) {
elemCountInThisBlock = QMOImageToLocalUInt32(dyldImage, *(uint32_t *) (listChunk + elemCount * elemSize + kNImagesOffset) );
// Process each element in the chunk. If the element is valid, place
// the image into the output array (if there's size)
for (elemIndex = 0; elemIndex < elemCountInThisBlock; elemIndex++) {
if ( QMOImageToLocalUInt32(dyldImage, *(uint32_t *) (listChunk + elemIndex * elemSize + kValidOffset)) != 0 ) {
if ( (imageArray != NULL) && (*imageCountPtr < imageArraySize) ) {
QTMAddr machHeader;
QTMAddr filePathAddr;
char filePath[PATH_MAX];
machHeader = QMOImageToLocalUInt32(dyldImage, *(uint32_t *) (listChunk + elemIndex * elemSize + kMHOffset ));
filePathAddr = QMOImageToLocalUInt32(dyldImage, *(uint32_t *) (listChunk + elemIndex * elemSize + kPhysicalNameOffset));
err = QTMRead(task, filePathAddr, sizeof(filePath), filePath);
if (err == 0) {
filePath[sizeof(filePath) - 1] = 0; // guarantee NULL termination
err = QMOImageCreateFromTask(task, machHeader, filePath, &imageArray[*imageCountPtr]);
}
}
*imageCountPtr += 1;
if (err != 0) {
break;
}
}
}
}
// Move on to the next chunk.
if (err == 0) {
listChunkAddr = QMOImageToLocalUInt32(dyldImage, *(uint32_t *) (listChunk + elemCount * elemSize + kNextImagesOffset) );
}
QTMFree(listChunk, listChunkSize);
} while ( (err == 0) && (listChunkAddr != 0) );
}
return err;
}
static int ImageListForTaskOld(
task_t task,
QMOImageRef dyldImage,
QMOImageRef * imageArray,
size_t imageArraySize,
size_t * imageCountPtr
)
// Returns information about the prepared Mach-O images in a task
// using the old (pre-10.4) interface. See ImageListForTaskOldWithNames
// for a discussion of the mechanics of this.
//
// On entry, task must not be MACH_PORT_NULL
// On entry, dyldImage must not be NULL
// On entry, imageCountPtr must not be NULL
// On entry, *imageCountPtr is ignored
// On entry, if imageArray is NULL, imageArraySize must be 0
// On entry, if imageArray is not NULL, imageArraySize must be the number of
// elements available in imageArray
// On success, if imageArray is not NULL, up to imageArraySize image objects
// are returned in the array
// On success, *imageCountPtr is the number of images in the task; this may
// be larger than imageArraySize, in which case the imageArray wasn't big
// enough and the returned value tells you how big it needs to be
{
int err;
assert(task != MACH_PORT_NULL);
assert(dyldImage != NULL);
assert( (imageArray != NULL) || (imageArraySize == 0) );
assert(imageCountPtr != NULL);
// The old-style interface is only used for 32-bit.
assert( ! QMOImageIs64Bit(dyldImage) );
// The old interface makes a (strange) distinction between
// 'library' images and 'object' images. To avoid unnecessary
// code duplication I've factored this out into a subroutine
// that I call twice.
*imageCountPtr = 0;
err = ImageListForTaskOldWithNames(
task,
dyldImage,
"_library_images",
"_gdb_nlibrary_images",
"_gdb_library_image_size",
imageArray,
imageArraySize,
imageCountPtr
);
if (err == 0) {
err = ImageListForTaskOldWithNames(
task,
dyldImage,
"_object_images",
"_gdb_nobject_images",
"_gdb_object_image_size",
imageArray,
imageArraySize,
imageCountPtr
);
}
return err;
}
/////////////////////////////////////////////////////////////////
#pragma mark ***** Exported Abstraction Layer
static int SortByMachHeader(const void *lhs, const void *rhs)
{
QTMAddr lhsAddr;
QTMAddr rhsAddr;
lhsAddr = QMOImageGetMachHeaderOffset(*((QMOImageRef *) lhs));
rhsAddr = QMOImageGetMachHeaderOffset(*((QMOImageRef *) rhs));
// Can't do this using the standard (lhsAddr - rhsAddr) trick because the
// difference can be so big that it overflows an int, which is the type of
// the return value.
if (lhsAddr < rhsAddr) {
return -1;
} else if (lhsAddr > rhsAddr) {
return +1;
} else {
return 0;
}
}
extern void QMOImageListDestroy(QMOImageRef *imageArray, size_t imageArrayCount)
// See comments in header.
{
uint32_t imageIndex;
assert( (imageArray != NULL) || (imageArrayCount == 0) );
if (imageArray != NULL) {
for (imageIndex = 0; imageIndex < imageArrayCount; imageIndex++) {
QMOImageDestroy(imageArray[imageIndex]);
imageArray[imageIndex] = NULL;
}
}
}
bool kQMachOImageListTestOldDyldOnNew = false;
// By changing this variable you can test the old code when looking
// at a new dyld. For compatibility reasons, the new dyld continues
// to support the old interface (at least for 32-bit PowerPC), so this
// makes it possible to test the old code on 10.4, which makes life a lot
// easier.
//
// This is exported for the benefit of the unit test.
bool kQMachOImageListTestSelfShortCircuit = true;
// By changing this variable to false you can disable the short circuit that
// we apply when targetting the local task.
//
// This is exported for the benefit of the unit test.
extern int QMOImageListFromSelf(
QMOImageRef * imageArray,
size_t imageArrayCount,
size_t * imageCountPtr
)
// See comments in header.
{
int err;
uint32_t imageIndex;
assert( (imageArray != NULL) || (imageArrayCount == 0) );
assert(imageCountPtr != NULL);
// Blast the output array to NULL to aid in error recovery.
if (imageArray != NULL) {
memset(imageArray, 0, imageArrayCount * sizeof(*imageArray));
}
// Do the job.
err = 0;
*imageCountPtr = _dyld_image_count();
for (imageIndex = 0; imageIndex < *imageCountPtr; imageIndex++) {
if ( (imageArray != NULL) && (imageIndex < imageArrayCount) ) {
const struct mach_header * machHeader;
const char * filePath;
machHeader = _dyld_get_image_header(imageIndex);
filePath = _dyld_get_image_name(imageIndex);
// fprintf(stderr, "filePath = %s\n", filePath);
err = QMOImageCreateFromLocalImage(machHeader, filePath, &imageArray[imageIndex]);
if (err != 0) {
break;
}
}
}
if ( (err == 0) && (imageArray != NULL) ) {
qsort(imageArray, MIN(imageArrayCount, *imageCountPtr), sizeof(*imageArray), SortByMachHeader);
}
// On error, destroy any partial results.
if (err != 0) {
QMOImageListDestroy(imageArray, imageArrayCount);
}
return err;
}
extern int QMOImageListFromTask(
task_t task,
cpu_type_t cputype,
QMOImageRef * imageArray,
size_t imageArrayCount,
size_t * imageCountPtr
)
// See comments in header.
{
int err;
assert(task != MACH_PORT_NULL);
assert( (imageArray != NULL) || (imageArrayCount == 0) );
assert(imageCountPtr != NULL);
if ( (task == mach_task_self()) && (cputype == QMOGetLocalCPUType()) && kQMachOImageListTestSelfShortCircuit ) {
err = QMOImageListFromSelf(imageArray, imageArrayCount, imageCountPtr);
} else {
QMOImageRef dyldImage;
QTMAddr allImageInfoAddr;
// Blast the output array to NULL to aid in error recovery.
if (imageArray != NULL) {
memset(imageArray, 0, imageArrayCount * sizeof(*imageArray));
}
dyldImage = NULL;
// The information about the prepared images within a task is published
// in various global variables exported by dyld. So, to start off, we
// have to find the address of dyld in the remote task and then create
// an QMOImage for it.
err = QMOImageCreateFromTaskDyld(task, cputype, &dyldImage);
// Now we look for the dyld_all_image_infos variable exported by dyld.
// If it exists, we looking at a new dyld (10.4 and later), and we use
// the new mechanism for finding the prepared images. If it's not present,
// we're looking at an old dyld and we use the old mechanism.
if (err == 0) {
err = QMOImageLookupSymbol(dyldImage, "_dyld_all_image_infos", &allImageInfoAddr);
if ( ! kQMachOImageListTestOldDyldOnNew && (err == 0) ) {
err = ImageListForTaskNew(task, dyldImage, allImageInfoAddr, imageArray, imageArrayCount, imageCountPtr);
} else {
err = ImageListForTaskOld(task, dyldImage, imageArray, imageArrayCount, imageCountPtr);
}
}
if ( (err == 0) && (imageArray != NULL) ) {
qsort(imageArray, MIN(imageArrayCount, *imageCountPtr), sizeof(*imageArray), SortByMachHeader);
}
// Clean up.
QMOImageDestroy(dyldImage);
// On error, destroy any partial results.
if (err != 0) {
QMOImageListDestroy(imageArray, imageArrayCount);
}
}
return err;
}