early-access version 1284
This commit is contained in:
		| @@ -1,7 +1,7 @@ | ||||
| yuzu emulator early access | ||||
| ============= | ||||
|  | ||||
| This is the source code for early-access 1283. | ||||
| This is the source code for early-access 1284. | ||||
|  | ||||
| ## Legal Notice | ||||
|  | ||||
|   | ||||
| @@ -168,8 +168,6 @@ add_library(video_core STATIC | ||||
|     renderer_vulkan/vk_texture_cache.h | ||||
|     renderer_vulkan/vk_update_descriptor.cpp | ||||
|     renderer_vulkan/vk_update_descriptor.h | ||||
|     renderer_vulkan/wrapper.cpp | ||||
|     renderer_vulkan/wrapper.h | ||||
|     shader_cache.h | ||||
|     shader_notify.cpp | ||||
|     shader_notify.h | ||||
| @@ -258,6 +256,16 @@ add_library(video_core STATIC | ||||
|     textures/texture.h | ||||
|     video_core.cpp | ||||
|     video_core.h | ||||
|     vulkan_common/vulkan_debug_callback.cpp | ||||
|     vulkan_common/vulkan_debug_callback.h | ||||
|     vulkan_common/vulkan_instance.cpp | ||||
|     vulkan_common/vulkan_instance.h | ||||
|     vulkan_common/vulkan_library.cpp | ||||
|     vulkan_common/vulkan_library.h | ||||
|     vulkan_common/vulkan_surface.cpp | ||||
|     vulkan_common/vulkan_surface.h | ||||
|     vulkan_common/vulkan_wrapper.cpp | ||||
|     vulkan_common/vulkan_wrapper.h | ||||
| ) | ||||
|  | ||||
| create_target_directory_groups(video_core) | ||||
|   | ||||
| @@ -17,8 +17,8 @@ | ||||
| #include "video_core/renderer_vulkan/vk_state_tracker.h" | ||||
| #include "video_core/renderer_vulkan/vk_texture_cache.h" | ||||
| #include "video_core/renderer_vulkan/vk_update_descriptor.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/surface.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -8,8 +8,8 @@ | ||||
|  | ||||
| #include "video_core/engines/fermi_2d.h" | ||||
| #include "video_core/renderer_vulkan/vk_descriptor_pool.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/texture_cache/types.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -10,8 +10,8 @@ | ||||
| #include "video_core/engines/maxwell_3d.h" | ||||
| #include "video_core/renderer_vulkan/maxwell_to_vk.h" | ||||
| #include "video_core/renderer_vulkan/vk_device.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/surface.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan::MaxwellToVK { | ||||
|  | ||||
|   | ||||
| @@ -7,9 +7,9 @@ | ||||
| #include "common/common_types.h" | ||||
| #include "video_core/engines/maxwell_3d.h" | ||||
| #include "video_core/renderer_vulkan/vk_device.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/surface.h" | ||||
| #include "video_core/textures/texture.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan::MaxwellToVK { | ||||
|  | ||||
|   | ||||
| @@ -12,8 +12,6 @@ | ||||
|  | ||||
| #include <fmt/format.h> | ||||
|  | ||||
| #include "common/dynamic_library.h" | ||||
| #include "common/file_util.h" | ||||
| #include "common/logging/log.h" | ||||
| #include "common/telemetry.h" | ||||
| #include "core/core.h" | ||||
| @@ -31,169 +29,14 @@ | ||||
| #include "video_core/renderer_vulkan/vk_scheduler.h" | ||||
| #include "video_core/renderer_vulkan/vk_state_tracker.h" | ||||
| #include "video_core/renderer_vulkan/vk_swapchain.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
|  | ||||
| // Include these late to avoid polluting previous headers | ||||
| #ifdef _WIN32 | ||||
| #include <windows.h> | ||||
| // ensure include order | ||||
| #include <vulkan/vulkan_win32.h> | ||||
| #endif | ||||
|  | ||||
| #if !defined(_WIN32) && !defined(__APPLE__) | ||||
| #include <X11/Xlib.h> | ||||
| #include <vulkan/vulkan_wayland.h> | ||||
| #include <vulkan/vulkan_xlib.h> | ||||
| #endif | ||||
| #include "video_core/vulkan_common/vulkan_debug_callback.h" | ||||
| #include "video_core/vulkan_common/vulkan_instance.h" | ||||
| #include "video_core/vulkan_common/vulkan_library.h" | ||||
| #include "video_core/vulkan_common/vulkan_surface.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
| namespace { | ||||
|  | ||||
| using Core::Frontend::WindowSystemType; | ||||
|  | ||||
| VkBool32 DebugCallback(VkDebugUtilsMessageSeverityFlagBitsEXT severity, | ||||
|                        VkDebugUtilsMessageTypeFlagsEXT type, | ||||
|                        const VkDebugUtilsMessengerCallbackDataEXT* data, | ||||
|                        [[maybe_unused]] void* user_data) { | ||||
|     const char* const message{data->pMessage}; | ||||
|  | ||||
|     if (severity & VK_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT) { | ||||
|         LOG_CRITICAL(Render_Vulkan, "{}", message); | ||||
|     } else if (severity & VK_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT) { | ||||
|         LOG_WARNING(Render_Vulkan, "{}", message); | ||||
|     } else if (severity & VK_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT) { | ||||
|         LOG_INFO(Render_Vulkan, "{}", message); | ||||
|     } else if (severity & VK_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT) { | ||||
|         LOG_DEBUG(Render_Vulkan, "{}", message); | ||||
|     } | ||||
|     return VK_FALSE; | ||||
| } | ||||
|  | ||||
| Common::DynamicLibrary OpenVulkanLibrary() { | ||||
|     Common::DynamicLibrary library; | ||||
| #ifdef __APPLE__ | ||||
|     // Check if a path to a specific Vulkan library has been specified. | ||||
|     char* libvulkan_env = getenv("LIBVULKAN_PATH"); | ||||
|     if (!libvulkan_env || !library.Open(libvulkan_env)) { | ||||
|         // Use the libvulkan.dylib from the application bundle. | ||||
|         const std::string filename = | ||||
|             Common::FS::GetBundleDirectory() + "/Contents/Frameworks/libvulkan.dylib"; | ||||
|         library.Open(filename.c_str()); | ||||
|     } | ||||
| #else | ||||
|     std::string filename = Common::DynamicLibrary::GetVersionedFilename("vulkan", 1); | ||||
|     if (!library.Open(filename.c_str())) { | ||||
|         // Android devices may not have libvulkan.so.1, only libvulkan.so. | ||||
|         filename = Common::DynamicLibrary::GetVersionedFilename("vulkan"); | ||||
|         (void)library.Open(filename.c_str()); | ||||
|     } | ||||
| #endif | ||||
|     return library; | ||||
| } | ||||
|  | ||||
| std::pair<vk::Instance, u32> CreateInstance(Common::DynamicLibrary& library, | ||||
|                                             vk::InstanceDispatch& dld, WindowSystemType window_type, | ||||
|                                             bool enable_debug_utils, bool enable_layers) { | ||||
|     if (!library.IsOpen()) { | ||||
|         LOG_ERROR(Render_Vulkan, "Vulkan library not available"); | ||||
|         return {}; | ||||
|     } | ||||
|     if (!library.GetSymbol("vkGetInstanceProcAddr", &dld.vkGetInstanceProcAddr)) { | ||||
|         LOG_ERROR(Render_Vulkan, "vkGetInstanceProcAddr not present in Vulkan"); | ||||
|         return {}; | ||||
|     } | ||||
|     if (!vk::Load(dld)) { | ||||
|         LOG_ERROR(Render_Vulkan, "Failed to load Vulkan function pointers"); | ||||
|         return {}; | ||||
|     } | ||||
|  | ||||
|     std::vector<const char*> extensions; | ||||
|     extensions.reserve(6); | ||||
|     switch (window_type) { | ||||
|     case Core::Frontend::WindowSystemType::Headless: | ||||
|         break; | ||||
| #ifdef _WIN32 | ||||
|     case Core::Frontend::WindowSystemType::Windows: | ||||
|         extensions.push_back(VK_KHR_WIN32_SURFACE_EXTENSION_NAME); | ||||
|         break; | ||||
| #endif | ||||
| #if !defined(_WIN32) && !defined(__APPLE__) | ||||
|     case Core::Frontend::WindowSystemType::X11: | ||||
|         extensions.push_back(VK_KHR_XLIB_SURFACE_EXTENSION_NAME); | ||||
|         break; | ||||
|     case Core::Frontend::WindowSystemType::Wayland: | ||||
|         extensions.push_back(VK_KHR_WAYLAND_SURFACE_EXTENSION_NAME); | ||||
|         break; | ||||
| #endif | ||||
|     default: | ||||
|         LOG_ERROR(Render_Vulkan, "Presentation not supported on this platform"); | ||||
|         break; | ||||
|     } | ||||
|     if (window_type != Core::Frontend::WindowSystemType::Headless) { | ||||
|         extensions.push_back(VK_KHR_SURFACE_EXTENSION_NAME); | ||||
|     } | ||||
|     if (enable_debug_utils) { | ||||
|         extensions.push_back(VK_EXT_DEBUG_UTILS_EXTENSION_NAME); | ||||
|     } | ||||
|     extensions.push_back(VK_KHR_GET_PHYSICAL_DEVICE_PROPERTIES_2_EXTENSION_NAME); | ||||
|  | ||||
|     const std::optional properties = vk::EnumerateInstanceExtensionProperties(dld); | ||||
|     if (!properties) { | ||||
|         LOG_ERROR(Render_Vulkan, "Failed to query extension properties"); | ||||
|         return {}; | ||||
|     } | ||||
|  | ||||
|     for (const char* extension : extensions) { | ||||
|         const auto it = | ||||
|             std::find_if(properties->begin(), properties->end(), [extension](const auto& prop) { | ||||
|                 return !std::strcmp(extension, prop.extensionName); | ||||
|             }); | ||||
|         if (it == properties->end()) { | ||||
|             LOG_ERROR(Render_Vulkan, "Required instance extension {} is not available", extension); | ||||
|             return {}; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     std::vector<const char*> layers; | ||||
|     layers.reserve(1); | ||||
|     if (enable_layers) { | ||||
|         layers.push_back("VK_LAYER_KHRONOS_validation"); | ||||
|     } | ||||
|  | ||||
|     const std::optional layer_properties = vk::EnumerateInstanceLayerProperties(dld); | ||||
|     if (!layer_properties) { | ||||
|         LOG_ERROR(Render_Vulkan, "Failed to query layer properties, disabling layers"); | ||||
|         layers.clear(); | ||||
|     } | ||||
|  | ||||
|     for (auto layer_it = layers.begin(); layer_it != layers.end();) { | ||||
|         const char* const layer = *layer_it; | ||||
|         const auto it = std::find_if( | ||||
|             layer_properties->begin(), layer_properties->end(), | ||||
|             [layer](const VkLayerProperties& prop) { return !std::strcmp(layer, prop.layerName); }); | ||||
|         if (it == layer_properties->end()) { | ||||
|             LOG_ERROR(Render_Vulkan, "Layer {} not available, removing it", layer); | ||||
|             layer_it = layers.erase(layer_it); | ||||
|         } else { | ||||
|             ++layer_it; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // Limit the maximum version of Vulkan to avoid using untested version. | ||||
|     const u32 version = std::min(vk::AvailableVersion(dld), static_cast<u32>(VK_API_VERSION_1_1)); | ||||
|  | ||||
|     vk::Instance instance = vk::Instance::Create(version, layers, extensions, dld); | ||||
|     if (!instance) { | ||||
|         LOG_ERROR(Render_Vulkan, "Failed to create Vulkan instance"); | ||||
|         return {}; | ||||
|     } | ||||
|     if (!vk::Load(*instance, dld)) { | ||||
|         LOG_ERROR(Render_Vulkan, "Failed to load Vulkan instance function pointers"); | ||||
|     } | ||||
|     return std::make_pair(std::move(instance), version); | ||||
| } | ||||
|  | ||||
| std::string GetReadableVersion(u32 version) { | ||||
|     return fmt::format("{}.{}.{}", VK_VERSION_MAJOR(version), VK_VERSION_MINOR(version), | ||||
|                        VK_VERSION_PATCH(version)); | ||||
| @@ -216,7 +59,6 @@ std::string GetDriverVersion(const VKDevice& device) { | ||||
|         const u32 minor = version & 0x3fff; | ||||
|         return fmt::format("{}.{}", major, minor); | ||||
|     } | ||||
|  | ||||
|     return GetReadableVersion(version); | ||||
| } | ||||
|  | ||||
| @@ -255,7 +97,6 @@ void RendererVulkan::SwapBuffers(const Tegra::FramebufferConfig* framebuffer) { | ||||
|     if (!framebuffer) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     const auto& layout = render_window.GetFramebufferLayout(); | ||||
|     if (layout.width > 0 && layout.height > 0 && render_window.IsShown()) { | ||||
|         const VAddr framebuffer_addr = framebuffer->address + framebuffer->offset; | ||||
| @@ -284,14 +125,16 @@ void RendererVulkan::SwapBuffers(const Tegra::FramebufferConfig* framebuffer) { | ||||
|     render_window.OnFrameDisplayed(); | ||||
| } | ||||
|  | ||||
| bool RendererVulkan::Init() { | ||||
|     library = OpenVulkanLibrary(); | ||||
|     std::tie(instance, instance_version) = CreateInstance( | ||||
|         library, dld, render_window.GetWindowInfo().type, true, Settings::values.renderer_debug); | ||||
|     if (!instance || !CreateDebugCallback() || !CreateSurface() || !PickDevices()) { | ||||
|         return false; | ||||
| bool RendererVulkan::Init() try { | ||||
|     library = OpenLibrary(); | ||||
|     instance = CreateInstance(library, dld, VK_API_VERSION_1_1, render_window.GetWindowInfo().type, | ||||
|                               true, Settings::values.renderer_debug); | ||||
|     if (Settings::values.renderer_debug) { | ||||
|         debug_callback = CreateDebugCallback(instance); | ||||
|     } | ||||
|     surface = CreateSurface(instance, render_window); | ||||
|  | ||||
|     InitializeDevice(); | ||||
|     Report(); | ||||
|  | ||||
|     memory_manager = std::make_unique<VKMemoryManager>(*device); | ||||
| @@ -311,8 +154,11 @@ bool RendererVulkan::Init() { | ||||
|     blit_screen = | ||||
|         std::make_unique<VKBlitScreen>(cpu_memory, render_window, *rasterizer, *device, | ||||
|                                        *memory_manager, *swapchain, *scheduler, screen_info); | ||||
|  | ||||
|     return true; | ||||
|  | ||||
| } catch (const vk::Exception& exception) { | ||||
|     LOG_ERROR(Render_Vulkan, "Vulkan initialization failed with error: {}", exception.what()); | ||||
|     return false; | ||||
| } | ||||
|  | ||||
| void RendererVulkan::ShutDown() { | ||||
| @@ -322,7 +168,6 @@ void RendererVulkan::ShutDown() { | ||||
|     if (const auto& dev = device->GetLogical()) { | ||||
|         dev.WaitIdle(); | ||||
|     } | ||||
|  | ||||
|     rasterizer.reset(); | ||||
|     blit_screen.reset(); | ||||
|     scheduler.reset(); | ||||
| @@ -331,95 +176,15 @@ void RendererVulkan::ShutDown() { | ||||
|     device.reset(); | ||||
| } | ||||
|  | ||||
| bool RendererVulkan::CreateDebugCallback() { | ||||
|     if (!Settings::values.renderer_debug) { | ||||
|         return true; | ||||
|     } | ||||
|     debug_callback = instance.TryCreateDebugCallback(DebugCallback); | ||||
|     if (!debug_callback) { | ||||
|         LOG_ERROR(Render_Vulkan, "Failed to create debug callback"); | ||||
|         return false; | ||||
|     } | ||||
|     return true; | ||||
| } | ||||
|  | ||||
| bool RendererVulkan::CreateSurface() { | ||||
|     [[maybe_unused]] const auto& window_info = render_window.GetWindowInfo(); | ||||
|     VkSurfaceKHR unsafe_surface = nullptr; | ||||
|  | ||||
| #ifdef _WIN32 | ||||
|     if (window_info.type == Core::Frontend::WindowSystemType::Windows) { | ||||
|         const HWND hWnd = static_cast<HWND>(window_info.render_surface); | ||||
|         const VkWin32SurfaceCreateInfoKHR win32_ci{VK_STRUCTURE_TYPE_WIN32_SURFACE_CREATE_INFO_KHR, | ||||
|                                                    nullptr, 0, nullptr, hWnd}; | ||||
|         const auto vkCreateWin32SurfaceKHR = reinterpret_cast<PFN_vkCreateWin32SurfaceKHR>( | ||||
|             dld.vkGetInstanceProcAddr(*instance, "vkCreateWin32SurfaceKHR")); | ||||
|         if (!vkCreateWin32SurfaceKHR || | ||||
|             vkCreateWin32SurfaceKHR(*instance, &win32_ci, nullptr, &unsafe_surface) != VK_SUCCESS) { | ||||
|             LOG_ERROR(Render_Vulkan, "Failed to initialize Win32 surface"); | ||||
|             return false; | ||||
|         } | ||||
|     } | ||||
| #endif | ||||
| #if !defined(_WIN32) && !defined(__APPLE__) | ||||
|     if (window_info.type == Core::Frontend::WindowSystemType::X11) { | ||||
|         const VkXlibSurfaceCreateInfoKHR xlib_ci{ | ||||
|             VK_STRUCTURE_TYPE_XLIB_SURFACE_CREATE_INFO_KHR, nullptr, 0, | ||||
|             static_cast<Display*>(window_info.display_connection), | ||||
|             reinterpret_cast<Window>(window_info.render_surface)}; | ||||
|         const auto vkCreateXlibSurfaceKHR = reinterpret_cast<PFN_vkCreateXlibSurfaceKHR>( | ||||
|             dld.vkGetInstanceProcAddr(*instance, "vkCreateXlibSurfaceKHR")); | ||||
|         if (!vkCreateXlibSurfaceKHR || | ||||
|             vkCreateXlibSurfaceKHR(*instance, &xlib_ci, nullptr, &unsafe_surface) != VK_SUCCESS) { | ||||
|             LOG_ERROR(Render_Vulkan, "Failed to initialize Xlib surface"); | ||||
|             return false; | ||||
|         } | ||||
|     } | ||||
|     if (window_info.type == Core::Frontend::WindowSystemType::Wayland) { | ||||
|         const VkWaylandSurfaceCreateInfoKHR wayland_ci{ | ||||
|             VK_STRUCTURE_TYPE_WAYLAND_SURFACE_CREATE_INFO_KHR, nullptr, 0, | ||||
|             static_cast<wl_display*>(window_info.display_connection), | ||||
|             static_cast<wl_surface*>(window_info.render_surface)}; | ||||
|         const auto vkCreateWaylandSurfaceKHR = reinterpret_cast<PFN_vkCreateWaylandSurfaceKHR>( | ||||
|             dld.vkGetInstanceProcAddr(*instance, "vkCreateWaylandSurfaceKHR")); | ||||
|         if (!vkCreateWaylandSurfaceKHR || | ||||
|             vkCreateWaylandSurfaceKHR(*instance, &wayland_ci, nullptr, &unsafe_surface) != | ||||
|                 VK_SUCCESS) { | ||||
|             LOG_ERROR(Render_Vulkan, "Failed to initialize Wayland surface"); | ||||
|             return false; | ||||
|         } | ||||
|     } | ||||
| #endif | ||||
|     if (!unsafe_surface) { | ||||
|         LOG_ERROR(Render_Vulkan, "Presentation not supported on this platform"); | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     surface = vk::SurfaceKHR(unsafe_surface, *instance, dld); | ||||
|     return true; | ||||
| } | ||||
|  | ||||
| bool RendererVulkan::PickDevices() { | ||||
|     const auto devices = instance.EnumeratePhysicalDevices(); | ||||
|     if (!devices) { | ||||
|         LOG_ERROR(Render_Vulkan, "Failed to enumerate physical devices"); | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
| void RendererVulkan::InitializeDevice() { | ||||
|     const std::vector<VkPhysicalDevice> devices = instance.EnumeratePhysicalDevices(); | ||||
|     const s32 device_index = Settings::values.vulkan_device.GetValue(); | ||||
|     if (device_index < 0 || device_index >= static_cast<s32>(devices->size())) { | ||||
|     if (device_index < 0 || device_index >= static_cast<s32>(devices.size())) { | ||||
|         LOG_ERROR(Render_Vulkan, "Invalid device index {}!", device_index); | ||||
|         return false; | ||||
|         throw vk::Exception(VK_ERROR_INITIALIZATION_FAILED); | ||||
|     } | ||||
|     const vk::PhysicalDevice physical_device((*devices)[static_cast<std::size_t>(device_index)], | ||||
|                                              dld); | ||||
|     if (!VKDevice::IsSuitable(physical_device, *surface)) { | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     device = | ||||
|         std::make_unique<VKDevice>(*instance, instance_version, physical_device, *surface, dld); | ||||
|     return device->Create(); | ||||
|     const vk::PhysicalDevice physical_device(devices[static_cast<size_t>(device_index)], dld); | ||||
|     device = std::make_unique<VKDevice>(*instance, physical_device, *surface, dld); | ||||
| } | ||||
|  | ||||
| void RendererVulkan::Report() const { | ||||
| @@ -444,26 +209,21 @@ void RendererVulkan::Report() const { | ||||
|     telemetry_session.AddField(field, "GPU_Vulkan_Extensions", extensions); | ||||
| } | ||||
|  | ||||
| std::vector<std::string> RendererVulkan::EnumerateDevices() { | ||||
| std::vector<std::string> RendererVulkan::EnumerateDevices() try { | ||||
|     vk::InstanceDispatch dld; | ||||
|     Common::DynamicLibrary library = OpenVulkanLibrary(); | ||||
|     vk::Instance instance = | ||||
|         CreateInstance(library, dld, WindowSystemType::Headless, false, false).first; | ||||
|     if (!instance) { | ||||
|         return {}; | ||||
|     } | ||||
|  | ||||
|     const std::optional physical_devices = instance.EnumeratePhysicalDevices(); | ||||
|     if (!physical_devices) { | ||||
|         return {}; | ||||
|     } | ||||
|  | ||||
|     const Common::DynamicLibrary library = OpenLibrary(); | ||||
|     const vk::Instance instance = CreateInstance(library, dld, VK_API_VERSION_1_0); | ||||
|     const std::vector<VkPhysicalDevice> physical_devices = instance.EnumeratePhysicalDevices(); | ||||
|     std::vector<std::string> names; | ||||
|     names.reserve(physical_devices->size()); | ||||
|     for (const auto& device : *physical_devices) { | ||||
|     names.reserve(physical_devices.size()); | ||||
|     for (const VkPhysicalDevice device : physical_devices) { | ||||
|         names.push_back(vk::PhysicalDevice(device, dld).GetProperties().deviceName); | ||||
|     } | ||||
|     return names; | ||||
|  | ||||
| } catch (const vk::Exception& exception) { | ||||
|     LOG_ERROR(Render_Vulkan, "Failed to enumerate devices with error: {}", exception.what()); | ||||
|     return {}; | ||||
| } | ||||
|  | ||||
| } // namespace Vulkan | ||||
|   | ||||
| @@ -11,7 +11,7 @@ | ||||
| #include "common/dynamic_library.h" | ||||
|  | ||||
| #include "video_core/renderer_base.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Core { | ||||
| class TelemetrySession; | ||||
| @@ -56,11 +56,7 @@ public: | ||||
|     static std::vector<std::string> EnumerateDevices(); | ||||
|  | ||||
| private: | ||||
|     bool CreateDebugCallback(); | ||||
|  | ||||
|     bool CreateSurface(); | ||||
|  | ||||
|     bool PickDevices(); | ||||
|     void InitializeDevice(); | ||||
|  | ||||
|     void Report() const; | ||||
|  | ||||
| @@ -72,13 +68,12 @@ private: | ||||
|     vk::InstanceDispatch dld; | ||||
|  | ||||
|     vk::Instance instance; | ||||
|     u32 instance_version{}; | ||||
|  | ||||
|     vk::SurfaceKHR surface; | ||||
|  | ||||
|     VKScreenInfo screen_info; | ||||
|  | ||||
|     vk::DebugCallback debug_callback; | ||||
|     vk::DebugUtilsMessenger debug_callback; | ||||
|     std::unique_ptr<VKDevice> device; | ||||
|     std::unique_ptr<VKMemoryManager> memory_manager; | ||||
|     std::unique_ptr<StateTracker> state_tracker; | ||||
|   | ||||
| @@ -27,9 +27,9 @@ | ||||
| #include "video_core/renderer_vulkan/vk_scheduler.h" | ||||
| #include "video_core/renderer_vulkan/vk_shader_util.h" | ||||
| #include "video_core/renderer_vulkan/vk_swapchain.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/surface.h" | ||||
| #include "video_core/textures/decoders.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -7,7 +7,7 @@ | ||||
| #include <memory> | ||||
|  | ||||
| #include "video_core/renderer_vulkan/vk_memory_manager.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Core { | ||||
| class System; | ||||
|   | ||||
| @@ -12,7 +12,7 @@ | ||||
| #include "video_core/renderer_vulkan/vk_device.h" | ||||
| #include "video_core/renderer_vulkan/vk_scheduler.h" | ||||
| #include "video_core/renderer_vulkan/vk_stream_buffer.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -11,7 +11,7 @@ | ||||
| #include "video_core/renderer_vulkan/vk_memory_manager.h" | ||||
| #include "video_core/renderer_vulkan/vk_staging_buffer_pool.h" | ||||
| #include "video_core/renderer_vulkan/vk_stream_buffer.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -6,7 +6,7 @@ | ||||
|  | ||||
| #include "video_core/renderer_vulkan/vk_command_pool.h" | ||||
| #include "video_core/renderer_vulkan/vk_device.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
| #include <vector> | ||||
|  | ||||
| #include "video_core/renderer_vulkan/vk_resource_pool.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -19,7 +19,7 @@ | ||||
| #include "video_core/renderer_vulkan/vk_scheduler.h" | ||||
| #include "video_core/renderer_vulkan/vk_staging_buffer_pool.h" | ||||
| #include "video_core/renderer_vulkan/vk_update_descriptor.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -11,7 +11,7 @@ | ||||
| #include "common/common_types.h" | ||||
| #include "video_core/engines/maxwell_3d.h" | ||||
| #include "video_core/renderer_vulkan/vk_descriptor_pool.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -11,7 +11,7 @@ | ||||
| #include "video_core/renderer_vulkan/vk_scheduler.h" | ||||
| #include "video_core/renderer_vulkan/vk_shader_decompiler.h" | ||||
| #include "video_core/renderer_vulkan/vk_update_descriptor.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -7,7 +7,7 @@ | ||||
| #include "common/common_types.h" | ||||
| #include "video_core/renderer_vulkan/vk_descriptor_pool.h" | ||||
| #include "video_core/renderer_vulkan/vk_shader_decompiler.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -9,7 +9,7 @@ | ||||
| #include "video_core/renderer_vulkan/vk_device.h" | ||||
| #include "video_core/renderer_vulkan/vk_resource_pool.h" | ||||
| #include "video_core/renderer_vulkan/vk_scheduler.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -7,7 +7,7 @@ | ||||
| #include <vector> | ||||
|  | ||||
| #include "video_core/renderer_vulkan/vk_resource_pool.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -14,7 +14,7 @@ | ||||
| #include "common/assert.h" | ||||
| #include "core/settings.h" | ||||
| #include "video_core/renderer_vulkan/vk_device.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
| @@ -206,17 +206,14 @@ std::unordered_map<VkFormat, VkFormatProperties> GetFormatProperties( | ||||
|  | ||||
| } // Anonymous namespace | ||||
|  | ||||
| VKDevice::VKDevice(VkInstance instance_, u32 instance_version_, vk::PhysicalDevice physical_, | ||||
|                    VkSurfaceKHR surface, const vk::InstanceDispatch& dld_) | ||||
| VKDevice::VKDevice(VkInstance instance_, vk::PhysicalDevice physical_, VkSurfaceKHR surface, | ||||
|                    const vk::InstanceDispatch& dld_) | ||||
|     : instance{instance_}, dld{dld_}, physical{physical_}, properties{physical.GetProperties()}, | ||||
|       instance_version{instance_version_}, format_properties{GetFormatProperties(physical, dld)} { | ||||
|       format_properties{GetFormatProperties(physical, dld)} { | ||||
|     CheckSuitability(); | ||||
|     SetupFamilies(surface); | ||||
|     SetupFeatures(); | ||||
| } | ||||
|  | ||||
| VKDevice::~VKDevice() = default; | ||||
|  | ||||
| bool VKDevice::Create() { | ||||
|     const auto queue_cis = GetDeviceQueueCreateInfos(); | ||||
|     const std::vector extensions = LoadExtensions(); | ||||
|  | ||||
| @@ -426,12 +423,7 @@ bool VKDevice::Create() { | ||||
|         }; | ||||
|         first_next = &diagnostics_nv; | ||||
|     } | ||||
|  | ||||
|     logical = vk::Device::Create(physical, queue_cis, extensions, first_next, dld); | ||||
|     if (!logical) { | ||||
|         LOG_ERROR(Render_Vulkan, "Failed to create logical device"); | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     CollectTelemetryParameters(); | ||||
|     CollectToolingInfo(); | ||||
| @@ -455,9 +447,10 @@ bool VKDevice::Create() { | ||||
|     present_queue = logical.GetQueue(present_family); | ||||
|  | ||||
|     use_asynchronous_shaders = Settings::values.use_asynchronous_shaders.GetValue(); | ||||
|     return true; | ||||
| } | ||||
|  | ||||
| VKDevice::~VKDevice() = default; | ||||
|  | ||||
| VkFormat VKDevice::GetSupportedFormat(VkFormat wanted_format, VkFormatFeatureFlags wanted_usage, | ||||
|                                       FormatType format_type) const { | ||||
|     if (IsFormatSupported(wanted_format, wanted_usage, format_type)) { | ||||
| @@ -556,64 +549,45 @@ bool VKDevice::IsFormatSupported(VkFormat wanted_format, VkFormatFeatureFlags wa | ||||
|     return (supported_usage & wanted_usage) == wanted_usage; | ||||
| } | ||||
|  | ||||
| bool VKDevice::IsSuitable(vk::PhysicalDevice physical, VkSurfaceKHR surface) { | ||||
|     bool is_suitable = true; | ||||
| void VKDevice::CheckSuitability() const { | ||||
|     std::bitset<REQUIRED_EXTENSIONS.size()> available_extensions; | ||||
|  | ||||
|     for (const auto& prop : physical.EnumerateDeviceExtensionProperties()) { | ||||
|     for (const VkExtensionProperties& property : physical.EnumerateDeviceExtensionProperties()) { | ||||
|         for (std::size_t i = 0; i < REQUIRED_EXTENSIONS.size(); ++i) { | ||||
|             if (available_extensions[i]) { | ||||
|                 continue; | ||||
|             } | ||||
|             const std::string_view name{prop.extensionName}; | ||||
|             const std::string_view name{property.extensionName}; | ||||
|             available_extensions[i] = name == REQUIRED_EXTENSIONS[i]; | ||||
|         } | ||||
|     } | ||||
|     if (!available_extensions.all()) { | ||||
|         for (std::size_t i = 0; i < REQUIRED_EXTENSIONS.size(); ++i) { | ||||
|             if (available_extensions[i]) { | ||||
|                 continue; | ||||
|             } | ||||
|             LOG_ERROR(Render_Vulkan, "Missing required extension: {}", REQUIRED_EXTENSIONS[i]); | ||||
|             is_suitable = false; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     bool has_graphics{}, has_present{}; | ||||
|     const std::vector queue_family_properties = physical.GetQueueFamilyProperties(); | ||||
|     for (u32 i = 0; i < static_cast<u32>(queue_family_properties.size()); ++i) { | ||||
|         const auto& family = queue_family_properties[i]; | ||||
|         if (family.queueCount == 0) { | ||||
|     for (size_t i = 0; i < REQUIRED_EXTENSIONS.size(); ++i) { | ||||
|         if (available_extensions[i]) { | ||||
|             continue; | ||||
|         } | ||||
|         has_graphics |= family.queueFlags & VK_QUEUE_GRAPHICS_BIT; | ||||
|         has_present |= physical.GetSurfaceSupportKHR(i, surface); | ||||
|         LOG_ERROR(Render_Vulkan, "Missing required extension: {}", REQUIRED_EXTENSIONS[i]); | ||||
|         throw vk::Exception(VK_ERROR_EXTENSION_NOT_PRESENT); | ||||
|     } | ||||
|     if (!has_graphics || !has_present) { | ||||
|         LOG_ERROR(Render_Vulkan, "Device lacks a graphics and present queue"); | ||||
|         is_suitable = false; | ||||
|     struct LimitTuple { | ||||
|         u32 minimum; | ||||
|         u32 value; | ||||
|         const char* name; | ||||
|     }; | ||||
|     const VkPhysicalDeviceLimits& limits{properties.limits}; | ||||
|     const std::array limits_report{ | ||||
|         LimitTuple{65536, limits.maxUniformBufferRange, "maxUniformBufferRange"}, | ||||
|         LimitTuple{16, limits.maxViewports, "maxViewports"}, | ||||
|         LimitTuple{8, limits.maxColorAttachments, "maxColorAttachments"}, | ||||
|         LimitTuple{8, limits.maxClipDistances, "maxClipDistances"}, | ||||
|     }; | ||||
|     for (const auto& tuple : limits_report) { | ||||
|         if (tuple.value < tuple.minimum) { | ||||
|             LOG_ERROR(Render_Vulkan, "{} has to be {} or greater but it is {}", tuple.name, | ||||
|                       tuple.minimum, tuple.value); | ||||
|             throw vk::Exception(VK_ERROR_FEATURE_NOT_PRESENT); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // TODO(Rodrigo): Check if the device matches all requeriments. | ||||
|     const auto properties{physical.GetProperties()}; | ||||
|     const auto& limits{properties.limits}; | ||||
|  | ||||
|     constexpr u32 required_ubo_size = 65536; | ||||
|     if (limits.maxUniformBufferRange < required_ubo_size) { | ||||
|         LOG_ERROR(Render_Vulkan, "Device UBO size {} is too small, {} is required", | ||||
|                   limits.maxUniformBufferRange, required_ubo_size); | ||||
|         is_suitable = false; | ||||
|     } | ||||
|  | ||||
|     constexpr u32 required_num_viewports = 16; | ||||
|     if (limits.maxViewports < required_num_viewports) { | ||||
|         LOG_INFO(Render_Vulkan, "Device number of viewports {} is too small, {} is required", | ||||
|                  limits.maxViewports, required_num_viewports); | ||||
|         is_suitable = false; | ||||
|     } | ||||
|  | ||||
|     const auto features{physical.GetFeatures()}; | ||||
|     const std::array feature_report = { | ||||
|     const VkPhysicalDeviceFeatures features{physical.GetFeatures()}; | ||||
|     const std::array feature_report{ | ||||
|         std::make_pair(features.vertexPipelineStoresAndAtomics, "vertexPipelineStoresAndAtomics"), | ||||
|         std::make_pair(features.imageCubeArray, "imageCubeArray"), | ||||
|         std::make_pair(features.independentBlend, "independentBlend"), | ||||
| @@ -631,19 +605,13 @@ bool VKDevice::IsSuitable(vk::PhysicalDevice physical, VkSurfaceKHR surface) { | ||||
|         std::make_pair(features.shaderStorageImageWriteWithoutFormat, | ||||
|                        "shaderStorageImageWriteWithoutFormat"), | ||||
|     }; | ||||
|     for (const auto& [supported, name] : feature_report) { | ||||
|         if (supported) { | ||||
|     for (const auto& [is_supported, name] : feature_report) { | ||||
|         if (is_supported) { | ||||
|             continue; | ||||
|         } | ||||
|         LOG_ERROR(Render_Vulkan, "Missing required feature: {}", name); | ||||
|         is_suitable = false; | ||||
|         throw vk::Exception(VK_ERROR_FEATURE_NOT_PRESENT); | ||||
|     } | ||||
|  | ||||
|     if (!is_suitable) { | ||||
|         LOG_ERROR(Render_Vulkan, "{} is not suitable", properties.deviceName); | ||||
|     } | ||||
|  | ||||
|     return is_suitable; | ||||
| } | ||||
|  | ||||
| std::vector<const char*> VKDevice::LoadExtensions() { | ||||
| @@ -685,9 +653,7 @@ std::vector<const char*> VKDevice::LoadExtensions() { | ||||
|         test(has_ext_custom_border_color, VK_EXT_CUSTOM_BORDER_COLOR_EXTENSION_NAME, false); | ||||
|         test(has_ext_extended_dynamic_state, VK_EXT_EXTENDED_DYNAMIC_STATE_EXTENSION_NAME, false); | ||||
|         test(has_ext_robustness2, VK_EXT_ROBUSTNESS_2_EXTENSION_NAME, false); | ||||
|         if (instance_version >= VK_API_VERSION_1_1) { | ||||
|             test(has_ext_subgroup_size_control, VK_EXT_SUBGROUP_SIZE_CONTROL_EXTENSION_NAME, false); | ||||
|         } | ||||
|         test(has_ext_subgroup_size_control, VK_EXT_SUBGROUP_SIZE_CONTROL_EXTENSION_NAME, false); | ||||
|         if (Settings::values.renderer_debug) { | ||||
|             test(nv_device_diagnostics_config, VK_NV_DEVICE_DIAGNOSTICS_CONFIG_EXTENSION_NAME, | ||||
|                  true); | ||||
| @@ -802,28 +768,34 @@ std::vector<const char*> VKDevice::LoadExtensions() { | ||||
| } | ||||
|  | ||||
| void VKDevice::SetupFamilies(VkSurfaceKHR surface) { | ||||
|     std::optional<u32> graphics_family_, present_family_; | ||||
|  | ||||
|     const std::vector queue_family_properties = physical.GetQueueFamilyProperties(); | ||||
|     for (u32 i = 0; i < static_cast<u32>(queue_family_properties.size()); ++i) { | ||||
|         if (graphics_family_ && present_family_) | ||||
|     std::optional<u32> graphics; | ||||
|     std::optional<u32> present; | ||||
|     for (u32 index = 0; index < static_cast<u32>(queue_family_properties.size()); ++index) { | ||||
|         if (graphics && present) { | ||||
|             break; | ||||
|  | ||||
|         const auto& queue_family = queue_family_properties[i]; | ||||
|         if (queue_family.queueCount == 0) | ||||
|             continue; | ||||
|  | ||||
|         if (queue_family.queueFlags & VK_QUEUE_GRAPHICS_BIT) { | ||||
|             graphics_family_ = i; | ||||
|         } | ||||
|         if (physical.GetSurfaceSupportKHR(i, surface)) { | ||||
|             present_family_ = i; | ||||
|         const VkQueueFamilyProperties& queue_family = queue_family_properties[index]; | ||||
|         if (queue_family.queueCount == 0) { | ||||
|             continue; | ||||
|         } | ||||
|         if (queue_family.queueFlags & VK_QUEUE_GRAPHICS_BIT) { | ||||
|             graphics = index; | ||||
|         } | ||||
|         if (physical.GetSurfaceSupportKHR(index, surface)) { | ||||
|             present = index; | ||||
|         } | ||||
|     } | ||||
|     ASSERT(graphics_family_ && present_family_); | ||||
|  | ||||
|     graphics_family = *graphics_family_; | ||||
|     present_family = *present_family_; | ||||
|     if (!graphics) { | ||||
|         LOG_ERROR(Render_Vulkan, "Device lacks a graphics queue"); | ||||
|         throw vk::Exception(VK_ERROR_FEATURE_NOT_PRESENT); | ||||
|     } | ||||
|     if (!present) { | ||||
|         LOG_ERROR(Render_Vulkan, "Device lacks a present queue"); | ||||
|         throw vk::Exception(VK_ERROR_FEATURE_NOT_PRESENT); | ||||
|     } | ||||
|     graphics_family = *graphics; | ||||
|     present_family = *present; | ||||
| } | ||||
|  | ||||
| void VKDevice::SetupFeatures() { | ||||
|   | ||||
| @@ -11,7 +11,7 @@ | ||||
|  | ||||
| #include "common/common_types.h" | ||||
| #include "video_core/renderer_vulkan/nsight_aftermath_tracker.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
| @@ -24,13 +24,10 @@ const u32 GuestWarpSize = 32; | ||||
| /// Handles data specific to a physical device. | ||||
| class VKDevice final { | ||||
| public: | ||||
|     explicit VKDevice(VkInstance instance, u32 instance_version, vk::PhysicalDevice physical, | ||||
|                       VkSurfaceKHR surface, const vk::InstanceDispatch& dld); | ||||
|     explicit VKDevice(VkInstance instance, vk::PhysicalDevice physical, VkSurfaceKHR surface, | ||||
|                       const vk::InstanceDispatch& dld); | ||||
|     ~VKDevice(); | ||||
|  | ||||
|     /// Initializes the device. Returns true on success. | ||||
|     bool Create(); | ||||
|  | ||||
|     /** | ||||
|      * Returns a format supported by the device for the passed requeriments. | ||||
|      * @param wanted_format The ideal format to be returned. It may not be the returned format. | ||||
| @@ -82,11 +79,6 @@ public: | ||||
|         return present_family; | ||||
|     } | ||||
|  | ||||
|     /// Returns the current instance Vulkan API version in Vulkan-formatted version numbers. | ||||
|     u32 InstanceApiVersion() const { | ||||
|         return instance_version; | ||||
|     } | ||||
|  | ||||
|     /// Returns the current Vulkan API version provided in Vulkan-formatted version numbers. | ||||
|     u32 ApiVersion() const { | ||||
|         return properties.apiVersion; | ||||
| @@ -232,10 +224,10 @@ public: | ||||
|         return use_asynchronous_shaders; | ||||
|     } | ||||
|  | ||||
|     /// Checks if the physical device is suitable. | ||||
|     static bool IsSuitable(vk::PhysicalDevice physical, VkSurfaceKHR surface); | ||||
|  | ||||
| private: | ||||
|     /// Checks if the physical device is suitable. | ||||
|     void CheckSuitability() const; | ||||
|  | ||||
|     /// Loads extensions into a vector and stores available ones in this object. | ||||
|     std::vector<const char*> LoadExtensions(); | ||||
|  | ||||
|   | ||||
| @@ -10,7 +10,7 @@ | ||||
| #include "video_core/renderer_vulkan/vk_fence_manager.h" | ||||
| #include "video_core/renderer_vulkan/vk_scheduler.h" | ||||
| #include "video_core/renderer_vulkan/vk_texture_cache.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -9,7 +9,7 @@ | ||||
| #include "video_core/fence_manager.h" | ||||
| #include "video_core/renderer_vulkan/vk_buffer_cache.h" | ||||
| #include "video_core/renderer_vulkan/vk_texture_cache.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Core { | ||||
| class System; | ||||
|   | ||||
| @@ -17,7 +17,7 @@ | ||||
| #include "video_core/renderer_vulkan/vk_pipeline_cache.h" | ||||
| #include "video_core/renderer_vulkan/vk_scheduler.h" | ||||
| #include "video_core/renderer_vulkan/vk_update_descriptor.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -13,7 +13,7 @@ | ||||
| #include "video_core/renderer_vulkan/fixed_pipeline_state.h" | ||||
| #include "video_core/renderer_vulkan/vk_descriptor_pool.h" | ||||
| #include "video_core/renderer_vulkan/vk_shader_decompiler.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
| #include "core/settings.h" | ||||
| #include "video_core/renderer_vulkan/vk_device.h" | ||||
| #include "video_core/renderer_vulkan/vk_master_semaphore.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
| #include <thread> | ||||
|  | ||||
| #include "common/common_types.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -13,7 +13,7 @@ | ||||
| #include "common/logging/log.h" | ||||
| #include "video_core/renderer_vulkan/vk_device.h" | ||||
| #include "video_core/renderer_vulkan/vk_memory_manager.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -9,7 +9,7 @@ | ||||
| #include <utility> | ||||
| #include <vector> | ||||
| #include "common/common_types.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -25,11 +25,11 @@ | ||||
| #include "video_core/renderer_vulkan/vk_rasterizer.h" | ||||
| #include "video_core/renderer_vulkan/vk_scheduler.h" | ||||
| #include "video_core/renderer_vulkan/vk_update_descriptor.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/shader/compiler_settings.h" | ||||
| #include "video_core/shader/memory_util.h" | ||||
| #include "video_core/shader_cache.h" | ||||
| #include "video_core/shader_notify.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -20,12 +20,12 @@ | ||||
| #include "video_core/renderer_vulkan/fixed_pipeline_state.h" | ||||
| #include "video_core/renderer_vulkan/vk_graphics_pipeline.h" | ||||
| #include "video_core/renderer_vulkan/vk_shader_decompiler.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/shader/async_shaders.h" | ||||
| #include "video_core/shader/memory_util.h" | ||||
| #include "video_core/shader/registry.h" | ||||
| #include "video_core/shader/shader_ir.h" | ||||
| #include "video_core/shader_cache.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Core { | ||||
| class System; | ||||
|   | ||||
| @@ -11,7 +11,7 @@ | ||||
| #include "video_core/renderer_vulkan/vk_query_cache.h" | ||||
| #include "video_core/renderer_vulkan/vk_resource_pool.h" | ||||
| #include "video_core/renderer_vulkan/vk_scheduler.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -12,7 +12,7 @@ | ||||
| #include "common/common_types.h" | ||||
| #include "video_core/query_cache.h" | ||||
| #include "video_core/renderer_vulkan/vk_resource_pool.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace VideoCore { | ||||
| class RasterizerInterface; | ||||
|   | ||||
| @@ -36,9 +36,9 @@ | ||||
| #include "video_core/renderer_vulkan/vk_state_tracker.h" | ||||
| #include "video_core/renderer_vulkan/vk_texture_cache.h" | ||||
| #include "video_core/renderer_vulkan/vk_update_descriptor.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/shader_cache.h" | ||||
| #include "video_core/texture_cache/texture_cache.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -29,8 +29,8 @@ | ||||
| #include "video_core/renderer_vulkan/vk_stream_buffer.h" | ||||
| #include "video_core/renderer_vulkan/vk_texture_cache.h" | ||||
| #include "video_core/renderer_vulkan/vk_update_descriptor.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/shader/async_shaders.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Core { | ||||
| class System; | ||||
|   | ||||
| @@ -17,7 +17,7 @@ | ||||
| #include "video_core/renderer_vulkan/vk_scheduler.h" | ||||
| #include "video_core/renderer_vulkan/vk_state_tracker.h" | ||||
| #include "video_core/renderer_vulkan/vk_texture_cache.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -12,7 +12,7 @@ | ||||
| #include <utility> | ||||
| #include "common/common_types.h" | ||||
| #include "common/threadsafe_queue.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -272,19 +272,12 @@ bool IsPrecise(Operation operand) { | ||||
|     return false; | ||||
| } | ||||
|  | ||||
| u32 ShaderVersion(const VKDevice& device) { | ||||
|     if (device.InstanceApiVersion() < VK_API_VERSION_1_1) { | ||||
|         return 0x00010000; | ||||
|     } | ||||
|     return 0x00010300; | ||||
| } | ||||
|  | ||||
| class SPIRVDecompiler final : public Sirit::Module { | ||||
| public: | ||||
|     explicit SPIRVDecompiler(const VKDevice& device_, const ShaderIR& ir_, ShaderType stage_, | ||||
|                              const Registry& registry_, const Specialization& specialization_) | ||||
|         : Module(ShaderVersion(device_)), device{device_}, ir{ir_}, stage{stage_}, | ||||
|           header{ir_.GetHeader()}, registry{registry_}, specialization{specialization_} { | ||||
|         : Module(0x00010300), device{device_}, ir{ir_}, stage{stage_}, header{ir_.GetHeader()}, | ||||
|           registry{registry_}, specialization{specialization_} { | ||||
|         if (stage_ != ShaderType::Compute) { | ||||
|             transform_feedback = BuildTransformFeedback(registry_.GetGraphicsInfo()); | ||||
|         } | ||||
|   | ||||
| @@ -9,7 +9,7 @@ | ||||
| #include "common/common_types.h" | ||||
| #include "video_core/renderer_vulkan/vk_device.h" | ||||
| #include "video_core/renderer_vulkan/vk_shader_util.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -7,7 +7,7 @@ | ||||
| #include <span> | ||||
|  | ||||
| #include "common/common_types.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -12,7 +12,7 @@ | ||||
| #include "video_core/renderer_vulkan/vk_device.h" | ||||
| #include "video_core/renderer_vulkan/vk_scheduler.h" | ||||
| #include "video_core/renderer_vulkan/vk_staging_buffer_pool.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -10,7 +10,7 @@ | ||||
| #include "common/common_types.h" | ||||
|  | ||||
| #include "video_core/renderer_vulkan/vk_memory_manager.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -13,7 +13,7 @@ | ||||
| #include "video_core/renderer_vulkan/vk_device.h" | ||||
| #include "video_core/renderer_vulkan/vk_scheduler.h" | ||||
| #include "video_core/renderer_vulkan/vk_stream_buffer.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -9,7 +9,7 @@ | ||||
| #include <vector> | ||||
|  | ||||
| #include "common/common_types.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -14,7 +14,7 @@ | ||||
| #include "video_core/renderer_vulkan/vk_device.h" | ||||
| #include "video_core/renderer_vulkan/vk_scheduler.h" | ||||
| #include "video_core/renderer_vulkan/vk_swapchain.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -7,7 +7,7 @@ | ||||
| #include <vector> | ||||
|  | ||||
| #include "common/common_types.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Layout { | ||||
| struct FramebufferLayout; | ||||
|   | ||||
| @@ -14,7 +14,7 @@ | ||||
| #include "video_core/renderer_vulkan/vk_scheduler.h" | ||||
| #include "video_core/renderer_vulkan/vk_staging_buffer_pool.h" | ||||
| #include "video_core/renderer_vulkan/vk_texture_cache.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -8,8 +8,8 @@ | ||||
| #include <span> | ||||
|  | ||||
| #include "video_core/renderer_vulkan/vk_memory_manager.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/texture_cache/texture_cache.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -10,7 +10,7 @@ | ||||
| #include "video_core/renderer_vulkan/vk_device.h" | ||||
| #include "video_core/renderer_vulkan/vk_scheduler.h" | ||||
| #include "video_core/renderer_vulkan/vk_update_descriptor.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
| #include <boost/container/static_vector.hpp> | ||||
|  | ||||
| #include "common/common_types.h" | ||||
| #include "video_core/renderer_vulkan/wrapper.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
|   | ||||
							
								
								
									
										45
									
								
								src/video_core/vulkan_common/vulkan_debug_callback.cpp
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										45
									
								
								src/video_core/vulkan_common/vulkan_debug_callback.cpp
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,45 @@ | ||||
| // Copyright 2020 yuzu Emulator Project | ||||
| // Licensed under GPLv2 or any later version | ||||
| // Refer to the license.txt file included. | ||||
|  | ||||
| #include <string_view> | ||||
| #include "common/logging/log.h" | ||||
| #include "video_core/vulkan_common/vulkan_debug_callback.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
| namespace { | ||||
| VkBool32 Callback(VkDebugUtilsMessageSeverityFlagBitsEXT severity, | ||||
|                   VkDebugUtilsMessageTypeFlagsEXT type, | ||||
|                   const VkDebugUtilsMessengerCallbackDataEXT* data, | ||||
|                   [[maybe_unused]] void* user_data) { | ||||
|     const std::string_view message{data->pMessage}; | ||||
|     if (severity & VK_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT) { | ||||
|         LOG_CRITICAL(Render_Vulkan, "{}", message); | ||||
|     } else if (severity & VK_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT) { | ||||
|         LOG_WARNING(Render_Vulkan, "{}", message); | ||||
|     } else if (severity & VK_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT) { | ||||
|         LOG_INFO(Render_Vulkan, "{}", message); | ||||
|     } else if (severity & VK_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT) { | ||||
|         LOG_DEBUG(Render_Vulkan, "{}", message); | ||||
|     } | ||||
|     return VK_FALSE; | ||||
| } | ||||
| } // Anonymous namespace | ||||
|  | ||||
| vk::DebugUtilsMessenger CreateDebugCallback(const vk::Instance& instance) { | ||||
|     return instance.CreateDebugUtilsMessenger(VkDebugUtilsMessengerCreateInfoEXT{ | ||||
|         .sType = VK_STRUCTURE_TYPE_DEBUG_UTILS_MESSENGER_CREATE_INFO_EXT, | ||||
|         .pNext = nullptr, | ||||
|         .flags = 0, | ||||
|         .messageSeverity = VK_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT | | ||||
|                            VK_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT | | ||||
|                            VK_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT | | ||||
|                            VK_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, | ||||
|         .messageType = VK_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT | | ||||
|                        VK_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT | | ||||
|                        VK_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, | ||||
|         .pfnUserCallback = Callback, | ||||
|     }); | ||||
| } | ||||
|  | ||||
| } // namespace Vulkan | ||||
							
								
								
									
										11
									
								
								src/video_core/vulkan_common/vulkan_debug_callback.h
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										11
									
								
								src/video_core/vulkan_common/vulkan_debug_callback.h
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| // Copyright 2020 yuzu Emulator Project | ||||
| // Licensed under GPLv2 or any later version | ||||
| // Refer to the license.txt file included. | ||||
|  | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
| vk::DebugUtilsMessenger CreateDebugCallback(const vk::Instance& instance); | ||||
|  | ||||
| } // namespace Vulkan | ||||
							
								
								
									
										151
									
								
								src/video_core/vulkan_common/vulkan_instance.cpp
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										151
									
								
								src/video_core/vulkan_common/vulkan_instance.cpp
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,151 @@ | ||||
| // Copyright 2020 yuzu Emulator Project | ||||
| // Licensed under GPLv2 or any later version | ||||
| // Refer to the license.txt file included. | ||||
|  | ||||
| #include <algorithm> | ||||
| #include <optional> | ||||
| #include <span> | ||||
| #include <utility> | ||||
| #include <vector> | ||||
|  | ||||
| #include "common/common_types.h" | ||||
| #include "common/dynamic_library.h" | ||||
| #include "common/logging/log.h" | ||||
| #include "core/frontend/emu_window.h" | ||||
| #include "video_core/vulkan_common/vulkan_instance.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| // Include these late to avoid polluting previous headers | ||||
| #ifdef _WIN32 | ||||
| #include <windows.h> | ||||
| // ensure include order | ||||
| #include <vulkan/vulkan_win32.h> | ||||
| #endif | ||||
|  | ||||
| #if !defined(_WIN32) && !defined(__APPLE__) | ||||
| #include <X11/Xlib.h> | ||||
| #include <vulkan/vulkan_wayland.h> | ||||
| #include <vulkan/vulkan_xlib.h> | ||||
| #endif | ||||
|  | ||||
| namespace Vulkan { | ||||
| namespace { | ||||
| [[nodiscard]] std::vector<const char*> RequiredExtensions( | ||||
|     Core::Frontend::WindowSystemType window_type, bool enable_debug_utils) { | ||||
|     std::vector<const char*> extensions; | ||||
|     extensions.reserve(6); | ||||
|     switch (window_type) { | ||||
|     case Core::Frontend::WindowSystemType::Headless: | ||||
|         break; | ||||
| #ifdef _WIN32 | ||||
|     case Core::Frontend::WindowSystemType::Windows: | ||||
|         extensions.push_back(VK_KHR_WIN32_SURFACE_EXTENSION_NAME); | ||||
|         break; | ||||
| #endif | ||||
| #if !defined(_WIN32) && !defined(__APPLE__) | ||||
|     case Core::Frontend::WindowSystemType::X11: | ||||
|         extensions.push_back(VK_KHR_XLIB_SURFACE_EXTENSION_NAME); | ||||
|         break; | ||||
|     case Core::Frontend::WindowSystemType::Wayland: | ||||
|         extensions.push_back(VK_KHR_WAYLAND_SURFACE_EXTENSION_NAME); | ||||
|         break; | ||||
| #endif | ||||
|     default: | ||||
|         LOG_ERROR(Render_Vulkan, "Presentation not supported on this platform"); | ||||
|         break; | ||||
|     } | ||||
|     if (window_type != Core::Frontend::WindowSystemType::Headless) { | ||||
|         extensions.push_back(VK_KHR_SURFACE_EXTENSION_NAME); | ||||
|     } | ||||
|     if (enable_debug_utils) { | ||||
|         extensions.push_back(VK_EXT_DEBUG_UTILS_EXTENSION_NAME); | ||||
|     } | ||||
|     extensions.push_back(VK_KHR_GET_PHYSICAL_DEVICE_PROPERTIES_2_EXTENSION_NAME); | ||||
|     return extensions; | ||||
| } | ||||
|  | ||||
| [[nodiscard]] bool AreExtensionsSupported(const vk::InstanceDispatch& dld, | ||||
|                                           std::span<const char* const> extensions) { | ||||
|     const std::optional properties = vk::EnumerateInstanceExtensionProperties(dld); | ||||
|     if (!properties) { | ||||
|         LOG_ERROR(Render_Vulkan, "Failed to query extension properties"); | ||||
|         return false; | ||||
|     } | ||||
|     for (const char* extension : extensions) { | ||||
|         const auto it = std::ranges::find_if(*properties, [extension](const auto& prop) { | ||||
|             return std::strcmp(extension, prop.extensionName) == 0; | ||||
|         }); | ||||
|         if (it == properties->end()) { | ||||
|             LOG_ERROR(Render_Vulkan, "Required instance extension {} is not available", extension); | ||||
|             return false; | ||||
|         } | ||||
|     } | ||||
|     return true; | ||||
| } | ||||
|  | ||||
| [[nodiscard]] std::vector<const char*> Layers(bool enable_layers) { | ||||
|     std::vector<const char*> layers; | ||||
|     if (enable_layers) { | ||||
|         layers.push_back("VK_LAYER_KHRONOS_validation"); | ||||
|     } | ||||
|     return layers; | ||||
| } | ||||
|  | ||||
| void RemoveUnavailableLayers(const vk::InstanceDispatch& dld, std::vector<const char*>& layers) { | ||||
|     const std::optional layer_properties = vk::EnumerateInstanceLayerProperties(dld); | ||||
|     if (!layer_properties) { | ||||
|         LOG_ERROR(Render_Vulkan, "Failed to query layer properties, disabling layers"); | ||||
|         layers.clear(); | ||||
|     } | ||||
|     std::erase_if(layers, [&layer_properties](const char* layer) { | ||||
|         const auto comp = [layer](const VkLayerProperties& layer_property) { | ||||
|             return std::strcmp(layer, layer_property.layerName) == 0; | ||||
|         }; | ||||
|         const auto it = std::ranges::find_if(*layer_properties, comp); | ||||
|         if (it == layer_properties->end()) { | ||||
|             LOG_ERROR(Render_Vulkan, "Layer {} not available, removing it", layer); | ||||
|             return true; | ||||
|         } | ||||
|         return false; | ||||
|     }); | ||||
| } | ||||
| } // Anonymous namespace | ||||
|  | ||||
| vk::Instance CreateInstance(const Common::DynamicLibrary& library, vk::InstanceDispatch& dld, | ||||
|                             u32 required_version, Core::Frontend::WindowSystemType window_type, | ||||
|                             bool enable_debug_utils, bool enable_layers) { | ||||
|     if (!library.IsOpen()) { | ||||
|         LOG_ERROR(Render_Vulkan, "Vulkan library not available"); | ||||
|         throw vk::Exception(VK_ERROR_INITIALIZATION_FAILED); | ||||
|     } | ||||
|     if (!library.GetSymbol("vkGetInstanceProcAddr", &dld.vkGetInstanceProcAddr)) { | ||||
|         LOG_ERROR(Render_Vulkan, "vkGetInstanceProcAddr not present in Vulkan"); | ||||
|         throw vk::Exception(VK_ERROR_INITIALIZATION_FAILED); | ||||
|     } | ||||
|     if (!vk::Load(dld)) { | ||||
|         LOG_ERROR(Render_Vulkan, "Failed to load Vulkan function pointers"); | ||||
|         throw vk::Exception(VK_ERROR_INITIALIZATION_FAILED); | ||||
|     } | ||||
|     const std::vector<const char*> extensions = RequiredExtensions(window_type, enable_debug_utils); | ||||
|     if (!AreExtensionsSupported(dld, extensions)) { | ||||
|         throw vk::Exception(VK_ERROR_EXTENSION_NOT_PRESENT); | ||||
|     } | ||||
|     std::vector<const char*> layers = Layers(enable_layers); | ||||
|     RemoveUnavailableLayers(dld, layers); | ||||
|  | ||||
|     const u32 available_version = vk::AvailableVersion(dld); | ||||
|     if (available_version < required_version) { | ||||
|         LOG_ERROR(Render_Vulkan, "Vulkan {}.{} is not supported, {}.{} is required", | ||||
|                   VK_VERSION_MAJOR(available_version), VK_VERSION_MINOR(available_version), | ||||
|                   VK_VERSION_MAJOR(required_version), VK_VERSION_MINOR(required_version)); | ||||
|         throw vk::Exception(VK_ERROR_INCOMPATIBLE_DRIVER); | ||||
|     } | ||||
|     vk::Instance instance = vk::Instance::Create(required_version, layers, extensions, dld); | ||||
|     if (!vk::Load(*instance, dld)) { | ||||
|         LOG_ERROR(Render_Vulkan, "Failed to load Vulkan instance function pointers"); | ||||
|         throw vk::Exception(VK_ERROR_INITIALIZATION_FAILED); | ||||
|     } | ||||
|     return instance; | ||||
| } | ||||
|  | ||||
| } // namespace Vulkan | ||||
							
								
								
									
										32
									
								
								src/video_core/vulkan_common/vulkan_instance.h
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										32
									
								
								src/video_core/vulkan_common/vulkan_instance.h
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,32 @@ | ||||
| // Copyright 2020 yuzu Emulator Project | ||||
| // Licensed under GPLv2 or any later version | ||||
| // Refer to the license.txt file included. | ||||
|  | ||||
| #pragma once | ||||
|  | ||||
| #include "common/common_types.h" | ||||
| #include "common/dynamic_library.h" | ||||
| #include "core/frontend/emu_window.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
| /** | ||||
|  * Create a Vulkan instance | ||||
|  * | ||||
|  * @param library            Dynamic library to load the Vulkan instance from | ||||
|  * @param dld                Dispatch table to load function pointers into | ||||
|  * @param required_version   Required Vulkan version (for example, VK_API_VERSION_1_1) | ||||
|  * @param window_type        Window system type's enabled extension | ||||
|  * @param enable_debug_utils Whether to enable VK_EXT_debug_utils_extension_name or not | ||||
|  * @param enable_layers      Whether to enable Vulkan validation layers or not | ||||
|  * | ||||
|  * @return A new Vulkan instance | ||||
|  * @throw vk::Exception on failure | ||||
|  */ | ||||
| [[nodiscard]] vk::Instance CreateInstance( | ||||
|     const Common::DynamicLibrary& library, vk::InstanceDispatch& dld, u32 required_version, | ||||
|     Core::Frontend::WindowSystemType window_type = Core::Frontend::WindowSystemType::Headless, | ||||
|     bool enable_debug_utils = false, bool enable_layers = false); | ||||
|  | ||||
| } // namespace Vulkan | ||||
							
								
								
									
										36
									
								
								src/video_core/vulkan_common/vulkan_library.cpp
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										36
									
								
								src/video_core/vulkan_common/vulkan_library.cpp
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,36 @@ | ||||
| // Copyright 2020 yuzu Emulator Project | ||||
| // Licensed under GPLv2 or any later version | ||||
| // Refer to the license.txt file included. | ||||
|  | ||||
| #include <cstdlib> | ||||
| #include <string> | ||||
|  | ||||
| #include "common/dynamic_library.h" | ||||
| #include "common/file_util.h" | ||||
| #include "video_core/vulkan_common/vulkan_library.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
| Common::DynamicLibrary OpenLibrary() { | ||||
|     Common::DynamicLibrary library; | ||||
| #ifdef __APPLE__ | ||||
|     // Check if a path to a specific Vulkan library has been specified. | ||||
|     char* const libvulkan_env = std::getenv("LIBVULKAN_PATH"); | ||||
|     if (!libvulkan_env || !library.Open(libvulkan_env)) { | ||||
|         // Use the libvulkan.dylib from the application bundle. | ||||
|         const std::string filename = | ||||
|             Common::FS::GetBundleDirectory() + "/Contents/Frameworks/libvulkan.dylib"; | ||||
|         library.Open(filename.c_str()); | ||||
|     } | ||||
| #else | ||||
|     std::string filename = Common::DynamicLibrary::GetVersionedFilename("vulkan", 1); | ||||
|     if (!library.Open(filename.c_str())) { | ||||
|         // Android devices may not have libvulkan.so.1, only libvulkan.so. | ||||
|         filename = Common::DynamicLibrary::GetVersionedFilename("vulkan"); | ||||
|         void(library.Open(filename.c_str())); | ||||
|     } | ||||
| #endif | ||||
|     return library; | ||||
| } | ||||
|  | ||||
| } // namespace Vulkan | ||||
							
								
								
									
										13
									
								
								src/video_core/vulkan_common/vulkan_library.h
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										13
									
								
								src/video_core/vulkan_common/vulkan_library.h
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| // Copyright 2020 yuzu Emulator Project | ||||
| // Licensed under GPLv2 or any later version | ||||
| // Refer to the license.txt file included. | ||||
|  | ||||
| #pragma once | ||||
|  | ||||
| #include "common/dynamic_library.h" | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
| Common::DynamicLibrary OpenLibrary(); | ||||
|  | ||||
| } // namespace Vulkan | ||||
							
								
								
									
										81
									
								
								src/video_core/vulkan_common/vulkan_surface.cpp
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										81
									
								
								src/video_core/vulkan_common/vulkan_surface.cpp
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,81 @@ | ||||
| // Copyright 2020 yuzu Emulator Project | ||||
| // Licensed under GPLv2 or any later version | ||||
| // Refer to the license.txt file included. | ||||
|  | ||||
| #include "common/logging/log.h" | ||||
| #include "core/frontend/emu_window.h" | ||||
| #include "video_core/vulkan_common/vulkan_surface.h" | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| // Include these late to avoid polluting previous headers | ||||
| #ifdef _WIN32 | ||||
| #include <windows.h> | ||||
| // ensure include order | ||||
| #include <vulkan/vulkan_win32.h> | ||||
| #endif | ||||
|  | ||||
| #if !defined(_WIN32) && !defined(__APPLE__) | ||||
| #include <X11/Xlib.h> | ||||
| #include <vulkan/vulkan_wayland.h> | ||||
| #include <vulkan/vulkan_xlib.h> | ||||
| #endif | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
| vk::SurfaceKHR CreateSurface(const vk::Instance& instance, | ||||
|                              const Core::Frontend::EmuWindow& emu_window) { | ||||
|     [[maybe_unused]] const vk::InstanceDispatch& dld = instance.Dispatch(); | ||||
|     [[maybe_unused]] const auto& window_info = emu_window.GetWindowInfo(); | ||||
|     VkSurfaceKHR unsafe_surface = nullptr; | ||||
|  | ||||
| #ifdef _WIN32 | ||||
|     if (window_info.type == Core::Frontend::WindowSystemType::Windows) { | ||||
|         const HWND hWnd = static_cast<HWND>(window_info.render_surface); | ||||
|         const VkWin32SurfaceCreateInfoKHR win32_ci{VK_STRUCTURE_TYPE_WIN32_SURFACE_CREATE_INFO_KHR, | ||||
|                                                    nullptr, 0, nullptr, hWnd}; | ||||
|         const auto vkCreateWin32SurfaceKHR = reinterpret_cast<PFN_vkCreateWin32SurfaceKHR>( | ||||
|             dld.vkGetInstanceProcAddr(*instance, "vkCreateWin32SurfaceKHR")); | ||||
|         if (!vkCreateWin32SurfaceKHR || | ||||
|             vkCreateWin32SurfaceKHR(*instance, &win32_ci, nullptr, &unsafe_surface) != VK_SUCCESS) { | ||||
|             LOG_ERROR(Render_Vulkan, "Failed to initialize Win32 surface"); | ||||
|             throw vk::Exception(VK_ERROR_INITIALIZATION_FAILED); | ||||
|         } | ||||
|     } | ||||
| #endif | ||||
| #if !defined(_WIN32) && !defined(__APPLE__) | ||||
|     if (window_info.type == Core::Frontend::WindowSystemType::X11) { | ||||
|         const VkXlibSurfaceCreateInfoKHR xlib_ci{ | ||||
|             VK_STRUCTURE_TYPE_XLIB_SURFACE_CREATE_INFO_KHR, nullptr, 0, | ||||
|             static_cast<Display*>(window_info.display_connection), | ||||
|             reinterpret_cast<Window>(window_info.render_surface)}; | ||||
|         const auto vkCreateXlibSurfaceKHR = reinterpret_cast<PFN_vkCreateXlibSurfaceKHR>( | ||||
|             dld.vkGetInstanceProcAddr(*instance, "vkCreateXlibSurfaceKHR")); | ||||
|         if (!vkCreateXlibSurfaceKHR || | ||||
|             vkCreateXlibSurfaceKHR(*instance, &xlib_ci, nullptr, &unsafe_surface) != VK_SUCCESS) { | ||||
|             LOG_ERROR(Render_Vulkan, "Failed to initialize Xlib surface"); | ||||
|             throw vk::Exception(VK_ERROR_INITIALIZATION_FAILED); | ||||
|         } | ||||
|     } | ||||
|     if (window_info.type == Core::Frontend::WindowSystemType::Wayland) { | ||||
|         const VkWaylandSurfaceCreateInfoKHR wayland_ci{ | ||||
|             VK_STRUCTURE_TYPE_WAYLAND_SURFACE_CREATE_INFO_KHR, nullptr, 0, | ||||
|             static_cast<wl_display*>(window_info.display_connection), | ||||
|             static_cast<wl_surface*>(window_info.render_surface)}; | ||||
|         const auto vkCreateWaylandSurfaceKHR = reinterpret_cast<PFN_vkCreateWaylandSurfaceKHR>( | ||||
|             dld.vkGetInstanceProcAddr(*instance, "vkCreateWaylandSurfaceKHR")); | ||||
|         if (!vkCreateWaylandSurfaceKHR || | ||||
|             vkCreateWaylandSurfaceKHR(*instance, &wayland_ci, nullptr, &unsafe_surface) != | ||||
|                 VK_SUCCESS) { | ||||
|             LOG_ERROR(Render_Vulkan, "Failed to initialize Wayland surface"); | ||||
|             throw vk::Exception(VK_ERROR_INITIALIZATION_FAILED); | ||||
|         } | ||||
|     } | ||||
| #endif | ||||
|     if (!unsafe_surface) { | ||||
|         LOG_ERROR(Render_Vulkan, "Presentation not supported on this platform"); | ||||
|         throw vk::Exception(VK_ERROR_INITIALIZATION_FAILED); | ||||
|     } | ||||
|     return vk::SurfaceKHR(unsafe_surface, *instance, dld); | ||||
| } | ||||
|  | ||||
| } // namespace Vulkan | ||||
							
								
								
									
										18
									
								
								src/video_core/vulkan_common/vulkan_surface.h
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										18
									
								
								src/video_core/vulkan_common/vulkan_surface.h
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| // Copyright 2020 yuzu Emulator Project | ||||
| // Licensed under GPLv2 or any later version | ||||
| // Refer to the license.txt file included. | ||||
|  | ||||
| #pragma once | ||||
|  | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Core::Frontend { | ||||
| class EmuWindow; | ||||
| } | ||||
|  | ||||
| namespace Vulkan { | ||||
|  | ||||
| [[nodiscard]] vk::SurfaceKHR CreateSurface(const vk::Instance& instance, | ||||
|                                            const Core::Frontend::EmuWindow& emu_window); | ||||
|  | ||||
| } // namespace Vulkan | ||||
							
								
								
									
										900
									
								
								src/video_core/vulkan_common/vulkan_wrapper.cpp
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										900
									
								
								src/video_core/vulkan_common/vulkan_wrapper.cpp
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,900 @@ | ||||
| // Copyright 2020 yuzu Emulator Project | ||||
| // Licensed under GPLv2 or any later version | ||||
| // Refer to the license.txt file included. | ||||
|  | ||||
| #include <algorithm> | ||||
| #include <exception> | ||||
| #include <memory> | ||||
| #include <optional> | ||||
| #include <string_view> | ||||
| #include <utility> | ||||
| #include <vector> | ||||
|  | ||||
| #include "common/common_types.h" | ||||
| #include "common/logging/log.h" | ||||
|  | ||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||
|  | ||||
| namespace Vulkan::vk { | ||||
|  | ||||
| namespace { | ||||
|  | ||||
| template <typename Func> | ||||
| void SortPhysicalDevices(std::vector<VkPhysicalDevice>& devices, const InstanceDispatch& dld, | ||||
|                          Func&& func) { | ||||
|     // Calling GetProperties calls Vulkan more than needed. But they are supposed to be cheap | ||||
|     // functions. | ||||
|     std::stable_sort(devices.begin(), devices.end(), | ||||
|                      [&dld, &func](VkPhysicalDevice lhs, VkPhysicalDevice rhs) { | ||||
|                          return func(vk::PhysicalDevice(lhs, dld).GetProperties(), | ||||
|                                      vk::PhysicalDevice(rhs, dld).GetProperties()); | ||||
|                      }); | ||||
| } | ||||
|  | ||||
| void SortPhysicalDevicesPerVendor(std::vector<VkPhysicalDevice>& devices, | ||||
|                                   const InstanceDispatch& dld, | ||||
|                                   std::initializer_list<u32> vendor_ids) { | ||||
|     for (auto it = vendor_ids.end(); it != vendor_ids.begin();) { | ||||
|         --it; | ||||
|         SortPhysicalDevices(devices, dld, [id = *it](const auto& lhs, const auto& rhs) { | ||||
|             return lhs.vendorID == id && rhs.vendorID != id; | ||||
|         }); | ||||
|     } | ||||
| } | ||||
|  | ||||
| void SortPhysicalDevices(std::vector<VkPhysicalDevice>& devices, const InstanceDispatch& dld) { | ||||
|     // Sort by name, this will set a base and make GPUs with higher numbers appear first | ||||
|     // (e.g. GTX 1650 will intentionally be listed before a GTX 1080). | ||||
|     SortPhysicalDevices(devices, dld, [](const auto& lhs, const auto& rhs) { | ||||
|         return std::string_view{lhs.deviceName} > std::string_view{rhs.deviceName}; | ||||
|     }); | ||||
|     // Prefer discrete over non-discrete | ||||
|     SortPhysicalDevices(devices, dld, [](const auto& lhs, const auto& rhs) { | ||||
|         return lhs.deviceType == VK_PHYSICAL_DEVICE_TYPE_DISCRETE_GPU && | ||||
|                rhs.deviceType != VK_PHYSICAL_DEVICE_TYPE_DISCRETE_GPU; | ||||
|     }); | ||||
|     // Prefer Nvidia over AMD, AMD over Intel, Intel over the rest. | ||||
|     SortPhysicalDevicesPerVendor(devices, dld, {0x10DE, 0x1002, 0x8086}); | ||||
| } | ||||
|  | ||||
| template <typename T> | ||||
| bool Proc(T& result, const InstanceDispatch& dld, const char* proc_name, | ||||
|           VkInstance instance = nullptr) noexcept { | ||||
|     result = reinterpret_cast<T>(dld.vkGetInstanceProcAddr(instance, proc_name)); | ||||
|     return result != nullptr; | ||||
| } | ||||
|  | ||||
| template <typename T> | ||||
| void Proc(T& result, const DeviceDispatch& dld, const char* proc_name, VkDevice device) noexcept { | ||||
|     result = reinterpret_cast<T>(dld.vkGetDeviceProcAddr(device, proc_name)); | ||||
| } | ||||
|  | ||||
| void Load(VkDevice device, DeviceDispatch& dld) noexcept { | ||||
| #define X(name) Proc(dld.name, dld, #name, device) | ||||
|     X(vkAcquireNextImageKHR); | ||||
|     X(vkAllocateCommandBuffers); | ||||
|     X(vkAllocateDescriptorSets); | ||||
|     X(vkAllocateMemory); | ||||
|     X(vkBeginCommandBuffer); | ||||
|     X(vkBindBufferMemory); | ||||
|     X(vkBindImageMemory); | ||||
|     X(vkCmdBeginQuery); | ||||
|     X(vkCmdBeginRenderPass); | ||||
|     X(vkCmdBeginTransformFeedbackEXT); | ||||
|     X(vkCmdBeginDebugUtilsLabelEXT); | ||||
|     X(vkCmdBindDescriptorSets); | ||||
|     X(vkCmdBindIndexBuffer); | ||||
|     X(vkCmdBindPipeline); | ||||
|     X(vkCmdBindTransformFeedbackBuffersEXT); | ||||
|     X(vkCmdBindVertexBuffers); | ||||
|     X(vkCmdBlitImage); | ||||
|     X(vkCmdClearAttachments); | ||||
|     X(vkCmdCopyBuffer); | ||||
|     X(vkCmdCopyBufferToImage); | ||||
|     X(vkCmdCopyImage); | ||||
|     X(vkCmdCopyImageToBuffer); | ||||
|     X(vkCmdDispatch); | ||||
|     X(vkCmdDraw); | ||||
|     X(vkCmdDrawIndexed); | ||||
|     X(vkCmdEndQuery); | ||||
|     X(vkCmdEndRenderPass); | ||||
|     X(vkCmdEndTransformFeedbackEXT); | ||||
|     X(vkCmdEndDebugUtilsLabelEXT); | ||||
|     X(vkCmdFillBuffer); | ||||
|     X(vkCmdPipelineBarrier); | ||||
|     X(vkCmdPushConstants); | ||||
|     X(vkCmdSetBlendConstants); | ||||
|     X(vkCmdSetDepthBias); | ||||
|     X(vkCmdSetDepthBounds); | ||||
|     X(vkCmdSetEvent); | ||||
|     X(vkCmdSetScissor); | ||||
|     X(vkCmdSetStencilCompareMask); | ||||
|     X(vkCmdSetStencilReference); | ||||
|     X(vkCmdSetStencilWriteMask); | ||||
|     X(vkCmdSetViewport); | ||||
|     X(vkCmdWaitEvents); | ||||
|     X(vkCmdBindVertexBuffers2EXT); | ||||
|     X(vkCmdSetCullModeEXT); | ||||
|     X(vkCmdSetDepthBoundsTestEnableEXT); | ||||
|     X(vkCmdSetDepthCompareOpEXT); | ||||
|     X(vkCmdSetDepthTestEnableEXT); | ||||
|     X(vkCmdSetDepthWriteEnableEXT); | ||||
|     X(vkCmdSetFrontFaceEXT); | ||||
|     X(vkCmdSetPrimitiveTopologyEXT); | ||||
|     X(vkCmdSetStencilOpEXT); | ||||
|     X(vkCmdSetStencilTestEnableEXT); | ||||
|     X(vkCmdResolveImage); | ||||
|     X(vkCreateBuffer); | ||||
|     X(vkCreateBufferView); | ||||
|     X(vkCreateCommandPool); | ||||
|     X(vkCreateComputePipelines); | ||||
|     X(vkCreateDescriptorPool); | ||||
|     X(vkCreateDescriptorSetLayout); | ||||
|     X(vkCreateDescriptorUpdateTemplateKHR); | ||||
|     X(vkCreateEvent); | ||||
|     X(vkCreateFence); | ||||
|     X(vkCreateFramebuffer); | ||||
|     X(vkCreateGraphicsPipelines); | ||||
|     X(vkCreateImage); | ||||
|     X(vkCreateImageView); | ||||
|     X(vkCreatePipelineLayout); | ||||
|     X(vkCreateQueryPool); | ||||
|     X(vkCreateRenderPass); | ||||
|     X(vkCreateSampler); | ||||
|     X(vkCreateSemaphore); | ||||
|     X(vkCreateShaderModule); | ||||
|     X(vkCreateSwapchainKHR); | ||||
|     X(vkDestroyBuffer); | ||||
|     X(vkDestroyBufferView); | ||||
|     X(vkDestroyCommandPool); | ||||
|     X(vkDestroyDescriptorPool); | ||||
|     X(vkDestroyDescriptorSetLayout); | ||||
|     X(vkDestroyDescriptorUpdateTemplateKHR); | ||||
|     X(vkDestroyEvent); | ||||
|     X(vkDestroyFence); | ||||
|     X(vkDestroyFramebuffer); | ||||
|     X(vkDestroyImage); | ||||
|     X(vkDestroyImageView); | ||||
|     X(vkDestroyPipeline); | ||||
|     X(vkDestroyPipelineLayout); | ||||
|     X(vkDestroyQueryPool); | ||||
|     X(vkDestroyRenderPass); | ||||
|     X(vkDestroySampler); | ||||
|     X(vkDestroySemaphore); | ||||
|     X(vkDestroyShaderModule); | ||||
|     X(vkDestroySwapchainKHR); | ||||
|     X(vkDeviceWaitIdle); | ||||
|     X(vkEndCommandBuffer); | ||||
|     X(vkFreeCommandBuffers); | ||||
|     X(vkFreeDescriptorSets); | ||||
|     X(vkFreeMemory); | ||||
|     X(vkGetBufferMemoryRequirements); | ||||
|     X(vkGetDeviceQueue); | ||||
|     X(vkGetEventStatus); | ||||
|     X(vkGetFenceStatus); | ||||
|     X(vkGetImageMemoryRequirements); | ||||
|     X(vkGetQueryPoolResults); | ||||
|     X(vkGetSemaphoreCounterValueKHR); | ||||
|     X(vkMapMemory); | ||||
|     X(vkQueueSubmit); | ||||
|     X(vkResetFences); | ||||
|     X(vkResetQueryPoolEXT); | ||||
|     X(vkSetDebugUtilsObjectNameEXT); | ||||
|     X(vkSetDebugUtilsObjectTagEXT); | ||||
|     X(vkUnmapMemory); | ||||
|     X(vkUpdateDescriptorSetWithTemplateKHR); | ||||
|     X(vkUpdateDescriptorSets); | ||||
|     X(vkWaitForFences); | ||||
|     X(vkWaitSemaphoresKHR); | ||||
| #undef X | ||||
| } | ||||
|  | ||||
| template <typename T> | ||||
| void SetObjectName(const DeviceDispatch* dld, VkDevice device, T handle, VkObjectType type, | ||||
|                    const char* name) { | ||||
|     const VkDebugUtilsObjectNameInfoEXT name_info{ | ||||
|         .sType = VK_STRUCTURE_TYPE_DEBUG_UTILS_OBJECT_NAME_INFO_EXT, | ||||
|         .pNext = nullptr, | ||||
|         .objectType = VK_OBJECT_TYPE_IMAGE, | ||||
|         .objectHandle = reinterpret_cast<u64>(handle), | ||||
|         .pObjectName = name, | ||||
|     }; | ||||
|     Check(dld->vkSetDebugUtilsObjectNameEXT(device, &name_info)); | ||||
| } | ||||
|  | ||||
| } // Anonymous namespace | ||||
|  | ||||
| bool Load(InstanceDispatch& dld) noexcept { | ||||
| #define X(name) Proc(dld.name, dld, #name) | ||||
|     return X(vkCreateInstance) && X(vkEnumerateInstanceExtensionProperties) && | ||||
|            X(vkEnumerateInstanceLayerProperties); | ||||
| #undef X | ||||
| } | ||||
|  | ||||
| bool Load(VkInstance instance, InstanceDispatch& dld) noexcept { | ||||
| #define X(name) Proc(dld.name, dld, #name, instance) | ||||
|     // These functions may fail to load depending on the enabled extensions. | ||||
|     // Don't return a failure on these. | ||||
|     X(vkCreateDebugUtilsMessengerEXT); | ||||
|     X(vkDestroyDebugUtilsMessengerEXT); | ||||
|     X(vkDestroySurfaceKHR); | ||||
|     X(vkGetPhysicalDeviceFeatures2KHR); | ||||
|     X(vkGetPhysicalDeviceProperties2KHR); | ||||
|     X(vkGetPhysicalDeviceSurfaceCapabilitiesKHR); | ||||
|     X(vkGetPhysicalDeviceSurfaceFormatsKHR); | ||||
|     X(vkGetPhysicalDeviceSurfacePresentModesKHR); | ||||
|     X(vkGetPhysicalDeviceSurfaceSupportKHR); | ||||
|     X(vkGetSwapchainImagesKHR); | ||||
|     X(vkQueuePresentKHR); | ||||
|  | ||||
|     return X(vkCreateDevice) && X(vkDestroyDevice) && X(vkDestroyDevice) && | ||||
|            X(vkEnumerateDeviceExtensionProperties) && X(vkEnumeratePhysicalDevices) && | ||||
|            X(vkGetDeviceProcAddr) && X(vkGetPhysicalDeviceFormatProperties) && | ||||
|            X(vkGetPhysicalDeviceMemoryProperties) && X(vkGetPhysicalDeviceProperties) && | ||||
|            X(vkGetPhysicalDeviceQueueFamilyProperties); | ||||
| #undef X | ||||
| } | ||||
|  | ||||
| const char* Exception::what() const noexcept { | ||||
|     return ToString(result); | ||||
| } | ||||
|  | ||||
| const char* ToString(VkResult result) noexcept { | ||||
|     switch (result) { | ||||
|     case VkResult::VK_SUCCESS: | ||||
|         return "VK_SUCCESS"; | ||||
|     case VkResult::VK_NOT_READY: | ||||
|         return "VK_NOT_READY"; | ||||
|     case VkResult::VK_TIMEOUT: | ||||
|         return "VK_TIMEOUT"; | ||||
|     case VkResult::VK_EVENT_SET: | ||||
|         return "VK_EVENT_SET"; | ||||
|     case VkResult::VK_EVENT_RESET: | ||||
|         return "VK_EVENT_RESET"; | ||||
|     case VkResult::VK_INCOMPLETE: | ||||
|         return "VK_INCOMPLETE"; | ||||
|     case VkResult::VK_ERROR_OUT_OF_HOST_MEMORY: | ||||
|         return "VK_ERROR_OUT_OF_HOST_MEMORY"; | ||||
|     case VkResult::VK_ERROR_OUT_OF_DEVICE_MEMORY: | ||||
|         return "VK_ERROR_OUT_OF_DEVICE_MEMORY"; | ||||
|     case VkResult::VK_ERROR_INITIALIZATION_FAILED: | ||||
|         return "VK_ERROR_INITIALIZATION_FAILED"; | ||||
|     case VkResult::VK_ERROR_DEVICE_LOST: | ||||
|         return "VK_ERROR_DEVICE_LOST"; | ||||
|     case VkResult::VK_ERROR_MEMORY_MAP_FAILED: | ||||
|         return "VK_ERROR_MEMORY_MAP_FAILED"; | ||||
|     case VkResult::VK_ERROR_LAYER_NOT_PRESENT: | ||||
|         return "VK_ERROR_LAYER_NOT_PRESENT"; | ||||
|     case VkResult::VK_ERROR_EXTENSION_NOT_PRESENT: | ||||
|         return "VK_ERROR_EXTENSION_NOT_PRESENT"; | ||||
|     case VkResult::VK_ERROR_FEATURE_NOT_PRESENT: | ||||
|         return "VK_ERROR_FEATURE_NOT_PRESENT"; | ||||
|     case VkResult::VK_ERROR_INCOMPATIBLE_DRIVER: | ||||
|         return "VK_ERROR_INCOMPATIBLE_DRIVER"; | ||||
|     case VkResult::VK_ERROR_TOO_MANY_OBJECTS: | ||||
|         return "VK_ERROR_TOO_MANY_OBJECTS"; | ||||
|     case VkResult::VK_ERROR_FORMAT_NOT_SUPPORTED: | ||||
|         return "VK_ERROR_FORMAT_NOT_SUPPORTED"; | ||||
|     case VkResult::VK_ERROR_FRAGMENTED_POOL: | ||||
|         return "VK_ERROR_FRAGMENTED_POOL"; | ||||
|     case VkResult::VK_ERROR_OUT_OF_POOL_MEMORY: | ||||
|         return "VK_ERROR_OUT_OF_POOL_MEMORY"; | ||||
|     case VkResult::VK_ERROR_INVALID_EXTERNAL_HANDLE: | ||||
|         return "VK_ERROR_INVALID_EXTERNAL_HANDLE"; | ||||
|     case VkResult::VK_ERROR_SURFACE_LOST_KHR: | ||||
|         return "VK_ERROR_SURFACE_LOST_KHR"; | ||||
|     case VkResult::VK_ERROR_NATIVE_WINDOW_IN_USE_KHR: | ||||
|         return "VK_ERROR_NATIVE_WINDOW_IN_USE_KHR"; | ||||
|     case VkResult::VK_SUBOPTIMAL_KHR: | ||||
|         return "VK_SUBOPTIMAL_KHR"; | ||||
|     case VkResult::VK_ERROR_OUT_OF_DATE_KHR: | ||||
|         return "VK_ERROR_OUT_OF_DATE_KHR"; | ||||
|     case VkResult::VK_ERROR_INCOMPATIBLE_DISPLAY_KHR: | ||||
|         return "VK_ERROR_INCOMPATIBLE_DISPLAY_KHR"; | ||||
|     case VkResult::VK_ERROR_VALIDATION_FAILED_EXT: | ||||
|         return "VK_ERROR_VALIDATION_FAILED_EXT"; | ||||
|     case VkResult::VK_ERROR_INVALID_SHADER_NV: | ||||
|         return "VK_ERROR_INVALID_SHADER_NV"; | ||||
|     case VkResult::VK_ERROR_INVALID_DRM_FORMAT_MODIFIER_PLANE_LAYOUT_EXT: | ||||
|         return "VK_ERROR_INVALID_DRM_FORMAT_MODIFIER_PLANE_LAYOUT_EXT"; | ||||
|     case VkResult::VK_ERROR_FRAGMENTATION_EXT: | ||||
|         return "VK_ERROR_FRAGMENTATION_EXT"; | ||||
|     case VkResult::VK_ERROR_NOT_PERMITTED_EXT: | ||||
|         return "VK_ERROR_NOT_PERMITTED_EXT"; | ||||
|     case VkResult::VK_ERROR_INVALID_DEVICE_ADDRESS_EXT: | ||||
|         return "VK_ERROR_INVALID_DEVICE_ADDRESS_EXT"; | ||||
|     case VkResult::VK_ERROR_FULL_SCREEN_EXCLUSIVE_MODE_LOST_EXT: | ||||
|         return "VK_ERROR_FULL_SCREEN_EXCLUSIVE_MODE_LOST_EXT"; | ||||
|     case VkResult::VK_ERROR_UNKNOWN: | ||||
|         return "VK_ERROR_UNKNOWN"; | ||||
|     case VkResult::VK_ERROR_INCOMPATIBLE_VERSION_KHR: | ||||
|         return "VK_ERROR_INCOMPATIBLE_VERSION_KHR"; | ||||
|     case VkResult::VK_THREAD_IDLE_KHR: | ||||
|         return "VK_THREAD_IDLE_KHR"; | ||||
|     case VkResult::VK_THREAD_DONE_KHR: | ||||
|         return "VK_THREAD_DONE_KHR"; | ||||
|     case VkResult::VK_OPERATION_DEFERRED_KHR: | ||||
|         return "VK_OPERATION_DEFERRED_KHR"; | ||||
|     case VkResult::VK_OPERATION_NOT_DEFERRED_KHR: | ||||
|         return "VK_OPERATION_NOT_DEFERRED_KHR"; | ||||
|     case VkResult::VK_PIPELINE_COMPILE_REQUIRED_EXT: | ||||
|         return "VK_PIPELINE_COMPILE_REQUIRED_EXT"; | ||||
|     case VkResult::VK_RESULT_MAX_ENUM: | ||||
|         return "VK_RESULT_MAX_ENUM"; | ||||
|     } | ||||
|     return "Unknown"; | ||||
| } | ||||
|  | ||||
| void Destroy(VkInstance instance, const InstanceDispatch& dld) noexcept { | ||||
|     dld.vkDestroyInstance(instance, nullptr); | ||||
| } | ||||
|  | ||||
| void Destroy(VkDevice device, const InstanceDispatch& dld) noexcept { | ||||
|     dld.vkDestroyDevice(device, nullptr); | ||||
| } | ||||
|  | ||||
| void Destroy(VkDevice device, VkBuffer handle, const DeviceDispatch& dld) noexcept { | ||||
|     dld.vkDestroyBuffer(device, handle, nullptr); | ||||
| } | ||||
|  | ||||
| void Destroy(VkDevice device, VkBufferView handle, const DeviceDispatch& dld) noexcept { | ||||
|     dld.vkDestroyBufferView(device, handle, nullptr); | ||||
| } | ||||
|  | ||||
| void Destroy(VkDevice device, VkCommandPool handle, const DeviceDispatch& dld) noexcept { | ||||
|     dld.vkDestroyCommandPool(device, handle, nullptr); | ||||
| } | ||||
|  | ||||
| void Destroy(VkDevice device, VkDescriptorPool handle, const DeviceDispatch& dld) noexcept { | ||||
|     dld.vkDestroyDescriptorPool(device, handle, nullptr); | ||||
| } | ||||
|  | ||||
| void Destroy(VkDevice device, VkDescriptorSetLayout handle, const DeviceDispatch& dld) noexcept { | ||||
|     dld.vkDestroyDescriptorSetLayout(device, handle, nullptr); | ||||
| } | ||||
|  | ||||
| void Destroy(VkDevice device, VkDescriptorUpdateTemplateKHR handle, | ||||
|              const DeviceDispatch& dld) noexcept { | ||||
|     dld.vkDestroyDescriptorUpdateTemplateKHR(device, handle, nullptr); | ||||
| } | ||||
|  | ||||
| void Destroy(VkDevice device, VkDeviceMemory handle, const DeviceDispatch& dld) noexcept { | ||||
|     dld.vkFreeMemory(device, handle, nullptr); | ||||
| } | ||||
|  | ||||
| void Destroy(VkDevice device, VkEvent handle, const DeviceDispatch& dld) noexcept { | ||||
|     dld.vkDestroyEvent(device, handle, nullptr); | ||||
| } | ||||
|  | ||||
| void Destroy(VkDevice device, VkFence handle, const DeviceDispatch& dld) noexcept { | ||||
|     dld.vkDestroyFence(device, handle, nullptr); | ||||
| } | ||||
|  | ||||
| void Destroy(VkDevice device, VkFramebuffer handle, const DeviceDispatch& dld) noexcept { | ||||
|     dld.vkDestroyFramebuffer(device, handle, nullptr); | ||||
| } | ||||
|  | ||||
| void Destroy(VkDevice device, VkImage handle, const DeviceDispatch& dld) noexcept { | ||||
|     dld.vkDestroyImage(device, handle, nullptr); | ||||
| } | ||||
|  | ||||
| void Destroy(VkDevice device, VkImageView handle, const DeviceDispatch& dld) noexcept { | ||||
|     dld.vkDestroyImageView(device, handle, nullptr); | ||||
| } | ||||
|  | ||||
| void Destroy(VkDevice device, VkPipeline handle, const DeviceDispatch& dld) noexcept { | ||||
|     dld.vkDestroyPipeline(device, handle, nullptr); | ||||
| } | ||||
|  | ||||
| void Destroy(VkDevice device, VkPipelineLayout handle, const DeviceDispatch& dld) noexcept { | ||||
|     dld.vkDestroyPipelineLayout(device, handle, nullptr); | ||||
| } | ||||
|  | ||||
| void Destroy(VkDevice device, VkQueryPool handle, const DeviceDispatch& dld) noexcept { | ||||
|     dld.vkDestroyQueryPool(device, handle, nullptr); | ||||
| } | ||||
|  | ||||
| void Destroy(VkDevice device, VkRenderPass handle, const DeviceDispatch& dld) noexcept { | ||||
|     dld.vkDestroyRenderPass(device, handle, nullptr); | ||||
| } | ||||
|  | ||||
| void Destroy(VkDevice device, VkSampler handle, const DeviceDispatch& dld) noexcept { | ||||
|     dld.vkDestroySampler(device, handle, nullptr); | ||||
| } | ||||
|  | ||||
| void Destroy(VkDevice device, VkSwapchainKHR handle, const DeviceDispatch& dld) noexcept { | ||||
|     dld.vkDestroySwapchainKHR(device, handle, nullptr); | ||||
| } | ||||
|  | ||||
| void Destroy(VkDevice device, VkSemaphore handle, const DeviceDispatch& dld) noexcept { | ||||
|     dld.vkDestroySemaphore(device, handle, nullptr); | ||||
| } | ||||
|  | ||||
| void Destroy(VkDevice device, VkShaderModule handle, const DeviceDispatch& dld) noexcept { | ||||
|     dld.vkDestroyShaderModule(device, handle, nullptr); | ||||
| } | ||||
|  | ||||
| void Destroy(VkInstance instance, VkDebugUtilsMessengerEXT handle, | ||||
|              const InstanceDispatch& dld) noexcept { | ||||
|     dld.vkDestroyDebugUtilsMessengerEXT(instance, handle, nullptr); | ||||
| } | ||||
|  | ||||
| void Destroy(VkInstance instance, VkSurfaceKHR handle, const InstanceDispatch& dld) noexcept { | ||||
|     dld.vkDestroySurfaceKHR(instance, handle, nullptr); | ||||
| } | ||||
|  | ||||
| VkResult Free(VkDevice device, VkDescriptorPool handle, Span<VkDescriptorSet> sets, | ||||
|               const DeviceDispatch& dld) noexcept { | ||||
|     return dld.vkFreeDescriptorSets(device, handle, sets.size(), sets.data()); | ||||
| } | ||||
|  | ||||
| VkResult Free(VkDevice device, VkCommandPool handle, Span<VkCommandBuffer> buffers, | ||||
|               const DeviceDispatch& dld) noexcept { | ||||
|     dld.vkFreeCommandBuffers(device, handle, buffers.size(), buffers.data()); | ||||
|     return VK_SUCCESS; | ||||
| } | ||||
|  | ||||
| Instance Instance::Create(u32 version, Span<const char*> layers, Span<const char*> extensions, | ||||
|                           InstanceDispatch& dispatch) { | ||||
|     const VkApplicationInfo application_info{ | ||||
|         .sType = VK_STRUCTURE_TYPE_APPLICATION_INFO, | ||||
|         .pNext = nullptr, | ||||
|         .pApplicationName = "yuzu Emulator", | ||||
|         .applicationVersion = VK_MAKE_VERSION(0, 1, 0), | ||||
|         .pEngineName = "yuzu Emulator", | ||||
|         .engineVersion = VK_MAKE_VERSION(0, 1, 0), | ||||
|         .apiVersion = version, | ||||
|     }; | ||||
|     const VkInstanceCreateInfo ci{ | ||||
|         .sType = VK_STRUCTURE_TYPE_INSTANCE_CREATE_INFO, | ||||
|         .pNext = nullptr, | ||||
|         .flags = 0, | ||||
|         .pApplicationInfo = &application_info, | ||||
|         .enabledLayerCount = layers.size(), | ||||
|         .ppEnabledLayerNames = layers.data(), | ||||
|         .enabledExtensionCount = extensions.size(), | ||||
|         .ppEnabledExtensionNames = extensions.data(), | ||||
|     }; | ||||
|     VkInstance instance; | ||||
|     Check(dispatch.vkCreateInstance(&ci, nullptr, &instance)); | ||||
|     if (!Proc(dispatch.vkDestroyInstance, dispatch, "vkDestroyInstance", instance)) { | ||||
|         // We successfully created an instance but the destroy function couldn't be loaded. | ||||
|         // This is a good moment to panic. | ||||
|         throw vk::Exception(VK_ERROR_INITIALIZATION_FAILED); | ||||
|     } | ||||
|     return Instance(instance, dispatch); | ||||
| } | ||||
|  | ||||
| std::vector<VkPhysicalDevice> Instance::EnumeratePhysicalDevices() const { | ||||
|     u32 num; | ||||
|     Check(dld->vkEnumeratePhysicalDevices(handle, &num, nullptr)); | ||||
|     std::vector<VkPhysicalDevice> physical_devices(num); | ||||
|     Check(dld->vkEnumeratePhysicalDevices(handle, &num, physical_devices.data())); | ||||
|     SortPhysicalDevices(physical_devices, *dld); | ||||
|     return physical_devices; | ||||
| } | ||||
|  | ||||
| DebugUtilsMessenger Instance::CreateDebugUtilsMessenger( | ||||
|     const VkDebugUtilsMessengerCreateInfoEXT& create_info) const { | ||||
|     VkDebugUtilsMessengerEXT object; | ||||
|     Check(dld->vkCreateDebugUtilsMessengerEXT(handle, &create_info, nullptr, &object)); | ||||
|     return DebugUtilsMessenger(object, handle, *dld); | ||||
| } | ||||
|  | ||||
| void Buffer::BindMemory(VkDeviceMemory memory, VkDeviceSize offset) const { | ||||
|     Check(dld->vkBindBufferMemory(owner, handle, memory, offset)); | ||||
| } | ||||
|  | ||||
| void Buffer::SetObjectNameEXT(const char* name) const { | ||||
|     SetObjectName(dld, owner, handle, VK_OBJECT_TYPE_BUFFER, name); | ||||
| } | ||||
|  | ||||
| void BufferView::SetObjectNameEXT(const char* name) const { | ||||
|     SetObjectName(dld, owner, handle, VK_OBJECT_TYPE_BUFFER_VIEW, name); | ||||
| } | ||||
|  | ||||
| void Image::BindMemory(VkDeviceMemory memory, VkDeviceSize offset) const { | ||||
|     Check(dld->vkBindImageMemory(owner, handle, memory, offset)); | ||||
| } | ||||
|  | ||||
| void Image::SetObjectNameEXT(const char* name) const { | ||||
|     SetObjectName(dld, owner, handle, VK_OBJECT_TYPE_IMAGE, name); | ||||
| } | ||||
|  | ||||
| void ImageView::SetObjectNameEXT(const char* name) const { | ||||
|     SetObjectName(dld, owner, handle, VK_OBJECT_TYPE_IMAGE_VIEW, name); | ||||
| } | ||||
|  | ||||
| void DeviceMemory::SetObjectNameEXT(const char* name) const { | ||||
|     SetObjectName(dld, owner, handle, VK_OBJECT_TYPE_DEVICE_MEMORY, name); | ||||
| } | ||||
|  | ||||
| void Fence::SetObjectNameEXT(const char* name) const { | ||||
|     SetObjectName(dld, owner, handle, VK_OBJECT_TYPE_FENCE, name); | ||||
| } | ||||
|  | ||||
| void Framebuffer::SetObjectNameEXT(const char* name) const { | ||||
|     SetObjectName(dld, owner, handle, VK_OBJECT_TYPE_FRAMEBUFFER, name); | ||||
| } | ||||
|  | ||||
| DescriptorSets DescriptorPool::Allocate(const VkDescriptorSetAllocateInfo& ai) const { | ||||
|     const std::size_t num = ai.descriptorSetCount; | ||||
|     std::unique_ptr sets = std::make_unique<VkDescriptorSet[]>(num); | ||||
|     switch (const VkResult result = dld->vkAllocateDescriptorSets(owner, &ai, sets.get())) { | ||||
|     case VK_SUCCESS: | ||||
|         return DescriptorSets(std::move(sets), num, owner, handle, *dld); | ||||
|     case VK_ERROR_OUT_OF_POOL_MEMORY: | ||||
|         return {}; | ||||
|     default: | ||||
|         throw Exception(result); | ||||
|     } | ||||
| } | ||||
|  | ||||
| void DescriptorPool::SetObjectNameEXT(const char* name) const { | ||||
|     SetObjectName(dld, owner, handle, VK_OBJECT_TYPE_DESCRIPTOR_POOL, name); | ||||
| } | ||||
|  | ||||
| CommandBuffers CommandPool::Allocate(std::size_t num_buffers, VkCommandBufferLevel level) const { | ||||
|     const VkCommandBufferAllocateInfo ai{ | ||||
|         .sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_ALLOCATE_INFO, | ||||
|         .pNext = nullptr, | ||||
|         .commandPool = handle, | ||||
|         .level = level, | ||||
|         .commandBufferCount = static_cast<u32>(num_buffers), | ||||
|     }; | ||||
|  | ||||
|     std::unique_ptr buffers = std::make_unique<VkCommandBuffer[]>(num_buffers); | ||||
|     switch (const VkResult result = dld->vkAllocateCommandBuffers(owner, &ai, buffers.get())) { | ||||
|     case VK_SUCCESS: | ||||
|         return CommandBuffers(std::move(buffers), num_buffers, owner, handle, *dld); | ||||
|     case VK_ERROR_OUT_OF_POOL_MEMORY: | ||||
|         return {}; | ||||
|     default: | ||||
|         throw Exception(result); | ||||
|     } | ||||
| } | ||||
|  | ||||
| void CommandPool::SetObjectNameEXT(const char* name) const { | ||||
|     SetObjectName(dld, owner, handle, VK_OBJECT_TYPE_COMMAND_POOL, name); | ||||
| } | ||||
|  | ||||
| std::vector<VkImage> SwapchainKHR::GetImages() const { | ||||
|     u32 num; | ||||
|     Check(dld->vkGetSwapchainImagesKHR(owner, handle, &num, nullptr)); | ||||
|     std::vector<VkImage> images(num); | ||||
|     Check(dld->vkGetSwapchainImagesKHR(owner, handle, &num, images.data())); | ||||
|     return images; | ||||
| } | ||||
|  | ||||
| void Event::SetObjectNameEXT(const char* name) const { | ||||
|     SetObjectName(dld, owner, handle, VK_OBJECT_TYPE_EVENT, name); | ||||
| } | ||||
|  | ||||
| void ShaderModule::SetObjectNameEXT(const char* name) const { | ||||
|     SetObjectName(dld, owner, handle, VK_OBJECT_TYPE_SHADER_MODULE, name); | ||||
| } | ||||
|  | ||||
| void Semaphore::SetObjectNameEXT(const char* name) const { | ||||
|     SetObjectName(dld, owner, handle, VK_OBJECT_TYPE_SEMAPHORE, name); | ||||
| } | ||||
|  | ||||
| Device Device::Create(VkPhysicalDevice physical_device, Span<VkDeviceQueueCreateInfo> queues_ci, | ||||
|                       Span<const char*> enabled_extensions, const void* next, | ||||
|                       DeviceDispatch& dispatch) { | ||||
|     const VkDeviceCreateInfo ci{ | ||||
|         .sType = VK_STRUCTURE_TYPE_DEVICE_CREATE_INFO, | ||||
|         .pNext = next, | ||||
|         .flags = 0, | ||||
|         .queueCreateInfoCount = queues_ci.size(), | ||||
|         .pQueueCreateInfos = queues_ci.data(), | ||||
|         .enabledLayerCount = 0, | ||||
|         .ppEnabledLayerNames = nullptr, | ||||
|         .enabledExtensionCount = enabled_extensions.size(), | ||||
|         .ppEnabledExtensionNames = enabled_extensions.data(), | ||||
|         .pEnabledFeatures = nullptr, | ||||
|     }; | ||||
|     VkDevice device; | ||||
|     Check(dispatch.vkCreateDevice(physical_device, &ci, nullptr, &device)); | ||||
|     Load(device, dispatch); | ||||
|     return Device(device, dispatch); | ||||
| } | ||||
|  | ||||
| Queue Device::GetQueue(u32 family_index) const noexcept { | ||||
|     VkQueue queue; | ||||
|     dld->vkGetDeviceQueue(handle, family_index, 0, &queue); | ||||
|     return Queue(queue, *dld); | ||||
| } | ||||
|  | ||||
| Buffer Device::CreateBuffer(const VkBufferCreateInfo& ci) const { | ||||
|     VkBuffer object; | ||||
|     Check(dld->vkCreateBuffer(handle, &ci, nullptr, &object)); | ||||
|     return Buffer(object, handle, *dld); | ||||
| } | ||||
|  | ||||
| BufferView Device::CreateBufferView(const VkBufferViewCreateInfo& ci) const { | ||||
|     VkBufferView object; | ||||
|     Check(dld->vkCreateBufferView(handle, &ci, nullptr, &object)); | ||||
|     return BufferView(object, handle, *dld); | ||||
| } | ||||
|  | ||||
| Image Device::CreateImage(const VkImageCreateInfo& ci) const { | ||||
|     VkImage object; | ||||
|     Check(dld->vkCreateImage(handle, &ci, nullptr, &object)); | ||||
|     return Image(object, handle, *dld); | ||||
| } | ||||
|  | ||||
| ImageView Device::CreateImageView(const VkImageViewCreateInfo& ci) const { | ||||
|     VkImageView object; | ||||
|     Check(dld->vkCreateImageView(handle, &ci, nullptr, &object)); | ||||
|     return ImageView(object, handle, *dld); | ||||
| } | ||||
|  | ||||
| Semaphore Device::CreateSemaphore() const { | ||||
|     static constexpr VkSemaphoreCreateInfo ci{ | ||||
|         .sType = VK_STRUCTURE_TYPE_SEMAPHORE_CREATE_INFO, | ||||
|         .pNext = nullptr, | ||||
|         .flags = 0, | ||||
|     }; | ||||
|     return CreateSemaphore(ci); | ||||
| } | ||||
|  | ||||
| Semaphore Device::CreateSemaphore(const VkSemaphoreCreateInfo& ci) const { | ||||
|     VkSemaphore object; | ||||
|     Check(dld->vkCreateSemaphore(handle, &ci, nullptr, &object)); | ||||
|     return Semaphore(object, handle, *dld); | ||||
| } | ||||
|  | ||||
| Fence Device::CreateFence(const VkFenceCreateInfo& ci) const { | ||||
|     VkFence object; | ||||
|     Check(dld->vkCreateFence(handle, &ci, nullptr, &object)); | ||||
|     return Fence(object, handle, *dld); | ||||
| } | ||||
|  | ||||
| DescriptorPool Device::CreateDescriptorPool(const VkDescriptorPoolCreateInfo& ci) const { | ||||
|     VkDescriptorPool object; | ||||
|     Check(dld->vkCreateDescriptorPool(handle, &ci, nullptr, &object)); | ||||
|     return DescriptorPool(object, handle, *dld); | ||||
| } | ||||
|  | ||||
| RenderPass Device::CreateRenderPass(const VkRenderPassCreateInfo& ci) const { | ||||
|     VkRenderPass object; | ||||
|     Check(dld->vkCreateRenderPass(handle, &ci, nullptr, &object)); | ||||
|     return RenderPass(object, handle, *dld); | ||||
| } | ||||
|  | ||||
| DescriptorSetLayout Device::CreateDescriptorSetLayout( | ||||
|     const VkDescriptorSetLayoutCreateInfo& ci) const { | ||||
|     VkDescriptorSetLayout object; | ||||
|     Check(dld->vkCreateDescriptorSetLayout(handle, &ci, nullptr, &object)); | ||||
|     return DescriptorSetLayout(object, handle, *dld); | ||||
| } | ||||
|  | ||||
| PipelineLayout Device::CreatePipelineLayout(const VkPipelineLayoutCreateInfo& ci) const { | ||||
|     VkPipelineLayout object; | ||||
|     Check(dld->vkCreatePipelineLayout(handle, &ci, nullptr, &object)); | ||||
|     return PipelineLayout(object, handle, *dld); | ||||
| } | ||||
|  | ||||
| Pipeline Device::CreateGraphicsPipeline(const VkGraphicsPipelineCreateInfo& ci) const { | ||||
|     VkPipeline object; | ||||
|     Check(dld->vkCreateGraphicsPipelines(handle, nullptr, 1, &ci, nullptr, &object)); | ||||
|     return Pipeline(object, handle, *dld); | ||||
| } | ||||
|  | ||||
| Pipeline Device::CreateComputePipeline(const VkComputePipelineCreateInfo& ci) const { | ||||
|     VkPipeline object; | ||||
|     Check(dld->vkCreateComputePipelines(handle, nullptr, 1, &ci, nullptr, &object)); | ||||
|     return Pipeline(object, handle, *dld); | ||||
| } | ||||
|  | ||||
| Sampler Device::CreateSampler(const VkSamplerCreateInfo& ci) const { | ||||
|     VkSampler object; | ||||
|     Check(dld->vkCreateSampler(handle, &ci, nullptr, &object)); | ||||
|     return Sampler(object, handle, *dld); | ||||
| } | ||||
|  | ||||
| Framebuffer Device::CreateFramebuffer(const VkFramebufferCreateInfo& ci) const { | ||||
|     VkFramebuffer object; | ||||
|     Check(dld->vkCreateFramebuffer(handle, &ci, nullptr, &object)); | ||||
|     return Framebuffer(object, handle, *dld); | ||||
| } | ||||
|  | ||||
| CommandPool Device::CreateCommandPool(const VkCommandPoolCreateInfo& ci) const { | ||||
|     VkCommandPool object; | ||||
|     Check(dld->vkCreateCommandPool(handle, &ci, nullptr, &object)); | ||||
|     return CommandPool(object, handle, *dld); | ||||
| } | ||||
|  | ||||
| DescriptorUpdateTemplateKHR Device::CreateDescriptorUpdateTemplateKHR( | ||||
|     const VkDescriptorUpdateTemplateCreateInfoKHR& ci) const { | ||||
|     VkDescriptorUpdateTemplateKHR object; | ||||
|     Check(dld->vkCreateDescriptorUpdateTemplateKHR(handle, &ci, nullptr, &object)); | ||||
|     return DescriptorUpdateTemplateKHR(object, handle, *dld); | ||||
| } | ||||
|  | ||||
| QueryPool Device::CreateQueryPool(const VkQueryPoolCreateInfo& ci) const { | ||||
|     VkQueryPool object; | ||||
|     Check(dld->vkCreateQueryPool(handle, &ci, nullptr, &object)); | ||||
|     return QueryPool(object, handle, *dld); | ||||
| } | ||||
|  | ||||
| ShaderModule Device::CreateShaderModule(const VkShaderModuleCreateInfo& ci) const { | ||||
|     VkShaderModule object; | ||||
|     Check(dld->vkCreateShaderModule(handle, &ci, nullptr, &object)); | ||||
|     return ShaderModule(object, handle, *dld); | ||||
| } | ||||
|  | ||||
| Event Device::CreateEvent() const { | ||||
|     static constexpr VkEventCreateInfo ci{ | ||||
|         .sType = VK_STRUCTURE_TYPE_EVENT_CREATE_INFO, | ||||
|         .pNext = nullptr, | ||||
|         .flags = 0, | ||||
|     }; | ||||
|  | ||||
|     VkEvent object; | ||||
|     Check(dld->vkCreateEvent(handle, &ci, nullptr, &object)); | ||||
|     return Event(object, handle, *dld); | ||||
| } | ||||
|  | ||||
| SwapchainKHR Device::CreateSwapchainKHR(const VkSwapchainCreateInfoKHR& ci) const { | ||||
|     VkSwapchainKHR object; | ||||
|     Check(dld->vkCreateSwapchainKHR(handle, &ci, nullptr, &object)); | ||||
|     return SwapchainKHR(object, handle, *dld); | ||||
| } | ||||
|  | ||||
| DeviceMemory Device::TryAllocateMemory(const VkMemoryAllocateInfo& ai) const noexcept { | ||||
|     VkDeviceMemory memory; | ||||
|     if (dld->vkAllocateMemory(handle, &ai, nullptr, &memory) != VK_SUCCESS) { | ||||
|         return {}; | ||||
|     } | ||||
|     return DeviceMemory(memory, handle, *dld); | ||||
| } | ||||
|  | ||||
| DeviceMemory Device::AllocateMemory(const VkMemoryAllocateInfo& ai) const { | ||||
|     VkDeviceMemory memory; | ||||
|     Check(dld->vkAllocateMemory(handle, &ai, nullptr, &memory)); | ||||
|     return DeviceMemory(memory, handle, *dld); | ||||
| } | ||||
|  | ||||
| VkMemoryRequirements Device::GetBufferMemoryRequirements(VkBuffer buffer) const noexcept { | ||||
|     VkMemoryRequirements requirements; | ||||
|     dld->vkGetBufferMemoryRequirements(handle, buffer, &requirements); | ||||
|     return requirements; | ||||
| } | ||||
|  | ||||
| VkMemoryRequirements Device::GetImageMemoryRequirements(VkImage image) const noexcept { | ||||
|     VkMemoryRequirements requirements; | ||||
|     dld->vkGetImageMemoryRequirements(handle, image, &requirements); | ||||
|     return requirements; | ||||
| } | ||||
|  | ||||
| void Device::UpdateDescriptorSets(Span<VkWriteDescriptorSet> writes, | ||||
|                                   Span<VkCopyDescriptorSet> copies) const noexcept { | ||||
|     dld->vkUpdateDescriptorSets(handle, writes.size(), writes.data(), copies.size(), copies.data()); | ||||
| } | ||||
|  | ||||
| VkPhysicalDeviceProperties PhysicalDevice::GetProperties() const noexcept { | ||||
|     VkPhysicalDeviceProperties properties; | ||||
|     dld->vkGetPhysicalDeviceProperties(physical_device, &properties); | ||||
|     return properties; | ||||
| } | ||||
|  | ||||
| void PhysicalDevice::GetProperties2KHR(VkPhysicalDeviceProperties2KHR& properties) const noexcept { | ||||
|     dld->vkGetPhysicalDeviceProperties2KHR(physical_device, &properties); | ||||
| } | ||||
|  | ||||
| VkPhysicalDeviceFeatures PhysicalDevice::GetFeatures() const noexcept { | ||||
|     VkPhysicalDeviceFeatures2KHR features2; | ||||
|     features2.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_FEATURES_2_KHR; | ||||
|     features2.pNext = nullptr; | ||||
|     dld->vkGetPhysicalDeviceFeatures2KHR(physical_device, &features2); | ||||
|     return features2.features; | ||||
| } | ||||
|  | ||||
| void PhysicalDevice::GetFeatures2KHR(VkPhysicalDeviceFeatures2KHR& features) const noexcept { | ||||
|     dld->vkGetPhysicalDeviceFeatures2KHR(physical_device, &features); | ||||
| } | ||||
|  | ||||
| VkFormatProperties PhysicalDevice::GetFormatProperties(VkFormat format) const noexcept { | ||||
|     VkFormatProperties properties; | ||||
|     dld->vkGetPhysicalDeviceFormatProperties(physical_device, format, &properties); | ||||
|     return properties; | ||||
| } | ||||
|  | ||||
| std::vector<VkExtensionProperties> PhysicalDevice::EnumerateDeviceExtensionProperties() const { | ||||
|     u32 num; | ||||
|     dld->vkEnumerateDeviceExtensionProperties(physical_device, nullptr, &num, nullptr); | ||||
|     std::vector<VkExtensionProperties> properties(num); | ||||
|     dld->vkEnumerateDeviceExtensionProperties(physical_device, nullptr, &num, properties.data()); | ||||
|     return properties; | ||||
| } | ||||
|  | ||||
| std::vector<VkQueueFamilyProperties> PhysicalDevice::GetQueueFamilyProperties() const { | ||||
|     u32 num; | ||||
|     dld->vkGetPhysicalDeviceQueueFamilyProperties(physical_device, &num, nullptr); | ||||
|     std::vector<VkQueueFamilyProperties> properties(num); | ||||
|     dld->vkGetPhysicalDeviceQueueFamilyProperties(physical_device, &num, properties.data()); | ||||
|     return properties; | ||||
| } | ||||
|  | ||||
| bool PhysicalDevice::GetSurfaceSupportKHR(u32 queue_family_index, VkSurfaceKHR surface) const { | ||||
|     VkBool32 supported; | ||||
|     Check(dld->vkGetPhysicalDeviceSurfaceSupportKHR(physical_device, queue_family_index, surface, | ||||
|                                                     &supported)); | ||||
|     return supported == VK_TRUE; | ||||
| } | ||||
|  | ||||
| VkSurfaceCapabilitiesKHR PhysicalDevice::GetSurfaceCapabilitiesKHR(VkSurfaceKHR surface) const { | ||||
|     VkSurfaceCapabilitiesKHR capabilities; | ||||
|     Check(dld->vkGetPhysicalDeviceSurfaceCapabilitiesKHR(physical_device, surface, &capabilities)); | ||||
|     return capabilities; | ||||
| } | ||||
|  | ||||
| std::vector<VkSurfaceFormatKHR> PhysicalDevice::GetSurfaceFormatsKHR(VkSurfaceKHR surface) const { | ||||
|     u32 num; | ||||
|     Check(dld->vkGetPhysicalDeviceSurfaceFormatsKHR(physical_device, surface, &num, nullptr)); | ||||
|     std::vector<VkSurfaceFormatKHR> formats(num); | ||||
|     Check( | ||||
|         dld->vkGetPhysicalDeviceSurfaceFormatsKHR(physical_device, surface, &num, formats.data())); | ||||
|     return formats; | ||||
| } | ||||
|  | ||||
| std::vector<VkPresentModeKHR> PhysicalDevice::GetSurfacePresentModesKHR( | ||||
|     VkSurfaceKHR surface) const { | ||||
|     u32 num; | ||||
|     Check(dld->vkGetPhysicalDeviceSurfacePresentModesKHR(physical_device, surface, &num, nullptr)); | ||||
|     std::vector<VkPresentModeKHR> modes(num); | ||||
|     Check(dld->vkGetPhysicalDeviceSurfacePresentModesKHR(physical_device, surface, &num, | ||||
|                                                          modes.data())); | ||||
|     return modes; | ||||
| } | ||||
|  | ||||
| VkPhysicalDeviceMemoryProperties PhysicalDevice::GetMemoryProperties() const noexcept { | ||||
|     VkPhysicalDeviceMemoryProperties properties; | ||||
|     dld->vkGetPhysicalDeviceMemoryProperties(physical_device, &properties); | ||||
|     return properties; | ||||
| } | ||||
|  | ||||
| u32 AvailableVersion(const InstanceDispatch& dld) noexcept { | ||||
|     PFN_vkEnumerateInstanceVersion vkEnumerateInstanceVersion; | ||||
|     if (!Proc(vkEnumerateInstanceVersion, dld, "vkEnumerateInstanceVersion")) { | ||||
|         // If the procedure is not found, Vulkan 1.0 is assumed | ||||
|         return VK_API_VERSION_1_0; | ||||
|     } | ||||
|     u32 version; | ||||
|     if (const VkResult result = vkEnumerateInstanceVersion(&version); result != VK_SUCCESS) { | ||||
|         LOG_ERROR(Render_Vulkan, "vkEnumerateInstanceVersion returned {}, assuming Vulkan 1.1", | ||||
|                   ToString(result)); | ||||
|         return VK_API_VERSION_1_1; | ||||
|     } | ||||
|     return version; | ||||
| } | ||||
|  | ||||
| std::optional<std::vector<VkExtensionProperties>> EnumerateInstanceExtensionProperties( | ||||
|     const InstanceDispatch& dld) { | ||||
|     u32 num; | ||||
|     if (dld.vkEnumerateInstanceExtensionProperties(nullptr, &num, nullptr) != VK_SUCCESS) { | ||||
|         return std::nullopt; | ||||
|     } | ||||
|     std::vector<VkExtensionProperties> properties(num); | ||||
|     if (dld.vkEnumerateInstanceExtensionProperties(nullptr, &num, properties.data()) != | ||||
|         VK_SUCCESS) { | ||||
|         return std::nullopt; | ||||
|     } | ||||
|     return properties; | ||||
| } | ||||
|  | ||||
| std::optional<std::vector<VkLayerProperties>> EnumerateInstanceLayerProperties( | ||||
|     const InstanceDispatch& dld) { | ||||
|     u32 num; | ||||
|     if (dld.vkEnumerateInstanceLayerProperties(&num, nullptr) != VK_SUCCESS) { | ||||
|         return std::nullopt; | ||||
|     } | ||||
|     std::vector<VkLayerProperties> properties(num); | ||||
|     if (dld.vkEnumerateInstanceLayerProperties(&num, properties.data()) != VK_SUCCESS) { | ||||
|         return std::nullopt; | ||||
|     } | ||||
|     return properties; | ||||
| } | ||||
|  | ||||
| } // namespace Vulkan::vk | ||||
							
								
								
									
										1222
									
								
								src/video_core/vulkan_common/vulkan_wrapper.h
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										1222
									
								
								src/video_core/vulkan_common/vulkan_wrapper.h
									
									
									
									
									
										Executable file
									
								
							
										
											
												Plik diff jest za duży
												Load Diff
											
										
									
								
							
		Reference in New Issue
	
	Block a user