// Copyright 2014 Citra Emulator Project // Licensed under GPLv2 or any later version // Refer to the license.txt file included. #include "core/arm/dyncom/arm_dyncom.h" #include #include #include "core/arm/dyncom/arm_dyncom_interpreter.h" #include "core/arm/dyncom/arm_dyncom_run.h" #include "core/arm/dyncom/arm_dyncom_trans.h" #include "core/arm/skyeye_common/armstate.h" #include "core/arm/skyeye_common/armsupp.h" #include "core/arm/skyeye_common/vfp/vfp.h" #include "core/core.h" #include "core/core_timing.h" ARM_DynCom::ARM_DynCom(PrivilegeMode initial_mode) { state = std::make_unique(initial_mode); } ARM_DynCom::~ARM_DynCom() {} void ARM_DynCom::ClearInstructionCache() { state->instruction_cache.clear(); trans_cache_buf_top = 0; } void ARM_DynCom::SetPC(u32 pc) { state->Reg[15] = pc; } u32 ARM_DynCom::GetPC() const { return state->Reg[15]; } u32 ARM_DynCom::GetReg(int index) const { return state->Reg[index]; } void ARM_DynCom::SetReg(int index, u32 value) { state->Reg[index] = value; } u32 ARM_DynCom::GetVFPReg(int index) const { return state->ExtReg[index]; } void ARM_DynCom::SetVFPReg(int index, u32 value) { state->ExtReg[index] = value; } u32 ARM_DynCom::GetVFPSystemReg(VFPSystemRegister reg) const { return state->VFP[reg]; } void ARM_DynCom::SetVFPSystemReg(VFPSystemRegister reg, u32 value) { state->VFP[reg] = value; } u32 ARM_DynCom::GetCPSR() const { return state->Cpsr; } void ARM_DynCom::SetCPSR(u32 cpsr) { state->Cpsr = cpsr; } u32 ARM_DynCom::GetCP15Register(CP15Register reg) { return state->CP15[reg]; } void ARM_DynCom::SetCP15Register(CP15Register reg, u32 value) { state->CP15[reg] = value; } void ARM_DynCom::AddTicks(u64 ticks) { down_count -= ticks; if (down_count < 0) CoreTiming::Advance(); } void ARM_DynCom::ExecuteInstructions(int num_instructions) { state->NumInstrsToExecute = num_instructions; // Dyncom only breaks on instruction dispatch. This only happens on every instruction when // executing one instruction at a time. Otherwise, if a block is being executed, more // instructions may actually be executed than specified. unsigned ticks_executed = InterpreterMainLoop(state.get()); AddTicks(ticks_executed); } void ARM_DynCom::SaveContext(Core::ThreadContext& ctx) { memcpy(ctx.cpu_registers, state->Reg.data(), sizeof(ctx.cpu_registers)); memcpy(ctx.fpu_registers, state->ExtReg.data(), sizeof(ctx.fpu_registers)); ctx.sp = state->Reg[13]; ctx.lr = state->Reg[14]; ctx.pc = state->Reg[15]; ctx.cpsr = state->Cpsr; ctx.fpscr = state->VFP[VFP_FPSCR]; ctx.fpexc = state->VFP[VFP_FPEXC]; } void ARM_DynCom::LoadContext(const Core::ThreadContext& ctx) { memcpy(state->Reg.data(), ctx.cpu_registers, sizeof(ctx.cpu_registers)); memcpy(state->ExtReg.data(), ctx.fpu_registers, sizeof(ctx.fpu_registers)); state->Reg[13] = ctx.sp; state->Reg[14] = ctx.lr; state->Reg[15] = ctx.pc; state->Cpsr = ctx.cpsr; state->VFP[VFP_FPSCR] = ctx.fpscr; state->VFP[VFP_FPEXC] = ctx.fpexc; } void ARM_DynCom::PrepareReschedule() { state->NumInstrsToExecute = 0; }