/*
* Copyright (c) 2008, 2009, 2010, 2011, 2012, 2013, 2014, 2015, 2016, 2017,
* 2018
* Jonathan Schleifer <js@heap.zone>
*
* All rights reserved.
*
* This file is part of ObjFW. It may be distributed under the terms of the
* Q Public License 1.0, which can be found in the file LICENSE.QPL included in
* the packaging of this file.
*
* Alternatively, it may be distributed under the terms of the GNU General
* Public License, either version 2 or 3, which can be found in the file
* LICENSE.GPLv2 or LICENSE.GPLv3 respectively included in the packaging of this
* file.
*/
#include "config.h"
#import "ObjFW_RT.h"
#import "private.h"
#import "macros.h"
#include <proto/exec.h>
#include <stdio.h>
#include <stdlib.h>
extern int _Unwind_RaiseException(void *);
extern void _Unwind_DeleteException(void *);
extern void *_Unwind_GetLanguageSpecificData(void *);
extern uintptr_t _Unwind_GetRegionStart(void *);
extern uintptr_t _Unwind_GetDataRelBase(void *);
extern uintptr_t _Unwind_GetTextRelBase(void *);
extern uintptr_t _Unwind_GetIP(void *);
extern uintptr_t _Unwind_GetGR(void *, int);
extern void _Unwind_SetIP(void *, uintptr_t);
extern void _Unwind_SetGR(void *, int, uintptr_t);
extern void _Unwind_Resume(void *);
extern void __register_frame_info(const void *, void *);
extern void __deregister_frame_info(const void *);
struct Library *ObjFWRTBase;
static void __attribute__((__constructor__))
init(void)
{
static bool initialized = false;
static struct objc_libc libc;
if (initialized)
return;
if ((ObjFWRTBase = OpenLibrary("objfw_rt.library", 0)) == NULL) {
fputs("Failed to open objfw_rt.library!\n", stderr);
abort();
}
libc = (struct objc_libc){
.malloc = malloc,
.calloc = calloc,
.realloc = realloc,
.free = free,
.vfprintf = vfprintf,
.fputs = fputs,
.exit = exit,
.abort = abort,
._Unwind_RaiseException = _Unwind_RaiseException,
._Unwind_DeleteException = _Unwind_DeleteException,
._Unwind_GetLanguageSpecificData =
_Unwind_GetLanguageSpecificData,
._Unwind_GetRegionStart = _Unwind_GetRegionStart,
._Unwind_GetDataRelBase = _Unwind_GetDataRelBase,
._Unwind_GetTextRelBase = _Unwind_GetTextRelBase,
._Unwind_GetIP = _Unwind_GetIP,
._Unwind_GetGR = _Unwind_GetGR,
._Unwind_SetIP = _Unwind_SetIP,
._Unwind_SetGR = _Unwind_SetGR,
._Unwind_Resume = _Unwind_Resume,
.__register_frame_info = __register_frame_info,
.__deregister_frame_info = __deregister_frame_info,
};
objc_init(&libc, stdout, stderr);
initialized = true;
}
OF_DESTRUCTOR()
{
CloseLibrary(ObjFWRTBase);
}
void
__objc_exec_class(void *module)
{
/*
* The compiler generates constructors that call into this, so it is
* possible that we are not set up yet when we get called.
*/
init();
glue___objc_exec_class(module);
}
IMP
objc_msg_lookup(id obj, SEL sel)
{
return glue_objc_msg_lookup(obj, sel);
}
IMP
objc_msg_lookup_stret(id obj, SEL sel)
{
return glue_objc_msg_lookup_stret(obj, sel);
}
IMP
objc_msg_lookup_super(struct objc_super *super, SEL sel)
{
return glue_objc_msg_lookup_super(super, sel);
}
IMP
objc_msg_lookup_super_stret(struct objc_super *super, SEL sel)
{
return glue_objc_msg_lookup_super_stret(super, sel);
}
Class
objc_lookUpClass(const char *name)
{
return glue_objc_lookUpClass(name);
}
Class
objc_getClass(const char *name)
{
return glue_objc_getClass(name);
}
Class
objc_getRequiredClass(const char *name)
{
return glue_objc_getRequiredClass(name);
}
Class
objc_lookup_class(const char *name)
{
return glue_objc_lookup_class(name);
}
Class
objc_get_class(const char *name)
{
return glue_objc_get_class(name);
}
void
objc_exception_throw(id object)
{
/*
* This does not use the glue code to hack around a compiler bug.
*
* When using the generated inline stubs, the compiler does not emit
* any frame information, making the unwind fail. As unwind always
* starts from objc_exception_throw(), this means exceptions would
* never work. If, however, we're using a function pointer instead of
* the inline stub, the compiler does generate a frame and everything
* works fine.
*/
register void *a6 OBJC_M68K_REG("a6") = ObjFWRTBase;
uintptr_t throw = (((uintptr_t)ObjFWRTBase) - 0x60);
((void (*)(id OBJC_M68K_REG("a0")))throw)(object);
(void)a6;
OF_UNREACHABLE
}
int
objc_sync_enter(id object)
{
return glue_objc_sync_enter(object);
}
int
objc_sync_exit(id object)
{
return glue_objc_sync_exit(object);
}
id
objc_getProperty(id self, SEL _cmd, ptrdiff_t offset, bool atomic)
{
return glue_objc_getProperty(self, _cmd, offset, atomic);
}
void
objc_setProperty(id self, SEL _cmd, ptrdiff_t offset, id value, bool atomic,
signed char copy)
{
glue_objc_setProperty(self, _cmd, offset, value, atomic, copy);
}
void
objc_getPropertyStruct(void *dest, const void *src, ptrdiff_t size, bool atomic,
bool strong)
{
glue_objc_getPropertyStruct(dest, src, size, atomic, strong);
}
void
objc_setPropertyStruct(void *dest, const void *src, ptrdiff_t size, bool atomic,
bool strong)
{
glue_objc_setPropertyStruct(dest, src, size, atomic, strong);
}
void
objc_enumerationMutation(id object)
{
/*
* This does not use the glue code to hack around a compiler bug.
*
* When using the generated inline stubs, the compiler does not emit
* any frame information, making the unwind fail. As a result
* objc_enumerationMutation() might throw an exception that could never
* be caught. If, however, we're using a function pointer instead of
* the inline stub, the compiler does generate a frame and everything
* works fine.
*/
register void *a6 OBJC_M68K_REG("a6") = ObjFWRTBase;
uintptr_t enumerationMutation = (((uintptr_t)ObjFWRTBase) - 0x8A);
((void (*)(id OBJC_M68K_REG("a0")))enumerationMutation)(object);
(void)a6;
OF_UNREACHABLE
}
int
__gnu_objc_personality_v0(int version, int actions, uint64_t ex_class,
void *ex, void *ctx)
{
return glue___gnu_objc_personality_v0(version, actions, &ex_class,
ex, ctx);
}