|
|
@ -10,6 +10,7 @@ |
|
|
#include "common/assert.h"
|
|
|
#include "common/assert.h"
|
|
|
#include "common/common_types.h"
|
|
|
#include "common/common_types.h"
|
|
|
#include "common/logging/log.h"
|
|
|
#include "common/logging/log.h"
|
|
|
|
|
|
#include "common/page_table.h"
|
|
|
#include "common/swap.h"
|
|
|
#include "common/swap.h"
|
|
|
#include "core/arm/arm_interface.h"
|
|
|
#include "core/arm/arm_interface.h"
|
|
|
#include "core/core.h"
|
|
|
#include "core/core.h"
|
|
|
@ -23,9 +24,9 @@ |
|
|
|
|
|
|
|
|
namespace Memory { |
|
|
namespace Memory { |
|
|
|
|
|
|
|
|
static PageTable* current_page_table = nullptr; |
|
|
|
|
|
|
|
|
static Common::PageTable* current_page_table = nullptr; |
|
|
|
|
|
|
|
|
void SetCurrentPageTable(PageTable* page_table) { |
|
|
|
|
|
|
|
|
void SetCurrentPageTable(Common::PageTable* page_table) { |
|
|
current_page_table = page_table; |
|
|
current_page_table = page_table; |
|
|
|
|
|
|
|
|
auto& system = Core::System::GetInstance(); |
|
|
auto& system = Core::System::GetInstance(); |
|
|
@ -37,34 +38,12 @@ void SetCurrentPageTable(PageTable* page_table) { |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
PageTable* GetCurrentPageTable() { |
|
|
|
|
|
|
|
|
Common::PageTable* GetCurrentPageTable() { |
|
|
return current_page_table; |
|
|
return current_page_table; |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
PageTable::PageTable() = default; |
|
|
|
|
|
|
|
|
|
|
|
PageTable::PageTable(std::size_t address_space_width_in_bits) { |
|
|
|
|
|
Resize(address_space_width_in_bits); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
PageTable::~PageTable() = default; |
|
|
|
|
|
|
|
|
|
|
|
void PageTable::Resize(std::size_t address_space_width_in_bits) { |
|
|
|
|
|
const std::size_t num_page_table_entries = 1ULL << (address_space_width_in_bits - PAGE_BITS); |
|
|
|
|
|
|
|
|
|
|
|
pointers.resize(num_page_table_entries); |
|
|
|
|
|
attributes.resize(num_page_table_entries); |
|
|
|
|
|
|
|
|
|
|
|
// The default is a 39-bit address space, which causes an initial 1GB allocation size. If the
|
|
|
|
|
|
// vector size is subsequently decreased (via resize), the vector might not automatically
|
|
|
|
|
|
// actually reallocate/resize its underlying allocation, which wastes up to ~800 MB for
|
|
|
|
|
|
// 36-bit titles. Call shrink_to_fit to reduce capacity to what's actually in use.
|
|
|
|
|
|
|
|
|
|
|
|
pointers.shrink_to_fit(); |
|
|
|
|
|
attributes.shrink_to_fit(); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
static void MapPages(PageTable& page_table, VAddr base, u64 size, u8* memory, PageType type) { |
|
|
|
|
|
|
|
|
static void MapPages(Common::PageTable& page_table, VAddr base, u64 size, u8* memory, |
|
|
|
|
|
Common::PageType type) { |
|
|
LOG_DEBUG(HW_Memory, "Mapping {} onto {:016X}-{:016X}", fmt::ptr(memory), base * PAGE_SIZE, |
|
|
LOG_DEBUG(HW_Memory, "Mapping {} onto {:016X}-{:016X}", fmt::ptr(memory), base * PAGE_SIZE, |
|
|
(base + size) * PAGE_SIZE); |
|
|
(base + size) * PAGE_SIZE); |
|
|
|
|
|
|
|
|
@ -92,41 +71,47 @@ static void MapPages(PageTable& page_table, VAddr base, u64 size, u8* memory, Pa |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
void MapMemoryRegion(PageTable& page_table, VAddr base, u64 size, u8* target) { |
|
|
|
|
|
|
|
|
void MapMemoryRegion(Common::PageTable& page_table, VAddr base, u64 size, u8* target) { |
|
|
ASSERT_MSG((size & PAGE_MASK) == 0, "non-page aligned size: {:016X}", 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); |
|
|
ASSERT_MSG((base & PAGE_MASK) == 0, "non-page aligned base: {:016X}", base); |
|
|
MapPages(page_table, base / PAGE_SIZE, size / PAGE_SIZE, target, PageType::Memory); |
|
|
|
|
|
|
|
|
MapPages(page_table, base / PAGE_SIZE, size / PAGE_SIZE, target, Common::PageType::Memory); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
void MapIoRegion(PageTable& page_table, VAddr base, u64 size, MemoryHookPointer mmio_handler) { |
|
|
|
|
|
|
|
|
void MapIoRegion(Common::PageTable& page_table, VAddr base, u64 size, |
|
|
|
|
|
Common::MemoryHookPointer mmio_handler) { |
|
|
ASSERT_MSG((size & PAGE_MASK) == 0, "non-page aligned size: {:016X}", 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); |
|
|
ASSERT_MSG((base & PAGE_MASK) == 0, "non-page aligned base: {:016X}", base); |
|
|
MapPages(page_table, base / PAGE_SIZE, size / PAGE_SIZE, nullptr, PageType::Special); |
|
|
|
|
|
|
|
|
MapPages(page_table, base / PAGE_SIZE, size / PAGE_SIZE, nullptr, Common::PageType::Special); |
|
|
|
|
|
|
|
|
auto interval = boost::icl::discrete_interval<VAddr>::closed(base, base + size - 1); |
|
|
auto interval = boost::icl::discrete_interval<VAddr>::closed(base, base + size - 1); |
|
|
SpecialRegion region{SpecialRegion::Type::IODevice, std::move(mmio_handler)}; |
|
|
|
|
|
page_table.special_regions.add(std::make_pair(interval, std::set<SpecialRegion>{region})); |
|
|
|
|
|
|
|
|
Common::SpecialRegion region{Common::SpecialRegion::Type::IODevice, std::move(mmio_handler)}; |
|
|
|
|
|
page_table.special_regions.add( |
|
|
|
|
|
std::make_pair(interval, std::set<Common::SpecialRegion>{region})); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
void UnmapRegion(PageTable& page_table, VAddr base, u64 size) { |
|
|
|
|
|
|
|
|
void UnmapRegion(Common::PageTable& page_table, VAddr base, u64 size) { |
|
|
ASSERT_MSG((size & PAGE_MASK) == 0, "non-page aligned size: {:016X}", 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); |
|
|
ASSERT_MSG((base & PAGE_MASK) == 0, "non-page aligned base: {:016X}", base); |
|
|
MapPages(page_table, base / PAGE_SIZE, size / PAGE_SIZE, nullptr, PageType::Unmapped); |
|
|
|
|
|
|
|
|
MapPages(page_table, base / PAGE_SIZE, size / PAGE_SIZE, nullptr, Common::PageType::Unmapped); |
|
|
|
|
|
|
|
|
auto interval = boost::icl::discrete_interval<VAddr>::closed(base, base + size - 1); |
|
|
auto interval = boost::icl::discrete_interval<VAddr>::closed(base, base + size - 1); |
|
|
page_table.special_regions.erase(interval); |
|
|
page_table.special_regions.erase(interval); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
void AddDebugHook(PageTable& page_table, VAddr base, u64 size, MemoryHookPointer hook) { |
|
|
|
|
|
|
|
|
void AddDebugHook(Common::PageTable& page_table, VAddr base, u64 size, |
|
|
|
|
|
Common::MemoryHookPointer hook) { |
|
|
auto interval = boost::icl::discrete_interval<VAddr>::closed(base, base + size - 1); |
|
|
auto interval = boost::icl::discrete_interval<VAddr>::closed(base, base + size - 1); |
|
|
SpecialRegion region{SpecialRegion::Type::DebugHook, std::move(hook)}; |
|
|
|
|
|
page_table.special_regions.add(std::make_pair(interval, std::set<SpecialRegion>{region})); |
|
|
|
|
|
|
|
|
Common::SpecialRegion region{Common::SpecialRegion::Type::DebugHook, std::move(hook)}; |
|
|
|
|
|
page_table.special_regions.add( |
|
|
|
|
|
std::make_pair(interval, std::set<Common::SpecialRegion>{region})); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
void RemoveDebugHook(PageTable& page_table, VAddr base, u64 size, MemoryHookPointer hook) { |
|
|
|
|
|
|
|
|
void RemoveDebugHook(Common::PageTable& page_table, VAddr base, u64 size, |
|
|
|
|
|
Common::MemoryHookPointer hook) { |
|
|
auto interval = boost::icl::discrete_interval<VAddr>::closed(base, base + size - 1); |
|
|
auto interval = boost::icl::discrete_interval<VAddr>::closed(base, base + size - 1); |
|
|
SpecialRegion region{SpecialRegion::Type::DebugHook, std::move(hook)}; |
|
|
|
|
|
page_table.special_regions.subtract(std::make_pair(interval, std::set<SpecialRegion>{region})); |
|
|
|
|
|
|
|
|
Common::SpecialRegion region{Common::SpecialRegion::Type::DebugHook, std::move(hook)}; |
|
|
|
|
|
page_table.special_regions.subtract( |
|
|
|
|
|
std::make_pair(interval, std::set<Common::SpecialRegion>{region})); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
/**
|
|
|
/**
|
|
|
@ -175,15 +160,15 @@ T Read(const VAddr vaddr) { |
|
|
return value; |
|
|
return value; |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
PageType type = current_page_table->attributes[vaddr >> PAGE_BITS]; |
|
|
|
|
|
|
|
|
Common::PageType type = current_page_table->attributes[vaddr >> PAGE_BITS]; |
|
|
switch (type) { |
|
|
switch (type) { |
|
|
case PageType::Unmapped: |
|
|
|
|
|
|
|
|
case Common::PageType::Unmapped: |
|
|
LOG_ERROR(HW_Memory, "Unmapped Read{} @ 0x{:08X}", sizeof(T) * 8, vaddr); |
|
|
LOG_ERROR(HW_Memory, "Unmapped Read{} @ 0x{:08X}", sizeof(T) * 8, vaddr); |
|
|
return 0; |
|
|
return 0; |
|
|
case PageType::Memory: |
|
|
|
|
|
|
|
|
case Common::PageType::Memory: |
|
|
ASSERT_MSG(false, "Mapped memory page without a pointer @ {:016X}", vaddr); |
|
|
ASSERT_MSG(false, "Mapped memory page without a pointer @ {:016X}", vaddr); |
|
|
break; |
|
|
break; |
|
|
case PageType::RasterizerCachedMemory: { |
|
|
|
|
|
|
|
|
case Common::PageType::RasterizerCachedMemory: { |
|
|
auto host_ptr{GetPointerFromVMA(vaddr)}; |
|
|
auto host_ptr{GetPointerFromVMA(vaddr)}; |
|
|
Core::System::GetInstance().GPU().FlushRegion(ToCacheAddr(host_ptr), sizeof(T)); |
|
|
Core::System::GetInstance().GPU().FlushRegion(ToCacheAddr(host_ptr), sizeof(T)); |
|
|
T value; |
|
|
T value; |
|
|
@ -205,16 +190,16 @@ void Write(const VAddr vaddr, const T data) { |
|
|
return; |
|
|
return; |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
PageType type = current_page_table->attributes[vaddr >> PAGE_BITS]; |
|
|
|
|
|
|
|
|
Common::PageType type = current_page_table->attributes[vaddr >> PAGE_BITS]; |
|
|
switch (type) { |
|
|
switch (type) { |
|
|
case PageType::Unmapped: |
|
|
|
|
|
|
|
|
case Common::PageType::Unmapped: |
|
|
LOG_ERROR(HW_Memory, "Unmapped Write{} 0x{:08X} @ 0x{:016X}", sizeof(data) * 8, |
|
|
LOG_ERROR(HW_Memory, "Unmapped Write{} 0x{:08X} @ 0x{:016X}", sizeof(data) * 8, |
|
|
static_cast<u32>(data), vaddr); |
|
|
static_cast<u32>(data), vaddr); |
|
|
return; |
|
|
return; |
|
|
case PageType::Memory: |
|
|
|
|
|
|
|
|
case Common::PageType::Memory: |
|
|
ASSERT_MSG(false, "Mapped memory page without a pointer @ {:016X}", vaddr); |
|
|
ASSERT_MSG(false, "Mapped memory page without a pointer @ {:016X}", vaddr); |
|
|
break; |
|
|
break; |
|
|
case PageType::RasterizerCachedMemory: { |
|
|
|
|
|
|
|
|
case Common::PageType::RasterizerCachedMemory: { |
|
|
auto host_ptr{GetPointerFromVMA(vaddr)}; |
|
|
auto host_ptr{GetPointerFromVMA(vaddr)}; |
|
|
Core::System::GetInstance().GPU().InvalidateRegion(ToCacheAddr(host_ptr), sizeof(T)); |
|
|
Core::System::GetInstance().GPU().InvalidateRegion(ToCacheAddr(host_ptr), sizeof(T)); |
|
|
std::memcpy(host_ptr, &data, sizeof(T)); |
|
|
std::memcpy(host_ptr, &data, sizeof(T)); |
|
|
@ -232,10 +217,10 @@ bool IsValidVirtualAddress(const Kernel::Process& process, const VAddr vaddr) { |
|
|
if (page_pointer) |
|
|
if (page_pointer) |
|
|
return true; |
|
|
return true; |
|
|
|
|
|
|
|
|
if (page_table.attributes[vaddr >> PAGE_BITS] == PageType::RasterizerCachedMemory) |
|
|
|
|
|
|
|
|
if (page_table.attributes[vaddr >> PAGE_BITS] == Common::PageType::RasterizerCachedMemory) |
|
|
return true; |
|
|
return true; |
|
|
|
|
|
|
|
|
if (page_table.attributes[vaddr >> PAGE_BITS] != PageType::Special) |
|
|
|
|
|
|
|
|
if (page_table.attributes[vaddr >> PAGE_BITS] != Common::PageType::Special) |
|
|
return false; |
|
|
return false; |
|
|
|
|
|
|
|
|
return false; |
|
|
return false; |
|
|
@ -255,7 +240,8 @@ u8* GetPointer(const VAddr vaddr) { |
|
|
return page_pointer + (vaddr & PAGE_MASK); |
|
|
return page_pointer + (vaddr & PAGE_MASK); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
if (current_page_table->attributes[vaddr >> PAGE_BITS] == PageType::RasterizerCachedMemory) { |
|
|
|
|
|
|
|
|
if (current_page_table->attributes[vaddr >> PAGE_BITS] == |
|
|
|
|
|
Common::PageType::RasterizerCachedMemory) { |
|
|
return GetPointerFromVMA(vaddr); |
|
|
return GetPointerFromVMA(vaddr); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
@ -289,20 +275,20 @@ void RasterizerMarkRegionCached(VAddr vaddr, u64 size, bool cached) { |
|
|
|
|
|
|
|
|
u64 num_pages = ((vaddr + size - 1) >> PAGE_BITS) - (vaddr >> PAGE_BITS) + 1; |
|
|
u64 num_pages = ((vaddr + size - 1) >> PAGE_BITS) - (vaddr >> PAGE_BITS) + 1; |
|
|
for (unsigned i = 0; i < num_pages; ++i, vaddr += PAGE_SIZE) { |
|
|
for (unsigned i = 0; i < num_pages; ++i, vaddr += PAGE_SIZE) { |
|
|
PageType& page_type = current_page_table->attributes[vaddr >> PAGE_BITS]; |
|
|
|
|
|
|
|
|
Common::PageType& page_type = current_page_table->attributes[vaddr >> PAGE_BITS]; |
|
|
|
|
|
|
|
|
if (cached) { |
|
|
if (cached) { |
|
|
// Switch page type to cached if now cached
|
|
|
// Switch page type to cached if now cached
|
|
|
switch (page_type) { |
|
|
switch (page_type) { |
|
|
case PageType::Unmapped: |
|
|
|
|
|
|
|
|
case Common::PageType::Unmapped: |
|
|
// It is not necessary for a process to have this region mapped into its address
|
|
|
// 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.
|
|
|
// space, for example, a system module need not have a VRAM mapping.
|
|
|
break; |
|
|
break; |
|
|
case PageType::Memory: |
|
|
|
|
|
page_type = PageType::RasterizerCachedMemory; |
|
|
|
|
|
|
|
|
case Common::PageType::Memory: |
|
|
|
|
|
page_type = Common::PageType::RasterizerCachedMemory; |
|
|
current_page_table->pointers[vaddr >> PAGE_BITS] = nullptr; |
|
|
current_page_table->pointers[vaddr >> PAGE_BITS] = nullptr; |
|
|
break; |
|
|
break; |
|
|
case PageType::RasterizerCachedMemory: |
|
|
|
|
|
|
|
|
case Common::PageType::RasterizerCachedMemory: |
|
|
// There can be more than one GPU region mapped per CPU region, so it's common that
|
|
|
// There can be more than one GPU region mapped per CPU region, so it's common that
|
|
|
// this area is already marked as cached.
|
|
|
// this area is already marked as cached.
|
|
|
break; |
|
|
break; |
|
|
@ -312,23 +298,23 @@ void RasterizerMarkRegionCached(VAddr vaddr, u64 size, bool cached) { |
|
|
} else { |
|
|
} else { |
|
|
// Switch page type to uncached if now uncached
|
|
|
// Switch page type to uncached if now uncached
|
|
|
switch (page_type) { |
|
|
switch (page_type) { |
|
|
case PageType::Unmapped: |
|
|
|
|
|
|
|
|
case Common::PageType::Unmapped: |
|
|
// It is not necessary for a process to have this region mapped into its address
|
|
|
// 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.
|
|
|
// space, for example, a system module need not have a VRAM mapping.
|
|
|
break; |
|
|
break; |
|
|
case PageType::Memory: |
|
|
|
|
|
|
|
|
case Common::PageType::Memory: |
|
|
// There can be more than one GPU region mapped per CPU region, so it's common that
|
|
|
// There can be more than one GPU region mapped per CPU region, so it's common that
|
|
|
// this area is already unmarked as cached.
|
|
|
// this area is already unmarked as cached.
|
|
|
break; |
|
|
break; |
|
|
case PageType::RasterizerCachedMemory: { |
|
|
|
|
|
|
|
|
case Common::PageType::RasterizerCachedMemory: { |
|
|
u8* pointer = GetPointerFromVMA(vaddr & ~PAGE_MASK); |
|
|
u8* pointer = GetPointerFromVMA(vaddr & ~PAGE_MASK); |
|
|
if (pointer == nullptr) { |
|
|
if (pointer == nullptr) { |
|
|
// It's possible that this function has been called while updating the pagetable
|
|
|
// 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,
|
|
|
// after unmapping a VMA. In that case the underlying VMA will no longer exist,
|
|
|
// and we should just leave the pagetable entry blank.
|
|
|
// and we should just leave the pagetable entry blank.
|
|
|
page_type = PageType::Unmapped; |
|
|
|
|
|
|
|
|
page_type = Common::PageType::Unmapped; |
|
|
} else { |
|
|
} else { |
|
|
page_type = PageType::Memory; |
|
|
|
|
|
|
|
|
page_type = Common::PageType::Memory; |
|
|
current_page_table->pointers[vaddr >> PAGE_BITS] = pointer; |
|
|
current_page_table->pointers[vaddr >> PAGE_BITS] = pointer; |
|
|
} |
|
|
} |
|
|
break; |
|
|
break; |
|
|
@ -370,21 +356,21 @@ void ReadBlock(const Kernel::Process& process, const VAddr src_addr, void* dest_ |
|
|
const VAddr current_vaddr = static_cast<VAddr>((page_index << PAGE_BITS) + page_offset); |
|
|
const VAddr current_vaddr = static_cast<VAddr>((page_index << PAGE_BITS) + page_offset); |
|
|
|
|
|
|
|
|
switch (page_table.attributes[page_index]) { |
|
|
switch (page_table.attributes[page_index]) { |
|
|
case PageType::Unmapped: { |
|
|
|
|
|
|
|
|
case Common::PageType::Unmapped: { |
|
|
LOG_ERROR(HW_Memory, |
|
|
LOG_ERROR(HW_Memory, |
|
|
"Unmapped ReadBlock @ 0x{:016X} (start address = 0x{:016X}, size = {})", |
|
|
"Unmapped ReadBlock @ 0x{:016X} (start address = 0x{:016X}, size = {})", |
|
|
current_vaddr, src_addr, size); |
|
|
current_vaddr, src_addr, size); |
|
|
std::memset(dest_buffer, 0, copy_amount); |
|
|
std::memset(dest_buffer, 0, copy_amount); |
|
|
break; |
|
|
break; |
|
|
} |
|
|
} |
|
|
case PageType::Memory: { |
|
|
|
|
|
|
|
|
case Common::PageType::Memory: { |
|
|
DEBUG_ASSERT(page_table.pointers[page_index]); |
|
|
DEBUG_ASSERT(page_table.pointers[page_index]); |
|
|
|
|
|
|
|
|
const u8* src_ptr = page_table.pointers[page_index] + page_offset; |
|
|
const u8* src_ptr = page_table.pointers[page_index] + page_offset; |
|
|
std::memcpy(dest_buffer, src_ptr, copy_amount); |
|
|
std::memcpy(dest_buffer, src_ptr, copy_amount); |
|
|
break; |
|
|
break; |
|
|
} |
|
|
} |
|
|
case PageType::RasterizerCachedMemory: { |
|
|
|
|
|
|
|
|
case Common::PageType::RasterizerCachedMemory: { |
|
|
const auto& host_ptr{GetPointerFromVMA(process, current_vaddr)}; |
|
|
const auto& host_ptr{GetPointerFromVMA(process, current_vaddr)}; |
|
|
Core::System::GetInstance().GPU().FlushRegion(ToCacheAddr(host_ptr), copy_amount); |
|
|
Core::System::GetInstance().GPU().FlushRegion(ToCacheAddr(host_ptr), copy_amount); |
|
|
std::memcpy(dest_buffer, host_ptr, copy_amount); |
|
|
std::memcpy(dest_buffer, host_ptr, copy_amount); |
|
|
@ -434,20 +420,20 @@ void WriteBlock(const Kernel::Process& process, const VAddr dest_addr, const voi |
|
|
const VAddr current_vaddr = static_cast<VAddr>((page_index << PAGE_BITS) + page_offset); |
|
|
const VAddr current_vaddr = static_cast<VAddr>((page_index << PAGE_BITS) + page_offset); |
|
|
|
|
|
|
|
|
switch (page_table.attributes[page_index]) { |
|
|
switch (page_table.attributes[page_index]) { |
|
|
case PageType::Unmapped: { |
|
|
|
|
|
|
|
|
case Common::PageType::Unmapped: { |
|
|
LOG_ERROR(HW_Memory, |
|
|
LOG_ERROR(HW_Memory, |
|
|
"Unmapped WriteBlock @ 0x{:016X} (start address = 0x{:016X}, size = {})", |
|
|
"Unmapped WriteBlock @ 0x{:016X} (start address = 0x{:016X}, size = {})", |
|
|
current_vaddr, dest_addr, size); |
|
|
current_vaddr, dest_addr, size); |
|
|
break; |
|
|
break; |
|
|
} |
|
|
} |
|
|
case PageType::Memory: { |
|
|
|
|
|
|
|
|
case Common::PageType::Memory: { |
|
|
DEBUG_ASSERT(page_table.pointers[page_index]); |
|
|
DEBUG_ASSERT(page_table.pointers[page_index]); |
|
|
|
|
|
|
|
|
u8* dest_ptr = page_table.pointers[page_index] + page_offset; |
|
|
u8* dest_ptr = page_table.pointers[page_index] + page_offset; |
|
|
std::memcpy(dest_ptr, src_buffer, copy_amount); |
|
|
std::memcpy(dest_ptr, src_buffer, copy_amount); |
|
|
break; |
|
|
break; |
|
|
} |
|
|
} |
|
|
case PageType::RasterizerCachedMemory: { |
|
|
|
|
|
|
|
|
case Common::PageType::RasterizerCachedMemory: { |
|
|
const auto& host_ptr{GetPointerFromVMA(process, current_vaddr)}; |
|
|
const auto& host_ptr{GetPointerFromVMA(process, current_vaddr)}; |
|
|
Core::System::GetInstance().GPU().InvalidateRegion(ToCacheAddr(host_ptr), copy_amount); |
|
|
Core::System::GetInstance().GPU().InvalidateRegion(ToCacheAddr(host_ptr), copy_amount); |
|
|
std::memcpy(host_ptr, src_buffer, copy_amount); |
|
|
std::memcpy(host_ptr, src_buffer, copy_amount); |
|
|
@ -480,20 +466,20 @@ void ZeroBlock(const Kernel::Process& process, const VAddr dest_addr, const std: |
|
|
const VAddr current_vaddr = static_cast<VAddr>((page_index << PAGE_BITS) + page_offset); |
|
|
const VAddr current_vaddr = static_cast<VAddr>((page_index << PAGE_BITS) + page_offset); |
|
|
|
|
|
|
|
|
switch (page_table.attributes[page_index]) { |
|
|
switch (page_table.attributes[page_index]) { |
|
|
case PageType::Unmapped: { |
|
|
|
|
|
|
|
|
case Common::PageType::Unmapped: { |
|
|
LOG_ERROR(HW_Memory, |
|
|
LOG_ERROR(HW_Memory, |
|
|
"Unmapped ZeroBlock @ 0x{:016X} (start address = 0x{:016X}, size = {})", |
|
|
"Unmapped ZeroBlock @ 0x{:016X} (start address = 0x{:016X}, size = {})", |
|
|
current_vaddr, dest_addr, size); |
|
|
current_vaddr, dest_addr, size); |
|
|
break; |
|
|
break; |
|
|
} |
|
|
} |
|
|
case PageType::Memory: { |
|
|
|
|
|
|
|
|
case Common::PageType::Memory: { |
|
|
DEBUG_ASSERT(page_table.pointers[page_index]); |
|
|
DEBUG_ASSERT(page_table.pointers[page_index]); |
|
|
|
|
|
|
|
|
u8* dest_ptr = page_table.pointers[page_index] + page_offset; |
|
|
u8* dest_ptr = page_table.pointers[page_index] + page_offset; |
|
|
std::memset(dest_ptr, 0, copy_amount); |
|
|
std::memset(dest_ptr, 0, copy_amount); |
|
|
break; |
|
|
break; |
|
|
} |
|
|
} |
|
|
case PageType::RasterizerCachedMemory: { |
|
|
|
|
|
|
|
|
case Common::PageType::RasterizerCachedMemory: { |
|
|
const auto& host_ptr{GetPointerFromVMA(process, current_vaddr)}; |
|
|
const auto& host_ptr{GetPointerFromVMA(process, current_vaddr)}; |
|
|
Core::System::GetInstance().GPU().InvalidateRegion(ToCacheAddr(host_ptr), copy_amount); |
|
|
Core::System::GetInstance().GPU().InvalidateRegion(ToCacheAddr(host_ptr), copy_amount); |
|
|
std::memset(host_ptr, 0, copy_amount); |
|
|
std::memset(host_ptr, 0, copy_amount); |
|
|
@ -522,20 +508,20 @@ void CopyBlock(const Kernel::Process& process, VAddr dest_addr, VAddr src_addr, |
|
|
const VAddr current_vaddr = static_cast<VAddr>((page_index << PAGE_BITS) + page_offset); |
|
|
const VAddr current_vaddr = static_cast<VAddr>((page_index << PAGE_BITS) + page_offset); |
|
|
|
|
|
|
|
|
switch (page_table.attributes[page_index]) { |
|
|
switch (page_table.attributes[page_index]) { |
|
|
case PageType::Unmapped: { |
|
|
|
|
|
|
|
|
case Common::PageType::Unmapped: { |
|
|
LOG_ERROR(HW_Memory, |
|
|
LOG_ERROR(HW_Memory, |
|
|
"Unmapped CopyBlock @ 0x{:016X} (start address = 0x{:016X}, size = {})", |
|
|
"Unmapped CopyBlock @ 0x{:016X} (start address = 0x{:016X}, size = {})", |
|
|
current_vaddr, src_addr, size); |
|
|
current_vaddr, src_addr, size); |
|
|
ZeroBlock(process, dest_addr, copy_amount); |
|
|
ZeroBlock(process, dest_addr, copy_amount); |
|
|
break; |
|
|
break; |
|
|
} |
|
|
} |
|
|
case PageType::Memory: { |
|
|
|
|
|
|
|
|
case Common::PageType::Memory: { |
|
|
DEBUG_ASSERT(page_table.pointers[page_index]); |
|
|
DEBUG_ASSERT(page_table.pointers[page_index]); |
|
|
const u8* src_ptr = page_table.pointers[page_index] + page_offset; |
|
|
const u8* src_ptr = page_table.pointers[page_index] + page_offset; |
|
|
WriteBlock(process, dest_addr, src_ptr, copy_amount); |
|
|
WriteBlock(process, dest_addr, src_ptr, copy_amount); |
|
|
break; |
|
|
break; |
|
|
} |
|
|
} |
|
|
case PageType::RasterizerCachedMemory: { |
|
|
|
|
|
|
|
|
case Common::PageType::RasterizerCachedMemory: { |
|
|
const auto& host_ptr{GetPointerFromVMA(process, current_vaddr)}; |
|
|
const auto& host_ptr{GetPointerFromVMA(process, current_vaddr)}; |
|
|
Core::System::GetInstance().GPU().FlushRegion(ToCacheAddr(host_ptr), copy_amount); |
|
|
Core::System::GetInstance().GPU().FlushRegion(ToCacheAddr(host_ptr), copy_amount); |
|
|
WriteBlock(process, dest_addr, host_ptr, copy_amount); |
|
|
WriteBlock(process, dest_addr, host_ptr, copy_amount); |
|
|
|