mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-21 19:32:16 +00:00
Initial implementation of exiting CompilationCallback
This should save all argument registers on entry and restore on exit, despite that, simple things seem to work!!! git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@18161 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
22cd028eef
commit
d9d06b3af1
@ -54,16 +54,19 @@ static void CompilationCallback() {
|
|||||||
//uint64_t FPRegs[13];
|
//uint64_t FPRegs[13];
|
||||||
unsigned *CameFromStub = (unsigned*)__builtin_return_address(0);
|
unsigned *CameFromStub = (unsigned*)__builtin_return_address(0);
|
||||||
unsigned *CameFromOrig = (unsigned*)__builtin_return_address(1);
|
unsigned *CameFromOrig = (unsigned*)__builtin_return_address(1);
|
||||||
|
unsigned *CCStackPtr = (unsigned*)__builtin_frame_address(0);
|
||||||
|
//unsigned *StubStackPtr = (unsigned*)__builtin_frame_address(1);
|
||||||
|
unsigned *OrigStackPtr = (unsigned*)__builtin_frame_address(2);
|
||||||
|
|
||||||
// Adjust our pointers to the branches, not the return addresses.
|
// Adjust pointer to the branch, not the return address.
|
||||||
--CameFromStub; --CameFromOrig;
|
--CameFromStub;
|
||||||
|
|
||||||
void *Target = JITCompilerFunction(CameFromStub);
|
void *Target = JITCompilerFunction(CameFromStub);
|
||||||
|
|
||||||
// Check to see if CameFromOrig[-1] is a 'bl' instruction, and if we can
|
// Check to see if CameFromOrig[-1] is a 'bl' instruction, and if we can
|
||||||
// rewrite it to branch directly to the destination. If so, rewrite it so it
|
// rewrite it to branch directly to the destination. If so, rewrite it so it
|
||||||
// does not need to go through the stub anymore.
|
// does not need to go through the stub anymore.
|
||||||
unsigned CameFromOrigInst = *CameFromOrig;
|
unsigned CameFromOrigInst = CameFromOrig[-1];
|
||||||
if ((CameFromOrigInst >> 26) == 18) { // Direct call.
|
if ((CameFromOrigInst >> 26) == 18) { // Direct call.
|
||||||
intptr_t Offset = ((intptr_t)Target-(intptr_t)CameFromOrig) >> 2;
|
intptr_t Offset = ((intptr_t)Target-(intptr_t)CameFromOrig) >> 2;
|
||||||
if (Offset >= -(1 << 23) && Offset < (1 << 23)) { // In range?
|
if (Offset >= -(1 << 23) && Offset < (1 << 23)) { // In range?
|
||||||
@ -71,26 +74,38 @@ static void CompilationCallback() {
|
|||||||
// Clear the original target out:
|
// Clear the original target out:
|
||||||
CameFromOrigInst &= (63 << 26) | 3;
|
CameFromOrigInst &= (63 << 26) | 3;
|
||||||
CameFromOrigInst |= Offset << 2;
|
CameFromOrigInst |= Offset << 2;
|
||||||
*CameFromOrig = CameFromOrigInst;
|
CameFromOrig[-1] = CameFromOrigInst;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Locate the start of the stub. If this is a short call, adjust backwards
|
// Locate the start of the stub. If this is a short call, adjust backwards
|
||||||
// the short amount, otherwise the full amount.
|
// the short amount, otherwise the full amount.
|
||||||
bool isShortStub = (*CameFromStub >> 26) == 18;
|
bool isShortStub = (*CameFromStub >> 26) == 18;
|
||||||
CameFromStub -= isShortStub ? 3 : 7;
|
CameFromStub -= isShortStub ? 2 : 6;
|
||||||
|
|
||||||
// Rewrite the stub with an unconditional branch to the target, for any users
|
// Rewrite the stub with an unconditional branch to the target, for any users
|
||||||
// who took the address of the stub.
|
// who took the address of the stub.
|
||||||
EmitBranchToAt(CameFromStub, Target, false);
|
EmitBranchToAt(CameFromStub, Target, false);
|
||||||
|
|
||||||
|
// Change the SP so that we pop two stack frames off when we return.
|
||||||
|
*CCStackPtr = (intptr_t)OrigStackPtr;
|
||||||
|
|
||||||
|
// Put the address of the stub and the LR value that originally came into the
|
||||||
|
// stub in a place that is easy to get on the stack after we restore all regs.
|
||||||
|
CCStackPtr[2] = (intptr_t)CameFromStub;
|
||||||
|
CCStackPtr[1] = (intptr_t)CameFromOrig;
|
||||||
|
|
||||||
// FIXME: Need to restore the registers from IntRegs/FPRegs.
|
// FIXME: Need to restore the registers from IntRegs/FPRegs.
|
||||||
|
|
||||||
// FIXME: Need to pop two frames off of the stack and return to a place where
|
// Note, this is not a standard epilog!
|
||||||
// we magically reexecute the call, or jump directly to the caller. This
|
#if defined(__POWERPC__) || defined (__ppc__) || defined(_POWER)
|
||||||
// requires inline asm majik.
|
__asm__ __volatile__ ("lwz r0,4(r1)\n" // Get CameFromOrig (LR into stub)
|
||||||
assert(0 && "CompilationCallback not finished yet!");
|
"mtlr r0\n" // Put it in the LR register
|
||||||
|
"lwz r0,8(r1)\n" // Get "CameFromStub"
|
||||||
|
"mtctr r0\n" // Put it into the CTR register
|
||||||
|
"lwz r1,0(r1)\n" // Pop two frames off
|
||||||
|
"bctr\n" ::); // Return to stub!
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user