Files
kaizen/external/parallel-rdp/parallel-rdp-standalone/vulkan/context.hpp
T
iris 00cc9309cb Squashed 'external/ircolib/' changes from ce3cd726c..de6e324bd
de6e324bd separate emu thread
10d3daf86 Roms List improvements
95d202f37 Let's make the rom list process on a separate thread so the emulator doesnt take ages to load.
fc306967f Wow the ROM Header was just completely busted. Game list view works now
bad1691ee fuck this shit
2b59e5f46 game list in progress
d26417b83 remappable inputs in progress
ac4af8106 input
e72abc240 update readme
430139dc9 Qt6 frontend
3080d4d45 Fix this small bug too
08cd13b85 Cop0 unused functions do not actually pose a threat (as per manual). They don't do anything, so shall we.
61bb4fb44 make idle loop detection a little more specific with where the load goes
b037de4c3 SAZDFsdff
12e81e73e need to figure out why n64-systemtest loops indefinitely at some address that appears to be valid (i think it's me not invalidating the cache properly)
204f0e13b idle skipping seems to work!
cb8bb634a sdkfjlasdf
58e5c89c1 Fix compilation issue on my machine (no idea)
24fb2898e attempting more serious idle skipping
214719577 Place rsp.Step inside cached interpreter. Gains about 3 more fps
bb97dcc23 mmmmm
920b77d38 wjkhasdfjhkasdf
430ccdab4 it's a start...
4f42a673a Cached interpreter plays Mario 64. Start looking into RSP as well
c9a030787 idle skipping works!
5fbda03ce new idea
366637aba Idle skipping... maybe?
609fa2fb0 Cache instructions implemented but broken lmao. Commented out for now
e140a6d12 - Stop using inheritance for CPU, instead use composition. - Introduce KAIZEN_JIT_ENABLED optional define instead of relying on __aarch64__ and the like. - More cache work
68e613057 prep cache impl
811b4d809 fix clang format
fda755f7d idk
d5024ebbf small MI refactor in preparation of (eventually) implementing the RDRAM interface properly
694b45341 Merge commit '206dcdedf195fb320913584180edb12c7731e396' as 'external/SDL'
206dcdedf Squashed 'external/SDL/' content from commit 4d17b99d0a
4d16e1cb4 need to update sdl
848b19920 Fix compilation error
db61b5299 Merge commit 'e94a94559f28e49678fbcf72199a5258137b0fe9' as 'external/imgui'
e94a94559 Squashed 'external/imgui/' content from commit 02e9b8cac
52edb3757 need to update imgui
c1a705e86 Emulate weird JALR behaviour
4b4c32f4b Fix exception for "unusable COP1" in 4 instructions i missed accidentally (again)
df5828142 Bug putting 0s in the log everywhere
f8b580048 Make isviewer a sink to file
8241e9735 Fix exception for "unusable COP1" in 4 instructions i missed accidentally
b29715f20 small changes
d9a620bc1 make use of my new small utility library
0d1aa938e Add 'external/ircolib/' from commit 'ce3cd726c8df8388d554abf8bb55d55020eb4450'
e64eb40b3 Fuck git

git-subtree-dir: external/ircolib
git-subtree-split: de6e324bde
2026-06-15 11:56:38 +02:00

403 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>;
}