2012-11-01 15:19:01 +00:00
|
|
|
// Copyright (C) 2003 Dolphin Project.
|
|
|
|
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
2012-11-04 22:01:49 +00:00
|
|
|
// the Free Software Foundation, version 2.0 or later versions.
|
2012-11-01 15:19:01 +00:00
|
|
|
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License 2.0 for more details.
|
|
|
|
|
|
|
|
// A copy of the GPL 2.0 should have been included with the program.
|
|
|
|
// If not, see http://www.gnu.org/licenses/
|
|
|
|
|
|
|
|
// Official SVN repository and contact information can be found at
|
|
|
|
// http://code.google.com/p/dolphin-emu/
|
|
|
|
|
2016-10-12 15:32:52 +00:00
|
|
|
#include "ppsspp_config.h"
|
|
|
|
#if PPSSPP_ARCH(X86) || PPSSPP_ARCH(AMD64)
|
|
|
|
|
2020-10-03 22:25:21 +00:00
|
|
|
#include "Common/Math/math_util.h"
|
2013-10-21 10:21:22 +00:00
|
|
|
|
2012-11-01 15:19:01 +00:00
|
|
|
#include "ABI.h"
|
|
|
|
#include "x64Emitter.h"
|
|
|
|
|
2013-03-29 19:51:14 +00:00
|
|
|
#include "Core/Core.h"
|
2013-10-21 10:21:22 +00:00
|
|
|
#include "Core/MemMap.h"
|
2013-03-29 19:51:14 +00:00
|
|
|
#include "Core/System.h"
|
|
|
|
#include "Core/MIPS/MIPS.h"
|
|
|
|
#include "Core/CoreTiming.h"
|
|
|
|
#include "Common/MemoryUtil.h"
|
2012-11-01 15:19:01 +00:00
|
|
|
|
2013-10-21 10:21:22 +00:00
|
|
|
#include "Core/MIPS/JitCommon/JitCommon.h"
|
|
|
|
#include "Core/MIPS/x86/Jit.h"
|
2012-11-01 15:19:01 +00:00
|
|
|
|
|
|
|
using namespace Gen;
|
2014-12-13 20:11:36 +00:00
|
|
|
using namespace X64JitConstants;
|
2012-11-01 15:19:01 +00:00
|
|
|
|
2015-10-08 20:15:28 +00:00
|
|
|
extern volatile CoreState coreState;
|
|
|
|
|
|
|
|
namespace MIPSComp
|
|
|
|
{
|
|
|
|
|
2012-11-01 15:19:01 +00:00
|
|
|
//TODO - make an option
|
|
|
|
//#if _DEBUG
|
2016-05-08 19:38:03 +00:00
|
|
|
static bool enableDebug = false;
|
2015-10-08 17:58:37 +00:00
|
|
|
|
2012-11-01 15:19:01 +00:00
|
|
|
//#else
|
|
|
|
// bool enableDebug = false;
|
|
|
|
//#endif
|
|
|
|
|
|
|
|
//static bool enableStatistics = false; //unused?
|
|
|
|
|
|
|
|
//GLOBAL STATIC ALLOCATIONS x86
|
|
|
|
//EAX - ubiquitous scratch register - EVERYBODY scratches this
|
2014-12-17 09:11:33 +00:00
|
|
|
//EBP - Pointer to fpr/gpr regs
|
2012-11-01 15:19:01 +00:00
|
|
|
|
|
|
|
//GLOBAL STATIC ALLOCATIONS x64
|
|
|
|
//EAX - ubiquitous scratch register - EVERYBODY scratches this
|
|
|
|
//RBX - Base pointer of memory
|
2014-12-17 09:11:33 +00:00
|
|
|
//R14 - Pointer to fpr/gpr regs
|
|
|
|
//R15 - Pointer to array of block pointers
|
2012-11-01 15:19:01 +00:00
|
|
|
|
2015-10-08 17:58:37 +00:00
|
|
|
void ImHere() {
|
2013-01-03 11:46:10 +00:00
|
|
|
DEBUG_LOG(CPU, "JIT Here: %08x", currentMIPS->pc);
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|
|
|
|
|
2015-10-08 20:15:28 +00:00
|
|
|
void Jit::GenerateFixedCode(JitOptions &jo) {
|
2021-02-15 17:28:07 +00:00
|
|
|
AlignCodePage();
|
2016-08-28 12:52:08 +00:00
|
|
|
BeginWrite();
|
2015-10-08 17:58:37 +00:00
|
|
|
|
|
|
|
restoreRoundingMode = AlignCode16(); {
|
2017-07-05 11:12:42 +00:00
|
|
|
STMXCSR(MIPSSTATE_VAR(temp));
|
2015-10-08 17:58:37 +00:00
|
|
|
// Clear the rounding mode and flush-to-zero bits back to 0.
|
2017-07-05 11:12:42 +00:00
|
|
|
AND(32, MIPSSTATE_VAR(temp), Imm32(~(7 << 13)));
|
|
|
|
LDMXCSR(MIPSSTATE_VAR(temp));
|
2015-10-08 17:58:37 +00:00
|
|
|
RET();
|
|
|
|
}
|
|
|
|
|
|
|
|
applyRoundingMode = AlignCode16(); {
|
2017-07-05 11:12:42 +00:00
|
|
|
MOV(32, R(EAX), MIPSSTATE_VAR(fcr31));
|
2015-10-10 11:11:38 +00:00
|
|
|
AND(32, R(EAX), Imm32(0x01000003));
|
2015-10-08 17:58:37 +00:00
|
|
|
|
2015-10-10 11:11:38 +00:00
|
|
|
// If it's 0 (nearest + no flush0), we don't actually bother setting - we cleared the rounding
|
|
|
|
// mode out in restoreRoundingMode anyway. This is the most common.
|
2015-10-08 17:58:37 +00:00
|
|
|
FixupBranch skip = J_CC(CC_Z);
|
2017-07-05 11:12:42 +00:00
|
|
|
STMXCSR(MIPSSTATE_VAR(temp));
|
2015-10-08 17:58:37 +00:00
|
|
|
|
|
|
|
// The MIPS bits don't correspond exactly, so we have to adjust.
|
|
|
|
// 0 -> 0 (skip2), 1 -> 3, 2 -> 2 (skip2), 3 -> 1
|
|
|
|
TEST(8, R(AL), Imm8(1));
|
|
|
|
FixupBranch skip2 = J_CC(CC_Z);
|
|
|
|
XOR(32, R(EAX), Imm8(2));
|
|
|
|
SetJumpTarget(skip2);
|
|
|
|
|
2015-10-10 11:11:38 +00:00
|
|
|
// Adjustment complete, now reconstruct MXCSR
|
2015-10-08 17:58:37 +00:00
|
|
|
SHL(32, R(EAX), Imm8(13));
|
2015-10-10 11:11:38 +00:00
|
|
|
// Before setting new bits, we must clear the old ones.
|
2017-07-05 11:12:42 +00:00
|
|
|
AND(32, MIPSSTATE_VAR(temp), Imm32(~(7 << 13))); // Clearing bits 13-14 (rounding mode) and 15 (flush to zero)
|
|
|
|
OR(32, MIPSSTATE_VAR(temp), R(EAX));
|
2015-10-08 17:58:37 +00:00
|
|
|
|
2017-07-05 11:12:42 +00:00
|
|
|
TEST(32, MIPSSTATE_VAR(fcr31), Imm32(1 << 24));
|
2015-10-08 17:58:37 +00:00
|
|
|
FixupBranch skip3 = J_CC(CC_Z);
|
2017-07-05 11:12:42 +00:00
|
|
|
OR(32, MIPSSTATE_VAR(temp), Imm32(1 << 15));
|
2015-10-08 17:58:37 +00:00
|
|
|
SetJumpTarget(skip3);
|
|
|
|
|
2017-07-05 11:12:42 +00:00
|
|
|
LDMXCSR(MIPSSTATE_VAR(temp));
|
2015-10-08 17:58:37 +00:00
|
|
|
SetJumpTarget(skip);
|
|
|
|
RET();
|
|
|
|
}
|
|
|
|
|
|
|
|
enterDispatcher = AlignCode16();
|
2012-11-01 15:19:01 +00:00
|
|
|
ABI_PushAllCalleeSavedRegsAndAdjustStack();
|
2021-03-03 05:49:21 +00:00
|
|
|
#if PPSSPP_ARCH(AMD64)
|
2012-11-01 15:19:01 +00:00
|
|
|
// Two statically allocated registers.
|
2014-12-17 09:11:33 +00:00
|
|
|
MOV(64, R(MEMBASEREG), ImmPtr(Memory::base));
|
2015-10-08 20:15:28 +00:00
|
|
|
uintptr_t jitbase = (uintptr_t)GetBasePtr();
|
|
|
|
if (jitbase > 0x7FFFFFFFULL) {
|
|
|
|
MOV(64, R(JITBASEREG), ImmPtr(GetBasePtr()));
|
|
|
|
jo.reserveR15ForAsm = true;
|
2014-12-17 16:09:59 +00:00
|
|
|
}
|
2012-11-01 15:19:01 +00:00
|
|
|
#endif
|
2014-12-17 09:00:21 +00:00
|
|
|
// From the start of the FP reg, a single byte offset can reach all GPR + all FPR (but no VFPUR)
|
2015-10-08 20:15:28 +00:00
|
|
|
MOV(PTRBITS, R(CTXREG), ImmPtr(&mips_->f[0]));
|
2012-11-01 15:19:01 +00:00
|
|
|
|
|
|
|
outerLoop = GetCodePtr();
|
2015-10-08 20:15:28 +00:00
|
|
|
RestoreRoundingMode(true);
|
2012-11-01 15:19:01 +00:00
|
|
|
ABI_CallFunction(reinterpret_cast<void *>(&CoreTiming::Advance));
|
2015-10-08 20:15:28 +00:00
|
|
|
ApplyRoundingMode(true);
|
2017-03-14 11:32:20 +00:00
|
|
|
FixupBranch skipToCoreStateCheck = J(); //skip the downcount check
|
2012-12-25 08:01:17 +00:00
|
|
|
|
|
|
|
dispatcherCheckCoreState = GetCodePtr();
|
|
|
|
|
2013-01-24 06:24:20 +00:00
|
|
|
// The result of slice decrementation should be in flags if somebody jumped here
|
|
|
|
// IMPORTANT - We jump on negative, not carry!!!
|
|
|
|
FixupBranch bailCoreState = J_CC(CC_S, true);
|
|
|
|
|
2017-03-14 11:32:20 +00:00
|
|
|
SetJumpTarget(skipToCoreStateCheck);
|
2017-07-05 13:06:44 +00:00
|
|
|
if (RipAccessible((const void *)&coreState)) {
|
2017-07-05 13:59:17 +00:00
|
|
|
CMP(32, M(&coreState), Imm32(0)); // rip accessible
|
2017-07-05 13:06:44 +00:00
|
|
|
} else {
|
|
|
|
MOV(PTRBITS, R(RAX), ImmPtr((const void *)&coreState));
|
|
|
|
CMP(32, MatR(RAX), Imm32(0));
|
|
|
|
}
|
2012-12-25 08:01:17 +00:00
|
|
|
FixupBranch badCoreState = J_CC(CC_NZ, true);
|
2013-01-08 22:52:11 +00:00
|
|
|
FixupBranch skipToRealDispatch2 = J(); //skip the sync and compare first time
|
2012-12-25 08:01:17 +00:00
|
|
|
|
2012-11-01 15:19:01 +00:00
|
|
|
dispatcher = GetCodePtr();
|
2013-01-24 06:24:20 +00:00
|
|
|
|
2012-11-01 15:19:01 +00:00
|
|
|
// The result of slice decrementation should be in flags if somebody jumped here
|
|
|
|
// IMPORTANT - We jump on negative, not carry!!!
|
2013-01-08 20:24:42 +00:00
|
|
|
FixupBranch bail = J_CC(CC_S, true);
|
2012-11-01 15:19:01 +00:00
|
|
|
|
2013-01-08 22:52:11 +00:00
|
|
|
SetJumpTarget(skipToRealDispatch2);
|
2012-11-01 15:19:01 +00:00
|
|
|
|
|
|
|
dispatcherNoCheck = GetCodePtr();
|
|
|
|
|
2017-07-05 11:12:42 +00:00
|
|
|
MOV(32, R(EAX), MIPSSTATE_VAR(pc));
|
2014-12-17 09:12:41 +00:00
|
|
|
dispatcherInEAXNoCheck = GetCodePtr();
|
2014-12-17 09:00:21 +00:00
|
|
|
|
2017-02-28 00:47:13 +00:00
|
|
|
#ifdef MASKED_PSP_MEMORY
|
2012-11-01 15:19:01 +00:00
|
|
|
AND(32, R(EAX), Imm32(Memory::MEMVIEW32_MASK));
|
2017-02-28 00:47:13 +00:00
|
|
|
#endif
|
2021-02-09 00:30:31 +00:00
|
|
|
dispatcherFetch = GetCodePtr();
|
2021-03-03 05:49:21 +00:00
|
|
|
#if PPSSPP_ARCH(X86)
|
2020-07-19 15:47:02 +00:00
|
|
|
_assert_msg_( Memory::base != 0, "Memory base bogus");
|
2013-01-21 05:07:57 +00:00
|
|
|
MOV(32, R(EAX), MDisp(EAX, (u32)Memory::base));
|
2021-03-03 05:49:21 +00:00
|
|
|
#elif PPSSPP_ARCH(AMD64)
|
2014-12-17 09:11:33 +00:00
|
|
|
MOV(32, R(EAX), MComplex(MEMBASEREG, RAX, SCALE_1, 0));
|
2012-11-01 15:19:01 +00:00
|
|
|
#endif
|
|
|
|
MOV(32, R(EDX), R(EAX));
|
2020-07-19 15:47:02 +00:00
|
|
|
_assert_msg_(MIPS_JITBLOCK_MASK == 0xFF000000, "Hardcoded assumption of emuhack mask");
|
2014-12-17 16:25:18 +00:00
|
|
|
SHR(32, R(EDX), Imm8(24));
|
|
|
|
CMP(32, R(EDX), Imm8(MIPS_EMUHACK_OPCODE >> 24));
|
|
|
|
FixupBranch notfound = J_CC(CC_NE);
|
2017-07-05 11:12:42 +00:00
|
|
|
if (enableDebug) {
|
|
|
|
ADD(32, MIPSSTATE_VAR(debugCount), Imm8(1));
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|
|
|
|
//grab from list and jump to it
|
|
|
|
AND(32, R(EAX), Imm32(MIPS_EMUHACK_VALUE_MASK));
|
2021-03-03 05:49:21 +00:00
|
|
|
#if PPSSPP_ARCH(X86)
|
2015-10-08 20:15:28 +00:00
|
|
|
ADD(32, R(EAX), ImmPtr(GetBasePtr()));
|
2021-03-03 05:49:21 +00:00
|
|
|
#elif PPSSPP_ARCH(AMD64)
|
2022-01-30 23:15:38 +00:00
|
|
|
if (jo.reserveR15ForAsm) {
|
2014-12-17 16:09:59 +00:00
|
|
|
ADD(64, R(RAX), R(JITBASEREG));
|
2022-01-30 23:15:38 +00:00
|
|
|
} else {
|
|
|
|
// See above, reserveR15ForAsm is used when above 0x7FFFFFFF.
|
|
|
|
ADD(64, R(EAX), Imm32((u32)jitbase));
|
|
|
|
}
|
2012-11-01 15:19:01 +00:00
|
|
|
#endif
|
2013-04-26 22:22:18 +00:00
|
|
|
JMPptr(R(EAX));
|
2012-11-01 15:19:01 +00:00
|
|
|
SetJumpTarget(notfound);
|
|
|
|
|
|
|
|
//Ok, no block, let's jit
|
2015-10-08 20:15:28 +00:00
|
|
|
RestoreRoundingMode(true);
|
2014-12-12 23:09:37 +00:00
|
|
|
ABI_CallFunction(&MIPSComp::JitAt);
|
2015-10-08 20:15:28 +00:00
|
|
|
ApplyRoundingMode(true);
|
2014-09-03 16:50:24 +00:00
|
|
|
JMP(dispatcherNoCheck, true); // Let's just dispatch again, we'll enter the block since we know it's there.
|
2012-11-01 15:19:01 +00:00
|
|
|
|
|
|
|
SetJumpTarget(bail);
|
2013-01-24 06:24:20 +00:00
|
|
|
SetJumpTarget(bailCoreState);
|
2012-12-25 08:01:17 +00:00
|
|
|
|
2017-07-05 13:06:44 +00:00
|
|
|
if (RipAccessible((const void *)&coreState)) {
|
2017-07-05 13:59:17 +00:00
|
|
|
CMP(32, M(&coreState), Imm32(0)); // rip accessible
|
2017-07-05 13:06:44 +00:00
|
|
|
} else {
|
|
|
|
MOV(PTRBITS, R(RAX), ImmPtr((const void *)&coreState));
|
|
|
|
CMP(32, MatR(RAX), Imm32(0));
|
|
|
|
}
|
2012-11-01 15:19:01 +00:00
|
|
|
J_CC(CC_Z, outerLoop, true);
|
|
|
|
|
2019-02-12 09:58:20 +00:00
|
|
|
const uint8_t *quitLoop = GetCodePtr();
|
2012-12-25 08:01:17 +00:00
|
|
|
SetJumpTarget(badCoreState);
|
2015-10-08 20:15:28 +00:00
|
|
|
RestoreRoundingMode(true);
|
2012-11-01 15:19:01 +00:00
|
|
|
ABI_PopAllCalleeSavedRegsAndAdjustStack();
|
|
|
|
RET();
|
|
|
|
|
2019-02-12 09:58:20 +00:00
|
|
|
crashHandler = GetCodePtr();
|
|
|
|
if (RipAccessible((const void *)&coreState)) {
|
|
|
|
MOV(32, M(&coreState), Imm32(CORE_RUNTIME_ERROR));
|
|
|
|
} else {
|
|
|
|
MOV(PTRBITS, R(RAX), ImmPtr((const void *)&coreState));
|
|
|
|
MOV(32, MatR(RAX), Imm32(CORE_RUNTIME_ERROR));
|
|
|
|
}
|
|
|
|
JMP(quitLoop, true);
|
|
|
|
|
2016-08-28 12:52:08 +00:00
|
|
|
// Let's spare the pre-generated code from unprotect-reprotect.
|
2016-08-28 11:35:27 +00:00
|
|
|
endOfPregeneratedCode = AlignCodePage();
|
2016-08-28 12:52:08 +00:00
|
|
|
EndWrite();
|
2014-09-03 16:50:24 +00:00
|
|
|
}
|
2015-10-08 20:15:28 +00:00
|
|
|
|
2016-10-12 15:32:52 +00:00
|
|
|
} // namespace
|
|
|
|
|
|
|
|
#endif // PPSSPP_ARCH(X86) || PPSSPP_ARCH(AMD64)
|