2013-04-17 23:09:55 -04:00
|
|
|
// Copyright 2013 Dolphin Emulator Project
|
|
|
|
// Licensed under GPLv2
|
|
|
|
// Refer to the license.txt file included.
|
2008-12-08 05:30:24 +00:00
|
|
|
|
|
|
|
#include <map>
|
|
|
|
|
|
|
|
#include "Common.h"
|
|
|
|
#include "MemoryUtil.h"
|
2013-02-26 13:49:00 -06:00
|
|
|
#include "x64ABI.h"
|
2008-12-19 21:24:52 +00:00
|
|
|
#include "Thunk.h"
|
2008-12-08 05:30:24 +00:00
|
|
|
|
|
|
|
#define THUNK_ARENA_SIZE 1024*1024*1
|
|
|
|
|
2008-12-19 21:24:52 +00:00
|
|
|
namespace
|
|
|
|
{
|
2008-12-08 05:30:24 +00:00
|
|
|
|
2008-12-19 21:24:52 +00:00
|
|
|
static u8 GC_ALIGNED32(saved_fp_state[16 * 4 * 4]);
|
|
|
|
static u8 GC_ALIGNED32(saved_gpr_state[16 * 8]);
|
2008-12-08 05:30:24 +00:00
|
|
|
static u16 saved_mxcsr;
|
|
|
|
|
2008-12-19 21:24:52 +00:00
|
|
|
} // namespace
|
|
|
|
|
|
|
|
using namespace Gen;
|
2008-12-08 05:30:24 +00:00
|
|
|
|
2008-12-19 21:24:52 +00:00
|
|
|
void ThunkManager::Init()
|
|
|
|
{
|
|
|
|
AllocCodeSpace(THUNK_ARENA_SIZE);
|
2008-12-08 05:30:24 +00:00
|
|
|
save_regs = GetCodePtr();
|
|
|
|
for (int i = 2; i < ABI_GetNumXMMRegs(); i++)
|
|
|
|
MOVAPS(M(saved_fp_state + i * 16), (X64Reg)(XMM0 + i));
|
|
|
|
STMXCSR(M(&saved_mxcsr));
|
|
|
|
#ifdef _M_X64
|
|
|
|
MOV(64, M(saved_gpr_state + 0 ), R(RCX));
|
|
|
|
MOV(64, M(saved_gpr_state + 8 ), R(RDX));
|
|
|
|
MOV(64, M(saved_gpr_state + 16), R(R8) );
|
|
|
|
MOV(64, M(saved_gpr_state + 24), R(R9) );
|
|
|
|
MOV(64, M(saved_gpr_state + 32), R(R10));
|
|
|
|
MOV(64, M(saved_gpr_state + 40), R(R11));
|
|
|
|
#ifndef _WIN32
|
|
|
|
MOV(64, M(saved_gpr_state + 48), R(RSI));
|
|
|
|
MOV(64, M(saved_gpr_state + 56), R(RDI));
|
|
|
|
#endif
|
|
|
|
MOV(64, M(saved_gpr_state + 64), R(RBX));
|
|
|
|
#else
|
|
|
|
MOV(32, M(saved_gpr_state + 0 ), R(RCX));
|
|
|
|
MOV(32, M(saved_gpr_state + 4 ), R(RDX));
|
|
|
|
#endif
|
|
|
|
RET();
|
|
|
|
load_regs = GetCodePtr();
|
|
|
|
LDMXCSR(M(&saved_mxcsr));
|
|
|
|
for (int i = 2; i < ABI_GetNumXMMRegs(); i++)
|
|
|
|
MOVAPS((X64Reg)(XMM0 + i), M(saved_fp_state + i * 16));
|
|
|
|
#ifdef _M_X64
|
|
|
|
MOV(64, R(RCX), M(saved_gpr_state + 0 ));
|
|
|
|
MOV(64, R(RDX), M(saved_gpr_state + 8 ));
|
|
|
|
MOV(64, R(R8) , M(saved_gpr_state + 16));
|
|
|
|
MOV(64, R(R9) , M(saved_gpr_state + 24));
|
|
|
|
MOV(64, R(R10), M(saved_gpr_state + 32));
|
|
|
|
MOV(64, R(R11), M(saved_gpr_state + 40));
|
|
|
|
#ifndef _WIN32
|
|
|
|
MOV(64, R(RSI), M(saved_gpr_state + 48));
|
|
|
|
MOV(64, R(RDI), M(saved_gpr_state + 56));
|
|
|
|
#endif
|
|
|
|
MOV(64, R(RBX), M(saved_gpr_state + 64));
|
|
|
|
#else
|
|
|
|
MOV(32, R(RCX), M(saved_gpr_state + 0 ));
|
|
|
|
MOV(32, R(RDX), M(saved_gpr_state + 4 ));
|
|
|
|
#endif
|
|
|
|
RET();
|
|
|
|
}
|
|
|
|
|
2008-12-19 21:24:52 +00:00
|
|
|
void ThunkManager::Reset()
|
2008-12-08 05:30:24 +00:00
|
|
|
{
|
|
|
|
thunks.clear();
|
2008-12-19 21:24:52 +00:00
|
|
|
ResetCodePtr();
|
2008-12-08 05:30:24 +00:00
|
|
|
}
|
|
|
|
|
2008-12-19 21:24:52 +00:00
|
|
|
void ThunkManager::Shutdown()
|
2008-12-08 05:30:24 +00:00
|
|
|
{
|
2008-12-19 21:24:52 +00:00
|
|
|
Reset();
|
|
|
|
FreeCodeSpace();
|
2008-12-08 05:30:24 +00:00
|
|
|
}
|
|
|
|
|
2008-12-19 21:24:52 +00:00
|
|
|
void *ThunkManager::ProtectFunction(void *function, int num_params)
|
2008-12-08 05:30:24 +00:00
|
|
|
{
|
|
|
|
std::map<void *, const u8 *>::iterator iter;
|
|
|
|
iter = thunks.find(function);
|
|
|
|
if (iter != thunks.end())
|
|
|
|
return (void *)iter->second;
|
2008-12-19 21:24:52 +00:00
|
|
|
if (!region)
|
2008-12-08 05:30:24 +00:00
|
|
|
PanicAlert("Trying to protect functions before the emu is started. Bad bad bad.");
|
|
|
|
|
|
|
|
const u8 *call_point = GetCodePtr();
|
2013-09-22 14:29:35 -04:00
|
|
|
#ifdef _M_X64
|
2013-09-22 15:48:27 -04:00
|
|
|
// Make sure to align stack.
|
|
|
|
ABI_AlignStack(0, true);
|
2008-12-08 05:30:24 +00:00
|
|
|
CALL((void*)save_regs);
|
|
|
|
CALL((void*)function);
|
|
|
|
CALL((void*)load_regs);
|
2013-09-22 15:48:27 -04:00
|
|
|
ABI_RestoreStack(0, true);
|
2008-12-08 05:30:24 +00:00
|
|
|
RET();
|
|
|
|
#else
|
|
|
|
CALL((void*)save_regs);
|
|
|
|
// Since parameters are in the previous stack frame, not in registers, this takes some
|
|
|
|
// trickery : we simply re-push the parameters. might not be optimal, but that doesn't really
|
|
|
|
// matter.
|
2013-09-22 15:48:27 -04:00
|
|
|
ABI_AlignStack(num_params * 4, true);
|
2013-09-29 16:36:26 -04:00
|
|
|
unsigned int alignedSize = ABI_GetAlignedFrameSize(num_params * 4, true);
|
2008-12-08 05:30:24 +00:00
|
|
|
for (int i = 0; i < num_params; i++) {
|
|
|
|
// ESP is changing, so we do not need i
|
2013-09-22 15:48:27 -04:00
|
|
|
PUSH(32, MDisp(ESP, alignedSize));
|
2008-12-08 05:30:24 +00:00
|
|
|
}
|
|
|
|
CALL(function);
|
2013-09-22 15:48:27 -04:00
|
|
|
ABI_RestoreStack(num_params * 4, true);
|
2008-12-08 05:30:24 +00:00
|
|
|
CALL((void*)load_regs);
|
|
|
|
RET();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
thunks[function] = call_point;
|
|
|
|
return (void *)call_point;
|
|
|
|
}
|