2016-08-13 14:57:50 +02:00
|
|
|
// Copyright 2016 Dolphin Emulator Project
|
2021-07-05 03:22:19 +02:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2016-08-13 14:57:50 +02:00
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <memory>
|
2019-10-02 13:27:32 +02:00
|
|
|
#include <optional>
|
2019-09-30 17:07:36 +02:00
|
|
|
#include <string>
|
2016-08-13 14:57:50 +02:00
|
|
|
#include <vector>
|
|
|
|
|
2016-10-01 05:07:50 +02:00
|
|
|
#include "Common/CommonTypes.h"
|
2018-10-24 06:47:48 +02:00
|
|
|
#include "Common/WindowSystemInfo.h"
|
2016-08-13 14:57:50 +02:00
|
|
|
#include "VideoBackends/Vulkan/Constants.h"
|
|
|
|
#include "VideoCommon/VideoConfig.h"
|
|
|
|
|
|
|
|
namespace Vulkan
|
|
|
|
{
|
|
|
|
class VulkanContext
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
VulkanContext(VkInstance instance, VkPhysicalDevice physical_device);
|
|
|
|
~VulkanContext();
|
|
|
|
|
|
|
|
// Determines if the Vulkan validation layer is available on the system.
|
|
|
|
static bool CheckValidationLayerAvailablility();
|
|
|
|
|
|
|
|
// Helper method to create a Vulkan instance.
|
2022-10-27 21:25:32 +02:00
|
|
|
static VkInstance CreateVulkanInstance(WindowSystemType wstype, bool enable_debug_utils,
|
2022-10-07 23:36:16 +02:00
|
|
|
bool enable_validation_layer, u32* out_vk_api_version);
|
2016-08-13 14:57:50 +02:00
|
|
|
|
|
|
|
// Returns a list of Vulkan-compatible GPUs.
|
|
|
|
using GPUList = std::vector<VkPhysicalDevice>;
|
|
|
|
static GPUList EnumerateGPUs(VkInstance instance);
|
|
|
|
|
|
|
|
// Populates backend/video config.
|
|
|
|
// These are public so that the backend info can be populated without creating a context.
|
|
|
|
static void PopulateBackendInfo(VideoConfig* config);
|
|
|
|
static void PopulateBackendInfoAdapters(VideoConfig* config, const GPUList& gpu_list);
|
|
|
|
static void PopulateBackendInfoFeatures(VideoConfig* config, VkPhysicalDevice gpu,
|
2017-03-09 15:01:23 +01:00
|
|
|
const VkPhysicalDeviceProperties& properties,
|
2016-08-13 14:57:50 +02:00
|
|
|
const VkPhysicalDeviceFeatures& features);
|
|
|
|
static void PopulateBackendInfoMultisampleModes(VideoConfig* config, VkPhysicalDevice gpu,
|
|
|
|
const VkPhysicalDeviceProperties& properties);
|
|
|
|
|
|
|
|
// Creates a Vulkan device context.
|
|
|
|
// This assumes that PopulateBackendInfo and PopulateBackendInfoAdapters has already
|
|
|
|
// been called for the specified VideoConfig.
|
|
|
|
static std::unique_ptr<VulkanContext> Create(VkInstance instance, VkPhysicalDevice gpu,
|
2022-10-27 21:25:32 +02:00
|
|
|
VkSurfaceKHR surface, bool enable_debug_utils,
|
2022-10-07 23:36:16 +02:00
|
|
|
bool enable_validation_layer, u32 api_version);
|
2016-08-13 14:57:50 +02:00
|
|
|
|
|
|
|
// Enable/disable debug message runtime.
|
2022-10-27 21:25:32 +02:00
|
|
|
bool EnableDebugUtils();
|
|
|
|
void DisableDebugUtils();
|
2016-08-13 14:57:50 +02:00
|
|
|
|
|
|
|
// Global state accessors
|
|
|
|
VkInstance GetVulkanInstance() const { return m_instance; }
|
|
|
|
VkPhysicalDevice GetPhysicalDevice() const { return m_physical_device; }
|
|
|
|
VkDevice GetDevice() const { return m_device; }
|
|
|
|
VkQueue GetGraphicsQueue() const { return m_graphics_queue; }
|
|
|
|
u32 GetGraphicsQueueFamilyIndex() const { return m_graphics_queue_family_index; }
|
2017-09-04 23:35:53 +02:00
|
|
|
VkQueue GetPresentQueue() const { return m_present_queue; }
|
|
|
|
u32 GetPresentQueueFamilyIndex() const { return m_present_queue_family_index; }
|
2016-08-13 14:57:50 +02:00
|
|
|
const VkQueueFamilyProperties& GetGraphicsQueueProperties() const
|
|
|
|
{
|
|
|
|
return m_graphics_queue_properties;
|
|
|
|
}
|
|
|
|
const VkPhysicalDeviceMemoryProperties& GetDeviceMemoryProperties() const
|
|
|
|
{
|
|
|
|
return m_device_memory_properties;
|
|
|
|
}
|
|
|
|
const VkPhysicalDeviceProperties& GetDeviceProperties() const { return m_device_properties; }
|
|
|
|
const VkPhysicalDeviceFeatures& GetDeviceFeatures() const { return m_device_features; }
|
|
|
|
const VkPhysicalDeviceLimits& GetDeviceLimits() const { return m_device_properties.limits; }
|
|
|
|
// Support bits
|
|
|
|
bool SupportsAnisotropicFiltering() const
|
|
|
|
{
|
|
|
|
return m_device_features.samplerAnisotropy == VK_TRUE;
|
|
|
|
}
|
|
|
|
bool SupportsPreciseOcclusionQueries() const
|
|
|
|
{
|
|
|
|
return m_device_features.occlusionQueryPrecise == VK_TRUE;
|
|
|
|
}
|
2019-03-22 11:39:37 +01:00
|
|
|
u32 GetShaderSubgroupSize() const { return m_shader_subgroup_size; }
|
|
|
|
bool SupportsShaderSubgroupOperations() const { return m_supports_shader_subgroup_operations; }
|
2018-08-29 05:12:19 +02:00
|
|
|
|
2016-08-13 14:57:50 +02:00
|
|
|
// Helpers for getting constants
|
|
|
|
VkDeviceSize GetUniformBufferAlignment() const
|
|
|
|
{
|
|
|
|
return m_device_properties.limits.minUniformBufferOffsetAlignment;
|
|
|
|
}
|
|
|
|
VkDeviceSize GetTexelBufferAlignment() const
|
|
|
|
{
|
|
|
|
return m_device_properties.limits.minUniformBufferOffsetAlignment;
|
|
|
|
}
|
|
|
|
VkDeviceSize GetBufferImageGranularity() const
|
|
|
|
{
|
|
|
|
return m_device_properties.limits.bufferImageGranularity;
|
|
|
|
}
|
2016-10-02 15:53:09 +02:00
|
|
|
float GetMaxSamplerAnisotropy() const { return m_device_properties.limits.maxSamplerAnisotropy; }
|
2016-08-13 14:57:50 +02:00
|
|
|
|
2019-09-30 17:07:36 +02:00
|
|
|
// Returns true if the specified extension is supported and enabled.
|
|
|
|
bool SupportsDeviceExtension(const char* name) const;
|
|
|
|
|
2019-09-30 17:10:08 +02:00
|
|
|
// Returns true if exclusive fullscreen is supported for the given surface.
|
|
|
|
bool SupportsExclusiveFullscreen(const WindowSystemInfo& wsi, VkSurfaceKHR surface);
|
|
|
|
|
2022-10-07 23:36:16 +02:00
|
|
|
VmaAllocator GetMemoryAllocator() const { return m_allocator; }
|
|
|
|
|
2019-09-30 17:10:08 +02:00
|
|
|
#ifdef WIN32
|
|
|
|
// Returns the platform-specific exclusive fullscreen structure.
|
|
|
|
VkSurfaceFullScreenExclusiveWin32InfoEXT
|
|
|
|
GetPlatformExclusiveFullscreenInfo(const WindowSystemInfo& wsi);
|
|
|
|
#endif
|
|
|
|
|
2016-08-13 14:57:50 +02:00
|
|
|
private:
|
2019-09-30 17:07:36 +02:00
|
|
|
static bool SelectInstanceExtensions(std::vector<const char*>* extension_list,
|
2022-10-27 21:25:32 +02:00
|
|
|
WindowSystemType wstype, bool enable_debug_utils,
|
|
|
|
bool validation_layer_enabled);
|
2019-09-30 17:07:36 +02:00
|
|
|
bool SelectDeviceExtensions(bool enable_surface);
|
2016-08-13 14:57:50 +02:00
|
|
|
bool SelectDeviceFeatures();
|
|
|
|
bool CreateDevice(VkSurfaceKHR surface, bool enable_validation_layer);
|
2017-09-19 13:11:33 +02:00
|
|
|
void InitDriverDetails();
|
2019-03-22 11:39:37 +01:00
|
|
|
void PopulateShaderSubgroupSupport();
|
2022-10-07 23:36:16 +02:00
|
|
|
bool CreateAllocator(u32 vk_api_version);
|
2016-08-13 14:57:50 +02:00
|
|
|
|
|
|
|
VkInstance m_instance = VK_NULL_HANDLE;
|
|
|
|
VkPhysicalDevice m_physical_device = VK_NULL_HANDLE;
|
|
|
|
VkDevice m_device = VK_NULL_HANDLE;
|
2022-10-07 23:36:16 +02:00
|
|
|
VmaAllocator m_allocator = VK_NULL_HANDLE;
|
2016-08-13 14:57:50 +02:00
|
|
|
|
|
|
|
VkQueue m_graphics_queue = VK_NULL_HANDLE;
|
|
|
|
u32 m_graphics_queue_family_index = 0;
|
2017-09-04 23:35:53 +02:00
|
|
|
VkQueue m_present_queue = VK_NULL_HANDLE;
|
|
|
|
u32 m_present_queue_family_index = 0;
|
2016-08-13 14:57:50 +02:00
|
|
|
VkQueueFamilyProperties m_graphics_queue_properties = {};
|
|
|
|
|
2022-10-27 21:25:32 +02:00
|
|
|
VkDebugUtilsMessengerEXT m_debug_utils_messenger = VK_NULL_HANDLE;
|
2016-08-13 14:57:50 +02:00
|
|
|
|
|
|
|
VkPhysicalDeviceFeatures m_device_features = {};
|
|
|
|
VkPhysicalDeviceProperties m_device_properties = {};
|
|
|
|
VkPhysicalDeviceMemoryProperties m_device_memory_properties = {};
|
2019-03-22 11:39:37 +01:00
|
|
|
|
|
|
|
u32 m_shader_subgroup_size = 1;
|
|
|
|
bool m_supports_shader_subgroup_operations = false;
|
2019-09-30 17:07:36 +02:00
|
|
|
|
|
|
|
std::vector<std::string> m_device_extensions;
|
2016-08-13 14:57:50 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
extern std::unique_ptr<VulkanContext> g_vulkan_context;
|
|
|
|
|
|
|
|
} // namespace Vulkan
|