|
|
|
@ -1,8 +1,10 @@
|
|
|
|
|
// SPDX-FileCopyrightText: 2015 Citra Emulator Project
|
|
|
|
|
// SPDX-FileCopyrightText: 2018 yuzu Emulator Project
|
|
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
|
#include <cstring>
|
|
|
|
|
#include <mutex>
|
|
|
|
|
#include <span>
|
|
|
|
|
|
|
|
|
|
#include "common/assert.h"
|
|
|
|
@ -10,6 +12,7 @@
|
|
|
|
|
#include "common/common_types.h"
|
|
|
|
|
#include "common/logging/log.h"
|
|
|
|
|
#include "common/page_table.h"
|
|
|
|
|
#include "common/scope_exit.h"
|
|
|
|
|
#include "common/settings.h"
|
|
|
|
|
#include "common/swap.h"
|
|
|
|
|
#include "core/core.h"
|
|
|
|
@ -318,7 +321,7 @@ struct Memory::Impl {
|
|
|
|
|
[&](const Common::ProcessAddress current_vaddr, const std::size_t copy_amount,
|
|
|
|
|
u8* const host_ptr) {
|
|
|
|
|
if constexpr (!UNSAFE) {
|
|
|
|
|
system.GPU().InvalidateRegion(GetInteger(current_vaddr), copy_amount);
|
|
|
|
|
HandleRasterizerWrite(GetInteger(current_vaddr), copy_amount);
|
|
|
|
|
}
|
|
|
|
|
std::memcpy(host_ptr, src_buffer, copy_amount);
|
|
|
|
|
},
|
|
|
|
@ -351,7 +354,7 @@ struct Memory::Impl {
|
|
|
|
|
},
|
|
|
|
|
[&](const Common::ProcessAddress current_vaddr, const std::size_t copy_amount,
|
|
|
|
|
u8* const host_ptr) {
|
|
|
|
|
system.GPU().InvalidateRegion(GetInteger(current_vaddr), copy_amount);
|
|
|
|
|
HandleRasterizerWrite(GetInteger(current_vaddr), copy_amount);
|
|
|
|
|
std::memset(host_ptr, 0, copy_amount);
|
|
|
|
|
},
|
|
|
|
|
[](const std::size_t copy_amount) {});
|
|
|
|
@ -420,7 +423,7 @@ struct Memory::Impl {
|
|
|
|
|
const std::size_t block_size) {
|
|
|
|
|
// dc cvac: Store to point of coherency
|
|
|
|
|
// CPU flush -> GPU invalidate
|
|
|
|
|
system.GPU().InvalidateRegion(GetInteger(current_vaddr), block_size);
|
|
|
|
|
HandleRasterizerWrite(GetInteger(current_vaddr), block_size);
|
|
|
|
|
};
|
|
|
|
|
return PerformCacheOperation(dest_addr, size, on_rasterizer);
|
|
|
|
|
}
|
|
|
|
@ -430,7 +433,7 @@ struct Memory::Impl {
|
|
|
|
|
const std::size_t block_size) {
|
|
|
|
|
// dc civac: Store to point of coherency, and invalidate from cache
|
|
|
|
|
// CPU flush -> GPU invalidate
|
|
|
|
|
system.GPU().InvalidateRegion(GetInteger(current_vaddr), block_size);
|
|
|
|
|
HandleRasterizerWrite(GetInteger(current_vaddr), block_size);
|
|
|
|
|
};
|
|
|
|
|
return PerformCacheOperation(dest_addr, size, on_rasterizer);
|
|
|
|
|
}
|
|
|
|
@ -767,7 +770,18 @@ struct Memory::Impl {
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void HandleRasterizerWrite(VAddr address, size_t size) {
|
|
|
|
|
const size_t core = system.GetCurrentHostThreadID();
|
|
|
|
|
constexpr size_t sys_core = Core::Hardware::NUM_CPU_CORES - 1;
|
|
|
|
|
const size_t core = std::min(system.GetCurrentHostThreadID(),
|
|
|
|
|
sys_core); // any other calls threads go to syscore.
|
|
|
|
|
// Guard on sys_core;
|
|
|
|
|
if (core == sys_core) [[unlikely]] {
|
|
|
|
|
sys_core_guard.lock();
|
|
|
|
|
}
|
|
|
|
|
SCOPE_EXIT({
|
|
|
|
|
if (core == sys_core) [[unlikely]] {
|
|
|
|
|
sys_core_guard.unlock();
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
auto& current_area = rasterizer_write_areas[core];
|
|
|
|
|
VAddr subaddress = address >> YUZU_PAGEBITS;
|
|
|
|
|
bool do_collection = current_area.last_address == subaddress;
|
|
|
|
@ -799,6 +813,7 @@ struct Memory::Impl {
|
|
|
|
|
rasterizer_read_areas{};
|
|
|
|
|
std::array<GPUDirtyState, Core::Hardware::NUM_CPU_CORES> rasterizer_write_areas{};
|
|
|
|
|
std::span<Core::GPUDirtyMemoryManager> gpu_dirty_managers;
|
|
|
|
|
std::mutex sys_core_guard;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
Memory::Memory(Core::System& system_) : system{system_} {
|
|
|
|
|