mirror of
				https://github.com/dolphin-emu/dolphin.git
				synced 2025-10-26 01:49:46 +00:00 
			
		
		
		
	SPDX standardizes how source code conveys its copyright and licensing information. See https://spdx.github.io/spdx-spec/1-rationale/ . SPDX tags are adopted in many large projects, including things like the Linux kernel.
		
			
				
	
	
		
			89 lines
		
	
	
	
		
			2.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			89 lines
		
	
	
	
		
			2.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2014 Dolphin Emulator Project
 | |
| // SPDX-License-Identifier: GPL-2.0-or-later
 | |
| 
 | |
| #include <chrono>
 | |
| 
 | |
| #include "Common/CommonTypes.h"
 | |
| #include "Common/Timer.h"
 | |
| #include "Core/MemTools.h"
 | |
| #include "Core/PowerPC/JitCommon/JitBase.h"
 | |
| #include "Core/PowerPC/JitInterface.h"
 | |
| 
 | |
| // include order is important
 | |
| #include <gtest/gtest.h>  // NOLINT
 | |
| 
 | |
| enum
 | |
| {
 | |
| #ifdef _WIN32
 | |
|   PAGE_GRAN = 0x10000
 | |
| #else
 | |
|   PAGE_GRAN = 0x1000
 | |
| #endif
 | |
| };
 | |
| 
 | |
| class PageFaultFakeJit : public JitBase
 | |
| {
 | |
| public:
 | |
|   // CPUCoreBase methods
 | |
|   void Init() override {}
 | |
|   void Shutdown() override {}
 | |
|   void ClearCache() override {}
 | |
|   void Run() override {}
 | |
|   void SingleStep() override {}
 | |
|   const char* GetName() const override { return nullptr; }
 | |
|   // JitBase methods
 | |
|   JitBaseBlockCache* GetBlockCache() override { return nullptr; }
 | |
|   void Jit(u32 em_address) override {}
 | |
|   const CommonAsmRoutinesBase* GetAsmRoutines() override { return nullptr; }
 | |
|   virtual bool HandleFault(uintptr_t access_address, SContext* ctx) override
 | |
|   {
 | |
|     m_pre_unprotect_time = std::chrono::high_resolution_clock::now();
 | |
|     Common::UnWriteProtectMemory(m_data, PAGE_GRAN, /*allowExecute*/ false);
 | |
|     m_post_unprotect_time = std::chrono::high_resolution_clock::now();
 | |
|     return true;
 | |
|   }
 | |
| 
 | |
|   void* m_data;
 | |
|   std::chrono::time_point<std::chrono::high_resolution_clock> m_pre_unprotect_time,
 | |
|       m_post_unprotect_time;
 | |
| };
 | |
| 
 | |
| #ifdef _MSC_VER
 | |
| #define ASAN_DISABLE __declspec(no_sanitize_address)
 | |
| #else
 | |
| #define ASAN_DISABLE
 | |
| #endif
 | |
| 
 | |
| static void ASAN_DISABLE perform_invalid_access(void* data)
 | |
| {
 | |
|   *(volatile int*)data = 5;
 | |
| }
 | |
| 
 | |
| TEST(PageFault, PageFault)
 | |
| {
 | |
|   EMM::InstallExceptionHandler();
 | |
|   void* data = Common::AllocateMemoryPages(PAGE_GRAN);
 | |
|   EXPECT_NE(data, nullptr);
 | |
|   Common::WriteProtectMemory(data, PAGE_GRAN, false);
 | |
| 
 | |
|   PageFaultFakeJit pfjit;
 | |
|   JitInterface::SetJit(&pfjit);
 | |
|   pfjit.m_data = data;
 | |
| 
 | |
|   auto start = std::chrono::high_resolution_clock::now();
 | |
|   perform_invalid_access(data);
 | |
|   auto end = std::chrono::high_resolution_clock::now();
 | |
| 
 | |
| #define AS_NS(diff)                                                                                \
 | |
|   ((unsigned long long)std::chrono::duration_cast<std::chrono::nanoseconds>(diff).count())
 | |
| 
 | |
|   EMM::UninstallExceptionHandler();
 | |
|   JitInterface::SetJit(nullptr);
 | |
| 
 | |
|   printf("page fault timing:\n");
 | |
|   printf("start->HandleFault     %llu ns\n", AS_NS(pfjit.m_pre_unprotect_time - start));
 | |
|   printf("UnWriteProtectMemory   %llu ns\n",
 | |
|          AS_NS(pfjit.m_post_unprotect_time - pfjit.m_pre_unprotect_time));
 | |
|   printf("HandleFault->end       %llu ns\n", AS_NS(end - pfjit.m_post_unprotect_time));
 | |
|   printf("total                  %llu ns\n", AS_NS(end - start));
 | |
| }
 |