mirror of
https://github.com/skyline-emu/skyline.git
synced 2025-01-17 05:37:55 +03:00
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:
parent
8a62f8d37b
commit
da464d84bc
@ -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;
|
||||||
}
|
}
|
||||||
|
@ -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) {
|
||||||
|
@ -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;
|
||||||
|
@ -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
|
||||||
|
Loading…
x
Reference in New Issue
Block a user