|
|
@ -199,7 +199,15 @@ const u8* MemoryManager::GetPointer(GPUVAddr addr) const {
|
|
|
|
return {};
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void MemoryManager::ReadBlock(GPUVAddr src_addr, void* dest_buffer, std::size_t size) const {
|
|
|
|
bool MemoryManager::IsBlockContinous(const GPUVAddr start, const std::size_t size) {
|
|
|
|
|
|
|
|
const GPUVAddr end = start + size;
|
|
|
|
|
|
|
|
const auto host_ptr_start = reinterpret_cast<std::uintptr_t>(GetPointer(start));
|
|
|
|
|
|
|
|
const auto host_ptr_end = reinterpret_cast<std::uintptr_t>(GetPointer(end));
|
|
|
|
|
|
|
|
const std::size_t range = static_cast<std::size_t>(host_ptr_end - host_ptr_start);
|
|
|
|
|
|
|
|
return range == size;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void MemoryManager::ReadBlock(GPUVAddr src_addr, void* dest_buffer, const std::size_t size) const {
|
|
|
|
std::size_t remaining_size{size};
|
|
|
|
std::size_t remaining_size{size};
|
|
|
|
std::size_t page_index{src_addr >> page_bits};
|
|
|
|
std::size_t page_index{src_addr >> page_bits};
|
|
|
|
std::size_t page_offset{src_addr & page_mask};
|
|
|
|
std::size_t page_offset{src_addr & page_mask};
|
|
|
@ -226,7 +234,30 @@ void MemoryManager::ReadBlock(GPUVAddr src_addr, void* dest_buffer, std::size_t
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void MemoryManager::WriteBlock(GPUVAddr dest_addr, const void* src_buffer, std::size_t size) {
|
|
|
|
void MemoryManager::ReadBlockUnsafe(GPUVAddr src_addr, void* dest_buffer,
|
|
|
|
|
|
|
|
const std::size_t size) const {
|
|
|
|
|
|
|
|
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 u8* page_pointer = page_table.pointers[page_index];
|
|
|
|
|
|
|
|
if (page_pointer) {
|
|
|
|
|
|
|
|
const u8* src_ptr{page_pointer + page_offset};
|
|
|
|
|
|
|
|
std::memcpy(dest_buffer, src_ptr, copy_amount);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
std::memset(dest_buffer, 0, copy_amount);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
page_index++;
|
|
|
|
|
|
|
|
page_offset = 0;
|
|
|
|
|
|
|
|
dest_buffer = static_cast<u8*>(dest_buffer) + copy_amount;
|
|
|
|
|
|
|
|
remaining_size -= copy_amount;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void MemoryManager::WriteBlock(GPUVAddr dest_addr, const void* src_buffer, const std::size_t size) {
|
|
|
|
std::size_t remaining_size{size};
|
|
|
|
std::size_t remaining_size{size};
|
|
|
|
std::size_t page_index{dest_addr >> page_bits};
|
|
|
|
std::size_t page_index{dest_addr >> page_bits};
|
|
|
|
std::size_t page_offset{dest_addr & page_mask};
|
|
|
|
std::size_t page_offset{dest_addr & page_mask};
|
|
|
@ -253,7 +284,28 @@ void MemoryManager::WriteBlock(GPUVAddr dest_addr, const void* src_buffer, std::
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void MemoryManager::CopyBlock(GPUVAddr dest_addr, GPUVAddr src_addr, std::size_t size) {
|
|
|
|
void MemoryManager::WriteBlockUnsafe(GPUVAddr dest_addr, const void* src_buffer,
|
|
|
|
|
|
|
|
const std::size_t size) {
|
|
|
|
|
|
|
|
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)};
|
|
|
|
|
|
|
|
u8* page_pointer = page_table.pointers[page_index];
|
|
|
|
|
|
|
|
if (page_pointer) {
|
|
|
|
|
|
|
|
u8* dest_ptr{page_pointer + page_offset};
|
|
|
|
|
|
|
|
std::memcpy(dest_ptr, src_buffer, copy_amount);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
page_index++;
|
|
|
|
|
|
|
|
page_offset = 0;
|
|
|
|
|
|
|
|
src_buffer = static_cast<const u8*>(src_buffer) + copy_amount;
|
|
|
|
|
|
|
|
remaining_size -= copy_amount;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void MemoryManager::CopyBlock(GPUVAddr dest_addr, GPUVAddr src_addr, const std::size_t size) {
|
|
|
|
std::size_t remaining_size{size};
|
|
|
|
std::size_t remaining_size{size};
|
|
|
|
std::size_t page_index{src_addr >> page_bits};
|
|
|
|
std::size_t page_index{src_addr >> page_bits};
|
|
|
|
std::size_t page_offset{src_addr & page_mask};
|
|
|
|
std::size_t page_offset{src_addr & page_mask};
|
|
|
@ -281,6 +333,12 @@ void MemoryManager::CopyBlock(GPUVAddr dest_addr, GPUVAddr src_addr, std::size_t
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void MemoryManager::CopyBlockUnsafe(GPUVAddr dest_addr, GPUVAddr src_addr, const std::size_t size) {
|
|
|
|
|
|
|
|
std::vector<u8> tmp_buffer(size);
|
|
|
|
|
|
|
|
ReadBlockUnsafe(src_addr, tmp_buffer.data(), size);
|
|
|
|
|
|
|
|
WriteBlockUnsafe(dest_addr, tmp_buffer.data(), size);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void MemoryManager::MapPages(GPUVAddr base, u64 size, u8* memory, Common::PageType type,
|
|
|
|
void MemoryManager::MapPages(GPUVAddr base, u64 size, u8* memory, Common::PageType type,
|
|
|
|
VAddr backing_addr) {
|
|
|
|
VAddr backing_addr) {
|
|
|
|
LOG_DEBUG(HW_GPU, "Mapping {} onto {:016X}-{:016X}", fmt::ptr(memory), base * page_size,
|
|
|
|
LOG_DEBUG(HW_GPU, "Mapping {} onto {:016X}-{:016X}", fmt::ptr(memory), base * page_size,
|
|
|
|