Merge pull request #3238 from ReinUsesLisp/vk-resource-manager
vk_resource_manager: Catch device losses and other changes
This commit is contained in:
commit
1e76655f83
|
@ -3,12 +3,15 @@
|
||||||
// Refer to the license.txt file included.
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
#include <bitset>
|
#include <bitset>
|
||||||
|
#include <chrono>
|
||||||
#include <cstdlib>
|
#include <cstdlib>
|
||||||
#include <optional>
|
#include <optional>
|
||||||
#include <set>
|
#include <set>
|
||||||
#include <string_view>
|
#include <string_view>
|
||||||
|
#include <thread>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include "common/assert.h"
|
#include "common/assert.h"
|
||||||
|
#include "core/settings.h"
|
||||||
#include "video_core/renderer_vulkan/declarations.h"
|
#include "video_core/renderer_vulkan/declarations.h"
|
||||||
#include "video_core/renderer_vulkan/vk_device.h"
|
#include "video_core/renderer_vulkan/vk_device.h"
|
||||||
|
|
||||||
|
@ -201,6 +204,22 @@ vk::Format VKDevice::GetSupportedFormat(vk::Format wanted_format,
|
||||||
return wanted_format;
|
return wanted_format;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void VKDevice::ReportLoss() const {
|
||||||
|
LOG_CRITICAL(Render_Vulkan, "Device loss occured!");
|
||||||
|
|
||||||
|
// Wait some time to let the log flush
|
||||||
|
std::this_thread::sleep_for(std::chrono::seconds{1});
|
||||||
|
|
||||||
|
if (!nv_device_diagnostic_checkpoints) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
[[maybe_unused]] const std::vector data = graphics_queue.getCheckpointDataNV(dld);
|
||||||
|
// Catch here in debug builds (or with optimizations disabled) the last graphics pipeline to be
|
||||||
|
// executed. It can be done on a debugger by evaluating the expression:
|
||||||
|
// *(VKGraphicsPipeline*)data[0]
|
||||||
|
}
|
||||||
|
|
||||||
bool VKDevice::IsOptimalAstcSupported(const vk::PhysicalDeviceFeatures& features,
|
bool VKDevice::IsOptimalAstcSupported(const vk::PhysicalDeviceFeatures& features,
|
||||||
const vk::DispatchLoaderDynamic& dldi) const {
|
const vk::DispatchLoaderDynamic& dldi) const {
|
||||||
// Disable for now to avoid converting ASTC twice.
|
// Disable for now to avoid converting ASTC twice.
|
||||||
|
@ -381,6 +400,8 @@ std::vector<const char*> VKDevice::LoadExtensions(const vk::DispatchLoaderDynami
|
||||||
VK_EXT_SHADER_VIEWPORT_INDEX_LAYER_EXTENSION_NAME, true);
|
VK_EXT_SHADER_VIEWPORT_INDEX_LAYER_EXTENSION_NAME, true);
|
||||||
Test(extension, ext_subgroup_size_control, VK_EXT_SUBGROUP_SIZE_CONTROL_EXTENSION_NAME,
|
Test(extension, ext_subgroup_size_control, VK_EXT_SUBGROUP_SIZE_CONTROL_EXTENSION_NAME,
|
||||||
false);
|
false);
|
||||||
|
Test(extension, nv_device_diagnostic_checkpoints,
|
||||||
|
VK_NV_DEVICE_DIAGNOSTIC_CHECKPOINTS_EXTENSION_NAME, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (khr_shader_float16_int8) {
|
if (khr_shader_float16_int8) {
|
||||||
|
@ -464,6 +485,7 @@ std::vector<vk::DeviceQueueCreateInfo> VKDevice::GetDeviceQueueCreateInfos() con
|
||||||
std::unordered_map<vk::Format, vk::FormatProperties> VKDevice::GetFormatProperties(
|
std::unordered_map<vk::Format, vk::FormatProperties> VKDevice::GetFormatProperties(
|
||||||
const vk::DispatchLoaderDynamic& dldi, vk::PhysicalDevice physical) {
|
const vk::DispatchLoaderDynamic& dldi, vk::PhysicalDevice physical) {
|
||||||
static constexpr std::array formats{vk::Format::eA8B8G8R8UnormPack32,
|
static constexpr std::array formats{vk::Format::eA8B8G8R8UnormPack32,
|
||||||
|
vk::Format::eA8B8G8R8UintPack32,
|
||||||
vk::Format::eA8B8G8R8SnormPack32,
|
vk::Format::eA8B8G8R8SnormPack32,
|
||||||
vk::Format::eA8B8G8R8SrgbPack32,
|
vk::Format::eA8B8G8R8SrgbPack32,
|
||||||
vk::Format::eB5G6R5UnormPack16,
|
vk::Format::eB5G6R5UnormPack16,
|
||||||
|
|
|
@ -39,6 +39,9 @@ public:
|
||||||
vk::Format GetSupportedFormat(vk::Format wanted_format, vk::FormatFeatureFlags wanted_usage,
|
vk::Format GetSupportedFormat(vk::Format wanted_format, vk::FormatFeatureFlags wanted_usage,
|
||||||
FormatType format_type) const;
|
FormatType format_type) const;
|
||||||
|
|
||||||
|
/// Reports a device loss.
|
||||||
|
void ReportLoss() const;
|
||||||
|
|
||||||
/// Returns the dispatch loader with direct function pointers of the device.
|
/// Returns the dispatch loader with direct function pointers of the device.
|
||||||
const vk::DispatchLoaderDynamic& GetDispatchLoader() const {
|
const vk::DispatchLoaderDynamic& GetDispatchLoader() const {
|
||||||
return dld;
|
return dld;
|
||||||
|
@ -159,6 +162,11 @@ public:
|
||||||
return ext_shader_viewport_index_layer;
|
return ext_shader_viewport_index_layer;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns true if the device supports VK_NV_device_diagnostic_checkpoints.
|
||||||
|
bool IsNvDeviceDiagnosticCheckpoints() const {
|
||||||
|
return nv_device_diagnostic_checkpoints;
|
||||||
|
}
|
||||||
|
|
||||||
/// Returns the vendor name reported from Vulkan.
|
/// Returns the vendor name reported from Vulkan.
|
||||||
std::string_view GetVendorName() const {
|
std::string_view GetVendorName() const {
|
||||||
return vendor_name;
|
return vendor_name;
|
||||||
|
@ -218,6 +226,7 @@ private:
|
||||||
bool ext_index_type_uint8{}; ///< Support for VK_EXT_index_type_uint8.
|
bool ext_index_type_uint8{}; ///< Support for VK_EXT_index_type_uint8.
|
||||||
bool ext_depth_range_unrestricted{}; ///< Support for VK_EXT_depth_range_unrestricted.
|
bool ext_depth_range_unrestricted{}; ///< Support for VK_EXT_depth_range_unrestricted.
|
||||||
bool ext_shader_viewport_index_layer{}; ///< Support for VK_EXT_shader_viewport_index_layer.
|
bool ext_shader_viewport_index_layer{}; ///< Support for VK_EXT_shader_viewport_index_layer.
|
||||||
|
bool nv_device_diagnostic_checkpoints{}; ///< Support for VK_NV_device_diagnostic_checkpoints.
|
||||||
|
|
||||||
// Telemetry parameters
|
// Telemetry parameters
|
||||||
std::string vendor_name; ///< Device's driver name.
|
std::string vendor_name; ///< Device's driver name.
|
||||||
|
|
|
@ -72,12 +72,22 @@ VKFence::VKFence(const VKDevice& device, UniqueFence handle)
|
||||||
VKFence::~VKFence() = default;
|
VKFence::~VKFence() = default;
|
||||||
|
|
||||||
void VKFence::Wait() {
|
void VKFence::Wait() {
|
||||||
|
static constexpr u64 timeout = std::numeric_limits<u64>::max();
|
||||||
const auto dev = device.GetLogical();
|
const auto dev = device.GetLogical();
|
||||||
const auto& dld = device.GetDispatchLoader();
|
const auto& dld = device.GetDispatchLoader();
|
||||||
dev.waitForFences({*handle}, true, std::numeric_limits<u64>::max(), dld);
|
switch (const auto result = dev.waitForFences(1, &*handle, true, timeout, dld)) {
|
||||||
|
case vk::Result::eSuccess:
|
||||||
|
return;
|
||||||
|
case vk::Result::eErrorDeviceLost:
|
||||||
|
device.ReportLoss();
|
||||||
|
[[fallthrough]];
|
||||||
|
default:
|
||||||
|
vk::throwResultException(result, "vk::waitForFences");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void VKFence::Release() {
|
void VKFence::Release() {
|
||||||
|
ASSERT(is_owned);
|
||||||
is_owned = false;
|
is_owned = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -133,8 +143,32 @@ void VKFence::Unprotect(VKResource* resource) {
|
||||||
protected_resources.erase(it);
|
protected_resources.erase(it);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void VKFence::RedirectProtection(VKResource* old_resource, VKResource* new_resource) noexcept {
|
||||||
|
std::replace(std::begin(protected_resources), std::end(protected_resources), old_resource,
|
||||||
|
new_resource);
|
||||||
|
}
|
||||||
|
|
||||||
VKFenceWatch::VKFenceWatch() = default;
|
VKFenceWatch::VKFenceWatch() = default;
|
||||||
|
|
||||||
|
VKFenceWatch::VKFenceWatch(VKFence& initial_fence) {
|
||||||
|
Watch(initial_fence);
|
||||||
|
}
|
||||||
|
|
||||||
|
VKFenceWatch::VKFenceWatch(VKFenceWatch&& rhs) noexcept {
|
||||||
|
fence = std::exchange(rhs.fence, nullptr);
|
||||||
|
if (fence) {
|
||||||
|
fence->RedirectProtection(&rhs, this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
VKFenceWatch& VKFenceWatch::operator=(VKFenceWatch&& rhs) noexcept {
|
||||||
|
fence = std::exchange(rhs.fence, nullptr);
|
||||||
|
if (fence) {
|
||||||
|
fence->RedirectProtection(&rhs, this);
|
||||||
|
}
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
VKFenceWatch::~VKFenceWatch() {
|
VKFenceWatch::~VKFenceWatch() {
|
||||||
if (fence) {
|
if (fence) {
|
||||||
fence->Unprotect(this);
|
fence->Unprotect(this);
|
||||||
|
|
|
@ -65,6 +65,9 @@ public:
|
||||||
/// Removes protection for a resource.
|
/// Removes protection for a resource.
|
||||||
void Unprotect(VKResource* resource);
|
void Unprotect(VKResource* resource);
|
||||||
|
|
||||||
|
/// Redirects one protected resource to a new address.
|
||||||
|
void RedirectProtection(VKResource* old_resource, VKResource* new_resource) noexcept;
|
||||||
|
|
||||||
/// Retreives the fence.
|
/// Retreives the fence.
|
||||||
operator vk::Fence() const {
|
operator vk::Fence() const {
|
||||||
return *handle;
|
return *handle;
|
||||||
|
@ -97,8 +100,13 @@ private:
|
||||||
class VKFenceWatch final : public VKResource {
|
class VKFenceWatch final : public VKResource {
|
||||||
public:
|
public:
|
||||||
explicit VKFenceWatch();
|
explicit VKFenceWatch();
|
||||||
|
VKFenceWatch(VKFence& initial_fence);
|
||||||
|
VKFenceWatch(VKFenceWatch&&) noexcept;
|
||||||
|
VKFenceWatch(const VKFenceWatch&) = delete;
|
||||||
~VKFenceWatch() override;
|
~VKFenceWatch() override;
|
||||||
|
|
||||||
|
VKFenceWatch& operator=(VKFenceWatch&&) noexcept;
|
||||||
|
|
||||||
/// Waits for the fence to be released.
|
/// Waits for the fence to be released.
|
||||||
void Wait();
|
void Wait();
|
||||||
|
|
||||||
|
@ -116,6 +124,14 @@ public:
|
||||||
|
|
||||||
void OnFenceRemoval(VKFence* signaling_fence) override;
|
void OnFenceRemoval(VKFence* signaling_fence) override;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Do not use it paired with Watch. Use TryWatch instead.
|
||||||
|
* Returns true when the watch is free.
|
||||||
|
*/
|
||||||
|
bool IsUsed() const {
|
||||||
|
return fence != nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
VKFence* fence{}; ///< Fence watching this resource. nullptr when the watch is free.
|
VKFence* fence{}; ///< Fence watching this resource. nullptr when the watch is free.
|
||||||
};
|
};
|
||||||
|
|
Loading…
Reference in a new issue