mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-19 04:32:19 +00:00
Fix tail call support in VC++ builds
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@22143 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
ad38cd6ad0
commit
8bc6f934e8
@ -28,20 +28,16 @@ void X86JITInfo::replaceMachineCodeForFunction(void *Old, void *New) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#ifdef _MSC_VER
|
|
||||||
#pragma optimize("y", off)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/// JITCompilerFunction - This contains the address of the JIT function used to
|
/// JITCompilerFunction - This contains the address of the JIT function used to
|
||||||
/// compile a function lazily.
|
/// compile a function lazily.
|
||||||
static TargetJITInfo::JITCompilerFn JITCompilerFunction;
|
static TargetJITInfo::JITCompilerFn JITCompilerFunction;
|
||||||
|
|
||||||
// Provide a wrapper for X86CompilationCallback2 that saves non-traditional
|
// Provide a wrapper for X86CompilationCallback2 that saves non-traditional
|
||||||
// callee saved registers, for the fastcc calling convention.
|
// callee saved registers, for the fastcc calling convention.
|
||||||
extern "C" void X86CompilationCallback(void);
|
extern "C" {
|
||||||
|
#if defined(__i386__) || defined(i386) || defined(_M_IX86)
|
||||||
#if defined(__i386__) || defined(i386)
|
|
||||||
#ifndef _MSC_VER
|
#ifndef _MSC_VER
|
||||||
|
void X86CompilationCallback(void);
|
||||||
asm(
|
asm(
|
||||||
".text\n"
|
".text\n"
|
||||||
".align 8\n"
|
".align 8\n"
|
||||||
@ -57,7 +53,21 @@ asm(
|
|||||||
"popl %ebp\n"
|
"popl %ebp\n"
|
||||||
"ret\n");
|
"ret\n");
|
||||||
#else
|
#else
|
||||||
// FIXME: implement this for VC++
|
extern "C" void *_AddressOfReturnAddress(void);
|
||||||
|
#pragma intrinsic(_AddressOfReturnAddress)
|
||||||
|
|
||||||
|
void X86CompilationCallback2(void);
|
||||||
|
|
||||||
|
_declspec(naked) void X86CompilationCallback(void) {
|
||||||
|
__asm {
|
||||||
|
push eax
|
||||||
|
push edx
|
||||||
|
call X86CompilationCallback2
|
||||||
|
pop edx
|
||||||
|
pop eax
|
||||||
|
ret
|
||||||
|
}
|
||||||
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#else
|
#else
|
||||||
@ -67,6 +77,7 @@ void X86CompilationCallback() {
|
|||||||
abort();
|
abort();
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
/// X86CompilationCallback - This is the target-specific function invoked by the
|
/// X86CompilationCallback - This is the target-specific function invoked by the
|
||||||
/// function stub when we did not know the real target of a call. This function
|
/// function stub when we did not know the real target of a call. This function
|
||||||
@ -74,14 +85,14 @@ void X86CompilationCallback() {
|
|||||||
/// compiler function.
|
/// compiler function.
|
||||||
extern "C" void X86CompilationCallback2() {
|
extern "C" void X86CompilationCallback2() {
|
||||||
#ifdef _MSC_VER
|
#ifdef _MSC_VER
|
||||||
// FIXME: This needs to go up one more level!
|
assert(sizeof(size_t) == 4); // FIXME: handle Win64
|
||||||
unsigned *StackPtr, RetAddr;
|
unsigned *RetAddrLoc = (unsigned *)_AddressOfReturnAddress();
|
||||||
__asm mov StackPtr, ebp;
|
RetAddrLoc += 3; // skip over ret addr, edx, eax
|
||||||
__asm mov eax, DWORD PTR [ebp + 4];
|
unsigned RetAddr = *RetAddrLoc;
|
||||||
__asm mov RetAddr, eax;
|
|
||||||
#else
|
#else
|
||||||
unsigned *StackPtr = (unsigned*)__builtin_frame_address(1);
|
unsigned *StackPtr = (unsigned*)__builtin_frame_address(1);
|
||||||
unsigned RetAddr = (unsigned)(intptr_t)__builtin_return_address(1);
|
unsigned RetAddr = (unsigned)(intptr_t)__builtin_return_address(1);
|
||||||
|
unsigned *RetAddrLoc = &StackPtr[1];
|
||||||
|
|
||||||
// NOTE: __builtin_frame_address doesn't work if frame pointer elimination has
|
// NOTE: __builtin_frame_address doesn't work if frame pointer elimination has
|
||||||
// been performed. Having a variable sized alloca disables frame pointer
|
// been performed. Having a variable sized alloca disables frame pointer
|
||||||
@ -89,7 +100,7 @@ extern "C" void X86CompilationCallback2() {
|
|||||||
alloca(10+(RetAddr >> 31));
|
alloca(10+(RetAddr >> 31));
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
assert(StackPtr[1] == RetAddr &&
|
assert(*RetAddrLoc == RetAddr &&
|
||||||
"Could not find return address on the stack!");
|
"Could not find return address on the stack!");
|
||||||
|
|
||||||
// It's a stub if there is an interrupt marker after the call.
|
// It's a stub if there is an interrupt marker after the call.
|
||||||
@ -123,13 +134,9 @@ extern "C" void X86CompilationCallback2() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Change the return address to reexecute the call instruction...
|
// Change the return address to reexecute the call instruction...
|
||||||
StackPtr[1] -= 5;
|
*RetAddrLoc -= 5;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef _MSC_VER
|
|
||||||
#pragma optimize( "", on )
|
|
||||||
#endif
|
|
||||||
|
|
||||||
TargetJITInfo::LazyResolverFn
|
TargetJITInfo::LazyResolverFn
|
||||||
X86JITInfo::getLazyResolverFunction(JITCompilerFn F) {
|
X86JITInfo::getLazyResolverFunction(JITCompilerFn F) {
|
||||||
JITCompilerFunction = F;
|
JITCompilerFunction = F;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user