#ifndef SDI_HOOK_H #define SDI_HOOK_H /* Includeheader Name: SDI_hook.h Versionstring: $VER: SDI_hook.h 1.27 (04.04.2015) Authors: Dirk Stoecker, Jens Maus Distribution: PD Project page: https://github.com/adtools/SDI Description: defines to hide compiler specific hook stuff 1.0 21.06.02 : based on the work made for freeciv and YAM with additional texts partly taken from YAM_hook.h changes made by Jens Langner, largely reworked the mechanism 1.1 07.10.02 : added HOOKPROTONP and HOOKPROTONONP requested by Jens 1.2 18.10.02 : reverted to old MorphOS-method for GCC 1.3 08.02.04 : modified to get it compatible to AmigaOS4 1.4 17.02.04 : modified to get compatible to latest SDI_compiler.h changes 1.5 02.03.04 : added UNUSED define to OS4 hook specification so that the compiler can ignore some warnings. 1.6 02.03.04 : added (APTR) casts to MorphOS prototype definition to reduce compiler warnings. 1.7 04.07.04 : removed static from all DISPATCHERPROTO definitions as there may be dispatchers that are of course non static. 1.8 07.04.05 : added MakeHookWithData (Sebastian Bauer) 1.9 08.04.05 : changed MorphOS hooks to use HookEntry (Ilkka Lehtoranta) 1.10 16.05.05 : simplified and fixed for vbcc/MorphOS (Frank Wille) 1.11 17.05.05 : changed cast in DISPATCHERPROTO from (void(*)()) to APTR cause SDI version of the EmulLibEntry uses APTR for easy usage. Added #ifndef SDI_TRAP_LIB to avoid double defines when combining with SDI_interrupt.h or SDI_misc.h (Guido Mersmann) 1.12 18.05.05 : DISPATCHERPROTO wasn't working, because of the missing REG_Ax definitions. Added include (Guido Mersmann) 1.13 11.12.05 : fixed a minor typo in the PPC HOOKPROTONP macro. (Jens Langner) 1.14 20.04.06 : unified static of MorphOs with non-MorphOS vesion 1.15 30.04.06 : modified to get it compatible to AROS. (Guido Mersmann) 1.16 06.10.06 : added new DISPATCHER() macro and separated it from the DISPATCHERPROTO() definition. Now the DISPATCHERPROTO() should only be used to get the correct prototype and the plain DISPATCHER() for defining the dispatcher itself. 1.17 14.07.08 : added "_" to all UNUSED variable specifications to make sure a user does not use those definition on accident. 1.18 20.03.09 : modified macros to be somewhat more compatible for an AROS usage (Pavel Fedin) 1.19 25.03.09 : fixed the DISPATCHERPROTO() macros for x86_64 AROS. 1.20 26.03.09 : fixed m68k define checks. 1.21 19.05.09 : added SDISPATCHER() to generate a static dispatcher. 1.22 24.06.10 : fixed AROS macros (Matthias Rustler). 1.23 12.08.10 : added missing proto/alib.h include for AROS 1.24 03.03.11 : fixed AROS macros for m68k (Jason McMullan) 1.25 18.07.12 : added crosscall macros for functions with 1 and 2 parameters. These macros are to be used in case i.e. PPC code is to be called from m68k code. The function pointer must be passed with ENTRY(). 1.26 17.10.12 : added crosscall macros for functions with 1 and 2 parameters but no return value. 1.27 04.04.15 : fixed HOOKPROTO for vbcc (Fredrik Wikstrom) 1.28 12.07.16 : fixed HOOKPROTO for vbcc (O. Sezer) */ /* ** This is PD (Public Domain). This means you can do with it whatever you want ** without any restrictions. I only ask you to tell me improvements, so I may ** fix the main line of this files as well. ** ** To keep confusion level low: When changing this file, please note it in ** above history list and indicate that the change was not made by myself ** (e.g. add your name or nick name). ** ** Find the latest version of this file at: ** https://github.com/adtools/SDI ** ** Jens Maus ** Dirk Stoecker */ #include "SDI_compiler.h" /* ** Hook macros to handle the creation of Hooks/Dispatchers for different ** Operating System versions. ** Currently AmigaOS, AROS, and MorphOS are supported. ** ** For more information about hooks see include file or ** the relevant descriptions in utility.library autodocs. ** ** Example: ** ** Creates a hook with the name "TestHook" that calls a corresponding ** function "TestFunc" that will be called with a pointer "text" ** (in register A1) and returns a long. ** ** HOOKPROTONHNO(TestFunc, LONG, STRPTR text) ** { ** Printf(text); ** return 0; ** } ** MakeHook(TestHook, TestFunc); ** ** Every function that is created with HOOKPROTO* must have a MakeHook() or ** MakeStaticHook() to create the corresponding hook. Best is to call this ** directly after the hook function. This is required by the GCC macros. ** ** The naming convention for the Hook Prototype macros is as followed: ** ** HOOKPROTO[NH][NO][NP] ** ^^ ^^ ^^ ** NoHook | NoParameter ** NoObject ** ** So a plain HOOKPROTO() creates you a Hook function that requires ** 4 parameters, the "name" of the hookfunction, the "obj" in REG_A2, ** the "param" in REG_A1 and a "hook" in REG_A0. Usually you will always ** use NH, as the hook structure itself is nearly never required. ** ** The DISPATCHERPROTO macro is for MUI dispatcher functions. It gets the ** functionname as argument. To supply this function for use by MUI, use ** The ENTRY macro, which also gets the function name as argument. */ #if !defined(__AROS__) && (defined(_M68000) || defined(__M68000) || defined(__mc68000) || defined(__M68K__)) #define HOOKPROTO(name, ret, obj, param) static SAVEDS ASM ret \ name(REG(a0, struct Hook *hook), REG(a2, obj), REG(a1, param)) #define HOOKPROTONO(name, ret, param) static SAVEDS ASM ret \ name(REG(a0, struct Hook *hook), REG(a1, param)) #define HOOKPROTONP(name, ret, obj) static SAVEDS ASM ret \ name(REG(a0, struct Hook *hook), REG(a2, obj)) #define HOOKPROTONONP(name, ret) static SAVEDS ASM ret \ name(REG(a0, struct Hook *hook)) #define HOOKPROTONH(name, ret, obj, param) static SAVEDS ASM ret \ name(REG(a2, obj), REG(a1, param)) #define HOOKPROTONHNO(name, ret, param) static SAVEDS ASM ret \ name(REG(a1, param)) #define HOOKPROTONHNP(name, ret, obj) static SAVEDS ASM ret \ name(REG(a2, obj)) #define HOOKPROTONHNONP(name, ret) static SAVEDS ret name(void) #else /* NOTE: This is fine for AROS, since HookEntry will handle stack params */ #define HOOKPROTO(name, ret, obj, param) static SAVEDS ret \ name(struct Hook *hook, obj, param) #define HOOKPROTONO(name, ret, param) static SAVEDS ret \ name(struct Hook *hook, UNUSED APTR _obj, param) #define HOOKPROTONP(name, ret, obj) static SAVEDS ret \ name(struct Hook *hook, obj, UNUSED APTR _param) #define HOOKPROTONONP(name, ret) static SAVEDS ret \ name(struct Hook *hook, UNUSED APTR _obj, UNUSED APTR _param) #define HOOKPROTONH(name, ret, obj, param) static SAVEDS ret \ name(UNUSED struct Hook *_hook, obj, param) #define HOOKPROTONHNO(name, ret, param) static SAVEDS ret \ name(UNUSED struct Hook *_hook, UNUSED APTR _obj, param) #define HOOKPROTONHNP(name, ret, obj) static SAVEDS ret \ name(UNUSED struct Hook *_hook, obj, UNUSED APTR _param) #define HOOKPROTONHNONP(name, ret) static SAVEDS ret name(void) #endif #ifdef __MORPHOS__ #ifndef SDI_TRAP_LIB /* avoid defining this twice */ #include #include #define SDI_TRAP_LIB 0xFF00 /* SDI prefix to reduce conflicts */ struct SDI_EmulLibEntry { UWORD Trap; UWORD pad; APTR Func; }; #endif #define MakeHook(hookname, funcname) struct Hook hookname = {{NULL, NULL}, \ (HOOKFUNC)HookEntry, (HOOKFUNC)funcname, NULL} #define MakeHookWithData(hookname, funcname, data) struct Hook hookname = \ {{NULL, NULL}, (HOOKFUNC)HookEntry, (HOOKFUNC)funcname, (APTR)data} #define MakeStaticHook(hookname, funcname) static struct Hook hookname = \ {{NULL, NULL}, (HOOKFUNC)HookEntry, (HOOKFUNC)funcname, NULL} #define DISPATCHERPROTO(name) ULONG name(struct IClass * cl, Object * obj, \ Msg msg); \ extern const struct SDI_EmulLibEntry Gate_##name #define DISPATCHER(name) \ struct IClass; \ ULONG name(struct IClass * cl, Object * obj, Msg msg); \ static ULONG Trampoline_##name(void) {return name((struct IClass *) \ REG_A0, (Object *) REG_A2, (Msg) REG_A1);} \ const struct SDI_EmulLibEntry Gate_##name = {SDI_TRAP_LIB, 0, \ (APTR) Trampoline_##name}; \ ULONG name(struct IClass * cl, Object * obj, Msg msg) #define SDISPATCHER(name) \ struct IClass; \ static ULONG name(struct IClass * cl, Object * obj, Msg msg); \ static ULONG Trampoline_##name(void) {return name((struct IClass *) \ REG_A0, (Object *) REG_A2, (Msg) REG_A1);} \ static const struct SDI_EmulLibEntry Gate_##name = {SDI_TRAP_LIB, 0, \ (APTR) Trampoline_##name}; \ static ULONG name(struct IClass * cl, Object * obj, Msg msg) #define CROSSCALL1(name, ret, type1, param1) \ static STDARGS SAVEDS ret name(type1 param1); \ static ret Trampoline_##name(void) \ { \ ULONG *stk = (ULONG *)REG_A7; \ type1 param1 = (type1)stk[1]; \ return name(param1); \ } \ static const struct SDI_EmulLibEntry Gate_##name = {SDI_TRAP_LIB, 0, \ (APTR) Trampoline_##name}; \ static STDARGS SAVEDS ret name(type1 param1, type2 param2) #define CROSSCALL1NR(name, type1, param1) \ static STDARGS SAVEDS void name(type1 param1); \ static void Trampoline_##name(void) \ { \ ULONG *stk = (ULONG *)REG_A7; \ type1 param1 = (type1)stk[1]; \ name(param1); \ } \ static const struct SDI_EmulLibEntry Gate_##name = {SDI_TRAP_LIB, 0, \ (APTR) Trampoline_##name}; \ static STDARGS SAVEDS void name(type1 param1) #define CROSSCALL2(name, ret, type1, param1, type2, param2) \ static STDARGS SAVEDS ret name(type1 param1, type2 param2); \ static ret Trampoline_##name(void) \ { \ ULONG *stk = (ULONG *)REG_A7; \ type1 param1 = (type1)stk[1]; \ type2 param2 = (type2)stk[2]; \ return name(param1, param2); \ } \ static const struct SDI_EmulLibEntry Gate_##name = {SDI_TRAP_LIB, 0, \ (APTR) Trampoline_##name}; \ static STDARGS SAVEDS ret name(type1 param1, type2 param2) #define CROSSCALL2NR(name, type1, param1, type2, param2) \ static STDARGS SAVEDS void name(type1 param1, type2 param2); \ static void Trampoline_##name(void) \ { \ ULONG *stk = (ULONG *)REG_A7; \ type1 param1 = (type1)stk[1]; \ type2 param2 = (type2)stk[2]; \ name(param1, param2); \ } \ static const struct SDI_EmulLibEntry Gate_##name = {SDI_TRAP_LIB, 0, \ (APTR) Trampoline_##name}; \ static STDARGS SAVEDS void name(type1 param1, type2 param2) #define ENTRY(func) (APTR)&Gate_##func #elif defined(__AROS__) #include #define MakeHook(hookname, funcname) struct Hook hookname = {{NULL, NULL}, \ (HOOKFUNC)HookEntry, (HOOKFUNC)funcname, NULL} #define MakeHookWithData(hookname, funcname, data) struct Hook hookname = \ {{NULL, NULL}, (HOOKFUNC)HookEntry, (HOOKFUNC)funcname, (APTR)data} #define MakeStaticHook(hookname, funcname) static struct Hook hookname = \ {{NULL, NULL}, (HOOKFUNC)HookEntry, (HOOKFUNC)funcname, NULL} #define DISPATCHERPROTO(name) \ IPTR name(struct IClass * cl, Object * obj, Msg msg); \ AROS_UFP3(IPTR, Gate_##name, \ AROS_UFPA(struct IClass *, cl, A0), \ AROS_UFPA(Object *, obj, A2), \ AROS_UFPA(Msg, msg, A1)) #define DISPATCHERx(x,name) \ x IPTR name(struct IClass * cl, Object * obj, Msg msg); \ x AROS_UFH3(IPTR, Gate_##name, \ AROS_UFHA(struct IClass *, cl, A0), \ AROS_UFHA(Object *, obj, A2), \ AROS_UFHA(Msg, msg, A1)) \ { AROS_USERFUNC_INIT \ return name(cl, obj, msg); \ AROS_USERFUNC_EXIT \ } \ x IPTR name(struct IClass * cl, Object * obj, Msg msg) #define DISPATCHER(name) DISPATCHERx(,name) #define SDISPATCHER(name) DISPATCHERx(static,name) #define CROSSCALL1(name, ret, type1, param1) \ static STDARGS SAVEDS ret Gate_##name(type1 param1) #define CROSSCALL1NR(name, type1, param1) \ static STDARGS SAVEDS void Gate_##name(type1 param1) #define CROSSCALL2(name, ret, type1, param1, type2, param2) \ static STDARGS SAVEDS ret Gate_##name(type1 param1, type2 param2) #define CROSSCALL2NR(name, type1, param1, type2, param2) \ static STDARGS SAVEDS void Gate_##name(type1 param1, type2 param2) #define ENTRY(func) (APTR)Gate_##func #else /* !__MORPHOS__ && !__AROS__*/ #define MakeHook(hookname, funcname) struct Hook hookname = {{NULL, NULL}, \ (HOOKFUNC)funcname, NULL, NULL} #define MakeHookWithData(hookname, funcname, data) struct Hook hookname = \ {{NULL, NULL}, (HOOKFUNC)funcname, NULL, (APTR)data} #define MakeStaticHook(hookname, funcname) static struct Hook hookname = \ {{NULL, NULL}, (HOOKFUNC)funcname, NULL, NULL} #define DISPATCHERPROTO(name) SAVEDS ASM IPTR name(REG(a0, \ struct IClass * cl), REG(a2, Object * obj), REG(a1, Msg msg)) #define DISPATCHER(name) DISPATCHERPROTO(name) #define SDISPATCHER(name) static DISPATCHERPROTO(name) #define CROSSCALL1(name, ret, type1, param1) \ static STDARGS SAVEDS ret name(type1 param1) #define CROSSCALL1NR(name, type1, param1) \ static STDARGS SAVEDS void name(type1 param1) #define CROSSCALL2(name, ret, type1, param1, type2, param2) \ static STDARGS SAVEDS ret name(type1 param1, type2 param2) #define CROSSCALL2NR(name, type1, param1, type2, param2) \ static STDARGS SAVEDS void name(type1 param1, type2 param2) #define ENTRY(func) (APTR)func #endif #define InitHook(hook, orighook, data) ((hook)->h_Entry = (orighook).h_Entry,\ (hook)->h_SubEntry = (orighook).h_SubEntry,(hook)->h_Data = (APTR)(data)) #endif /* SDI_HOOK_H */