forked from eden-emu/eden
		
	Merge pull request #9303 from liamwhite/new-vulkan-init
Vulkan: update initialization
This commit is contained in:
		
						commit
						39df80ff00
					
				
					 15 changed files with 194 additions and 104 deletions
				
			
		
							
								
								
									
										2
									
								
								.gitmodules
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitmodules
									
										
									
									
										vendored
									
									
								
							|  | @ -27,7 +27,7 @@ | ||||||
|     url = https://github.com/KhronosGroup/Vulkan-Headers.git |     url = https://github.com/KhronosGroup/Vulkan-Headers.git | ||||||
| [submodule "sirit"] | [submodule "sirit"] | ||||||
|     path = externals/sirit |     path = externals/sirit | ||||||
|     url = https://github.com/ReinUsesLisp/sirit |     url = https://github.com/yuzu-emu/sirit | ||||||
| [submodule "mbedtls"] | [submodule "mbedtls"] | ||||||
|     path = externals/mbedtls |     path = externals/mbedtls | ||||||
|     url = https://github.com/yuzu-emu/mbedtls |     url = https://github.com/yuzu-emu/mbedtls | ||||||
|  |  | ||||||
							
								
								
									
										2
									
								
								externals/sirit
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								externals/sirit
									
										
									
									
										vendored
									
									
								
							|  | @ -1 +1 @@ | ||||||
| Subproject commit aa292d56650bc28f2b2d75973fab2e61d0136f9c | Subproject commit d7ad93a88864bda94e282e95028f90b5784e4d20 | ||||||
|  | @ -17,6 +17,8 @@ enum class WindowSystemType { | ||||||
|     Windows, |     Windows, | ||||||
|     X11, |     X11, | ||||||
|     Wayland, |     Wayland, | ||||||
|  |     Cocoa, | ||||||
|  |     Android, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  |  | ||||||
|  | @ -402,8 +402,10 @@ void SetupCapabilities(const Profile& profile, const Info& info, EmitContext& ct | ||||||
|         ctx.AddCapability(spv::Capability::SparseResidency); |         ctx.AddCapability(spv::Capability::SparseResidency); | ||||||
|     } |     } | ||||||
|     if (info.uses_demote_to_helper_invocation && profile.support_demote_to_helper_invocation) { |     if (info.uses_demote_to_helper_invocation && profile.support_demote_to_helper_invocation) { | ||||||
|  |         if (profile.supported_spirv < 0x00010600) { | ||||||
|             ctx.AddExtension("SPV_EXT_demote_to_helper_invocation"); |             ctx.AddExtension("SPV_EXT_demote_to_helper_invocation"); | ||||||
|         ctx.AddCapability(spv::Capability::DemoteToHelperInvocationEXT); |         } | ||||||
|  |         ctx.AddCapability(spv::Capability::DemoteToHelperInvocation); | ||||||
|     } |     } | ||||||
|     if (info.stores[IR::Attribute::ViewportIndex]) { |     if (info.stores[IR::Attribute::ViewportIndex]) { | ||||||
|         ctx.AddCapability(spv::Capability::MultiViewport); |         ctx.AddCapability(spv::Capability::MultiViewport); | ||||||
|  | @ -426,12 +428,11 @@ void SetupCapabilities(const Profile& profile, const Info& info, EmitContext& ct | ||||||
|     if ((info.uses_subgroup_vote || info.uses_subgroup_invocation_id || |     if ((info.uses_subgroup_vote || info.uses_subgroup_invocation_id || | ||||||
|          info.uses_subgroup_shuffles) && |          info.uses_subgroup_shuffles) && | ||||||
|         profile.support_vote) { |         profile.support_vote) { | ||||||
|         ctx.AddExtension("SPV_KHR_shader_ballot"); |         ctx.AddCapability(spv::Capability::GroupNonUniformBallot); | ||||||
|         ctx.AddCapability(spv::Capability::SubgroupBallotKHR); |         ctx.AddCapability(spv::Capability::GroupNonUniformShuffle); | ||||||
|         if (!profile.warp_size_potentially_larger_than_guest) { |         if (!profile.warp_size_potentially_larger_than_guest) { | ||||||
|             // vote ops are only used when not taking the long path
 |             // vote ops are only used when not taking the long path
 | ||||||
|             ctx.AddExtension("SPV_KHR_subgroup_vote"); |             ctx.AddCapability(spv::Capability::GroupNonUniformVote); | ||||||
|             ctx.AddCapability(spv::Capability::SubgroupVoteKHR); |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     if (info.uses_int64_bit_atomics && profile.support_int64_atomics) { |     if (info.uses_int64_bit_atomics && profile.support_int64_atomics) { | ||||||
|  |  | ||||||
|  | @ -12,7 +12,7 @@ void EmitJoin(EmitContext&) { | ||||||
| 
 | 
 | ||||||
| void EmitDemoteToHelperInvocation(EmitContext& ctx) { | void EmitDemoteToHelperInvocation(EmitContext& ctx) { | ||||||
|     if (ctx.profile.support_demote_to_helper_invocation) { |     if (ctx.profile.support_demote_to_helper_invocation) { | ||||||
|         ctx.OpDemoteToHelperInvocationEXT(); |         ctx.OpDemoteToHelperInvocation(); | ||||||
|     } else { |     } else { | ||||||
|         const Id kill_label{ctx.OpLabel()}; |         const Id kill_label{ctx.OpLabel()}; | ||||||
|         const Id impossible_label{ctx.OpLabel()}; |         const Id impossible_label{ctx.OpLabel()}; | ||||||
|  |  | ||||||
|  | @ -6,6 +6,10 @@ | ||||||
| 
 | 
 | ||||||
| namespace Shader::Backend::SPIRV { | namespace Shader::Backend::SPIRV { | ||||||
| namespace { | namespace { | ||||||
|  | Id SubgroupScope(EmitContext& ctx) { | ||||||
|  |     return ctx.Const(static_cast<u32>(spv::Scope::Subgroup)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| Id GetThreadId(EmitContext& ctx) { | Id GetThreadId(EmitContext& ctx) { | ||||||
|     return ctx.OpLoad(ctx.U32[1], ctx.subgroup_local_invocation_id); |     return ctx.OpLoad(ctx.U32[1], ctx.subgroup_local_invocation_id); | ||||||
| } | } | ||||||
|  | @ -49,8 +53,9 @@ Id GetMaxThreadId(EmitContext& ctx, Id thread_id, Id clamp, Id segmentation_mask | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| Id SelectValue(EmitContext& ctx, Id in_range, Id value, Id src_thread_id) { | Id SelectValue(EmitContext& ctx, Id in_range, Id value, Id src_thread_id) { | ||||||
|     return ctx.OpSelect(ctx.U32[1], in_range, |     return ctx.OpSelect( | ||||||
|                         ctx.OpSubgroupReadInvocationKHR(ctx.U32[1], value, src_thread_id), value); |         ctx.U32[1], in_range, | ||||||
|  |         ctx.OpGroupNonUniformShuffle(ctx.U32[1], SubgroupScope(ctx), value, src_thread_id), value); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| Id GetUpperClamp(EmitContext& ctx, Id invocation_id, Id clamp) { | Id GetUpperClamp(EmitContext& ctx, Id invocation_id, Id clamp) { | ||||||
|  | @ -71,40 +76,46 @@ Id EmitLaneId(EmitContext& ctx) { | ||||||
| 
 | 
 | ||||||
| Id EmitVoteAll(EmitContext& ctx, Id pred) { | Id EmitVoteAll(EmitContext& ctx, Id pred) { | ||||||
|     if (!ctx.profile.warp_size_potentially_larger_than_guest) { |     if (!ctx.profile.warp_size_potentially_larger_than_guest) { | ||||||
|         return ctx.OpSubgroupAllKHR(ctx.U1, pred); |         return ctx.OpGroupNonUniformAll(ctx.U1, SubgroupScope(ctx), pred); | ||||||
|     } |     } | ||||||
|     const Id mask_ballot{ctx.OpSubgroupBallotKHR(ctx.U32[4], ctx.true_value)}; |     const Id mask_ballot{ | ||||||
|  |         ctx.OpGroupNonUniformBallot(ctx.U32[4], SubgroupScope(ctx), ctx.true_value)}; | ||||||
|     const Id active_mask{WarpExtract(ctx, mask_ballot)}; |     const Id active_mask{WarpExtract(ctx, mask_ballot)}; | ||||||
|     const Id ballot{WarpExtract(ctx, ctx.OpSubgroupBallotKHR(ctx.U32[4], pred))}; |     const Id ballot{ | ||||||
|  |         WarpExtract(ctx, ctx.OpGroupNonUniformBallot(ctx.U32[4], SubgroupScope(ctx), pred))}; | ||||||
|     const Id lhs{ctx.OpBitwiseAnd(ctx.U32[1], ballot, active_mask)}; |     const Id lhs{ctx.OpBitwiseAnd(ctx.U32[1], ballot, active_mask)}; | ||||||
|     return ctx.OpIEqual(ctx.U1, lhs, active_mask); |     return ctx.OpIEqual(ctx.U1, lhs, active_mask); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| Id EmitVoteAny(EmitContext& ctx, Id pred) { | Id EmitVoteAny(EmitContext& ctx, Id pred) { | ||||||
|     if (!ctx.profile.warp_size_potentially_larger_than_guest) { |     if (!ctx.profile.warp_size_potentially_larger_than_guest) { | ||||||
|         return ctx.OpSubgroupAnyKHR(ctx.U1, pred); |         return ctx.OpGroupNonUniformAny(ctx.U1, SubgroupScope(ctx), pred); | ||||||
|     } |     } | ||||||
|     const Id mask_ballot{ctx.OpSubgroupBallotKHR(ctx.U32[4], ctx.true_value)}; |     const Id mask_ballot{ | ||||||
|  |         ctx.OpGroupNonUniformBallot(ctx.U32[4], SubgroupScope(ctx), ctx.true_value)}; | ||||||
|     const Id active_mask{WarpExtract(ctx, mask_ballot)}; |     const Id active_mask{WarpExtract(ctx, mask_ballot)}; | ||||||
|     const Id ballot{WarpExtract(ctx, ctx.OpSubgroupBallotKHR(ctx.U32[4], pred))}; |     const Id ballot{ | ||||||
|  |         WarpExtract(ctx, ctx.OpGroupNonUniformBallot(ctx.U32[4], SubgroupScope(ctx), pred))}; | ||||||
|     const Id lhs{ctx.OpBitwiseAnd(ctx.U32[1], ballot, active_mask)}; |     const Id lhs{ctx.OpBitwiseAnd(ctx.U32[1], ballot, active_mask)}; | ||||||
|     return ctx.OpINotEqual(ctx.U1, lhs, ctx.u32_zero_value); |     return ctx.OpINotEqual(ctx.U1, lhs, ctx.u32_zero_value); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| Id EmitVoteEqual(EmitContext& ctx, Id pred) { | Id EmitVoteEqual(EmitContext& ctx, Id pred) { | ||||||
|     if (!ctx.profile.warp_size_potentially_larger_than_guest) { |     if (!ctx.profile.warp_size_potentially_larger_than_guest) { | ||||||
|         return ctx.OpSubgroupAllEqualKHR(ctx.U1, pred); |         return ctx.OpGroupNonUniformAllEqual(ctx.U1, SubgroupScope(ctx), pred); | ||||||
|     } |     } | ||||||
|     const Id mask_ballot{ctx.OpSubgroupBallotKHR(ctx.U32[4], ctx.true_value)}; |     const Id mask_ballot{ | ||||||
|  |         ctx.OpGroupNonUniformBallot(ctx.U32[4], SubgroupScope(ctx), ctx.true_value)}; | ||||||
|     const Id active_mask{WarpExtract(ctx, mask_ballot)}; |     const Id active_mask{WarpExtract(ctx, mask_ballot)}; | ||||||
|     const Id ballot{WarpExtract(ctx, ctx.OpSubgroupBallotKHR(ctx.U32[4], pred))}; |     const Id ballot{ | ||||||
|  |         WarpExtract(ctx, ctx.OpGroupNonUniformBallot(ctx.U32[4], SubgroupScope(ctx), pred))}; | ||||||
|     const Id lhs{ctx.OpBitwiseXor(ctx.U32[1], ballot, active_mask)}; |     const Id lhs{ctx.OpBitwiseXor(ctx.U32[1], ballot, active_mask)}; | ||||||
|     return ctx.OpLogicalOr(ctx.U1, ctx.OpIEqual(ctx.U1, lhs, ctx.u32_zero_value), |     return ctx.OpLogicalOr(ctx.U1, ctx.OpIEqual(ctx.U1, lhs, ctx.u32_zero_value), | ||||||
|                            ctx.OpIEqual(ctx.U1, lhs, active_mask)); |                            ctx.OpIEqual(ctx.U1, lhs, active_mask)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| Id EmitSubgroupBallot(EmitContext& ctx, Id pred) { | Id EmitSubgroupBallot(EmitContext& ctx, Id pred) { | ||||||
|     const Id ballot{ctx.OpSubgroupBallotKHR(ctx.U32[4], pred)}; |     const Id ballot{ctx.OpGroupNonUniformBallot(ctx.U32[4], SubgroupScope(ctx), pred)}; | ||||||
|     if (!ctx.profile.warp_size_potentially_larger_than_guest) { |     if (!ctx.profile.warp_size_potentially_larger_than_guest) { | ||||||
|         return ctx.OpCompositeExtract(ctx.U32[1], ballot, 0U); |         return ctx.OpCompositeExtract(ctx.U32[1], ballot, 0U); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -54,7 +54,7 @@ using VideoCommon::FileEnvironment; | ||||||
| using VideoCommon::GenericEnvironment; | using VideoCommon::GenericEnvironment; | ||||||
| using VideoCommon::GraphicsEnvironment; | using VideoCommon::GraphicsEnvironment; | ||||||
| 
 | 
 | ||||||
| constexpr u32 CACHE_VERSION = 7; | constexpr u32 CACHE_VERSION = 8; | ||||||
| 
 | 
 | ||||||
| template <typename Container> | template <typename Container> | ||||||
| auto MakeSpan(Container& container) { | auto MakeSpan(Container& container) { | ||||||
|  | @ -289,7 +289,7 @@ PipelineCache::PipelineCache(RasterizerVulkan& rasterizer_, const Device& device | ||||||
|     const auto& float_control{device.FloatControlProperties()}; |     const auto& float_control{device.FloatControlProperties()}; | ||||||
|     const VkDriverIdKHR driver_id{device.GetDriverID()}; |     const VkDriverIdKHR driver_id{device.GetDriverID()}; | ||||||
|     profile = Shader::Profile{ |     profile = Shader::Profile{ | ||||||
|         .supported_spirv = device.IsKhrSpirv1_4Supported() ? 0x00010400U : 0x00010000U, |         .supported_spirv = device.SupportedSpirvVersion(), | ||||||
|         .unified_descriptor_binding = true, |         .unified_descriptor_binding = true, | ||||||
|         .support_descriptor_aliasing = true, |         .support_descriptor_aliasing = true, | ||||||
|         .support_int8 = device.IsInt8Supported(), |         .support_int8 = device.IsInt8Supported(), | ||||||
|  |  | ||||||
|  | @ -74,23 +74,14 @@ enum class NvidiaArchitecture { | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| constexpr std::array REQUIRED_EXTENSIONS{ | constexpr std::array REQUIRED_EXTENSIONS{ | ||||||
|     VK_KHR_MAINTENANCE1_EXTENSION_NAME, |  | ||||||
|     VK_KHR_STORAGE_BUFFER_STORAGE_CLASS_EXTENSION_NAME, |  | ||||||
|     VK_KHR_SHADER_DRAW_PARAMETERS_EXTENSION_NAME, |  | ||||||
|     VK_KHR_16BIT_STORAGE_EXTENSION_NAME, |  | ||||||
|     VK_KHR_8BIT_STORAGE_EXTENSION_NAME, |  | ||||||
|     VK_KHR_DRIVER_PROPERTIES_EXTENSION_NAME, |  | ||||||
|     VK_KHR_DESCRIPTOR_UPDATE_TEMPLATE_EXTENSION_NAME, |  | ||||||
|     VK_KHR_TIMELINE_SEMAPHORE_EXTENSION_NAME, |  | ||||||
|     VK_KHR_SAMPLER_MIRROR_CLAMP_TO_EDGE_EXTENSION_NAME, |  | ||||||
|     VK_KHR_SHADER_FLOAT_CONTROLS_EXTENSION_NAME, |  | ||||||
|     VK_KHR_VARIABLE_POINTERS_EXTENSION_NAME, |  | ||||||
|     VK_EXT_VERTEX_ATTRIBUTE_DIVISOR_EXTENSION_NAME, |     VK_EXT_VERTEX_ATTRIBUTE_DIVISOR_EXTENSION_NAME, | ||||||
|     VK_EXT_SHADER_SUBGROUP_BALLOT_EXTENSION_NAME, |  | ||||||
|     VK_EXT_SHADER_SUBGROUP_VOTE_EXTENSION_NAME, |  | ||||||
|     VK_EXT_ROBUSTNESS_2_EXTENSION_NAME, |     VK_EXT_ROBUSTNESS_2_EXTENSION_NAME, | ||||||
|  | 
 | ||||||
|  |     // Core in 1.2, but required due to use of extension methods,
 | ||||||
|  |     // and well-supported by drivers
 | ||||||
|  |     VK_KHR_TIMELINE_SEMAPHORE_EXTENSION_NAME, | ||||||
|  |     VK_KHR_DESCRIPTOR_UPDATE_TEMPLATE_EXTENSION_NAME, | ||||||
|     VK_EXT_HOST_QUERY_RESET_EXTENSION_NAME, |     VK_EXT_HOST_QUERY_RESET_EXTENSION_NAME, | ||||||
|     VK_EXT_SHADER_DEMOTE_TO_HELPER_INVOCATION_EXTENSION_NAME, |  | ||||||
| #ifdef _WIN32 | #ifdef _WIN32 | ||||||
|     VK_KHR_EXTERNAL_MEMORY_WIN32_EXTENSION_NAME, |     VK_KHR_EXTERNAL_MEMORY_WIN32_EXTENSION_NAME, | ||||||
| #endif | #endif | ||||||
|  | @ -99,6 +90,17 @@ constexpr std::array REQUIRED_EXTENSIONS{ | ||||||
| #endif | #endif | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | constexpr std::array REQUIRED_EXTENSIONS_BEFORE_1_2{ | ||||||
|  |     VK_KHR_8BIT_STORAGE_EXTENSION_NAME, | ||||||
|  |     VK_KHR_SHADER_FLOAT_CONTROLS_EXTENSION_NAME, | ||||||
|  |     VK_KHR_SAMPLER_MIRROR_CLAMP_TO_EDGE_EXTENSION_NAME, | ||||||
|  |     VK_KHR_DRIVER_PROPERTIES_EXTENSION_NAME, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | constexpr std::array REQUIRED_EXTENSIONS_BEFORE_1_3{ | ||||||
|  |     VK_EXT_SHADER_DEMOTE_TO_HELPER_INVOCATION_EXTENSION_NAME, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
| template <typename T> | template <typename T> | ||||||
| void SetNext(void**& next, T& data) { | void SetNext(void**& next, T& data) { | ||||||
|     *next = &data; |     *next = &data; | ||||||
|  | @ -327,7 +329,8 @@ NvidiaArchitecture GetNvidiaArchitecture(vk::PhysicalDevice physical, | ||||||
| Device::Device(VkInstance instance_, vk::PhysicalDevice physical_, VkSurfaceKHR surface, | Device::Device(VkInstance instance_, vk::PhysicalDevice physical_, VkSurfaceKHR surface, | ||||||
|                const vk::InstanceDispatch& dld_) |                const vk::InstanceDispatch& dld_) | ||||||
|     : instance{instance_}, dld{dld_}, physical{physical_}, properties{physical.GetProperties()}, |     : instance{instance_}, dld{dld_}, physical{physical_}, properties{physical.GetProperties()}, | ||||||
|       supported_extensions{GetSupportedExtensions(physical)}, |       instance_version{properties.apiVersion}, supported_extensions{GetSupportedExtensions( | ||||||
|  |                                                    physical)}, | ||||||
|       format_properties(GetFormatProperties(physical)) { |       format_properties(GetFormatProperties(physical)) { | ||||||
|     CheckSuitability(surface != nullptr); |     CheckSuitability(surface != nullptr); | ||||||
|     SetupFamilies(surface); |     SetupFamilies(surface); | ||||||
|  | @ -451,8 +454,8 @@ Device::Device(VkInstance instance_, vk::PhysicalDevice physical_, VkSurfaceKHR | ||||||
|     }; |     }; | ||||||
|     SetNext(next, variable_pointers); |     SetNext(next, variable_pointers); | ||||||
| 
 | 
 | ||||||
|     VkPhysicalDeviceShaderDemoteToHelperInvocationFeaturesEXT demote{ |     VkPhysicalDeviceShaderDemoteToHelperInvocationFeatures demote{ | ||||||
|         .sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_SHADER_DEMOTE_TO_HELPER_INVOCATION_FEATURES_EXT, |         .sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_SHADER_DEMOTE_TO_HELPER_INVOCATION_FEATURES, | ||||||
|         .pNext = nullptr, |         .pNext = nullptr, | ||||||
|         .shaderDemoteToHelperInvocation = true, |         .shaderDemoteToHelperInvocation = true, | ||||||
|     }; |     }; | ||||||
|  | @ -896,28 +899,51 @@ std::string Device::GetDriverName() const { | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static std::vector<const char*> ExtensionsRequiredForInstanceVersion(u32 available_version) { | ||||||
|  |     std::vector<const char*> extensions{REQUIRED_EXTENSIONS.begin(), REQUIRED_EXTENSIONS.end()}; | ||||||
|  | 
 | ||||||
|  |     if (available_version < VK_API_VERSION_1_2) { | ||||||
|  |         extensions.insert(extensions.end(), REQUIRED_EXTENSIONS_BEFORE_1_2.begin(), | ||||||
|  |                           REQUIRED_EXTENSIONS_BEFORE_1_2.end()); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (available_version < VK_API_VERSION_1_3) { | ||||||
|  |         extensions.insert(extensions.end(), REQUIRED_EXTENSIONS_BEFORE_1_3.begin(), | ||||||
|  |                           REQUIRED_EXTENSIONS_BEFORE_1_3.end()); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return extensions; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void Device::CheckSuitability(bool requires_swapchain) const { | void Device::CheckSuitability(bool requires_swapchain) const { | ||||||
|     std::bitset<REQUIRED_EXTENSIONS.size()> available_extensions; |     std::vector<const char*> required_extensions = | ||||||
|     bool has_swapchain = false; |         ExtensionsRequiredForInstanceVersion(instance_version); | ||||||
|     for (const VkExtensionProperties& property : physical.EnumerateDeviceExtensionProperties()) { |     std::vector<const char*> available_extensions; | ||||||
|         const std::string_view name{property.extensionName}; | 
 | ||||||
|         for (size_t i = 0; i < REQUIRED_EXTENSIONS.size(); ++i) { |     if (requires_swapchain) { | ||||||
|             if (available_extensions[i]) { |         required_extensions.push_back(VK_KHR_SWAPCHAIN_EXTENSION_NAME); | ||||||
|                 continue; |  | ||||||
|     } |     } | ||||||
|             available_extensions[i] = name == REQUIRED_EXTENSIONS[i]; | 
 | ||||||
|  |     auto extension_properties = physical.EnumerateDeviceExtensionProperties(); | ||||||
|  | 
 | ||||||
|  |     for (const VkExtensionProperties& property : extension_properties) { | ||||||
|  |         available_extensions.push_back(property.extensionName); | ||||||
|     } |     } | ||||||
|         has_swapchain = has_swapchain || name == VK_KHR_SWAPCHAIN_EXTENSION_NAME; | 
 | ||||||
|  |     bool has_all_required_extensions = true; | ||||||
|  |     for (const char* requirement_name : required_extensions) { | ||||||
|  |         const bool found = | ||||||
|  |             std::ranges::any_of(available_extensions, [&](const char* extension_name) { | ||||||
|  |                 return std::strcmp(requirement_name, extension_name) == 0; | ||||||
|  |             }); | ||||||
|  | 
 | ||||||
|  |         if (!found) { | ||||||
|  |             LOG_ERROR(Render_Vulkan, "Missing required extension: {}", requirement_name); | ||||||
|  |             has_all_required_extensions = false; | ||||||
|         } |         } | ||||||
|     for (size_t i = 0; i < REQUIRED_EXTENSIONS.size(); ++i) { |  | ||||||
|         if (available_extensions[i]) { |  | ||||||
|             continue; |  | ||||||
|     } |     } | ||||||
|         LOG_ERROR(Render_Vulkan, "Missing required extension: {}", REQUIRED_EXTENSIONS[i]); | 
 | ||||||
|         throw vk::Exception(VK_ERROR_EXTENSION_NOT_PRESENT); |     if (!has_all_required_extensions) { | ||||||
|     } |  | ||||||
|     if (requires_swapchain && !has_swapchain) { |  | ||||||
|         LOG_ERROR(Render_Vulkan, "Missing required extension: VK_KHR_swapchain"); |  | ||||||
|         throw vk::Exception(VK_ERROR_EXTENSION_NOT_PRESENT); |         throw vk::Exception(VK_ERROR_EXTENSION_NOT_PRESENT); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -940,9 +966,8 @@ void Device::CheckSuitability(bool requires_swapchain) const { | ||||||
|             throw vk::Exception(VK_ERROR_FEATURE_NOT_PRESENT); |             throw vk::Exception(VK_ERROR_FEATURE_NOT_PRESENT); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     VkPhysicalDeviceShaderDemoteToHelperInvocationFeaturesEXT demote{}; |     VkPhysicalDeviceShaderDemoteToHelperInvocationFeatures demote{}; | ||||||
|     demote.sType = |     demote.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_SHADER_DEMOTE_TO_HELPER_INVOCATION_FEATURES; | ||||||
|         VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_SHADER_DEMOTE_TO_HELPER_INVOCATION_FEATURES_EXT; |  | ||||||
|     demote.pNext = nullptr; |     demote.pNext = nullptr; | ||||||
| 
 | 
 | ||||||
|     VkPhysicalDeviceVariablePointerFeaturesKHR variable_pointers{}; |     VkPhysicalDeviceVariablePointerFeaturesKHR variable_pointers{}; | ||||||
|  | @ -960,7 +985,7 @@ void Device::CheckSuitability(bool requires_swapchain) const { | ||||||
|     physical.GetFeatures2KHR(features2); |     physical.GetFeatures2KHR(features2); | ||||||
| 
 | 
 | ||||||
|     const VkPhysicalDeviceFeatures& features{features2.features}; |     const VkPhysicalDeviceFeatures& features{features2.features}; | ||||||
|     const std::array feature_report{ |     std::vector feature_report{ | ||||||
|         std::make_pair(features.robustBufferAccess, "robustBufferAccess"), |         std::make_pair(features.robustBufferAccess, "robustBufferAccess"), | ||||||
|         std::make_pair(features.vertexPipelineStoresAndAtomics, "vertexPipelineStoresAndAtomics"), |         std::make_pair(features.vertexPipelineStoresAndAtomics, "vertexPipelineStoresAndAtomics"), | ||||||
|         std::make_pair(features.imageCubeArray, "imageCubeArray"), |         std::make_pair(features.imageCubeArray, "imageCubeArray"), | ||||||
|  | @ -983,27 +1008,30 @@ void Device::CheckSuitability(bool requires_swapchain) const { | ||||||
|                        "shaderStorageImageWriteWithoutFormat"), |                        "shaderStorageImageWriteWithoutFormat"), | ||||||
|         std::make_pair(features.shaderClipDistance, "shaderClipDistance"), |         std::make_pair(features.shaderClipDistance, "shaderClipDistance"), | ||||||
|         std::make_pair(features.shaderCullDistance, "shaderCullDistance"), |         std::make_pair(features.shaderCullDistance, "shaderCullDistance"), | ||||||
|         std::make_pair(demote.shaderDemoteToHelperInvocation, "shaderDemoteToHelperInvocation"), |  | ||||||
|         std::make_pair(variable_pointers.variablePointers, "variablePointers"), |         std::make_pair(variable_pointers.variablePointers, "variablePointers"), | ||||||
|         std::make_pair(variable_pointers.variablePointersStorageBuffer, |         std::make_pair(variable_pointers.variablePointersStorageBuffer, | ||||||
|                        "variablePointersStorageBuffer"), |                        "variablePointersStorageBuffer"), | ||||||
|         std::make_pair(robustness2.robustBufferAccess2, "robustBufferAccess2"), |         std::make_pair(robustness2.robustBufferAccess2, "robustBufferAccess2"), | ||||||
|         std::make_pair(robustness2.robustImageAccess2, "robustImageAccess2"), |         std::make_pair(robustness2.robustImageAccess2, "robustImageAccess2"), | ||||||
|         std::make_pair(robustness2.nullDescriptor, "nullDescriptor"), |         std::make_pair(robustness2.nullDescriptor, "nullDescriptor"), | ||||||
|  |         std::make_pair(demote.shaderDemoteToHelperInvocation, "shaderDemoteToHelperInvocation"), | ||||||
|     }; |     }; | ||||||
|  | 
 | ||||||
|  |     bool has_all_required_features = true; | ||||||
|     for (const auto& [is_supported, name] : feature_report) { |     for (const auto& [is_supported, name] : feature_report) { | ||||||
|         if (is_supported) { |         if (!is_supported) { | ||||||
|             continue; |  | ||||||
|         } |  | ||||||
|             LOG_ERROR(Render_Vulkan, "Missing required feature: {}", name); |             LOG_ERROR(Render_Vulkan, "Missing required feature: {}", name); | ||||||
|  |             has_all_required_features = false; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (!has_all_required_features) { | ||||||
|         throw vk::Exception(VK_ERROR_FEATURE_NOT_PRESENT); |         throw vk::Exception(VK_ERROR_FEATURE_NOT_PRESENT); | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| std::vector<const char*> Device::LoadExtensions(bool requires_surface) { | std::vector<const char*> Device::LoadExtensions(bool requires_surface) { | ||||||
|     std::vector<const char*> extensions; |     std::vector<const char*> extensions = ExtensionsRequiredForInstanceVersion(instance_version); | ||||||
|     extensions.reserve(8 + REQUIRED_EXTENSIONS.size()); |  | ||||||
|     extensions.insert(extensions.begin(), REQUIRED_EXTENSIONS.begin(), REQUIRED_EXTENSIONS.end()); |  | ||||||
|     if (requires_surface) { |     if (requires_surface) { | ||||||
|         extensions.push_back(VK_KHR_SWAPCHAIN_EXTENSION_NAME); |         extensions.push_back(VK_KHR_SWAPCHAIN_EXTENSION_NAME); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -211,11 +211,6 @@ public: | ||||||
|         return khr_uniform_buffer_standard_layout; |         return khr_uniform_buffer_standard_layout; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /// Returns true if the device supports VK_KHR_spirv_1_4.
 |  | ||||||
|     bool IsKhrSpirv1_4Supported() const { |  | ||||||
|         return khr_spirv_1_4; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /// Returns true if the device supports VK_KHR_push_descriptor.
 |     /// Returns true if the device supports VK_KHR_push_descriptor.
 | ||||||
|     bool IsKhrPushDescriptorSupported() const { |     bool IsKhrPushDescriptorSupported() const { | ||||||
|         return khr_push_descriptor; |         return khr_push_descriptor; | ||||||
|  | @ -316,6 +311,17 @@ public: | ||||||
|         return ext_shader_atomic_int64; |         return ext_shader_atomic_int64; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /// Returns the minimum supported version of SPIR-V.
 | ||||||
|  |     u32 SupportedSpirvVersion() const { | ||||||
|  |         if (instance_version >= VK_API_VERSION_1_3) { | ||||||
|  |             return 0x00010600U; | ||||||
|  |         } | ||||||
|  |         if (khr_spirv_1_4) { | ||||||
|  |             return 0x00010400U; | ||||||
|  |         } | ||||||
|  |         return 0x00010000U; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     /// Returns true when a known debugging tool is attached.
 |     /// Returns true when a known debugging tool is attached.
 | ||||||
|     bool HasDebuggingToolAttached() const { |     bool HasDebuggingToolAttached() const { | ||||||
|         return has_renderdoc || has_nsight_graphics; |         return has_renderdoc || has_nsight_graphics; | ||||||
|  |  | ||||||
|  | @ -14,13 +14,15 @@ | ||||||
| #include "video_core/vulkan_common/vulkan_wrapper.h" | #include "video_core/vulkan_common/vulkan_wrapper.h" | ||||||
| 
 | 
 | ||||||
| // Include these late to avoid polluting previous headers
 | // Include these late to avoid polluting previous headers
 | ||||||
| #ifdef _WIN32 | #if defined(_WIN32) | ||||||
| #include <windows.h> | #include <windows.h> | ||||||
| // ensure include order
 | // ensure include order
 | ||||||
| #include <vulkan/vulkan_win32.h> | #include <vulkan/vulkan_win32.h> | ||||||
| #endif | #elif defined(__APPLE__) | ||||||
| 
 | #include <vulkan/vulkan_macos.h> | ||||||
| #if !defined(_WIN32) && !defined(__APPLE__) | #elif defined(__ANDROID__) | ||||||
|  | #include <vulkan/vulkan_android.h> | ||||||
|  | #else | ||||||
| #include <X11/Xlib.h> | #include <X11/Xlib.h> | ||||||
| #include <vulkan/vulkan_wayland.h> | #include <vulkan/vulkan_wayland.h> | ||||||
| #include <vulkan/vulkan_xlib.h> | #include <vulkan/vulkan_xlib.h> | ||||||
|  | @ -39,8 +41,15 @@ namespace { | ||||||
|     case Core::Frontend::WindowSystemType::Windows: |     case Core::Frontend::WindowSystemType::Windows: | ||||||
|         extensions.push_back(VK_KHR_WIN32_SURFACE_EXTENSION_NAME); |         extensions.push_back(VK_KHR_WIN32_SURFACE_EXTENSION_NAME); | ||||||
|         break; |         break; | ||||||
| #endif | #elif defined(__APPLE__) | ||||||
| #if !defined(_WIN32) && !defined(__APPLE__) |     case Core::Frontend::WindowSystemType::Cocoa: | ||||||
|  |         extensions.push_back(VK_MVK_MACOS_SURFACE_EXTENSION_NAME); | ||||||
|  |         break; | ||||||
|  | #elif defined(__ANDROID__) | ||||||
|  |     case Core::Frontend::WindowSystemType::Android: | ||||||
|  |         extensions.push_back(VK_KHR_ANDROID_SURFACE_EXTENSION_NAME); | ||||||
|  |         break; | ||||||
|  | #else | ||||||
|     case Core::Frontend::WindowSystemType::X11: |     case Core::Frontend::WindowSystemType::X11: | ||||||
|         extensions.push_back(VK_KHR_XLIB_SURFACE_EXTENSION_NAME); |         extensions.push_back(VK_KHR_XLIB_SURFACE_EXTENSION_NAME); | ||||||
|         break; |         break; | ||||||
|  | @ -59,6 +68,10 @@ namespace { | ||||||
|         extensions.push_back(VK_EXT_DEBUG_UTILS_EXTENSION_NAME); |         extensions.push_back(VK_EXT_DEBUG_UTILS_EXTENSION_NAME); | ||||||
|     } |     } | ||||||
|     extensions.push_back(VK_KHR_GET_PHYSICAL_DEVICE_PROPERTIES_2_EXTENSION_NAME); |     extensions.push_back(VK_KHR_GET_PHYSICAL_DEVICE_PROPERTIES_2_EXTENSION_NAME); | ||||||
|  | 
 | ||||||
|  | #ifdef __APPLE__ | ||||||
|  |     extensions.push_back(VK_KHR_PORTABILITY_ENUMERATION_EXTENSION_NAME); | ||||||
|  | #endif | ||||||
|     return extensions; |     return extensions; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -140,7 +153,7 @@ vk::Instance CreateInstance(const Common::DynamicLibrary& library, vk::InstanceD | ||||||
|     } |     } | ||||||
|     vk::Instance instance = |     vk::Instance instance = | ||||||
|         std::async([&] { |         std::async([&] { | ||||||
|             return vk::Instance::Create(required_version, layers, extensions, dld); |             return vk::Instance::Create(available_version, layers, extensions, dld); | ||||||
|         }).get(); |         }).get(); | ||||||
|     if (!vk::Load(*instance, dld)) { |     if (!vk::Load(*instance, dld)) { | ||||||
|         LOG_ERROR(Render_Vulkan, "Failed to load Vulkan instance function pointers"); |         LOG_ERROR(Render_Vulkan, "Failed to load Vulkan instance function pointers"); | ||||||
|  |  | ||||||
|  | @ -11,9 +11,11 @@ | ||||||
| #include <windows.h> | #include <windows.h> | ||||||
| // ensure include order
 | // ensure include order
 | ||||||
| #include <vulkan/vulkan_win32.h> | #include <vulkan/vulkan_win32.h> | ||||||
| #endif | #elif defined(__APPLE__) | ||||||
| 
 | #include <vulkan/vulkan_macos.h> | ||||||
| #if !defined(_WIN32) && !defined(__APPLE__) | #elif defined(__ANDROID__) | ||||||
|  | #include <vulkan/vulkan_android.h> | ||||||
|  | #else | ||||||
| #include <X11/Xlib.h> | #include <X11/Xlib.h> | ||||||
| #include <vulkan/vulkan_wayland.h> | #include <vulkan/vulkan_wayland.h> | ||||||
| #include <vulkan/vulkan_xlib.h> | #include <vulkan/vulkan_xlib.h> | ||||||
|  | @ -40,8 +42,33 @@ vk::SurfaceKHR CreateSurface(const vk::Instance& instance, | ||||||
|             throw vk::Exception(VK_ERROR_INITIALIZATION_FAILED); |             throw vk::Exception(VK_ERROR_INITIALIZATION_FAILED); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| #endif | #elif defined(__APPLE__) | ||||||
| #if !defined(_WIN32) && !defined(__APPLE__) |     if (window_info.type == Core::Frontend::WindowSystemType::Cocoa) { | ||||||
|  |         const VkMacOSSurfaceCreateInfoMVK mvk_ci{VK_STRUCTURE_TYPE_MACOS_SURFACE_CREATE_INFO_MVK, | ||||||
|  |                                                  nullptr, 0, window_info.render_surface}; | ||||||
|  |         const auto vkCreateMacOSSurfaceMVK = reinterpret_cast<PFN_vkCreateMacOSSurfaceMVK>( | ||||||
|  |             dld.vkGetInstanceProcAddr(*instance, "vkCreateMacOSSurfaceMVK")); | ||||||
|  |         if (!vkCreateMacOSSurfaceMVK || | ||||||
|  |             vkCreateMacOSSurfaceMVK(*instance, &mvk_ci, nullptr, &unsafe_surface) != VK_SUCCESS) { | ||||||
|  |             LOG_ERROR(Render_Vulkan, "Failed to initialize Metal surface"); | ||||||
|  |             throw vk::Exception(VK_ERROR_INITIALIZATION_FAILED); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | #elif defined(__ANDROID__) | ||||||
|  |     if (window_info.type == Core::Frontend::WindowSystemType::Android) { | ||||||
|  |         const VkAndroidSurfaceCreateInfoKHR android_ci{ | ||||||
|  |             VK_STRUCTURE_TYPE_ANDROID_SURFACE_CREATE_INFO_KHR, nullptr, 0, | ||||||
|  |             reinterpret_cast<ANativeWindow*>(window_info.render_surface)}; | ||||||
|  |         const auto vkCreateAndroidSurfaceKHR = reinterpret_cast<PFN_vkCreateAndroidSurfaceKHR>( | ||||||
|  |             dld.vkGetInstanceProcAddr(*instance, "vkCreateAndroidSurfaceKHR")); | ||||||
|  |         if (!vkCreateAndroidSurfaceKHR || | ||||||
|  |             vkCreateAndroidSurfaceKHR(*instance, &android_ci, nullptr, &unsafe_surface) != | ||||||
|  |                 VK_SUCCESS) { | ||||||
|  |             LOG_ERROR(Render_Vulkan, "Failed to initialize Android surface"); | ||||||
|  |             throw vk::Exception(VK_ERROR_INITIALIZATION_FAILED); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | #else | ||||||
|     if (window_info.type == Core::Frontend::WindowSystemType::X11) { |     if (window_info.type == Core::Frontend::WindowSystemType::X11) { | ||||||
|         const VkXlibSurfaceCreateInfoKHR xlib_ci{ |         const VkXlibSurfaceCreateInfoKHR xlib_ci{ | ||||||
|             VK_STRUCTURE_TYPE_XLIB_SURFACE_CREATE_INFO_KHR, nullptr, 0, |             VK_STRUCTURE_TYPE_XLIB_SURFACE_CREATE_INFO_KHR, nullptr, 0, | ||||||
|  | @ -70,6 +97,7 @@ vk::SurfaceKHR CreateSurface(const vk::Instance& instance, | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| #endif | #endif | ||||||
|  | 
 | ||||||
|     if (!unsafe_surface) { |     if (!unsafe_surface) { | ||||||
|         LOG_ERROR(Render_Vulkan, "Presentation not supported on this platform"); |         LOG_ERROR(Render_Vulkan, "Presentation not supported on this platform"); | ||||||
|         throw vk::Exception(VK_ERROR_INITIALIZATION_FAILED); |         throw vk::Exception(VK_ERROR_INITIALIZATION_FAILED); | ||||||
|  |  | ||||||
|  | @ -267,6 +267,10 @@ static Core::Frontend::WindowSystemType GetWindowSystemType() { | ||||||
|         return Core::Frontend::WindowSystemType::X11; |         return Core::Frontend::WindowSystemType::X11; | ||||||
|     else if (platform_name == QStringLiteral("wayland")) |     else if (platform_name == QStringLiteral("wayland")) | ||||||
|         return Core::Frontend::WindowSystemType::Wayland; |         return Core::Frontend::WindowSystemType::Wayland; | ||||||
|  |     else if (platform_name == QStringLiteral("cocoa")) | ||||||
|  |         return Core::Frontend::WindowSystemType::Cocoa; | ||||||
|  |     else if (platform_name == QStringLiteral("android")) | ||||||
|  |         return Core::Frontend::WindowSystemType::Android; | ||||||
| 
 | 
 | ||||||
|     LOG_CRITICAL(Frontend, "Unknown Qt platform!"); |     LOG_CRITICAL(Frontend, "Unknown Qt platform!"); | ||||||
|     return Core::Frontend::WindowSystemType::Windows; |     return Core::Frontend::WindowSystemType::Windows; | ||||||
|  |  | ||||||
|  | @ -360,7 +360,7 @@ void ConfigureGraphics::RetrieveVulkanDevices() try { | ||||||
| 
 | 
 | ||||||
|     vk::InstanceDispatch dld; |     vk::InstanceDispatch dld; | ||||||
|     const Common::DynamicLibrary library = OpenLibrary(); |     const Common::DynamicLibrary library = OpenLibrary(); | ||||||
|     const vk::Instance instance = CreateInstance(library, dld, VK_API_VERSION_1_0); |     const vk::Instance instance = CreateInstance(library, dld, VK_API_VERSION_1_1); | ||||||
|     const std::vector<VkPhysicalDevice> physical_devices = instance.EnumeratePhysicalDevices(); |     const std::vector<VkPhysicalDevice> physical_devices = instance.EnumeratePhysicalDevices(); | ||||||
| 
 | 
 | ||||||
|     vulkan_devices.clear(); |     vulkan_devices.clear(); | ||||||
|  |  | ||||||
|  | @ -27,7 +27,7 @@ void CheckVulkan() { | ||||||
|         Vulkan::vk::InstanceDispatch dld; |         Vulkan::vk::InstanceDispatch dld; | ||||||
|         const Common::DynamicLibrary library = Vulkan::OpenLibrary(); |         const Common::DynamicLibrary library = Vulkan::OpenLibrary(); | ||||||
|         const Vulkan::vk::Instance instance = |         const Vulkan::vk::Instance instance = | ||||||
|             Vulkan::CreateInstance(library, dld, VK_API_VERSION_1_0); |             Vulkan::CreateInstance(library, dld, VK_API_VERSION_1_1); | ||||||
| 
 | 
 | ||||||
|     } catch (const Vulkan::vk::Exception& exception) { |     } catch (const Vulkan::vk::Exception& exception) { | ||||||
|         fmt::print(stderr, "Failed to initialize Vulkan: {}\n", exception.what()); |         fmt::print(stderr, "Failed to initialize Vulkan: {}\n", exception.what()); | ||||||
|  |  | ||||||
|  | @ -51,11 +51,6 @@ EmuWindow_SDL2_VK::EmuWindow_SDL2_VK(InputCommon::InputSubsystem* input_subsyste | ||||||
|         window_info.type = Core::Frontend::WindowSystemType::Windows; |         window_info.type = Core::Frontend::WindowSystemType::Windows; | ||||||
|         window_info.render_surface = reinterpret_cast<void*>(wm.info.win.window); |         window_info.render_surface = reinterpret_cast<void*>(wm.info.win.window); | ||||||
|         break; |         break; | ||||||
| #else |  | ||||||
|     case SDL_SYSWM_TYPE::SDL_SYSWM_WINDOWS: |  | ||||||
|         LOG_CRITICAL(Frontend, "Window manager subsystem Windows not compiled"); |  | ||||||
|         std::exit(EXIT_FAILURE); |  | ||||||
|         break; |  | ||||||
| #endif | #endif | ||||||
| #ifdef SDL_VIDEO_DRIVER_X11 | #ifdef SDL_VIDEO_DRIVER_X11 | ||||||
|     case SDL_SYSWM_TYPE::SDL_SYSWM_X11: |     case SDL_SYSWM_TYPE::SDL_SYSWM_X11: | ||||||
|  | @ -63,11 +58,6 @@ EmuWindow_SDL2_VK::EmuWindow_SDL2_VK(InputCommon::InputSubsystem* input_subsyste | ||||||
|         window_info.display_connection = wm.info.x11.display; |         window_info.display_connection = wm.info.x11.display; | ||||||
|         window_info.render_surface = reinterpret_cast<void*>(wm.info.x11.window); |         window_info.render_surface = reinterpret_cast<void*>(wm.info.x11.window); | ||||||
|         break; |         break; | ||||||
| #else |  | ||||||
|     case SDL_SYSWM_TYPE::SDL_SYSWM_X11: |  | ||||||
|         LOG_CRITICAL(Frontend, "Window manager subsystem X11 not compiled"); |  | ||||||
|         std::exit(EXIT_FAILURE); |  | ||||||
|         break; |  | ||||||
| #endif | #endif | ||||||
| #ifdef SDL_VIDEO_DRIVER_WAYLAND | #ifdef SDL_VIDEO_DRIVER_WAYLAND | ||||||
|     case SDL_SYSWM_TYPE::SDL_SYSWM_WAYLAND: |     case SDL_SYSWM_TYPE::SDL_SYSWM_WAYLAND: | ||||||
|  | @ -75,14 +65,21 @@ EmuWindow_SDL2_VK::EmuWindow_SDL2_VK(InputCommon::InputSubsystem* input_subsyste | ||||||
|         window_info.display_connection = wm.info.wl.display; |         window_info.display_connection = wm.info.wl.display; | ||||||
|         window_info.render_surface = wm.info.wl.surface; |         window_info.render_surface = wm.info.wl.surface; | ||||||
|         break; |         break; | ||||||
| #else | #endif | ||||||
|     case SDL_SYSWM_TYPE::SDL_SYSWM_WAYLAND: | #ifdef SDL_VIDEO_DRIVER_COCOA | ||||||
|         LOG_CRITICAL(Frontend, "Window manager subsystem Wayland not compiled"); |     case SDL_SYSWM_TYPE::SDL_SYSWM_COCOA: | ||||||
|         std::exit(EXIT_FAILURE); |         window_info.type = Core::Frontend::WindowSystemType::Cocoa; | ||||||
|  |         window_info.render_surface = SDL_Metal_CreateView(render_window); | ||||||
|  |         break; | ||||||
|  | #endif | ||||||
|  | #ifdef SDL_VIDEO_DRIVER_ANDROID | ||||||
|  |     case SDL_SYSWM_TYPE::SDL_SYSWM_ANDROID: | ||||||
|  |         window_info.type = Core::Frontend::WindowSystemType::Android; | ||||||
|  |         window_info.render_surface = reinterpret_cast<void*>(wm.info.android.window); | ||||||
|         break; |         break; | ||||||
| #endif | #endif | ||||||
|     default: |     default: | ||||||
|         LOG_CRITICAL(Frontend, "Window manager subsystem not implemented"); |         LOG_CRITICAL(Frontend, "Window manager subsystem {} not implemented", wm.subsystem); | ||||||
|         std::exit(EXIT_FAILURE); |         std::exit(EXIT_FAILURE); | ||||||
|         break; |         break; | ||||||
|     } |     } | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Matías Locatti
						Matías Locatti