JITs: Outline FreeRanges Function

As I have done for Cached Interpreter 2.0
This commit is contained in:
mitaclaw 2024-07-26 20:58:52 -07:00
parent c431cd2e1e
commit d26dc1ba32
4 changed files with 43 additions and 33 deletions

View File

@ -310,6 +310,17 @@ void Jit64::ClearCache()
ResetFreeMemoryRanges(); ResetFreeMemoryRanges();
} }
void Jit64::FreeRanges()
{
// Check if any code blocks have been freed in the block cache and transfer this information to
// the local rangesets to allow overwriting them with new code.
for (const auto& [from, to] : blocks.GetRangesToFreeNear())
m_free_ranges_near.insert(from, to);
for (const auto& [from, to] : blocks.GetRangesToFreeFar())
m_free_ranges_far.insert(from, to);
blocks.ClearRangesToFree();
}
void Jit64::ResetFreeMemoryRanges() void Jit64::ResetFreeMemoryRanges()
{ {
// Set the entire near and far code regions as unused. // Set the entire near and far code regions as unused.
@ -746,14 +757,7 @@ void Jit64::Jit(u32 em_address, bool clear_cache_and_retry_on_failure)
} }
ClearCache(); ClearCache();
} }
FreeRanges();
// Check if any code blocks have been freed in the block cache and transfer this information to
// the local rangesets to allow overwriting them with new code.
for (auto range : blocks.GetRangesToFreeNear())
m_free_ranges_near.insert(range.first, range.second);
for (auto range : blocks.GetRangesToFreeFar())
m_free_ranges_far.insert(range.first, range.second);
blocks.ClearRangesToFree();
std::size_t block_size = m_code_buffer.size(); std::size_t block_size = m_code_buffer.size();

View File

@ -266,6 +266,7 @@ private:
bool HandleFunctionHooking(u32 address); bool HandleFunctionHooking(u32 address);
void FreeRanges();
void ResetFreeMemoryRanges(); void ResetFreeMemoryRanges();
static void ImHere(Jit64& jit); static void ImHere(Jit64& jit);

View File

@ -188,6 +188,34 @@ void JitArm64::GenerateAsmAndResetFreeMemoryRanges()
routines_far_end - routines_far_start); routines_far_end - routines_far_start);
} }
void JitArm64::FreeRanges()
{
// Check if any code blocks have been freed in the block cache and transfer this information to
// the local rangesets to allow overwriting them with new code.
for (const auto& [from, to] : blocks.GetRangesToFreeNear())
{
const auto first_fastmem_area = m_fault_to_handler.upper_bound(from);
auto last_fastmem_area = first_fastmem_area;
const auto end = m_fault_to_handler.end();
while (last_fastmem_area != end && last_fastmem_area->first <= to)
++last_fastmem_area;
m_fault_to_handler.erase(first_fastmem_area, last_fastmem_area);
if (from < m_near_code_0.GetCodeEnd())
m_free_ranges_near_0.insert(from, to);
else
m_free_ranges_near_1.insert(from, to);
}
for (const auto& [from, to] : blocks.GetRangesToFreeFar())
{
if (from < m_far_code_0.GetCodeEnd())
m_free_ranges_far_0.insert(from, to);
else
m_free_ranges_far_1.insert(from, to);
}
blocks.ClearRangesToFree();
}
void JitArm64::ResetFreeMemoryRanges(size_t routines_near_size, size_t routines_far_size) void JitArm64::ResetFreeMemoryRanges(size_t routines_near_size, size_t routines_far_size)
{ {
// Set the near and far code regions as unused. // Set the near and far code regions as unused.
@ -911,31 +939,7 @@ void JitArm64::Jit(u32 em_address, bool clear_cache_and_retry_on_failure)
if (SConfig::GetInstance().bJITNoBlockCache) if (SConfig::GetInstance().bJITNoBlockCache)
ClearCache(); ClearCache();
FreeRanges();
// Check if any code blocks have been freed in the block cache and transfer this information to
// the local rangesets to allow overwriting them with new code.
for (auto range : blocks.GetRangesToFreeNear())
{
auto first_fastmem_area = m_fault_to_handler.upper_bound(range.first);
auto last_fastmem_area = first_fastmem_area;
auto end = m_fault_to_handler.end();
while (last_fastmem_area != end && last_fastmem_area->first <= range.second)
++last_fastmem_area;
m_fault_to_handler.erase(first_fastmem_area, last_fastmem_area);
if (range.first < m_near_code_0.GetCodeEnd())
m_free_ranges_near_0.insert(range.first, range.second);
else
m_free_ranges_near_1.insert(range.first, range.second);
}
for (auto range : blocks.GetRangesToFreeFar())
{
if (range.first < m_far_code_0.GetCodeEnd())
m_free_ranges_far_0.insert(range.first, range.second);
else
m_free_ranges_far_1.insert(range.first, range.second);
}
blocks.ClearRangesToFree();
const Common::ScopedJITPageWriteAndNoExecute enable_jit_page_writes; const Common::ScopedJITPageWriteAndNoExecute enable_jit_page_writes;

View File

@ -294,6 +294,7 @@ protected:
void Cleanup(); void Cleanup();
void ResetStack(); void ResetStack();
void FreeRanges();
void GenerateAsmAndResetFreeMemoryRanges(); void GenerateAsmAndResetFreeMemoryRanges();
void ResetFreeMemoryRanges(size_t routines_near_size, size_t routines_far_size); void ResetFreeMemoryRanges(size_t routines_near_size, size_t routines_far_size);