fixes for issue 129 : reviewed by Waylonis

git-svn-id: http://google-breakpad.googlecode.com/svn/trunk@125 4c0a9323-5329-0410-9bdc-e9ce6186880e
This commit is contained in:
ladderbreaker 2007-03-09 21:45:42 +00:00
parent 530a7ad99e
commit 1f3d2571d1
2 changed files with 101 additions and 24 deletions

View file

@ -42,7 +42,8 @@
NSMutableDictionary *sources_; // Address and Source file paths (STRONG) NSMutableDictionary *sources_; // Address and Source file paths (STRONG)
NSMutableArray *cppAddresses_; // Addresses of C++ symbols (STRONG) NSMutableArray *cppAddresses_; // Addresses of C++ symbols (STRONG)
NSMutableDictionary *headers_; // Mach-o header information (STRONG) NSMutableDictionary *headers_; // Mach-o header information (STRONG)
NSNumber *lastFunctionStart_; // Address of last function symbol parsed NSMutableDictionary *lastFunctionStartDict_; // Keyed by section# (STRONG)
NSMutableDictionary *sectionNumbers_; // Keyed by seg/sect name (STRONG)
} }
- (id)initWithContentsOfFile:(NSString *)machoFile; - (id)initWithContentsOfFile:(NSString *)machoFile;

View file

@ -66,11 +66,12 @@ static const int kTextSection = 1;
standardIn:(NSFileHandle *)standardIn; standardIn:(NSFileHandle *)standardIn;
- (NSArray *)convertCPlusPlusSymbols:(NSArray *)symbols; - (NSArray *)convertCPlusPlusSymbols:(NSArray *)symbols;
- (void)convertSymbols; - (void)convertSymbols;
- (void)addFunction:(NSString *)name line:(int)line address:(uint64_t)address; - (void)addFunction:(NSString *)name line:(int)line address:(uint64_t)address section:(int)section;
- (BOOL)processSymbolItem:(struct nlist_64 *)list stringTable:(char *)table; - (BOOL)processSymbolItem:(struct nlist_64 *)list stringTable:(char *)table;
- (BOOL)loadSymbolInfo:(void *)base offset:(uint32_t)offset; - (BOOL)loadSymbolInfo:(void *)base offset:(uint32_t)offset;
- (BOOL)loadSymbolInfo64:(void *)base offset:(uint32_t)offset; - (BOOL)loadSymbolInfo64:(void *)base offset:(uint32_t)offset;
- (BOOL)loadSymbolInfoForArchitecture; - (BOOL)loadSymbolInfoForArchitecture;
- (void)generateSectionDictionary:(struct mach_header*)header;
- (BOOL)loadHeader:(void *)base offset:(uint32_t)offset; - (BOOL)loadHeader:(void *)base offset:(uint32_t)offset;
- (BOOL)loadHeader64:(void *)base offset:(uint32_t)offset; - (BOOL)loadHeader64:(void *)base offset:(uint32_t)offset;
- (BOOL)loadModuleInfo; - (BOOL)loadModuleInfo;
@ -221,7 +222,7 @@ static BOOL StringTailMatches(NSString *str, NSString *tail) {
} }
//============================================================================= //=============================================================================
- (void)addFunction:(NSString *)name line:(int)line address:(uint64_t)address { - (void)addFunction:(NSString *)name line:(int)line address:(uint64_t)address section:(int)section {
NSNumber *addressNum = [NSNumber numberWithUnsignedLongLong:address]; NSNumber *addressNum = [NSNumber numberWithUnsignedLongLong:address];
if (!address) if (!address)
@ -276,8 +277,9 @@ static BOOL StringTailMatches(NSString *str, NSString *tail) {
forKey:kAddressSourceLineKey]; forKey:kAddressSourceLineKey];
// Save the function name so that we can add the end of function address // Save the function name so that we can add the end of function address
if ([name length]) if ([name length]) {
lastFunctionStart_ = addressNum; [lastFunctionStartDict_ setObject:addressNum forKey:[NSNumber numberWithUnsignedInt:section] ];
}
} }
//============================================================================= //=============================================================================
@ -285,10 +287,18 @@ static BOOL StringTailMatches(NSString *str, NSString *tail) {
uint32_t n_strx = list->n_un.n_strx; uint32_t n_strx = list->n_un.n_strx;
BOOL result = NO; BOOL result = NO;
// We only care about symbols in the __text sect // We don't care about non-section specific information
if (list->n_sect != kTextSection) if (list->n_sect == 0 )
return NO; return NO;
int line = list->n_desc;
// We only care about line number information in __TEXT __text
uint32_t mainSection = [[sectionNumbers_ objectForKey:@"__TEXT__text" ] unsignedLongValue];
if(list->n_sect != mainSection) {
line = 0;
}
// Extract debugging information: // Extract debugging information:
// Doc: http://developer.apple.com/documentation/DeveloperTools/gdb/stabs/stabs_toc.html // Doc: http://developer.apple.com/documentation/DeveloperTools/gdb/stabs/stabs_toc.html
// Header: /usr/include/mach-o/stab.h: // Header: /usr/include/mach-o/stab.h:
@ -317,27 +327,32 @@ static BOOL StringTailMatches(NSString *str, NSString *tail) {
// The function has a ":" followed by some stuff // The function has a ":" followed by some stuff
fn = [fn substringToIndex:range.location]; fn = [fn substringToIndex:range.location];
[self addFunction:fn line:list->n_desc address:list->n_value]; [self addFunction:fn line:line address:list->n_value section:list->n_sect ];
result = YES; result = YES;
} else if (list->n_type == N_SLINE) { } else if (list->n_type == N_SLINE && list->n_sect == mainSection ) {
[self addFunction:nil line:list->n_desc address:list->n_value]; [self addFunction:nil line:line address:list->n_value section:list->n_sect ];
result = YES; result = YES;
} else if (((list->n_type & N_TYPE) == N_SECT) && !(list->n_type & N_STAB)) { } else if (((list->n_type & N_TYPE) == N_SECT) && !(list->n_type & N_STAB)) {
// Regular symbols or ones that are external // Regular symbols or ones that are external
NSString *fn = [NSString stringWithUTF8String:&table[n_strx]]; NSString *fn = [NSString stringWithUTF8String:&table[n_strx]];
[self addFunction:fn line:0 address:list->n_value]; [self addFunction:fn line:0 address:list->n_value section:list->n_sect ];
result = YES; result = YES;
} else if (list->n_type == N_ENSYM) { } else if (list->n_type == N_ENSYM && list->n_sect == mainSection ) {
if (lastFunctionStart_) { NSNumber *lastFunctionStart = [lastFunctionStartDict_
unsigned long long start = [lastFunctionStart_ unsignedLongLongValue]; objectForKey:[NSNumber numberWithUnsignedLongLong:list->n_sect] ];
if (lastFunctionStart) {
unsigned long long start = [lastFunctionStart unsignedLongLongValue];
unsigned long long size = list->n_value - start; unsigned long long size = list->n_value - start;
NSMutableDictionary *dict = [addresses_ objectForKey:lastFunctionStart_]; NSMutableDictionary *dict = [addresses_ objectForKey:lastFunctionStart];
assert(dict); assert(dict);
assert(list->n_value > start); assert(list->n_value > start);
[dict setObject:[NSNumber numberWithUnsignedLongLong:size] [dict setObject:[NSNumber numberWithUnsignedLongLong:size]
forKey:kFunctionSizeKey]; forKey:kFunctionSizeKey];
lastFunctionStart_ = nil;
[lastFunctionStartDict_ removeObjectForKey:[NSNumber numberWithUnsignedLongLong:list->n_sect] ];
} }
} }
@ -361,7 +376,7 @@ static BOOL StringTailMatches(NSString *str, NSString *tail) {
if (!addresses_) if (!addresses_)
addresses_ = [[NSMutableDictionary alloc] init]; addresses_ = [[NSMutableDictionary alloc] init];
for (uint32_t i = 0; cmd && (i < count); i++) { for (uint32_t i = 0; cmd && (i < count); ++i) {
if (cmd->cmd == symbolTableCommand) { if (cmd->cmd == symbolTableCommand) {
struct symtab_command *symtab = (struct symtab_command *)cmd; struct symtab_command *symtab = (struct symtab_command *)cmd;
uint32_t ncmds = SwapLongIfNeeded(symtab->nsyms); uint32_t ncmds = SwapLongIfNeeded(symtab->nsyms);
@ -453,6 +468,46 @@ static BOOL StringTailMatches(NSString *str, NSString *tail) {
return result; return result;
} }
//=============================================================================
// build a dictionary of section numbers keyed off a string
// which is the concatenation of the segment name and the section name
- (void)generateSectionDictionary:(struct mach_header*)header {
BOOL swap = (header->magic == MH_CIGAM);
uint32_t count = SwapLongIfNeeded(header->ncmds);
struct load_command *cmd =
(struct load_command *)((uint32_t)header + sizeof(struct mach_header));
uint32_t segmentCommand = SwapLongIfNeeded(LC_SEGMENT);
uint32_t sectionNumber = 1; // section numbers are counted from 1
if (!sectionNumbers_)
sectionNumbers_ = [[NSMutableDictionary alloc] init];
// loop through every segment command, then through every section
// contained inside each of them
for (uint32_t i = 0; cmd && (i < count); ++i) {
if (cmd->cmd == segmentCommand) {
struct segment_command *seg = (struct segment_command *)cmd;
section *sect = (section *)((uint32_t)cmd + sizeof(segment_command));
uint32_t nsects = SwapLongIfNeeded(seg->nsects);
for (uint32_t j = 0; j < nsects; ++j) {
//printf("%d: %s %s\n", sectionNumber, seg->segname, sect->sectname );
NSString *segSectName = [NSString stringWithFormat:@"%s%s",
seg->segname, sect->sectname ];
[sectionNumbers_ setValue:[NSNumber numberWithUnsignedLong:sectionNumber]
forKey:segSectName ];
++sect;
++sectionNumber;
}
}
uint32_t cmdSize = SwapLongIfNeeded(cmd->cmdsize);
cmd = (struct load_command *)((uint32_t)cmd + cmdSize);
}
}
//============================================================================= //=============================================================================
- (BOOL)loadHeader:(void *)base offset:(uint32_t)offset { - (BOOL)loadHeader:(void *)base offset:(uint32_t)offset {
struct mach_header *header = (struct mach_header *)((uint32_t)base + offset); struct mach_header *header = (struct mach_header *)((uint32_t)base + offset);
@ -460,11 +515,14 @@ static BOOL StringTailMatches(NSString *str, NSString *tail) {
uint32_t count = SwapLongIfNeeded(header->ncmds); uint32_t count = SwapLongIfNeeded(header->ncmds);
struct load_command *cmd = struct load_command *cmd =
(struct load_command *)((uint32_t)header + sizeof(struct mach_header)); (struct load_command *)((uint32_t)header + sizeof(struct mach_header));
uint32_t segmentCommand = SwapLongIfNeeded(LC_SEGMENT);
for (uint32_t i = 0; cmd && (i < count); i++) { [self generateSectionDictionary:header];
uint32_t segmentCommand = SwapLongIfNeeded(LC_SEGMENT);
for (uint32_t i = 0; cmd && (i < count); ++i) {
if (cmd->cmd == segmentCommand) { if (cmd->cmd == segmentCommand) {
struct segment_command *seg = (struct segment_command *)cmd; struct segment_command *seg = (struct segment_command *)cmd;
if (!strcmp(seg->segname, "__TEXT")) { if (!strcmp(seg->segname, "__TEXT")) {
uint32_t addr = SwapLongIfNeeded(seg->vmaddr); uint32_t addr = SwapLongIfNeeded(seg->vmaddr);
uint32_t size = SwapLongIfNeeded(seg->vmsize); uint32_t size = SwapLongIfNeeded(seg->vmsize);
@ -500,7 +558,7 @@ static BOOL StringTailMatches(NSString *str, NSString *tail) {
struct load_command *cmd = struct load_command *cmd =
(struct load_command *)((uint32_t)header + sizeof(struct mach_header_64)); (struct load_command *)((uint32_t)header + sizeof(struct mach_header_64));
for (uint32_t i = 0; cmd && (i < count); i++) { for (uint32_t i = 0; cmd && (i < count); ++i) {
uint32_t segmentCommand = SwapLongIfNeeded(LC_SEGMENT_64); uint32_t segmentCommand = SwapLongIfNeeded(LC_SEGMENT_64);
if (cmd->cmd == segmentCommand) { if (cmd->cmd == segmentCommand) {
struct segment_command_64 *seg = (struct segment_command_64 *)cmd; struct segment_command_64 *seg = (struct segment_command_64 *)cmd;
@ -610,7 +668,7 @@ static BOOL WriteFormat(int fd, const char *fmt, ...) {
uint64_t moduleSize = uint64_t moduleSize =
moduleSizeNum ? [moduleSizeNum unsignedLongLongValue] : 0; moduleSizeNum ? [moduleSizeNum unsignedLongLongValue] : 0;
lastFunctionStart_ = nil; [lastFunctionStartDict_ removeAllObjects];
// Gather the information // Gather the information
[self loadSymbolInfoForArchitecture]; [self loadSymbolInfoForArchitecture];
@ -684,6 +742,18 @@ static BOOL WriteFormat(int fd, const char *fmt, ...) {
symbol = [dict objectForKey:kAddressSymbolKey]; symbol = [dict objectForKey:kAddressSymbolKey];
// Skip some symbols // Skip some symbols
if (StringHeadMatches(symbol, @"vtable for"))
continue;
if (StringHeadMatches(symbol, @"__static_initialization_and_destruction_0"))
continue;
if (StringHeadMatches(symbol, @"_GLOBAL__I__"))
continue;
if (StringHeadMatches(symbol, @"__func__."))
continue;
if (StringHeadMatches(symbol, @"__gnu")) if (StringHeadMatches(symbol, @"__gnu"))
continue; continue;
@ -710,9 +780,10 @@ static BOOL WriteFormat(int fd, const char *fmt, ...) {
} }
} }
NSNumber *functionLength = [dict objectForKey:kFunctionSizeKey];
if (line) { if (line) {
if (symbol) { if (symbol && functionLength) {
NSNumber *functionLength = [dict objectForKey:kFunctionSizeKey];
uint64_t functionLengthVal = [functionLength unsignedLongLongValue]; uint64_t functionLengthVal = [functionLength unsignedLongLongValue];
// Function // Function
@ -753,6 +824,10 @@ static BOOL WriteFormat(int fd, const char *fmt, ...) {
return nil; return nil;
} }
// keep track of last function start address on a per-section basis
if (!lastFunctionStartDict_)
lastFunctionStartDict_ = [[NSMutableDictionary alloc] init];
// If there's more than one, use the native one // If there's more than one, use the native one
if ([headers_ count] > 1) { if ([headers_ count] > 1) {
const NXArchInfo *localArchInfo = NXGetLocalArchInfo(); const NXArchInfo *localArchInfo = NXGetLocalArchInfo();
@ -790,6 +865,7 @@ static BOOL WriteFormat(int fd, const char *fmt, ...) {
[addresses_ release]; [addresses_ release];
[sources_ release]; [sources_ release];
[headers_ release]; [headers_ release];
[lastFunctionStartDict_ release];
[super dealloc]; [super dealloc];
} }