| 
									
										
										
										
											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); | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2011-01-09 14:09:27 +00:00
										 |  |  | 	void* ptr = mmap(0, size, PROT_READ | PROT_WRITE | PROT_EXEC, | 
					
						
							| 
									
										
										
										
											2010-07-18 05:15:16 +00:00
										 |  |  | 		MAP_ANON | MAP_PRIVATE | 
					
						
							| 
									
										
										
										
											2011-01-09 14:09:27 +00:00
										 |  |  | #if defined __linux__ && defined __x86_64__
 | 
					
						
							|  |  |  | 		| (low ? MAP_32BIT : 0) | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-01-09 14:09:27 +00:00
										 |  |  | 		, -1, 0); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-09 14:09:27 +00:00
										 |  |  | 	// printf("Mapped executable memory at %p (size %ld)\n", ptr,
 | 
					
						
							|  |  |  | 	//	(unsigned long)size);
 | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if (ptr == NULL) | 
					
						
							|  |  |  | 		PanicAlert("Failed to allocate executable memory"); | 
					
						
							| 
									
										
										
										
											2011-01-12 21:25:33 +00:00
										 |  |  | #ifdef _M_X64
 | 
					
						
							| 
									
										
										
										
											2011-01-11 04:09:11 +00:00
										 |  |  | 	if ((u64)ptr >= 0x80000000 && low == true) | 
					
						
							| 
									
										
										
										
											2011-01-09 14:09:27 +00:00
										 |  |  | 		PanicAlert("Executable memory ended up above 2GB!"); | 
					
						
							| 
									
										
										
										
											2011-01-12 21:25:33 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-01-09 14:09:27 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return ptr; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2011-01-09 14:09:27 +00:00
										 |  |  | 	void* ptr = mmap(0, size, PROT_READ | PROT_WRITE, | 
					
						
							|  |  |  | 			MAP_ANON | MAP_PRIVATE, -1, 0); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-09 14:09:27 +00:00
										 |  |  | 	// printf("Mapped memory at %p (size %ld)\n", ptr,
 | 
					
						
							|  |  |  | 	//	(unsigned long)size);
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-09 14:09:27 +00:00
										 |  |  | 	if (ptr == NULL) | 
					
						
							|  |  |  | 		PanicAlert("Failed to allocate raw memory"); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-09 14:09:27 +00:00
										 |  |  | 	return ptr; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-25 20:35:05 +00:00
										 |  |  | void* AllocateAlignedMemory(size_t size,size_t alignment) | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | { | 
					
						
							|  |  |  | #ifdef _WIN32
 | 
					
						
							| 
									
										
										
										
											2011-02-25 20:35:05 +00:00
										 |  |  | 	void* ptr =  _aligned_malloc(size,alignment); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 	void* ptr = NULL; | 
					
						
							|  |  |  | 	posix_memalign(&ptr, alignment, size); | 
					
						
							|  |  |  | ; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// printf("Mapped memory at %p (size %ld)\n", ptr,
 | 
					
						
							|  |  |  | 	//	(unsigned long)size);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (ptr == NULL) | 
					
						
							|  |  |  | 		PanicAlert("Failed to allocate aligned memory"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void FreeMemoryPages(void* ptr, size_t size) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 	if (ptr) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2011-02-25 20:35:05 +00:00
										 |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											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?
 | 
					
						
							| 
									
										
										
										
											2011-02-25 20:35:05 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 		munmap(ptr, size); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-02-25 20:35:05 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void FreeAlignedMemory(void* ptr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (ptr) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 		_aligned_free(ptr); | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2011-02-25 20:35:05 +00:00
										 |  |  | 	free(ptr); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-02-25 20:35:05 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-09 14:09:27 +00:00
										 |  |  | #ifdef _WIN32
 | 
					
						
							| 
									
										
										
										
											2009-09-07 12:40:43 +00:00
										 |  |  | 	DWORD processID = GetCurrentProcessId(); | 
					
						
							| 
									
										
										
										
											2011-01-09 14:09:27 +00:00
										 |  |  | 	HANDLE hProcess; | 
					
						
							|  |  |  | 	PROCESS_MEMORY_COUNTERS pmc; | 
					
						
							| 
									
										
										
										
											2009-09-07 12:40:43 +00:00
										 |  |  | 	std::string Ret; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-09 14:09:27 +00:00
										 |  |  | 	// Print information about the memory usage of the process.
 | 
					
						
							| 
									
										
										
										
											2009-09-07 12:40:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-09 14:09:27 +00:00
										 |  |  | 	hProcess = OpenProcess(PROCESS_QUERY_INFORMATION | PROCESS_VM_READ, FALSE, processID); | 
					
						
							|  |  |  | 	if (NULL == hProcess) return "MemUsage Error"; | 
					
						
							| 
									
										
										
										
											2009-09-07 12:40:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-09 14:09:27 +00:00
										 |  |  | 	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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-09 14:09:27 +00:00
										 |  |  | 	CloseHandle(hProcess); | 
					
						
							| 
									
										
										
										
											2009-09-07 12:40:43 +00:00
										 |  |  | 	return Ret; | 
					
						
							| 
									
										
										
										
											2011-01-09 14:09:27 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2009-09-07 12:40:43 +00:00
										 |  |  | 	return ""; | 
					
						
							| 
									
										
										
										
											2011-01-09 14:09:27 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-11-15 22:26:39 +00:00
										 |  |  | } |