mirror of
https://github.com/SimoneN64/Kaizen.git
synced 2025-04-02 10:41:53 -04:00
402 lines
14 KiB
C++
402 lines
14 KiB
C++
/* Copyright (c) 2017-2023 Hans-Kristian Arntzen
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining
|
|
* a copy of this software and associated documentation files (the
|
|
* "Software"), to deal in the Software without restriction, including
|
|
* without limitation the rights to use, copy, modify, merge, publish,
|
|
* distribute, sublicense, and/or sell copies of the Software, and to
|
|
* permit persons to whom the Software is furnished to do so, subject to
|
|
* the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be
|
|
* included in all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
|
* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
|
|
* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
|
|
* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
|
|
* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include "vulkan_headers.hpp"
|
|
#include "logging.hpp"
|
|
#include "vulkan_common.hpp"
|
|
#include <memory>
|
|
#include <functional>
|
|
|
|
#ifdef GRANITE_VULKAN_FOSSILIZE
|
|
#include "cli/fossilize_feature_filter.hpp"
|
|
#endif
|
|
|
|
namespace Util
|
|
{
|
|
class TimelineTraceFile;
|
|
}
|
|
|
|
namespace Granite
|
|
{
|
|
class Filesystem;
|
|
class ThreadGroup;
|
|
class AssetManager;
|
|
}
|
|
|
|
namespace Vulkan
|
|
{
|
|
struct DeviceFeatures
|
|
{
|
|
bool supports_debug_utils = false;
|
|
bool supports_external_memory_host = false;
|
|
bool supports_surface_capabilities2 = false;
|
|
bool supports_full_screen_exclusive = false;
|
|
bool supports_conservative_rasterization = false;
|
|
bool supports_calibrated_timestamps = false;
|
|
bool supports_memory_budget = false;
|
|
bool supports_video_queue = false;
|
|
bool supports_driver_properties = false;
|
|
bool supports_video_decode_queue = false;
|
|
bool supports_video_decode_h264 = false;
|
|
bool supports_video_decode_h265 = false;
|
|
bool supports_astc_decode_mode = false;
|
|
bool supports_image_format_list = false;
|
|
bool supports_format_feature_flags2 = false;
|
|
bool supports_video_encode_queue = false;
|
|
bool supports_video_encode_h264 = false;
|
|
bool supports_video_encode_h265 = false;
|
|
bool supports_external = false;
|
|
bool supports_tooling_info = false;
|
|
bool supports_hdr_metadata = false;
|
|
bool supports_swapchain_colorspace = false;
|
|
bool supports_surface_maintenance1 = false;
|
|
bool supports_store_op_none = false;
|
|
bool supports_push_descriptor = false;
|
|
|
|
VkPhysicalDeviceFeatures enabled_features = {};
|
|
|
|
VkPhysicalDeviceVulkan11Features vk11_features = {};
|
|
VkPhysicalDeviceVulkan12Features vk12_features = {};
|
|
VkPhysicalDeviceVulkan13Features vk13_features = {};
|
|
VkPhysicalDeviceVulkan11Properties vk11_props = {};
|
|
VkPhysicalDeviceVulkan12Properties vk12_props = {};
|
|
VkPhysicalDeviceVulkan13Properties vk13_props = {};
|
|
|
|
// KHR
|
|
VkPhysicalDevicePerformanceQueryFeaturesKHR performance_query_features = {};
|
|
VkPhysicalDevicePresentIdFeaturesKHR present_id_features = {};
|
|
VkPhysicalDevicePresentWaitFeaturesKHR present_wait_features = {};
|
|
VkPhysicalDeviceFragmentShaderBarycentricFeaturesKHR barycentric_features = {};
|
|
VkPhysicalDeviceVideoMaintenance1FeaturesKHR video_maintenance1_features = {};
|
|
|
|
// EXT
|
|
VkPhysicalDeviceExternalMemoryHostPropertiesEXT host_memory_properties = {};
|
|
VkPhysicalDeviceConservativeRasterizationPropertiesEXT conservative_rasterization_properties = {};
|
|
VkPhysicalDeviceMemoryPriorityFeaturesEXT memory_priority_features = {};
|
|
VkPhysicalDeviceASTCDecodeFeaturesEXT astc_decode_features = {};
|
|
VkPhysicalDeviceSwapchainMaintenance1FeaturesEXT swapchain_maintenance1_features = {};
|
|
VkPhysicalDevicePageableDeviceLocalMemoryFeaturesEXT pageable_device_local_memory_features = {};
|
|
VkPhysicalDeviceMeshShaderFeaturesEXT mesh_shader_features = {};
|
|
VkPhysicalDeviceMeshShaderPropertiesEXT mesh_shader_properties = {};
|
|
VkPhysicalDeviceIndexTypeUint8FeaturesEXT index_type_uint8_features = {};
|
|
VkPhysicalDeviceRGBA10X6FormatsFeaturesEXT rgba10x6_formats_features = {};
|
|
VkPhysicalDeviceImageCompressionControlFeaturesEXT image_compression_control_features = {};
|
|
VkPhysicalDeviceImageCompressionControlSwapchainFeaturesEXT image_compression_control_swapchain_features = {};
|
|
|
|
// Vendor
|
|
VkPhysicalDeviceComputeShaderDerivativesFeaturesNV compute_shader_derivative_features = {};
|
|
VkPhysicalDeviceDeviceGeneratedCommandsFeaturesNV device_generated_commands_features = {};
|
|
VkPhysicalDeviceDeviceGeneratedCommandsComputeFeaturesNV device_generated_commands_compute_features = {};
|
|
VkPhysicalDeviceDeviceGeneratedCommandsPropertiesNV device_generated_commands_properties = {};
|
|
VkPhysicalDeviceDescriptorPoolOverallocationFeaturesNV descriptor_pool_overallocation_features = {};
|
|
|
|
// Fallback feature structs (Vulkan 1.1)
|
|
VkPhysicalDeviceHostQueryResetFeatures host_query_reset_features = {};
|
|
VkPhysicalDevice16BitStorageFeaturesKHR storage_16bit_features = {};
|
|
// Fallback feature structs (Vulkan 1.2)
|
|
VkPhysicalDeviceFloat16Int8FeaturesKHR float16_int8_features = {};
|
|
VkPhysicalDevice8BitStorageFeaturesKHR storage_8bit_features = {};
|
|
// Fallback feature structs (Vulkan 1.3)
|
|
VkPhysicalDeviceSubgroupSizeControlFeatures subgroup_size_control_features = {};
|
|
|
|
VkDriverId driver_id = {};
|
|
|
|
// References Vulkan::Context.
|
|
const VkPhysicalDeviceFeatures2 *pdf2 = nullptr;
|
|
const char * const * instance_extensions = nullptr;
|
|
uint32_t num_instance_extensions = 0;
|
|
const char * const * device_extensions = nullptr;
|
|
uint32_t num_device_extensions = 0;
|
|
|
|
uint32_t instance_api_core_version = VK_API_VERSION_1_1;
|
|
uint32_t device_api_core_version = VK_API_VERSION_1_1;
|
|
};
|
|
|
|
enum VendorID
|
|
{
|
|
VENDOR_ID_AMD = 0x1002,
|
|
VENDOR_ID_NVIDIA = 0x10de,
|
|
VENDOR_ID_INTEL = 0x8086,
|
|
VENDOR_ID_ARM = 0x13b5,
|
|
VENDOR_ID_QCOM = 0x5143
|
|
};
|
|
|
|
enum ContextCreationFlagBits
|
|
{
|
|
CONTEXT_CREATION_ENABLE_ADVANCED_WSI_BIT = 1 << 0,
|
|
CONTEXT_CREATION_ENABLE_VIDEO_DECODE_BIT = 1 << 1,
|
|
CONTEXT_CREATION_ENABLE_VIDEO_ENCODE_BIT = 1 << 2,
|
|
CONTEXT_CREATION_ENABLE_VIDEO_H264_BIT = 1 << 3,
|
|
CONTEXT_CREATION_ENABLE_VIDEO_H265_BIT = 1 << 4
|
|
};
|
|
using ContextCreationFlags = uint32_t;
|
|
|
|
struct QueueInfo
|
|
{
|
|
QueueInfo();
|
|
VkQueue queues[QUEUE_INDEX_COUNT] = {};
|
|
uint32_t family_indices[QUEUE_INDEX_COUNT];
|
|
uint32_t counts[QUEUE_INDEX_COUNT] = {};
|
|
uint32_t timestamp_valid_bits = 0;
|
|
};
|
|
|
|
struct InstanceFactory
|
|
{
|
|
virtual ~InstanceFactory() = default;
|
|
virtual VkInstance create_instance(const VkInstanceCreateInfo *info) = 0;
|
|
};
|
|
|
|
struct DeviceFactory
|
|
{
|
|
virtual ~DeviceFactory() = default;
|
|
virtual VkDevice create_device(VkPhysicalDevice gpu, const VkDeviceCreateInfo *info) = 0;
|
|
};
|
|
|
|
class CopiedApplicationInfo
|
|
{
|
|
public:
|
|
CopiedApplicationInfo();
|
|
const VkApplicationInfo &get_application_info() const;
|
|
void copy_assign(const VkApplicationInfo *info);
|
|
|
|
private:
|
|
std::string application;
|
|
std::string engine;
|
|
VkApplicationInfo app = {
|
|
VK_STRUCTURE_TYPE_APPLICATION_INFO, nullptr, "Granite", 0, "Granite", 0, VK_API_VERSION_1_1,
|
|
};
|
|
|
|
void set_default_app();
|
|
};
|
|
|
|
class Context
|
|
: public Util::IntrusivePtrEnabled<Context, std::default_delete<Context>, HandleCounter>
|
|
#ifdef GRANITE_VULKAN_FOSSILIZE
|
|
, public Fossilize::DeviceQueryInterface
|
|
#endif
|
|
{
|
|
public:
|
|
// If these interface are set, factory->create() calls are used instead of global vkCreateInstance and vkCreateDevice.
|
|
// For deeper API interop scenarios.
|
|
void set_instance_factory(InstanceFactory *factory);
|
|
void set_device_factory(DeviceFactory *factory);
|
|
|
|
// Only takes effect if profiles are enabled in build. (GRANITE_VULKAN_PROFILES)
|
|
// If profile is non-null, forces a specific profile.
|
|
// If not supported, initialization fails.
|
|
// If not set, ignore profiles.
|
|
// If strict is false, the profile should be seen as a baseline and Granite will augment features on top.
|
|
// If true, the profile is a strict limit for device functionality. For validation purposes.
|
|
void set_required_profile(const char *profile, bool strict);
|
|
|
|
// Call before initializing instances. app_info may be freed after returning.
|
|
// API_VERSION must be at least 1.1.
|
|
// By default, a Vulkan 1.1 instance is created.
|
|
void set_application_info(const VkApplicationInfo *app_info);
|
|
|
|
// Recommended interface.
|
|
// InstanceFactory can be used to override enabled instance layers and extensions.
|
|
// For simple WSI use, it is enough to just enable VK_KHR_surface and the platform.
|
|
bool init_instance(const char * const *instance_ext, uint32_t instance_ext_count,
|
|
ContextCreationFlags flags = 0);
|
|
// DeviceFactory can be used to override enabled device extensions.
|
|
// For simple WSI use, it is enough to just enable VK_KHR_swapchain.
|
|
bool init_device(VkPhysicalDevice gpu, VkSurfaceKHR surface_compat,
|
|
const char * const *device_ext, uint32_t device_ext_count,
|
|
ContextCreationFlags flags = 0);
|
|
|
|
// Simplified initialization which calls init_instance and init_device in succession with NULL GPU and surface.
|
|
// Provided for compat with older code.
|
|
bool init_instance_and_device(const char * const *instance_ext, uint32_t instance_ext_count,
|
|
const char * const *device_ext, uint32_t device_ext_count,
|
|
ContextCreationFlags flags = 0);
|
|
|
|
// Deprecated. For libretro Vulkan context negotiation v1.
|
|
// Use InstanceFactory and DeviceFactory for more advanced scenarios in v2.
|
|
bool init_device_from_instance(VkInstance instance, VkPhysicalDevice gpu, VkSurfaceKHR surface,
|
|
const char **required_device_extensions,
|
|
unsigned num_required_device_extensions,
|
|
const VkPhysicalDeviceFeatures *required_features,
|
|
ContextCreationFlags flags = 0);
|
|
|
|
Context();
|
|
Context(const Context &) = delete;
|
|
void operator=(const Context &) = delete;
|
|
static bool init_loader(PFN_vkGetInstanceProcAddr addr, bool force_reload = false);
|
|
static PFN_vkGetInstanceProcAddr get_instance_proc_addr();
|
|
|
|
~Context();
|
|
|
|
VkInstance get_instance() const
|
|
{
|
|
return instance;
|
|
}
|
|
|
|
VkPhysicalDevice get_gpu() const
|
|
{
|
|
return gpu;
|
|
}
|
|
|
|
VkDevice get_device() const
|
|
{
|
|
return device;
|
|
}
|
|
|
|
const QueueInfo &get_queue_info() const
|
|
{
|
|
return queue_info;
|
|
}
|
|
|
|
const VkPhysicalDeviceProperties &get_gpu_props() const
|
|
{
|
|
return gpu_props;
|
|
}
|
|
|
|
const VkPhysicalDeviceMemoryProperties &get_mem_props() const
|
|
{
|
|
return mem_props;
|
|
}
|
|
|
|
void release_instance()
|
|
{
|
|
owned_instance = false;
|
|
}
|
|
|
|
void release_device()
|
|
{
|
|
owned_device = false;
|
|
}
|
|
|
|
const DeviceFeatures &get_enabled_device_features() const
|
|
{
|
|
return ext;
|
|
}
|
|
|
|
const VkApplicationInfo &get_application_info() const;
|
|
|
|
void notify_validation_error(const char *msg);
|
|
void set_notification_callback(std::function<void (const char *)> func);
|
|
|
|
void set_num_thread_indices(unsigned indices)
|
|
{
|
|
num_thread_indices = indices;
|
|
}
|
|
|
|
unsigned get_num_thread_indices() const
|
|
{
|
|
return num_thread_indices;
|
|
}
|
|
|
|
const VolkDeviceTable &get_device_table() const
|
|
{
|
|
return device_table;
|
|
}
|
|
|
|
struct SystemHandles
|
|
{
|
|
Util::TimelineTraceFile *timeline_trace_file = nullptr;
|
|
Granite::Filesystem *filesystem = nullptr;
|
|
Granite::ThreadGroup *thread_group = nullptr;
|
|
Granite::AssetManager *asset_manager = nullptr;
|
|
};
|
|
|
|
void set_system_handles(const SystemHandles &handles_)
|
|
{
|
|
handles = handles_;
|
|
}
|
|
|
|
const SystemHandles &get_system_handles() const
|
|
{
|
|
return handles;
|
|
}
|
|
|
|
#ifdef GRANITE_VULKAN_FOSSILIZE
|
|
const Fossilize::FeatureFilter &get_feature_filter() const
|
|
{
|
|
return feature_filter;
|
|
}
|
|
#endif
|
|
|
|
const VkPhysicalDeviceFeatures2 &get_physical_device_features() const
|
|
{
|
|
return pdf2;
|
|
}
|
|
|
|
private:
|
|
InstanceFactory *instance_factory = nullptr;
|
|
DeviceFactory *device_factory = nullptr;
|
|
VkDevice device = VK_NULL_HANDLE;
|
|
VkInstance instance = VK_NULL_HANDLE;
|
|
VkPhysicalDevice gpu = VK_NULL_HANDLE;
|
|
VolkDeviceTable device_table = {};
|
|
SystemHandles handles;
|
|
VkPhysicalDeviceProperties gpu_props = {};
|
|
VkPhysicalDeviceMemoryProperties mem_props = {};
|
|
|
|
CopiedApplicationInfo user_application_info;
|
|
|
|
QueueInfo queue_info;
|
|
unsigned num_thread_indices = 1;
|
|
|
|
bool create_instance(const char * const *instance_ext, uint32_t instance_ext_count, ContextCreationFlags flags);
|
|
bool create_device(VkPhysicalDevice gpu, VkSurfaceKHR surface,
|
|
const char * const *required_device_extensions, uint32_t num_required_device_extensions,
|
|
const VkPhysicalDeviceFeatures *required_features, ContextCreationFlags flags);
|
|
|
|
bool owned_instance = false;
|
|
bool owned_device = false;
|
|
DeviceFeatures ext;
|
|
VkPhysicalDeviceFeatures2 pdf2;
|
|
std::vector<const char *> enabled_device_extensions;
|
|
std::vector<const char *> enabled_instance_extensions;
|
|
|
|
std::string required_profile;
|
|
bool required_profile_strict = false;
|
|
|
|
#ifdef VULKAN_DEBUG
|
|
VkDebugUtilsMessengerEXT debug_messenger = VK_NULL_HANDLE;
|
|
bool force_no_validation = false;
|
|
#endif
|
|
std::function<void (const char *)> message_callback;
|
|
|
|
void destroy_instance();
|
|
void destroy_device();
|
|
|
|
bool physical_device_supports_surface_and_profile(VkPhysicalDevice candidate_gpu, VkSurfaceKHR surface) const;
|
|
|
|
#ifdef GRANITE_VULKAN_FOSSILIZE
|
|
Fossilize::FeatureFilter feature_filter;
|
|
bool format_is_supported(VkFormat format, VkFormatFeatureFlags features) override;
|
|
bool descriptor_set_layout_is_supported(const VkDescriptorSetLayoutCreateInfo *set_layout) override;
|
|
#endif
|
|
|
|
bool init_profile();
|
|
VkResult create_instance_from_profile(const VkInstanceCreateInfo &info, VkInstance *pInstance);
|
|
VkResult create_device_from_profile(const VkDeviceCreateInfo &info, VkDevice *pDevice);
|
|
|
|
VkApplicationInfo get_promoted_application_info() const;
|
|
};
|
|
|
|
using ContextHandle = Util::IntrusivePtr<Context>;
|
|
}
|