2019-11-19 10:30:04 +00:00
|
|
|
#include "cpu_code_cache.h"
|
2020-07-31 07:09:18 +00:00
|
|
|
#include "bus.h"
|
2020-07-31 17:53:53 +00:00
|
|
|
#include "common/assert.h"
|
2020-01-10 03:31:12 +00:00
|
|
|
#include "common/log.h"
|
2019-11-19 10:30:04 +00:00
|
|
|
#include "cpu_core.h"
|
2020-08-08 06:44:12 +00:00
|
|
|
#include "cpu_core_private.h"
|
2019-11-19 10:30:04 +00:00
|
|
|
#include "cpu_disasm.h"
|
2020-10-18 04:43:55 +00:00
|
|
|
#include "settings.h"
|
2019-11-19 14:15:14 +00:00
|
|
|
#include "system.h"
|
2020-07-31 07:09:18 +00:00
|
|
|
#include "timing_event.h"
|
2019-11-19 10:30:04 +00:00
|
|
|
Log_SetChannel(CPU::CodeCache);
|
|
|
|
|
2019-11-27 15:55:33 +00:00
|
|
|
#ifdef WITH_RECOMPILER
|
|
|
|
#include "cpu_recompiler_code_generator.h"
|
|
|
|
#endif
|
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
namespace CPU::CodeCache {
|
2019-11-19 10:30:04 +00:00
|
|
|
|
2019-11-23 03:16:43 +00:00
|
|
|
constexpr bool USE_BLOCK_LINKING = true;
|
2019-11-22 07:57:02 +00:00
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
#ifdef WITH_RECOMPILER
|
2020-08-22 03:24:57 +00:00
|
|
|
|
|
|
|
// Currently remapping the code buffer doesn't work in macOS or Haiku.
|
|
|
|
#if !defined(__HAIKU__) && !defined(__APPLE__)
|
|
|
|
#define USE_STATIC_CODE_BUFFER 1
|
|
|
|
#endif
|
|
|
|
|
2020-11-22 03:59:26 +00:00
|
|
|
#if defined(AARCH32)
|
|
|
|
// Use a smaller code buffer size on AArch32 to have a better chance of being in range.
|
|
|
|
static constexpr u32 RECOMPILER_CODE_CACHE_SIZE = 16 * 1024 * 1024;
|
|
|
|
static constexpr u32 RECOMPILER_FAR_CODE_CACHE_SIZE = 8 * 1024 * 1024;
|
|
|
|
#else
|
2019-12-03 10:45:14 +00:00
|
|
|
static constexpr u32 RECOMPILER_CODE_CACHE_SIZE = 32 * 1024 * 1024;
|
2020-11-22 03:59:26 +00:00
|
|
|
static constexpr u32 RECOMPILER_FAR_CODE_CACHE_SIZE = 16 * 1024 * 1024;
|
|
|
|
#endif
|
2020-11-07 08:49:59 +00:00
|
|
|
static constexpr u32 CODE_WRITE_FAULT_THRESHOLD_FOR_SLOWMEM = 10;
|
2020-08-22 03:24:57 +00:00
|
|
|
|
|
|
|
#ifdef USE_STATIC_CODE_BUFFER
|
2020-07-31 07:09:18 +00:00
|
|
|
static constexpr u32 RECOMPILER_GUARD_SIZE = 4096;
|
|
|
|
alignas(Recompiler::CODE_STORAGE_ALIGNMENT) static u8
|
|
|
|
s_code_storage[RECOMPILER_CODE_CACHE_SIZE + RECOMPILER_FAR_CODE_CACHE_SIZE];
|
2020-08-22 03:20:37 +00:00
|
|
|
#endif
|
2020-08-22 03:24:57 +00:00
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
static JitCodeBuffer s_code_buffer;
|
2020-08-08 05:42:11 +00:00
|
|
|
|
|
|
|
std::array<CodeBlock::HostCodePointer, FAST_MAP_TOTAL_SLOT_COUNT> s_fast_map;
|
2020-10-18 04:43:09 +00:00
|
|
|
DispatcherFunction s_asm_dispatcher;
|
|
|
|
SingleBlockDispatcherFunction s_single_block_asm_dispatcher;
|
2020-08-08 05:42:11 +00:00
|
|
|
|
|
|
|
ALWAYS_INLINE static u32 GetFastMapIndex(u32 pc)
|
|
|
|
{
|
|
|
|
return ((pc & PHYSICAL_MEMORY_ADDRESS_MASK) >= Bus::BIOS_BASE) ?
|
|
|
|
(FAST_MAP_RAM_SLOT_COUNT + ((pc & Bus::BIOS_MASK) >> 2)) :
|
|
|
|
((pc & Bus::RAM_MASK) >> 2);
|
|
|
|
}
|
|
|
|
|
2020-10-18 04:43:09 +00:00
|
|
|
static void CompileDispatcher();
|
2020-08-08 05:42:11 +00:00
|
|
|
static void FastCompileBlockFunction();
|
|
|
|
|
|
|
|
static void ResetFastMap()
|
|
|
|
{
|
|
|
|
s_fast_map.fill(FastCompileBlockFunction);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void SetFastMap(u32 pc, CodeBlock::HostCodePointer function)
|
|
|
|
{
|
|
|
|
s_fast_map[GetFastMapIndex(pc)] = function;
|
|
|
|
}
|
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
#endif
|
2019-11-19 10:30:04 +00:00
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
using BlockMap = std::unordered_map<u32, CodeBlock*>;
|
2020-10-18 04:43:55 +00:00
|
|
|
using HostCodeMap = std::map<CodeBlock::HostCodePointer, CodeBlock*>;
|
2019-11-19 10:30:04 +00:00
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
void LogCurrentState();
|
|
|
|
|
|
|
|
/// Returns the block key for the current execution state.
|
|
|
|
static CodeBlockKey GetNextBlockKey();
|
|
|
|
|
|
|
|
/// Looks up the block in the cache if it's already been compiled.
|
|
|
|
static CodeBlock* LookupBlock(CodeBlockKey key);
|
|
|
|
|
|
|
|
/// Can the current block execute? This will re-validate the block if necessary.
|
|
|
|
/// The block can also be flushed if recompilation failed, so ignore the pointer if false is returned.
|
|
|
|
static bool RevalidateBlock(CodeBlock* block);
|
|
|
|
|
|
|
|
static bool CompileBlock(CodeBlock* block);
|
2020-12-30 15:48:43 +00:00
|
|
|
static void RemoveReferencesToBlock(CodeBlock* block);
|
2020-07-31 07:09:18 +00:00
|
|
|
static void AddBlockToPageMap(CodeBlock* block);
|
|
|
|
static void RemoveBlockFromPageMap(CodeBlock* block);
|
|
|
|
|
|
|
|
/// Link block from to to.
|
|
|
|
static void LinkBlock(CodeBlock* from, CodeBlock* to);
|
|
|
|
|
|
|
|
/// Unlink all blocks which point to this block, and any that this block links to.
|
|
|
|
static void UnlinkBlock(CodeBlock* block);
|
2019-11-19 10:30:04 +00:00
|
|
|
|
2020-10-18 04:43:55 +00:00
|
|
|
static void ClearState();
|
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
static BlockMap s_blocks;
|
2020-11-22 15:06:25 +00:00
|
|
|
static std::array<std::vector<CodeBlock*>, Bus::RAM_CODE_PAGE_COUNT> m_ram_block_map;
|
2020-07-31 07:09:18 +00:00
|
|
|
|
2020-10-18 04:43:55 +00:00
|
|
|
#ifdef WITH_RECOMPILER
|
|
|
|
static HostCodeMap s_host_code_map;
|
|
|
|
|
|
|
|
static void AddBlockToHostCodeMap(CodeBlock* block);
|
|
|
|
static void RemoveBlockFromHostCodeMap(CodeBlock* block);
|
2020-11-20 15:56:51 +00:00
|
|
|
|
2020-10-18 04:43:55 +00:00
|
|
|
static bool InitializeFastmem();
|
|
|
|
static void ShutdownFastmem();
|
2020-11-22 15:06:25 +00:00
|
|
|
static Common::PageFaultHandler::HandlerResult LUTPageFaultHandler(void* exception_pc, void* fault_address,
|
|
|
|
bool is_write);
|
|
|
|
#ifdef WITH_MMAP_FASTMEM
|
|
|
|
static Common::PageFaultHandler::HandlerResult MMapPageFaultHandler(void* exception_pc, void* fault_address,
|
|
|
|
bool is_write);
|
|
|
|
#endif
|
2020-11-20 15:56:51 +00:00
|
|
|
#endif // WITH_RECOMPILER
|
2020-10-18 04:43:55 +00:00
|
|
|
|
|
|
|
void Initialize()
|
2019-11-19 10:30:04 +00:00
|
|
|
{
|
2020-07-31 07:09:18 +00:00
|
|
|
Assert(s_blocks.empty());
|
2019-11-19 10:30:04 +00:00
|
|
|
|
2019-11-27 15:55:33 +00:00
|
|
|
#ifdef WITH_RECOMPILER
|
2020-10-18 04:43:55 +00:00
|
|
|
if (g_settings.IsUsingRecompiler())
|
|
|
|
{
|
2020-08-22 03:24:57 +00:00
|
|
|
#ifdef USE_STATIC_CODE_BUFFER
|
2020-12-06 08:13:07 +00:00
|
|
|
const bool has_buffer = s_code_buffer.Initialize(s_code_storage, sizeof(s_code_storage),
|
|
|
|
RECOMPILER_FAR_CODE_CACHE_SIZE, RECOMPILER_GUARD_SIZE);
|
2020-08-22 03:20:37 +00:00
|
|
|
#else
|
2020-12-06 08:13:07 +00:00
|
|
|
const bool has_buffer = false;
|
2020-08-22 03:20:37 +00:00
|
|
|
#endif
|
2020-12-06 08:13:07 +00:00
|
|
|
if (!has_buffer && !s_code_buffer.Allocate(RECOMPILER_CODE_CACHE_SIZE, RECOMPILER_FAR_CODE_CACHE_SIZE))
|
2020-10-18 04:43:55 +00:00
|
|
|
{
|
|
|
|
Panic("Failed to initialize code space");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (g_settings.IsUsingFastmem() && !InitializeFastmem())
|
|
|
|
Panic("Failed to initialize fastmem");
|
|
|
|
|
|
|
|
ResetFastMap();
|
|
|
|
CompileDispatcher();
|
2020-07-31 07:09:18 +00:00
|
|
|
}
|
2020-10-18 04:43:55 +00:00
|
|
|
#endif
|
|
|
|
}
|
2020-08-08 05:42:11 +00:00
|
|
|
|
2020-10-18 04:43:55 +00:00
|
|
|
void ClearState()
|
|
|
|
{
|
|
|
|
Bus::ClearRAMCodePageFlags();
|
|
|
|
for (auto& it : m_ram_block_map)
|
|
|
|
it.clear();
|
|
|
|
|
|
|
|
for (const auto& it : s_blocks)
|
|
|
|
delete it.second;
|
|
|
|
|
|
|
|
s_blocks.clear();
|
|
|
|
#ifdef WITH_RECOMPILER
|
|
|
|
s_host_code_map.clear();
|
|
|
|
s_code_buffer.Reset();
|
2020-08-08 05:42:11 +00:00
|
|
|
ResetFastMap();
|
2019-11-27 15:55:33 +00:00
|
|
|
#endif
|
2019-11-19 10:30:04 +00:00
|
|
|
}
|
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
void Shutdown()
|
2019-11-19 10:30:04 +00:00
|
|
|
{
|
2020-10-18 04:43:55 +00:00
|
|
|
ClearState();
|
2020-07-31 17:53:53 +00:00
|
|
|
#ifdef WITH_RECOMPILER
|
2020-11-22 15:06:25 +00:00
|
|
|
ShutdownFastmem();
|
2020-07-31 07:09:18 +00:00
|
|
|
s_code_buffer.Destroy();
|
2020-07-31 17:53:53 +00:00
|
|
|
#endif
|
2020-07-31 07:09:18 +00:00
|
|
|
}
|
2019-11-19 15:19:03 +00:00
|
|
|
|
2020-08-19 13:26:57 +00:00
|
|
|
template<PGXPMode pgxp_mode>
|
|
|
|
static void ExecuteImpl()
|
2020-07-31 07:09:18 +00:00
|
|
|
{
|
|
|
|
CodeBlockKey next_block_key;
|
|
|
|
|
2020-12-16 15:18:02 +00:00
|
|
|
g_using_interpreter = false;
|
2020-07-31 07:09:18 +00:00
|
|
|
g_state.frame_done = false;
|
2020-12-16 15:18:02 +00:00
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
while (!g_state.frame_done)
|
2019-11-19 10:30:04 +00:00
|
|
|
{
|
2020-10-16 15:28:08 +00:00
|
|
|
if (HasPendingInterrupt())
|
|
|
|
{
|
|
|
|
SafeReadInstruction(g_state.regs.pc, &g_state.next_instruction.bits);
|
|
|
|
DispatchInterrupt();
|
|
|
|
}
|
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
TimingEvents::UpdateCPUDowncount();
|
2019-11-19 10:30:04 +00:00
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
next_block_key = GetNextBlockKey();
|
|
|
|
while (g_state.pending_ticks < g_state.downcount)
|
2019-11-19 10:30:04 +00:00
|
|
|
{
|
2020-07-31 07:09:18 +00:00
|
|
|
CodeBlock* block = LookupBlock(next_block_key);
|
|
|
|
if (!block)
|
|
|
|
{
|
2021-01-01 07:16:54 +00:00
|
|
|
InterpretUncachedBlock<pgxp_mode>();
|
2020-07-31 07:09:18 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
reexecute_block:
|
2020-10-16 15:28:08 +00:00
|
|
|
Assert(!(HasPendingInterrupt()));
|
2019-11-23 15:10:51 +00:00
|
|
|
|
2019-11-22 06:45:13 +00:00
|
|
|
#if 0
|
2020-08-08 18:32:52 +00:00
|
|
|
const u32 tick = TimingEvents::GetGlobalTickCounter() + CPU::GetPendingTicks();
|
|
|
|
if (tick == 4188233674)
|
2020-07-31 07:09:18 +00:00
|
|
|
__debugbreak();
|
2019-11-22 06:45:13 +00:00
|
|
|
#endif
|
|
|
|
|
2019-11-23 15:10:51 +00:00
|
|
|
#if 0
|
2020-07-31 07:09:18 +00:00
|
|
|
LogCurrentState();
|
2019-11-23 15:10:51 +00:00
|
|
|
#endif
|
|
|
|
|
2020-08-29 12:07:33 +00:00
|
|
|
if (g_settings.cpu_recompiler_icache)
|
|
|
|
CheckAndUpdateICacheTags(block->icache_line_count, block->uncached_fetch_ticks);
|
|
|
|
|
2020-08-22 15:30:20 +00:00
|
|
|
InterpretCachedBlock<pgxp_mode>(*block);
|
2019-11-19 10:30:04 +00:00
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
if (g_state.pending_ticks >= g_state.downcount)
|
|
|
|
break;
|
2020-10-16 15:28:08 +00:00
|
|
|
else if (!USE_BLOCK_LINKING)
|
2020-07-31 07:09:18 +00:00
|
|
|
continue;
|
2019-11-23 03:16:43 +00:00
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
next_block_key = GetNextBlockKey();
|
|
|
|
if (next_block_key.bits == block->key.bits)
|
2019-11-21 14:32:40 +00:00
|
|
|
{
|
2020-07-31 07:09:18 +00:00
|
|
|
// we can jump straight to it if there's no pending interrupts
|
|
|
|
// ensure it's not a self-modifying block
|
|
|
|
if (!block->invalidated || RevalidateBlock(block))
|
|
|
|
goto reexecute_block;
|
|
|
|
}
|
|
|
|
else if (!block->invalidated)
|
|
|
|
{
|
|
|
|
// Try to find an already-linked block.
|
|
|
|
// TODO: Don't need to dereference the block, just store a pointer to the code.
|
|
|
|
for (CodeBlock* linked_block : block->link_successors)
|
2019-11-23 03:16:43 +00:00
|
|
|
{
|
2020-07-31 07:09:18 +00:00
|
|
|
if (linked_block->key.bits == next_block_key.bits)
|
2019-11-23 03:16:43 +00:00
|
|
|
{
|
2020-07-31 07:09:18 +00:00
|
|
|
if (linked_block->invalidated && !RevalidateBlock(linked_block))
|
|
|
|
{
|
|
|
|
// CanExecuteBlock can result in a block flush, so stop iterating here.
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Execute the linked block
|
|
|
|
block = linked_block;
|
|
|
|
goto reexecute_block;
|
2019-11-23 03:16:43 +00:00
|
|
|
}
|
2020-07-31 07:09:18 +00:00
|
|
|
}
|
2019-11-23 03:16:43 +00:00
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
// No acceptable blocks found in the successor list, try a new one.
|
|
|
|
CodeBlock* next_block = LookupBlock(next_block_key);
|
|
|
|
if (next_block)
|
|
|
|
{
|
|
|
|
// Link the previous block to this new block if we find a new block.
|
|
|
|
LinkBlock(block, next_block);
|
|
|
|
block = next_block;
|
2019-11-21 14:32:40 +00:00
|
|
|
goto reexecute_block;
|
2019-11-23 03:16:43 +00:00
|
|
|
}
|
|
|
|
}
|
2019-11-19 15:19:03 +00:00
|
|
|
}
|
2020-07-31 07:09:18 +00:00
|
|
|
|
|
|
|
TimingEvents::RunEvents();
|
2019-11-19 10:30:04 +00:00
|
|
|
}
|
2019-12-12 13:34:53 +00:00
|
|
|
|
|
|
|
// in case we switch to interpreter...
|
2020-07-31 07:09:18 +00:00
|
|
|
g_state.regs.npc = g_state.regs.pc;
|
2019-11-19 10:30:04 +00:00
|
|
|
}
|
|
|
|
|
2020-08-19 13:26:57 +00:00
|
|
|
void Execute()
|
|
|
|
{
|
|
|
|
if (g_settings.gpu_pgxp_enable)
|
|
|
|
{
|
|
|
|
if (g_settings.gpu_pgxp_cpu)
|
|
|
|
ExecuteImpl<PGXPMode::CPU>();
|
|
|
|
else
|
|
|
|
ExecuteImpl<PGXPMode::Memory>();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ExecuteImpl<PGXPMode::Disabled>();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-08 05:42:11 +00:00
|
|
|
#ifdef WITH_RECOMPILER
|
|
|
|
|
2020-10-18 04:43:09 +00:00
|
|
|
void CompileDispatcher()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
Recompiler::CodeGenerator cg(&s_code_buffer);
|
|
|
|
s_asm_dispatcher = cg.CompileDispatcher();
|
|
|
|
}
|
|
|
|
{
|
|
|
|
Recompiler::CodeGenerator cg(&s_code_buffer);
|
|
|
|
s_single_block_asm_dispatcher = cg.CompileSingleBlockDispatcher();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CodeBlock::HostCodePointer* GetFastMapPointer()
|
|
|
|
{
|
|
|
|
return s_fast_map.data();
|
|
|
|
}
|
|
|
|
|
2020-08-08 05:42:11 +00:00
|
|
|
void ExecuteRecompiler()
|
|
|
|
{
|
2020-12-16 15:18:02 +00:00
|
|
|
g_using_interpreter = false;
|
2020-08-08 05:42:11 +00:00
|
|
|
g_state.frame_done = false;
|
2020-12-16 15:18:02 +00:00
|
|
|
|
2020-10-18 04:43:09 +00:00
|
|
|
#if 0
|
2020-08-08 05:42:11 +00:00
|
|
|
while (!g_state.frame_done)
|
|
|
|
{
|
2020-10-16 15:28:08 +00:00
|
|
|
if (HasPendingInterrupt())
|
|
|
|
{
|
|
|
|
SafeReadInstruction(g_state.regs.pc, &g_state.next_instruction.bits);
|
|
|
|
DispatchInterrupt();
|
|
|
|
}
|
|
|
|
|
2020-08-08 05:42:11 +00:00
|
|
|
TimingEvents::UpdateCPUDowncount();
|
|
|
|
|
|
|
|
while (g_state.pending_ticks < g_state.downcount)
|
|
|
|
{
|
|
|
|
const u32 pc = g_state.regs.pc;
|
|
|
|
g_state.current_instruction_pc = pc;
|
|
|
|
const u32 fast_map_index = GetFastMapIndex(pc);
|
2020-10-26 11:56:56 +00:00
|
|
|
s_single_block_asm_dispatcher[fast_map_index]();
|
2020-08-08 05:42:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TimingEvents::RunEvents();
|
|
|
|
}
|
2020-10-18 04:43:09 +00:00
|
|
|
#else
|
|
|
|
s_asm_dispatcher();
|
|
|
|
#endif
|
2020-08-08 05:42:11 +00:00
|
|
|
|
|
|
|
// in case we switch to interpreter...
|
|
|
|
g_state.regs.npc = g_state.regs.pc;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2020-10-18 04:43:55 +00:00
|
|
|
void Reinitialize()
|
2019-11-23 10:22:09 +00:00
|
|
|
{
|
2020-10-18 04:43:55 +00:00
|
|
|
ClearState();
|
|
|
|
|
2019-11-27 15:55:33 +00:00
|
|
|
#ifdef WITH_RECOMPILER
|
2019-11-23 10:22:09 +00:00
|
|
|
|
2020-10-18 04:43:55 +00:00
|
|
|
ShutdownFastmem();
|
|
|
|
s_code_buffer.Destroy();
|
|
|
|
|
|
|
|
if (g_settings.IsUsingRecompiler())
|
|
|
|
{
|
|
|
|
|
|
|
|
#ifdef USE_STATIC_CODE_BUFFER
|
|
|
|
if (!s_code_buffer.Initialize(s_code_storage, sizeof(s_code_storage), RECOMPILER_FAR_CODE_CACHE_SIZE,
|
|
|
|
RECOMPILER_GUARD_SIZE))
|
|
|
|
#else
|
|
|
|
if (!s_code_buffer.Allocate(RECOMPILER_CODE_CACHE_SIZE, RECOMPILER_FAR_CODE_CACHE_SIZE))
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
Panic("Failed to initialize code space");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (g_settings.IsUsingFastmem() && !InitializeFastmem())
|
|
|
|
Panic("Failed to initialize fastmem");
|
|
|
|
|
|
|
|
ResetFastMap();
|
|
|
|
CompileDispatcher();
|
|
|
|
}
|
2019-11-27 15:55:33 +00:00
|
|
|
#endif
|
2019-11-23 10:22:09 +00:00
|
|
|
}
|
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
void Flush()
|
2019-11-19 10:30:04 +00:00
|
|
|
{
|
2020-10-18 04:43:55 +00:00
|
|
|
ClearState();
|
2020-11-20 15:56:51 +00:00
|
|
|
#ifdef WITH_RECOMPILER
|
2020-10-18 04:43:55 +00:00
|
|
|
if (g_settings.IsUsingRecompiler())
|
|
|
|
CompileDispatcher();
|
2020-11-20 15:56:51 +00:00
|
|
|
#endif
|
2019-11-19 10:30:04 +00:00
|
|
|
}
|
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
void LogCurrentState()
|
2019-11-21 13:33:58 +00:00
|
|
|
{
|
2020-07-31 07:09:18 +00:00
|
|
|
const auto& regs = g_state.regs;
|
2019-12-12 13:34:53 +00:00
|
|
|
WriteToExecutionLog("tick=%u pc=%08X zero=%08X at=%08X v0=%08X v1=%08X a0=%08X a1=%08X a2=%08X a3=%08X t0=%08X "
|
|
|
|
"t1=%08X t2=%08X t3=%08X t4=%08X t5=%08X t6=%08X t7=%08X s0=%08X s1=%08X s2=%08X s3=%08X s4=%08X "
|
|
|
|
"s5=%08X s6=%08X s7=%08X t8=%08X t9=%08X k0=%08X k1=%08X gp=%08X sp=%08X fp=%08X ra=%08X ldr=%s "
|
|
|
|
"ldv=%08X\n",
|
2020-07-31 07:09:18 +00:00
|
|
|
TimingEvents::GetGlobalTickCounter() + GetPendingTicks(), regs.pc, regs.zero, regs.at, regs.v0,
|
|
|
|
regs.v1, regs.a0, regs.a1, regs.a2, regs.a3, regs.t0, regs.t1, regs.t2, regs.t3, regs.t4, regs.t5,
|
|
|
|
regs.t6, regs.t7, regs.s0, regs.s1, regs.s2, regs.s3, regs.s4, regs.s5, regs.s6, regs.s7, regs.t8,
|
|
|
|
regs.t9, regs.k0, regs.k1, regs.gp, regs.sp, regs.fp, regs.ra,
|
|
|
|
(g_state.next_load_delay_reg == Reg::count) ? "NONE" : GetRegName(g_state.next_load_delay_reg),
|
|
|
|
(g_state.next_load_delay_reg == Reg::count) ? 0 : g_state.next_load_delay_value);
|
2019-11-21 13:33:58 +00:00
|
|
|
}
|
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
CodeBlockKey GetNextBlockKey()
|
2019-11-19 10:30:04 +00:00
|
|
|
{
|
|
|
|
CodeBlockKey key = {};
|
2020-07-31 07:09:18 +00:00
|
|
|
key.SetPC(g_state.regs.pc);
|
|
|
|
key.user_mode = InUserMode();
|
2019-11-19 15:19:03 +00:00
|
|
|
return key;
|
|
|
|
}
|
2019-11-19 10:30:04 +00:00
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
CodeBlock* LookupBlock(CodeBlockKey key)
|
2019-11-19 15:19:03 +00:00
|
|
|
{
|
2020-07-31 07:09:18 +00:00
|
|
|
BlockMap::iterator iter = s_blocks.find(key.bits);
|
|
|
|
if (iter != s_blocks.end())
|
2019-11-21 14:32:40 +00:00
|
|
|
{
|
|
|
|
// ensure it hasn't been invalidated
|
|
|
|
CodeBlock* existing_block = iter->second;
|
|
|
|
if (!existing_block || !existing_block->invalidated || RevalidateBlock(existing_block))
|
|
|
|
return existing_block;
|
|
|
|
}
|
2019-11-19 10:30:04 +00:00
|
|
|
|
2019-11-21 14:32:40 +00:00
|
|
|
CodeBlock* block = new CodeBlock(key);
|
2019-11-19 10:30:04 +00:00
|
|
|
if (CompileBlock(block))
|
|
|
|
{
|
2019-11-21 14:32:40 +00:00
|
|
|
// add it to the page map if it's in ram
|
|
|
|
AddBlockToPageMap(block);
|
2020-08-08 05:42:11 +00:00
|
|
|
|
|
|
|
#ifdef WITH_RECOMPILER
|
|
|
|
SetFastMap(block->GetPC(), block->host_code);
|
2020-10-30 10:44:21 +00:00
|
|
|
AddBlockToHostCodeMap(block);
|
2020-08-08 05:42:11 +00:00
|
|
|
#endif
|
2019-11-19 10:30:04 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-11-19 15:19:03 +00:00
|
|
|
Log_ErrorPrintf("Failed to compile block at PC=0x%08X", key.GetPC());
|
2019-11-21 14:32:40 +00:00
|
|
|
delete block;
|
|
|
|
block = nullptr;
|
2019-11-19 10:30:04 +00:00
|
|
|
}
|
|
|
|
|
2020-10-30 10:44:21 +00:00
|
|
|
s_blocks.emplace(key.bits, block);
|
2019-11-19 10:30:04 +00:00
|
|
|
return block;
|
|
|
|
}
|
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
bool RevalidateBlock(CodeBlock* block)
|
2019-11-21 14:32:40 +00:00
|
|
|
{
|
|
|
|
for (const CodeBlockInstruction& cbi : block->instructions)
|
|
|
|
{
|
2020-08-29 12:07:33 +00:00
|
|
|
u32 new_code = 0;
|
|
|
|
SafeReadInstruction(cbi.pc, &new_code);
|
2019-11-21 14:32:40 +00:00
|
|
|
if (cbi.instruction.bits != new_code)
|
|
|
|
{
|
|
|
|
Log_DebugPrintf("Block 0x%08X changed at PC 0x%08X - %08X to %08X - recompiling.", block->GetPC(), cbi.pc,
|
|
|
|
cbi.instruction.bits, new_code);
|
|
|
|
goto recompile;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// re-add it to the page map since it's still up-to-date
|
|
|
|
block->invalidated = false;
|
|
|
|
AddBlockToPageMap(block);
|
2020-08-08 05:42:11 +00:00
|
|
|
#ifdef WITH_RECOMPILER
|
|
|
|
SetFastMap(block->GetPC(), block->host_code);
|
|
|
|
#endif
|
2019-11-21 14:32:40 +00:00
|
|
|
return true;
|
|
|
|
|
|
|
|
recompile:
|
2020-12-30 15:48:43 +00:00
|
|
|
// remove any references to the block from the lookup table.
|
|
|
|
// this is an edge case where compiling causes a flush-all due to no space,
|
|
|
|
// and we don't want to nuke the block we're compiling...
|
|
|
|
RemoveReferencesToBlock(block);
|
|
|
|
|
2020-11-20 15:56:51 +00:00
|
|
|
#ifdef WITH_RECOMPILER
|
2020-10-18 04:43:55 +00:00
|
|
|
RemoveBlockFromHostCodeMap(block);
|
2020-11-20 15:56:51 +00:00
|
|
|
#endif
|
2020-10-18 04:43:55 +00:00
|
|
|
|
2019-11-21 14:32:40 +00:00
|
|
|
block->instructions.clear();
|
|
|
|
if (!CompileBlock(block))
|
|
|
|
{
|
|
|
|
Log_WarningPrintf("Failed to recompile block 0x%08X - flushing.", block->GetPC());
|
2020-12-30 15:48:43 +00:00
|
|
|
delete block;
|
2019-11-21 14:32:40 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-12-30 15:48:43 +00:00
|
|
|
AddBlockToPageMap(block);
|
|
|
|
|
2020-11-20 15:56:51 +00:00
|
|
|
#ifdef WITH_RECOMPILER
|
2019-11-21 14:32:40 +00:00
|
|
|
// re-add to page map again
|
2020-12-30 15:48:43 +00:00
|
|
|
SetFastMap(block->GetPC(), block->host_code);
|
2020-10-18 04:43:55 +00:00
|
|
|
AddBlockToHostCodeMap(block);
|
2020-11-20 15:56:51 +00:00
|
|
|
#endif
|
2019-11-21 14:32:40 +00:00
|
|
|
|
2020-12-30 15:48:43 +00:00
|
|
|
// re-insert into the block map since we removed it earlier.
|
|
|
|
s_blocks.emplace(block->key.bits, block);
|
2019-11-21 14:32:40 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
bool CompileBlock(CodeBlock* block)
|
2019-11-19 10:30:04 +00:00
|
|
|
{
|
|
|
|
u32 pc = block->GetPC();
|
|
|
|
bool is_branch_delay_slot = false;
|
2020-11-18 10:34:25 +00:00
|
|
|
bool is_unconditional_branch_delay_slot = false;
|
2019-11-19 10:30:04 +00:00
|
|
|
bool is_load_delay_slot = false;
|
|
|
|
|
2019-11-22 07:57:02 +00:00
|
|
|
#if 0
|
|
|
|
if (pc == 0x0005aa90)
|
|
|
|
__debugbreak();
|
|
|
|
#endif
|
|
|
|
|
2020-12-30 15:48:43 +00:00
|
|
|
block->icache_line_count = 0;
|
|
|
|
block->uncached_fetch_ticks = 0;
|
|
|
|
block->contains_double_branches = false;
|
|
|
|
block->contains_loadstore_instructions = false;
|
|
|
|
|
2020-08-29 12:07:33 +00:00
|
|
|
u32 last_cache_line = ICACHE_LINES;
|
|
|
|
|
2019-11-19 10:30:04 +00:00
|
|
|
for (;;)
|
|
|
|
{
|
|
|
|
CodeBlockInstruction cbi = {};
|
2020-08-29 12:07:33 +00:00
|
|
|
if (!SafeReadInstruction(pc, &cbi.instruction.bits) || !IsInvalidInstruction(cbi.instruction))
|
2019-11-19 10:30:04 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
cbi.pc = pc;
|
|
|
|
cbi.is_branch_delay_slot = is_branch_delay_slot;
|
|
|
|
cbi.is_load_delay_slot = is_load_delay_slot;
|
2019-11-19 14:15:14 +00:00
|
|
|
cbi.is_branch_instruction = IsBranchInstruction(cbi.instruction);
|
2020-11-18 10:34:25 +00:00
|
|
|
cbi.is_unconditional_branch_instruction = IsUnconditionalBranchInstruction(cbi.instruction);
|
2019-11-19 14:15:14 +00:00
|
|
|
cbi.is_load_instruction = IsMemoryLoadInstruction(cbi.instruction);
|
|
|
|
cbi.is_store_instruction = IsMemoryStoreInstruction(cbi.instruction);
|
|
|
|
cbi.has_load_delay = InstructionHasLoadDelay(cbi.instruction);
|
2020-07-31 07:09:18 +00:00
|
|
|
cbi.can_trap = CanInstructionTrap(cbi.instruction, InUserMode());
|
2019-11-19 10:30:04 +00:00
|
|
|
|
2020-08-29 12:07:33 +00:00
|
|
|
if (g_settings.cpu_recompiler_icache)
|
|
|
|
{
|
|
|
|
const u32 icache_line = GetICacheLine(pc);
|
|
|
|
if (icache_line != last_cache_line)
|
|
|
|
{
|
|
|
|
block->icache_line_count++;
|
|
|
|
last_cache_line = icache_line;
|
|
|
|
}
|
|
|
|
block->uncached_fetch_ticks += GetInstructionReadTicks(pc);
|
|
|
|
}
|
|
|
|
|
2020-10-18 04:43:55 +00:00
|
|
|
block->contains_loadstore_instructions |= cbi.is_load_instruction;
|
|
|
|
block->contains_loadstore_instructions |= cbi.is_store_instruction;
|
|
|
|
|
2020-11-18 10:34:25 +00:00
|
|
|
pc += sizeof(cbi.instruction.bits);
|
|
|
|
|
|
|
|
if (is_branch_delay_slot && cbi.is_branch_instruction)
|
|
|
|
{
|
|
|
|
if (!is_unconditional_branch_delay_slot)
|
|
|
|
{
|
|
|
|
Log_WarningPrintf("Conditional branch delay slot at %08X, skipping block", cbi.pc);
|
|
|
|
return false;
|
|
|
|
}
|
2020-12-13 15:21:49 +00:00
|
|
|
if (!IsDirectBranchInstruction(cbi.instruction))
|
|
|
|
{
|
|
|
|
Log_WarningPrintf("Indirect branch in delay slot at %08X, skipping block", cbi.pc);
|
|
|
|
return false;
|
|
|
|
}
|
2020-11-18 10:34:25 +00:00
|
|
|
|
|
|
|
// change the pc for the second branch's delay slot, it comes from the first branch
|
|
|
|
const CodeBlockInstruction& prev_cbi = block->instructions.back();
|
|
|
|
pc = GetBranchInstructionTarget(prev_cbi.instruction, prev_cbi.pc);
|
|
|
|
Log_DevPrintf("Double branch at %08X, using delay slot from %08X -> %08X", cbi.pc, prev_cbi.pc, pc);
|
|
|
|
}
|
|
|
|
|
2019-11-19 10:30:04 +00:00
|
|
|
// instruction is decoded now
|
|
|
|
block->instructions.push_back(cbi);
|
|
|
|
|
|
|
|
// if we're in a branch delay slot, the block is now done
|
|
|
|
// except if this is a branch in a branch delay slot, then we grab the one after that, and so on...
|
2019-11-19 14:15:14 +00:00
|
|
|
if (is_branch_delay_slot && !cbi.is_branch_instruction)
|
2019-11-19 10:30:04 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
// if this is a branch, we grab the next instruction (delay slot), and then exit
|
2019-11-19 14:15:14 +00:00
|
|
|
is_branch_delay_slot = cbi.is_branch_instruction;
|
2020-11-18 10:34:25 +00:00
|
|
|
is_unconditional_branch_delay_slot = cbi.is_unconditional_branch_instruction;
|
2019-11-19 14:15:14 +00:00
|
|
|
|
|
|
|
// same for load delay
|
|
|
|
is_load_delay_slot = cbi.has_load_delay;
|
2019-11-19 10:30:04 +00:00
|
|
|
|
|
|
|
// is this a non-branchy exit? (e.g. syscall)
|
|
|
|
if (IsExitBlockInstruction(cbi.instruction))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!block->instructions.empty())
|
|
|
|
{
|
|
|
|
block->instructions.back().is_last_instruction = true;
|
|
|
|
|
|
|
|
#ifdef _DEBUG
|
|
|
|
SmallString disasm;
|
|
|
|
Log_DebugPrintf("Block at 0x%08X", block->GetPC());
|
|
|
|
for (const CodeBlockInstruction& cbi : block->instructions)
|
|
|
|
{
|
2020-12-16 15:18:02 +00:00
|
|
|
CPU::DisassembleInstruction(&disasm, cbi.pc, cbi.instruction.bits);
|
2019-11-19 10:30:04 +00:00
|
|
|
Log_DebugPrintf("[%s %s 0x%08X] %08X %s", cbi.is_branch_delay_slot ? "BD" : " ",
|
|
|
|
cbi.is_load_delay_slot ? "LD" : " ", cbi.pc, cbi.instruction.bits, disasm.GetCharArray());
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Log_WarningPrintf("Empty block compiled at 0x%08X", block->key.GetPC());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-11-27 15:55:33 +00:00
|
|
|
#ifdef WITH_RECOMPILER
|
2020-10-18 04:43:55 +00:00
|
|
|
if (g_settings.IsUsingRecompiler())
|
2019-11-19 10:30:04 +00:00
|
|
|
{
|
|
|
|
// Ensure we're not going to run out of space while compiling this block.
|
2020-07-31 07:09:18 +00:00
|
|
|
if (s_code_buffer.GetFreeCodeSpace() <
|
2019-11-22 07:57:02 +00:00
|
|
|
(block->instructions.size() * Recompiler::MAX_NEAR_HOST_BYTES_PER_INSTRUCTION) ||
|
2020-07-31 07:09:18 +00:00
|
|
|
s_code_buffer.GetFreeFarCodeSpace() <
|
2019-11-22 07:57:02 +00:00
|
|
|
(block->instructions.size() * Recompiler::MAX_FAR_HOST_BYTES_PER_INSTRUCTION))
|
2019-11-19 10:30:04 +00:00
|
|
|
{
|
|
|
|
Log_WarningPrintf("Out of code space, flushing all blocks.");
|
2019-11-23 10:22:09 +00:00
|
|
|
Flush();
|
2019-11-19 10:30:04 +00:00
|
|
|
}
|
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
Recompiler::CodeGenerator codegen(&s_code_buffer);
|
2019-11-19 10:30:04 +00:00
|
|
|
if (!codegen.CompileBlock(block, &block->host_code, &block->host_code_size))
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("Failed to compile host code for block at 0x%08X", block->key.GetPC());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2019-11-27 15:55:33 +00:00
|
|
|
#endif
|
2019-11-19 10:30:04 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-08-08 05:42:11 +00:00
|
|
|
#ifdef WITH_RECOMPILER
|
|
|
|
|
|
|
|
void FastCompileBlockFunction()
|
|
|
|
{
|
|
|
|
CodeBlock* block = LookupBlock(GetNextBlockKey());
|
|
|
|
if (block)
|
2020-10-18 04:43:09 +00:00
|
|
|
s_single_block_asm_dispatcher(block->host_code);
|
2021-01-01 07:16:54 +00:00
|
|
|
else if (g_settings.gpu_pgxp_enable)
|
|
|
|
InterpretUncachedBlock<PGXPMode::Memory>();
|
2020-08-08 05:42:11 +00:00
|
|
|
else
|
2021-01-01 07:16:54 +00:00
|
|
|
InterpretUncachedBlock<PGXPMode::Disabled>();
|
2020-08-08 05:42:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
void InvalidateBlocksWithPageIndex(u32 page_index)
|
2019-11-19 10:30:04 +00:00
|
|
|
{
|
2020-11-22 15:06:25 +00:00
|
|
|
DebugAssert(page_index < Bus::RAM_CODE_PAGE_COUNT);
|
2019-11-19 10:30:04 +00:00
|
|
|
auto& blocks = m_ram_block_map[page_index];
|
2019-11-21 14:32:40 +00:00
|
|
|
for (CodeBlock* block : blocks)
|
|
|
|
{
|
|
|
|
// Invalidate forces the block to be checked again.
|
|
|
|
Log_DebugPrintf("Invalidating block at 0x%08X", block->GetPC());
|
|
|
|
block->invalidated = true;
|
2020-08-08 05:42:11 +00:00
|
|
|
#ifdef WITH_RECOMPILER
|
|
|
|
SetFastMap(block->GetPC(), FastCompileBlockFunction);
|
|
|
|
#endif
|
2019-11-21 14:32:40 +00:00
|
|
|
}
|
2019-11-19 10:30:04 +00:00
|
|
|
|
2019-11-21 14:32:40 +00:00
|
|
|
// Block will be re-added next execution.
|
|
|
|
blocks.clear();
|
2020-07-31 07:09:18 +00:00
|
|
|
Bus::ClearRAMCodePage(page_index);
|
2019-11-19 10:30:04 +00:00
|
|
|
}
|
|
|
|
|
2020-12-30 15:48:43 +00:00
|
|
|
void RemoveReferencesToBlock(CodeBlock* block)
|
2019-11-19 10:30:04 +00:00
|
|
|
{
|
2020-07-31 07:09:18 +00:00
|
|
|
BlockMap::iterator iter = s_blocks.find(block->key.GetPC());
|
|
|
|
Assert(iter != s_blocks.end() && iter->second == block);
|
2019-11-19 10:30:04 +00:00
|
|
|
|
2020-08-08 05:42:11 +00:00
|
|
|
#ifdef WITH_RECOMPILER
|
|
|
|
SetFastMap(block->GetPC(), FastCompileBlockFunction);
|
|
|
|
#endif
|
|
|
|
|
2019-11-21 14:32:40 +00:00
|
|
|
// if it's been invalidated it won't be in the page map
|
2020-12-29 04:44:45 +00:00
|
|
|
if (!block->invalidated)
|
2019-11-21 14:32:40 +00:00
|
|
|
RemoveBlockFromPageMap(block);
|
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
UnlinkBlock(block);
|
2020-10-18 04:43:55 +00:00
|
|
|
#ifdef WITH_RECOMPILER
|
2020-12-29 04:44:45 +00:00
|
|
|
if (!block->invalidated)
|
|
|
|
RemoveBlockFromHostCodeMap(block);
|
2020-10-18 04:43:55 +00:00
|
|
|
#endif
|
2020-07-31 07:09:18 +00:00
|
|
|
|
|
|
|
s_blocks.erase(iter);
|
2019-11-21 14:32:40 +00:00
|
|
|
}
|
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
void AddBlockToPageMap(CodeBlock* block)
|
2019-11-21 14:32:40 +00:00
|
|
|
{
|
|
|
|
if (!block->IsInRAM())
|
|
|
|
return;
|
|
|
|
|
2019-11-19 10:30:04 +00:00
|
|
|
const u32 start_page = block->GetStartPageIndex();
|
|
|
|
const u32 end_page = block->GetEndPageIndex();
|
2019-11-26 09:45:36 +00:00
|
|
|
for (u32 page = start_page; page <= end_page; page++)
|
2019-11-19 10:30:04 +00:00
|
|
|
{
|
2019-11-21 14:32:40 +00:00
|
|
|
m_ram_block_map[page].push_back(block);
|
2020-07-31 07:09:18 +00:00
|
|
|
Bus::SetRAMCodePage(page);
|
2019-11-19 10:30:04 +00:00
|
|
|
}
|
2019-11-21 14:32:40 +00:00
|
|
|
}
|
2019-11-19 10:30:04 +00:00
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
void RemoveBlockFromPageMap(CodeBlock* block)
|
2019-11-21 14:32:40 +00:00
|
|
|
{
|
|
|
|
if (!block->IsInRAM())
|
|
|
|
return;
|
2019-11-19 10:30:04 +00:00
|
|
|
|
2019-11-21 14:32:40 +00:00
|
|
|
const u32 start_page = block->GetStartPageIndex();
|
|
|
|
const u32 end_page = block->GetEndPageIndex();
|
2019-11-26 09:45:36 +00:00
|
|
|
for (u32 page = start_page; page <= end_page; page++)
|
2019-11-19 10:30:04 +00:00
|
|
|
{
|
2019-11-21 14:32:40 +00:00
|
|
|
auto& page_blocks = m_ram_block_map[page];
|
|
|
|
auto page_block_iter = std::find(page_blocks.begin(), page_blocks.end(), block);
|
|
|
|
Assert(page_block_iter != page_blocks.end());
|
|
|
|
page_blocks.erase(page_block_iter);
|
2019-11-19 10:30:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
void LinkBlock(CodeBlock* from, CodeBlock* to)
|
2019-11-23 03:16:43 +00:00
|
|
|
{
|
|
|
|
Log_DebugPrintf("Linking block %p(%08x) to %p(%08x)", from, from->GetPC(), to, to->GetPC());
|
|
|
|
from->link_successors.push_back(to);
|
|
|
|
to->link_predecessors.push_back(from);
|
|
|
|
}
|
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
void UnlinkBlock(CodeBlock* block)
|
2019-11-23 03:16:43 +00:00
|
|
|
{
|
|
|
|
for (CodeBlock* predecessor : block->link_predecessors)
|
|
|
|
{
|
|
|
|
auto iter = std::find(predecessor->link_successors.begin(), predecessor->link_successors.end(), block);
|
|
|
|
Assert(iter != predecessor->link_successors.end());
|
|
|
|
predecessor->link_successors.erase(iter);
|
|
|
|
}
|
|
|
|
block->link_predecessors.clear();
|
|
|
|
|
|
|
|
for (CodeBlock* successor : block->link_successors)
|
|
|
|
{
|
|
|
|
auto iter = std::find(successor->link_predecessors.begin(), successor->link_predecessors.end(), block);
|
|
|
|
Assert(iter != successor->link_predecessors.end());
|
|
|
|
successor->link_predecessors.erase(iter);
|
|
|
|
}
|
|
|
|
block->link_successors.clear();
|
|
|
|
}
|
|
|
|
|
2020-10-18 04:43:55 +00:00
|
|
|
#ifdef WITH_RECOMPILER
|
|
|
|
|
|
|
|
void AddBlockToHostCodeMap(CodeBlock* block)
|
|
|
|
{
|
|
|
|
if (!g_settings.IsUsingRecompiler())
|
|
|
|
return;
|
|
|
|
|
|
|
|
auto ir = s_host_code_map.emplace(block->host_code, block);
|
|
|
|
Assert(ir.second);
|
|
|
|
}
|
|
|
|
|
|
|
|
void RemoveBlockFromHostCodeMap(CodeBlock* block)
|
|
|
|
{
|
|
|
|
if (!g_settings.IsUsingRecompiler())
|
|
|
|
return;
|
|
|
|
|
|
|
|
HostCodeMap::iterator hc_iter = s_host_code_map.find(block->host_code);
|
|
|
|
Assert(hc_iter != s_host_code_map.end());
|
|
|
|
s_host_code_map.erase(hc_iter);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool InitializeFastmem()
|
|
|
|
{
|
2020-11-22 15:06:25 +00:00
|
|
|
const CPUFastmemMode mode = g_settings.cpu_fastmem_mode;
|
|
|
|
Assert(mode != CPUFastmemMode::Disabled);
|
|
|
|
|
|
|
|
#ifdef WITH_MMAP_FASTMEM
|
|
|
|
const auto handler = (mode == CPUFastmemMode::MMap) ? MMapPageFaultHandler : LUTPageFaultHandler;
|
|
|
|
#else
|
|
|
|
const auto handler = LUTPageFaultHandler;
|
|
|
|
Assert(mode != CPUFastmemMode::MMap);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (!Common::PageFaultHandler::InstallHandler(&s_host_code_map, handler))
|
2020-10-18 04:43:55 +00:00
|
|
|
{
|
|
|
|
Log_ErrorPrintf("Failed to install page fault handler");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-11-22 15:06:25 +00:00
|
|
|
Bus::UpdateFastmemViews(mode, g_state.cop0_regs.sr.Isc);
|
2020-10-18 04:43:55 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ShutdownFastmem()
|
|
|
|
{
|
|
|
|
Common::PageFaultHandler::RemoveHandler(&s_host_code_map);
|
2020-11-22 15:06:25 +00:00
|
|
|
Bus::UpdateFastmemViews(CPUFastmemMode::Disabled, false);
|
2020-10-18 04:43:55 +00:00
|
|
|
}
|
|
|
|
|
2020-11-22 15:06:25 +00:00
|
|
|
#ifdef WITH_MMAP_FASTMEM
|
|
|
|
|
|
|
|
Common::PageFaultHandler::HandlerResult MMapPageFaultHandler(void* exception_pc, void* fault_address, bool is_write)
|
2020-10-18 04:43:55 +00:00
|
|
|
{
|
|
|
|
if (static_cast<u8*>(fault_address) < g_state.fastmem_base ||
|
2021-02-06 09:19:55 +00:00
|
|
|
(static_cast<u8*>(fault_address) - g_state.fastmem_base) >= static_cast<ptrdiff_t>(Bus::FASTMEM_REGION_SIZE))
|
2020-10-18 04:43:55 +00:00
|
|
|
{
|
|
|
|
return Common::PageFaultHandler::HandlerResult::ExecuteNextHandler;
|
|
|
|
}
|
|
|
|
|
|
|
|
const PhysicalMemoryAddress fastmem_address =
|
|
|
|
static_cast<PhysicalMemoryAddress>(static_cast<ptrdiff_t>(static_cast<u8*>(fault_address) - g_state.fastmem_base));
|
|
|
|
|
|
|
|
Log_DevPrintf("Page fault handler invoked at PC=%p Address=%p %s, fastmem offset 0x%08X", exception_pc, fault_address,
|
|
|
|
is_write ? "(write)" : "(read)", fastmem_address);
|
|
|
|
|
|
|
|
// use upper_bound to find the next block after the pc
|
|
|
|
HostCodeMap::iterator upper_iter =
|
|
|
|
s_host_code_map.upper_bound(reinterpret_cast<CodeBlock::HostCodePointer>(exception_pc));
|
|
|
|
if (upper_iter == s_host_code_map.begin())
|
|
|
|
return Common::PageFaultHandler::HandlerResult::ExecuteNextHandler;
|
|
|
|
|
|
|
|
// then decrement it by one to (hopefully) get the block we want
|
|
|
|
upper_iter--;
|
|
|
|
|
|
|
|
// find the loadstore info in the code block
|
|
|
|
CodeBlock* block = upper_iter->second;
|
|
|
|
for (auto bpi_iter = block->loadstore_backpatch_info.begin(); bpi_iter != block->loadstore_backpatch_info.end();
|
|
|
|
++bpi_iter)
|
|
|
|
{
|
2020-11-07 08:49:59 +00:00
|
|
|
Recompiler::LoadStoreBackpatchInfo& lbi = *bpi_iter;
|
2020-10-18 04:43:55 +00:00
|
|
|
if (lbi.host_pc == exception_pc)
|
|
|
|
{
|
2020-11-07 08:49:59 +00:00
|
|
|
if (is_write && !g_state.cop0_regs.sr.Isc && Bus::IsRAMAddress(fastmem_address))
|
|
|
|
{
|
|
|
|
// this is probably a code page, since we aren't going to fault due to requiring fastmem on RAM.
|
|
|
|
const u32 code_page_index = Bus::GetRAMCodePageIndex(fastmem_address);
|
|
|
|
if (Bus::IsRAMCodePage(code_page_index))
|
|
|
|
{
|
|
|
|
if (++lbi.fault_count < CODE_WRITE_FAULT_THRESHOLD_FOR_SLOWMEM)
|
|
|
|
{
|
|
|
|
InvalidateBlocksWithPageIndex(code_page_index);
|
|
|
|
return Common::PageFaultHandler::HandlerResult::ContinueExecution;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Log_DevPrintf("Backpatching code write at %p (%08X) address %p (%08X) to slowmem after threshold",
|
|
|
|
exception_pc, lbi.guest_pc, fault_address, fastmem_address);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-18 04:43:55 +00:00
|
|
|
// found it, do fixup
|
|
|
|
if (Recompiler::CodeGenerator::BackpatchLoadStore(lbi))
|
|
|
|
{
|
|
|
|
// remove the backpatch entry since we won't be coming back to this one
|
|
|
|
block->loadstore_backpatch_info.erase(bpi_iter);
|
|
|
|
return Common::PageFaultHandler::HandlerResult::ContinueExecution;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("Failed to backpatch %p in block 0x%08X", exception_pc, block->GetPC());
|
|
|
|
return Common::PageFaultHandler::HandlerResult::ExecuteNextHandler;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// we didn't find the pc in our list..
|
|
|
|
Log_ErrorPrintf("Loadstore PC not found for %p in block 0x%08X", exception_pc, block->GetPC());
|
|
|
|
return Common::PageFaultHandler::HandlerResult::ExecuteNextHandler;
|
|
|
|
}
|
|
|
|
|
2020-11-22 15:06:25 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
Common::PageFaultHandler::HandlerResult LUTPageFaultHandler(void* exception_pc, void* fault_address, bool is_write)
|
|
|
|
{
|
|
|
|
// use upper_bound to find the next block after the pc
|
|
|
|
HostCodeMap::iterator upper_iter =
|
|
|
|
s_host_code_map.upper_bound(reinterpret_cast<CodeBlock::HostCodePointer>(exception_pc));
|
|
|
|
if (upper_iter == s_host_code_map.begin())
|
|
|
|
return Common::PageFaultHandler::HandlerResult::ExecuteNextHandler;
|
|
|
|
|
|
|
|
// then decrement it by one to (hopefully) get the block we want
|
|
|
|
upper_iter--;
|
|
|
|
|
|
|
|
// find the loadstore info in the code block
|
|
|
|
CodeBlock* block = upper_iter->second;
|
|
|
|
for (auto bpi_iter = block->loadstore_backpatch_info.begin(); bpi_iter != block->loadstore_backpatch_info.end();
|
|
|
|
++bpi_iter)
|
|
|
|
{
|
|
|
|
Recompiler::LoadStoreBackpatchInfo& lbi = *bpi_iter;
|
|
|
|
if (lbi.host_pc == exception_pc)
|
|
|
|
{
|
|
|
|
// found it, do fixup
|
|
|
|
if (Recompiler::CodeGenerator::BackpatchLoadStore(lbi))
|
|
|
|
{
|
|
|
|
// remove the backpatch entry since we won't be coming back to this one
|
|
|
|
block->loadstore_backpatch_info.erase(bpi_iter);
|
|
|
|
return Common::PageFaultHandler::HandlerResult::ContinueExecution;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("Failed to backpatch %p in block 0x%08X", exception_pc, block->GetPC());
|
|
|
|
return Common::PageFaultHandler::HandlerResult::ExecuteNextHandler;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// we didn't find the pc in our list..
|
|
|
|
Log_ErrorPrintf("Loadstore PC not found for %p in block 0x%08X", exception_pc, block->GetPC());
|
|
|
|
return Common::PageFaultHandler::HandlerResult::ExecuteNextHandler;
|
|
|
|
}
|
2020-11-20 15:56:51 +00:00
|
|
|
|
|
|
|
#endif // WITH_RECOMPILER
|
2020-10-18 04:43:55 +00:00
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
} // namespace CPU::CodeCache
|