| 
									
										
										
										
											2015-05-12 23:38:56 -03:00
										 |  |  | // Copyright 2015 Citra Emulator Project
 | 
					
						
							| 
									
										
										
										
											2014-12-16 21:38:14 -08:00
										 |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							| 
									
										
										
										
											2014-04-08 19:15:46 -04:00
										 |  |  | // Refer to the license.txt file included.
 | 
					
						
							| 
									
										
										
										
											2013-09-18 23:52:51 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-27 15:16:39 +00:00
										 |  |  | #include <algorithm>
 | 
					
						
							| 
									
										
										
										
											2015-09-09 23:23:44 -04:00
										 |  |  | #include <cstring>
 | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  | #include <optional>
 | 
					
						
							| 
									
										
										
										
											2018-07-18 19:02:47 -04:00
										 |  |  | #include <utility>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-12 23:38:56 -03:00
										 |  |  | #include "common/assert.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-06 04:06:12 -03:00
										 |  |  | #include "common/common_types.h"
 | 
					
						
							|  |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2019-03-02 15:20:28 -05:00
										 |  |  | #include "common/page_table.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-06 04:06:12 -03:00
										 |  |  | #include "common/swap.h"
 | 
					
						
							| 
									
										
										
										
											2017-09-24 22:44:13 +01:00
										 |  |  | #include "core/arm/arm_interface.h"
 | 
					
						
							|  |  |  | #include "core/core.h"
 | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  | #include "core/device_memory.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/memory/page_table.h"
 | 
					
						
							| 
									
										
										
										
											2020-01-12 17:04:15 +01:00
										 |  |  | #include "core/hle/kernel/physical_memory.h"
 | 
					
						
							| 
									
										
										
										
											2015-08-05 21:26:52 -03:00
										 |  |  | #include "core/hle/kernel/process.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-20 23:52:38 -07:00
										 |  |  | #include "core/memory.h"
 | 
					
						
							| 
									
										
										
										
											2019-02-22 23:38:45 -05:00
										 |  |  | #include "video_core/gpu.h"
 | 
					
						
							| 
									
										
										
										
											2016-04-16 18:57:57 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-31 15:10:44 -04:00
										 |  |  | namespace Core::Memory { | 
					
						
							| 
									
										
										
										
											2015-05-12 23:38:56 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 12:33:20 -05:00
										 |  |  | // Implementation class used to keep the specifics of the memory subsystem hidden
 | 
					
						
							|  |  |  | // from outside classes. This also allows modification to the internals of the memory
 | 
					
						
							|  |  |  | // subsystem without needing to rebuild all files that make use of the memory interface.
 | 
					
						
							|  |  |  | struct Memory::Impl { | 
					
						
							|  |  |  |     explicit Impl(Core::System& system_) : system{system_} {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 18:34:30 -05:00
										 |  |  |     void SetCurrentPageTable(Kernel::Process& process) { | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |         current_page_table = &process.PageTable().PageTableImpl(); | 
					
						
							| 
									
										
										
										
											2019-11-26 18:34:30 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |         const std::size_t address_space_width = process.PageTable().GetAddressSpaceWidth(); | 
					
						
							| 
									
										
										
										
											2019-11-26 18:34:30 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         system.ArmInterface(0).PageTableChanged(*current_page_table, address_space_width); | 
					
						
							|  |  |  |         system.ArmInterface(1).PageTableChanged(*current_page_table, address_space_width); | 
					
						
							|  |  |  |         system.ArmInterface(2).PageTableChanged(*current_page_table, address_space_width); | 
					
						
							|  |  |  |         system.ArmInterface(3).PageTableChanged(*current_page_table, address_space_width); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |     void MapMemoryRegion(Common::PageTable& page_table, VAddr base, u64 size, PAddr target) { | 
					
						
							| 
									
										
										
										
											2019-11-26 13:09:12 -05:00
										 |  |  |         ASSERT_MSG((size & PAGE_MASK) == 0, "non-page aligned size: {:016X}", size); | 
					
						
							|  |  |  |         ASSERT_MSG((base & PAGE_MASK) == 0, "non-page aligned base: {:016X}", base); | 
					
						
							|  |  |  |         MapPages(page_table, base / PAGE_SIZE, size / PAGE_SIZE, target, Common::PageType::Memory); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-11-26 12:33:20 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 13:09:12 -05:00
										 |  |  |     void MapIoRegion(Common::PageTable& page_table, VAddr base, u64 size, | 
					
						
							|  |  |  |                      Common::MemoryHookPointer mmio_handler) { | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |         UNIMPLEMENTED(); | 
					
						
							| 
									
										
										
										
											2019-11-26 13:09:12 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-11-26 12:33:20 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 13:09:12 -05:00
										 |  |  |     void UnmapRegion(Common::PageTable& page_table, VAddr base, u64 size) { | 
					
						
							|  |  |  |         ASSERT_MSG((size & PAGE_MASK) == 0, "non-page aligned size: {:016X}", size); | 
					
						
							|  |  |  |         ASSERT_MSG((base & PAGE_MASK) == 0, "non-page aligned base: {:016X}", base); | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |         MapPages(page_table, base / PAGE_SIZE, size / PAGE_SIZE, 0, Common::PageType::Unmapped); | 
					
						
							| 
									
										
										
										
											2019-11-26 13:09:12 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-05-02 22:36:51 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 13:09:12 -05:00
										 |  |  |     void AddDebugHook(Common::PageTable& page_table, VAddr base, u64 size, | 
					
						
							|  |  |  |                       Common::MemoryHookPointer hook) { | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |         UNIMPLEMENTED(); | 
					
						
							| 
									
										
										
										
											2019-11-26 13:09:12 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-09-24 22:42:42 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 13:09:12 -05:00
										 |  |  |     void RemoveDebugHook(Common::PageTable& page_table, VAddr base, u64 size, | 
					
						
							|  |  |  |                          Common::MemoryHookPointer hook) { | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |         UNIMPLEMENTED(); | 
					
						
							| 
									
										
										
										
											2019-11-26 13:09:12 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 13:46:41 -05:00
										 |  |  |     bool IsValidVirtualAddress(const Kernel::Process& process, const VAddr vaddr) const { | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |         const auto& page_table = process.PageTable().PageTableImpl(); | 
					
						
							| 
									
										
										
										
											2019-11-26 13:46:41 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         const u8* const page_pointer = page_table.pointers[vaddr >> PAGE_BITS]; | 
					
						
							|  |  |  |         if (page_pointer != nullptr) { | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (page_table.attributes[vaddr >> PAGE_BITS] == Common::PageType::RasterizerCachedMemory) { | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (page_table.attributes[vaddr >> PAGE_BITS] != Common::PageType::Special) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool IsValidVirtualAddress(VAddr vaddr) const { | 
					
						
							|  |  |  |         return IsValidVirtualAddress(*system.CurrentProcess(), vaddr); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |     u8* GetPointerFromRasterizerCachedMemory(VAddr vaddr) const { | 
					
						
							|  |  |  |         const PAddr paddr{current_page_table->backing_addr[vaddr >> PAGE_BITS]}; | 
					
						
							| 
									
										
										
										
											2019-11-26 18:28:44 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |         if (!paddr) { | 
					
						
							|  |  |  |             return {}; | 
					
						
							| 
									
										
										
										
											2019-11-26 18:28:44 -05:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |         return system.DeviceMemory().GetPointer(paddr) + vaddr; | 
					
						
							| 
									
										
										
										
											2019-11-26 18:28:44 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |     u8* GetPointer(const VAddr vaddr) const { | 
					
						
							|  |  |  |         u8* const page_pointer{current_page_table->pointers[vaddr >> PAGE_BITS]}; | 
					
						
							|  |  |  |         if (page_pointer) { | 
					
						
							| 
									
										
										
										
											2019-12-31 00:11:45 +01:00
										 |  |  |             return page_pointer + vaddr; | 
					
						
							| 
									
										
										
										
											2019-11-26 15:19:15 -05:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (current_page_table->attributes[vaddr >> PAGE_BITS] == | 
					
						
							|  |  |  |             Common::PageType::RasterizerCachedMemory) { | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |             return GetPointerFromRasterizerCachedMemory(vaddr); | 
					
						
							| 
									
										
										
										
											2019-11-26 15:19:15 -05:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2019-11-26 15:19:15 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 16:29:34 -05:00
										 |  |  |     u8 Read8(const VAddr addr) { | 
					
						
							|  |  |  |         return Read<u8>(addr); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     u16 Read16(const VAddr addr) { | 
					
						
							| 
									
										
										
										
											2020-04-08 23:03:25 -04:00
										 |  |  |         if ((addr & 1) == 0) { | 
					
						
							|  |  |  |             return Read<u16_le>(addr); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             const u8 a{Read<u8>(addr)}; | 
					
						
							|  |  |  |             const u8 b{Read<u8>(addr + sizeof(u8))}; | 
					
						
							|  |  |  |             return (static_cast<u16>(b) << 8) | a; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-11-26 16:29:34 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     u32 Read32(const VAddr addr) { | 
					
						
							| 
									
										
										
										
											2020-04-08 23:03:25 -04:00
										 |  |  |         if ((addr & 3) == 0) { | 
					
						
							|  |  |  |             return Read<u32_le>(addr); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             const u16 a{Read16(addr)}; | 
					
						
							|  |  |  |             const u16 b{Read16(addr + sizeof(u16))}; | 
					
						
							|  |  |  |             return (static_cast<u32>(b) << 16) | a; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-11-26 16:29:34 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     u64 Read64(const VAddr addr) { | 
					
						
							| 
									
										
										
										
											2020-04-08 23:03:25 -04:00
										 |  |  |         if ((addr & 7) == 0) { | 
					
						
							|  |  |  |             return Read<u64_le>(addr); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             const u32 a{Read32(addr)}; | 
					
						
							|  |  |  |             const u32 b{Read32(addr + sizeof(u32))}; | 
					
						
							|  |  |  |             return (static_cast<u64>(b) << 32) | a; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-11-26 16:29:34 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 17:39:57 -05:00
										 |  |  |     void Write8(const VAddr addr, const u8 data) { | 
					
						
							|  |  |  |         Write<u8>(addr, data); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void Write16(const VAddr addr, const u16 data) { | 
					
						
							| 
									
										
										
										
											2020-04-08 23:03:25 -04:00
										 |  |  |         if ((addr & 1) == 0) { | 
					
						
							|  |  |  |             Write<u16_le>(addr, data); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             Write<u8>(addr, static_cast<u8>(data)); | 
					
						
							|  |  |  |             Write<u8>(addr + sizeof(u8), static_cast<u8>(data >> 8)); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-11-26 17:39:57 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void Write32(const VAddr addr, const u32 data) { | 
					
						
							| 
									
										
										
										
											2020-04-08 23:03:25 -04:00
										 |  |  |         if ((addr & 3) == 0) { | 
					
						
							|  |  |  |             Write<u32_le>(addr, data); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             Write16(addr, static_cast<u16>(data)); | 
					
						
							|  |  |  |             Write16(addr + sizeof(u16), static_cast<u16>(data >> 16)); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-11-26 17:39:57 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void Write64(const VAddr addr, const u64 data) { | 
					
						
							| 
									
										
										
										
											2020-04-08 23:03:25 -04:00
										 |  |  |         if ((addr & 7) == 0) { | 
					
						
							|  |  |  |             Write<u64_le>(addr, data); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             Write32(addr, static_cast<u32>(data)); | 
					
						
							|  |  |  |             Write32(addr + sizeof(u32), static_cast<u32>(data >> 32)); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-11-26 17:39:57 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 15:48:19 -05:00
										 |  |  |     std::string ReadCString(VAddr vaddr, std::size_t max_length) { | 
					
						
							|  |  |  |         std::string string; | 
					
						
							|  |  |  |         string.reserve(max_length); | 
					
						
							|  |  |  |         for (std::size_t i = 0; i < max_length; ++i) { | 
					
						
							|  |  |  |             const char c = Read8(vaddr); | 
					
						
							|  |  |  |             if (c == '\0') { | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             string.push_back(c); | 
					
						
							|  |  |  |             ++vaddr; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         string.shrink_to_fit(); | 
					
						
							|  |  |  |         return string; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 16:29:34 -05:00
										 |  |  |     void ReadBlock(const Kernel::Process& process, const VAddr src_addr, void* dest_buffer, | 
					
						
							|  |  |  |                    const std::size_t size) { | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |         const auto& page_table = process.PageTable().PageTableImpl(); | 
					
						
							| 
									
										
										
										
											2019-11-26 16:29:34 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         std::size_t remaining_size = size; | 
					
						
							|  |  |  |         std::size_t page_index = src_addr >> PAGE_BITS; | 
					
						
							|  |  |  |         std::size_t page_offset = src_addr & PAGE_MASK; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         while (remaining_size > 0) { | 
					
						
							|  |  |  |             const std::size_t copy_amount = | 
					
						
							|  |  |  |                 std::min(static_cast<std::size_t>(PAGE_SIZE) - page_offset, remaining_size); | 
					
						
							|  |  |  |             const auto current_vaddr = static_cast<VAddr>((page_index << PAGE_BITS) + page_offset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             switch (page_table.attributes[page_index]) { | 
					
						
							|  |  |  |             case Common::PageType::Unmapped: { | 
					
						
							|  |  |  |                 LOG_ERROR(HW_Memory, | 
					
						
							|  |  |  |                           "Unmapped ReadBlock @ 0x{:016X} (start address = 0x{:016X}, size = {})", | 
					
						
							|  |  |  |                           current_vaddr, src_addr, size); | 
					
						
							|  |  |  |                 std::memset(dest_buffer, 0, copy_amount); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             case Common::PageType::Memory: { | 
					
						
							|  |  |  |                 DEBUG_ASSERT(page_table.pointers[page_index]); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-31 00:11:45 +01:00
										 |  |  |                 const u8* const src_ptr = | 
					
						
							|  |  |  |                     page_table.pointers[page_index] + page_offset + (page_index << PAGE_BITS); | 
					
						
							| 
									
										
										
										
											2019-11-26 16:29:34 -05:00
										 |  |  |                 std::memcpy(dest_buffer, src_ptr, copy_amount); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             case Common::PageType::RasterizerCachedMemory: { | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |                 const u8* const host_ptr{GetPointerFromRasterizerCachedMemory(current_vaddr)}; | 
					
						
							| 
									
										
										
										
											2020-04-05 12:58:23 -04:00
										 |  |  |                 system.GPU().FlushRegion(current_vaddr, copy_amount); | 
					
						
							| 
									
										
										
										
											2019-11-26 16:29:34 -05:00
										 |  |  |                 std::memcpy(dest_buffer, host_ptr, copy_amount); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |                 UNREACHABLE(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             page_index++; | 
					
						
							|  |  |  |             page_offset = 0; | 
					
						
							|  |  |  |             dest_buffer = static_cast<u8*>(dest_buffer) + copy_amount; | 
					
						
							|  |  |  |             remaining_size -= copy_amount; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-05 17:23:49 -04:00
										 |  |  |     void ReadBlockUnsafe(const Kernel::Process& process, const VAddr src_addr, void* dest_buffer, | 
					
						
							|  |  |  |                          const std::size_t size) { | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |         const auto& page_table = process.PageTable().PageTableImpl(); | 
					
						
							| 
									
										
										
										
											2020-04-05 17:23:49 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         std::size_t remaining_size = size; | 
					
						
							|  |  |  |         std::size_t page_index = src_addr >> PAGE_BITS; | 
					
						
							|  |  |  |         std::size_t page_offset = src_addr & PAGE_MASK; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         while (remaining_size > 0) { | 
					
						
							|  |  |  |             const std::size_t copy_amount = | 
					
						
							|  |  |  |                 std::min(static_cast<std::size_t>(PAGE_SIZE) - page_offset, remaining_size); | 
					
						
							|  |  |  |             const auto current_vaddr = static_cast<VAddr>((page_index << PAGE_BITS) + page_offset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             switch (page_table.attributes[page_index]) { | 
					
						
							|  |  |  |             case Common::PageType::Unmapped: { | 
					
						
							|  |  |  |                 LOG_ERROR(HW_Memory, | 
					
						
							|  |  |  |                           "Unmapped ReadBlock @ 0x{:016X} (start address = 0x{:016X}, size = {})", | 
					
						
							|  |  |  |                           current_vaddr, src_addr, size); | 
					
						
							|  |  |  |                 std::memset(dest_buffer, 0, copy_amount); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             case Common::PageType::Memory: { | 
					
						
							|  |  |  |                 DEBUG_ASSERT(page_table.pointers[page_index]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 const u8* const src_ptr = | 
					
						
							|  |  |  |                     page_table.pointers[page_index] + page_offset + (page_index << PAGE_BITS); | 
					
						
							|  |  |  |                 std::memcpy(dest_buffer, src_ptr, copy_amount); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             case Common::PageType::RasterizerCachedMemory: { | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |                 const u8* const host_ptr{GetPointerFromRasterizerCachedMemory(current_vaddr)}; | 
					
						
							| 
									
										
										
										
											2020-04-05 17:23:49 -04:00
										 |  |  |                 std::memcpy(dest_buffer, host_ptr, copy_amount); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |                 UNREACHABLE(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             page_index++; | 
					
						
							|  |  |  |             page_offset = 0; | 
					
						
							|  |  |  |             dest_buffer = static_cast<u8*>(dest_buffer) + copy_amount; | 
					
						
							|  |  |  |             remaining_size -= copy_amount; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 16:29:34 -05:00
										 |  |  |     void ReadBlock(const VAddr src_addr, void* dest_buffer, const std::size_t size) { | 
					
						
							|  |  |  |         ReadBlock(*system.CurrentProcess(), src_addr, dest_buffer, size); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-05 17:23:49 -04:00
										 |  |  |     void ReadBlockUnsafe(const VAddr src_addr, void* dest_buffer, const std::size_t size) { | 
					
						
							|  |  |  |         ReadBlockUnsafe(*system.CurrentProcess(), src_addr, dest_buffer, size); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 17:39:57 -05:00
										 |  |  |     void WriteBlock(const Kernel::Process& process, const VAddr dest_addr, const void* src_buffer, | 
					
						
							|  |  |  |                     const std::size_t size) { | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |         const auto& page_table = process.PageTable().PageTableImpl(); | 
					
						
							| 
									
										
										
										
											2019-11-26 17:39:57 -05:00
										 |  |  |         std::size_t remaining_size = size; | 
					
						
							|  |  |  |         std::size_t page_index = dest_addr >> PAGE_BITS; | 
					
						
							|  |  |  |         std::size_t page_offset = dest_addr & PAGE_MASK; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         while (remaining_size > 0) { | 
					
						
							|  |  |  |             const std::size_t copy_amount = | 
					
						
							|  |  |  |                 std::min(static_cast<std::size_t>(PAGE_SIZE) - page_offset, remaining_size); | 
					
						
							|  |  |  |             const auto current_vaddr = static_cast<VAddr>((page_index << PAGE_BITS) + page_offset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             switch (page_table.attributes[page_index]) { | 
					
						
							|  |  |  |             case Common::PageType::Unmapped: { | 
					
						
							|  |  |  |                 LOG_ERROR(HW_Memory, | 
					
						
							|  |  |  |                           "Unmapped WriteBlock @ 0x{:016X} (start address = 0x{:016X}, size = {})", | 
					
						
							|  |  |  |                           current_vaddr, dest_addr, size); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             case Common::PageType::Memory: { | 
					
						
							|  |  |  |                 DEBUG_ASSERT(page_table.pointers[page_index]); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-31 00:11:45 +01:00
										 |  |  |                 u8* const dest_ptr = | 
					
						
							|  |  |  |                     page_table.pointers[page_index] + page_offset + (page_index << PAGE_BITS); | 
					
						
							| 
									
										
										
										
											2019-11-26 17:39:57 -05:00
										 |  |  |                 std::memcpy(dest_ptr, src_buffer, copy_amount); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             case Common::PageType::RasterizerCachedMemory: { | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |                 u8* const host_ptr{GetPointerFromRasterizerCachedMemory(current_vaddr)}; | 
					
						
							| 
									
										
										
										
											2020-04-05 12:58:23 -04:00
										 |  |  |                 system.GPU().InvalidateRegion(current_vaddr, copy_amount); | 
					
						
							| 
									
										
										
										
											2019-11-26 17:39:57 -05:00
										 |  |  |                 std::memcpy(host_ptr, src_buffer, copy_amount); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |                 UNREACHABLE(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             page_index++; | 
					
						
							|  |  |  |             page_offset = 0; | 
					
						
							|  |  |  |             src_buffer = static_cast<const u8*>(src_buffer) + copy_amount; | 
					
						
							|  |  |  |             remaining_size -= copy_amount; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-05 17:23:49 -04:00
										 |  |  |     void WriteBlockUnsafe(const Kernel::Process& process, const VAddr dest_addr, | 
					
						
							|  |  |  |                           const void* src_buffer, const std::size_t size) { | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |         const auto& page_table = process.PageTable().PageTableImpl(); | 
					
						
							| 
									
										
										
										
											2020-04-05 17:23:49 -04:00
										 |  |  |         std::size_t remaining_size = size; | 
					
						
							|  |  |  |         std::size_t page_index = dest_addr >> PAGE_BITS; | 
					
						
							|  |  |  |         std::size_t page_offset = dest_addr & PAGE_MASK; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         while (remaining_size > 0) { | 
					
						
							|  |  |  |             const std::size_t copy_amount = | 
					
						
							|  |  |  |                 std::min(static_cast<std::size_t>(PAGE_SIZE) - page_offset, remaining_size); | 
					
						
							|  |  |  |             const auto current_vaddr = static_cast<VAddr>((page_index << PAGE_BITS) + page_offset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             switch (page_table.attributes[page_index]) { | 
					
						
							|  |  |  |             case Common::PageType::Unmapped: { | 
					
						
							|  |  |  |                 LOG_ERROR(HW_Memory, | 
					
						
							|  |  |  |                           "Unmapped WriteBlock @ 0x{:016X} (start address = 0x{:016X}, size = {})", | 
					
						
							|  |  |  |                           current_vaddr, dest_addr, size); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             case Common::PageType::Memory: { | 
					
						
							|  |  |  |                 DEBUG_ASSERT(page_table.pointers[page_index]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 u8* const dest_ptr = | 
					
						
							|  |  |  |                     page_table.pointers[page_index] + page_offset + (page_index << PAGE_BITS); | 
					
						
							|  |  |  |                 std::memcpy(dest_ptr, src_buffer, copy_amount); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             case Common::PageType::RasterizerCachedMemory: { | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |                 u8* const host_ptr{GetPointerFromRasterizerCachedMemory(current_vaddr)}; | 
					
						
							| 
									
										
										
										
											2020-04-05 17:23:49 -04:00
										 |  |  |                 std::memcpy(host_ptr, src_buffer, copy_amount); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |                 UNREACHABLE(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             page_index++; | 
					
						
							|  |  |  |             page_offset = 0; | 
					
						
							|  |  |  |             src_buffer = static_cast<const u8*>(src_buffer) + copy_amount; | 
					
						
							|  |  |  |             remaining_size -= copy_amount; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 17:39:57 -05:00
										 |  |  |     void WriteBlock(const VAddr dest_addr, const void* src_buffer, const std::size_t size) { | 
					
						
							|  |  |  |         WriteBlock(*system.CurrentProcess(), dest_addr, src_buffer, size); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-05 17:23:49 -04:00
										 |  |  |     void WriteBlockUnsafe(const VAddr dest_addr, const void* src_buffer, const std::size_t size) { | 
					
						
							|  |  |  |         WriteBlockUnsafe(*system.CurrentProcess(), dest_addr, src_buffer, size); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 16:06:49 -05:00
										 |  |  |     void ZeroBlock(const Kernel::Process& process, const VAddr dest_addr, const std::size_t size) { | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |         const auto& page_table = process.PageTable().PageTableImpl(); | 
					
						
							| 
									
										
										
										
											2019-11-26 16:06:49 -05:00
										 |  |  |         std::size_t remaining_size = size; | 
					
						
							|  |  |  |         std::size_t page_index = dest_addr >> PAGE_BITS; | 
					
						
							|  |  |  |         std::size_t page_offset = dest_addr & PAGE_MASK; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         while (remaining_size > 0) { | 
					
						
							|  |  |  |             const std::size_t copy_amount = | 
					
						
							|  |  |  |                 std::min(static_cast<std::size_t>(PAGE_SIZE) - page_offset, remaining_size); | 
					
						
							|  |  |  |             const auto current_vaddr = static_cast<VAddr>((page_index << PAGE_BITS) + page_offset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             switch (page_table.attributes[page_index]) { | 
					
						
							|  |  |  |             case Common::PageType::Unmapped: { | 
					
						
							|  |  |  |                 LOG_ERROR(HW_Memory, | 
					
						
							|  |  |  |                           "Unmapped ZeroBlock @ 0x{:016X} (start address = 0x{:016X}, size = {})", | 
					
						
							|  |  |  |                           current_vaddr, dest_addr, size); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             case Common::PageType::Memory: { | 
					
						
							|  |  |  |                 DEBUG_ASSERT(page_table.pointers[page_index]); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-31 00:11:45 +01:00
										 |  |  |                 u8* dest_ptr = | 
					
						
							|  |  |  |                     page_table.pointers[page_index] + page_offset + (page_index << PAGE_BITS); | 
					
						
							| 
									
										
										
										
											2019-11-26 16:06:49 -05:00
										 |  |  |                 std::memset(dest_ptr, 0, copy_amount); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             case Common::PageType::RasterizerCachedMemory: { | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |                 u8* const host_ptr{GetPointerFromRasterizerCachedMemory(current_vaddr)}; | 
					
						
							| 
									
										
										
										
											2020-04-05 12:58:23 -04:00
										 |  |  |                 system.GPU().InvalidateRegion(current_vaddr, copy_amount); | 
					
						
							| 
									
										
										
										
											2019-11-26 16:06:49 -05:00
										 |  |  |                 std::memset(host_ptr, 0, copy_amount); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |                 UNREACHABLE(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             page_index++; | 
					
						
							|  |  |  |             page_offset = 0; | 
					
						
							|  |  |  |             remaining_size -= copy_amount; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void ZeroBlock(const VAddr dest_addr, const std::size_t size) { | 
					
						
							|  |  |  |         ZeroBlock(*system.CurrentProcess(), dest_addr, size); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void CopyBlock(const Kernel::Process& process, VAddr dest_addr, VAddr src_addr, | 
					
						
							|  |  |  |                    const std::size_t size) { | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |         const auto& page_table = process.PageTable().PageTableImpl(); | 
					
						
							| 
									
										
										
										
											2019-11-26 16:06:49 -05:00
										 |  |  |         std::size_t remaining_size = size; | 
					
						
							|  |  |  |         std::size_t page_index = src_addr >> PAGE_BITS; | 
					
						
							|  |  |  |         std::size_t page_offset = src_addr & PAGE_MASK; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         while (remaining_size > 0) { | 
					
						
							|  |  |  |             const std::size_t copy_amount = | 
					
						
							|  |  |  |                 std::min(static_cast<std::size_t>(PAGE_SIZE) - page_offset, remaining_size); | 
					
						
							|  |  |  |             const auto current_vaddr = static_cast<VAddr>((page_index << PAGE_BITS) + page_offset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             switch (page_table.attributes[page_index]) { | 
					
						
							|  |  |  |             case Common::PageType::Unmapped: { | 
					
						
							|  |  |  |                 LOG_ERROR(HW_Memory, | 
					
						
							|  |  |  |                           "Unmapped CopyBlock @ 0x{:016X} (start address = 0x{:016X}, size = {})", | 
					
						
							|  |  |  |                           current_vaddr, src_addr, size); | 
					
						
							|  |  |  |                 ZeroBlock(process, dest_addr, copy_amount); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             case Common::PageType::Memory: { | 
					
						
							|  |  |  |                 DEBUG_ASSERT(page_table.pointers[page_index]); | 
					
						
							| 
									
										
										
										
											2019-12-31 00:11:45 +01:00
										 |  |  |                 const u8* src_ptr = | 
					
						
							|  |  |  |                     page_table.pointers[page_index] + page_offset + (page_index << PAGE_BITS); | 
					
						
							| 
									
										
										
										
											2019-11-26 16:06:49 -05:00
										 |  |  |                 WriteBlock(process, dest_addr, src_ptr, copy_amount); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             case Common::PageType::RasterizerCachedMemory: { | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |                 const u8* const host_ptr{GetPointerFromRasterizerCachedMemory(current_vaddr)}; | 
					
						
							| 
									
										
										
										
											2020-04-05 12:58:23 -04:00
										 |  |  |                 system.GPU().FlushRegion(current_vaddr, copy_amount); | 
					
						
							| 
									
										
										
										
											2019-11-26 16:06:49 -05:00
										 |  |  |                 WriteBlock(process, dest_addr, host_ptr, copy_amount); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |                 UNREACHABLE(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             page_index++; | 
					
						
							|  |  |  |             page_offset = 0; | 
					
						
							|  |  |  |             dest_addr += static_cast<VAddr>(copy_amount); | 
					
						
							|  |  |  |             src_addr += static_cast<VAddr>(copy_amount); | 
					
						
							|  |  |  |             remaining_size -= copy_amount; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void CopyBlock(VAddr dest_addr, VAddr src_addr, std::size_t size) { | 
					
						
							|  |  |  |         return CopyBlock(*system.CurrentProcess(), dest_addr, src_addr, size); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 15:56:13 -05:00
										 |  |  |     void RasterizerMarkRegionCached(VAddr vaddr, u64 size, bool cached) { | 
					
						
							|  |  |  |         if (vaddr == 0) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Iterate over a contiguous CPU address space, which corresponds to the specified GPU
 | 
					
						
							|  |  |  |         // address space, marking the region as un/cached. The region is marked un/cached at a
 | 
					
						
							|  |  |  |         // granularity of CPU pages, hence why we iterate on a CPU page basis (note: GPU page size
 | 
					
						
							|  |  |  |         // is different). This assumes the specified GPU address region is contiguous as well.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         u64 num_pages = ((vaddr + size - 1) >> PAGE_BITS) - (vaddr >> PAGE_BITS) + 1; | 
					
						
							|  |  |  |         for (unsigned i = 0; i < num_pages; ++i, vaddr += PAGE_SIZE) { | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |             Common::PageType& page_type{current_page_table->attributes[vaddr >> PAGE_BITS]}; | 
					
						
							| 
									
										
										
										
											2019-11-26 15:56:13 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if (cached) { | 
					
						
							|  |  |  |                 // Switch page type to cached if now cached
 | 
					
						
							|  |  |  |                 switch (page_type) { | 
					
						
							|  |  |  |                 case Common::PageType::Unmapped: | 
					
						
							|  |  |  |                     // It is not necessary for a process to have this region mapped into its address
 | 
					
						
							|  |  |  |                     // space, for example, a system module need not have a VRAM mapping.
 | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 case Common::PageType::Memory: | 
					
						
							|  |  |  |                     page_type = Common::PageType::RasterizerCachedMemory; | 
					
						
							|  |  |  |                     current_page_table->pointers[vaddr >> PAGE_BITS] = nullptr; | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 case Common::PageType::RasterizerCachedMemory: | 
					
						
							|  |  |  |                     // There can be more than one GPU region mapped per CPU region, so it's common
 | 
					
						
							|  |  |  |                     // that this area is already marked as cached.
 | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 default: | 
					
						
							|  |  |  |                     UNREACHABLE(); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 // Switch page type to uncached if now uncached
 | 
					
						
							|  |  |  |                 switch (page_type) { | 
					
						
							|  |  |  |                 case Common::PageType::Unmapped: | 
					
						
							|  |  |  |                     // It is not necessary for a process to have this region mapped into its address
 | 
					
						
							|  |  |  |                     // space, for example, a system module need not have a VRAM mapping.
 | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 case Common::PageType::Memory: | 
					
						
							|  |  |  |                     // There can be more than one GPU region mapped per CPU region, so it's common
 | 
					
						
							|  |  |  |                     // that this area is already unmarked as cached.
 | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 case Common::PageType::RasterizerCachedMemory: { | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |                     u8* pointer{GetPointerFromRasterizerCachedMemory(vaddr & ~PAGE_MASK)}; | 
					
						
							| 
									
										
										
										
											2019-11-26 15:56:13 -05:00
										 |  |  |                     if (pointer == nullptr) { | 
					
						
							|  |  |  |                         // It's possible that this function has been called while updating the
 | 
					
						
							|  |  |  |                         // pagetable after unmapping a VMA. In that case the underlying VMA will no
 | 
					
						
							|  |  |  |                         // longer exist, and we should just leave the pagetable entry blank.
 | 
					
						
							|  |  |  |                         page_type = Common::PageType::Unmapped; | 
					
						
							|  |  |  |                     } else { | 
					
						
							|  |  |  |                         page_type = Common::PageType::Memory; | 
					
						
							| 
									
										
										
										
											2019-12-31 00:11:45 +01:00
										 |  |  |                         current_page_table->pointers[vaddr >> PAGE_BITS] = | 
					
						
							|  |  |  |                             pointer - (vaddr & ~PAGE_MASK); | 
					
						
							| 
									
										
										
										
											2019-11-26 15:56:13 -05:00
										 |  |  |                     } | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 default: | 
					
						
							|  |  |  |                     UNREACHABLE(); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 13:09:12 -05:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Maps a region of pages as a specific type. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param page_table The page table to use to perform the mapping. | 
					
						
							|  |  |  |      * @param base       The base address to begin mapping at. | 
					
						
							|  |  |  |      * @param size       The total size of the range in bytes. | 
					
						
							|  |  |  |      * @param memory     The memory to map. | 
					
						
							|  |  |  |      * @param type       The page type to map the memory as. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |     void MapPages(Common::PageTable& page_table, VAddr base, u64 size, PAddr target, | 
					
						
							| 
									
										
										
										
											2019-11-26 13:09:12 -05:00
										 |  |  |                   Common::PageType type) { | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |         LOG_DEBUG(HW_Memory, "Mapping {:016X} onto {:016X}-{:016X}", target, base * PAGE_SIZE, | 
					
						
							| 
									
										
										
										
											2019-11-26 13:09:12 -05:00
										 |  |  |                   (base + size) * PAGE_SIZE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // During boot, current_page_table might not be set yet, in which case we need not flush
 | 
					
						
							|  |  |  |         if (system.IsPoweredOn()) { | 
					
						
							|  |  |  |             auto& gpu = system.GPU(); | 
					
						
							|  |  |  |             for (u64 i = 0; i < size; i++) { | 
					
						
							|  |  |  |                 const auto page = base + i; | 
					
						
							|  |  |  |                 if (page_table.attributes[page] == Common::PageType::RasterizerCachedMemory) { | 
					
						
							|  |  |  |                     gpu.FlushAndInvalidateRegion(page << PAGE_BITS, PAGE_SIZE); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2019-09-18 21:50:21 -04:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-03-22 22:56:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 13:09:12 -05:00
										 |  |  |         const VAddr end = base + size; | 
					
						
							|  |  |  |         ASSERT_MSG(end <= page_table.pointers.size(), "out of range mapping at {:016X}", | 
					
						
							|  |  |  |                    base + page_table.pointers.size()); | 
					
						
							| 
									
										
										
										
											2015-05-12 23:38:56 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |         if (!target) { | 
					
						
							|  |  |  |             while (base != end) { | 
					
						
							|  |  |  |                 page_table.pointers[base] = nullptr; | 
					
						
							|  |  |  |                 page_table.attributes[base] = type; | 
					
						
							|  |  |  |                 page_table.backing_addr[base] = 0; | 
					
						
							| 
									
										
										
										
											2015-05-12 23:38:56 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |                 base += 1; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-11-26 13:09:12 -05:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             while (base != end) { | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |                 page_table.pointers[base] = | 
					
						
							|  |  |  |                     system.DeviceMemory().GetPointer(target) - (base << PAGE_BITS); | 
					
						
							|  |  |  |                 page_table.attributes[base] = type; | 
					
						
							|  |  |  |                 page_table.backing_addr[base] = target - (base << PAGE_BITS); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-31 00:11:45 +01:00
										 |  |  |                 ASSERT_MSG(page_table.pointers[base], | 
					
						
							|  |  |  |                            "memory mapping base yield a nullptr within the table"); | 
					
						
							| 
									
										
										
										
											2019-02-27 23:22:47 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 13:09:12 -05:00
										 |  |  |                 base += 1; | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |                 target += PAGE_SIZE; | 
					
						
							| 
									
										
										
										
											2019-11-26 13:09:12 -05:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-02-27 23:22:47 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2014-04-01 18:18:02 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-09-18 23:52:51 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 16:29:34 -05:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Reads a particular data type out of memory at the given virtual address. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param vaddr The virtual address to read the data type from. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @tparam T The data type to read out of memory. This type *must* be | 
					
						
							|  |  |  |      *           trivially copyable, otherwise the behavior of this function | 
					
						
							|  |  |  |      *           is undefined. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @returns The instance of T read from the specified virtual address. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     template <typename T> | 
					
						
							|  |  |  |     T Read(const VAddr vaddr) { | 
					
						
							|  |  |  |         const u8* const page_pointer = current_page_table->pointers[vaddr >> PAGE_BITS]; | 
					
						
							|  |  |  |         if (page_pointer != nullptr) { | 
					
						
							|  |  |  |             // NOTE: Avoid adding any extra logic to this fast-path block
 | 
					
						
							|  |  |  |             T value; | 
					
						
							| 
									
										
										
										
											2019-12-31 00:11:45 +01:00
										 |  |  |             std::memcpy(&value, &page_pointer[vaddr], sizeof(T)); | 
					
						
							| 
									
										
										
										
											2019-11-26 16:29:34 -05:00
										 |  |  |             return value; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const Common::PageType type = current_page_table->attributes[vaddr >> PAGE_BITS]; | 
					
						
							|  |  |  |         switch (type) { | 
					
						
							|  |  |  |         case Common::PageType::Unmapped: | 
					
						
							|  |  |  |             LOG_ERROR(HW_Memory, "Unmapped Read{} @ 0x{:08X}", sizeof(T) * 8, vaddr); | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |         case Common::PageType::Memory: | 
					
						
							|  |  |  |             ASSERT_MSG(false, "Mapped memory page without a pointer @ {:016X}", vaddr); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case Common::PageType::RasterizerCachedMemory: { | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |             const u8* const host_ptr{GetPointerFromRasterizerCachedMemory(vaddr)}; | 
					
						
							| 
									
										
										
										
											2020-04-05 12:58:23 -04:00
										 |  |  |             system.GPU().FlushRegion(vaddr, sizeof(T)); | 
					
						
							| 
									
										
										
										
											2019-11-26 16:29:34 -05:00
										 |  |  |             T value; | 
					
						
							|  |  |  |             std::memcpy(&value, host_ptr, sizeof(T)); | 
					
						
							|  |  |  |             return value; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             UNREACHABLE(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 17:39:57 -05:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Writes a particular data type to memory at the given virtual address. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param vaddr The virtual address to write the data type to. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @tparam T The data type to write to memory. This type *must* be | 
					
						
							|  |  |  |      *           trivially copyable, otherwise the behavior of this function | 
					
						
							|  |  |  |      *           is undefined. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @returns The instance of T write to the specified virtual address. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     template <typename T> | 
					
						
							|  |  |  |     void Write(const VAddr vaddr, const T data) { | 
					
						
							|  |  |  |         u8* const page_pointer = current_page_table->pointers[vaddr >> PAGE_BITS]; | 
					
						
							|  |  |  |         if (page_pointer != nullptr) { | 
					
						
							|  |  |  |             // NOTE: Avoid adding any extra logic to this fast-path block
 | 
					
						
							| 
									
										
										
										
											2019-12-31 00:11:45 +01:00
										 |  |  |             std::memcpy(&page_pointer[vaddr], &data, sizeof(T)); | 
					
						
							| 
									
										
										
										
											2019-11-26 17:39:57 -05:00
										 |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const Common::PageType type = current_page_table->attributes[vaddr >> PAGE_BITS]; | 
					
						
							|  |  |  |         switch (type) { | 
					
						
							|  |  |  |         case Common::PageType::Unmapped: | 
					
						
							|  |  |  |             LOG_ERROR(HW_Memory, "Unmapped Write{} 0x{:08X} @ 0x{:016X}", sizeof(data) * 8, | 
					
						
							|  |  |  |                       static_cast<u32>(data), vaddr); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         case Common::PageType::Memory: | 
					
						
							|  |  |  |             ASSERT_MSG(false, "Mapped memory page without a pointer @ {:016X}", vaddr); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case Common::PageType::RasterizerCachedMemory: { | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  |             u8* const host_ptr{GetPointerFromRasterizerCachedMemory(vaddr)}; | 
					
						
							| 
									
										
										
										
											2020-04-05 12:58:23 -04:00
										 |  |  |             system.GPU().InvalidateRegion(vaddr, sizeof(T)); | 
					
						
							| 
									
										
										
										
											2019-11-26 17:39:57 -05:00
										 |  |  |             std::memcpy(host_ptr, &data, sizeof(T)); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             UNREACHABLE(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 18:34:30 -05:00
										 |  |  |     Common::PageTable* current_page_table = nullptr; | 
					
						
							| 
									
										
										
										
											2019-11-26 13:09:12 -05:00
										 |  |  |     Core::System& system; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2014-04-26 01:27:25 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 13:09:12 -05:00
										 |  |  | Memory::Memory(Core::System& system) : impl{std::make_unique<Impl>(system)} {} | 
					
						
							|  |  |  | Memory::~Memory() = default; | 
					
						
							| 
									
										
										
										
											2016-01-30 18:41:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 18:34:30 -05:00
										 |  |  | void Memory::SetCurrentPageTable(Kernel::Process& process) { | 
					
						
							|  |  |  |     impl->SetCurrentPageTable(process); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 22:50:46 -04:00
										 |  |  | void Memory::MapMemoryRegion(Common::PageTable& page_table, VAddr base, u64 size, PAddr target) { | 
					
						
							| 
									
										
										
										
											2019-11-26 13:09:12 -05:00
										 |  |  |     impl->MapMemoryRegion(page_table, base, size, target); | 
					
						
							| 
									
										
										
										
											2015-05-12 23:38:56 -03:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2014-12-29 19:35:06 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 13:09:12 -05:00
										 |  |  | void Memory::MapIoRegion(Common::PageTable& page_table, VAddr base, u64 size, | 
					
						
							|  |  |  |                          Common::MemoryHookPointer mmio_handler) { | 
					
						
							|  |  |  |     impl->MapIoRegion(page_table, base, size, std::move(mmio_handler)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Memory::UnmapRegion(Common::PageTable& page_table, VAddr base, u64 size) { | 
					
						
							|  |  |  |     impl->UnmapRegion(page_table, base, size); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-04-16 18:57:57 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 13:09:12 -05:00
										 |  |  | void Memory::AddDebugHook(Common::PageTable& page_table, VAddr base, u64 size, | 
					
						
							|  |  |  |                           Common::MemoryHookPointer hook) { | 
					
						
							|  |  |  |     impl->AddDebugHook(page_table, base, size, std::move(hook)); | 
					
						
							| 
									
										
										
										
											2018-01-27 15:16:39 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-04-16 18:57:57 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 13:09:12 -05:00
										 |  |  | void Memory::RemoveDebugHook(Common::PageTable& page_table, VAddr base, u64 size, | 
					
						
							|  |  |  |                              Common::MemoryHookPointer hook) { | 
					
						
							|  |  |  |     impl->RemoveDebugHook(page_table, base, size, std::move(hook)); | 
					
						
							| 
									
										
										
										
											2016-04-16 18:57:57 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 13:46:41 -05:00
										 |  |  | bool Memory::IsValidVirtualAddress(const Kernel::Process& process, const VAddr vaddr) const { | 
					
						
							|  |  |  |     return impl->IsValidVirtualAddress(process, vaddr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Memory::IsValidVirtualAddress(const VAddr vaddr) const { | 
					
						
							|  |  |  |     return impl->IsValidVirtualAddress(vaddr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 15:19:15 -05:00
										 |  |  | u8* Memory::GetPointer(VAddr vaddr) { | 
					
						
							|  |  |  |     return impl->GetPointer(vaddr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const u8* Memory::GetPointer(VAddr vaddr) const { | 
					
						
							|  |  |  |     return impl->GetPointer(vaddr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 16:29:34 -05:00
										 |  |  | u8 Memory::Read8(const VAddr addr) { | 
					
						
							|  |  |  |     return impl->Read8(addr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | u16 Memory::Read16(const VAddr addr) { | 
					
						
							|  |  |  |     return impl->Read16(addr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | u32 Memory::Read32(const VAddr addr) { | 
					
						
							|  |  |  |     return impl->Read32(addr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | u64 Memory::Read64(const VAddr addr) { | 
					
						
							|  |  |  |     return impl->Read64(addr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 17:39:57 -05:00
										 |  |  | void Memory::Write8(VAddr addr, u8 data) { | 
					
						
							|  |  |  |     impl->Write8(addr, data); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Memory::Write16(VAddr addr, u16 data) { | 
					
						
							|  |  |  |     impl->Write16(addr, data); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Memory::Write32(VAddr addr, u32 data) { | 
					
						
							|  |  |  |     impl->Write32(addr, data); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Memory::Write64(VAddr addr, u64 data) { | 
					
						
							|  |  |  |     impl->Write64(addr, data); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 15:48:19 -05:00
										 |  |  | std::string Memory::ReadCString(VAddr vaddr, std::size_t max_length) { | 
					
						
							|  |  |  |     return impl->ReadCString(vaddr, max_length); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 16:29:34 -05:00
										 |  |  | void Memory::ReadBlock(const Kernel::Process& process, const VAddr src_addr, void* dest_buffer, | 
					
						
							|  |  |  |                        const std::size_t size) { | 
					
						
							|  |  |  |     impl->ReadBlock(process, src_addr, dest_buffer, size); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Memory::ReadBlock(const VAddr src_addr, void* dest_buffer, const std::size_t size) { | 
					
						
							|  |  |  |     impl->ReadBlock(src_addr, dest_buffer, size); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-05 17:23:49 -04:00
										 |  |  | void Memory::ReadBlockUnsafe(const Kernel::Process& process, const VAddr src_addr, | 
					
						
							|  |  |  |                              void* dest_buffer, const std::size_t size) { | 
					
						
							|  |  |  |     impl->ReadBlockUnsafe(process, src_addr, dest_buffer, size); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Memory::ReadBlockUnsafe(const VAddr src_addr, void* dest_buffer, const std::size_t size) { | 
					
						
							|  |  |  |     impl->ReadBlockUnsafe(src_addr, dest_buffer, size); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 17:39:57 -05:00
										 |  |  | void Memory::WriteBlock(const Kernel::Process& process, VAddr dest_addr, const void* src_buffer, | 
					
						
							|  |  |  |                         std::size_t size) { | 
					
						
							|  |  |  |     impl->WriteBlock(process, dest_addr, src_buffer, size); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Memory::WriteBlock(const VAddr dest_addr, const void* src_buffer, const std::size_t size) { | 
					
						
							|  |  |  |     impl->WriteBlock(dest_addr, src_buffer, size); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-05 17:23:49 -04:00
										 |  |  | void Memory::WriteBlockUnsafe(const Kernel::Process& process, VAddr dest_addr, | 
					
						
							|  |  |  |                               const void* src_buffer, std::size_t size) { | 
					
						
							|  |  |  |     impl->WriteBlockUnsafe(process, dest_addr, src_buffer, size); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Memory::WriteBlockUnsafe(const VAddr dest_addr, const void* src_buffer, | 
					
						
							|  |  |  |                               const std::size_t size) { | 
					
						
							|  |  |  |     impl->WriteBlockUnsafe(dest_addr, src_buffer, size); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 16:06:49 -05:00
										 |  |  | void Memory::ZeroBlock(const Kernel::Process& process, VAddr dest_addr, std::size_t size) { | 
					
						
							|  |  |  |     impl->ZeroBlock(process, dest_addr, size); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Memory::ZeroBlock(VAddr dest_addr, std::size_t size) { | 
					
						
							|  |  |  |     impl->ZeroBlock(dest_addr, size); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Memory::CopyBlock(const Kernel::Process& process, VAddr dest_addr, VAddr src_addr, | 
					
						
							|  |  |  |                        const std::size_t size) { | 
					
						
							|  |  |  |     impl->CopyBlock(process, dest_addr, src_addr, size); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Memory::CopyBlock(VAddr dest_addr, VAddr src_addr, std::size_t size) { | 
					
						
							|  |  |  |     impl->CopyBlock(dest_addr, src_addr, size); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 15:56:13 -05:00
										 |  |  | void Memory::RasterizerMarkRegionCached(VAddr vaddr, u64 size, bool cached) { | 
					
						
							|  |  |  |     impl->RasterizerMarkRegionCached(vaddr, size, cached); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-22 00:47:59 -06:00
										 |  |  | bool IsKernelVirtualAddress(const VAddr vaddr) { | 
					
						
							|  |  |  |     return KERNEL_REGION_VADDR <= vaddr && vaddr < KERNEL_REGION_END; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-31 15:10:44 -04:00
										 |  |  | } // namespace Core::Memory
 |