forked from eden-emu/eden
		
	core: hle: kernel: Add KPageTableSlabHeap.
This commit is contained in:
		
							parent
							
								
									019e425377
								
							
						
					
					
						commit
						315292275e
					
				
					 2 changed files with 94 additions and 0 deletions
				
			
		|  | @ -224,6 +224,7 @@ add_library(core STATIC | ||||||
|     hle/kernel/k_page_group.h |     hle/kernel/k_page_group.h | ||||||
|     hle/kernel/k_page_table.cpp |     hle/kernel/k_page_table.cpp | ||||||
|     hle/kernel/k_page_table.h |     hle/kernel/k_page_table.h | ||||||
|  |     hle/kernel/k_page_table_slab_heap.h | ||||||
|     hle/kernel/k_port.cpp |     hle/kernel/k_port.cpp | ||||||
|     hle/kernel/k_port.h |     hle/kernel/k_port.h | ||||||
|     hle/kernel/k_priority_queue.h |     hle/kernel/k_priority_queue.h | ||||||
|  |  | ||||||
							
								
								
									
										93
									
								
								src/core/hle/kernel/k_page_table_slab_heap.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										93
									
								
								src/core/hle/kernel/k_page_table_slab_heap.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,93 @@ | ||||||
|  | // SPDX-FileCopyrightText: Copyright 2022 yuzu Emulator Project
 | ||||||
|  | // SPDX-License-Identifier: GPL-2.0-or-later
 | ||||||
|  | 
 | ||||||
|  | #pragma once | ||||||
|  | 
 | ||||||
|  | #include <array> | ||||||
|  | #include <vector> | ||||||
|  | 
 | ||||||
|  | #include "common/common_types.h" | ||||||
|  | #include "core/hle/kernel/k_dynamic_slab_heap.h" | ||||||
|  | #include "core/hle/kernel/slab_helpers.h" | ||||||
|  | 
 | ||||||
|  | namespace Kernel { | ||||||
|  | 
 | ||||||
|  | namespace impl { | ||||||
|  | 
 | ||||||
|  | class PageTablePage { | ||||||
|  | public: | ||||||
|  |     // Do not initialize anything.
 | ||||||
|  |     PageTablePage() = default; | ||||||
|  | 
 | ||||||
|  | private: | ||||||
|  |     std::array<u8, PageSize> m_buffer{}; | ||||||
|  | }; | ||||||
|  | static_assert(sizeof(PageTablePage) == PageSize); | ||||||
|  | 
 | ||||||
|  | } // namespace impl
 | ||||||
|  | 
 | ||||||
|  | class KPageTableSlabHeap : public KDynamicSlabHeap<impl::PageTablePage, true> { | ||||||
|  | public: | ||||||
|  |     using RefCount = u16; | ||||||
|  |     static constexpr size_t PageTableSize = sizeof(impl::PageTablePage); | ||||||
|  |     static_assert(PageTableSize == PageSize); | ||||||
|  | 
 | ||||||
|  | public: | ||||||
|  |     KPageTableSlabHeap() = default; | ||||||
|  | 
 | ||||||
|  |     static constexpr size_t CalculateReferenceCountSize(size_t size) { | ||||||
|  |         return (size / PageSize) * sizeof(RefCount); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void Initialize(KDynamicPageManager* page_allocator, size_t object_count, RefCount* rc) { | ||||||
|  |         BaseHeap::Initialize(page_allocator, object_count); | ||||||
|  |         this->Initialize(rc); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     RefCount GetRefCount(VAddr addr) { | ||||||
|  |         ASSERT(this->IsInRange(addr)); | ||||||
|  |         return *this->GetRefCountPointer(addr); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void Open(VAddr addr, int count) { | ||||||
|  |         ASSERT(this->IsInRange(addr)); | ||||||
|  | 
 | ||||||
|  |         *this->GetRefCountPointer(addr) += static_cast<RefCount>(count); | ||||||
|  | 
 | ||||||
|  |         ASSERT(this->GetRefCount(addr) > 0); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     bool Close(VAddr addr, int count) { | ||||||
|  |         ASSERT(this->IsInRange(addr)); | ||||||
|  |         ASSERT(this->GetRefCount(addr) >= count); | ||||||
|  | 
 | ||||||
|  |         *this->GetRefCountPointer(addr) -= static_cast<RefCount>(count); | ||||||
|  |         return this->GetRefCount(addr) == 0; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     bool IsInPageTableHeap(VAddr addr) const { | ||||||
|  |         return this->IsInRange(addr); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | private: | ||||||
|  |     void Initialize([[maybe_unused]] RefCount* rc) { | ||||||
|  |         // TODO(bunnei): Use rc once we support kernel virtual memory allocations.
 | ||||||
|  |         const auto count = this->GetSize() / PageSize; | ||||||
|  |         m_ref_counts.resize(count); | ||||||
|  | 
 | ||||||
|  |         for (size_t i = 0; i < count; i++) { | ||||||
|  |             m_ref_counts[i] = 0; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     RefCount* GetRefCountPointer(VAddr addr) { | ||||||
|  |         return m_ref_counts.data() + ((addr - this->GetAddress()) / PageSize); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | private: | ||||||
|  |     using BaseHeap = KDynamicSlabHeap<impl::PageTablePage, true>; | ||||||
|  | 
 | ||||||
|  |     std::vector<RefCount> m_ref_counts; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | } // namespace Kernel
 | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 bunnei
						bunnei