2001-12-17 14:31:42 +00:00
|
|
|
|
/** NSException - Object encapsulation of a general exception handler
|
2013-11-07 14:41:58 +00:00
|
|
|
|
Copyright (C) 1993-2013 Free Software Foundation, Inc.
|
1995-03-18 17:15:15 +00:00
|
|
|
|
|
|
|
|
|
Written by: Adam Fedor <fedor@boulder.colorado.edu>
|
|
|
|
|
Date: Mar 1995
|
|
|
|
|
|
1996-05-12 00:56:10 +00:00
|
|
|
|
This file is part of the GNUstep Base Library.
|
1995-03-18 17:15:15 +00:00
|
|
|
|
|
|
|
|
|
This library is free software; you can redistribute it and/or
|
2007-09-14 11:36:11 +00:00
|
|
|
|
modify it under the terms of the GNU Lesser General Public
|
1995-03-18 17:15:15 +00:00
|
|
|
|
License as published by the Free Software Foundation; either
|
2008-06-08 10:38:33 +00:00
|
|
|
|
version 2 of the License, or (at your option) any later version.
|
1999-04-21 20:16:25 +00:00
|
|
|
|
|
1995-03-18 17:15:15 +00:00
|
|
|
|
This library 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
|
2019-12-09 23:36:00 +00:00
|
|
|
|
Lesser General Public License for more details.
|
1995-03-18 17:15:15 +00:00
|
|
|
|
|
2007-09-14 11:36:11 +00:00
|
|
|
|
You should have received a copy of the GNU Lesser General Public
|
1995-03-18 17:15:15 +00:00
|
|
|
|
License along with this library; if not, write to the Free
|
2024-11-07 13:37:59 +00:00
|
|
|
|
Software Foundation, Inc., 31 Milk Street #960789 Boston, MA 02196 USA.
|
2001-12-18 16:54:15 +00:00
|
|
|
|
|
|
|
|
|
$Date$ $Revision$
|
1995-03-18 17:15:15 +00:00
|
|
|
|
*/
|
1996-05-28 18:27:40 +00:00
|
|
|
|
|
2010-02-19 08:12:46 +00:00
|
|
|
|
#import "common.h"
|
2010-02-14 10:48:10 +00:00
|
|
|
|
#define EXPOSE_NSException_IVARS 1
|
2010-02-24 10:23:47 +00:00
|
|
|
|
#define EXPOSE_NSThread_IVARS 1
|
2007-12-03 14:13:57 +00:00
|
|
|
|
#import "GSPrivate.h"
|
2007-11-29 20:53:26 +00:00
|
|
|
|
#import "Foundation/NSEnumerator.h"
|
|
|
|
|
#import "Foundation/NSException.h"
|
|
|
|
|
#import "Foundation/NSArray.h"
|
|
|
|
|
#import "Foundation/NSCoder.h"
|
2018-03-26 13:49:13 +00:00
|
|
|
|
#import "Foundation/NSData.h"
|
2016-03-11 16:48:31 +00:00
|
|
|
|
#import "Foundation/NSLock.h"
|
2007-11-29 20:53:26 +00:00
|
|
|
|
#import "Foundation/NSNull.h"
|
|
|
|
|
#import "Foundation/NSThread.h"
|
|
|
|
|
#import "Foundation/NSLock.h"
|
|
|
|
|
#import "Foundation/NSDictionary.h"
|
|
|
|
|
#import "Foundation/NSValue.h"
|
2010-02-14 10:48:10 +00:00
|
|
|
|
#import "GNUstepBase/NSString+GNUstepBase.h"
|
|
|
|
|
|
2018-03-26 13:49:13 +00:00
|
|
|
|
#import "GSPThread.h"
|
2016-03-11 16:48:31 +00:00
|
|
|
|
|
2010-07-15 09:59:01 +00:00
|
|
|
|
#ifdef __GNUSTEP_RUNTIME__
|
|
|
|
|
#include <objc/hooks.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
2010-12-23 02:23:05 +00:00
|
|
|
|
#ifdef HAVE_SET_UNCAUGHT_EXCEPTION_HANDLER
|
|
|
|
|
#include <objc/objc-exception.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
2010-03-16 06:11:00 +00:00
|
|
|
|
#ifdef HAVE_MALLOC_H
|
2013-11-07 14:41:58 +00:00
|
|
|
|
#if !defined(__OpenBSD__)
|
2010-03-16 06:11:00 +00:00
|
|
|
|
#include <malloc.h>
|
|
|
|
|
#endif
|
2013-11-07 14:41:58 +00:00
|
|
|
|
#endif
|
2010-02-25 19:05:21 +00:00
|
|
|
|
#ifdef HAVE_ALLOCA_H
|
|
|
|
|
#include <alloca.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
2001-05-31 22:39:16 +00:00
|
|
|
|
#include <stdio.h>
|
2010-01-24 17:13:03 +00:00
|
|
|
|
|
2009-09-11 16:14:45 +00:00
|
|
|
|
#ifdef HAVE_BACKTRACE
|
|
|
|
|
#include <execinfo.h>
|
|
|
|
|
#endif
|
2002-03-27 09:55:57 +00:00
|
|
|
|
|
2016-03-11 16:48:31 +00:00
|
|
|
|
/*
|
2016-05-14 14:47:43 +00:00
|
|
|
|
* Turn off USE_BFD if we don't have bfd support for it.
|
2016-03-11 16:48:31 +00:00
|
|
|
|
*/
|
|
|
|
|
#if !(defined(HAVE_BFD_H) && defined(HAVE_LIBBFD) && defined(HAVE_LIBIBERTY))
|
2016-05-14 14:47:43 +00:00
|
|
|
|
# if defined(USE_BFD)
|
|
|
|
|
# undef USE_BFD
|
|
|
|
|
# endif
|
2016-03-11 16:48:31 +00:00
|
|
|
|
#endif
|
|
|
|
|
|
2016-05-14 14:47:43 +00:00
|
|
|
|
#if defined(_WIN32) && !defined(USE_BFD)
|
2016-03-11 16:48:31 +00:00
|
|
|
|
#include <windows.h>
|
|
|
|
|
#if defined(HAVE_DBGHELP_H)
|
|
|
|
|
#include <dbghelp.h>
|
|
|
|
|
#else
|
|
|
|
|
/* Supply the relevant bits from dbghelp.h if we could't find the header.
|
|
|
|
|
*/
|
|
|
|
|
#define SYMOPT_UNDNAME 0x00000002
|
|
|
|
|
#define SYMOPT_DEFERRED_LOADS 0x00000004
|
|
|
|
|
typedef struct _SYMBOL_INFO {
|
|
|
|
|
ULONG SizeOfStruct;
|
|
|
|
|
ULONG TypeIndex;
|
|
|
|
|
uint64_t Reserved[2];
|
|
|
|
|
ULONG Index;
|
|
|
|
|
ULONG Size;
|
|
|
|
|
uint64_t ModBase;
|
|
|
|
|
ULONG Flags;
|
|
|
|
|
uint64_t Value;
|
|
|
|
|
uint64_t Address;
|
|
|
|
|
ULONG Register;
|
|
|
|
|
ULONG Scope;
|
|
|
|
|
ULONG Tag;
|
|
|
|
|
ULONG NameLen;
|
|
|
|
|
ULONG MaxNameLen;
|
|
|
|
|
TCHAR Name[1];
|
|
|
|
|
} SYMBOL_INFO;
|
|
|
|
|
#endif
|
|
|
|
|
#endif
|
|
|
|
|
|
2010-02-08 20:33:40 +00:00
|
|
|
|
static NSUncaughtExceptionHandler *_NSUncaughtExceptionHandler = 0;
|
2007-12-05 16:13:24 +00:00
|
|
|
|
|
|
|
|
|
#define _e_info (((id*)_reserved)[0])
|
|
|
|
|
#define _e_stack (((id*)_reserved)[1])
|
|
|
|
|
|
2010-01-24 17:13:03 +00:00
|
|
|
|
|
|
|
|
|
@interface NSException (GSPrivate)
|
|
|
|
|
- (GSStackTrace*) _callStack;
|
2006-05-08 15:30:52 +00:00
|
|
|
|
@end
|
|
|
|
|
|
2024-05-07 10:24:26 +00:00
|
|
|
|
#if defined(_WIN32) || defined(USE_BFD)
|
|
|
|
|
static gs_mutex_t traceLock;
|
|
|
|
|
#endif
|
2006-05-08 15:30:52 +00:00
|
|
|
|
|
2016-03-09 13:16:16 +00:00
|
|
|
|
#if defined(_WIN32)
|
2016-05-14 14:47:43 +00:00
|
|
|
|
#if defined(USE_BFD)
|
2010-01-24 17:13:03 +00:00
|
|
|
|
static NSString *
|
|
|
|
|
GSPrivateBaseAddress(void *addr, void **base)
|
2006-05-08 15:30:52 +00:00
|
|
|
|
{
|
2010-01-25 09:58:52 +00:00
|
|
|
|
MEMORY_BASIC_INFORMATION info;
|
|
|
|
|
|
|
|
|
|
/* Found a note saying that according to Matt Pietrek's "Under the Hood"
|
|
|
|
|
* column for the April 1997 issue of Microsoft Systems Journal, the
|
|
|
|
|
* allocation base returned by VirtualQuery can be used as the handle
|
|
|
|
|
* to obtain module information for a loaded library.
|
|
|
|
|
*/
|
|
|
|
|
if (VirtualQuery (addr, &info, sizeof(info)) != 0)
|
|
|
|
|
{
|
2010-02-03 09:31:59 +00:00
|
|
|
|
HMODULE handle = (HMODULE) info.AllocationBase;
|
2010-01-25 09:58:52 +00:00
|
|
|
|
unichar path[MAX_PATH+1];
|
|
|
|
|
|
|
|
|
|
if (GetModuleFileNameW(handle, path, sizeof(path)-1) != 0)
|
|
|
|
|
{
|
|
|
|
|
path[sizeof(path)-1] = '\0';
|
|
|
|
|
|
|
|
|
|
*base = info.BaseAddress;
|
2010-02-03 09:31:59 +00:00
|
|
|
|
return [NSString stringWithCharacters: path length: wcslen(path)];
|
2010-01-25 09:58:52 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2010-01-24 17:13:03 +00:00
|
|
|
|
return nil;
|
2006-05-08 15:30:52 +00:00
|
|
|
|
}
|
2016-05-14 14:47:43 +00:00
|
|
|
|
#endif /* USE_BFD */
|
2016-03-09 13:16:16 +00:00
|
|
|
|
#else /* _WIN32 */
|
2010-01-24 17:13:03 +00:00
|
|
|
|
|
|
|
|
|
#include <dlfcn.h>
|
|
|
|
|
|
2016-05-14 14:47:43 +00:00
|
|
|
|
#if defined(USE_BFD)
|
2010-01-24 17:13:03 +00:00
|
|
|
|
static NSString *
|
|
|
|
|
GSPrivateBaseAddress(void *addr, void **base)
|
|
|
|
|
{
|
|
|
|
|
#ifdef HAVE_DLADDR
|
|
|
|
|
Dl_info info;
|
|
|
|
|
|
|
|
|
|
if (!dladdr(addr, &info))
|
|
|
|
|
return nil;
|
|
|
|
|
|
|
|
|
|
*base = info.dli_fbase;
|
|
|
|
|
|
|
|
|
|
return [NSString stringWithUTF8String: info.dli_fname];
|
|
|
|
|
#else
|
|
|
|
|
return nil;
|
|
|
|
|
#endif
|
|
|
|
|
}
|
2016-05-14 14:47:43 +00:00
|
|
|
|
#endif /* USE_BFD */
|
2016-03-09 13:16:16 +00:00
|
|
|
|
#endif /* _WIN32 */
|
2010-01-24 17:13:03 +00:00
|
|
|
|
|
2016-05-14 14:47:43 +00:00
|
|
|
|
#if defined(USE_BFD)
|
2010-01-24 17:13:03 +00:00
|
|
|
|
|
|
|
|
|
// GSStackTrace inspired by FYStackTrace.m
|
|
|
|
|
// created by Wim Oudshoorn on Mon 11-Apr-2006
|
|
|
|
|
// reworked by Lloyd Dupont @ NovaMind.com on 4-May-2006
|
|
|
|
|
|
|
|
|
|
#include <bfd.h>
|
|
|
|
|
|
|
|
|
|
@class GSBinaryFileInfo;
|
|
|
|
|
|
|
|
|
|
@interface GSFunctionInfo : NSObject
|
|
|
|
|
{
|
|
|
|
|
void *_address;
|
|
|
|
|
NSString *_fileName;
|
|
|
|
|
NSString *_functionName;
|
|
|
|
|
int _lineNo;
|
|
|
|
|
GSBinaryFileInfo *_module;
|
|
|
|
|
}
|
|
|
|
|
- (void*) address;
|
|
|
|
|
- (NSString *) fileName;
|
|
|
|
|
- (NSString *) function;
|
|
|
|
|
- (id) initWithModule: (GSBinaryFileInfo*)module
|
|
|
|
|
address: (void*)address
|
|
|
|
|
file: (NSString*)file
|
|
|
|
|
function: (NSString*)function
|
|
|
|
|
line: (int)lineNo;
|
|
|
|
|
- (int) lineNumber;
|
|
|
|
|
- (GSBinaryFileInfo*) module;
|
|
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@interface GSBinaryFileInfo : NSObject
|
2006-05-08 15:30:52 +00:00
|
|
|
|
{
|
2010-01-24 17:13:03 +00:00
|
|
|
|
NSString *_fileName;
|
|
|
|
|
bfd *_abfd;
|
|
|
|
|
asymbol **_symbols;
|
|
|
|
|
long _symbolCount;
|
2006-05-08 15:30:52 +00:00
|
|
|
|
}
|
2010-01-24 17:13:03 +00:00
|
|
|
|
- (NSString *) fileName;
|
|
|
|
|
- (GSFunctionInfo *) functionForAddress: (void*) address;
|
|
|
|
|
- (id) initWithBinaryFile: (NSString *)fileName;
|
|
|
|
|
- (id) init; // return info for the current executing process
|
|
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@implementation GSFunctionInfo
|
|
|
|
|
|
|
|
|
|
- (void*) address
|
2006-05-08 15:30:52 +00:00
|
|
|
|
{
|
2010-01-24 17:13:03 +00:00
|
|
|
|
return _address;
|
2006-05-08 15:30:52 +00:00
|
|
|
|
}
|
2010-01-24 17:13:03 +00:00
|
|
|
|
|
2006-05-08 15:30:52 +00:00
|
|
|
|
- (oneway void) dealloc
|
|
|
|
|
{
|
2010-01-24 17:13:03 +00:00
|
|
|
|
DESTROY(_module);
|
|
|
|
|
DESTROY(_fileName);
|
|
|
|
|
DESTROY(_functionName);
|
2009-09-11 19:19:05 +00:00
|
|
|
|
[super dealloc];
|
2006-05-08 15:30:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-01-24 17:13:03 +00:00
|
|
|
|
- (NSString *) description
|
|
|
|
|
{
|
|
|
|
|
return [NSString stringWithFormat: @"(%@: %p) %@ %@: %d",
|
|
|
|
|
[_module fileName], _address, _functionName, _fileName, _lineNo];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
- (NSString *) fileName
|
2006-05-08 15:30:52 +00:00
|
|
|
|
{
|
2010-01-24 17:13:03 +00:00
|
|
|
|
return _fileName;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
- (NSString *) function
|
|
|
|
|
{
|
|
|
|
|
return _functionName;
|
|
|
|
|
}
|
2009-09-11 19:19:05 +00:00
|
|
|
|
|
2010-01-24 17:13:03 +00:00
|
|
|
|
- (id) init
|
|
|
|
|
{
|
2010-02-25 18:49:31 +00:00
|
|
|
|
DESTROY(self);
|
2010-01-24 17:13:03 +00:00
|
|
|
|
return nil;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
- (id) initWithModule: (GSBinaryFileInfo*)module
|
|
|
|
|
address: (void*)address
|
|
|
|
|
file: (NSString*)file
|
|
|
|
|
function: (NSString*)function
|
|
|
|
|
line: (int)lineNo
|
|
|
|
|
{
|
|
|
|
|
_module = RETAIN(module);
|
|
|
|
|
_address = address;
|
|
|
|
|
_fileName = [file copy];
|
|
|
|
|
_functionName = [function copy];
|
|
|
|
|
_lineNo = lineNo;
|
|
|
|
|
|
|
|
|
|
return self;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
- (int) lineNumber
|
|
|
|
|
{
|
|
|
|
|
return _lineNo;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
- (GSBinaryFileInfo *) module
|
|
|
|
|
{
|
|
|
|
|
return _module;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@implementation GSBinaryFileInfo
|
|
|
|
|
|
|
|
|
|
+ (GSBinaryFileInfo*) infoWithBinaryFile: (NSString *)fileName
|
|
|
|
|
{
|
|
|
|
|
return [[[self alloc] initWithBinaryFile: fileName] autorelease];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
+ (void) initialize
|
|
|
|
|
{
|
|
|
|
|
static BOOL first = YES;
|
|
|
|
|
|
|
|
|
|
if (first == NO)
|
2009-09-11 19:19:05 +00:00
|
|
|
|
{
|
2010-01-24 17:13:03 +00:00
|
|
|
|
return;
|
2009-09-11 19:19:05 +00:00
|
|
|
|
}
|
2010-01-24 17:13:03 +00:00
|
|
|
|
first = NO;
|
|
|
|
|
bfd_init ();
|
2006-05-08 15:30:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-01-24 17:13:03 +00:00
|
|
|
|
- (oneway void) dealloc
|
2006-05-08 15:30:52 +00:00
|
|
|
|
{
|
2010-01-24 17:13:03 +00:00
|
|
|
|
DESTROY(_fileName);
|
|
|
|
|
if (_abfd)
|
2009-09-11 19:19:05 +00:00
|
|
|
|
{
|
2010-01-24 17:13:03 +00:00
|
|
|
|
bfd_close (_abfd);
|
|
|
|
|
_abfd = NULL;
|
|
|
|
|
}
|
|
|
|
|
if (_symbols)
|
|
|
|
|
{
|
2011-02-19 19:42:42 +00:00
|
|
|
|
free(_symbols);
|
2010-01-24 17:13:03 +00:00
|
|
|
|
_symbols = NULL;
|
|
|
|
|
}
|
|
|
|
|
[super dealloc];
|
|
|
|
|
}
|
2009-09-11 19:19:05 +00:00
|
|
|
|
|
2010-01-24 17:13:03 +00:00
|
|
|
|
- (NSString *) fileName
|
|
|
|
|
{
|
|
|
|
|
return _fileName;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
- (id) init
|
|
|
|
|
{
|
|
|
|
|
NSString *processName;
|
|
|
|
|
|
|
|
|
|
processName = [[[NSProcessInfo processInfo] arguments] objectAtIndex: 0];
|
|
|
|
|
return [self initWithBinaryFile: processName];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
- (id) initWithBinaryFile: (NSString *)fileName
|
|
|
|
|
{
|
|
|
|
|
int neededSpace;
|
|
|
|
|
|
|
|
|
|
// 1st initialize the bfd
|
|
|
|
|
if ([fileName length] == 0)
|
|
|
|
|
{
|
|
|
|
|
//NSLog (@"GSBinaryFileInfo: No File");
|
2010-02-25 18:49:31 +00:00
|
|
|
|
DESTROY(self);
|
2010-01-24 17:13:03 +00:00
|
|
|
|
return nil;
|
|
|
|
|
}
|
|
|
|
|
_fileName = [fileName copy];
|
|
|
|
|
_abfd = bfd_openr ([fileName cString], NULL);
|
|
|
|
|
if (!_abfd)
|
|
|
|
|
{
|
|
|
|
|
//NSLog (@"GSBinaryFileInfo: No Binary Info");
|
2010-02-25 18:49:31 +00:00
|
|
|
|
DESTROY(self);
|
2010-01-24 17:13:03 +00:00
|
|
|
|
return nil;
|
|
|
|
|
}
|
|
|
|
|
if (!bfd_check_format_matches (_abfd, bfd_object, NULL))
|
|
|
|
|
{
|
|
|
|
|
//NSLog (@"GSBinaryFileInfo: BFD format object error");
|
2010-02-25 18:49:31 +00:00
|
|
|
|
DESTROY(self);
|
2010-01-24 17:13:03 +00:00
|
|
|
|
return nil;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// second read the symbols from it
|
|
|
|
|
if (!(bfd_get_file_flags (_abfd) & HAS_SYMS))
|
|
|
|
|
{
|
|
|
|
|
//NSLog (@"GSBinaryFileInfo: BFD does not contain any symbols");
|
2010-02-25 18:49:31 +00:00
|
|
|
|
DESTROY(self);
|
2010-01-24 17:13:03 +00:00
|
|
|
|
return nil;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
neededSpace = bfd_get_symtab_upper_bound (_abfd);
|
|
|
|
|
if (neededSpace < 0)
|
|
|
|
|
{
|
|
|
|
|
//NSLog (@"GSBinaryFileInfo: BFD error while deducing needed space");
|
2010-02-25 18:49:31 +00:00
|
|
|
|
DESTROY(self);
|
2010-01-24 17:13:03 +00:00
|
|
|
|
return nil;
|
|
|
|
|
}
|
|
|
|
|
if (neededSpace == 0)
|
|
|
|
|
{
|
|
|
|
|
//NSLog (@"GSBinaryFileInfo: BFD no space for symbols needed");
|
2010-02-25 18:49:31 +00:00
|
|
|
|
DESTROY(self);
|
2010-01-24 17:13:03 +00:00
|
|
|
|
return nil;
|
|
|
|
|
}
|
2024-06-21 10:45:38 +00:00
|
|
|
|
_symbols = (asymbol**)malloc(neededSpace);
|
2010-01-24 17:13:03 +00:00
|
|
|
|
if (!_symbols)
|
|
|
|
|
{
|
2024-06-21 10:45:38 +00:00
|
|
|
|
//NSLog(@"GSBinaryFileInfo: Can't allocate buffer");
|
2010-02-25 18:49:31 +00:00
|
|
|
|
DESTROY(self);
|
2010-01-24 17:13:03 +00:00
|
|
|
|
return nil;
|
|
|
|
|
}
|
2024-06-21 10:45:38 +00:00
|
|
|
|
_symbolCount = bfd_canonicalize_symtab(_abfd, _symbols);
|
2010-01-24 17:13:03 +00:00
|
|
|
|
if (_symbolCount < 0)
|
|
|
|
|
{
|
2024-06-21 10:45:38 +00:00
|
|
|
|
//NSLog(@"GSBinaryFileInfo: BFD error while reading symbols");
|
2010-02-25 18:49:31 +00:00
|
|
|
|
DESTROY(self);
|
2010-01-24 17:13:03 +00:00
|
|
|
|
return nil;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return self;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct SearchAddressStruct
|
|
|
|
|
{
|
|
|
|
|
void *theAddress;
|
|
|
|
|
GSBinaryFileInfo *module;
|
|
|
|
|
asymbol **symbols;
|
|
|
|
|
GSFunctionInfo *theInfo;
|
|
|
|
|
};
|
|
|
|
|
|
2024-05-07 10:24:26 +00:00
|
|
|
|
static void find_address(bfd *abfd, asection *section,
|
2010-01-24 17:13:03 +00:00
|
|
|
|
struct SearchAddressStruct *info)
|
|
|
|
|
{
|
|
|
|
|
bfd_vma address;
|
|
|
|
|
bfd_vma vma;
|
|
|
|
|
unsigned size;
|
|
|
|
|
const char *fileName = 0;
|
|
|
|
|
const char *functionName = 0;
|
|
|
|
|
unsigned line = 0;
|
|
|
|
|
|
|
|
|
|
if (info->theInfo)
|
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
2021-03-18 21:35:34 +00:00
|
|
|
|
address = (bfd_vma) (uintptr_t)info->theAddress;
|
|
|
|
|
|
|
|
|
|
/* bfd_get_section_vma() was changed to bfd_section_vma() together with
|
|
|
|
|
* changes to a couple of other inline functions.
|
|
|
|
|
*/
|
|
|
|
|
#if defined(HAVE_BFD_SECTION_VMA)
|
|
|
|
|
if (!(bfd_section_flags(section) & SEC_ALLOC))
|
2010-01-24 17:13:03 +00:00
|
|
|
|
{
|
|
|
|
|
return; // Only debug in this section
|
|
|
|
|
}
|
2021-03-18 21:35:34 +00:00
|
|
|
|
if (bfd_section_flags(section) & SEC_DATA)
|
2010-01-24 17:13:03 +00:00
|
|
|
|
{
|
|
|
|
|
return; // Only data in this section
|
|
|
|
|
}
|
2021-03-18 21:35:34 +00:00
|
|
|
|
vma = bfd_section_vma(section);
|
|
|
|
|
size = bfd_section_size(section);
|
2010-01-24 17:13:03 +00:00
|
|
|
|
#else
|
2021-03-18 21:35:34 +00:00
|
|
|
|
if (!(bfd_get_section_flags(abfd, section) & SEC_ALLOC))
|
|
|
|
|
{
|
|
|
|
|
return; // Only debug in this section
|
|
|
|
|
}
|
|
|
|
|
if (bfd_get_section_flags(abfd, section) & SEC_DATA)
|
|
|
|
|
{
|
|
|
|
|
return; // Only data in this section
|
|
|
|
|
}
|
|
|
|
|
vma = bfd_get_section_vma(abfd, section);
|
|
|
|
|
size = bfd_section_size(abfd, section);
|
2010-01-24 17:13:03 +00:00
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
if (address < vma || address >= vma + size)
|
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2024-06-21 10:45:38 +00:00
|
|
|
|
if (bfd_find_nearest_line(abfd, section, info->symbols,
|
2010-01-24 17:13:03 +00:00
|
|
|
|
address - vma, &fileName, &functionName, &line))
|
|
|
|
|
{
|
|
|
|
|
GSFunctionInfo *fi;
|
|
|
|
|
NSString *file = nil;
|
|
|
|
|
NSString *func = nil;
|
|
|
|
|
|
|
|
|
|
if (fileName != 0)
|
|
|
|
|
{
|
|
|
|
|
file = [NSString stringWithCString: fileName
|
|
|
|
|
encoding: [NSString defaultCStringEncoding]];
|
2006-05-08 15:30:52 +00:00
|
|
|
|
}
|
2010-01-24 17:13:03 +00:00
|
|
|
|
if (functionName != 0)
|
|
|
|
|
{
|
|
|
|
|
func = [NSString stringWithCString: functionName
|
|
|
|
|
encoding: [NSString defaultCStringEncoding]];
|
|
|
|
|
}
|
|
|
|
|
fi = [GSFunctionInfo alloc];
|
|
|
|
|
fi = [fi initWithModule: info->module
|
|
|
|
|
address: info->theAddress
|
|
|
|
|
file: file
|
|
|
|
|
function: func
|
|
|
|
|
line: line];
|
|
|
|
|
[fi autorelease];
|
|
|
|
|
info->theInfo = fi;
|
2009-09-11 19:19:05 +00:00
|
|
|
|
}
|
2007-02-04 08:43:16 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-01-24 17:13:03 +00:00
|
|
|
|
- (GSFunctionInfo *) functionForAddress: (void*) address
|
|
|
|
|
{
|
|
|
|
|
struct SearchAddressStruct searchInfo =
|
|
|
|
|
{ address, self, _symbols, nil };
|
|
|
|
|
|
2024-05-07 10:24:26 +00:00
|
|
|
|
GS_MUTEX_LOCK(traceLock);
|
|
|
|
|
bfd_map_over_sections(_abfd,
|
2010-01-24 17:13:03 +00:00
|
|
|
|
(void (*) (bfd *, asection *, void *)) find_address, &searchInfo);
|
2024-05-07 10:24:26 +00:00
|
|
|
|
GS_MUTEX_UNLOCK(traceLock);
|
2010-01-24 17:13:03 +00:00
|
|
|
|
return searchInfo.theInfo;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
2021-07-28 14:17:47 +00:00
|
|
|
|
static gs_mutex_t modLock;
|
2010-01-24 17:13:03 +00:00
|
|
|
|
static NSMutableDictionary *stackModules = nil;
|
|
|
|
|
|
|
|
|
|
// initialize stack trace info
|
|
|
|
|
static id
|
|
|
|
|
GSLoadModule(NSString *fileName)
|
2006-05-08 15:30:52 +00:00
|
|
|
|
{
|
2010-01-24 17:13:03 +00:00
|
|
|
|
GSBinaryFileInfo *module = nil;
|
|
|
|
|
|
2021-07-28 14:17:47 +00:00
|
|
|
|
GS_MUTEX_LOCK(modLock);
|
2010-01-24 17:13:03 +00:00
|
|
|
|
|
|
|
|
|
if (stackModules == nil)
|
2009-09-11 19:19:05 +00:00
|
|
|
|
{
|
2010-01-24 17:13:03 +00:00
|
|
|
|
NSEnumerator *enumerator;
|
|
|
|
|
NSBundle *bundle;
|
2009-09-11 19:19:05 +00:00
|
|
|
|
|
2010-01-24 17:13:03 +00:00
|
|
|
|
stackModules = [NSMutableDictionary new];
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Try to ensure we have the main, base and gui library bundles.
|
|
|
|
|
*/
|
|
|
|
|
[NSBundle mainBundle];
|
|
|
|
|
[NSBundle bundleForClass: [NSObject class]];
|
|
|
|
|
[NSBundle bundleForClass: NSClassFromString(@"NSView")];
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Add file info for all bundles with code.
|
|
|
|
|
*/
|
|
|
|
|
enumerator = [[NSBundle allBundles] objectEnumerator];
|
|
|
|
|
while ((bundle = [enumerator nextObject]) != nil)
|
|
|
|
|
{
|
|
|
|
|
if ([bundle load] == YES)
|
|
|
|
|
{
|
|
|
|
|
GSLoadModule([bundle executablePath]);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ([fileName length] > 0)
|
|
|
|
|
{
|
|
|
|
|
module = [stackModules objectForKey: fileName];
|
|
|
|
|
if (module == nil)
|
2009-09-11 16:14:45 +00:00
|
|
|
|
{
|
2010-01-24 17:13:03 +00:00
|
|
|
|
module = [GSBinaryFileInfo infoWithBinaryFile: fileName];
|
|
|
|
|
if (module == nil)
|
|
|
|
|
{
|
|
|
|
|
module = (id)[NSNull null];
|
|
|
|
|
}
|
|
|
|
|
if ([stackModules objectForKey: fileName] == nil)
|
|
|
|
|
{
|
|
|
|
|
[stackModules setObject: module forKey: fileName];
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
module = [stackModules objectForKey: fileName];
|
|
|
|
|
}
|
2006-05-08 15:30:52 +00:00
|
|
|
|
}
|
2009-09-11 19:19:05 +00:00
|
|
|
|
}
|
2021-07-28 14:17:47 +00:00
|
|
|
|
GS_MUTEX_UNLOCK(modLock);
|
2010-01-24 17:13:03 +00:00
|
|
|
|
|
|
|
|
|
if (module == (id)[NSNull null])
|
|
|
|
|
{
|
|
|
|
|
module = nil;
|
|
|
|
|
}
|
|
|
|
|
return module;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static NSArray*
|
|
|
|
|
GSListModules()
|
|
|
|
|
{
|
|
|
|
|
NSArray *result;
|
|
|
|
|
|
|
|
|
|
GSLoadModule(nil); // initialise
|
2021-07-28 14:17:47 +00:00
|
|
|
|
GS_MUTEX_LOCK(modLock);
|
2010-01-24 17:13:03 +00:00
|
|
|
|
result = [stackModules allValues];
|
2021-07-28 14:17:47 +00:00
|
|
|
|
GS_MUTEX_UNLOCK(modLock);
|
2010-01-24 17:13:03 +00:00
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2016-05-14 14:47:43 +00:00
|
|
|
|
#endif /* USE_BFD */
|
2010-01-24 17:13:03 +00:00
|
|
|
|
|
|
|
|
|
|
2020-07-27 23:02:55 +00:00
|
|
|
|
#if defined(WITH_UNWIND) && !defined(HAVE_BACKTRACE)
|
2019-06-13 15:24:28 +00:00
|
|
|
|
|
|
|
|
|
#include <unwind.h>
|
2019-12-05 10:07:40 +00:00
|
|
|
|
#if !defined(_WIN32)
|
2019-06-13 15:24:28 +00:00
|
|
|
|
#include <dlfcn.h>
|
2019-12-05 10:07:40 +00:00
|
|
|
|
#endif
|
2019-06-13 15:24:28 +00:00
|
|
|
|
|
|
|
|
|
struct GSBacktraceState
|
|
|
|
|
{
|
|
|
|
|
void **current;
|
|
|
|
|
void **end;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static _Unwind_Reason_Code
|
|
|
|
|
GSUnwindCallback(struct _Unwind_Context* context, void* arg)
|
|
|
|
|
{
|
2025-01-09 11:06:31 +00:00
|
|
|
|
struct GSBacktraceState *state = (struct GSBacktraceState*)arg;
|
|
|
|
|
uintptr_t pc = _Unwind_GetIP(context);
|
|
|
|
|
|
|
|
|
|
if (pc)
|
|
|
|
|
{
|
|
|
|
|
if (state->current == state->end)
|
|
|
|
|
{
|
|
|
|
|
return _URC_END_OF_STACK;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
*state->current++ = (void*)pc;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return 0; //_URC_OK/_URC_NO_REASON
|
2019-06-13 15:24:28 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-07-27 23:02:55 +00:00
|
|
|
|
#endif /* WITH_UNWIND && !HAVE_BACKTRACE */
|
2019-06-13 15:24:28 +00:00
|
|
|
|
|
|
|
|
|
|
2016-05-14 14:47:43 +00:00
|
|
|
|
#if defined(_WIN32) && !defined(USE_BFD)
|
2016-03-11 16:48:31 +00:00
|
|
|
|
typedef USHORT (WINAPI *CaptureStackBackTraceType)(ULONG,ULONG,PVOID*,PULONG);
|
|
|
|
|
typedef BOOL (WINAPI *SymInitializeType)(HANDLE,char*,BOOL);
|
|
|
|
|
typedef DWORD (WINAPI *SymSetOptionsType)(DWORD);
|
|
|
|
|
typedef BOOL (WINAPI *SymFromAddrType)(HANDLE,DWORD64,PDWORD64,SYMBOL_INFO*);
|
|
|
|
|
|
|
|
|
|
static CaptureStackBackTraceType capture = 0;
|
|
|
|
|
static SymInitializeType initSym = 0;
|
|
|
|
|
static SymSetOptionsType optSym = 0;
|
|
|
|
|
static SymFromAddrType fromSym = 0;
|
|
|
|
|
static HANDLE hProcess = 0;
|
|
|
|
|
#define MAXFRAMES 62 /* Limitation of windows-xp */
|
2018-03-26 13:49:13 +00:00
|
|
|
|
#else
|
|
|
|
|
#define MAXFRAMES 128 /* 1KB buffer on 64bit machine */
|
2016-03-11 16:48:31 +00:00
|
|
|
|
#endif
|
|
|
|
|
|
2018-03-27 06:06:17 +00:00
|
|
|
|
#if !defined(HAVE_BUILTIN_EXTRACT_RETURN_ADDRESS)
|
|
|
|
|
# define __builtin_extract_return_address(X) X
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#define _NS_FRAME_HACK(a) \
|
|
|
|
|
case a: env->addr = __builtin_frame_address(a + 1); break;
|
|
|
|
|
#define _NS_RETURN_HACK(a) \
|
|
|
|
|
case a: env->addr = (__builtin_frame_address(a + 1) ? \
|
|
|
|
|
__builtin_extract_return_address(__builtin_return_address(a + 1)) : 0); break;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* The following horrible signal handling code is a workaround for the fact
|
|
|
|
|
* that the __builtin_frame_address() and __builtin_return_address()
|
|
|
|
|
* functions are not reliable (at least not on my EM64T based system) and
|
|
|
|
|
* will sometimes walk off the stack and access illegal memory locations.
|
|
|
|
|
* In order to prevent such an occurrance from crashing the application,
|
|
|
|
|
* we use sigsetjmp() and siglongjmp() to ensure that we can recover, and
|
|
|
|
|
* we keep the jump buffer in thread-local memory to avoid possible thread
|
|
|
|
|
* safety issues.
|
|
|
|
|
* Of course this will fail horribly if an exception occurs in one of the
|
|
|
|
|
* few methods we use to manage the per-thread jump buffer.
|
|
|
|
|
*/
|
|
|
|
|
#if defined(HAVE_SYS_SIGNAL_H)
|
|
|
|
|
# include <sys/signal.h>
|
|
|
|
|
#elif defined(HAVE_SIGNAL_H)
|
|
|
|
|
# include <signal.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#if defined(_WIN32)
|
|
|
|
|
#ifndef SIGBUS
|
|
|
|
|
#define SIGBUS SIGILL
|
|
|
|
|
#endif
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
/* sigsetjmp may be a function or a macro. The test for the function is
|
|
|
|
|
* done at configure time so we can tell here if either is available.
|
|
|
|
|
*/
|
|
|
|
|
#if !defined(HAVE_SIGSETJMP) && !defined(sigsetjmp)
|
|
|
|
|
#define siglongjmp(A,B) longjmp(A,B)
|
|
|
|
|
#define sigsetjmp(A,B) setjmp(A)
|
|
|
|
|
#define sigjmp_buf jmp_buf
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
|
sigjmp_buf buf;
|
|
|
|
|
void *addr;
|
|
|
|
|
void (*bus)(int);
|
|
|
|
|
void (*segv)(int);
|
|
|
|
|
} jbuf_type;
|
|
|
|
|
|
|
|
|
|
static jbuf_type *
|
|
|
|
|
jbuf()
|
|
|
|
|
{
|
|
|
|
|
NSMutableData *d;
|
|
|
|
|
NSMutableDictionary *dict;
|
|
|
|
|
|
|
|
|
|
dict = [[NSThread currentThread] threadDictionary];
|
|
|
|
|
d = [dict objectForKey: @"GSjbuf"];
|
|
|
|
|
if (d == nil)
|
|
|
|
|
{
|
|
|
|
|
d = [[NSMutableData alloc] initWithLength: sizeof(jbuf_type)];
|
|
|
|
|
[dict setObject: d forKey: @"GSjbuf"];
|
|
|
|
|
RELEASE(d);
|
|
|
|
|
}
|
|
|
|
|
return (jbuf_type*)[d mutableBytes];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
recover(int sig)
|
|
|
|
|
{
|
|
|
|
|
siglongjmp(jbuf()->buf, 1);
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-29 11:57:47 +00:00
|
|
|
|
#ifdef __clang__
|
|
|
|
|
#pragma clang diagnostic push
|
|
|
|
|
#pragma clang diagnostic ignored "-Wframe-address"
|
|
|
|
|
#else
|
|
|
|
|
#pragma GCC diagnostic push
|
|
|
|
|
#pragma GCC diagnostic ignored "-Wframe-address"
|
|
|
|
|
#endif
|
|
|
|
|
|
2018-03-27 06:06:17 +00:00
|
|
|
|
void *
|
|
|
|
|
NSFrameAddress(NSUInteger offset)
|
|
|
|
|
{
|
|
|
|
|
jbuf_type *env;
|
|
|
|
|
|
|
|
|
|
env = jbuf();
|
|
|
|
|
if (sigsetjmp(env->buf, 1) == 0)
|
|
|
|
|
{
|
|
|
|
|
env->segv = signal(SIGSEGV, recover);
|
|
|
|
|
env->bus = signal(SIGBUS, recover);
|
|
|
|
|
switch (offset)
|
|
|
|
|
{
|
|
|
|
|
_NS_FRAME_HACK(0); _NS_FRAME_HACK(1); _NS_FRAME_HACK(2);
|
|
|
|
|
_NS_FRAME_HACK(3); _NS_FRAME_HACK(4); _NS_FRAME_HACK(5);
|
|
|
|
|
_NS_FRAME_HACK(6); _NS_FRAME_HACK(7); _NS_FRAME_HACK(8);
|
|
|
|
|
_NS_FRAME_HACK(9); _NS_FRAME_HACK(10); _NS_FRAME_HACK(11);
|
|
|
|
|
_NS_FRAME_HACK(12); _NS_FRAME_HACK(13); _NS_FRAME_HACK(14);
|
|
|
|
|
_NS_FRAME_HACK(15); _NS_FRAME_HACK(16); _NS_FRAME_HACK(17);
|
|
|
|
|
_NS_FRAME_HACK(18); _NS_FRAME_HACK(19); _NS_FRAME_HACK(20);
|
|
|
|
|
_NS_FRAME_HACK(21); _NS_FRAME_HACK(22); _NS_FRAME_HACK(23);
|
|
|
|
|
_NS_FRAME_HACK(24); _NS_FRAME_HACK(25); _NS_FRAME_HACK(26);
|
|
|
|
|
_NS_FRAME_HACK(27); _NS_FRAME_HACK(28); _NS_FRAME_HACK(29);
|
|
|
|
|
_NS_FRAME_HACK(30); _NS_FRAME_HACK(31); _NS_FRAME_HACK(32);
|
|
|
|
|
_NS_FRAME_HACK(33); _NS_FRAME_HACK(34); _NS_FRAME_HACK(35);
|
|
|
|
|
_NS_FRAME_HACK(36); _NS_FRAME_HACK(37); _NS_FRAME_HACK(38);
|
|
|
|
|
_NS_FRAME_HACK(39); _NS_FRAME_HACK(40); _NS_FRAME_HACK(41);
|
|
|
|
|
_NS_FRAME_HACK(42); _NS_FRAME_HACK(43); _NS_FRAME_HACK(44);
|
|
|
|
|
_NS_FRAME_HACK(45); _NS_FRAME_HACK(46); _NS_FRAME_HACK(47);
|
|
|
|
|
_NS_FRAME_HACK(48); _NS_FRAME_HACK(49); _NS_FRAME_HACK(50);
|
|
|
|
|
_NS_FRAME_HACK(51); _NS_FRAME_HACK(52); _NS_FRAME_HACK(53);
|
|
|
|
|
_NS_FRAME_HACK(54); _NS_FRAME_HACK(55); _NS_FRAME_HACK(56);
|
|
|
|
|
_NS_FRAME_HACK(57); _NS_FRAME_HACK(58); _NS_FRAME_HACK(59);
|
|
|
|
|
_NS_FRAME_HACK(60); _NS_FRAME_HACK(61); _NS_FRAME_HACK(62);
|
|
|
|
|
_NS_FRAME_HACK(63); _NS_FRAME_HACK(64); _NS_FRAME_HACK(65);
|
|
|
|
|
_NS_FRAME_HACK(66); _NS_FRAME_HACK(67); _NS_FRAME_HACK(68);
|
|
|
|
|
_NS_FRAME_HACK(69); _NS_FRAME_HACK(70); _NS_FRAME_HACK(71);
|
|
|
|
|
_NS_FRAME_HACK(72); _NS_FRAME_HACK(73); _NS_FRAME_HACK(74);
|
|
|
|
|
_NS_FRAME_HACK(75); _NS_FRAME_HACK(76); _NS_FRAME_HACK(77);
|
|
|
|
|
_NS_FRAME_HACK(78); _NS_FRAME_HACK(79); _NS_FRAME_HACK(80);
|
|
|
|
|
_NS_FRAME_HACK(81); _NS_FRAME_HACK(82); _NS_FRAME_HACK(83);
|
|
|
|
|
_NS_FRAME_HACK(84); _NS_FRAME_HACK(85); _NS_FRAME_HACK(86);
|
|
|
|
|
_NS_FRAME_HACK(87); _NS_FRAME_HACK(88); _NS_FRAME_HACK(89);
|
|
|
|
|
_NS_FRAME_HACK(90); _NS_FRAME_HACK(91); _NS_FRAME_HACK(92);
|
|
|
|
|
_NS_FRAME_HACK(93); _NS_FRAME_HACK(94); _NS_FRAME_HACK(95);
|
|
|
|
|
_NS_FRAME_HACK(96); _NS_FRAME_HACK(97); _NS_FRAME_HACK(98);
|
|
|
|
|
_NS_FRAME_HACK(99);
|
|
|
|
|
default: env->addr = NULL; break;
|
|
|
|
|
}
|
|
|
|
|
signal(SIGSEGV, env->segv);
|
|
|
|
|
signal(SIGBUS, env->bus);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
env = jbuf();
|
|
|
|
|
signal(SIGSEGV, env->segv);
|
|
|
|
|
signal(SIGBUS, env->bus);
|
|
|
|
|
env->addr = NULL;
|
|
|
|
|
}
|
|
|
|
|
return env->addr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
NSUInteger NSCountFrames(void)
|
|
|
|
|
{
|
|
|
|
|
jbuf_type *env;
|
|
|
|
|
|
|
|
|
|
env = jbuf();
|
|
|
|
|
if (sigsetjmp(env->buf, 1) == 0)
|
|
|
|
|
{
|
|
|
|
|
env->segv = signal(SIGSEGV, recover);
|
|
|
|
|
env->bus = signal(SIGBUS, recover);
|
|
|
|
|
env->addr = 0;
|
|
|
|
|
|
|
|
|
|
#define _NS_COUNT_HACK(X) if (__builtin_frame_address(X + 1) == 0) \
|
|
|
|
|
goto done; else env->addr = (void*)(X + 1);
|
|
|
|
|
|
|
|
|
|
_NS_COUNT_HACK(0); _NS_COUNT_HACK(1); _NS_COUNT_HACK(2);
|
|
|
|
|
_NS_COUNT_HACK(3); _NS_COUNT_HACK(4); _NS_COUNT_HACK(5);
|
|
|
|
|
_NS_COUNT_HACK(6); _NS_COUNT_HACK(7); _NS_COUNT_HACK(8);
|
|
|
|
|
_NS_COUNT_HACK(9); _NS_COUNT_HACK(10); _NS_COUNT_HACK(11);
|
|
|
|
|
_NS_COUNT_HACK(12); _NS_COUNT_HACK(13); _NS_COUNT_HACK(14);
|
|
|
|
|
_NS_COUNT_HACK(15); _NS_COUNT_HACK(16); _NS_COUNT_HACK(17);
|
|
|
|
|
_NS_COUNT_HACK(18); _NS_COUNT_HACK(19); _NS_COUNT_HACK(20);
|
|
|
|
|
_NS_COUNT_HACK(21); _NS_COUNT_HACK(22); _NS_COUNT_HACK(23);
|
|
|
|
|
_NS_COUNT_HACK(24); _NS_COUNT_HACK(25); _NS_COUNT_HACK(26);
|
|
|
|
|
_NS_COUNT_HACK(27); _NS_COUNT_HACK(28); _NS_COUNT_HACK(29);
|
|
|
|
|
_NS_COUNT_HACK(30); _NS_COUNT_HACK(31); _NS_COUNT_HACK(32);
|
|
|
|
|
_NS_COUNT_HACK(33); _NS_COUNT_HACK(34); _NS_COUNT_HACK(35);
|
|
|
|
|
_NS_COUNT_HACK(36); _NS_COUNT_HACK(37); _NS_COUNT_HACK(38);
|
|
|
|
|
_NS_COUNT_HACK(39); _NS_COUNT_HACK(40); _NS_COUNT_HACK(41);
|
|
|
|
|
_NS_COUNT_HACK(42); _NS_COUNT_HACK(43); _NS_COUNT_HACK(44);
|
|
|
|
|
_NS_COUNT_HACK(45); _NS_COUNT_HACK(46); _NS_COUNT_HACK(47);
|
|
|
|
|
_NS_COUNT_HACK(48); _NS_COUNT_HACK(49); _NS_COUNT_HACK(50);
|
|
|
|
|
_NS_COUNT_HACK(51); _NS_COUNT_HACK(52); _NS_COUNT_HACK(53);
|
|
|
|
|
_NS_COUNT_HACK(54); _NS_COUNT_HACK(55); _NS_COUNT_HACK(56);
|
|
|
|
|
_NS_COUNT_HACK(57); _NS_COUNT_HACK(58); _NS_COUNT_HACK(59);
|
|
|
|
|
_NS_COUNT_HACK(60); _NS_COUNT_HACK(61); _NS_COUNT_HACK(62);
|
|
|
|
|
_NS_COUNT_HACK(63); _NS_COUNT_HACK(64); _NS_COUNT_HACK(65);
|
|
|
|
|
_NS_COUNT_HACK(66); _NS_COUNT_HACK(67); _NS_COUNT_HACK(68);
|
|
|
|
|
_NS_COUNT_HACK(69); _NS_COUNT_HACK(70); _NS_COUNT_HACK(71);
|
|
|
|
|
_NS_COUNT_HACK(72); _NS_COUNT_HACK(73); _NS_COUNT_HACK(74);
|
|
|
|
|
_NS_COUNT_HACK(75); _NS_COUNT_HACK(76); _NS_COUNT_HACK(77);
|
|
|
|
|
_NS_COUNT_HACK(78); _NS_COUNT_HACK(79); _NS_COUNT_HACK(80);
|
|
|
|
|
_NS_COUNT_HACK(81); _NS_COUNT_HACK(82); _NS_COUNT_HACK(83);
|
|
|
|
|
_NS_COUNT_HACK(84); _NS_COUNT_HACK(85); _NS_COUNT_HACK(86);
|
|
|
|
|
_NS_COUNT_HACK(87); _NS_COUNT_HACK(88); _NS_COUNT_HACK(89);
|
|
|
|
|
_NS_COUNT_HACK(90); _NS_COUNT_HACK(91); _NS_COUNT_HACK(92);
|
|
|
|
|
_NS_COUNT_HACK(93); _NS_COUNT_HACK(94); _NS_COUNT_HACK(95);
|
|
|
|
|
_NS_COUNT_HACK(96); _NS_COUNT_HACK(97); _NS_COUNT_HACK(98);
|
|
|
|
|
_NS_COUNT_HACK(99);
|
|
|
|
|
|
|
|
|
|
done:
|
|
|
|
|
signal(SIGSEGV, env->segv);
|
|
|
|
|
signal(SIGBUS, env->bus);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
env = jbuf();
|
|
|
|
|
signal(SIGSEGV, env->segv);
|
|
|
|
|
signal(SIGBUS, env->bus);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return (uintptr_t)env->addr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void *
|
|
|
|
|
NSReturnAddress(NSUInteger offset)
|
|
|
|
|
{
|
|
|
|
|
jbuf_type *env;
|
|
|
|
|
|
|
|
|
|
env = jbuf();
|
|
|
|
|
if (sigsetjmp(env->buf, 1) == 0)
|
|
|
|
|
{
|
|
|
|
|
env->segv = signal(SIGSEGV, recover);
|
|
|
|
|
env->bus = signal(SIGBUS, recover);
|
|
|
|
|
switch (offset)
|
|
|
|
|
{
|
|
|
|
|
_NS_RETURN_HACK(0); _NS_RETURN_HACK(1); _NS_RETURN_HACK(2);
|
|
|
|
|
_NS_RETURN_HACK(3); _NS_RETURN_HACK(4); _NS_RETURN_HACK(5);
|
|
|
|
|
_NS_RETURN_HACK(6); _NS_RETURN_HACK(7); _NS_RETURN_HACK(8);
|
|
|
|
|
_NS_RETURN_HACK(9); _NS_RETURN_HACK(10); _NS_RETURN_HACK(11);
|
|
|
|
|
_NS_RETURN_HACK(12); _NS_RETURN_HACK(13); _NS_RETURN_HACK(14);
|
|
|
|
|
_NS_RETURN_HACK(15); _NS_RETURN_HACK(16); _NS_RETURN_HACK(17);
|
|
|
|
|
_NS_RETURN_HACK(18); _NS_RETURN_HACK(19); _NS_RETURN_HACK(20);
|
|
|
|
|
_NS_RETURN_HACK(21); _NS_RETURN_HACK(22); _NS_RETURN_HACK(23);
|
|
|
|
|
_NS_RETURN_HACK(24); _NS_RETURN_HACK(25); _NS_RETURN_HACK(26);
|
|
|
|
|
_NS_RETURN_HACK(27); _NS_RETURN_HACK(28); _NS_RETURN_HACK(29);
|
|
|
|
|
_NS_RETURN_HACK(30); _NS_RETURN_HACK(31); _NS_RETURN_HACK(32);
|
|
|
|
|
_NS_RETURN_HACK(33); _NS_RETURN_HACK(34); _NS_RETURN_HACK(35);
|
|
|
|
|
_NS_RETURN_HACK(36); _NS_RETURN_HACK(37); _NS_RETURN_HACK(38);
|
|
|
|
|
_NS_RETURN_HACK(39); _NS_RETURN_HACK(40); _NS_RETURN_HACK(41);
|
|
|
|
|
_NS_RETURN_HACK(42); _NS_RETURN_HACK(43); _NS_RETURN_HACK(44);
|
|
|
|
|
_NS_RETURN_HACK(45); _NS_RETURN_HACK(46); _NS_RETURN_HACK(47);
|
|
|
|
|
_NS_RETURN_HACK(48); _NS_RETURN_HACK(49); _NS_RETURN_HACK(50);
|
|
|
|
|
_NS_RETURN_HACK(51); _NS_RETURN_HACK(52); _NS_RETURN_HACK(53);
|
|
|
|
|
_NS_RETURN_HACK(54); _NS_RETURN_HACK(55); _NS_RETURN_HACK(56);
|
|
|
|
|
_NS_RETURN_HACK(57); _NS_RETURN_HACK(58); _NS_RETURN_HACK(59);
|
|
|
|
|
_NS_RETURN_HACK(60); _NS_RETURN_HACK(61); _NS_RETURN_HACK(62);
|
|
|
|
|
_NS_RETURN_HACK(63); _NS_RETURN_HACK(64); _NS_RETURN_HACK(65);
|
|
|
|
|
_NS_RETURN_HACK(66); _NS_RETURN_HACK(67); _NS_RETURN_HACK(68);
|
|
|
|
|
_NS_RETURN_HACK(69); _NS_RETURN_HACK(70); _NS_RETURN_HACK(71);
|
|
|
|
|
_NS_RETURN_HACK(72); _NS_RETURN_HACK(73); _NS_RETURN_HACK(74);
|
|
|
|
|
_NS_RETURN_HACK(75); _NS_RETURN_HACK(76); _NS_RETURN_HACK(77);
|
|
|
|
|
_NS_RETURN_HACK(78); _NS_RETURN_HACK(79); _NS_RETURN_HACK(80);
|
|
|
|
|
_NS_RETURN_HACK(81); _NS_RETURN_HACK(82); _NS_RETURN_HACK(83);
|
|
|
|
|
_NS_RETURN_HACK(84); _NS_RETURN_HACK(85); _NS_RETURN_HACK(86);
|
|
|
|
|
_NS_RETURN_HACK(87); _NS_RETURN_HACK(88); _NS_RETURN_HACK(89);
|
|
|
|
|
_NS_RETURN_HACK(90); _NS_RETURN_HACK(91); _NS_RETURN_HACK(92);
|
|
|
|
|
_NS_RETURN_HACK(93); _NS_RETURN_HACK(94); _NS_RETURN_HACK(95);
|
|
|
|
|
_NS_RETURN_HACK(96); _NS_RETURN_HACK(97); _NS_RETURN_HACK(98);
|
|
|
|
|
_NS_RETURN_HACK(99);
|
|
|
|
|
default: env->addr = NULL; break;
|
|
|
|
|
}
|
|
|
|
|
signal(SIGSEGV, env->segv);
|
|
|
|
|
signal(SIGBUS, env->bus);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
env = jbuf();
|
|
|
|
|
signal(SIGSEGV, env->segv);
|
|
|
|
|
signal(SIGBUS, env->bus);
|
|
|
|
|
env->addr = NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return env->addr;
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-29 11:57:47 +00:00
|
|
|
|
#ifdef __clang__
|
|
|
|
|
#pragma clang diagnostic pop
|
|
|
|
|
#else
|
|
|
|
|
#pragma GCC diagnostic pop
|
|
|
|
|
#endif
|
|
|
|
|
|
2018-04-04 11:58:06 +00:00
|
|
|
|
unsigned
|
|
|
|
|
GSPrivateReturnAddresses(NSUInteger **returns)
|
2018-03-26 13:49:13 +00:00
|
|
|
|
{
|
2018-04-04 11:58:06 +00:00
|
|
|
|
unsigned numReturns;
|
2021-03-18 20:17:35 +00:00
|
|
|
|
#if defined(_WIN32) && !defined(USE_BFD)
|
2016-03-11 16:48:31 +00:00
|
|
|
|
NSUInteger addr[MAXFRAMES];
|
|
|
|
|
|
2021-07-28 14:17:47 +00:00
|
|
|
|
GS_MUTEX_LOCK(traceLock);
|
2016-03-11 16:48:31 +00:00
|
|
|
|
if (0 == hProcess)
|
|
|
|
|
{
|
|
|
|
|
hProcess = GetCurrentProcess();
|
|
|
|
|
|
|
|
|
|
if (0 == capture)
|
|
|
|
|
{
|
|
|
|
|
HANDLE hModule;
|
|
|
|
|
|
|
|
|
|
hModule = LoadLibrary("kernel32.dll");
|
|
|
|
|
if (0 == hModule)
|
|
|
|
|
{
|
|
|
|
|
fprintf(stderr, "Failed to load kernel32.dll with error: %d\n",
|
2016-06-27 19:06:12 +00:00
|
|
|
|
(int)GetLastError());
|
2021-07-28 14:17:47 +00:00
|
|
|
|
GS_MUTEX_UNLOCK(traceLock);
|
2018-04-04 11:58:06 +00:00
|
|
|
|
return 0;
|
2016-03-11 16:48:31 +00:00
|
|
|
|
}
|
|
|
|
|
capture = (CaptureStackBackTraceType)GetProcAddress(
|
|
|
|
|
hModule, "RtlCaptureStackBackTrace");
|
|
|
|
|
if (0 == capture)
|
|
|
|
|
{
|
|
|
|
|
fprintf(stderr, "Failed to find RtlCaptureStackBackTrace: %d\n",
|
2016-06-27 19:06:12 +00:00
|
|
|
|
(int)GetLastError());
|
2021-07-28 14:17:47 +00:00
|
|
|
|
GS_MUTEX_UNLOCK(traceLock);
|
2018-04-04 11:58:06 +00:00
|
|
|
|
return 0;
|
2016-03-11 16:48:31 +00:00
|
|
|
|
}
|
|
|
|
|
hModule = LoadLibrary("dbghelp.dll");
|
|
|
|
|
if (0 == hModule)
|
|
|
|
|
{
|
|
|
|
|
fprintf(stderr, "Failed to load dbghelp.dll with error: %d\n",
|
2016-06-27 19:06:12 +00:00
|
|
|
|
(int)GetLastError());
|
2021-07-28 14:17:47 +00:00
|
|
|
|
GS_MUTEX_UNLOCK(traceLock);
|
2018-04-04 11:58:06 +00:00
|
|
|
|
return 0;
|
2016-03-11 16:48:31 +00:00
|
|
|
|
}
|
|
|
|
|
optSym = (SymSetOptionsType)GetProcAddress(
|
|
|
|
|
hModule, "SymSetOptions");
|
|
|
|
|
if (0 == optSym)
|
|
|
|
|
{
|
|
|
|
|
fprintf(stderr, "Failed to find SymSetOptions: %d\n",
|
2016-06-27 19:06:12 +00:00
|
|
|
|
(int)GetLastError());
|
2021-07-28 14:17:47 +00:00
|
|
|
|
GS_MUTEX_UNLOCK(traceLock);
|
2018-04-04 11:58:06 +00:00
|
|
|
|
return 0;
|
2016-03-11 16:48:31 +00:00
|
|
|
|
}
|
|
|
|
|
initSym = (SymInitializeType)GetProcAddress(
|
|
|
|
|
hModule, "SymInitialize");
|
|
|
|
|
if (0 == initSym)
|
|
|
|
|
{
|
|
|
|
|
fprintf(stderr, "Failed to find SymInitialize: %d\n",
|
2016-06-27 19:06:12 +00:00
|
|
|
|
(int)GetLastError());
|
2021-07-28 14:17:47 +00:00
|
|
|
|
GS_MUTEX_UNLOCK(traceLock);
|
2018-04-04 11:58:06 +00:00
|
|
|
|
return 0;
|
2016-03-11 16:48:31 +00:00
|
|
|
|
}
|
|
|
|
|
fromSym = (SymFromAddrType)GetProcAddress(
|
|
|
|
|
hModule, "SymFromAddr");
|
|
|
|
|
if (0 == fromSym)
|
|
|
|
|
{
|
|
|
|
|
fprintf(stderr, "Failed to find SymFromAddr: %d\n",
|
2016-06-27 19:06:12 +00:00
|
|
|
|
(int)GetLastError());
|
2021-07-28 14:17:47 +00:00
|
|
|
|
GS_MUTEX_UNLOCK(traceLock);
|
2018-04-04 11:58:06 +00:00
|
|
|
|
return 0;
|
2016-03-11 16:48:31 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
(optSym)(SYMOPT_UNDNAME | SYMOPT_DEFERRED_LOADS);
|
|
|
|
|
|
|
|
|
|
if (!(initSym)(hProcess, NULL, TRUE))
|
|
|
|
|
{
|
|
|
|
|
fprintf(stderr, "SymInitialize failed with error: %d\n",
|
2016-06-27 19:06:12 +00:00
|
|
|
|
(int)GetLastError());
|
2016-03-11 16:48:31 +00:00
|
|
|
|
fromSym = 0;
|
2021-07-28 14:17:47 +00:00
|
|
|
|
GS_MUTEX_UNLOCK(traceLock);
|
2018-04-04 11:58:06 +00:00
|
|
|
|
return 0;
|
2016-03-11 16:48:31 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (0 == capture)
|
|
|
|
|
{
|
2021-07-28 14:17:47 +00:00
|
|
|
|
GS_MUTEX_UNLOCK(traceLock);
|
2018-04-04 11:58:06 +00:00
|
|
|
|
return 0;
|
2016-03-11 16:48:31 +00:00
|
|
|
|
}
|
|
|
|
|
|
2018-03-26 13:49:13 +00:00
|
|
|
|
numReturns = (capture)(0, MAXFRAMES, (void**)addr, NULL);
|
|
|
|
|
if (numReturns > 0)
|
2016-03-11 16:48:31 +00:00
|
|
|
|
{
|
2024-06-21 10:45:38 +00:00
|
|
|
|
*returns = (NSUInteger*)malloc(numReturns * sizeof(NSUInteger));
|
|
|
|
|
memcpy(*returns, addr, numReturns * sizeof(NSUInteger));
|
2016-03-11 16:48:31 +00:00
|
|
|
|
}
|
2018-03-26 13:49:13 +00:00
|
|
|
|
|
2021-07-28 14:17:47 +00:00
|
|
|
|
GS_MUTEX_UNLOCK(traceLock);
|
2016-03-11 16:48:31 +00:00
|
|
|
|
|
2021-03-18 20:17:35 +00:00
|
|
|
|
#elif defined(HAVE_BACKTRACE)
|
2025-01-09 11:06:31 +00:00
|
|
|
|
void *addr[MAXFRAMES];
|
2021-03-18 20:17:35 +00:00
|
|
|
|
|
|
|
|
|
numReturns = backtrace(addr, MAXFRAMES);
|
|
|
|
|
if (numReturns > 0)
|
|
|
|
|
{
|
2024-06-21 10:45:38 +00:00
|
|
|
|
*returns = (NSUInteger*)malloc(numReturns * sizeof(NSUInteger));
|
|
|
|
|
memcpy(*returns, addr, numReturns * sizeof(NSUInteger));
|
2021-03-18 20:17:35 +00:00
|
|
|
|
}
|
|
|
|
|
#elif defined(WITH_UNWIND)
|
|
|
|
|
void *addr[MAXFRAMES];
|
|
|
|
|
|
|
|
|
|
struct GSBacktraceState state = {addr, addr + MAXFRAMES};
|
|
|
|
|
_Unwind_Backtrace(GSUnwindCallback, &state);
|
|
|
|
|
|
|
|
|
|
numReturns = state.current - addr;
|
|
|
|
|
if (numReturns > 0)
|
|
|
|
|
{
|
2024-06-21 10:45:38 +00:00
|
|
|
|
*returns = (NSUInteger*)malloc(numReturns * sizeof(NSUInteger));
|
|
|
|
|
memcpy(*returns, addr, numReturns * sizeof(NSUInteger));
|
2021-03-18 20:17:35 +00:00
|
|
|
|
}
|
2018-03-26 13:49:13 +00:00
|
|
|
|
#else
|
|
|
|
|
int n;
|
2016-03-11 16:48:31 +00:00
|
|
|
|
|
2018-03-26 13:49:13 +00:00
|
|
|
|
n = NSCountFrames();
|
|
|
|
|
/* There should be more frame addresses than return addresses.
|
|
|
|
|
*/
|
|
|
|
|
if (n > 0)
|
|
|
|
|
{
|
|
|
|
|
n--;
|
|
|
|
|
}
|
|
|
|
|
if (n > 0)
|
|
|
|
|
{
|
|
|
|
|
n--;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ((numReturns = n) > 0)
|
|
|
|
|
{
|
|
|
|
|
jbuf_type *env;
|
|
|
|
|
|
2024-06-21 10:45:38 +00:00
|
|
|
|
*returns = (NSUInteger*)malloc(numReturns * sizeof(NSUInteger));
|
2018-03-26 13:49:13 +00:00
|
|
|
|
|
|
|
|
|
env = jbuf();
|
|
|
|
|
if (sigsetjmp(env->buf, 1) == 0)
|
|
|
|
|
{
|
|
|
|
|
unsigned i;
|
|
|
|
|
|
|
|
|
|
env->segv = signal(SIGSEGV, recover);
|
|
|
|
|
env->bus = signal(SIGBUS, recover);
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < n; i++)
|
|
|
|
|
{
|
|
|
|
|
switch (i)
|
|
|
|
|
{
|
|
|
|
|
_NS_RETURN_HACK(0); _NS_RETURN_HACK(1); _NS_RETURN_HACK(2);
|
|
|
|
|
_NS_RETURN_HACK(3); _NS_RETURN_HACK(4); _NS_RETURN_HACK(5);
|
|
|
|
|
_NS_RETURN_HACK(6); _NS_RETURN_HACK(7); _NS_RETURN_HACK(8);
|
|
|
|
|
_NS_RETURN_HACK(9); _NS_RETURN_HACK(10); _NS_RETURN_HACK(11);
|
|
|
|
|
_NS_RETURN_HACK(12); _NS_RETURN_HACK(13); _NS_RETURN_HACK(14);
|
|
|
|
|
_NS_RETURN_HACK(15); _NS_RETURN_HACK(16); _NS_RETURN_HACK(17);
|
|
|
|
|
_NS_RETURN_HACK(18); _NS_RETURN_HACK(19); _NS_RETURN_HACK(20);
|
|
|
|
|
_NS_RETURN_HACK(21); _NS_RETURN_HACK(22); _NS_RETURN_HACK(23);
|
|
|
|
|
_NS_RETURN_HACK(24); _NS_RETURN_HACK(25); _NS_RETURN_HACK(26);
|
|
|
|
|
_NS_RETURN_HACK(27); _NS_RETURN_HACK(28); _NS_RETURN_HACK(29);
|
|
|
|
|
_NS_RETURN_HACK(30); _NS_RETURN_HACK(31); _NS_RETURN_HACK(32);
|
|
|
|
|
_NS_RETURN_HACK(33); _NS_RETURN_HACK(34); _NS_RETURN_HACK(35);
|
|
|
|
|
_NS_RETURN_HACK(36); _NS_RETURN_HACK(37); _NS_RETURN_HACK(38);
|
|
|
|
|
_NS_RETURN_HACK(39); _NS_RETURN_HACK(40); _NS_RETURN_HACK(41);
|
|
|
|
|
_NS_RETURN_HACK(42); _NS_RETURN_HACK(43); _NS_RETURN_HACK(44);
|
|
|
|
|
_NS_RETURN_HACK(45); _NS_RETURN_HACK(46); _NS_RETURN_HACK(47);
|
|
|
|
|
_NS_RETURN_HACK(48); _NS_RETURN_HACK(49); _NS_RETURN_HACK(50);
|
|
|
|
|
_NS_RETURN_HACK(51); _NS_RETURN_HACK(52); _NS_RETURN_HACK(53);
|
|
|
|
|
_NS_RETURN_HACK(54); _NS_RETURN_HACK(55); _NS_RETURN_HACK(56);
|
|
|
|
|
_NS_RETURN_HACK(57); _NS_RETURN_HACK(58); _NS_RETURN_HACK(59);
|
|
|
|
|
_NS_RETURN_HACK(60); _NS_RETURN_HACK(61); _NS_RETURN_HACK(62);
|
|
|
|
|
_NS_RETURN_HACK(63); _NS_RETURN_HACK(64); _NS_RETURN_HACK(65);
|
|
|
|
|
_NS_RETURN_HACK(66); _NS_RETURN_HACK(67); _NS_RETURN_HACK(68);
|
|
|
|
|
_NS_RETURN_HACK(69); _NS_RETURN_HACK(70); _NS_RETURN_HACK(71);
|
|
|
|
|
_NS_RETURN_HACK(72); _NS_RETURN_HACK(73); _NS_RETURN_HACK(74);
|
|
|
|
|
_NS_RETURN_HACK(75); _NS_RETURN_HACK(76); _NS_RETURN_HACK(77);
|
|
|
|
|
_NS_RETURN_HACK(78); _NS_RETURN_HACK(79); _NS_RETURN_HACK(80);
|
|
|
|
|
_NS_RETURN_HACK(81); _NS_RETURN_HACK(82); _NS_RETURN_HACK(83);
|
|
|
|
|
_NS_RETURN_HACK(84); _NS_RETURN_HACK(85); _NS_RETURN_HACK(86);
|
|
|
|
|
_NS_RETURN_HACK(87); _NS_RETURN_HACK(88); _NS_RETURN_HACK(89);
|
|
|
|
|
_NS_RETURN_HACK(90); _NS_RETURN_HACK(91); _NS_RETURN_HACK(92);
|
|
|
|
|
_NS_RETURN_HACK(93); _NS_RETURN_HACK(94); _NS_RETURN_HACK(95);
|
|
|
|
|
_NS_RETURN_HACK(96); _NS_RETURN_HACK(97); _NS_RETURN_HACK(98);
|
|
|
|
|
_NS_RETURN_HACK(99);
|
|
|
|
|
default: env->addr = 0; break;
|
|
|
|
|
}
|
|
|
|
|
if (env->addr == 0)
|
|
|
|
|
{
|
|
|
|
|
break;
|
|
|
|
|
}
|
2024-06-21 10:45:38 +00:00
|
|
|
|
memcpy(&(*returns)[i], env->addr, sizeof(NSUInteger));
|
2018-03-26 13:49:13 +00:00
|
|
|
|
}
|
|
|
|
|
signal(SIGSEGV, env->segv);
|
|
|
|
|
signal(SIGBUS, env->bus);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
env = jbuf();
|
|
|
|
|
signal(SIGSEGV, env->segv);
|
|
|
|
|
signal(SIGBUS, env->bus);
|
|
|
|
|
}
|
2010-01-24 17:13:03 +00:00
|
|
|
|
}
|
|
|
|
|
#endif
|
2018-04-04 11:58:06 +00:00
|
|
|
|
return numReturns;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@implementation GSStackTrace : NSObject
|
|
|
|
|
|
|
|
|
|
/** Offset from the top of the stack (when we generate a trace) to the
|
|
|
|
|
* first frame likely to be of interest for debugging.
|
|
|
|
|
*/
|
|
|
|
|
#define FrameOffset 4
|
|
|
|
|
|
|
|
|
|
+ (void) initialize
|
|
|
|
|
{
|
2024-05-07 10:24:26 +00:00
|
|
|
|
#if defined(_WIN32) || defined(USE_BFD)
|
2021-07-28 14:17:47 +00:00
|
|
|
|
GS_MUTEX_INIT_RECURSIVE(traceLock);
|
2018-04-04 11:58:06 +00:00
|
|
|
|
#endif
|
|
|
|
|
#if defined(USE_BFD)
|
2021-07-28 14:17:47 +00:00
|
|
|
|
GS_MUTEX_INIT_RECURSIVE(modLock);
|
2018-04-04 11:58:06 +00:00
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
- (NSArray*) addresses
|
|
|
|
|
{
|
|
|
|
|
if (nil == addresses && numReturns > FrameOffset)
|
|
|
|
|
{
|
2020-02-08 16:42:17 +00:00
|
|
|
|
ENTER_POOL
|
2018-04-04 11:58:06 +00:00
|
|
|
|
NSInteger count = numReturns - FrameOffset;
|
|
|
|
|
NSValue *objects[count];
|
|
|
|
|
NSUInteger index;
|
2018-04-13 06:11:15 +00:00
|
|
|
|
void **ptrs = (void **)returns;
|
2018-04-04 11:58:06 +00:00
|
|
|
|
|
|
|
|
|
for (index = 0; index < count; index++)
|
|
|
|
|
{
|
|
|
|
|
objects[index] = [NSValue valueWithPointer: ptrs[FrameOffset+index]];
|
|
|
|
|
}
|
|
|
|
|
addresses = [[NSArray alloc] initWithObjects: objects count: count];
|
2020-02-08 16:42:17 +00:00
|
|
|
|
LEAVE_POOL
|
2018-04-04 11:58:06 +00:00
|
|
|
|
}
|
|
|
|
|
return addresses;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
- (oneway void) dealloc
|
|
|
|
|
{
|
|
|
|
|
DESTROY(addresses);
|
|
|
|
|
DESTROY(symbols);
|
|
|
|
|
if (returns != NULL)
|
|
|
|
|
{
|
|
|
|
|
free(returns);
|
|
|
|
|
returns = NULL;
|
|
|
|
|
}
|
|
|
|
|
[super dealloc];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
- (NSString*) description
|
|
|
|
|
{
|
|
|
|
|
NSMutableString *result;
|
|
|
|
|
NSArray *s;
|
|
|
|
|
int i;
|
|
|
|
|
int n;
|
|
|
|
|
|
|
|
|
|
result = [NSMutableString string];
|
|
|
|
|
s = [self symbols];
|
|
|
|
|
n = [s count];
|
|
|
|
|
for (i = 0; i < n; i++)
|
|
|
|
|
{
|
|
|
|
|
NSString *line = [s objectAtIndex: i];
|
|
|
|
|
|
|
|
|
|
[result appendFormat: @"%3d: %@\n", i, line];
|
|
|
|
|
}
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
- (id) init
|
|
|
|
|
{
|
2009-09-11 19:19:05 +00:00
|
|
|
|
return self;
|
2006-05-08 15:30:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-01-24 17:13:03 +00:00
|
|
|
|
- (NSArray*) symbols
|
|
|
|
|
{
|
2018-04-13 06:11:15 +00:00
|
|
|
|
if (nil == symbols && numReturns > FrameOffset)
|
2010-01-24 17:13:03 +00:00
|
|
|
|
{
|
2018-03-26 13:49:13 +00:00
|
|
|
|
NSInteger count = numReturns - FrameOffset;
|
2018-04-13 06:11:15 +00:00
|
|
|
|
NSUInteger i;
|
2010-01-24 17:13:03 +00:00
|
|
|
|
|
2016-05-14 14:47:43 +00:00
|
|
|
|
#if defined(USE_BFD)
|
2018-04-13 06:11:15 +00:00
|
|
|
|
void **ptrs = (void**)&returns[FrameOffset];
|
|
|
|
|
NSMutableArray *a;
|
2010-01-24 17:13:03 +00:00
|
|
|
|
|
2018-04-13 06:11:15 +00:00
|
|
|
|
a = [[NSMutableArray alloc] initWithCapacity: count];
|
2006-05-08 15:30:52 +00:00
|
|
|
|
|
2018-04-13 06:11:15 +00:00
|
|
|
|
for (i = 0; i < count; i++)
|
|
|
|
|
{
|
|
|
|
|
GSFunctionInfo *aFrame = nil;
|
|
|
|
|
void *address = (void*)*ptrs++;
|
|
|
|
|
void *base;
|
|
|
|
|
NSString *modulePath;
|
|
|
|
|
GSBinaryFileInfo *bfi;
|
|
|
|
|
|
|
|
|
|
modulePath = GSPrivateBaseAddress(address, &base);
|
|
|
|
|
if (modulePath != nil && (bfi = GSLoadModule(modulePath)) != nil)
|
|
|
|
|
{
|
|
|
|
|
aFrame = [bfi functionForAddress: (void*)(address - base)];
|
|
|
|
|
if (aFrame == nil)
|
|
|
|
|
{
|
|
|
|
|
/* We know we have the right module but function lookup
|
|
|
|
|
* failed ... perhaps we need to use the absolute
|
|
|
|
|
* address rather than offest by 'base' in this case.
|
|
|
|
|
*/
|
|
|
|
|
aFrame = [bfi functionForAddress: address];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
NSArray *modules;
|
|
|
|
|
int j;
|
|
|
|
|
int m;
|
|
|
|
|
|
|
|
|
|
modules = GSListModules();
|
|
|
|
|
m = [modules count];
|
|
|
|
|
for (j = 0; j < m; j++)
|
|
|
|
|
{
|
|
|
|
|
bfi = [modules objectAtIndex: j];
|
|
|
|
|
|
|
|
|
|
if ((id)bfi != (id)[NSNull null])
|
|
|
|
|
{
|
|
|
|
|
aFrame = [bfi functionForAddress: address];
|
|
|
|
|
if (aFrame != nil)
|
|
|
|
|
{
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// not found (?!), add an 'unknown' function
|
|
|
|
|
if (aFrame == nil)
|
|
|
|
|
{
|
|
|
|
|
aFrame = [GSFunctionInfo alloc];
|
|
|
|
|
[aFrame initWithModule: nil
|
|
|
|
|
address: address
|
|
|
|
|
file: nil
|
|
|
|
|
function: nil
|
|
|
|
|
line: 0];
|
|
|
|
|
[aFrame autorelease];
|
|
|
|
|
}
|
|
|
|
|
[a addObject: [aFrame description]];
|
|
|
|
|
}
|
|
|
|
|
symbols = [a copy];
|
|
|
|
|
[a release];
|
2016-03-11 16:48:31 +00:00
|
|
|
|
#elif defined(_WIN32)
|
2018-04-13 06:11:15 +00:00
|
|
|
|
void **ptrs = (void**)&returns[FrameOffset];
|
|
|
|
|
SYMBOL_INFO *symbol;
|
|
|
|
|
NSString *syms[MAXFRAMES];
|
2016-03-11 16:48:31 +00:00
|
|
|
|
|
2018-04-13 06:11:15 +00:00
|
|
|
|
symbol = (SYMBOL_INFO *)calloc(sizeof(SYMBOL_INFO)
|
|
|
|
|
+ 1024 * sizeof(char), 1);
|
|
|
|
|
symbol->MaxNameLen = 1024;
|
|
|
|
|
symbol->SizeOfStruct = sizeof(SYMBOL_INFO);
|
2016-03-11 16:48:31 +00:00
|
|
|
|
|
2021-07-28 14:17:47 +00:00
|
|
|
|
GS_MUTEX_LOCK(traceLock);
|
2018-04-13 06:11:15 +00:00
|
|
|
|
for (i = 0; i < count; i++)
|
|
|
|
|
{
|
|
|
|
|
NSUInteger addr = (NSUInteger)*ptrs++;
|
2016-03-11 16:48:31 +00:00
|
|
|
|
|
2018-04-13 06:11:15 +00:00
|
|
|
|
if ((fromSym)(hProcess, (DWORD64)addr, 0, symbol))
|
|
|
|
|
{
|
|
|
|
|
syms[i] = [NSString stringWithFormat:
|
2021-08-10 16:49:29 +00:00
|
|
|
|
@"%s - %lx", symbol->Name, (unsigned long)addr];
|
2018-04-13 06:11:15 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
syms[i] = [NSString stringWithFormat:
|
2021-08-10 16:49:29 +00:00
|
|
|
|
@"unknown - %lx", (unsigned long)addr];
|
2018-04-13 06:11:15 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2021-07-28 14:17:47 +00:00
|
|
|
|
GS_MUTEX_UNLOCK(traceLock);
|
2018-04-13 06:11:15 +00:00
|
|
|
|
free(symbol);
|
|
|
|
|
|
|
|
|
|
symbols = [[NSArray alloc] initWithObjects: syms count: count];
|
2016-03-11 16:48:31 +00:00
|
|
|
|
#elif defined(HAVE_BACKTRACE)
|
2018-04-13 06:11:15 +00:00
|
|
|
|
void **ptrs = (void**)&returns[FrameOffset];
|
|
|
|
|
char **strs;
|
|
|
|
|
NSString **symbolArray;
|
2016-03-11 16:48:31 +00:00
|
|
|
|
|
2018-04-13 06:11:15 +00:00
|
|
|
|
strs = backtrace_symbols(ptrs, count);
|
|
|
|
|
symbolArray = alloca(count * sizeof(NSString*));
|
|
|
|
|
for (i = 0; i < count; i++)
|
|
|
|
|
{
|
|
|
|
|
symbolArray[i] = [NSString stringWithUTF8String: strs[i]];
|
|
|
|
|
}
|
|
|
|
|
symbols = [[NSArray alloc] initWithObjects: symbolArray count: count];
|
|
|
|
|
free(strs);
|
2021-01-18 13:20:14 +00:00
|
|
|
|
#elif defined(WITH_UNWIND)
|
2019-06-13 15:24:28 +00:00
|
|
|
|
void **ptrs = (void**)&returns[FrameOffset];
|
|
|
|
|
NSString **symbolArray;
|
|
|
|
|
|
|
|
|
|
symbolArray = alloca(count * sizeof(NSString*));
|
|
|
|
|
for (i = 0; i < count; i++)
|
|
|
|
|
{
|
|
|
|
|
const void *addr = ptrs[i];
|
|
|
|
|
Dl_info info;
|
|
|
|
|
if (dladdr(addr, &info)) {
|
|
|
|
|
const char *libname = "unknown";
|
|
|
|
|
if (info.dli_fname) {
|
|
|
|
|
// strip library path
|
|
|
|
|
char *delim = strrchr(info.dli_fname, '/');
|
|
|
|
|
libname = delim ? delim + 1 : info.dli_fname;
|
|
|
|
|
}
|
|
|
|
|
if (info.dli_sname) {
|
|
|
|
|
symbolArray[i] = [NSString stringWithFormat:
|
|
|
|
|
@"%lu: %p %s %s + %d", (unsigned long)i, addr, libname,
|
|
|
|
|
info.dli_sname, (int)(addr - info.dli_saddr)];
|
|
|
|
|
} else {
|
|
|
|
|
symbolArray[i] = [NSString stringWithFormat:
|
|
|
|
|
@"%lu: %p %s unknown", (unsigned long)i, addr, libname];
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
symbolArray[i] = [NSString stringWithFormat:
|
|
|
|
|
@"%lu: %p unknown", (unsigned long)i, addr];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
symbols = [[NSArray alloc] initWithObjects: symbolArray count: count];
|
2018-04-13 06:11:15 +00:00
|
|
|
|
#else
|
|
|
|
|
NSMutableArray *a;
|
|
|
|
|
|
|
|
|
|
symbols = a = [[self addresses] mutableCopy];
|
|
|
|
|
for (i = 0; i < count; i++)
|
|
|
|
|
{
|
|
|
|
|
NSString *s;
|
|
|
|
|
|
2019-06-13 15:24:28 +00:00
|
|
|
|
s = [[NSString alloc] initWithFormat: @"%p: symbol not available",
|
|
|
|
|
[[a objectAtIndex: i] pointerValue]];
|
2018-04-13 06:11:15 +00:00
|
|
|
|
[a replaceObjectAtIndex: i withObject: s];
|
|
|
|
|
RELEASE(s);
|
|
|
|
|
}
|
2013-04-14 09:04:40 +00:00
|
|
|
|
#endif
|
2010-01-24 17:13:03 +00:00
|
|
|
|
}
|
|
|
|
|
return symbols;
|
|
|
|
|
}
|
|
|
|
|
|
2018-04-04 11:58:06 +00:00
|
|
|
|
- (void) trace
|
|
|
|
|
{
|
|
|
|
|
DESTROY(addresses);
|
|
|
|
|
DESTROY(symbols);
|
|
|
|
|
if (returns != NULL)
|
|
|
|
|
{
|
|
|
|
|
free(returns);
|
|
|
|
|
returns = NULL;
|
|
|
|
|
}
|
|
|
|
|
numReturns = GSPrivateReturnAddresses(&returns);
|
|
|
|
|
}
|
|
|
|
|
|
2010-01-24 17:13:03 +00:00
|
|
|
|
@end
|
|
|
|
|
|
2006-05-08 15:30:52 +00:00
|
|
|
|
|
2021-01-18 13:20:14 +00:00
|
|
|
|
GS_DECLARE NSString* const NSCharacterConversionException
|
2008-03-17 05:45:55 +00:00
|
|
|
|
= @"NSCharacterConversionException";
|
|
|
|
|
|
2021-01-18 13:20:14 +00:00
|
|
|
|
GS_DECLARE NSString* const NSGenericException
|
2002-06-09 11:36:13 +00:00
|
|
|
|
= @"NSGenericException";
|
|
|
|
|
|
2021-01-18 13:20:14 +00:00
|
|
|
|
GS_DECLARE NSString* const NSInternalInconsistencyException
|
2002-06-09 11:36:13 +00:00
|
|
|
|
= @"NSInternalInconsistencyException";
|
|
|
|
|
|
2021-01-18 13:20:14 +00:00
|
|
|
|
GS_DECLARE NSString* const NSInvalidArgumentException
|
2002-06-09 11:36:13 +00:00
|
|
|
|
= @"NSInvalidArgumentException";
|
|
|
|
|
|
2021-01-18 13:20:14 +00:00
|
|
|
|
GS_DECLARE NSString* const NSMallocException
|
2002-06-09 11:36:13 +00:00
|
|
|
|
= @"NSMallocException";
|
|
|
|
|
|
2021-01-18 13:20:14 +00:00
|
|
|
|
GS_DECLARE NSString* const NSOldStyleException
|
2008-03-17 05:45:55 +00:00
|
|
|
|
= @"NSOldStyleException";
|
2002-06-09 11:36:13 +00:00
|
|
|
|
|
2021-01-18 13:20:14 +00:00
|
|
|
|
GS_DECLARE NSString* const NSParseErrorException
|
2002-06-09 11:36:13 +00:00
|
|
|
|
= @"NSParseErrorException";
|
|
|
|
|
|
2021-01-18 13:20:14 +00:00
|
|
|
|
GS_DECLARE NSString* const NSRangeException
|
2008-03-17 05:45:55 +00:00
|
|
|
|
= @"NSRangeException";
|
|
|
|
|
|
2005-07-01 08:53:32 +00:00
|
|
|
|
static void _terminate()
|
1995-03-18 17:15:15 +00:00
|
|
|
|
{
|
2010-01-24 17:13:03 +00:00
|
|
|
|
BOOL shouldAbort;
|
2001-01-25 08:05:52 +00:00
|
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
2005-07-01 08:53:32 +00:00
|
|
|
|
shouldAbort = YES; // abort() by default.
|
2001-01-25 08:05:52 +00:00
|
|
|
|
#else
|
2005-07-01 08:53:32 +00:00
|
|
|
|
shouldAbort = NO; // exit() by default.
|
2001-01-25 08:05:52 +00:00
|
|
|
|
#endif
|
2006-10-20 10:56:27 +00:00
|
|
|
|
shouldAbort = GSPrivateEnvironmentFlag("CRASH_ON_ABORT", shouldAbort);
|
2005-07-01 08:53:32 +00:00
|
|
|
|
if (shouldAbort == YES)
|
2001-01-25 08:05:52 +00:00
|
|
|
|
{
|
|
|
|
|
abort();
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
exit(1);
|
|
|
|
|
}
|
1995-03-18 17:15:15 +00:00
|
|
|
|
}
|
|
|
|
|
|
2005-07-01 08:53:32 +00:00
|
|
|
|
static void
|
|
|
|
|
_NSFoundationUncaughtExceptionHandler (NSException *exception)
|
|
|
|
|
{
|
2011-02-28 19:49:57 +00:00
|
|
|
|
NSAutoreleasePool *pool = [NSAutoreleasePool new];
|
|
|
|
|
|
2006-10-09 14:00:01 +00:00
|
|
|
|
fprintf(stderr, "%s: Uncaught exception %s, reason: %s\n",
|
2006-10-20 10:56:27 +00:00
|
|
|
|
GSPrivateArgZero(),
|
2005-07-01 08:53:32 +00:00
|
|
|
|
[[exception name] lossyCString], [[exception reason] lossyCString]);
|
|
|
|
|
fflush(stderr); /* NEEDED UNDER MINGW */
|
2018-06-12 15:43:15 +00:00
|
|
|
|
if (GSPrivateEnvironmentFlag("GNUSTEP_STACK_TRACE", NO) == YES
|
|
|
|
|
|| GSPrivateDefaultsFlag(GSExceptionStackTrace) == YES)
|
2007-02-04 08:43:16 +00:00
|
|
|
|
{
|
2009-09-11 19:19:05 +00:00
|
|
|
|
fprintf(stderr, "Stack\n%s\n",
|
|
|
|
|
[[[exception _callStack] description] lossyCString]);
|
2007-02-04 08:43:16 +00:00
|
|
|
|
}
|
|
|
|
|
fflush(stderr); /* NEEDED UNDER MINGW */
|
2011-05-27 11:48:44 +00:00
|
|
|
|
[pool drain];
|
2005-07-01 08:53:32 +00:00
|
|
|
|
_terminate();
|
|
|
|
|
}
|
|
|
|
|
|
2008-11-28 15:38:48 +00:00
|
|
|
|
static void
|
|
|
|
|
callUncaughtHandler(id value)
|
|
|
|
|
{
|
|
|
|
|
if (_NSUncaughtExceptionHandler != NULL)
|
|
|
|
|
{
|
|
|
|
|
(*_NSUncaughtExceptionHandler)(value);
|
|
|
|
|
}
|
2010-02-08 20:24:12 +00:00
|
|
|
|
/* The uncaught exception handler which is set has not exited,
|
|
|
|
|
* so we MUST call the builtin handler, (normal behavior of MacOS-X).
|
|
|
|
|
* The standard handler is guaranteed to exit/abort, which is the
|
|
|
|
|
* required behavior for OSX compatibility.
|
|
|
|
|
* NB Cocoa's Exception Handling framework might bypass this behavior
|
|
|
|
|
* somehow (it's not clear if it does that or simply wraps various
|
|
|
|
|
* things with its own exception handlers thus preventing the
|
|
|
|
|
* uncaught handler from ever being needed) ... if anyone contributes
|
|
|
|
|
* an implementation, perhaps we could integrate it here.
|
|
|
|
|
*/
|
2010-02-08 17:52:36 +00:00
|
|
|
|
_NSFoundationUncaughtExceptionHandler(value);
|
2008-11-28 15:38:48 +00:00
|
|
|
|
}
|
|
|
|
|
|
1995-03-18 17:15:15 +00:00
|
|
|
|
@implementation NSException
|
|
|
|
|
|
2007-02-06 11:47:32 +00:00
|
|
|
|
+ (void) initialize
|
|
|
|
|
{
|
2018-03-28 08:28:56 +00:00
|
|
|
|
if (self == [NSException class])
|
2010-01-24 17:13:03 +00:00
|
|
|
|
{
|
2010-12-23 02:23:05 +00:00
|
|
|
|
#if defined(_NATIVE_OBJC_EXCEPTIONS)
|
|
|
|
|
# ifdef HAVE_SET_UNCAUGHT_EXCEPTION_HANDLER
|
2018-03-28 08:28:56 +00:00
|
|
|
|
objc_setUncaughtExceptionHandler(callUncaughtHandler);
|
2010-12-23 02:23:05 +00:00
|
|
|
|
# elif defined(HAVE_SET_UNEXPECTED)
|
2018-03-28 08:28:56 +00:00
|
|
|
|
objc_set_unexpected(callUncaughtHandler);
|
2024-02-12 06:10:44 +00:00
|
|
|
|
# elif defined(HAVE_UNEXPECTED)
|
|
|
|
|
_objc_unexpected_exception = callUncaughtHandler;
|
2010-12-23 02:23:05 +00:00
|
|
|
|
# endif
|
2008-11-28 15:38:48 +00:00
|
|
|
|
#endif
|
2018-03-28 08:28:56 +00:00
|
|
|
|
}
|
2008-11-28 15:38:48 +00:00
|
|
|
|
}
|
2007-02-06 11:47:32 +00:00
|
|
|
|
|
2000-04-05 14:21:05 +00:00
|
|
|
|
+ (NSException*) exceptionWithName: (NSString*)name
|
|
|
|
|
reason: (NSString*)reason
|
|
|
|
|
userInfo: (NSDictionary*)userInfo
|
1995-03-18 17:15:15 +00:00
|
|
|
|
{
|
1999-04-21 20:16:25 +00:00
|
|
|
|
return AUTORELEASE([[self alloc] initWithName: name reason: reason
|
2002-01-03 10:01:25 +00:00
|
|
|
|
userInfo: userInfo]);
|
1995-03-18 17:15:15 +00:00
|
|
|
|
}
|
|
|
|
|
|
2000-04-05 14:21:05 +00:00
|
|
|
|
+ (void) raise: (NSString*)name
|
|
|
|
|
format: (NSString*)format,...
|
1995-03-18 17:15:15 +00:00
|
|
|
|
{
|
1999-04-21 20:16:25 +00:00
|
|
|
|
va_list args;
|
1995-03-18 17:15:15 +00:00
|
|
|
|
|
1999-04-21 20:16:25 +00:00
|
|
|
|
va_start(args, format);
|
|
|
|
|
[self raise: name format: format arguments: args];
|
2002-10-27 01:38:30 +00:00
|
|
|
|
// This probably doesn't matter, but va_end won't get called
|
1999-04-21 20:16:25 +00:00
|
|
|
|
va_end(args);
|
2018-01-25 10:05:52 +00:00
|
|
|
|
while (1); // does not return
|
1995-03-18 17:15:15 +00:00
|
|
|
|
}
|
|
|
|
|
|
2000-04-05 14:21:05 +00:00
|
|
|
|
+ (void) raise: (NSString*)name
|
|
|
|
|
format: (NSString*)format
|
|
|
|
|
arguments: (va_list)argList
|
1995-03-18 17:15:15 +00:00
|
|
|
|
{
|
1999-04-21 20:16:25 +00:00
|
|
|
|
NSString *reason;
|
|
|
|
|
NSException *except;
|
|
|
|
|
|
|
|
|
|
reason = [NSString stringWithFormat: format arguments: argList];
|
|
|
|
|
except = [self exceptionWithName: name reason: reason userInfo: nil];
|
|
|
|
|
[except raise];
|
2018-01-25 10:05:52 +00:00
|
|
|
|
while (1); // does not return
|
1995-03-18 17:15:15 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-01-24 17:13:03 +00:00
|
|
|
|
/* For OSX compatibility -init returns nil.
|
|
|
|
|
*/
|
|
|
|
|
- (id) init
|
|
|
|
|
{
|
2010-02-25 18:49:31 +00:00
|
|
|
|
DESTROY(self);
|
2010-01-24 17:13:03 +00:00
|
|
|
|
return nil;
|
|
|
|
|
}
|
|
|
|
|
|
2000-04-05 14:21:05 +00:00
|
|
|
|
- (id) initWithName: (NSString*)name
|
|
|
|
|
reason: (NSString*)reason
|
|
|
|
|
userInfo: (NSDictionary*)userInfo
|
1995-03-18 17:15:15 +00:00
|
|
|
|
{
|
1999-09-16 07:21:34 +00:00
|
|
|
|
ASSIGN(_e_name, name);
|
|
|
|
|
ASSIGN(_e_reason, reason);
|
2007-12-05 16:13:24 +00:00
|
|
|
|
if (userInfo != nil)
|
|
|
|
|
{
|
2010-01-24 17:13:03 +00:00
|
|
|
|
if (_reserved == 0)
|
|
|
|
|
{
|
|
|
|
|
_reserved = NSZoneCalloc([self zone], 2, sizeof(id));
|
|
|
|
|
}
|
2007-12-05 16:13:24 +00:00
|
|
|
|
ASSIGN(_e_info, userInfo);
|
|
|
|
|
}
|
1999-04-21 20:16:25 +00:00
|
|
|
|
return self;
|
1995-03-18 17:15:15 +00:00
|
|
|
|
}
|
2010-01-24 06:53:49 +00:00
|
|
|
|
|
2007-12-05 16:13:24 +00:00
|
|
|
|
- (NSArray*) callStackReturnAddresses
|
|
|
|
|
{
|
2010-01-24 17:13:03 +00:00
|
|
|
|
if (_reserved == 0)
|
|
|
|
|
{
|
|
|
|
|
return nil;
|
|
|
|
|
}
|
2009-09-11 16:14:45 +00:00
|
|
|
|
return [_e_stack addresses];
|
|
|
|
|
}
|
2010-01-24 17:13:03 +00:00
|
|
|
|
|
2009-09-11 19:19:05 +00:00
|
|
|
|
- (NSArray *) callStackSymbols
|
2009-09-11 16:14:45 +00:00
|
|
|
|
{
|
2010-01-24 17:13:03 +00:00
|
|
|
|
if (_reserved == 0)
|
|
|
|
|
{
|
|
|
|
|
return nil;
|
|
|
|
|
}
|
2009-09-11 19:19:05 +00:00
|
|
|
|
return [_e_stack symbols];
|
2009-09-11 16:14:45 +00:00
|
|
|
|
}
|
2007-12-05 16:13:24 +00:00
|
|
|
|
|
2000-04-05 14:21:05 +00:00
|
|
|
|
- (void) dealloc
|
1997-09-13 17:52:31 +00:00
|
|
|
|
{
|
1999-09-16 07:21:34 +00:00
|
|
|
|
DESTROY(_e_name);
|
|
|
|
|
DESTROY(_e_reason);
|
2007-12-05 16:13:24 +00:00
|
|
|
|
if (_reserved != 0)
|
|
|
|
|
{
|
|
|
|
|
DESTROY(_e_info);
|
|
|
|
|
DESTROY(_e_stack);
|
|
|
|
|
NSZoneFree([self zone], _reserved);
|
|
|
|
|
_reserved = 0;
|
|
|
|
|
}
|
1997-09-13 17:52:31 +00:00
|
|
|
|
[super dealloc];
|
|
|
|
|
}
|
|
|
|
|
|
2010-03-14 05:37:49 +00:00
|
|
|
|
- (NSString*) description
|
|
|
|
|
{
|
2011-02-28 19:49:57 +00:00
|
|
|
|
NSAutoreleasePool *pool = [NSAutoreleasePool new];
|
|
|
|
|
NSString *result;
|
2010-03-14 05:37:49 +00:00
|
|
|
|
|
|
|
|
|
if (_e_name == nil)
|
|
|
|
|
{
|
|
|
|
|
[NSException raise: NSInvalidArgumentException
|
|
|
|
|
format: @"Atttempt to use uninitialised NSException"];
|
|
|
|
|
}
|
|
|
|
|
if (_reserved != 0)
|
|
|
|
|
{
|
|
|
|
|
if (_e_stack != nil
|
2018-06-12 15:43:15 +00:00
|
|
|
|
&& (GSPrivateEnvironmentFlag("GNUSTEP_STACK_TRACE", NO) == YES
|
|
|
|
|
|| GSPrivateDefaultsFlag(GSExceptionStackTrace) == YES))
|
2010-03-14 05:37:49 +00:00
|
|
|
|
{
|
|
|
|
|
if (_e_info != nil)
|
|
|
|
|
{
|
|
|
|
|
result = [NSString stringWithFormat:
|
|
|
|
|
@"%@ NAME:%@ REASON:%@ INFO:%@ STACK:%@",
|
|
|
|
|
[super description], _e_name, _e_reason, _e_info, _e_stack];
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
result = [NSString stringWithFormat:
|
|
|
|
|
@"%@ NAME:%@ REASON:%@ STACK:%@",
|
|
|
|
|
[super description], _e_name, _e_reason, _e_stack];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
result = [NSString stringWithFormat:
|
|
|
|
|
@"%@ NAME:%@ REASON:%@ INFO:%@",
|
|
|
|
|
[super description], _e_name, _e_reason, _e_info];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
result = [NSString stringWithFormat: @"%@ NAME:%@ REASON:%@",
|
|
|
|
|
[super description], _e_name, _e_reason];
|
|
|
|
|
}
|
2011-02-28 19:49:57 +00:00
|
|
|
|
[result retain];
|
2011-05-27 11:48:44 +00:00
|
|
|
|
[pool drain];
|
2011-02-28 19:49:57 +00:00
|
|
|
|
return [result autorelease];
|
2010-03-14 05:37:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
2000-04-05 14:21:05 +00:00
|
|
|
|
- (void) raise
|
1995-03-18 17:15:15 +00:00
|
|
|
|
{
|
2007-12-05 16:13:24 +00:00
|
|
|
|
if (_reserved == 0)
|
|
|
|
|
{
|
|
|
|
|
_reserved = NSZoneCalloc([self zone], 2, sizeof(id));
|
|
|
|
|
}
|
2012-01-03 09:34:10 +00:00
|
|
|
|
if (nil == _e_stack)
|
|
|
|
|
{
|
|
|
|
|
// Only set the stack when first raised
|
|
|
|
|
_e_stack = [GSStackTrace new];
|
2018-04-04 11:58:06 +00:00
|
|
|
|
[_e_stack trace];
|
2012-01-03 09:34:10 +00:00
|
|
|
|
}
|
2008-10-29 06:19:17 +00:00
|
|
|
|
|
2010-03-08 12:07:11 +00:00
|
|
|
|
#if defined(_NATIVE_OBJC_EXCEPTIONS)
|
2006-05-08 15:30:52 +00:00
|
|
|
|
@throw self;
|
|
|
|
|
#else
|
2010-03-08 12:07:11 +00:00
|
|
|
|
{
|
|
|
|
|
NSThread *thread;
|
|
|
|
|
NSHandler *handler;
|
|
|
|
|
|
2010-02-24 10:23:47 +00:00
|
|
|
|
thread = GSCurrentThread();
|
1999-04-21 20:16:25 +00:00
|
|
|
|
handler = thread->_exception_handler;
|
2011-02-10 10:52:54 +00:00
|
|
|
|
if (NULL == handler)
|
1999-04-21 20:16:25 +00:00
|
|
|
|
{
|
2007-05-15 12:53:33 +00:00
|
|
|
|
static int recursion = 0;
|
2005-07-01 08:53:32 +00:00
|
|
|
|
|
|
|
|
|
/*
|
2007-05-15 12:53:33 +00:00
|
|
|
|
* Set/check a counter to prevent recursive uncaught exceptions.
|
|
|
|
|
* Allow a little recursion in case we have different handlers
|
|
|
|
|
* being tried.
|
2005-07-01 08:53:32 +00:00
|
|
|
|
*/
|
2007-05-15 12:53:33 +00:00
|
|
|
|
if (recursion++ > 3)
|
2005-07-01 08:53:32 +00:00
|
|
|
|
{
|
|
|
|
|
fprintf(stderr,
|
|
|
|
|
"recursion encountered handling uncaught exception\n");
|
|
|
|
|
fflush(stderr); /* NEEDED UNDER MINGW */
|
|
|
|
|
_terminate();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Call the uncaught exception handler (if there is one).
|
2010-02-08 20:24:12 +00:00
|
|
|
|
* The calls the built-in default handler to terminate the program!
|
2005-07-01 08:53:32 +00:00
|
|
|
|
*/
|
2008-11-28 15:38:48 +00:00
|
|
|
|
callUncaughtHandler(self);
|
1999-04-21 20:16:25 +00:00
|
|
|
|
}
|
2011-02-10 10:52:54 +00:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
thread->_exception_handler = handler->next;
|
|
|
|
|
handler->exception = self;
|
|
|
|
|
longjmp(handler->jumpState, 1);
|
|
|
|
|
}
|
2010-03-08 12:07:11 +00:00
|
|
|
|
}
|
2006-04-26 14:15:03 +00:00
|
|
|
|
#endif
|
2018-01-25 10:05:52 +00:00
|
|
|
|
while (1); // does not return
|
1995-03-18 17:15:15 +00:00
|
|
|
|
}
|
|
|
|
|
|
2000-04-05 14:21:05 +00:00
|
|
|
|
- (NSString*) name
|
1995-03-18 17:15:15 +00:00
|
|
|
|
{
|
2000-09-07 16:53:14 +00:00
|
|
|
|
if (_e_name != nil)
|
|
|
|
|
{
|
|
|
|
|
return _e_name;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
return NSStringFromClass([self class]);
|
|
|
|
|
}
|
1995-03-18 17:15:15 +00:00
|
|
|
|
}
|
|
|
|
|
|
2000-04-05 14:21:05 +00:00
|
|
|
|
- (NSString*) reason
|
1995-03-18 17:15:15 +00:00
|
|
|
|
{
|
2000-09-07 16:53:14 +00:00
|
|
|
|
if (_e_reason != nil)
|
|
|
|
|
{
|
|
|
|
|
return _e_reason;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
return @"unspecified reason";
|
|
|
|
|
}
|
1995-03-18 17:15:15 +00:00
|
|
|
|
}
|
|
|
|
|
|
2000-04-05 14:21:05 +00:00
|
|
|
|
- (NSDictionary*) userInfo
|
1995-03-18 17:15:15 +00:00
|
|
|
|
{
|
2007-12-05 16:13:24 +00:00
|
|
|
|
if (_reserved == 0)
|
|
|
|
|
{
|
|
|
|
|
return nil;
|
|
|
|
|
}
|
1999-09-16 07:21:34 +00:00
|
|
|
|
return _e_info;
|
1995-03-18 17:15:15 +00:00
|
|
|
|
}
|
|
|
|
|
|
1998-10-17 18:56:49 +00:00
|
|
|
|
- (Class) classForPortCoder
|
|
|
|
|
{
|
|
|
|
|
return [self class];
|
|
|
|
|
}
|
|
|
|
|
|
1999-04-21 20:16:25 +00:00
|
|
|
|
- (id) replacementObjectForPortCoder: (NSPortCoder*)aCoder
|
1998-10-17 18:56:49 +00:00
|
|
|
|
{
|
1999-04-21 20:16:25 +00:00
|
|
|
|
return self;
|
1998-10-17 18:56:49 +00:00
|
|
|
|
}
|
1995-03-18 17:15:15 +00:00
|
|
|
|
|
1999-04-21 20:16:25 +00:00
|
|
|
|
- (void) encodeWithCoder: (NSCoder*)aCoder
|
1995-03-18 17:15:15 +00:00
|
|
|
|
{
|
2007-12-05 16:13:24 +00:00
|
|
|
|
id info = (_reserved == 0) ? nil : _e_info;
|
|
|
|
|
|
1999-09-16 07:21:34 +00:00
|
|
|
|
[aCoder encodeValueOfObjCType: @encode(id) at: &_e_name];
|
|
|
|
|
[aCoder encodeValueOfObjCType: @encode(id) at: &_e_reason];
|
2007-12-05 16:13:24 +00:00
|
|
|
|
[aCoder encodeValueOfObjCType: @encode(id) at: &info];
|
1995-03-18 17:15:15 +00:00
|
|
|
|
}
|
|
|
|
|
|
1999-04-21 20:16:25 +00:00
|
|
|
|
- (id) initWithCoder: (NSCoder*)aDecoder
|
1995-03-18 17:15:15 +00:00
|
|
|
|
{
|
2007-12-05 16:13:24 +00:00
|
|
|
|
id info;
|
|
|
|
|
|
1999-09-16 07:21:34 +00:00
|
|
|
|
[aDecoder decodeValueOfObjCType: @encode(id) at: &_e_name];
|
|
|
|
|
[aDecoder decodeValueOfObjCType: @encode(id) at: &_e_reason];
|
2007-12-05 16:13:24 +00:00
|
|
|
|
[aDecoder decodeValueOfObjCType: @encode(id) at: &info];
|
|
|
|
|
if (info != nil)
|
|
|
|
|
{
|
|
|
|
|
if (_reserved == 0)
|
|
|
|
|
{
|
|
|
|
|
_reserved = NSZoneCalloc([self zone], 2, sizeof(id));
|
|
|
|
|
}
|
|
|
|
|
_e_info = info;
|
|
|
|
|
}
|
1999-04-21 20:16:25 +00:00
|
|
|
|
return self;
|
1995-03-18 17:15:15 +00:00
|
|
|
|
}
|
|
|
|
|
|
2000-04-05 14:21:05 +00:00
|
|
|
|
- (id) copyWithZone: (NSZone*)zone
|
1995-03-18 17:15:15 +00:00
|
|
|
|
{
|
1999-04-21 20:16:25 +00:00
|
|
|
|
if (NSShouldRetainWithZone(self, zone))
|
2007-12-05 16:13:24 +00:00
|
|
|
|
{
|
|
|
|
|
return RETAIN(self);
|
|
|
|
|
}
|
1999-04-21 20:16:25 +00:00
|
|
|
|
else
|
2007-12-05 16:13:24 +00:00
|
|
|
|
{
|
|
|
|
|
return [[[self class] alloc] initWithName: [self name]
|
|
|
|
|
reason: [self reason]
|
|
|
|
|
userInfo: [self userInfo]];
|
|
|
|
|
}
|
1995-03-18 17:15:15 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
2010-01-24 17:13:03 +00:00
|
|
|
|
@implementation NSException (GSPrivate)
|
|
|
|
|
|
|
|
|
|
- (GSStackTrace*) _callStack
|
|
|
|
|
{
|
|
|
|
|
if (_reserved == 0)
|
|
|
|
|
{
|
|
|
|
|
return nil;
|
|
|
|
|
}
|
|
|
|
|
return _e_stack;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
2013-10-18 02:00:54 +00:00
|
|
|
|
@implementation NSThread (CallStackSymbols)
|
|
|
|
|
|
|
|
|
|
+ (NSArray *) callStackSymbols
|
|
|
|
|
{
|
2018-04-04 11:58:06 +00:00
|
|
|
|
GSStackTrace *stackTrace = AUTORELEASE([GSStackTrace new]);
|
|
|
|
|
[stackTrace trace];
|
|
|
|
|
return [stackTrace symbols];
|
2013-10-18 02:00:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@end
|
1995-03-18 17:15:15 +00:00
|
|
|
|
|
1999-04-21 20:16:25 +00:00
|
|
|
|
void
|
2000-04-05 14:21:05 +00:00
|
|
|
|
_NSAddHandler (NSHandler* handler)
|
1995-03-18 17:15:15 +00:00
|
|
|
|
{
|
2010-02-24 10:23:47 +00:00
|
|
|
|
NSThread *thread;
|
1995-03-18 17:15:15 +00:00
|
|
|
|
|
2010-02-24 10:23:47 +00:00
|
|
|
|
thread = GSCurrentThread();
|
2016-03-09 13:16:16 +00:00
|
|
|
|
#if defined(_WIN32) && defined(DEBUG)
|
2005-10-23 15:11:19 +00:00
|
|
|
|
if (thread->_exception_handler
|
|
|
|
|
&& IsBadReadPtr(thread->_exception_handler, sizeof(NSHandler)))
|
|
|
|
|
{
|
2005-10-23 21:30:24 +00:00
|
|
|
|
fprintf(stderr, "ERROR: Current exception handler is bogus.\n");
|
2005-10-23 15:11:19 +00:00
|
|
|
|
}
|
|
|
|
|
#endif
|
1999-04-19 14:29:52 +00:00
|
|
|
|
handler->next = thread->_exception_handler;
|
|
|
|
|
thread->_exception_handler = handler;
|
1995-03-18 17:15:15 +00:00
|
|
|
|
}
|
|
|
|
|
|
1999-04-21 20:16:25 +00:00
|
|
|
|
void
|
2000-04-05 14:21:05 +00:00
|
|
|
|
_NSRemoveHandler (NSHandler* handler)
|
1995-03-18 17:15:15 +00:00
|
|
|
|
{
|
2010-02-24 10:23:47 +00:00
|
|
|
|
NSThread *thread;
|
1995-03-18 17:15:15 +00:00
|
|
|
|
|
2010-02-24 10:23:47 +00:00
|
|
|
|
thread = GSCurrentThread();
|
2005-10-23 15:11:19 +00:00
|
|
|
|
#if defined(DEBUG)
|
|
|
|
|
if (thread->_exception_handler != handler)
|
|
|
|
|
{
|
2005-10-23 15:14:35 +00:00
|
|
|
|
fprintf(stderr, "ERROR: Removing exception handler that is not on top "
|
2005-10-23 21:30:24 +00:00
|
|
|
|
"of the stack. (You probably called return in an NS_DURING block.)\n");
|
2005-10-23 15:11:19 +00:00
|
|
|
|
}
|
2016-03-09 13:16:16 +00:00
|
|
|
|
#if defined(_WIN32)
|
2005-10-23 15:11:19 +00:00
|
|
|
|
if (IsBadReadPtr(handler, sizeof(NSHandler)))
|
|
|
|
|
{
|
2005-10-23 15:14:35 +00:00
|
|
|
|
fprintf(stderr, "ERROR: Could not remove exception handler, "
|
2005-10-23 21:30:24 +00:00
|
|
|
|
"handler is bad pointer.\n");
|
2005-10-23 15:11:19 +00:00
|
|
|
|
thread->_exception_handler = 0;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (handler->next && IsBadReadPtr(handler->next, sizeof(NSHandler)))
|
|
|
|
|
{
|
2005-10-23 15:14:35 +00:00
|
|
|
|
fprintf(stderr, "ERROR: Could not restore exception handler, "
|
2005-10-23 21:30:24 +00:00
|
|
|
|
"handler->next is bad pointer.\n");
|
2005-10-23 15:11:19 +00:00
|
|
|
|
thread->_exception_handler = 0;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
#endif
|
|
|
|
|
thread->_exception_handler = handler->next;
|
1997-03-03 19:51:04 +00:00
|
|
|
|
}
|
2008-06-23 07:15:10 +00:00
|
|
|
|
|
|
|
|
|
NSUncaughtExceptionHandler *
|
|
|
|
|
NSGetUncaughtExceptionHandler()
|
|
|
|
|
{
|
|
|
|
|
return _NSUncaughtExceptionHandler;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
NSSetUncaughtExceptionHandler(NSUncaughtExceptionHandler *handler)
|
|
|
|
|
{
|
|
|
|
|
_NSUncaughtExceptionHandler = handler;
|
|
|
|
|
}
|