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 | } // 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; | 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); |     R_UNLESS(this->CanContain(addr, size, state), ResultInvalidCurrentMemory); | ||||||
|  |  | ||||||
|     // Lock the table. |     // Lock the table. | ||||||
|     std::lock_guard lock{page_table_lock}; |     KScopedLightLock lk(general_lock); | ||||||
|  |  | ||||||
|     // Verify that the destination memory is unmapped. |     // Verify that the destination memory is unmapped. | ||||||
|     R_TRY(this->CheckMemoryState(addr, size, KMemoryState::All, KMemoryState::Free, |     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) { | 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}; |     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) { | 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) { |     if (!size) { | ||||||
|         return ResultSuccess; |         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, | ResultCode KPageTable::UnmapProcessMemory(VAddr dst_addr, std::size_t size, | ||||||
|                                           KPageTable& src_page_table, VAddr src_addr) { |                                           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}; |     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) { | ResultCode KPageTable::MapPhysicalMemory(VAddr addr, std::size_t size) { | ||||||
|     // Lock the physical memory lock. |     // 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. |     // Lock the table. | ||||||
|     std::lock_guard lock{page_table_lock}; |     KScopedLightLock lk(general_lock); | ||||||
|  |  | ||||||
|     std::size_t mapped_size{}; |     std::size_t mapped_size{}; | ||||||
|     const VAddr end_addr{addr + 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) { | 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}; |     const VAddr end_addr{addr + size}; | ||||||
|     ResultCode result{ResultSuccess}; |     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) { | 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{}; |     KMemoryState src_state{}; | ||||||
|     CASCADE_CODE(CheckMemoryState( |     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) { | 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{}; |     KMemoryState src_state{}; | ||||||
|     CASCADE_CODE(CheckMemoryState( |     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, | ResultCode KPageTable::MapPages(VAddr addr, const KPageLinkedList& page_linked_list, | ||||||
|                                 KMemoryPermission perm) { |                                 KMemoryPermission perm) { | ||||||
|  |     ASSERT(this->IsLockedByCurrentThread()); | ||||||
|  |  | ||||||
|     VAddr cur_addr{addr}; |     VAddr cur_addr{addr}; | ||||||
|  |  | ||||||
|     for (const auto& node : page_linked_list.Nodes()) { |     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); |     R_UNLESS(this->CanContain(address, size, state), ResultInvalidCurrentMemory); | ||||||
|  |  | ||||||
|     // Lock the table. |     // Lock the table. | ||||||
|     std::lock_guard lock{page_table_lock}; |     KScopedLightLock lk(general_lock); | ||||||
|  |  | ||||||
|     // Check the memory state. |     // Check the memory state. | ||||||
|     R_TRY(this->CheckMemoryState(address, size, KMemoryState::All, KMemoryState::Free, |     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) { | ResultCode KPageTable::UnmapPages(VAddr addr, const KPageLinkedList& page_linked_list) { | ||||||
|  |     ASSERT(this->IsLockedByCurrentThread()); | ||||||
|  |  | ||||||
|     VAddr cur_addr{addr}; |     VAddr cur_addr{addr}; | ||||||
|  |  | ||||||
|     for (const auto& node : page_linked_list.Nodes()) { |     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); |     R_UNLESS(this->Contains(addr, size), ResultInvalidCurrentMemory); | ||||||
|  |  | ||||||
|     // Lock the table. |     // Lock the table. | ||||||
|     std::lock_guard lock{page_table_lock}; |     KScopedLightLock lk(general_lock); | ||||||
|  |  | ||||||
|     // Check the memory state. |     // Check the memory state. | ||||||
|     R_TRY(this->CheckMemoryState(addr, size, KMemoryState::All, state, KMemoryPermission::None, |     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; |     const size_t num_pages = size / PageSize; | ||||||
|  |  | ||||||
|     // Lock the table. |     // Lock the table. | ||||||
|     std::lock_guard lock{page_table_lock}; |     KScopedLightLock lk(general_lock); | ||||||
|  |  | ||||||
|     // Verify we can change the memory permission. |     // Verify we can change the memory permission. | ||||||
|     KMemoryState old_state; |     KMemoryState old_state; | ||||||
| @@ -768,7 +777,7 @@ ResultCode KPageTable::SetProcessMemoryPermission(VAddr addr, std::size_t size, | |||||||
| } | } | ||||||
|  |  | ||||||
| KMemoryInfo KPageTable::QueryInfoImpl(VAddr addr) { | KMemoryInfo KPageTable::QueryInfoImpl(VAddr addr) { | ||||||
|     std::lock_guard lock{page_table_lock}; |     KScopedLightLock lk(general_lock); | ||||||
|  |  | ||||||
|     return block_manager->FindBlock(addr).GetMemoryInfo(); |     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) { | ResultCode KPageTable::ReserveTransferMemory(VAddr addr, std::size_t size, KMemoryPermission perm) { | ||||||
|     std::lock_guard lock{page_table_lock}; |     KScopedLightLock lk(general_lock); | ||||||
|  |  | ||||||
|     KMemoryState state{}; |     KMemoryState state{}; | ||||||
|     KMemoryAttribute attribute{}; |     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) { | ResultCode KPageTable::ResetTransferMemory(VAddr addr, std::size_t size) { | ||||||
|     std::lock_guard lock{page_table_lock}; |     KScopedLightLock lk(general_lock); | ||||||
|  |  | ||||||
|     KMemoryState state{}; |     KMemoryState state{}; | ||||||
|  |  | ||||||
| @@ -820,7 +829,7 @@ ResultCode KPageTable::SetMemoryPermission(VAddr addr, std::size_t size, | |||||||
|     const size_t num_pages = size / PageSize; |     const size_t num_pages = size / PageSize; | ||||||
|  |  | ||||||
|     // Lock the table. |     // Lock the table. | ||||||
|     std::lock_guard lock{page_table_lock}; |     KScopedLightLock lk(general_lock); | ||||||
|  |  | ||||||
|     // Verify we can change the memory permission. |     // Verify we can change the memory permission. | ||||||
|     KMemoryState old_state; |     KMemoryState old_state; | ||||||
| @@ -849,7 +858,7 @@ ResultCode KPageTable::SetMemoryAttribute(VAddr addr, std::size_t size, u32 mask | |||||||
|            KMemoryAttribute::SetMask); |            KMemoryAttribute::SetMask); | ||||||
|  |  | ||||||
|     // Lock the table. |     // Lock the table. | ||||||
|     std::lock_guard lock{page_table_lock}; |     KScopedLightLock lk(general_lock); | ||||||
|  |  | ||||||
|     // Verify we can change the memory attribute. |     // Verify we can change the memory attribute. | ||||||
|     KMemoryState old_state; |     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) { | ResultCode KPageTable::SetMaxHeapSize(std::size_t size) { | ||||||
|     // Lock the table. |     // Lock the table. | ||||||
|     std::lock_guard lock{page_table_lock}; |     KScopedLightLock lk(general_lock); | ||||||
|  |  | ||||||
|     // Only process page tables are allowed to set heap size. |     // Only process page tables are allowed to set heap size. | ||||||
|     ASSERT(!this->IsKernel()); |     ASSERT(!this->IsKernel()); | ||||||
| @@ -891,15 +900,15 @@ ResultCode KPageTable::SetMaxHeapSize(std::size_t size) { | |||||||
| } | } | ||||||
|  |  | ||||||
| ResultCode KPageTable::SetHeapSize(VAddr* out, std::size_t size) { | ResultCode KPageTable::SetHeapSize(VAddr* out, std::size_t size) { | ||||||
|     // Lock the physical memory lock. |     // Lock the physical memory mutex. | ||||||
|     std::lock_guard phys_lk(map_physical_memory_lock); |     KScopedLightLock map_phys_mem_lk(map_physical_memory_lock); | ||||||
|  |  | ||||||
|     // Try to perform a reduction in heap, instead of an extension. |     // Try to perform a reduction in heap, instead of an extension. | ||||||
|     VAddr cur_address{}; |     VAddr cur_address{}; | ||||||
|     std::size_t allocation_size{}; |     std::size_t allocation_size{}; | ||||||
|     { |     { | ||||||
|         // Lock the table. |         // Lock the table. | ||||||
|         std::lock_guard lk(page_table_lock); |         KScopedLightLock lk(general_lock); | ||||||
|  |  | ||||||
|         // Validate that setting heap size is possible at all. |         // Validate that setting heap size is possible at all. | ||||||
|         R_UNLESS(!is_kernel, ResultOutOfMemory); |         R_UNLESS(!is_kernel, ResultOutOfMemory); | ||||||
| @@ -964,7 +973,7 @@ ResultCode KPageTable::SetHeapSize(VAddr* out, std::size_t size) { | |||||||
|     // Map the pages. |     // Map the pages. | ||||||
|     { |     { | ||||||
|         // Lock the table. |         // 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. |         // Ensure that the heap hasn't changed since we began executing. | ||||||
|         ASSERT(cur_address == current_heap_end); |         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, |                                                   bool is_map_only, VAddr region_start, | ||||||
|                                                   std::size_t region_num_pages, KMemoryState state, |                                                   std::size_t region_num_pages, KMemoryState state, | ||||||
|                                                   KMemoryPermission perm, PAddr map_addr) { |                                                   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)) { |     if (!CanContain(region_start, region_num_pages * PageSize, state)) { | ||||||
|         return ResultInvalidCurrentMemory; |         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) { | ResultCode KPageTable::LockForDeviceAddressSpace(VAddr addr, std::size_t size) { | ||||||
|     std::lock_guard lock{page_table_lock}; |     KScopedLightLock lk(general_lock); | ||||||
|  |  | ||||||
|     KMemoryPermission perm{}; |     KMemoryPermission perm{}; | ||||||
|     if (const ResultCode result{CheckMemoryState( |     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) { | ResultCode KPageTable::UnlockForDeviceAddressSpace(VAddr addr, std::size_t size) { | ||||||
|     std::lock_guard lock{page_table_lock}; |     KScopedLightLock lk(general_lock); | ||||||
|  |  | ||||||
|     KMemoryPermission perm{}; |     KMemoryPermission perm{}; | ||||||
|     if (const ResultCode result{CheckMemoryState( |     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) { | 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; |     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) { | 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; |     KMemoryPermission new_perm = KMemoryPermission::UserReadWrite; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -5,12 +5,12 @@ | |||||||
| #pragma once | #pragma once | ||||||
|  |  | ||||||
| #include <memory> | #include <memory> | ||||||
| #include <mutex> |  | ||||||
|  |  | ||||||
| #include "common/common_funcs.h" | #include "common/common_funcs.h" | ||||||
| #include "common/common_types.h" | #include "common/common_types.h" | ||||||
| #include "common/page_table.h" | #include "common/page_table.h" | ||||||
| #include "core/file_sys/program_metadata.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_block.h" | ||||||
| #include "core/hle/kernel/k_memory_manager.h" | #include "core/hle/kernel/k_memory_manager.h" | ||||||
| #include "core/hle/result.h" | #include "core/hle/result.h" | ||||||
| @@ -147,11 +147,12 @@ private: | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     bool IsLockedByCurrentThread() const { |     bool IsLockedByCurrentThread() const { | ||||||
|         return true; |         return general_lock.IsLockedByCurrentThread(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     std::recursive_mutex page_table_lock; |     mutable KLightLock general_lock; | ||||||
|     std::mutex map_physical_memory_lock; |     mutable KLightLock map_physical_memory_lock; | ||||||
|  |  | ||||||
|     std::unique_ptr<KMemoryBlockManager> block_manager; |     std::unique_ptr<KMemoryBlockManager> block_manager; | ||||||
|  |  | ||||||
| public: | public: | ||||||
| @@ -210,7 +211,7 @@ public: | |||||||
|         return alias_code_region_end - alias_code_region_start; |         return alias_code_region_end - alias_code_region_start; | ||||||
|     } |     } | ||||||
|     size_t GetNormalMemorySize() { |     size_t GetNormalMemorySize() { | ||||||
|         std::lock_guard lk(page_table_lock); |         KScopedLightLock lk(general_lock); | ||||||
|         return GetHeapSize() + mapped_physical_memory_size; |         return GetHeapSize() + mapped_physical_memory_size; | ||||||
|     } |     } | ||||||
|     constexpr std::size_t GetAddressSpaceWidth() const { |     constexpr std::size_t GetAddressSpaceWidth() const { | ||||||
| @@ -252,7 +253,9 @@ public: | |||||||
|     constexpr bool IsInsideASLRRegion(VAddr address, std::size_t size) const { |     constexpr bool IsInsideASLRRegion(VAddr address, std::size_t size) const { | ||||||
|         return !IsOutsideASLRRegion(address, size); |         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]; |         const auto backing_addr = page_table_impl.backing_addr[addr >> PageBits]; | ||||||
|         ASSERT(backing_addr); |         ASSERT(backing_addr); | ||||||
|         return backing_addr + addr; |         return backing_addr + addr; | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 bunnei
					bunnei