#pragma once #include "types.h" class JitCodeBuffer { public: JitCodeBuffer(); JitCodeBuffer(u32 size, u32 far_code_size); JitCodeBuffer(void* buffer, u32 size, u32 far_code_size, u32 guard_size); ~JitCodeBuffer(); bool Allocate(u32 size = 64 * 1024 * 1024, u32 far_code_size = 0); bool Initialize(void* buffer, u32 size, u32 far_code_size = 0, u32 guard_size = 0); void Destroy(); void Reset(); u8* GetFreeCodePointer() const { return m_free_code_ptr; } u32 GetFreeCodeSpace() const { return static_cast(m_code_size - m_code_used); } void CommitCode(u32 length); u8* GetFreeFarCodePointer() const { return m_free_far_code_ptr; } u32 GetFreeFarCodeSpace() const { return static_cast(m_far_code_size - m_far_code_used); } void CommitFarCode(u32 length); /// Adjusts the free code pointer to the specified alignment, padding with bytes. /// Assumes alignment is a power-of-two. void Align(u32 alignment, u8 padding_value); /// Flushes the instruction cache on the host for the specified range. static void FlushInstructionCache(void* address, u32 size); /// For Apple Silicon - Toggles write protection on the JIT space. #if defined(__APPLE__) && defined(__aarch64__) static void WriteProtect(bool enabled); #else ALWAYS_INLINE static void WriteProtect(bool enabled) {} #endif private: u8* m_code_ptr = nullptr; u8* m_free_code_ptr = nullptr; u32 m_code_size = 0; u32 m_code_used = 0; u8* m_far_code_ptr = nullptr; u8* m_free_far_code_ptr = nullptr; u32 m_far_code_size = 0; u32 m_far_code_used = 0; u32 m_total_size = 0; u32 m_guard_size = 0; u32 m_old_protection = 0; bool m_owns_buffer = false; };