mirror of
https://github.com/darlinghq/darling-libobjc2.git
synced 2024-11-27 14:10:33 +00:00
6317b79a10
clang trunk if -fobjc-runtime-1.7 is specified and provides significantly better interoperability with foreign exceptions. Note: Most of the exception tests will not pass with gcc or clang < 3.3. They test things that are impossible to implement quite correctly with the GCC-compatible exception ABI. Also updated the release notes to reflect recent improvements.
244 lines
6.3 KiB
C++
244 lines
6.3 KiB
C++
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include "dwarf_eh.h"
|
|
#include "objcxx_eh.h"
|
|
#include <exception>
|
|
|
|
extern "C"
|
|
{
|
|
#include "objc/runtime.h"
|
|
};
|
|
namespace __cxxabiv1
|
|
{
|
|
struct __class_type_info;
|
|
}
|
|
|
|
using __cxxabiv1::__class_type_info;
|
|
|
|
namespace std
|
|
{
|
|
/**
|
|
* std::type_info defined with the GCC ABI. This may not be exposed in
|
|
* public headers, but is required for correctly implementing the unified
|
|
* exception model.
|
|
*/
|
|
class type_info
|
|
{
|
|
public:
|
|
virtual ~type_info();
|
|
bool operator==(const type_info &) const;
|
|
bool operator!=(const type_info &) const;
|
|
bool before(const type_info &) const;
|
|
type_info();
|
|
private:
|
|
type_info(const type_info& rhs);
|
|
type_info& operator= (const type_info& rhs);
|
|
const char *__type_name;
|
|
protected:
|
|
type_info(const char *name): __type_name(name) { }
|
|
public:
|
|
const char* name() const { return __type_name; }
|
|
virtual bool __is_pointer_p() const;
|
|
virtual bool __is_function_p() const;
|
|
virtual bool __do_catch(const type_info *thrown_type,
|
|
void **thrown_object,
|
|
unsigned outer) const;
|
|
virtual bool __do_upcast(
|
|
const __class_type_info *target,
|
|
void **thrown_object) const;
|
|
};
|
|
}
|
|
|
|
using namespace std;
|
|
|
|
|
|
static BOOL isKindOfClass(Class thrown, Class type)
|
|
{
|
|
do
|
|
{
|
|
if (thrown == type)
|
|
{
|
|
return YES;
|
|
}
|
|
thrown = class_getSuperclass(thrown);
|
|
} while (Nil != thrown);
|
|
|
|
return NO;
|
|
}
|
|
|
|
/**
|
|
* C++ Exception structure. From the Itanium ABI spec
|
|
*/
|
|
struct __cxa_exception
|
|
{
|
|
std::type_info *exceptionType;
|
|
void (*exceptionDestructor) (void *);
|
|
unexpected_handler unexpectedHandler;
|
|
terminate_handler terminateHandler;
|
|
__cxa_exception *nextException;
|
|
unsigned int handlerCount;
|
|
int handlerSwitchValue;
|
|
const char *actionRecord;
|
|
const char *languageSpecificData;
|
|
void *catchTemp;
|
|
void *adjustedPtr;
|
|
_Unwind_Exception unwindHeader;
|
|
};
|
|
|
|
|
|
|
|
|
|
namespace gnustep
|
|
{
|
|
namespace libobjc
|
|
{
|
|
struct __objc_id_type_info : std::type_info
|
|
{
|
|
__objc_id_type_info() : type_info("@id") {};
|
|
virtual ~__objc_id_type_info();
|
|
virtual bool __do_catch(const type_info *thrownType,
|
|
void **obj,
|
|
unsigned outer) const;
|
|
};
|
|
struct __objc_class_type_info : std::type_info
|
|
{
|
|
virtual ~__objc_class_type_info();
|
|
virtual bool __do_catch(const type_info *thrownType,
|
|
void **obj,
|
|
unsigned outer) const;
|
|
};
|
|
}
|
|
};
|
|
|
|
|
|
static bool AppleCompatibleMode = true;
|
|
extern "C" int objc_set_apple_compatible_objcxx_exceptions(int newValue)
|
|
{
|
|
bool old = AppleCompatibleMode;
|
|
AppleCompatibleMode = newValue;
|
|
return old;
|
|
}
|
|
|
|
gnustep::libobjc::__objc_class_type_info::~__objc_class_type_info() {}
|
|
gnustep::libobjc::__objc_id_type_info::~__objc_id_type_info() {}
|
|
bool gnustep::libobjc::__objc_class_type_info::__do_catch(const type_info *thrownType,
|
|
void **obj,
|
|
unsigned outer) const
|
|
{
|
|
id thrown = (id)obj;
|
|
bool found = false;
|
|
// Id throw matches any ObjC catch. This may be a silly idea!
|
|
if (dynamic_cast<const __objc_id_type_info*>(thrownType)
|
|
|| (AppleCompatibleMode &&
|
|
dynamic_cast<const __objc_class_type_info*>(thrownType)))
|
|
{
|
|
thrown = **(id**)obj;
|
|
// nil only matches id catch handlers in Apple-compatible mode, or when thrown as an id
|
|
if (0 == thrown)
|
|
{
|
|
return false;
|
|
}
|
|
// Check whether the real thrown object matches the catch type.
|
|
found = isKindOfClass(object_getClass(thrown),
|
|
(Class)objc_getClass(name()));
|
|
}
|
|
else if (dynamic_cast<const __objc_class_type_info*>(thrownType))
|
|
{
|
|
thrown = **(id**)obj;
|
|
found = isKindOfClass((Class)objc_getClass(thrownType->name()),
|
|
(Class)objc_getClass(name()));
|
|
}
|
|
if (found)
|
|
{
|
|
*obj = (void*)thrown;
|
|
}
|
|
return found;
|
|
};
|
|
|
|
bool gnustep::libobjc::__objc_id_type_info::__do_catch(const type_info *thrownType,
|
|
void **obj,
|
|
unsigned outer) const
|
|
{
|
|
// Id catch matches any ObjC throw
|
|
if (dynamic_cast<const __objc_class_type_info*>(thrownType))
|
|
{
|
|
*obj = **(id**)obj;
|
|
return true;
|
|
}
|
|
if (dynamic_cast<const __objc_id_type_info*>(thrownType))
|
|
{
|
|
*obj = **(id**)obj;
|
|
return true;
|
|
}
|
|
return false;
|
|
};
|
|
|
|
/**
|
|
* Public interface to the Objective-C++ exception mechanism
|
|
*/
|
|
extern "C"
|
|
{
|
|
/**
|
|
* The public symbol that the compiler uses to indicate the Objective-C id type.
|
|
*/
|
|
gnustep::libobjc::__objc_id_type_info __objc_id_type_info;
|
|
|
|
/**
|
|
* Exception cleanup function for C++ exceptions that wrap Objective-C
|
|
* exceptions.
|
|
*/
|
|
static void exception_cleanup(_Unwind_Reason_Code reason,
|
|
struct _Unwind_Exception *ex)
|
|
{
|
|
__cxa_exception *cxxex = (__cxa_exception*) ((char*)ex - offsetof(struct __cxa_exception, unwindHeader));
|
|
if (cxxex->exceptionType != &__objc_id_type_info)
|
|
{
|
|
delete cxxex->exceptionType;
|
|
}
|
|
__cxa_free_exception((void*)ex);
|
|
}
|
|
|
|
struct _Unwind_Exception *objc_init_cxx_exception(void *thrown_exception)
|
|
{
|
|
__cxa_exception *ex = ((__cxa_exception*)thrown_exception) - 1;
|
|
|
|
std::type_info *tinfo = &__objc_id_type_info;
|
|
|
|
ex->exceptionType = tinfo;
|
|
|
|
ex->exceptionDestructor = 0;
|
|
|
|
ex->unwindHeader.exception_class = EXCEPTION_CLASS('G','N','U','C','C','+','+','\0');
|
|
ex->unwindHeader.exception_cleanup = exception_cleanup;
|
|
|
|
return &ex->unwindHeader;
|
|
}
|
|
|
|
void* objc_object_for_cxx_exception(void *thrown_exception, int *isValid)
|
|
{
|
|
__cxa_exception *ex = (__cxa_exception*) ((char*)thrown_exception -
|
|
offsetof(struct __cxa_exception, unwindHeader));
|
|
const std::type_info *thrownType = ex->exceptionType;
|
|
if (!dynamic_cast<const gnustep::libobjc::__objc_id_type_info*>(thrownType) &&
|
|
!dynamic_cast<const gnustep::libobjc::__objc_class_type_info*>(thrownType))
|
|
{
|
|
*isValid = 0;
|
|
return 0;
|
|
}
|
|
*isValid = 1;
|
|
return *(id*)(ex+1);
|
|
}
|
|
|
|
/*
|
|
void print_type_info(void *thrown_exception)
|
|
{
|
|
__cxa_exception *ex = (__cxa_exception*) ((char*)thrown_exception -
|
|
offsetof(struct __cxa_exception, unwindHeader));
|
|
fprintf(stderr, "Type info: %s\n", ex->exceptionType->name());
|
|
fprintf(stderr, "offset is: %d\n", offsetof(struct __cxa_exception, unwindHeader));
|
|
}
|
|
*/
|
|
|
|
} // extern "C"
|
|
|