mirror of
https://github.com/hrydgard/ppsspp.git
synced 2025-04-02 11:01:50 -04:00
241 lines
4.9 KiB
C++
241 lines
4.9 KiB
C++
// Copyright (c) 2012- PPSSPP 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
|
|
// the Free Software Foundation, version 2.0 or later versions.
|
|
|
|
// 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 git repository and contact information can be found at
|
|
// https://github.com/hrydgard/ppsspp and http://www.ppsspp.org/.
|
|
|
|
#include "Common/Serialize/Serializer.h"
|
|
#include "Common/Serialize/SerializeFuncs.h"
|
|
#include "Core/Reporting.h"
|
|
#include "Core/Config.h"
|
|
#include "Core/Core.h"
|
|
#include "Core/CoreTiming.h"
|
|
#include "Core/Debugger/Breakpoints.h"
|
|
#include "Core/Debugger/SymbolMap.h"
|
|
#include "Core/MemMap.h"
|
|
#include "Core/MIPS/MIPS.h"
|
|
#include "Core/MIPS/MIPSCodeUtils.h"
|
|
#include "Core/MIPS/MIPSInt.h"
|
|
#include "Core/MIPS/MIPSTables.h"
|
|
#include "Core/HLE/ReplaceTables.h"
|
|
|
|
#include "FakeEmitter.h"
|
|
#include "FakeJit.h"
|
|
#include "CPUDetect.h"
|
|
|
|
void DisassembleFake(const u8 *data, int size) {
|
|
}
|
|
|
|
namespace MIPSComp
|
|
{
|
|
|
|
FakeJit::FakeJit(MIPSState *mipsState) : blocks(mipsState, this), mips_(mipsState)
|
|
{
|
|
logBlocks = 0;
|
|
dontLogBlocks = 0;
|
|
blocks.Init();
|
|
}
|
|
|
|
void FakeJit::DoState(PointerWrap &p) {
|
|
auto s = p.Section("Jit", 1, 2);
|
|
if (!s)
|
|
return;
|
|
|
|
Do(p, js.startDefaultPrefix);
|
|
if (s >= 2) {
|
|
Do(p, js.hasSetRounding);
|
|
js.lastSetRounding = 0;
|
|
} else {
|
|
js.hasSetRounding = 1;
|
|
}
|
|
|
|
// The debugger sets this so that "go" on a breakpoint will actually... go.
|
|
// But if they load a state, we can end up hitting it by mistake, since it's based on PC and ticks.
|
|
CBreakPoints::SetSkipFirst(0);
|
|
}
|
|
|
|
// This is here so the savestate matches between jit and non-jit.
|
|
void FakeJit::DoDummyState(PointerWrap &p)
|
|
{
|
|
auto s = p.Section("FakeJit", 1, 2);
|
|
if (!s)
|
|
return;
|
|
|
|
bool dummy = false;
|
|
Do(p, dummy);
|
|
if (s >= 2) {
|
|
dummy = true;
|
|
Do(p, dummy);
|
|
}
|
|
}
|
|
|
|
void FakeJit::FlushAll()
|
|
{
|
|
//gpr.FlushAll();
|
|
//fpr.FlushAll();
|
|
FlushPrefixV();
|
|
}
|
|
|
|
void FakeJit::FlushPrefixV()
|
|
{
|
|
}
|
|
|
|
void FakeJit::ClearCache()
|
|
{
|
|
blocks.Clear();
|
|
ClearCodeSpace(0);
|
|
//GenerateFixedCode();
|
|
}
|
|
|
|
void FakeJit::InvalidateCacheAt(u32 em_address, int length) {
|
|
if (blocks.RangeMayHaveEmuHacks(em_address, em_address + length)) {
|
|
blocks.InvalidateICache(em_address, length);
|
|
}
|
|
}
|
|
|
|
void FakeJit::EatInstruction(MIPSOpcode op) {
|
|
MIPSInfo info = MIPSGetInfo(op);
|
|
if (info & DELAYSLOT) {
|
|
ERROR_LOG_REPORT_ONCE(ateDelaySlot, JIT, "Ate a branch op.");
|
|
}
|
|
if (js.inDelaySlot) {
|
|
ERROR_LOG_REPORT_ONCE(ateInDelaySlot, JIT, "Ate an instruction inside a delay slot.");
|
|
}
|
|
|
|
js.numInstructions++;
|
|
js.compilerPC += 4;
|
|
js.downcountAmount += MIPSGetInstructionCycleEstimate(op);
|
|
}
|
|
|
|
void FakeJit::CompileDelaySlot(int flags)
|
|
{
|
|
}
|
|
|
|
|
|
void FakeJit::Compile(u32 em_address) {
|
|
}
|
|
|
|
void FakeJit::RunLoopUntil(u64 globalticks) {
|
|
MIPSInterpret_RunUntil(globalticks);
|
|
}
|
|
|
|
const u8 *FakeJit::DoJit(u32 em_address, JitBlock *b) {
|
|
_assert_(false);
|
|
return nullptr;
|
|
}
|
|
|
|
void FakeJit::AddContinuedBlock(u32 dest)
|
|
{
|
|
}
|
|
|
|
bool FakeJit::DescribeCodePtr(const u8 *ptr, std::string &name)
|
|
{
|
|
// TODO: Not used by anything yet.
|
|
return false;
|
|
}
|
|
|
|
void FakeJit::Comp_RunBlock(MIPSOpcode op)
|
|
{
|
|
// This shouldn't be necessary, the dispatcher should catch us before we get here.
|
|
ERROR_LOG(JIT, "Comp_RunBlock should never be reached!");
|
|
}
|
|
|
|
bool FakeJit::ReplaceJalTo(u32 dest) {
|
|
return true;
|
|
}
|
|
|
|
void FakeJit::Comp_ReplacementFunc(MIPSOpcode op)
|
|
{
|
|
}
|
|
|
|
void FakeJit::Comp_Generic(MIPSOpcode op)
|
|
{
|
|
FlushAll();
|
|
MIPSInterpretFunc func = MIPSGetInterpretFunc(op);
|
|
if (func)
|
|
{
|
|
SaveDowncount();
|
|
RestoreDowncount();
|
|
}
|
|
|
|
const MIPSInfo info = MIPSGetInfo(op);
|
|
if ((info & IS_VFPU) != 0 && (info & VFPU_NO_PREFIX) == 0)
|
|
{
|
|
// If it does eat them, it'll happen in MIPSCompileOp().
|
|
if ((info & OUT_EAT_PREFIX) == 0)
|
|
js.PrefixUnknown();
|
|
}
|
|
}
|
|
|
|
void FakeJit::MovFromPC(FakeReg r) {
|
|
}
|
|
|
|
void FakeJit::MovToPC(FakeReg r) {
|
|
}
|
|
|
|
void FakeJit::SaveDowncount() {
|
|
}
|
|
|
|
void FakeJit::RestoreDowncount() {
|
|
}
|
|
|
|
void FakeJit::WriteDownCount(int offset) {
|
|
}
|
|
|
|
// Abuses R2
|
|
void FakeJit::WriteDownCountR(FakeReg reg) {
|
|
}
|
|
|
|
void FakeJit::RestoreRoundingMode(bool force) {
|
|
}
|
|
|
|
void FakeJit::ApplyRoundingMode(bool force) {
|
|
}
|
|
|
|
void FakeJit::UpdateRoundingMode() {
|
|
}
|
|
|
|
void FakeJit::WriteExit(u32 destination, int exit_num)
|
|
{
|
|
}
|
|
|
|
void FakeJit::WriteExitDestInR(FakeReg Reg)
|
|
{
|
|
}
|
|
|
|
void FakeJit::WriteSyscallExit()
|
|
{
|
|
}
|
|
|
|
#define _RS ((op>>21) & 0x1F)
|
|
#define _RT ((op>>16) & 0x1F)
|
|
#define _RD ((op>>11) & 0x1F)
|
|
#define _FS ((op>>11) & 0x1F)
|
|
#define _FT ((op>>16) & 0x1F)
|
|
#define _FD ((op>>6) & 0x1F)
|
|
#define _POS ((op>>6) & 0x1F)
|
|
#define _SIZE ((op>>11) & 0x1F)
|
|
|
|
//memory regions:
|
|
//
|
|
// 08-0A
|
|
// 48-4A
|
|
// 04-05
|
|
// 44-45
|
|
// mov eax, addrreg
|
|
// shr eax, 28
|
|
// mov eax, [table+eax]
|
|
// mov dreg, [eax+offreg]
|
|
|
|
}
|