forked from dolphin-emu/dolphin
		
	
		
			
	
	
		
			77 lines
		
	
	
		
			1.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			77 lines
		
	
	
		
			1.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
|   | // Copyright 2013 Dolphin Emulator Project
 | ||
|  | // Licensed under GPLv2
 | ||
|  | // Refer to the license.txt file included.
 | ||
|  | 
 | ||
|  | #pragma once
 | ||
|  | 
 | ||
|  | #include "Common/Common.h"
 | ||
|  | #include "Common/MemoryUtil.h"
 | ||
|  | 
 | ||
|  | // Everything that needs to generate code should inherit from this.
 | ||
|  | // You get memory management for free, plus, you can use all emitter functions without
 | ||
|  | // having to prefix them with gen-> or something similar.
 | ||
|  | // Example implementation:
 | ||
|  | // class JIT : public CodeBlock<ARMXEmitter> {}
 | ||
|  | template<class T> class CodeBlock : public T, NonCopyable | ||
|  | { | ||
|  | private: | ||
|  | 	// A privately used function to set the executable RAM space to something invalid.
 | ||
|  | 	// For debugging usefulness it should be used to set the RAM to a host specific breakpoint instruction
 | ||
|  | 	virtual void PoisonMemory() = 0; | ||
|  | 
 | ||
|  | protected: | ||
|  | 	u8 *region; | ||
|  | 	size_t region_size; | ||
|  | 
 | ||
|  | public: | ||
|  | 	CodeBlock() : region(nullptr), region_size(0) {} | ||
|  | 	virtual ~CodeBlock() { if (region) FreeCodeSpace(); } | ||
|  | 
 | ||
|  | 	// Call this before you generate any code.
 | ||
|  | 	void AllocCodeSpace(int size) | ||
|  | 	{ | ||
|  | 		region_size = size; | ||
|  | 		region = (u8*)AllocateExecutableMemory(region_size); | ||
|  | 		T::SetCodePtr(region); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	// Always clear code space with breakpoints, so that if someone accidentally executes
 | ||
|  | 	// uninitialized, it just breaks into the debugger.
 | ||
|  | 	void ClearCodeSpace() | ||
|  | 	{ | ||
|  | 		PoisonMemory(); | ||
|  | 		ResetCodePtr(); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	// Call this when shutting down. Don't rely on the destructor, even though it'll do the job.
 | ||
|  | 	void FreeCodeSpace() | ||
|  | 	{ | ||
|  | 		FreeMemoryPages(region, region_size); | ||
|  | 		region = nullptr; | ||
|  | 		region_size = 0; | ||
|  | 	} | ||
|  | 
 | ||
|  | 	bool IsInSpace(u8 *ptr) | ||
|  | 	{ | ||
|  | 		return (ptr >= region) && (ptr < (region + region_size)); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	// Cannot currently be undone. Will write protect the entire code region.
 | ||
|  | 	// Start over if you need to change the code (call FreeCodeSpace(), AllocCodeSpace()).
 | ||
|  | 	void WriteProtect() | ||
|  | 	{ | ||
|  | 		WriteProtectMemory(region, region_size, true); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	void ResetCodePtr() | ||
|  | 	{ | ||
|  | 		T::SetCodePtr(region); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	size_t GetSpaceLeft() const | ||
|  | 	{ | ||
|  | 		return region_size - (T::GetCodePtr() - region); | ||
|  | 	} | ||
|  | }; | ||
|  | 
 |