Consolidate NCE::TrapRegions functionality into CreateTrap

`NCE::TrapRegions` was a bit too overloaded as a method as it implicitly trapped which was unnecessary in all current usage cases, this has now been made more explicit by consolidating the functionality into `NCE::CreateTrap` which handles just creation of the trap and nothing past that, `RetrapRegions` has been renamed to `TrapRegions` and handles all trapping now.

Co-authored-by: Billy Laws <blaws05@gmail.com>
This commit is contained in:
PixelyIon 2022-07-29 02:29:18 +05:30
parent 8a62f8d37b
commit da464d84bc
No known key found for this signature in database
GPG Key ID: 11BC6C3201BC2C05
4 changed files with 19 additions and 22 deletions

View File

@ -17,7 +17,7 @@ namespace skyline::gpu {
// We can't just capture this in the lambda since the lambda could exceed the lifetime of the buffer // We can't just capture this in the lambda since the lambda could exceed the lifetime of the buffer
std::weak_ptr<Buffer> weakThis{shared_from_this()}; std::weak_ptr<Buffer> weakThis{shared_from_this()};
trapHandle = gpu.state.nce->TrapRegions(*guest, true, [weakThis] { trapHandle = gpu.state.nce->CreateTrap(*guest, [weakThis] {
auto buffer{weakThis.lock()}; auto buffer{weakThis.lock()};
if (!buffer) if (!buffer)
return; return;
@ -107,7 +107,7 @@ namespace skyline::gpu {
BlockAllCpuBackingWrites(); BlockAllCpuBackingWrites();
AdvanceSequence(); // The GPU will modify buffer contents so advance to the next sequence AdvanceSequence(); // The GPU will modify buffer contents so advance to the next sequence
gpu.state.nce->RetrapRegions(*trapHandle, false); gpu.state.nce->TrapRegions(*trapHandle, false);
} }
void Buffer::WaitOnFence() { void Buffer::WaitOnFence() {
@ -163,7 +163,7 @@ namespace skyline::gpu {
AdvanceSequence(); // We are modifying GPU backing contents so advance to the next sequence AdvanceSequence(); // We are modifying GPU backing contents so advance to the next sequence
if (!skipTrap) if (!skipTrap)
gpu.state.nce->RetrapRegions(*trapHandle, true); // Trap any future CPU writes to this buffer, must be done before the memcpy so that any modifications during the copy are tracked gpu.state.nce->TrapRegions(*trapHandle, true); // Trap any future CPU writes to this buffer, must be done before the memcpy so that any modifications during the copy are tracked
std::memcpy(backing.data(), mirror.data(), mirror.size()); std::memcpy(backing.data(), mirror.data(), mirror.size());
} }
@ -189,7 +189,7 @@ namespace skyline::gpu {
} }
if (!skipTrap) if (!skipTrap)
gpu.state.nce->RetrapRegions(*trapHandle, true); gpu.state.nce->TrapRegions(*trapHandle, true);
return true; return true;
} }

View File

@ -145,7 +145,7 @@ namespace skyline::gpu {
// We can't just capture `this` in the lambda since the lambda could exceed the lifetime of the buffer // We can't just capture `this` in the lambda since the lambda could exceed the lifetime of the buffer
std::weak_ptr<Texture> weakThis{weak_from_this()}; std::weak_ptr<Texture> weakThis{weak_from_this()};
trapHandle = gpu.state.nce->TrapRegions(mappings, true, [weakThis] { trapHandle = gpu.state.nce->CreateTrap(mappings, [weakThis] {
auto texture{weakThis.lock()}; auto texture{weakThis.lock()};
if (!texture) if (!texture)
return; return;
@ -690,7 +690,7 @@ namespace skyline::gpu {
if (gpuDirty && dirtyState == DirtyState::Clean) { if (gpuDirty && dirtyState == DirtyState::Clean) {
// If a texture is Clean then we can just transition it to being GPU dirty and retrap it // If a texture is Clean then we can just transition it to being GPU dirty and retrap it
dirtyState = DirtyState::GpuDirty; dirtyState = DirtyState::GpuDirty;
gpu.state.nce->RetrapRegions(*trapHandle, false); gpu.state.nce->TrapRegions(*trapHandle, false);
return; return;
} else if (dirtyState != DirtyState::CpuDirty) { } else if (dirtyState != DirtyState::CpuDirty) {
return; // If the texture has not been modified on the CPU, there is no need to synchronize it return; // If the texture has not been modified on the CPU, there is no need to synchronize it
@ -709,7 +709,7 @@ namespace skyline::gpu {
cycle = lCycle; cycle = lCycle;
} }
gpu.state.nce->RetrapRegions(*trapHandle, !gpuDirty); // Trap any future CPU reads (optionally) + writes to this texture gpu.state.nce->TrapRegions(*trapHandle, !gpuDirty); // Trap any future CPU reads (optionally) + writes to this texture
} }
void Texture::SynchronizeHostInline(const vk::raii::CommandBuffer &commandBuffer, const std::shared_ptr<FenceCycle> &pCycle, bool gpuDirty) { void Texture::SynchronizeHostInline(const vk::raii::CommandBuffer &commandBuffer, const std::shared_ptr<FenceCycle> &pCycle, bool gpuDirty) {
@ -722,7 +722,7 @@ namespace skyline::gpu {
std::scoped_lock lock{stateMutex}; std::scoped_lock lock{stateMutex};
if (gpuDirty && dirtyState == DirtyState::Clean) { if (gpuDirty && dirtyState == DirtyState::Clean) {
dirtyState = DirtyState::GpuDirty; dirtyState = DirtyState::GpuDirty;
gpu.state.nce->RetrapRegions(*trapHandle, false); gpu.state.nce->TrapRegions(*trapHandle, false);
return; return;
} else if (dirtyState != DirtyState::CpuDirty) { } else if (dirtyState != DirtyState::CpuDirty) {
return; return;
@ -739,7 +739,7 @@ namespace skyline::gpu {
cycle = pCycle; cycle = pCycle;
} }
gpu.state.nce->RetrapRegions(*trapHandle, !gpuDirty); // Trap any future CPU reads (optionally) + writes to this texture gpu.state.nce->TrapRegions(*trapHandle, !gpuDirty); // Trap any future CPU reads (optionally) + writes to this texture
} }
void Texture::SynchronizeGuest(bool cpuDirty, bool skipTrap) { void Texture::SynchronizeGuest(bool cpuDirty, bool skipTrap) {
@ -790,7 +790,7 @@ namespace skyline::gpu {
if (cpuDirty) if (cpuDirty)
gpu.state.nce->DeleteTrap(*trapHandle); gpu.state.nce->DeleteTrap(*trapHandle);
else else
gpu.state.nce->RetrapRegions(*trapHandle, true); // Trap any future CPU writes to this texture gpu.state.nce->TrapRegions(*trapHandle, true); // Trap any future CPU writes to this texture
} }
std::shared_ptr<TextureView> Texture::GetView(vk::ImageViewType type, vk::ImageSubresourceRange range, texture::Format pFormat, vk::ComponentMapping mapping) { std::shared_ptr<TextureView> Texture::GetView(vk::ImageViewType type, vk::ImageSubresourceRange range, texture::Format pFormat, vk::ComponentMapping mapping) {

View File

@ -535,17 +535,15 @@ namespace skyline::nce {
constexpr NCE::TrapHandle::TrapHandle(const TrapMap::GroupHandle &handle) : TrapMap::GroupHandle(handle) {} constexpr NCE::TrapHandle::TrapHandle(const TrapMap::GroupHandle &handle) : TrapMap::GroupHandle(handle) {}
NCE::TrapHandle NCE::TrapRegions(span<span<u8>> regions, bool writeOnly, const LockCallback &lockCallback, const TrapCallback &readCallback, const TrapCallback &writeCallback) { NCE::TrapHandle NCE::CreateTrap(span<span<u8>> regions, const LockCallback &lockCallback, const TrapCallback &readCallback, const TrapCallback &writeCallback) {
TRACE_EVENT("host", "NCE::TrapRegions"); TRACE_EVENT("host", "NCE::CreateTrap");
std::scoped_lock lock(trapMutex); std::scoped_lock lock(trapMutex);
auto protection{writeOnly ? TrapProtection::WriteOnly : TrapProtection::ReadWrite}; TrapHandle handle{trapMap.Insert(regions, CallbackEntry{TrapProtection::None, lockCallback, readCallback, writeCallback})};
TrapHandle handle{trapMap.Insert(regions, CallbackEntry{protection, lockCallback, readCallback, writeCallback})};
ReprotectIntervals(handle->intervals, protection);
return handle; return handle;
} }
void NCE::RetrapRegions(TrapHandle handle, bool writeOnly) { void NCE::TrapRegions(TrapHandle handle, bool writeOnly) {
TRACE_EVENT("host", "NCE::RetrapRegions"); TRACE_EVENT("host", "NCE::TrapRegions");
std::scoped_lock lock(trapMutex); std::scoped_lock lock(trapMutex);
auto protection{writeOnly ? TrapProtection::WriteOnly : TrapProtection::ReadWrite}; auto protection{writeOnly ? TrapProtection::WriteOnly : TrapProtection::ReadWrite};
handle->value.protection = protection; handle->value.protection = protection;

View File

@ -102,23 +102,22 @@ namespace skyline::nce {
}; };
/** /**
* @brief Traps a region of guest memory with a callback for when an access to it has been made * @brief Creates a region of guest memory that can be trapped with a callback for when an access to it has been made
* @param writeOnly If the trap is optimally for write-only accesses initially, this is not guarenteed
* @param lockCallback A callback to lock the resource that is being trapped, it must block until the resource is locked but unlock it prior to returning * @param lockCallback A callback to lock the resource that is being trapped, it must block until the resource is locked but unlock it prior to returning
* @param readCallback A callback for read accesses to the trapped region, it must not block and return a boolean if it would block * @param readCallback A callback for read accesses to the trapped region, it must not block and return a boolean if it would block
* @param writeCallback A callback for write accesses to the trapped region, it must not block and return a boolean if it would block * @param writeCallback A callback for write accesses to the trapped region, it must not block and return a boolean if it would block
* @note The handle **must** be deleted using DeleteTrap before the NCE instance is destroyed * @note The handle **must** be deleted using DeleteTrap before the NCE instance is destroyed
* @note It is UB to supply a region of host memory rather than guest memory * @note It is UB to supply a region of host memory rather than guest memory
* @note Any regions trapped without writeOnly may have their data (except border pages) paged out and it needs to be paged back in inside the callbacks * @note This doesn't trap the region in itself, any trapping must be done via TrapRegions(...)
*/ */
TrapHandle TrapRegions(span<span<u8>> regions, bool writeOnly, const LockCallback& lockCallback, const TrapCallback& readCallback, const TrapCallback& writeCallback); TrapHandle CreateTrap(span<span<u8>> regions, const LockCallback& lockCallback, const TrapCallback& readCallback, const TrapCallback& writeCallback);
/** /**
* @brief Re-traps a region of memory after protections were removed * @brief Re-traps a region of memory after protections were removed
* @param writeOnly If the trap is optimally for write-only accesses, this is not guarenteed * @param writeOnly If the trap is optimally for write-only accesses, this is not guarenteed
* @note Any regions trapped without writeOnly may have their data (except border pages) paged out and it needs to be paged back in inside the callbacks * @note Any regions trapped without writeOnly may have their data (except border pages) paged out and it needs to be paged back in inside the callbacks
*/ */
void RetrapRegions(TrapHandle handle, bool writeOnly); void TrapRegions(TrapHandle handle, bool writeOnly);
/** /**
* @brief Removes protections from a region of memory * @brief Removes protections from a region of memory