/* This file is part of the dynarmic project. * Copyright (c) 2016 MerryMage * SPDX-License-Identifier: 0BSD */ #include #include #include #include #include #include #include "backend/x64/a32_emit_x64.h" #include "backend/x64/a32_jitstate.h" #include "backend/x64/block_of_code.h" #include "backend/x64/callback.h" #include "backend/x64/devirtualize.h" #include "backend/x64/jitstate_info.h" #include "common/assert.h" #include "common/cast_util.h" #include "common/common_types.h" #include "common/llvm_disassemble.h" #include "common/scope_exit.h" #include "frontend/A32/translate/translate.h" #include "frontend/ir/basic_block.h" #include "frontend/ir/location_descriptor.h" #include "ir_opt/passes.h" namespace Dynarmic::A32 { using namespace Backend::X64; static RunCodeCallbacks GenRunCodeCallbacks(A32::UserCallbacks* cb, CodePtr (*LookupBlock)(void* lookup_block_arg), void* arg) { return RunCodeCallbacks{ std::make_unique(LookupBlock, reinterpret_cast(arg)), std::make_unique(Devirtualize<&A32::UserCallbacks::AddTicks>(cb)), std::make_unique(Devirtualize<&A32::UserCallbacks::GetTicksRemaining>(cb)), }; } static std::function GenRCP(const A32::UserConfig& conf) { return [conf](BlockOfCode& code) { if (conf.page_table) { code.mov(code.r14, Common::BitCast(conf.page_table)); } if (conf.fastmem_pointer) { code.mov(code.r13, Common::BitCast(conf.fastmem_pointer)); } }; } struct Jit::Impl { Impl(Jit* jit, A32::UserConfig conf) : block_of_code(GenRunCodeCallbacks(conf.callbacks, &GetCurrentBlockThunk, this), JitStateInfo{jit_state}, GenRCP(conf)) , emitter(block_of_code, conf, jit) , conf(std::move(conf)) , jit_interface(jit) {} A32JitState jit_state; BlockOfCode block_of_code; A32EmitX64 emitter; A32::UserConfig conf; // Requests made during execution to invalidate the cache are queued up here. size_t invalid_cache_generation = 0; boost::icl::interval_set invalid_cache_ranges; bool invalidate_entire_cache = false; void Execute() { const CodePtr current_codeptr = [this]{ // RSB optimization const u32 new_rsb_ptr = (jit_state.rsb_ptr - 1) & A32JitState::RSBPtrMask; if (jit_state.GetUniqueHash() == jit_state.rsb_location_descriptors[new_rsb_ptr]) { jit_state.rsb_ptr = new_rsb_ptr; return reinterpret_cast(jit_state.rsb_codeptrs[new_rsb_ptr]); } return GetCurrentBlock(); }(); block_of_code.RunCode(&jit_state, current_codeptr); } void Step() { block_of_code.StepCode(&jit_state, GetCurrentSingleStep()); } void ExceptionalExit() { if (!conf.wall_clock_cntpct) { const s64 ticks = jit_state.cycles_to_run - jit_state.cycles_remaining; conf.callbacks->AddTicks(ticks); } PerformCacheInvalidation(); } void ChangeProcessorID(size_t value) { conf.processor_id = value; emitter.ChangeProcessorID(value); } void ClearExclusiveState() { jit_state.exclusive_state = 0; } std::string Disassemble(const IR::LocationDescriptor& descriptor) { auto block = GetBasicBlock(descriptor); std::string result = fmt::format("address: {}\nsize: {} bytes\n", block.entrypoint, block.size); result += Common::DisassembleX64(block.entrypoint, reinterpret_cast(block.entrypoint) + block.size); return result; } void PerformCacheInvalidation() { if (invalidate_entire_cache) { jit_state.ResetRSB(); block_of_code.ClearCache(); emitter.ClearCache(); invalid_cache_ranges.clear(); invalidate_entire_cache = false; invalid_cache_generation++; return; } if (invalid_cache_ranges.empty()) { return; } jit_state.ResetRSB(); emitter.InvalidateCacheRanges(invalid_cache_ranges); invalid_cache_ranges.clear(); invalid_cache_generation++; } void RequestCacheInvalidation() { if (jit_interface->is_executing) { jit_state.halt_requested = true; return; } PerformCacheInvalidation(); } private: Jit* jit_interface; static CodePtr GetCurrentBlockThunk(void* this_voidptr) { Jit::Impl& this_ = *static_cast(this_voidptr); return this_.GetCurrentBlock(); } IR::LocationDescriptor GetCurrentLocation() const { return IR::LocationDescriptor{jit_state.GetUniqueHash()}; } CodePtr GetCurrentBlock() { return GetBasicBlock(GetCurrentLocation()).entrypoint; } CodePtr GetCurrentSingleStep() { return GetBasicBlock(A32::LocationDescriptor{GetCurrentLocation()}.SetSingleStepping(true)).entrypoint; } A32EmitX64::BlockDescriptor GetBasicBlock(IR::LocationDescriptor descriptor) { auto block = emitter.GetBasicBlock(descriptor); if (block) return *block; constexpr size_t MINIMUM_REMAINING_CODESIZE = 1 * 1024 * 1024; if (block_of_code.SpaceRemaining() < MINIMUM_REMAINING_CODESIZE) { invalidate_entire_cache = true; PerformCacheInvalidation(); } IR::Block ir_block = A32::Translate(A32::LocationDescriptor{descriptor}, [this](u32 vaddr) { return conf.callbacks->MemoryReadCode(vaddr); }, {conf.define_unpredictable_behaviour, conf.hook_hint_instructions}); if (conf.HasOptimization(OptimizationFlag::GetSetElimination)) { Optimization::A32GetSetElimination(ir_block); Optimization::DeadCodeElimination(ir_block); } if (conf.HasOptimization(OptimizationFlag::ConstProp)) { Optimization::A32ConstantMemoryReads(ir_block, conf.callbacks); Optimization::ConstantPropagation(ir_block); Optimization::DeadCodeElimination(ir_block); } Optimization::VerificationPass(ir_block); return emitter.Emit(ir_block); } }; Jit::Jit(UserConfig conf) : impl(std::make_unique(this, std::move(conf))) {} Jit::~Jit() = default; void Jit::Run() { ASSERT(!is_executing); is_executing = true; SCOPE_EXIT { this->is_executing = false; }; impl->jit_state.halt_requested = false; impl->Execute(); impl->PerformCacheInvalidation(); } void Jit::Step() { ASSERT(!is_executing); is_executing = true; SCOPE_EXIT { this->is_executing = false; }; impl->jit_state.halt_requested = true; impl->Step(); impl->PerformCacheInvalidation(); } void Jit::ClearCache() { impl->invalidate_entire_cache = true; impl->RequestCacheInvalidation(); } void Jit::InvalidateCacheRange(std::uint32_t start_address, std::size_t length) { impl->invalid_cache_ranges.add(boost::icl::discrete_interval::closed(start_address, static_cast(start_address + length - 1))); impl->RequestCacheInvalidation(); } void Jit::Reset() { ASSERT(!is_executing); impl->jit_state = {}; } void Jit::HaltExecution() { impl->jit_state.halt_requested = true; } void Jit::ExceptionalExit() { impl->ExceptionalExit(); is_executing = false; } void Jit::ClearExclusiveState() { impl->ClearExclusiveState(); } void Jit::ChangeProcessorID(size_t new_processor) { impl->ChangeProcessorID(new_processor); } std::array& Jit::Regs() { return impl->jit_state.Reg; } const std::array& Jit::Regs() const { return impl->jit_state.Reg; } std::array& Jit::ExtRegs() { return impl->jit_state.ExtReg; } const std::array& Jit::ExtRegs() const { return impl->jit_state.ExtReg; } u32 Jit::Cpsr() const { return impl->jit_state.Cpsr(); } void Jit::SetCpsr(u32 value) { return impl->jit_state.SetCpsr(value); } u32 Jit::Fpscr() const { return impl->jit_state.Fpscr(); } void Jit::SetFpscr(u32 value) { return impl->jit_state.SetFpscr(value); } Context Jit::SaveContext() const { Context ctx; SaveContext(ctx); return ctx; } struct Context::Impl { A32JitState jit_state; size_t invalid_cache_generation; }; Context::Context() : impl(std::make_unique()) { impl->jit_state.ResetRSB(); } Context::~Context() = default; Context::Context(const Context& ctx) : impl(std::make_unique(*ctx.impl)) {} Context::Context(Context&& ctx) noexcept : impl(std::move(ctx.impl)) {} Context& Context::operator=(const Context& ctx) { *impl = *ctx.impl; return *this; } Context& Context::operator=(Context&& ctx) noexcept { impl = std::move(ctx.impl); return *this; } std::array& Context::Regs() { return impl->jit_state.Reg; } const std::array& Context::Regs() const { return impl->jit_state.Reg; } std::array& Context::ExtRegs() { return impl->jit_state.ExtReg; } const std::array& Context::ExtRegs() const { return impl->jit_state.ExtReg; } std::uint32_t Context::Cpsr() const { return impl->jit_state.Cpsr(); } void Context::SetCpsr(std::uint32_t value) { impl->jit_state.SetCpsr(value); } std::uint32_t Context::Fpscr() const { return impl->jit_state.Fpscr(); } void Context::SetFpscr(std::uint32_t value) { return impl->jit_state.SetFpscr(value); } void Jit::SaveContext(Context& ctx) const { ctx.impl->jit_state.TransferJitState(impl->jit_state, false); ctx.impl->invalid_cache_generation = impl->invalid_cache_generation; } void Jit::LoadContext(const Context& ctx) { bool reset_rsb = ctx.impl->invalid_cache_generation != impl->invalid_cache_generation; impl->jit_state.TransferJitState(ctx.impl->jit_state, reset_rsb); } std::string Jit::Disassemble() const { return Common::DisassembleX64(impl->block_of_code.GetCodeBegin(), impl->block_of_code.getCurr()); } } // namespace Dynarmic::A32