Duckstation/src/common/jit_code_buffer.cpp

112 lines
3.2 KiB
C++
Raw Normal View History

2019-09-09 07:01:26 +00:00
#include "jit_code_buffer.h"
#include "YBaseLib/Assert.h"
#if defined(Y_PLATFORM_WINDOWS)
#include "YBaseLib/Windows/WindowsHeaders.h"
#elif defined(Y_PLATFORM_LINUX) || defined(Y_PLATFORM_ANDROID)
#include <sys/mman.h>
#endif
JitCodeBuffer::JitCodeBuffer(u32 size /* = 64 * 1024 * 1024 */, u32 far_code_size /* = 0 */)
2019-09-09 07:01:26 +00:00
{
m_total_size = size + far_code_size;
2019-09-09 07:01:26 +00:00
#if defined(Y_PLATFORM_WINDOWS)
m_code_ptr = static_cast<u8*>(VirtualAlloc(nullptr, m_total_size, MEM_COMMIT, PAGE_EXECUTE_READWRITE));
2019-09-09 07:01:26 +00:00
#elif defined(Y_PLATFORM_LINUX) || defined(Y_PLATFORM_ANDROID)
m_code_ptr = static_cast<u8*>(
mmap(nullptr, m_total_size, PROT_READ | PROT_WRITE | PROT_EXEC, MAP_PRIVATE | MAP_ANONYMOUS, -1, 0));
2019-09-09 07:01:26 +00:00
#else
m_code_ptr = nullptr;
#endif
m_free_code_ptr = m_code_ptr;
m_code_size = size;
m_code_used = 0;
m_far_code_ptr = static_cast<u8*>(m_code_ptr) + size;
m_free_far_code_ptr = m_far_code_ptr;
m_far_code_size = far_code_size;
m_far_code_used = 0;
2019-09-09 07:01:26 +00:00
if (!m_code_ptr)
Panic("Failed to allocate code space.");
}
JitCodeBuffer::~JitCodeBuffer()
{
#if defined(Y_PLATFORM_WINDOWS)
VirtualFree(m_code_ptr, 0, MEM_RELEASE);
2019-09-09 07:01:26 +00:00
#elif defined(Y_PLATFORM_LINUX) || defined(Y_PLATFORM_ANDROID)
munmap(m_code_ptr, m_total_size);
2019-09-09 07:01:26 +00:00
#endif
}
void JitCodeBuffer::CommitCode(u32 length)
2019-09-09 07:01:26 +00:00
{
if (length == 0)
return;
#if defined(Y_CPU_ARM) || defined(Y_CPU_AARCH64)
// ARM instruction and data caches are not coherent, we need to flush after every block.
DoCacheFlush(m_free_code_ptr, length);
#endif
2019-09-09 07:01:26 +00:00
Assert(length <= (m_code_size - m_code_used));
m_free_code_ptr += length;
2019-09-09 07:01:26 +00:00
m_code_used += length;
}
void JitCodeBuffer::CommitFarCode(u32 length)
{
if (length == 0)
return;
#if defined(Y_CPU_ARM) || defined(Y_CPU_AARCH64)
// ARM instruction and data caches are not coherent, we need to flush after every block.
DoCacheFlush(m_free_far_code_ptr, length);
#endif
Assert(length <= (m_far_code_size - m_far_code_used));
m_free_far_code_ptr += length;
m_far_code_used += length;
}
2019-09-09 07:01:26 +00:00
void JitCodeBuffer::Reset()
{
std::memset(m_code_ptr, 0, m_code_size);
FlushInstructionCache(m_code_ptr, m_code_size);
if (m_far_code_size > 0)
{
std::memset(m_far_code_ptr, 0, m_far_code_size);
FlushInstructionCache(m_far_code_ptr, m_far_code_size);
}
2019-09-09 07:01:26 +00:00
m_free_code_ptr = m_code_ptr;
m_code_used = 0;
m_free_far_code_ptr = m_far_code_ptr;
m_far_code_used = 0;
2019-09-09 07:01:26 +00:00
}
void JitCodeBuffer::Align(u32 alignment, u8 padding_value)
{
DebugAssert(Common::IsPow2(alignment));
const u32 num_padding_bytes =
std::min(static_cast<u32>(Common::AlignUpPow2(reinterpret_cast<uintptr_t>(m_free_code_ptr), alignment) -
reinterpret_cast<uintptr_t>(m_free_code_ptr)),
2019-09-09 07:01:26 +00:00
GetFreeCodeSpace());
std::memset(m_free_code_ptr, padding_value, num_padding_bytes);
m_free_code_ptr += num_padding_bytes;
2019-09-09 07:01:26 +00:00
m_code_used += num_padding_bytes;
}
void JitCodeBuffer::FlushInstructionCache(void* address, u32 size)
{
#if defined(Y_PLATFORM_WINDOWS)
::FlushInstructionCache(GetCurrentProcess(), address, size);
#elif defined(Y_COMPILER_GCC) || defined(Y_COMPILER_CLANG)
__builtin___clear_cache(reinterpret_cast<char*>(address), reinterpret_cast<char*>(address + size));
#else
#error Unknown platform.
#endif
}