texture_cache: added memory protection for Windows

This commit is contained in:
psucien 2024-04-28 00:21:04 +02:00
parent fec7f6cdc2
commit e0a4c3f1a3
2 changed files with 41 additions and 4 deletions

View file

@ -11,6 +11,20 @@
#ifndef _WIN64 #ifndef _WIN64
#include <signal.h> #include <signal.h>
#include <sys/mman.h> #include <sys/mman.h>
#define PROT_READ_WRITE (PROT_READ | PROT_WRITE) // There is no option to combine bitflags like this on Windows
#else
#include <Windows.h>
#define PROT_NONE PAGE_NOACCESS
#define PROT_READ_WRITE PAGE_READWRITE
void mprotect(void *addr, size_t len, int prot) {
DWORD old_prot{};
BOOL result = VirtualProtect(addr, len, prot, &old_prot);
ASSERT_MSG(result != 0, "Region protection failed");
}
#endif #endif
namespace VideoCore { namespace VideoCore {
@ -28,6 +42,21 @@ void GuestFaultSignalHandler(int sig, siginfo_t* info, void* raw_context) {
UNREACHABLE(); UNREACHABLE();
} }
} }
#else
LONG WINAPI GuestFaultSignalHandler(EXCEPTION_POINTERS *pExp) noexcept {
const u32 ec = pExp->ExceptionRecord->ExceptionCode;
if (ec == EXCEPTION_ACCESS_VIOLATION) {
const auto info = pExp->ExceptionRecord->ExceptionInformation;
if (info[0] == 1) { // Write violation
g_texture_cache->OnCpuWrite(info[1]);
return EXCEPTION_CONTINUE_EXECUTION;
}
else {
UNREACHABLE();
}
}
return EXCEPTION_CONTINUE_SEARCH; // pass further
}
#endif #endif
static constexpr u64 StreamBufferSize = 128_MB; static constexpr u64 StreamBufferSize = 128_MB;
@ -50,11 +79,18 @@ TextureCache::TextureCache(const Vulkan::Instance& instance_, Vulkan::Scheduler&
guest_access_fault.sa_sigaction = &GuestFaultSignalHandler; guest_access_fault.sa_sigaction = &GuestFaultSignalHandler;
guest_access_fault.sa_mask = signal_mask; guest_access_fault.sa_mask = signal_mask;
sigaction(SIGSEGV, &guest_access_fault, nullptr); sigaction(SIGSEGV, &guest_access_fault, nullptr);
#else
veh_handle = AddVectoredExceptionHandler(0, GuestFaultSignalHandler);
ASSERT_MSG(veh_handle, "Failed to register an exception handler");
#endif #endif
g_texture_cache = this; g_texture_cache = this;
} }
TextureCache::~TextureCache() = default; TextureCache::~TextureCache() {
#if _WIN64
RemoveVectoredExceptionHandler(veh_handle);
#endif
}
void TextureCache::OnCpuWrite(VAddr address) { void TextureCache::OnCpuWrite(VAddr address) {
const VAddr address_aligned = address & ~((1 << PageBits) - 1); const VAddr address_aligned = address & ~((1 << PageBits) - 1);
@ -190,16 +226,14 @@ void TextureCache::UpdatePagesCachedCount(VAddr addr, u64 size, s32 delta) {
const VAddr interval_start_addr = boost::icl::first(interval) << PageBits; const VAddr interval_start_addr = boost::icl::first(interval) << PageBits;
const VAddr interval_end_addr = boost::icl::last_next(interval) << PageBits; const VAddr interval_end_addr = boost::icl::last_next(interval) << PageBits;
const u32 interval_size = interval_end_addr - interval_start_addr; const u32 interval_size = interval_end_addr - interval_start_addr;
#ifndef _WIN64
void* addr = reinterpret_cast<void*>(interval_start_addr); void* addr = reinterpret_cast<void*>(interval_start_addr);
if (delta > 0 && count == delta) { if (delta > 0 && count == delta) {
mprotect(addr, interval_size, PROT_NONE); mprotect(addr, interval_size, PROT_NONE);
} else if (delta < 0 && count == -delta) { } else if (delta < 0 && count == -delta) {
mprotect(addr, interval_size, PROT_READ | PROT_WRITE); mprotect(addr, interval_size, PROT_READ_WRITE);
} else { } else {
ASSERT(count >= 0); ASSERT(count >= 0);
} }
#endif
} }
if (delta < 0) { if (delta < 0) {

View file

@ -115,6 +115,9 @@ private:
SlotVector<Image> slot_images; SlotVector<Image> slot_images;
tsl::robin_pg_map<u64, std::vector<ImageId>> page_table; tsl::robin_pg_map<u64, std::vector<ImageId>> page_table;
boost::icl::interval_map<VAddr, s32> cached_pages; boost::icl::interval_map<VAddr, s32> cached_pages;
#ifdef _WIN64
void* veh_handle{};
#endif
}; };
} // namespace VideoCore } // namespace VideoCore