| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522 |
- /*
- AngelCode Scripting Library
- Copyright (c) 2003-2018 Andreas Jonsson
- This software is provided 'as-is', without any express or implied
- warranty. In no event will the authors be held liable for any
- damages arising from the use of this software.
- Permission is granted to anyone to use this software for any
- purpose, including commercial applications, and to alter it and
- redistribute it freely, subject to the following restrictions:
- 1. The origin of this software must not be misrepresented; you
- must not claim that you wrote the original software. If you use
- this software in a product, an acknowledgment in the product
- documentation would be appreciated but is not required.
- 2. Altered source versions must be plainly marked as such, and
- must not be misrepresented as being the original software.
- 3. This notice may not be removed or altered from any source
- distribution.
- The original version of this library can be located at:
- http://www.angelcode.com/angelscript/
- Andreas Jonsson
- [email protected]
- */
- // Modified for Urho3D
- //
- // as_callfunc_x86.cpp
- //
- // These functions handle the actual calling of system functions
- //
- // Added support for functor methods by Jordi Oliveras Rovira in April, 2014.
- //
- #include "as_config.h"
- #ifndef AS_MAX_PORTABILITY
- #ifdef AS_X86
- #include "as_callfunc.h"
- #include "as_scriptengine.h"
- #include "as_texts.h"
- #include "as_tokendef.h"
- #include "as_context.h"
- // Urho3D: work around Clang crash and assembler error on GCC
- // Because Urho3D's AngelScript API convention forbids C++ exceptions leaking to syscalls,
- // the lack of prologs / epilogs should not be critical
- #if defined(__clang__) || defined(__GNUC__)
- #undef __OPTIMIZE__
- #endif
- BEGIN_AS_NAMESPACE
- //
- // With some compile level optimizations the functions don't clear the FPU
- // stack themselves. So we have to do it as part of calling the native functions,
- // as the compiler will not be able to predict when it is supposed to do it by
- // itself due to the dynamic nature of scripts
- //
- // - fninit clears the FPU stack and the FPU control word
- // - emms only clears the FPU stack, while preserving the FPU control word
- //
- // By default I use fninit as it seems to be what works for most people,
- // but some may find it necessary to define this as emms instead.
- //
- // TODO: Figure out when one or the other must be used, and a way to
- // configure this automatically in as_config.h
- //
- #ifndef CLEAR_FPU_STACK
- #define CLEAR_FPU_STACK fninit
- #endif
- // These macros are just to allow me to use the above macro in the GNUC style inline assembly
- #define _S(x) _TOSTRING(x)
- #define _TOSTRING(x) #x
- // Prototypes
- asQWORD CallCDeclFunction(const asDWORD *args, int paramSize, asFUNCTION_t func);
- asQWORD CallCDeclFunctionObjLast(const void *obj, const asDWORD *args, int paramSize, asFUNCTION_t func);
- asQWORD CallCDeclFunctionObjFirst(const void *obj, const asDWORD *args, int paramSize, asFUNCTION_t func);
- asQWORD CallCDeclFunctionRetByRef(const asDWORD *args, int paramSize, asFUNCTION_t func, void *retPtr);
- asQWORD CallCDeclFunctionRetByRefObjLast(const void *obj, const asDWORD *args, int paramSize, asFUNCTION_t func, void *retPtr);
- asQWORD CallCDeclFunctionRetByRefObjFirst(const void *obj, const asDWORD *args, int paramSize, asFUNCTION_t func, void *retPtr);
- asQWORD CallSTDCallFunction(const asDWORD *args, int paramSize, asFUNCTION_t func);
- asQWORD CallThisCallFunction(const void *obj, const asDWORD *args, int paramSize, asFUNCTION_t func);
- asQWORD CallThisCallFunctionRetByRef(const void *, const asDWORD *, int, asFUNCTION_t, void *retPtr);
- asDWORD GetReturnedFloat();
- asQWORD GetReturnedDouble();
- asQWORD CallSystemFunctionNative(asCContext *context, asCScriptFunction *descr, void *obj, asDWORD *args, void *retPointer, asQWORD &/*retQW2*/, void *secondObject)
- {
- asCScriptEngine *engine = context->m_engine;
- asSSystemFunctionInterface *sysFunc = descr->sysFuncIntf;
- asQWORD retQW = 0;
- // Prepare the parameters
- asDWORD paramBuffer[64];
- int callConv = sysFunc->callConv;
- // Changed because need check for ICC_THISCALL_OBJFIRST or
- // ICC_THISCALL_OBJLAST if sysFunc->takesObjByVal (avoid copy code)
- // Check if is THISCALL_OBJ* calling convention (in this case needs to add secondObject pointer into stack).
- bool isThisCallMethod = callConv >= ICC_THISCALL_OBJLAST;
- int paramSize = isThisCallMethod || sysFunc->takesObjByVal ? 0 : sysFunc->paramSize;
- int dpos = 1;
- if( isThisCallMethod &&
- (callConv >= ICC_THISCALL_OBJFIRST &&
- callConv <= ICC_VIRTUAL_THISCALL_OBJFIRST_RETURNINMEM) )
- {
- // Add the object pointer as the first parameter
- paramBuffer[dpos++] = (asDWORD)secondObject;
- paramSize++;
- }
- if( sysFunc->takesObjByVal || isThisCallMethod )
- {
- int spos = 0;
- for( asUINT n = 0; n < descr->parameterTypes.GetLength(); n++ )
- {
- if( descr->parameterTypes[n].IsObject() && !descr->parameterTypes[n].IsObjectHandle() && !descr->parameterTypes[n].IsReference() )
- {
- #ifdef COMPLEX_OBJS_PASSED_BY_REF
- if( descr->parameterTypes[n].GetTypeInfo()->flags & COMPLEX_MASK )
- {
- paramBuffer[dpos++] = args[spos++];
- paramSize++;
- }
- else
- #endif
- {
- // Copy the object's memory to the buffer
- // TODO: bug: Must call the object's copy constructor instead of doing a memcpy,
- // as the object may hold a pointer to itself. It's not enough to
- // change only this memcpy as the assembler routine also makes a copy
- // of paramBuffer to the final stack location. To avoid the second
- // copy the C++ routine should point paramBuffer to the final stack
- // position and copy the values directly to that location. The assembler
- // routines then don't need to copy anything, and will just be
- // responsible for setting up the registers and the stack frame appropriately.
- memcpy(¶mBuffer[dpos], *(void**)(args+spos), descr->parameterTypes[n].GetSizeInMemoryBytes());
- // Delete the original memory
- engine->CallFree(*(char**)(args+spos));
- spos++;
- dpos += descr->parameterTypes[n].GetSizeInMemoryDWords();
- paramSize += descr->parameterTypes[n].GetSizeInMemoryDWords();
- }
- }
- else
- {
- // Copy the value directly
- paramBuffer[dpos++] = args[spos++];
- if( descr->parameterTypes[n].GetSizeOnStackDWords() > 1 )
- paramBuffer[dpos++] = args[spos++];
- paramSize += descr->parameterTypes[n].GetSizeOnStackDWords();
- }
- }
- // Keep a free location at the beginning
- args = ¶mBuffer[1];
- }
- if( isThisCallMethod &&
- (callConv >= ICC_THISCALL_OBJLAST &&
- callConv <= ICC_VIRTUAL_THISCALL_OBJLAST_RETURNINMEM) )
- {
- // Add the object pointer as the last parameter
- paramBuffer[dpos++] = (asDWORD)secondObject;
- paramSize++;
- }
- // Make the actual call
- asFUNCTION_t func = sysFunc->func;
- if( sysFunc->hostReturnInMemory )
- callConv++;
- switch( callConv )
- {
- case ICC_CDECL:
- retQW = CallCDeclFunction(args, paramSize<<2, func);
- break;
- case ICC_CDECL_RETURNINMEM:
- retQW = CallCDeclFunctionRetByRef(args, paramSize<<2, func, retPointer);
- break;
- case ICC_STDCALL:
- retQW = CallSTDCallFunction(args, paramSize<<2, func);
- break;
- case ICC_STDCALL_RETURNINMEM:
- // Push the return pointer on the stack
- paramSize++;
- args--;
- *(asPWORD*)args = (size_t)retPointer;
- retQW = CallSTDCallFunction(args, paramSize<<2, func);
- break;
- case ICC_THISCALL:
- case ICC_THISCALL_OBJFIRST:
- case ICC_THISCALL_OBJLAST:
- retQW = CallThisCallFunction(obj, args, paramSize<<2, func);
- break;
- case ICC_THISCALL_RETURNINMEM:
- case ICC_THISCALL_OBJFIRST_RETURNINMEM:
- case ICC_THISCALL_OBJLAST_RETURNINMEM:
- retQW = CallThisCallFunctionRetByRef(obj, args, paramSize<<2, func, retPointer);
- break;
- case ICC_VIRTUAL_THISCALL:
- case ICC_VIRTUAL_THISCALL_OBJFIRST:
- case ICC_VIRTUAL_THISCALL_OBJLAST:
- {
- // Get virtual function table from the object pointer
- asFUNCTION_t *vftable = *(asFUNCTION_t**)obj;
- retQW = CallThisCallFunction(obj, args, paramSize<<2, vftable[FuncPtrToUInt(func)>>2]);
- }
- break;
- case ICC_VIRTUAL_THISCALL_RETURNINMEM:
- case ICC_VIRTUAL_THISCALL_OBJFIRST_RETURNINMEM:
- case ICC_VIRTUAL_THISCALL_OBJLAST_RETURNINMEM:
- {
- // Get virtual function table from the object pointer
- asFUNCTION_t *vftable = *(asFUNCTION_t**)obj;
- retQW = CallThisCallFunctionRetByRef(obj, args, paramSize<<2, vftable[FuncPtrToUInt(func)>>2], retPointer);
- }
- break;
- case ICC_CDECL_OBJLAST:
- retQW = CallCDeclFunctionObjLast(obj, args, paramSize<<2, func);
- break;
- case ICC_CDECL_OBJLAST_RETURNINMEM:
- // Call the system object method as a cdecl with the obj ref as the last parameter
- retQW = CallCDeclFunctionRetByRefObjLast(obj, args, paramSize<<2, func, retPointer);
- break;
- case ICC_CDECL_OBJFIRST:
- // Call the system object method as a cdecl with the obj ref as the first parameter
- retQW = CallCDeclFunctionObjFirst(obj, args, paramSize<<2, func);
- break;
- case ICC_CDECL_OBJFIRST_RETURNINMEM:
- // Call the system object method as a cdecl with the obj ref as the first parameter
- retQW = CallCDeclFunctionRetByRefObjFirst(obj, args, paramSize<<2, func, retPointer);
- break;
- default:
- context->SetInternalException(TXT_INVALID_CALLING_CONVENTION);
- }
- // If the return is a float value we need to get the value from the FP register
- if( sysFunc->hostReturnFloat )
- {
- if( sysFunc->hostReturnSize == 1 )
- *(asDWORD*)&retQW = GetReturnedFloat();
- else
- retQW = GetReturnedDouble();
- }
- return retQW;
- }
- // On GCC we need to prevent the compiler from inlining these assembler routines when
- // optimizing for speed (-O3), as the loop labels get duplicated which cause compile errors.
- #ifdef __GNUC__
- #define NOINLINE __attribute ((__noinline__))
- #else
- #define NOINLINE
- #endif
- asQWORD NOINLINE CallCDeclFunction(const asDWORD *args, int paramSize, asFUNCTION_t func)
- {
- volatile asQWORD retQW = 0;
- #if defined ASM_INTEL
- // Copy the data to the real stack. If we fail to do
- // this we may run into trouble in case of exceptions.
- __asm
- {
- // We must save registers that are used
- push ecx
- // Clear the FPU stack, in case the called function doesn't do it by itself
- CLEAR_FPU_STACK
- // Copy arguments from script
- // stack to application stack
- mov ecx, paramSize
- mov eax, args
- add eax, ecx
- cmp ecx, 0
- je endcopy
- copyloop:
- sub eax, 4
- push dword ptr [eax]
- sub ecx, 4
- jne copyloop
- endcopy:
- // Call function
- call [func]
- // Pop arguments from stack
- add esp, paramSize
- // Copy return value from EAX:EDX
- lea ecx, retQW
- mov [ecx], eax
- mov 4[ecx], edx
- // Restore registers
- pop ecx
- }
- #elif defined ASM_AT_N_T
- // It is not possible to rely on ESP or BSP to refer to variables or arguments on the stack
- // depending on compiler settings BSP may not even be used, and the ESP is not always on the
- // same offset from the local variables. Because the code adjusts the ESP register it is not
- // possible to inform the arguments through symbolic names below.
- // It's not also not possible to rely on the memory layout of the function arguments, because
- // on some compiler versions and settings the arguments may be copied to local variables with a
- // different ordering before they are accessed by the rest of the code.
- // I'm copying the arguments into this array where I know the exact memory layout. The address
- // of this array will then be passed to the inline asm in the EDX register.
- volatile asPWORD a[] = {asPWORD(args), asPWORD(paramSize), asPWORD(func)};
- asm __volatile__(
- #ifdef __OPTIMIZE__
- // When compiled with optimizations the stack unwind doesn't work properly,
- // causing exceptions to crash the application. By adding this prologue
- // and the epilogue below, the stack unwind works as it should.
- // TODO: runtime optimize: The prologue/epilogue shouldn't be needed if the correct cfi directives are used below
- "pushl %%ebp \n"
- ".cfi_adjust_cfa_offset 4 \n"
- ".cfi_rel_offset ebp, 0 \n"
- "movl %%esp, %%ebp \n"
- ".cfi_def_cfa_register ebp \n"
- #endif
- _S(CLEAR_FPU_STACK) "\n"
- "pushl %%ebx \n"
- "movl %%edx, %%ebx \n"
- // Need to align the stack pointer so that it is aligned to 16 bytes when making the function call.
- // It is assumed that when entering this function, the stack pointer is already aligned, so we need
- // to calculate how much we will put on the stack during this call.
- "movl 4(%%ebx), %%eax \n" // paramSize
- "addl $4, %%eax \n" // counting esp that we will push on the stack
- "movl %%esp, %%ecx \n"
- "subl %%eax, %%ecx \n"
- "andl $15, %%ecx \n"
- "movl %%esp, %%eax \n"
- "subl %%ecx, %%esp \n"
- "pushl %%eax \n" // Store the original stack pointer
- // Copy all arguments to the stack and call the function
- "movl 4(%%ebx), %%ecx \n" // paramSize
- "movl 0(%%ebx), %%eax \n" // args
- "addl %%ecx, %%eax \n" // push arguments on the stack
- "cmp $0, %%ecx \n"
- "je endcopy \n"
- "copyloop: \n"
- "subl $4, %%eax \n"
- "pushl (%%eax) \n"
- "subl $4, %%ecx \n"
- "jne copyloop \n"
- "endcopy: \n"
- "call *8(%%ebx) \n"
- "addl 4(%%ebx), %%esp \n" // pop arguments
- // Pop the alignment bytes
- "popl %%esp \n"
- "popl %%ebx \n"
- #ifdef __OPTIMIZE__
- // Epilogue
- "movl %%ebp, %%esp \n"
- ".cfi_def_cfa_register esp \n"
- "popl %%ebp \n"
- ".cfi_adjust_cfa_offset -4 \n"
- ".cfi_restore ebp \n"
- #endif
- // Copy EAX:EDX to retQW. As the stack pointer has been
- // restored it is now safe to access the local variable
- "leal %1, %%ecx \n"
- "movl %%eax, 0(%%ecx) \n"
- "movl %%edx, 4(%%ecx) \n"
- : // output
- : "d"(a), "m"(retQW) // input - pass pointer of args in edx, pass pointer of retQW in memory argument
- : "%eax", "%ecx" // clobber
- );
- #endif
- return retQW;
- }
- asQWORD NOINLINE CallCDeclFunctionObjLast(const void *obj, const asDWORD *args, int paramSize, asFUNCTION_t func)
- {
- volatile asQWORD retQW = 0;
- #if defined ASM_INTEL
- // Copy the data to the real stack. If we fail to do
- // this we may run into trouble in case of exceptions.
- __asm
- {
- // We must save registers that are used
- push ecx
- // Clear the FPU stack, in case the called function doesn't do it by itself
- CLEAR_FPU_STACK
- // Push the object pointer as the last argument to the function
- push obj
- // Copy arguments from script
- // stack to application stack
- mov ecx, paramSize
- mov eax, args
- add eax, ecx
- cmp ecx, 0
- je endcopy
- copyloop:
- sub eax, 4
- push dword ptr [eax]
- sub ecx, 4
- jne copyloop
- endcopy:
- // Call function
- call [func]
- // Pop arguments from stack
- add esp, paramSize
- add esp, 4
- // Copy return value from EAX:EDX
- lea ecx, retQW
- mov [ecx], eax
- mov 4[ecx], edx
- // Restore registers
- pop ecx
- }
- #elif defined ASM_AT_N_T
- volatile asPWORD a[] = {asPWORD(obj), asPWORD(args), asPWORD(paramSize), asPWORD(func)};
- asm __volatile__ (
- #ifdef __OPTIMIZE__
- // When compiled with optimizations the stack unwind doesn't work properly,
- // causing exceptions to crash the application. By adding this prologue
- // and the epilogue below, the stack unwind works as it should.
- // TODO: runtime optimize: The prologue/epilogue shouldn't be needed if the correct cfi directives are used below
- "pushl %%ebp \n"
- ".cfi_adjust_cfa_offset 4 \n"
- ".cfi_rel_offset ebp, 0 \n"
- "movl %%esp, %%ebp \n"
- ".cfi_def_cfa_register ebp \n"
- #endif
- _S(CLEAR_FPU_STACK) "\n"
- "pushl %%ebx \n"
- "movl %%edx, %%ebx \n"
- // Need to align the stack pointer so that it is aligned to 16 bytes when making the function call.
- // It is assumed that when entering this function, the stack pointer is already aligned, so we need
- // to calculate how much we will put on the stack during this call.
- "movl 8(%%ebx), %%eax \n" // paramSize
- "addl $8, %%eax \n" // counting esp that we will push on the stack
- "movl %%esp, %%ecx \n"
- "subl %%eax, %%ecx \n"
- "andl $15, %%ecx \n"
- "movl %%esp, %%eax \n"
- "subl %%ecx, %%esp \n"
- "pushl %%eax \n" // Store the original stack pointer
- "pushl 0(%%ebx) \n" // obj
- "movl 8(%%ebx), %%ecx \n" // paramSize
- "movl 4(%%ebx), %%eax \n" // args
- "addl %%ecx, %%eax \n" // push arguments on the stack
- "cmp $0, %%ecx \n"
- "je endcopy8 \n"
- "copyloop8: \n"
- "subl $4, %%eax \n"
- "pushl (%%eax) \n"
- "subl $4, %%ecx \n"
- "jne copyloop8 \n"
- "endcopy8: \n"
- "call *12(%%ebx) \n"
- "addl 8(%%ebx), %%esp \n" // pop arguments
- "addl $4, %%esp \n" // pop obj
- // Pop the alignment bytes
- "popl %%esp \n"
- "popl %%ebx \n"
- #ifdef __OPTIMIZE__
- // Epilogue
- "movl %%ebp, %%esp \n"
- ".cfi_def_cfa_register esp \n"
- "popl %%ebp \n"
- ".cfi_adjust_cfa_offset -4 \n"
- ".cfi_restore ebp \n"
- #endif
- // Copy EAX:EDX to retQW. As the stack pointer has been
- // restored it is now safe to access the local variable
- "leal %1, %%ecx \n"
- "movl %%eax, 0(%%ecx) \n"
- "movl %%edx, 4(%%ecx) \n"
- : // output
- : "d"(a), "m"(retQW) // input - pass pointer of args in edx, pass pointer of retQW in memory argument
- : "%eax", "%ecx" // clobber
- );
- #endif
- return retQW;
- }
- asQWORD NOINLINE CallCDeclFunctionObjFirst(const void *obj, const asDWORD *args, int paramSize, asFUNCTION_t func)
- {
- volatile asQWORD retQW = 0;
- #if defined ASM_INTEL
- // Copy the data to the real stack. If we fail to do
- // this we may run into trouble in case of exceptions.
- __asm
- {
- // We must save registers that are used
- push ecx
- // Clear the FPU stack, in case the called function doesn't do it by itself
- CLEAR_FPU_STACK
- // Copy arguments from script
- // stack to application stack
- mov ecx, paramSize
- mov eax, args
- add eax, ecx
- cmp ecx, 0
- je endcopy
- copyloop:
- sub eax, 4
- push dword ptr [eax]
- sub ecx, 4
- jne copyloop
- endcopy:
- // push object as first parameter
- push obj
- // Call function
- call [func]
- // Pop arguments from stack
- add esp, paramSize
- add esp, 4
- // Copy return value from EAX:EDX
- lea ecx, retQW
- mov [ecx], eax
- mov 4[ecx], edx
- // Restore registers
- pop ecx
- }
- #elif defined ASM_AT_N_T
- volatile asPWORD a[] = {asPWORD(obj), asPWORD(args), asPWORD(paramSize), asPWORD(func)};
- asm __volatile__ (
- #ifdef __OPTIMIZE__
- // When compiled with optimizations the stack unwind doesn't work properly,
- // causing exceptions to crash the application. By adding this prologue
- // and the epilogue below, the stack unwind works as it should.
- // TODO: runtime optimize: The prologue/epilogue shouldn't be needed if the correct cfi directives are used below
- "pushl %%ebp \n"
- ".cfi_adjust_cfa_offset 4 \n"
- ".cfi_rel_offset ebp, 0 \n"
- "movl %%esp, %%ebp \n"
- ".cfi_def_cfa_register ebp \n"
- #endif
- _S(CLEAR_FPU_STACK) "\n"
- "pushl %%ebx \n"
- "movl %%edx, %%ebx \n"
- // Need to align the stack pointer so that it is aligned to 16 bytes when making the function call.
- // It is assumed that when entering this function, the stack pointer is already aligned, so we need
- // to calculate how much we will put on the stack during this call.
- "movl 8(%%ebx), %%eax \n" // paramSize
- "addl $8, %%eax \n" // counting esp that we will push on the stack
- "movl %%esp, %%ecx \n"
- "subl %%eax, %%ecx \n"
- "andl $15, %%ecx \n"
- "movl %%esp, %%eax \n"
- "subl %%ecx, %%esp \n"
- "pushl %%eax \n" // Store the original stack pointer
- "movl 8(%%ebx), %%ecx \n" // paramSize
- "movl 4(%%ebx), %%eax \n" // args
- "addl %%ecx, %%eax \n" // push arguments on the stack
- "cmp $0, %%ecx \n"
- "je endcopy6 \n"
- "copyloop6: \n"
- "subl $4, %%eax \n"
- "pushl (%%eax) \n"
- "subl $4, %%ecx \n"
- "jne copyloop6 \n"
- "endcopy6: \n"
- "pushl 0(%%ebx) \n" // push obj
- "call *12(%%ebx) \n"
- "addl 8(%%ebx), %%esp \n" // pop arguments
- "addl $4, %%esp \n" // pop obj
- // Pop the alignment bytes
- "popl %%esp \n"
- "popl %%ebx \n"
- #ifdef __OPTIMIZE__
- // Epilogue
- "movl %%ebp, %%esp \n"
- ".cfi_def_cfa_register esp \n"
- "popl %%ebp \n"
- ".cfi_adjust_cfa_offset -4 \n"
- ".cfi_restore ebp \n"
- #endif
- // Copy EAX:EDX to retQW. As the stack pointer has been
- // restored it is now safe to access the local variable
- "leal %1, %%ecx \n"
- "movl %%eax, 0(%%ecx) \n"
- "movl %%edx, 4(%%ecx) \n"
- : // output
- : "d"(a), "m"(retQW) // input - pass pointer of args in edx, pass pointer of retQW in memory argument
- : "%eax", "%ecx" // clobber
- );
- #endif
- return retQW;
- }
- asQWORD NOINLINE CallCDeclFunctionRetByRefObjFirst(const void *obj, const asDWORD *args, int paramSize, asFUNCTION_t func, void *retPtr)
- {
- volatile asQWORD retQW = 0;
- #if defined ASM_INTEL
- // Copy the data to the real stack. If we fail to do
- // this we may run into trouble in case of exceptions.
- __asm
- {
- // We must save registers that are used
- push ecx
- // Clear the FPU stack, in case the called function doesn't do it by itself
- CLEAR_FPU_STACK
- // Copy arguments from script
- // stack to application stack
- mov ecx, paramSize
- mov eax, args
- add eax, ecx
- cmp ecx, 0
- je endcopy
- copyloop:
- sub eax, 4
- push dword ptr [eax]
- sub ecx, 4
- jne copyloop
- endcopy:
- // Push the object pointer
- push obj
- // Push the return pointer
- push retPtr;
- // Call function
- call [func]
- // Pop arguments from stack
- add esp, paramSize
- #ifndef CALLEE_POPS_HIDDEN_RETURN_POINTER
- // Pop the return pointer
- add esp, 8
- #else
- add esp, 4
- #endif
- // Copy return value from EAX:EDX
- lea ecx, retQW
- mov [ecx], eax
- mov 4[ecx], edx
- // Restore registers
- pop ecx
- }
- #elif defined ASM_AT_N_T
- volatile asPWORD a[] = {asPWORD(obj), asPWORD(args), asPWORD(paramSize), asPWORD(func), asPWORD(retPtr)};
- asm __volatile__ (
- #ifdef __OPTIMIZE__
- // When compiled with optimizations the stack unwind doesn't work properly,
- // causing exceptions to crash the application. By adding this prologue
- // and the epilogue below, the stack unwind works as it should.
- // TODO: runtime optimize: The prologue/epilogue shouldn't be needed if the correct cfi directives are used below
- "pushl %%ebp \n"
- ".cfi_adjust_cfa_offset 4 \n"
- ".cfi_rel_offset ebp, 0 \n"
- "movl %%esp, %%ebp \n"
- ".cfi_def_cfa_register ebp \n"
- #endif
- _S(CLEAR_FPU_STACK) "\n"
- "pushl %%ebx \n"
- "movl %%edx, %%ebx \n"
- // Need to align the stack pointer so that it is aligned to 16 bytes when making the function call.
- // It is assumed that when entering this function, the stack pointer is already aligned, so we need
- // to calculate how much we will put on the stack during this call.
- "movl 8(%%ebx), %%eax \n" // paramSize
- "addl $12, %%eax \n" // counting esp that we will push on the stack
- "movl %%esp, %%ecx \n"
- "subl %%eax, %%ecx \n"
- "andl $15, %%ecx \n"
- "movl %%esp, %%eax \n"
- "subl %%ecx, %%esp \n"
- "pushl %%eax \n" // Store the original stack pointer
- "movl 8(%%ebx), %%ecx \n" // paramSize
- "movl 4(%%ebx), %%eax \n" // args
- "addl %%ecx, %%eax \n" // push arguments on the stack
- "cmp $0, %%ecx \n"
- "je endcopy5 \n"
- "copyloop5: \n"
- "subl $4, %%eax \n"
- "pushl (%%eax) \n"
- "subl $4, %%ecx \n"
- "jne copyloop5 \n"
- "endcopy5: \n"
- "pushl 0(%%ebx) \n" // push object first
- "pushl 16(%%ebx) \n" // retPtr
- "call *12(%%ebx) \n" // func
- "addl 8(%%ebx), %%esp \n" // pop arguments
- #ifndef CALLEE_POPS_HIDDEN_RETURN_POINTER
- "addl $8, %%esp \n" // Pop the return pointer and object pointer
- #else
- "addl $4, %%esp \n" // Pop the object pointer
- #endif
- // Pop the alignment bytes
- "popl %%esp \n"
- "popl %%ebx \n"
- #ifdef __OPTIMIZE__
- // Epilogue
- "movl %%ebp, %%esp \n"
- ".cfi_def_cfa_register esp \n"
- "popl %%ebp \n"
- ".cfi_adjust_cfa_offset -4 \n"
- ".cfi_restore ebp \n"
- #endif
- // Copy EAX:EDX to retQW. As the stack pointer has been
- // restored it is now safe to access the local variable
- "leal %1, %%ecx \n"
- "movl %%eax, 0(%%ecx) \n"
- "movl %%edx, 4(%%ecx) \n"
- : // output
- : "d"(a), "m"(retQW) // input - pass pointer of args in edx, pass pointer of retQW in memory argument
- : "%eax", "%ecx" // clobber
- );
- #endif
- return retQW;
- }
- asQWORD NOINLINE CallCDeclFunctionRetByRef(const asDWORD *args, int paramSize, asFUNCTION_t func, void *retPtr)
- {
- volatile asQWORD retQW = 0;
- #if defined ASM_INTEL
- // Copy the data to the real stack. If we fail to do
- // this we may run into trouble in case of exceptions.
- __asm
- {
- // We must save registers that are used
- push ecx
- // Clear the FPU stack, in case the called function doesn't do it by itself
- CLEAR_FPU_STACK
- // Copy arguments from script
- // stack to application stack
- mov ecx, paramSize
- mov eax, args
- add eax, ecx
- cmp ecx, 0
- je endcopy
- copyloop:
- sub eax, 4
- push dword ptr [eax]
- sub ecx, 4
- jne copyloop
- endcopy:
- // Push the return pointer
- push retPtr;
- // Call function
- call [func]
- // Pop arguments from stack
- add esp, paramSize
- #ifndef CALLEE_POPS_HIDDEN_RETURN_POINTER
- // Pop the return pointer
- add esp, 4
- #endif
- // Copy return value from EAX:EDX
- lea ecx, retQW
- mov [ecx], eax
- mov 4[ecx], edx
- // Restore registers
- pop ecx
- // return value in EAX or EAX:EDX
- }
- #elif defined ASM_AT_N_T
- volatile asPWORD a[] = {asPWORD(args), asPWORD(paramSize), asPWORD(func), asPWORD(retPtr)};
- asm __volatile__ (
- #ifdef __OPTIMIZE__
- // When compiled with optimizations the stack unwind doesn't work properly,
- // causing exceptions to crash the application. By adding this prologue
- // and the epilogue below, the stack unwind works as it should.
- // TODO: runtime optimize: The prologue/epilogue shouldn't be needed if the correct cfi directives are used below
- "pushl %%ebp \n"
- ".cfi_adjust_cfa_offset 4 \n"
- ".cfi_rel_offset ebp, 0 \n"
- "movl %%esp, %%ebp \n"
- ".cfi_def_cfa_register ebp \n"
- #endif
- _S(CLEAR_FPU_STACK) "\n"
- "pushl %%ebx \n"
- "movl %%edx, %%ebx \n"
- // Need to align the stack pointer so that it is aligned to 16 bytes when making the function call.
- // It is assumed that when entering this function, the stack pointer is already aligned, so we need
- // to calculate how much we will put on the stack during this call.
- "movl 4(%%ebx), %%eax \n" // paramSize
- "addl $8, %%eax \n" // counting esp that we will push on the stack
- "movl %%esp, %%ecx \n"
- "subl %%eax, %%ecx \n"
- "andl $15, %%ecx \n"
- "movl %%esp, %%eax \n"
- "subl %%ecx, %%esp \n"
- "pushl %%eax \n" // Store the original stack pointer
- "movl 4(%%ebx), %%ecx \n" // paramSize
- "movl 0(%%ebx), %%eax \n" // args
- "addl %%ecx, %%eax \n" // push arguments on the stack
- "cmp $0, %%ecx \n"
- "je endcopy7 \n"
- "copyloop7: \n"
- "subl $4, %%eax \n"
- "pushl (%%eax) \n"
- "subl $4, %%ecx \n"
- "jne copyloop7 \n"
- "endcopy7: \n"
- "pushl 12(%%ebx) \n" // retPtr
- "call *8(%%ebx) \n" // func
- "addl 4(%%ebx), %%esp \n" // pop arguments
- #ifndef CALLEE_POPS_HIDDEN_RETURN_POINTER
- "addl $4, %%esp \n" // Pop the return pointer
- #endif
- // Pop the alignment bytes
- "popl %%esp \n"
- "popl %%ebx \n"
- #ifdef __OPTIMIZE__
- // Epilogue
- "movl %%ebp, %%esp \n"
- ".cfi_def_cfa_register esp \n"
- "popl %%ebp \n"
- ".cfi_adjust_cfa_offset -4 \n"
- ".cfi_restore ebp \n"
- #endif
- // Copy EAX:EDX to retQW. As the stack pointer has been
- // restored it is now safe to access the local variable
- "leal %1, %%ecx \n"
- "movl %%eax, 0(%%ecx) \n"
- "movl %%edx, 4(%%ecx) \n"
- : // output
- : "d"(a), "m"(retQW) // input - pass pointer of args in edx, pass pointer of retQW in memory argument
- : "%eax", "%ecx" // clobber
- );
- #endif
- return retQW;
- }
- asQWORD NOINLINE CallCDeclFunctionRetByRefObjLast(const void *obj, const asDWORD *args, int paramSize, asFUNCTION_t func, void *retPtr)
- {
- volatile asQWORD retQW = 0;
- #if defined ASM_INTEL
- // Copy the data to the real stack. If we fail to do
- // this we may run into trouble in case of exceptions.
- __asm
- {
- // We must save registers that are used
- push ecx
- // Clear the FPU stack, in case the called function doesn't do it by itself
- CLEAR_FPU_STACK
- push obj
- // Copy arguments from script
- // stack to application stack
- mov ecx, paramSize
- mov eax, args
- add eax, ecx
- cmp ecx, 0
- je endcopy
- copyloop:
- sub eax, 4
- push dword ptr [eax]
- sub ecx, 4
- jne copyloop
- endcopy:
- // Push the return pointer
- push retPtr;
- // Call function
- call [func]
- // Pop arguments from stack
- add esp, paramSize
- add esp, 4
- #ifndef CALLEE_POPS_HIDDEN_RETURN_POINTER
- // Pop the return pointer
- add esp, 4
- #endif
- // Copy return value from EAX:EDX
- lea ecx, retQW
- mov [ecx], eax
- mov 4[ecx], edx
- // Restore registers
- pop ecx
- }
- #elif defined ASM_AT_N_T
- volatile asPWORD a[] = {asPWORD(obj), asPWORD(args), asPWORD(paramSize), asPWORD(func), asPWORD(retPtr)};
- asm __volatile__ (
- #ifdef __OPTIMIZE__
- // When compiled with optimizations the stack unwind doesn't work properly,
- // causing exceptions to crash the application. By adding this prologue
- // and the epilogue below, the stack unwind works as it should.
- // TODO: runtime optimize: The prologue/epilogue shouldn't be needed if the correct cfi directives are used below
- "pushl %%ebp \n"
- ".cfi_adjust_cfa_offset 4 \n"
- ".cfi_rel_offset ebp, 0 \n"
- "movl %%esp, %%ebp \n"
- ".cfi_def_cfa_register ebp \n"
- #endif
- _S(CLEAR_FPU_STACK) "\n"
- "pushl %%ebx \n"
- "movl %%edx, %%ebx \n"
- // Need to align the stack pointer so that it is aligned to 16 bytes when making the function call.
- // It is assumed that when entering this function, the stack pointer is already aligned, so we need
- // to calculate how much we will put on the stack during this call.
- "movl 8(%%ebx), %%eax \n" // paramSize
- "addl $12, %%eax \n" // counting esp that we will push on the stack
- "movl %%esp, %%ecx \n"
- "subl %%eax, %%ecx \n"
- "andl $15, %%ecx \n"
- "movl %%esp, %%eax \n"
- "subl %%ecx, %%esp \n"
- "pushl %%eax \n" // Store the original stack pointer
- "pushl 0(%%ebx) \n" // obj
- "movl 8(%%ebx), %%ecx \n" // paramSize
- "movl 4(%%ebx), %%eax \n" // args
- "addl %%ecx, %%eax \n" // push arguments on the stack
- "cmp $0, %%ecx \n"
- "je endcopy4 \n"
- "copyloop4: \n"
- "subl $4, %%eax \n"
- "pushl (%%eax) \n"
- "subl $4, %%ecx \n"
- "jne copyloop4 \n"
- "endcopy4: \n"
- "pushl 16(%%ebx) \n" // retPtr
- "call *12(%%ebx) \n" // func
- "addl 8(%%ebx), %%esp \n" // pop arguments
- #ifndef CALLEE_POPS_HIDDEN_RETURN_POINTER
- "addl $8, %%esp \n" // Pop the return pointer and object pointer
- #else
- "addl $4, %%esp \n" // Pop the object pointer
- #endif
- // Pop the alignment bytes
- "popl %%esp \n"
- "popl %%ebx \n"
- #ifdef __OPTIMIZE__
- // Epilogue
- "movl %%ebp, %%esp \n"
- ".cfi_def_cfa_register esp \n"
- "popl %%ebp \n"
- ".cfi_adjust_cfa_offset -4 \n"
- ".cfi_restore ebp \n"
- #endif
- // Copy EAX:EDX to retQW. As the stack pointer has been
- // restored it is now safe to access the local variable
- "leal %1, %%ecx \n"
- "movl %%eax, 0(%%ecx) \n"
- "movl %%edx, 4(%%ecx) \n"
- : // output
- : "d"(a), "m"(retQW) // input - pass pointer of args in edx, pass pointer of retQW in memory argument
- : "%eax", "%ecx" // clobber
- );
- #endif
- return retQW;
- }
- asQWORD NOINLINE CallSTDCallFunction(const asDWORD *args, int paramSize, asFUNCTION_t func)
- {
- volatile asQWORD retQW = 0;
- #if defined ASM_INTEL
- // Copy the data to the real stack. If we fail to do
- // this we may run into trouble in case of exceptions.
- __asm
- {
- // We must save registers that are used
- push ecx
- // Clear the FPU stack, in case the called function doesn't do it by itself
- CLEAR_FPU_STACK
- // Copy arguments from script
- // stack to application stack
- mov ecx, paramSize
- mov eax, args
- add eax, ecx
- cmp ecx, 0
- je endcopy
- copyloop:
- sub eax, 4
- push dword ptr [eax]
- sub ecx, 4
- jne copyloop
- endcopy:
- // Call function
- call [func]
- // The callee already removed parameters from the stack
- // Copy return value from EAX:EDX
- lea ecx, retQW
- mov [ecx], eax
- mov 4[ecx], edx
- // Restore registers
- pop ecx
- }
- #elif defined ASM_AT_N_T
- volatile asPWORD a[] = {asPWORD(args), asPWORD(paramSize), asPWORD(func)};
- asm __volatile__ (
- #ifdef __OPTIMIZE__
- // When compiled with optimizations the stack unwind doesn't work properly,
- // causing exceptions to crash the application. By adding this prologue
- // and the epilogue below, the stack unwind works as it should.
- // TODO: runtime optimize: The prologue/epilogue shouldn't be needed if the correct cfi directives are used below
- "pushl %%ebp \n"
- ".cfi_adjust_cfa_offset 4 \n"
- ".cfi_rel_offset ebp, 0 \n"
- "movl %%esp, %%ebp \n"
- ".cfi_def_cfa_register ebp \n"
- #endif
- _S(CLEAR_FPU_STACK) "\n"
- "pushl %%ebx \n"
- "movl %%edx, %%ebx \n"
- // Need to align the stack pointer so that it is aligned to 16 bytes when making the function call.
- // It is assumed that when entering this function, the stack pointer is already aligned, so we need
- // to calculate how much we will put on the stack during this call.
- "movl 4(%%ebx), %%eax \n" // paramSize
- "addl $4, %%eax \n" // counting esp that we will push on the stack
- "movl %%esp, %%ecx \n"
- "subl %%eax, %%ecx \n"
- "andl $15, %%ecx \n"
- "movl %%esp, %%eax \n"
- "subl %%ecx, %%esp \n"
- "pushl %%eax \n" // Store the original stack pointer
- "movl 4(%%ebx), %%ecx \n" // paramSize
- "movl 0(%%ebx), %%eax \n" // args
- "addl %%ecx, %%eax \n" // push arguments on the stack
- "cmp $0, %%ecx \n"
- "je endcopy2 \n"
- "copyloop2: \n"
- "subl $4, %%eax \n"
- "pushl (%%eax) \n"
- "subl $4, %%ecx \n"
- "jne copyloop2 \n"
- "endcopy2: \n"
- "call *8(%%ebx) \n" // callee pops the arguments
- // Pop the alignment bytes
- "popl %%esp \n"
- "popl %%ebx \n"
- #ifdef __OPTIMIZE__
- // Epilogue
- "movl %%ebp, %%esp \n"
- ".cfi_def_cfa_register esp \n"
- "popl %%ebp \n"
- ".cfi_adjust_cfa_offset -4 \n"
- ".cfi_restore ebp \n"
- #endif
- // Copy EAX:EDX to retQW. As the stack pointer has been
- // restored it is now safe to access the local variable
- "leal %1, %%ecx \n"
- "movl %%eax, 0(%%ecx) \n"
- "movl %%edx, 4(%%ecx) \n"
- : // output
- : "d"(a), "m"(retQW) // input - pass pointer of args in edx, pass pointer of retQW in memory argument
- : "%eax", "%ecx" // clobber
- );
- #endif
- return retQW;
- }
- asQWORD NOINLINE CallThisCallFunction(const void *obj, const asDWORD *args, int paramSize, asFUNCTION_t func)
- {
- volatile asQWORD retQW = 0;
- #if defined ASM_INTEL
- // Copy the data to the real stack. If we fail to do
- // this we may run into trouble in case of exceptions.
- __asm
- {
- // We must save registers that are used
- push ecx
- // Clear the FPU stack, in case the called function doesn't do it by itself
- CLEAR_FPU_STACK
- // Copy arguments from script
- // stack to application stack
- mov ecx, paramSize
- mov eax, args
- add eax, ecx
- cmp ecx, 0
- je endcopy
- copyloop:
- sub eax, 4
- push dword ptr [eax]
- sub ecx, 4
- jne copyloop
- endcopy:
- #ifdef THISCALL_PASS_OBJECT_POINTER_ON_THE_STACK
- // Push the object pointer on the stack
- push obj
- #else
- // Move object pointer to ECX
- mov ecx, obj
- #endif
- // Call function
- call [func]
- #ifndef THISCALL_CALLEE_POPS_ARGUMENTS
- // Pop arguments
- add esp, paramSize
- #ifdef THISCALL_PASS_OBJECT_POINTER_ON_THE_STACK
- // Pop object pointer
- add esp, 4
- #endif
- #endif
- // Copy return value from EAX:EDX
- lea ecx, retQW
- mov [ecx], eax
- mov 4[ecx], edx
- // Restore registers
- pop ecx
- }
- #elif defined ASM_AT_N_T
- volatile asPWORD a[] = {asPWORD(obj), asPWORD(args), asPWORD(paramSize), asPWORD(func)};
- asm __volatile__ (
- #ifdef __OPTIMIZE__
- // When compiled with optimizations the stack unwind doesn't work properly,
- // causing exceptions to crash the application. By adding this prologue
- // and the epilogue below, the stack unwind works as it should.
- // TODO: runtime optimize: The prologue/epilogue shouldn't be needed if the correct cfi directives are used below
- "pushl %%ebp \n"
- ".cfi_adjust_cfa_offset 4 \n"
- ".cfi_rel_offset ebp, 0 \n"
- "movl %%esp, %%ebp \n"
- ".cfi_def_cfa_register ebp \n"
- #endif
- _S(CLEAR_FPU_STACK) "\n"
- "pushl %%ebx \n"
- "movl %%edx, %%ebx \n"
- // Need to align the stack pointer so that it is aligned to 16 bytes when making the function call.
- // It is assumed that when entering this function, the stack pointer is already aligned, so we need
- // to calculate how much we will put on the stack during this call.
- "movl 8(%%ebx), %%eax \n" // paramSize
- "addl $8, %%eax \n" // counting esp that we will push on the stack
- "movl %%esp, %%ecx \n"
- "subl %%eax, %%ecx \n"
- "andl $15, %%ecx \n"
- "movl %%esp, %%eax \n"
- "subl %%ecx, %%esp \n"
- "pushl %%eax \n" // Store the original stack pointer
- "movl 8(%%ebx), %%ecx \n" // paramSize
- "movl 4(%%ebx), %%eax \n" // args
- "addl %%ecx, %%eax \n" // push all arguments on the stack
- "cmp $0, %%ecx \n"
- "je endcopy1 \n"
- "copyloop1: \n"
- "subl $4, %%eax \n"
- "pushl (%%eax) \n"
- "subl $4, %%ecx \n"
- "jne copyloop1 \n"
- "endcopy1: \n"
- "movl 0(%%ebx), %%ecx \n" // move obj into ECX
- #ifdef THISCALL_PASS_OBJECT_POINTER_ON_THE_STACK
- "pushl %%ecx \n" // push obj on the stack
- #endif
- "call *12(%%ebx) \n"
- #ifndef THISCALL_CALLEE_POPS_ARGUMENTS
- "addl 8(%%ebx), %%esp \n" // pop arguments
- #ifdef THISCALL_PASS_OBJECT_POINTER_ON_THE_STACK
- "addl $4, %%esp \n" // pop obj
- #endif
- #endif
- // Pop the alignment bytes
- "popl %%esp \n"
- "popl %%ebx \n"
- #ifdef __OPTIMIZE__
- // Epilogue
- "movl %%ebp, %%esp \n"
- ".cfi_def_cfa_register esp \n"
- "popl %%ebp \n"
- ".cfi_adjust_cfa_offset -4 \n"
- ".cfi_restore ebp \n"
- #endif
- // Copy EAX:EDX to retQW. As the stack pointer has been
- // restored it is now safe to access the local variable
- "leal %1, %%ecx \n"
- "movl %%eax, 0(%%ecx) \n"
- "movl %%edx, 4(%%ecx) \n"
- : // output
- : "d"(a), "m"(retQW) // input - pass pointer of args in edx, pass pointer of retQW in memory argument
- : "%eax", "%ecx" // clobber
- );
- #endif
- return retQW;
- }
- asQWORD NOINLINE CallThisCallFunctionRetByRef(const void *obj, const asDWORD *args, int paramSize, asFUNCTION_t func, void *retPtr)
- {
- volatile asQWORD retQW = 0;
- #if defined ASM_INTEL
- // Copy the data to the real stack. If we fail to do
- // this we may run into trouble in case of exceptions.
- __asm
- {
- // We must save registers that are used
- push ecx
- // Clear the FPU stack, in case the called function doesn't do it by itself
- CLEAR_FPU_STACK
- // Copy arguments from script
- // stack to application stack
- mov ecx, paramSize
- mov eax, args
- add eax, ecx
- cmp ecx, 0
- je endcopy
- copyloop:
- sub eax, 4
- push dword ptr [eax]
- sub ecx, 4
- jne copyloop
- endcopy:
- #ifdef THISCALL_PASS_OBJECT_POINTER_ON_THE_STACK
- // Push the object pointer on the stack
- push obj
- #else
- // Move object pointer to ECX
- mov ecx, obj
- #endif
- // Push the return pointer
- push retPtr
- // Call function
- call [func]
- #ifndef THISCALL_CALLEE_POPS_HIDDEN_RETURN_POINTER
- // Pop the return pointer
- add esp, 4
- #endif
- #ifndef THISCALL_CALLEE_POPS_ARGUMENTS
- // Pop arguments
- add esp, paramSize
- #ifdef THISCALL_PASS_OBJECT_POINTER_ON_THE_STACK
- // Pop object pointer
- add esp, 4
- #endif
- #endif
- // Copy return value from EAX:EDX
- lea ecx, retQW
- mov [ecx], eax
- mov 4[ecx], edx
- // Restore registers
- pop ecx
- }
- #elif defined ASM_AT_N_T
- volatile asPWORD a[] = {asPWORD(obj), asPWORD(args), asPWORD(paramSize), asPWORD(func), asPWORD(retPtr)};
- asm __volatile__ (
- #ifdef __OPTIMIZE__
- // When compiled with optimizations the stack unwind doesn't work properly,
- // causing exceptions to crash the application. By adding this prologue
- // and the epilogue below, the stack unwind works as it should.
- // TODO: runtime optimize: The prologue/epilogue shouldn't be needed if the correct cfi directives are used below
- "pushl %%ebp \n"
- ".cfi_adjust_cfa_offset 4 \n"
- ".cfi_rel_offset ebp, 0 \n"
- "movl %%esp, %%ebp \n"
- ".cfi_def_cfa_register ebp \n"
- #endif
- _S(CLEAR_FPU_STACK) "\n"
- "pushl %%ebx \n"
- "movl %%edx, %%ebx \n"
- // Need to align the stack pointer so that it is aligned to 16 bytes when making the function call.
- // It is assumed that when entering this function, the stack pointer is already aligned, so we need
- // to calculate how much we will put on the stack during this call.
- "movl 8(%%ebx), %%eax \n" // paramSize
- "addl $12, %%eax \n" // counting esp that we will push on the stack
- "movl %%esp, %%ecx \n"
- "subl %%eax, %%ecx \n"
- "andl $15, %%ecx \n"
- "movl %%esp, %%eax \n"
- "subl %%ecx, %%esp \n"
- "pushl %%eax \n" // Store the original stack pointer
- "movl 8(%%ebx), %%ecx \n" // paramSize
- "movl 4(%%ebx), %%eax \n" // args
- "addl %%ecx, %%eax \n" // push all arguments to the stack
- "cmp $0, %%ecx \n"
- "je endcopy3 \n"
- "copyloop3: \n"
- "subl $4, %%eax \n"
- "pushl (%%eax) \n"
- "subl $4, %%ecx \n"
- "jne copyloop3 \n"
- "endcopy3: \n"
- #ifdef AS_MINGW47
- // MinGW made some strange choices with 4.7 and the thiscall calling convention,
- // returning an object in memory is completely different from when not returning
- // in memory
- "pushl 0(%%ebx) \n" // push obj on the stack
- "movl 16(%%ebx), %%ecx \n" // move the return pointer into ECX
- "call *12(%%ebx) \n" // call the function
- #else
- "movl 0(%%ebx), %%ecx \n" // move obj into ECX
- #ifdef THISCALL_PASS_OBJECT_POINTER_ON_THE_STACK
- "pushl %%ecx \n" // push obj on the stack
- #endif
- "pushl 16(%%ebx) \n" // push retPtr on the stack
- "call *12(%%ebx) \n"
- #ifndef THISCALL_CALLEE_POPS_HIDDEN_RETURN_POINTER
- "addl $4, %%esp \n" // pop return pointer
- #endif
- #ifndef THISCALL_CALLEE_POPS_ARGUMENTS
- "addl 8(%%ebx), %%esp \n" // pop arguments
- #ifdef THISCALL_PASS_OBJECT_POINTER_ON_THE_STACK
- "addl $4, %%esp \n" // pop the object pointer
- #endif
- #endif
- #endif // AS_MINGW47
- // Pop the alignment bytes
- "popl %%esp \n"
- "popl %%ebx \n"
- #ifdef __OPTIMIZE__
- // Epilogue
- "movl %%ebp, %%esp \n"
- ".cfi_def_cfa_register esp \n"
- "popl %%ebp \n"
- ".cfi_adjust_cfa_offset -4 \n"
- ".cfi_restore ebp \n"
- #endif
- // Copy EAX:EDX to retQW. As the stack pointer has been
- // restored it is now safe to access the local variable
- "leal %1, %%ecx \n"
- "movl %%eax, 0(%%ecx) \n"
- "movl %%edx, 4(%%ecx) \n"
- : // output
- : "d"(a), "m"(retQW) // input - pass pointer of args in edx, pass pointer of retQW in memory argument
- : "%eax", "%ecx" // clobber
- );
- #endif
- return retQW;
- }
- asDWORD GetReturnedFloat()
- {
- asDWORD f;
- #if defined ASM_INTEL
- // Get the float value from ST0
- __asm fstp dword ptr [f]
- #elif defined ASM_AT_N_T
- asm("fstps %0 \n" : "=m" (f));
- #endif
- return f;
- }
- asQWORD GetReturnedDouble()
- {
- asQWORD d;
- #if defined ASM_INTEL
- // Get the double value from ST0
- __asm fstp qword ptr [d]
- #elif defined ASM_AT_N_T
- asm("fstpl %0 \n" : "=m" (d));
- #endif
- return d;
- }
- END_AS_NAMESPACE
- #endif // AS_X86
- #endif // AS_MAX_PORTABILITY
|