| 
									
										
										
										
											2009-07-28 21:32:10 +00:00
										 |  |  | // Copyright (C) 2003 Dolphin Project.
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | // This program is free software: you can redistribute it and/or modify
 | 
					
						
							|  |  |  | // it under the terms of the GNU General Public License as published by
 | 
					
						
							|  |  |  | // the Free Software Foundation, version 2.0.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // This program is distributed in the hope that it will be useful,
 | 
					
						
							|  |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
					
						
							|  |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
					
						
							|  |  |  | // GNU General Public License 2.0 for more details.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // A copy of the GPL 2.0 should have been included with the program.
 | 
					
						
							|  |  |  | // If not, see http://www.gnu.org/licenses/
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Official SVN repository and contact information can be found at
 | 
					
						
							|  |  |  | // http://code.google.com/p/dolphin-emu/
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "Common.h"
 | 
					
						
							|  |  |  | #include "MemoryUtil.h"
 | 
					
						
							| 
									
										
										
										
											2009-09-07 12:40:43 +00:00
										 |  |  | #include "StringUtil.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-16 02:58:34 +00:00
										 |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  | #include <windows.h>
 | 
					
						
							| 
									
										
										
										
											2009-09-07 12:40:43 +00:00
										 |  |  | #include <psapi.h>
 | 
					
						
							| 
									
										
										
										
											2010-07-18 05:15:16 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | #include <errno.h>
 | 
					
						
							|  |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-06 06:18:22 +00:00
										 |  |  | // This is purposely not a full wrapper for virtualalloc/mmap, but it
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | // provides exactly the primitive operations that Dolphin needs.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-19 21:24:52 +00:00
										 |  |  | void* AllocateExecutableMemory(size_t size, bool low) | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | { | 
					
						
							|  |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  | 	void* ptr = VirtualAlloc(0, size, MEM_COMMIT, PAGE_EXECUTE_READWRITE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if ((u64)ptr >= 0x80000000) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		PanicAlert("Executable memory ended up above 2GB!"); | 
					
						
							|  |  |  | 		// If this happens, we have to implement a free ram search scheme. ector knows how.
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-06 06:18:22 +00:00
										 |  |  | 	return ptr; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 	void* retval = mmap(0, size, PROT_READ | PROT_WRITE | PROT_EXEC, | 
					
						
							| 
									
										
										
										
											2010-07-18 05:15:16 +00:00
										 |  |  | 		MAP_ANON | MAP_PRIVATE | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | #ifdef __x86_64__
 | 
					
						
							|  |  |  | 		 | (low ? MAP_32BIT : 0) | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |          , -1, 0);  // | MAP_FIXED
 | 
					
						
							|  |  |  | 	// printf("Mapped executable memory at %p (size %i)\n", retval, size);
 | 
					
						
							| 
									
										
										
										
											2010-03-05 09:23:28 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 	if (!retval) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		PanicAlert("Failed to allocate executable memory, errno=%i", errno); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-06 06:18:22 +00:00
										 |  |  | 	return retval; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-19 21:24:52 +00:00
										 |  |  | void* AllocateMemoryPages(size_t size) | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | { | 
					
						
							|  |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  | 	void* ptr = VirtualAlloc(0, size, MEM_COMMIT, PAGE_READWRITE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!ptr) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		PanicAlert("Failed to allocate raw memory"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-06 06:18:22 +00:00
										 |  |  | 	return ptr; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 	void* retval = mmap(0, size, PROT_READ | PROT_WRITE, | 
					
						
							| 
									
										
										
										
											2010-07-18 05:15:16 +00:00
										 |  |  | 			MAP_ANON | MAP_PRIVATE, -1, 0); // | MAP_FIXED
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 	// printf("Mapped memory at %p (size %i)\n", retval, size);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!retval) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		PanicAlert("Failed to allocate raw memory, errno=%i", errno); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-06 06:18:22 +00:00
										 |  |  | 	return retval; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-19 21:24:52 +00:00
										 |  |  | void FreeMemoryPages(void* ptr, size_t size) | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | { | 
					
						
							|  |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  | 	if (ptr) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2009-08-06 06:18:22 +00:00
										 |  |  | 		if (!VirtualFree(ptr, 0, MEM_RELEASE)) | 
					
						
							|  |  |  | 			PanicAlert("FreeMemoryPages failed!\n%s", GetLastErrorMsg()); | 
					
						
							|  |  |  | 		ptr = NULL; // Is this our responsibility?
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 	munmap(ptr, size); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-19 21:24:52 +00:00
										 |  |  | void WriteProtectMemory(void* ptr, size_t size, bool allowExecute) | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | { | 
					
						
							|  |  |  | #ifdef _WIN32
 | 
					
						
							| 
									
										
										
										
											2009-08-06 06:18:22 +00:00
										 |  |  | 	DWORD oldValue; | 
					
						
							|  |  |  | 	if (!VirtualProtect(ptr, size, allowExecute ? PAGE_EXECUTE_READ : PAGE_READONLY, &oldValue)) | 
					
						
							|  |  |  | 		PanicAlert("WriteProtectMemory failed!\n%s", GetLastErrorMsg()); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | #else
 | 
					
						
							|  |  |  | 	mprotect(ptr, size, allowExecute ? (PROT_READ | PROT_EXEC) : PROT_READ); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-19 21:24:52 +00:00
										 |  |  | void UnWriteProtectMemory(void* ptr, size_t size, bool allowExecute) | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | { | 
					
						
							|  |  |  | #ifdef _WIN32
 | 
					
						
							| 
									
										
										
										
											2009-08-06 06:18:22 +00:00
										 |  |  | 	DWORD oldValue; | 
					
						
							|  |  |  | 	if (!VirtualProtect(ptr, size, allowExecute ? PAGE_EXECUTE_READWRITE : PAGE_READWRITE, &oldValue)) | 
					
						
							|  |  |  | 		PanicAlert("UnWriteProtectMemory failed!\n%s", GetLastErrorMsg()); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | #else
 | 
					
						
							|  |  |  | 	mprotect(ptr, size, allowExecute ? (PROT_READ | PROT_WRITE | PROT_EXEC) : PROT_WRITE | PROT_READ); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2009-09-07 12:40:43 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::string MemUsage() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	#ifdef _WIN32
 | 
					
						
							|  |  |  | 	DWORD processID = GetCurrentProcessId(); | 
					
						
							|  |  |  |     HANDLE hProcess; | 
					
						
							|  |  |  |     PROCESS_MEMORY_COUNTERS pmc; | 
					
						
							|  |  |  | 	std::string Ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Print information about the memory usage of the process.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     hProcess = OpenProcess(PROCESS_QUERY_INFORMATION | PROCESS_VM_READ, FALSE, processID); | 
					
						
							|  |  |  |     if (NULL == hProcess) return "MemUsage Error"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (GetProcessMemoryInfo(hProcess, &pmc, sizeof(pmc))) | 
					
						
							| 
									
										
										
										
											2010-11-10 04:12:31 +00:00
										 |  |  | 		Ret = StringFromFormat("%s K", ThousandSeparate(pmc.WorkingSetSize / 1024, 7).c_str()); | 
					
						
							| 
									
										
										
										
											2009-09-07 12:40:43 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     CloseHandle(hProcess); | 
					
						
							|  |  |  | 	return Ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	#else
 | 
					
						
							|  |  |  | 	return ""; | 
					
						
							|  |  |  | 	#endif
 | 
					
						
							| 
									
										
										
										
											2009-11-15 22:26:39 +00:00
										 |  |  | } |