Merge pull request #7835 from bunnei/page-table-lock
hle: kernel: KPageTable: Migrate locks to KScopedLightLock.
This commit is contained in:
		| @@ -61,7 +61,8 @@ constexpr std::size_t GetSizeInRange(const KMemoryInfo& info, VAddr start, VAddr | ||||
|  | ||||
| } // namespace | ||||
|  | ||||
| KPageTable::KPageTable(Core::System& system_) : system{system_} {} | ||||
| KPageTable::KPageTable(Core::System& system_) | ||||
|     : general_lock{system_.Kernel()}, map_physical_memory_lock{system_.Kernel()}, system{system_} {} | ||||
|  | ||||
| KPageTable::~KPageTable() = default; | ||||
|  | ||||
| @@ -284,7 +285,7 @@ ResultCode KPageTable::MapProcessCode(VAddr addr, std::size_t num_pages, KMemory | ||||
|     R_UNLESS(this->CanContain(addr, size, state), ResultInvalidCurrentMemory); | ||||
|  | ||||
|     // Lock the table. | ||||
|     std::lock_guard lock{page_table_lock}; | ||||
|     KScopedLightLock lk(general_lock); | ||||
|  | ||||
|     // Verify that the destination memory is unmapped. | ||||
|     R_TRY(this->CheckMemoryState(addr, size, KMemoryState::All, KMemoryState::Free, | ||||
| @@ -302,7 +303,7 @@ ResultCode KPageTable::MapProcessCode(VAddr addr, std::size_t num_pages, KMemory | ||||
| } | ||||
|  | ||||
| ResultCode KPageTable::MapCodeMemory(VAddr dst_addr, VAddr src_addr, std::size_t size) { | ||||
|     std::lock_guard lock{page_table_lock}; | ||||
|     KScopedLightLock lk(general_lock); | ||||
|  | ||||
|     const std::size_t num_pages{size / PageSize}; | ||||
|  | ||||
| @@ -339,7 +340,7 @@ ResultCode KPageTable::MapCodeMemory(VAddr dst_addr, VAddr src_addr, std::size_t | ||||
| } | ||||
|  | ||||
| ResultCode KPageTable::UnmapCodeMemory(VAddr dst_addr, VAddr src_addr, std::size_t size) { | ||||
|     std::lock_guard lock{page_table_lock}; | ||||
|     KScopedLightLock lk(general_lock); | ||||
|  | ||||
|     if (!size) { | ||||
|         return ResultSuccess; | ||||
| @@ -373,7 +374,7 @@ ResultCode KPageTable::UnmapCodeMemory(VAddr dst_addr, VAddr src_addr, std::size | ||||
|  | ||||
| ResultCode KPageTable::UnmapProcessMemory(VAddr dst_addr, std::size_t size, | ||||
|                                           KPageTable& src_page_table, VAddr src_addr) { | ||||
|     std::lock_guard lock{page_table_lock}; | ||||
|     KScopedLightLock lk(general_lock); | ||||
|  | ||||
|     const std::size_t num_pages{size / PageSize}; | ||||
|  | ||||
| @@ -401,10 +402,10 @@ ResultCode KPageTable::UnmapProcessMemory(VAddr dst_addr, std::size_t size, | ||||
|  | ||||
| ResultCode KPageTable::MapPhysicalMemory(VAddr addr, std::size_t size) { | ||||
|     // Lock the physical memory lock. | ||||
|     std::lock_guard phys_lk(map_physical_memory_lock); | ||||
|     KScopedLightLock map_phys_mem_lk(map_physical_memory_lock); | ||||
|  | ||||
|     // Lock the table. | ||||
|     std::lock_guard lock{page_table_lock}; | ||||
|     KScopedLightLock lk(general_lock); | ||||
|  | ||||
|     std::size_t mapped_size{}; | ||||
|     const VAddr end_addr{addr + size}; | ||||
| @@ -480,7 +481,11 @@ ResultCode KPageTable::MapPhysicalMemory(VAddr addr, std::size_t size) { | ||||
| } | ||||
|  | ||||
| ResultCode KPageTable::UnmapPhysicalMemory(VAddr addr, std::size_t size) { | ||||
|     std::lock_guard lock{page_table_lock}; | ||||
|     // Lock the physical memory lock. | ||||
|     KScopedLightLock map_phys_mem_lk(map_physical_memory_lock); | ||||
|  | ||||
|     // Lock the table. | ||||
|     KScopedLightLock lk(general_lock); | ||||
|  | ||||
|     const VAddr end_addr{addr + size}; | ||||
|     ResultCode result{ResultSuccess}; | ||||
| @@ -542,7 +547,7 @@ ResultCode KPageTable::UnmapPhysicalMemory(VAddr addr, std::size_t size) { | ||||
| } | ||||
|  | ||||
| ResultCode KPageTable::MapMemory(VAddr dst_addr, VAddr src_addr, std::size_t size) { | ||||
|     std::lock_guard lock{page_table_lock}; | ||||
|     KScopedLightLock lk(general_lock); | ||||
|  | ||||
|     KMemoryState src_state{}; | ||||
|     CASCADE_CODE(CheckMemoryState( | ||||
| @@ -581,7 +586,7 @@ ResultCode KPageTable::MapMemory(VAddr dst_addr, VAddr src_addr, std::size_t siz | ||||
| } | ||||
|  | ||||
| ResultCode KPageTable::UnmapMemory(VAddr dst_addr, VAddr src_addr, std::size_t size) { | ||||
|     std::lock_guard lock{page_table_lock}; | ||||
|     KScopedLightLock lk(general_lock); | ||||
|  | ||||
|     KMemoryState src_state{}; | ||||
|     CASCADE_CODE(CheckMemoryState( | ||||
| @@ -624,6 +629,8 @@ ResultCode KPageTable::UnmapMemory(VAddr dst_addr, VAddr src_addr, std::size_t s | ||||
|  | ||||
| ResultCode KPageTable::MapPages(VAddr addr, const KPageLinkedList& page_linked_list, | ||||
|                                 KMemoryPermission perm) { | ||||
|     ASSERT(this->IsLockedByCurrentThread()); | ||||
|  | ||||
|     VAddr cur_addr{addr}; | ||||
|  | ||||
|     for (const auto& node : page_linked_list.Nodes()) { | ||||
| @@ -652,7 +659,7 @@ ResultCode KPageTable::MapPages(VAddr address, KPageLinkedList& page_linked_list | ||||
|     R_UNLESS(this->CanContain(address, size, state), ResultInvalidCurrentMemory); | ||||
|  | ||||
|     // Lock the table. | ||||
|     std::lock_guard lock{page_table_lock}; | ||||
|     KScopedLightLock lk(general_lock); | ||||
|  | ||||
|     // Check the memory state. | ||||
|     R_TRY(this->CheckMemoryState(address, size, KMemoryState::All, KMemoryState::Free, | ||||
| @@ -669,6 +676,8 @@ ResultCode KPageTable::MapPages(VAddr address, KPageLinkedList& page_linked_list | ||||
| } | ||||
|  | ||||
| ResultCode KPageTable::UnmapPages(VAddr addr, const KPageLinkedList& page_linked_list) { | ||||
|     ASSERT(this->IsLockedByCurrentThread()); | ||||
|  | ||||
|     VAddr cur_addr{addr}; | ||||
|  | ||||
|     for (const auto& node : page_linked_list.Nodes()) { | ||||
| @@ -693,7 +702,7 @@ ResultCode KPageTable::UnmapPages(VAddr addr, KPageLinkedList& page_linked_list, | ||||
|     R_UNLESS(this->Contains(addr, size), ResultInvalidCurrentMemory); | ||||
|  | ||||
|     // Lock the table. | ||||
|     std::lock_guard lock{page_table_lock}; | ||||
|     KScopedLightLock lk(general_lock); | ||||
|  | ||||
|     // Check the memory state. | ||||
|     R_TRY(this->CheckMemoryState(addr, size, KMemoryState::All, state, KMemoryPermission::None, | ||||
| @@ -714,7 +723,7 @@ ResultCode KPageTable::SetProcessMemoryPermission(VAddr addr, std::size_t size, | ||||
|     const size_t num_pages = size / PageSize; | ||||
|  | ||||
|     // Lock the table. | ||||
|     std::lock_guard lock{page_table_lock}; | ||||
|     KScopedLightLock lk(general_lock); | ||||
|  | ||||
|     // Verify we can change the memory permission. | ||||
|     KMemoryState old_state; | ||||
| @@ -768,7 +777,7 @@ ResultCode KPageTable::SetProcessMemoryPermission(VAddr addr, std::size_t size, | ||||
| } | ||||
|  | ||||
| KMemoryInfo KPageTable::QueryInfoImpl(VAddr addr) { | ||||
|     std::lock_guard lock{page_table_lock}; | ||||
|     KScopedLightLock lk(general_lock); | ||||
|  | ||||
|     return block_manager->FindBlock(addr).GetMemoryInfo(); | ||||
| } | ||||
| @@ -783,7 +792,7 @@ KMemoryInfo KPageTable::QueryInfo(VAddr addr) { | ||||
| } | ||||
|  | ||||
| ResultCode KPageTable::ReserveTransferMemory(VAddr addr, std::size_t size, KMemoryPermission perm) { | ||||
|     std::lock_guard lock{page_table_lock}; | ||||
|     KScopedLightLock lk(general_lock); | ||||
|  | ||||
|     KMemoryState state{}; | ||||
|     KMemoryAttribute attribute{}; | ||||
| @@ -801,7 +810,7 @@ ResultCode KPageTable::ReserveTransferMemory(VAddr addr, std::size_t size, KMemo | ||||
| } | ||||
|  | ||||
| ResultCode KPageTable::ResetTransferMemory(VAddr addr, std::size_t size) { | ||||
|     std::lock_guard lock{page_table_lock}; | ||||
|     KScopedLightLock lk(general_lock); | ||||
|  | ||||
|     KMemoryState state{}; | ||||
|  | ||||
| @@ -820,7 +829,7 @@ ResultCode KPageTable::SetMemoryPermission(VAddr addr, std::size_t size, | ||||
|     const size_t num_pages = size / PageSize; | ||||
|  | ||||
|     // Lock the table. | ||||
|     std::lock_guard lock{page_table_lock}; | ||||
|     KScopedLightLock lk(general_lock); | ||||
|  | ||||
|     // Verify we can change the memory permission. | ||||
|     KMemoryState old_state; | ||||
| @@ -849,7 +858,7 @@ ResultCode KPageTable::SetMemoryAttribute(VAddr addr, std::size_t size, u32 mask | ||||
|            KMemoryAttribute::SetMask); | ||||
|  | ||||
|     // Lock the table. | ||||
|     std::lock_guard lock{page_table_lock}; | ||||
|     KScopedLightLock lk(general_lock); | ||||
|  | ||||
|     // Verify we can change the memory attribute. | ||||
|     KMemoryState old_state; | ||||
| @@ -880,7 +889,7 @@ ResultCode KPageTable::SetMemoryAttribute(VAddr addr, std::size_t size, u32 mask | ||||
|  | ||||
| ResultCode KPageTable::SetMaxHeapSize(std::size_t size) { | ||||
|     // Lock the table. | ||||
|     std::lock_guard lock{page_table_lock}; | ||||
|     KScopedLightLock lk(general_lock); | ||||
|  | ||||
|     // Only process page tables are allowed to set heap size. | ||||
|     ASSERT(!this->IsKernel()); | ||||
| @@ -891,15 +900,15 @@ ResultCode KPageTable::SetMaxHeapSize(std::size_t size) { | ||||
| } | ||||
|  | ||||
| ResultCode KPageTable::SetHeapSize(VAddr* out, std::size_t size) { | ||||
|     // Lock the physical memory lock. | ||||
|     std::lock_guard phys_lk(map_physical_memory_lock); | ||||
|     // Lock the physical memory mutex. | ||||
|     KScopedLightLock map_phys_mem_lk(map_physical_memory_lock); | ||||
|  | ||||
|     // Try to perform a reduction in heap, instead of an extension. | ||||
|     VAddr cur_address{}; | ||||
|     std::size_t allocation_size{}; | ||||
|     { | ||||
|         // Lock the table. | ||||
|         std::lock_guard lk(page_table_lock); | ||||
|         KScopedLightLock lk(general_lock); | ||||
|  | ||||
|         // Validate that setting heap size is possible at all. | ||||
|         R_UNLESS(!is_kernel, ResultOutOfMemory); | ||||
| @@ -964,7 +973,7 @@ ResultCode KPageTable::SetHeapSize(VAddr* out, std::size_t size) { | ||||
|     // Map the pages. | ||||
|     { | ||||
|         // Lock the table. | ||||
|         std::lock_guard lk(page_table_lock); | ||||
|         KScopedLightLock lk(general_lock); | ||||
|  | ||||
|         // Ensure that the heap hasn't changed since we began executing. | ||||
|         ASSERT(cur_address == current_heap_end); | ||||
| @@ -1006,7 +1015,7 @@ ResultVal<VAddr> KPageTable::AllocateAndMapMemory(std::size_t needed_num_pages, | ||||
|                                                   bool is_map_only, VAddr region_start, | ||||
|                                                   std::size_t region_num_pages, KMemoryState state, | ||||
|                                                   KMemoryPermission perm, PAddr map_addr) { | ||||
|     std::lock_guard lock{page_table_lock}; | ||||
|     KScopedLightLock lk(general_lock); | ||||
|  | ||||
|     if (!CanContain(region_start, region_num_pages * PageSize, state)) { | ||||
|         return ResultInvalidCurrentMemory; | ||||
| @@ -1037,7 +1046,7 @@ ResultVal<VAddr> KPageTable::AllocateAndMapMemory(std::size_t needed_num_pages, | ||||
| } | ||||
|  | ||||
| ResultCode KPageTable::LockForDeviceAddressSpace(VAddr addr, std::size_t size) { | ||||
|     std::lock_guard lock{page_table_lock}; | ||||
|     KScopedLightLock lk(general_lock); | ||||
|  | ||||
|     KMemoryPermission perm{}; | ||||
|     if (const ResultCode result{CheckMemoryState( | ||||
| @@ -1060,7 +1069,7 @@ ResultCode KPageTable::LockForDeviceAddressSpace(VAddr addr, std::size_t size) { | ||||
| } | ||||
|  | ||||
| ResultCode KPageTable::UnlockForDeviceAddressSpace(VAddr addr, std::size_t size) { | ||||
|     std::lock_guard lock{page_table_lock}; | ||||
|     KScopedLightLock lk(general_lock); | ||||
|  | ||||
|     KMemoryPermission perm{}; | ||||
|     if (const ResultCode result{CheckMemoryState( | ||||
| @@ -1083,7 +1092,7 @@ ResultCode KPageTable::UnlockForDeviceAddressSpace(VAddr addr, std::size_t size) | ||||
| } | ||||
|  | ||||
| ResultCode KPageTable::LockForCodeMemory(VAddr addr, std::size_t size) { | ||||
|     std::lock_guard lock{page_table_lock}; | ||||
|     KScopedLightLock lk(general_lock); | ||||
|  | ||||
|     KMemoryPermission new_perm = KMemoryPermission::NotMapped | KMemoryPermission::KernelReadWrite; | ||||
|  | ||||
| @@ -1110,7 +1119,7 @@ ResultCode KPageTable::LockForCodeMemory(VAddr addr, std::size_t size) { | ||||
| } | ||||
|  | ||||
| ResultCode KPageTable::UnlockForCodeMemory(VAddr addr, std::size_t size) { | ||||
|     std::lock_guard lock{page_table_lock}; | ||||
|     KScopedLightLock lk(general_lock); | ||||
|  | ||||
|     KMemoryPermission new_perm = KMemoryPermission::UserReadWrite; | ||||
|  | ||||
|   | ||||
| @@ -5,12 +5,12 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include <memory> | ||||
| #include <mutex> | ||||
|  | ||||
| #include "common/common_funcs.h" | ||||
| #include "common/common_types.h" | ||||
| #include "common/page_table.h" | ||||
| #include "core/file_sys/program_metadata.h" | ||||
| #include "core/hle/kernel/k_light_lock.h" | ||||
| #include "core/hle/kernel/k_memory_block.h" | ||||
| #include "core/hle/kernel/k_memory_manager.h" | ||||
| #include "core/hle/result.h" | ||||
| @@ -147,11 +147,12 @@ private: | ||||
|     } | ||||
|  | ||||
|     bool IsLockedByCurrentThread() const { | ||||
|         return true; | ||||
|         return general_lock.IsLockedByCurrentThread(); | ||||
|     } | ||||
|  | ||||
|     std::recursive_mutex page_table_lock; | ||||
|     std::mutex map_physical_memory_lock; | ||||
|     mutable KLightLock general_lock; | ||||
|     mutable KLightLock map_physical_memory_lock; | ||||
|  | ||||
|     std::unique_ptr<KMemoryBlockManager> block_manager; | ||||
|  | ||||
| public: | ||||
| @@ -210,7 +211,7 @@ public: | ||||
|         return alias_code_region_end - alias_code_region_start; | ||||
|     } | ||||
|     size_t GetNormalMemorySize() { | ||||
|         std::lock_guard lk(page_table_lock); | ||||
|         KScopedLightLock lk(general_lock); | ||||
|         return GetHeapSize() + mapped_physical_memory_size; | ||||
|     } | ||||
|     constexpr std::size_t GetAddressSpaceWidth() const { | ||||
| @@ -252,7 +253,9 @@ public: | ||||
|     constexpr bool IsInsideASLRRegion(VAddr address, std::size_t size) const { | ||||
|         return !IsOutsideASLRRegion(address, size); | ||||
|     } | ||||
|     constexpr PAddr GetPhysicalAddr(VAddr addr) { | ||||
|  | ||||
|     PAddr GetPhysicalAddr(VAddr addr) { | ||||
|         ASSERT(IsLockedByCurrentThread()); | ||||
|         const auto backing_addr = page_table_impl.backing_addr[addr >> PageBits]; | ||||
|         ASSERT(backing_addr); | ||||
|         return backing_addr + addr; | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 bunnei
					bunnei