forked from eden-emu/eden
		
	Merge pull request #2191 from ReinUsesLisp/maxwell-to-vk
maxwell_to_vk: Initial implementation
This commit is contained in:
		
						commit
						ae4fd2c619
					
				
					 4 changed files with 553 additions and 3 deletions
				
			
		|  | @ -112,6 +112,8 @@ add_library(video_core STATIC | ||||||
| if (ENABLE_VULKAN) | if (ENABLE_VULKAN) | ||||||
|     target_sources(video_core PRIVATE |     target_sources(video_core PRIVATE | ||||||
|         renderer_vulkan/declarations.h |         renderer_vulkan/declarations.h | ||||||
|  |         renderer_vulkan/maxwell_to_vk.cpp | ||||||
|  |         renderer_vulkan/maxwell_to_vk.h | ||||||
|         renderer_vulkan/vk_buffer_cache.cpp |         renderer_vulkan/vk_buffer_cache.cpp | ||||||
|         renderer_vulkan/vk_buffer_cache.h |         renderer_vulkan/vk_buffer_cache.h | ||||||
|         renderer_vulkan/vk_device.cpp |         renderer_vulkan/vk_device.cpp | ||||||
|  |  | ||||||
							
								
								
									
										483
									
								
								src/video_core/renderer_vulkan/maxwell_to_vk.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										483
									
								
								src/video_core/renderer_vulkan/maxwell_to_vk.cpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,483 @@ | ||||||
|  | // Copyright 2019 yuzu Emulator Project
 | ||||||
|  | // Licensed under GPLv2 or any later version
 | ||||||
|  | // Refer to the license.txt file included.
 | ||||||
|  | 
 | ||||||
|  | #include "common/assert.h" | ||||||
|  | #include "common/common_types.h" | ||||||
|  | #include "common/logging/log.h" | ||||||
|  | #include "video_core/engines/maxwell_3d.h" | ||||||
|  | #include "video_core/renderer_vulkan/declarations.h" | ||||||
|  | #include "video_core/renderer_vulkan/maxwell_to_vk.h" | ||||||
|  | #include "video_core/renderer_vulkan/vk_device.h" | ||||||
|  | #include "video_core/surface.h" | ||||||
|  | 
 | ||||||
|  | namespace Vulkan::MaxwellToVK { | ||||||
|  | 
 | ||||||
|  | namespace Sampler { | ||||||
|  | 
 | ||||||
|  | vk::Filter Filter(Tegra::Texture::TextureFilter filter) { | ||||||
|  |     switch (filter) { | ||||||
|  |     case Tegra::Texture::TextureFilter::Linear: | ||||||
|  |         return vk::Filter::eLinear; | ||||||
|  |     case Tegra::Texture::TextureFilter::Nearest: | ||||||
|  |         return vk::Filter::eNearest; | ||||||
|  |     } | ||||||
|  |     UNIMPLEMENTED_MSG("Unimplemented sampler filter={}", static_cast<u32>(filter)); | ||||||
|  |     return {}; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | vk::SamplerMipmapMode MipmapMode(Tegra::Texture::TextureMipmapFilter mipmap_filter) { | ||||||
|  |     switch (mipmap_filter) { | ||||||
|  |     case Tegra::Texture::TextureMipmapFilter::None: | ||||||
|  |         // TODO(Rodrigo): None seems to be mapped to OpenGL's mag and min filters without mipmapping
 | ||||||
|  |         // (e.g. GL_NEAREST and GL_LINEAR). Vulkan doesn't have such a thing, find out if we have to
 | ||||||
|  |         // use an image view with a single mipmap level to emulate this.
 | ||||||
|  |         return vk::SamplerMipmapMode::eLinear; | ||||||
|  |     case Tegra::Texture::TextureMipmapFilter::Linear: | ||||||
|  |         return vk::SamplerMipmapMode::eLinear; | ||||||
|  |     case Tegra::Texture::TextureMipmapFilter::Nearest: | ||||||
|  |         return vk::SamplerMipmapMode::eNearest; | ||||||
|  |     } | ||||||
|  |     UNIMPLEMENTED_MSG("Unimplemented sampler mipmap mode={}", static_cast<u32>(mipmap_filter)); | ||||||
|  |     return {}; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | vk::SamplerAddressMode WrapMode(Tegra::Texture::WrapMode wrap_mode) { | ||||||
|  |     switch (wrap_mode) { | ||||||
|  |     case Tegra::Texture::WrapMode::Wrap: | ||||||
|  |         return vk::SamplerAddressMode::eRepeat; | ||||||
|  |     case Tegra::Texture::WrapMode::Mirror: | ||||||
|  |         return vk::SamplerAddressMode::eMirroredRepeat; | ||||||
|  |     case Tegra::Texture::WrapMode::ClampToEdge: | ||||||
|  |         return vk::SamplerAddressMode::eClampToEdge; | ||||||
|  |     case Tegra::Texture::WrapMode::Border: | ||||||
|  |         return vk::SamplerAddressMode::eClampToBorder; | ||||||
|  |     case Tegra::Texture::WrapMode::ClampOGL: | ||||||
|  |         // TODO(Rodrigo): GL_CLAMP was removed as of OpenGL 3.1, to implement GL_CLAMP, we can use
 | ||||||
|  |         // eClampToBorder to get the border color of the texture, and then sample the edge to
 | ||||||
|  |         // manually mix them. However the shader part of this is not yet implemented.
 | ||||||
|  |         return vk::SamplerAddressMode::eClampToBorder; | ||||||
|  |     case Tegra::Texture::WrapMode::MirrorOnceClampToEdge: | ||||||
|  |         return vk::SamplerAddressMode::eMirrorClampToEdge; | ||||||
|  |     case Tegra::Texture::WrapMode::MirrorOnceBorder: | ||||||
|  |         UNIMPLEMENTED(); | ||||||
|  |         return vk::SamplerAddressMode::eMirrorClampToEdge; | ||||||
|  |     } | ||||||
|  |     UNIMPLEMENTED_MSG("Unimplemented wrap mode={}", static_cast<u32>(wrap_mode)); | ||||||
|  |     return {}; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | vk::CompareOp DepthCompareFunction(Tegra::Texture::DepthCompareFunc depth_compare_func) { | ||||||
|  |     switch (depth_compare_func) { | ||||||
|  |     case Tegra::Texture::DepthCompareFunc::Never: | ||||||
|  |         return vk::CompareOp::eNever; | ||||||
|  |     case Tegra::Texture::DepthCompareFunc::Less: | ||||||
|  |         return vk::CompareOp::eLess; | ||||||
|  |     case Tegra::Texture::DepthCompareFunc::LessEqual: | ||||||
|  |         return vk::CompareOp::eLessOrEqual; | ||||||
|  |     case Tegra::Texture::DepthCompareFunc::Equal: | ||||||
|  |         return vk::CompareOp::eEqual; | ||||||
|  |     case Tegra::Texture::DepthCompareFunc::NotEqual: | ||||||
|  |         return vk::CompareOp::eNotEqual; | ||||||
|  |     case Tegra::Texture::DepthCompareFunc::Greater: | ||||||
|  |         return vk::CompareOp::eGreater; | ||||||
|  |     case Tegra::Texture::DepthCompareFunc::GreaterEqual: | ||||||
|  |         return vk::CompareOp::eGreaterOrEqual; | ||||||
|  |     case Tegra::Texture::DepthCompareFunc::Always: | ||||||
|  |         return vk::CompareOp::eAlways; | ||||||
|  |     } | ||||||
|  |     UNIMPLEMENTED_MSG("Unimplemented sampler depth compare function={}", | ||||||
|  |                       static_cast<u32>(depth_compare_func)); | ||||||
|  |     return {}; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | } // namespace Sampler
 | ||||||
|  | 
 | ||||||
|  | struct FormatTuple { | ||||||
|  |     vk::Format format;            ///< Vulkan format
 | ||||||
|  |     ComponentType component_type; ///< Abstracted component type
 | ||||||
|  |     bool attachable;              ///< True when this format can be used as an attachment
 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static constexpr std::array<FormatTuple, VideoCore::Surface::MaxPixelFormat> tex_format_tuples = {{ | ||||||
|  |     {vk::Format::eA8B8G8R8UnormPack32, ComponentType::UNorm, true},    // ABGR8U
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // ABGR8S
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // ABGR8UI
 | ||||||
|  |     {vk::Format::eB5G6R5UnormPack16, ComponentType::UNorm, false},     // B5G6R5U
 | ||||||
|  |     {vk::Format::eA2B10G10R10UnormPack32, ComponentType::UNorm, true}, // A2B10G10R10U
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // A1B5G5R5U
 | ||||||
|  |     {vk::Format::eR8Unorm, ComponentType::UNorm, true},                // R8U
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // R8UI
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // RGBA16F
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // RGBA16U
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // RGBA16UI
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // R11FG11FB10F
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // RGBA32UI
 | ||||||
|  |     {vk::Format::eBc1RgbaUnormBlock, ComponentType::UNorm, false},     // DXT1
 | ||||||
|  |     {vk::Format::eBc2UnormBlock, ComponentType::UNorm, false},         // DXT23
 | ||||||
|  |     {vk::Format::eBc3UnormBlock, ComponentType::UNorm, false},         // DXT45
 | ||||||
|  |     {vk::Format::eBc4UnormBlock, ComponentType::UNorm, false},         // DXN1
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // DXN2UNORM
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // DXN2SNORM
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // BC7U
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // BC6H_UF16
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // BC6H_SF16
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // ASTC_2D_4X4
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // BGRA8
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // RGBA32F
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // RG32F
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // R32F
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // R16F
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // R16U
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // R16S
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // R16UI
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // R16I
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // RG16
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // RG16F
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // RG16UI
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // RG16I
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // RG16S
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // RGB32F
 | ||||||
|  |     {vk::Format::eA8B8G8R8SrgbPack32, ComponentType::UNorm, true},     // RGBA8_SRGB
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // RG8U
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // RG8S
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // RG32UI
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // R32UI
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // ASTC_2D_8X8
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // ASTC_2D_8X5
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},           // ASTC_2D_5X4
 | ||||||
|  | 
 | ||||||
|  |     // Compressed sRGB formats
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false}, // BGRA8_SRGB
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false}, // DXT1_SRGB
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false}, // DXT23_SRGB
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false}, // DXT45_SRGB
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false}, // BC7U_SRGB
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false}, // ASTC_2D_4X4_SRGB
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false}, // ASTC_2D_8X8_SRGB
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false}, // ASTC_2D_8X5_SRGB
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false}, // ASTC_2D_5X4_SRGB
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false}, // ASTC_2D_5X5
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false}, // ASTC_2D_5X5_SRGB
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false}, // ASTC_2D_10X8
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false}, // ASTC_2D_10X8_SRGB
 | ||||||
|  | 
 | ||||||
|  |     // Depth formats
 | ||||||
|  |     {vk::Format::eD32Sfloat, ComponentType::Float, true}, // Z32F
 | ||||||
|  |     {vk::Format::eD16Unorm, ComponentType::UNorm, true},  // Z16
 | ||||||
|  | 
 | ||||||
|  |     // DepthStencil formats
 | ||||||
|  |     {vk::Format::eD24UnormS8Uint, ComponentType::UNorm, true}, // Z24S8
 | ||||||
|  |     {vk::Format::eD24UnormS8Uint, ComponentType::UNorm, true}, // S8Z24 (emulated)
 | ||||||
|  |     {vk::Format::eUndefined, ComponentType::Invalid, false},   // Z32FS8
 | ||||||
|  | }}; | ||||||
|  | 
 | ||||||
|  | static constexpr bool IsZetaFormat(PixelFormat pixel_format) { | ||||||
|  |     return pixel_format >= PixelFormat::MaxColorFormat && | ||||||
|  |            pixel_format < PixelFormat::MaxDepthStencilFormat; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | std::pair<vk::Format, bool> SurfaceFormat(const VKDevice& device, FormatType format_type, | ||||||
|  |                                           PixelFormat pixel_format, ComponentType component_type) { | ||||||
|  |     ASSERT(static_cast<std::size_t>(pixel_format) < tex_format_tuples.size()); | ||||||
|  | 
 | ||||||
|  |     const auto tuple = tex_format_tuples[static_cast<u32>(pixel_format)]; | ||||||
|  |     UNIMPLEMENTED_IF_MSG(tuple.format == vk::Format::eUndefined, | ||||||
|  |                          "Unimplemented texture format with pixel format={} and component type={}", | ||||||
|  |                          static_cast<u32>(pixel_format), static_cast<u32>(component_type)); | ||||||
|  |     ASSERT_MSG(component_type == tuple.component_type, "Component type mismatch"); | ||||||
|  | 
 | ||||||
|  |     auto usage = vk::FormatFeatureFlagBits::eSampledImage | | ||||||
|  |                  vk::FormatFeatureFlagBits::eTransferDst | vk::FormatFeatureFlagBits::eTransferSrc; | ||||||
|  |     if (tuple.attachable) { | ||||||
|  |         usage |= IsZetaFormat(pixel_format) ? vk::FormatFeatureFlagBits::eDepthStencilAttachment | ||||||
|  |                                             : vk::FormatFeatureFlagBits::eColorAttachment; | ||||||
|  |     } | ||||||
|  |     return {device.GetSupportedFormat(tuple.format, usage, format_type), tuple.attachable}; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | vk::ShaderStageFlagBits ShaderStage(Maxwell::ShaderStage stage) { | ||||||
|  |     switch (stage) { | ||||||
|  |     case Maxwell::ShaderStage::Vertex: | ||||||
|  |         return vk::ShaderStageFlagBits::eVertex; | ||||||
|  |     case Maxwell::ShaderStage::TesselationControl: | ||||||
|  |         return vk::ShaderStageFlagBits::eTessellationControl; | ||||||
|  |     case Maxwell::ShaderStage::TesselationEval: | ||||||
|  |         return vk::ShaderStageFlagBits::eTessellationEvaluation; | ||||||
|  |     case Maxwell::ShaderStage::Geometry: | ||||||
|  |         return vk::ShaderStageFlagBits::eGeometry; | ||||||
|  |     case Maxwell::ShaderStage::Fragment: | ||||||
|  |         return vk::ShaderStageFlagBits::eFragment; | ||||||
|  |     } | ||||||
|  |     UNIMPLEMENTED_MSG("Unimplemented shader stage={}", static_cast<u32>(stage)); | ||||||
|  |     return {}; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | vk::PrimitiveTopology PrimitiveTopology(Maxwell::PrimitiveTopology topology) { | ||||||
|  |     switch (topology) { | ||||||
|  |     case Maxwell::PrimitiveTopology::Points: | ||||||
|  |         return vk::PrimitiveTopology::ePointList; | ||||||
|  |     case Maxwell::PrimitiveTopology::Lines: | ||||||
|  |         return vk::PrimitiveTopology::eLineList; | ||||||
|  |     case Maxwell::PrimitiveTopology::LineStrip: | ||||||
|  |         return vk::PrimitiveTopology::eLineStrip; | ||||||
|  |     case Maxwell::PrimitiveTopology::Triangles: | ||||||
|  |         return vk::PrimitiveTopology::eTriangleList; | ||||||
|  |     case Maxwell::PrimitiveTopology::TriangleStrip: | ||||||
|  |         return vk::PrimitiveTopology::eTriangleStrip; | ||||||
|  |     } | ||||||
|  |     UNIMPLEMENTED_MSG("Unimplemented topology={}", static_cast<u32>(topology)); | ||||||
|  |     return {}; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | vk::Format VertexFormat(Maxwell::VertexAttribute::Type type, Maxwell::VertexAttribute::Size size) { | ||||||
|  |     switch (type) { | ||||||
|  |     case Maxwell::VertexAttribute::Type::SignedNorm: | ||||||
|  |         break; | ||||||
|  |     case Maxwell::VertexAttribute::Type::UnsignedNorm: | ||||||
|  |         switch (size) { | ||||||
|  |         case Maxwell::VertexAttribute::Size::Size_8_8_8_8: | ||||||
|  |             return vk::Format::eR8G8B8A8Unorm; | ||||||
|  |         default: | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |         break; | ||||||
|  |     case Maxwell::VertexAttribute::Type::SignedInt: | ||||||
|  |         break; | ||||||
|  |     case Maxwell::VertexAttribute::Type::UnsignedInt: | ||||||
|  |         switch (size) { | ||||||
|  |         case Maxwell::VertexAttribute::Size::Size_32: | ||||||
|  |             return vk::Format::eR32Uint; | ||||||
|  |         default: | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |     case Maxwell::VertexAttribute::Type::UnsignedScaled: | ||||||
|  |     case Maxwell::VertexAttribute::Type::SignedScaled: | ||||||
|  |         break; | ||||||
|  |     case Maxwell::VertexAttribute::Type::Float: | ||||||
|  |         switch (size) { | ||||||
|  |         case Maxwell::VertexAttribute::Size::Size_32_32_32_32: | ||||||
|  |             return vk::Format::eR32G32B32A32Sfloat; | ||||||
|  |         case Maxwell::VertexAttribute::Size::Size_32_32_32: | ||||||
|  |             return vk::Format::eR32G32B32Sfloat; | ||||||
|  |         case Maxwell::VertexAttribute::Size::Size_32_32: | ||||||
|  |             return vk::Format::eR32G32Sfloat; | ||||||
|  |         case Maxwell::VertexAttribute::Size::Size_32: | ||||||
|  |             return vk::Format::eR32Sfloat; | ||||||
|  |         default: | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  |     UNIMPLEMENTED_MSG("Unimplemented vertex format of type={} and size={}", static_cast<u32>(type), | ||||||
|  |                       static_cast<u32>(size)); | ||||||
|  |     return {}; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | vk::CompareOp ComparisonOp(Maxwell::ComparisonOp comparison) { | ||||||
|  |     switch (comparison) { | ||||||
|  |     case Maxwell::ComparisonOp::Never: | ||||||
|  |     case Maxwell::ComparisonOp::NeverOld: | ||||||
|  |         return vk::CompareOp::eNever; | ||||||
|  |     case Maxwell::ComparisonOp::Less: | ||||||
|  |     case Maxwell::ComparisonOp::LessOld: | ||||||
|  |         return vk::CompareOp::eLess; | ||||||
|  |     case Maxwell::ComparisonOp::Equal: | ||||||
|  |     case Maxwell::ComparisonOp::EqualOld: | ||||||
|  |         return vk::CompareOp::eEqual; | ||||||
|  |     case Maxwell::ComparisonOp::LessEqual: | ||||||
|  |     case Maxwell::ComparisonOp::LessEqualOld: | ||||||
|  |         return vk::CompareOp::eLessOrEqual; | ||||||
|  |     case Maxwell::ComparisonOp::Greater: | ||||||
|  |     case Maxwell::ComparisonOp::GreaterOld: | ||||||
|  |         return vk::CompareOp::eGreater; | ||||||
|  |     case Maxwell::ComparisonOp::NotEqual: | ||||||
|  |     case Maxwell::ComparisonOp::NotEqualOld: | ||||||
|  |         return vk::CompareOp::eNotEqual; | ||||||
|  |     case Maxwell::ComparisonOp::GreaterEqual: | ||||||
|  |     case Maxwell::ComparisonOp::GreaterEqualOld: | ||||||
|  |         return vk::CompareOp::eGreaterOrEqual; | ||||||
|  |     case Maxwell::ComparisonOp::Always: | ||||||
|  |     case Maxwell::ComparisonOp::AlwaysOld: | ||||||
|  |         return vk::CompareOp::eAlways; | ||||||
|  |     } | ||||||
|  |     UNIMPLEMENTED_MSG("Unimplemented comparison op={}", static_cast<u32>(comparison)); | ||||||
|  |     return {}; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | vk::IndexType IndexFormat(Maxwell::IndexFormat index_format) { | ||||||
|  |     switch (index_format) { | ||||||
|  |     case Maxwell::IndexFormat::UnsignedByte: | ||||||
|  |         UNIMPLEMENTED_MSG("Vulkan does not support native u8 index format"); | ||||||
|  |         return vk::IndexType::eUint16; | ||||||
|  |     case Maxwell::IndexFormat::UnsignedShort: | ||||||
|  |         return vk::IndexType::eUint16; | ||||||
|  |     case Maxwell::IndexFormat::UnsignedInt: | ||||||
|  |         return vk::IndexType::eUint32; | ||||||
|  |     } | ||||||
|  |     UNIMPLEMENTED_MSG("Unimplemented index_format={}", static_cast<u32>(index_format)); | ||||||
|  |     return {}; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | vk::StencilOp StencilOp(Maxwell::StencilOp stencil_op) { | ||||||
|  |     switch (stencil_op) { | ||||||
|  |     case Maxwell::StencilOp::Keep: | ||||||
|  |     case Maxwell::StencilOp::KeepOGL: | ||||||
|  |         return vk::StencilOp::eKeep; | ||||||
|  |     case Maxwell::StencilOp::Zero: | ||||||
|  |     case Maxwell::StencilOp::ZeroOGL: | ||||||
|  |         return vk::StencilOp::eZero; | ||||||
|  |     case Maxwell::StencilOp::Replace: | ||||||
|  |     case Maxwell::StencilOp::ReplaceOGL: | ||||||
|  |         return vk::StencilOp::eReplace; | ||||||
|  |     case Maxwell::StencilOp::Incr: | ||||||
|  |     case Maxwell::StencilOp::IncrOGL: | ||||||
|  |         return vk::StencilOp::eIncrementAndClamp; | ||||||
|  |     case Maxwell::StencilOp::Decr: | ||||||
|  |     case Maxwell::StencilOp::DecrOGL: | ||||||
|  |         return vk::StencilOp::eDecrementAndClamp; | ||||||
|  |     case Maxwell::StencilOp::Invert: | ||||||
|  |     case Maxwell::StencilOp::InvertOGL: | ||||||
|  |         return vk::StencilOp::eInvert; | ||||||
|  |     case Maxwell::StencilOp::IncrWrap: | ||||||
|  |     case Maxwell::StencilOp::IncrWrapOGL: | ||||||
|  |         return vk::StencilOp::eIncrementAndWrap; | ||||||
|  |     case Maxwell::StencilOp::DecrWrap: | ||||||
|  |     case Maxwell::StencilOp::DecrWrapOGL: | ||||||
|  |         return vk::StencilOp::eDecrementAndWrap; | ||||||
|  |     } | ||||||
|  |     UNIMPLEMENTED_MSG("Unimplemented stencil op={}", static_cast<u32>(stencil_op)); | ||||||
|  |     return {}; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | vk::BlendOp BlendEquation(Maxwell::Blend::Equation equation) { | ||||||
|  |     switch (equation) { | ||||||
|  |     case Maxwell::Blend::Equation::Add: | ||||||
|  |     case Maxwell::Blend::Equation::AddGL: | ||||||
|  |         return vk::BlendOp::eAdd; | ||||||
|  |     case Maxwell::Blend::Equation::Subtract: | ||||||
|  |     case Maxwell::Blend::Equation::SubtractGL: | ||||||
|  |         return vk::BlendOp::eSubtract; | ||||||
|  |     case Maxwell::Blend::Equation::ReverseSubtract: | ||||||
|  |     case Maxwell::Blend::Equation::ReverseSubtractGL: | ||||||
|  |         return vk::BlendOp::eReverseSubtract; | ||||||
|  |     case Maxwell::Blend::Equation::Min: | ||||||
|  |     case Maxwell::Blend::Equation::MinGL: | ||||||
|  |         return vk::BlendOp::eMin; | ||||||
|  |     case Maxwell::Blend::Equation::Max: | ||||||
|  |     case Maxwell::Blend::Equation::MaxGL: | ||||||
|  |         return vk::BlendOp::eMax; | ||||||
|  |     } | ||||||
|  |     UNIMPLEMENTED_MSG("Unimplemented blend equation={}", static_cast<u32>(equation)); | ||||||
|  |     return {}; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | vk::BlendFactor BlendFactor(Maxwell::Blend::Factor factor) { | ||||||
|  |     switch (factor) { | ||||||
|  |     case Maxwell::Blend::Factor::Zero: | ||||||
|  |     case Maxwell::Blend::Factor::ZeroGL: | ||||||
|  |         return vk::BlendFactor::eZero; | ||||||
|  |     case Maxwell::Blend::Factor::One: | ||||||
|  |     case Maxwell::Blend::Factor::OneGL: | ||||||
|  |         return vk::BlendFactor::eOne; | ||||||
|  |     case Maxwell::Blend::Factor::SourceColor: | ||||||
|  |     case Maxwell::Blend::Factor::SourceColorGL: | ||||||
|  |         return vk::BlendFactor::eSrcColor; | ||||||
|  |     case Maxwell::Blend::Factor::OneMinusSourceColor: | ||||||
|  |     case Maxwell::Blend::Factor::OneMinusSourceColorGL: | ||||||
|  |         return vk::BlendFactor::eOneMinusSrcColor; | ||||||
|  |     case Maxwell::Blend::Factor::SourceAlpha: | ||||||
|  |     case Maxwell::Blend::Factor::SourceAlphaGL: | ||||||
|  |         return vk::BlendFactor::eSrcAlpha; | ||||||
|  |     case Maxwell::Blend::Factor::OneMinusSourceAlpha: | ||||||
|  |     case Maxwell::Blend::Factor::OneMinusSourceAlphaGL: | ||||||
|  |         return vk::BlendFactor::eOneMinusSrcAlpha; | ||||||
|  |     case Maxwell::Blend::Factor::DestAlpha: | ||||||
|  |     case Maxwell::Blend::Factor::DestAlphaGL: | ||||||
|  |         return vk::BlendFactor::eDstAlpha; | ||||||
|  |     case Maxwell::Blend::Factor::OneMinusDestAlpha: | ||||||
|  |     case Maxwell::Blend::Factor::OneMinusDestAlphaGL: | ||||||
|  |         return vk::BlendFactor::eOneMinusDstAlpha; | ||||||
|  |     case Maxwell::Blend::Factor::DestColor: | ||||||
|  |     case Maxwell::Blend::Factor::DestColorGL: | ||||||
|  |         return vk::BlendFactor::eDstColor; | ||||||
|  |     case Maxwell::Blend::Factor::OneMinusDestColor: | ||||||
|  |     case Maxwell::Blend::Factor::OneMinusDestColorGL: | ||||||
|  |         return vk::BlendFactor::eOneMinusDstColor; | ||||||
|  |     case Maxwell::Blend::Factor::SourceAlphaSaturate: | ||||||
|  |     case Maxwell::Blend::Factor::SourceAlphaSaturateGL: | ||||||
|  |         return vk::BlendFactor::eSrcAlphaSaturate; | ||||||
|  |     case Maxwell::Blend::Factor::Source1Color: | ||||||
|  |     case Maxwell::Blend::Factor::Source1ColorGL: | ||||||
|  |         return vk::BlendFactor::eSrc1Color; | ||||||
|  |     case Maxwell::Blend::Factor::OneMinusSource1Color: | ||||||
|  |     case Maxwell::Blend::Factor::OneMinusSource1ColorGL: | ||||||
|  |         return vk::BlendFactor::eOneMinusSrc1Color; | ||||||
|  |     case Maxwell::Blend::Factor::Source1Alpha: | ||||||
|  |     case Maxwell::Blend::Factor::Source1AlphaGL: | ||||||
|  |         return vk::BlendFactor::eSrc1Alpha; | ||||||
|  |     case Maxwell::Blend::Factor::OneMinusSource1Alpha: | ||||||
|  |     case Maxwell::Blend::Factor::OneMinusSource1AlphaGL: | ||||||
|  |         return vk::BlendFactor::eOneMinusSrc1Alpha; | ||||||
|  |     case Maxwell::Blend::Factor::ConstantColor: | ||||||
|  |     case Maxwell::Blend::Factor::ConstantColorGL: | ||||||
|  |         return vk::BlendFactor::eConstantColor; | ||||||
|  |     case Maxwell::Blend::Factor::OneMinusConstantColor: | ||||||
|  |     case Maxwell::Blend::Factor::OneMinusConstantColorGL: | ||||||
|  |         return vk::BlendFactor::eOneMinusConstantColor; | ||||||
|  |     case Maxwell::Blend::Factor::ConstantAlpha: | ||||||
|  |     case Maxwell::Blend::Factor::ConstantAlphaGL: | ||||||
|  |         return vk::BlendFactor::eConstantAlpha; | ||||||
|  |     case Maxwell::Blend::Factor::OneMinusConstantAlpha: | ||||||
|  |     case Maxwell::Blend::Factor::OneMinusConstantAlphaGL: | ||||||
|  |         return vk::BlendFactor::eOneMinusConstantAlpha; | ||||||
|  |     } | ||||||
|  |     UNIMPLEMENTED_MSG("Unimplemented blend factor={}", static_cast<u32>(factor)); | ||||||
|  |     return {}; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | vk::FrontFace FrontFace(Maxwell::Cull::FrontFace front_face) { | ||||||
|  |     switch (front_face) { | ||||||
|  |     case Maxwell::Cull::FrontFace::ClockWise: | ||||||
|  |         return vk::FrontFace::eClockwise; | ||||||
|  |     case Maxwell::Cull::FrontFace::CounterClockWise: | ||||||
|  |         return vk::FrontFace::eCounterClockwise; | ||||||
|  |     } | ||||||
|  |     UNIMPLEMENTED_MSG("Unimplemented front face={}", static_cast<u32>(front_face)); | ||||||
|  |     return {}; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | vk::CullModeFlags CullFace(Maxwell::Cull::CullFace cull_face) { | ||||||
|  |     switch (cull_face) { | ||||||
|  |     case Maxwell::Cull::CullFace::Front: | ||||||
|  |         return vk::CullModeFlagBits::eFront; | ||||||
|  |     case Maxwell::Cull::CullFace::Back: | ||||||
|  |         return vk::CullModeFlagBits::eBack; | ||||||
|  |     case Maxwell::Cull::CullFace::FrontAndBack: | ||||||
|  |         return vk::CullModeFlagBits::eFrontAndBack; | ||||||
|  |     } | ||||||
|  |     UNIMPLEMENTED_MSG("Unimplemented cull face={}", static_cast<u32>(cull_face)); | ||||||
|  |     return {}; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | vk::ComponentSwizzle SwizzleSource(Tegra::Texture::SwizzleSource swizzle) { | ||||||
|  |     switch (swizzle) { | ||||||
|  |     case Tegra::Texture::SwizzleSource::Zero: | ||||||
|  |         return vk::ComponentSwizzle::eZero; | ||||||
|  |     case Tegra::Texture::SwizzleSource::R: | ||||||
|  |         return vk::ComponentSwizzle::eR; | ||||||
|  |     case Tegra::Texture::SwizzleSource::G: | ||||||
|  |         return vk::ComponentSwizzle::eG; | ||||||
|  |     case Tegra::Texture::SwizzleSource::B: | ||||||
|  |         return vk::ComponentSwizzle::eB; | ||||||
|  |     case Tegra::Texture::SwizzleSource::A: | ||||||
|  |         return vk::ComponentSwizzle::eA; | ||||||
|  |     case Tegra::Texture::SwizzleSource::OneInt: | ||||||
|  |     case Tegra::Texture::SwizzleSource::OneFloat: | ||||||
|  |         return vk::ComponentSwizzle::eOne; | ||||||
|  |     } | ||||||
|  |     UNIMPLEMENTED_MSG("Unimplemented swizzle source={}", static_cast<u32>(swizzle)); | ||||||
|  |     return {}; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | } // namespace Vulkan::MaxwellToVK
 | ||||||
							
								
								
									
										58
									
								
								src/video_core/renderer_vulkan/maxwell_to_vk.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								src/video_core/renderer_vulkan/maxwell_to_vk.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,58 @@ | ||||||
|  | // Copyright 2019 yuzu Emulator Project
 | ||||||
|  | // Licensed under GPLv2 or any later version
 | ||||||
|  | // Refer to the license.txt file included.
 | ||||||
|  | 
 | ||||||
|  | #pragma once | ||||||
|  | 
 | ||||||
|  | #include <utility> | ||||||
|  | #include "common/common_types.h" | ||||||
|  | #include "video_core/engines/maxwell_3d.h" | ||||||
|  | #include "video_core/renderer_vulkan/declarations.h" | ||||||
|  | #include "video_core/renderer_vulkan/vk_device.h" | ||||||
|  | #include "video_core/surface.h" | ||||||
|  | #include "video_core/textures/texture.h" | ||||||
|  | 
 | ||||||
|  | namespace Vulkan::MaxwellToVK { | ||||||
|  | 
 | ||||||
|  | using Maxwell = Tegra::Engines::Maxwell3D::Regs; | ||||||
|  | using PixelFormat = VideoCore::Surface::PixelFormat; | ||||||
|  | using ComponentType = VideoCore::Surface::ComponentType; | ||||||
|  | 
 | ||||||
|  | namespace Sampler { | ||||||
|  | 
 | ||||||
|  | vk::Filter Filter(Tegra::Texture::TextureFilter filter); | ||||||
|  | 
 | ||||||
|  | vk::SamplerMipmapMode MipmapMode(Tegra::Texture::TextureMipmapFilter mipmap_filter); | ||||||
|  | 
 | ||||||
|  | vk::SamplerAddressMode WrapMode(Tegra::Texture::WrapMode wrap_mode); | ||||||
|  | 
 | ||||||
|  | vk::CompareOp DepthCompareFunction(Tegra::Texture::DepthCompareFunc depth_compare_func); | ||||||
|  | 
 | ||||||
|  | } // namespace Sampler
 | ||||||
|  | 
 | ||||||
|  | std::pair<vk::Format, bool> SurfaceFormat(const VKDevice& device, FormatType format_type, | ||||||
|  |                                           PixelFormat pixel_format, ComponentType component_type); | ||||||
|  | 
 | ||||||
|  | vk::ShaderStageFlagBits ShaderStage(Maxwell::ShaderStage stage); | ||||||
|  | 
 | ||||||
|  | vk::PrimitiveTopology PrimitiveTopology(Maxwell::PrimitiveTopology topology); | ||||||
|  | 
 | ||||||
|  | vk::Format VertexFormat(Maxwell::VertexAttribute::Type type, Maxwell::VertexAttribute::Size size); | ||||||
|  | 
 | ||||||
|  | vk::CompareOp ComparisonOp(Maxwell::ComparisonOp comparison); | ||||||
|  | 
 | ||||||
|  | vk::IndexType IndexFormat(Maxwell::IndexFormat index_format); | ||||||
|  | 
 | ||||||
|  | vk::StencilOp StencilOp(Maxwell::StencilOp stencil_op); | ||||||
|  | 
 | ||||||
|  | vk::BlendOp BlendEquation(Maxwell::Blend::Equation equation); | ||||||
|  | 
 | ||||||
|  | vk::BlendFactor BlendFactor(Maxwell::Blend::Factor factor); | ||||||
|  | 
 | ||||||
|  | vk::FrontFace FrontFace(Maxwell::Cull::FrontFace front_face); | ||||||
|  | 
 | ||||||
|  | vk::CullModeFlags CullFace(Maxwell::Cull::CullFace cull_face); | ||||||
|  | 
 | ||||||
|  | vk::ComponentSwizzle SwizzleSource(Tegra::Texture::SwizzleSource swizzle); | ||||||
|  | 
 | ||||||
|  | } // namespace Vulkan::MaxwellToVK
 | ||||||
|  | @ -122,8 +122,7 @@ bool VKDevice::IsFormatSupported(vk::Format wanted_format, vk::FormatFeatureFlag | ||||||
|                                  FormatType format_type) const { |                                  FormatType format_type) const { | ||||||
|     const auto it = format_properties.find(wanted_format); |     const auto it = format_properties.find(wanted_format); | ||||||
|     if (it == format_properties.end()) { |     if (it == format_properties.end()) { | ||||||
|         LOG_CRITICAL(Render_Vulkan, "Unimplemented format query={}", |         LOG_CRITICAL(Render_Vulkan, "Unimplemented format query={}", vk::to_string(wanted_format)); | ||||||
|                      static_cast<u32>(wanted_format)); |  | ||||||
|         UNREACHABLE(); |         UNREACHABLE(); | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
|  | @ -219,11 +218,19 @@ std::map<vk::Format, vk::FormatProperties> VKDevice::GetFormatProperties( | ||||||
|         format_properties.emplace(format, physical.getFormatProperties(format, dldi)); |         format_properties.emplace(format, physical.getFormatProperties(format, dldi)); | ||||||
|     }; |     }; | ||||||
|     AddFormatQuery(vk::Format::eA8B8G8R8UnormPack32); |     AddFormatQuery(vk::Format::eA8B8G8R8UnormPack32); | ||||||
|     AddFormatQuery(vk::Format::eR5G6B5UnormPack16); |     AddFormatQuery(vk::Format::eB5G6R5UnormPack16); | ||||||
|  |     AddFormatQuery(vk::Format::eA2B10G10R10UnormPack32); | ||||||
|  |     AddFormatQuery(vk::Format::eR8G8B8A8Srgb); | ||||||
|  |     AddFormatQuery(vk::Format::eR8Unorm); | ||||||
|     AddFormatQuery(vk::Format::eD32Sfloat); |     AddFormatQuery(vk::Format::eD32Sfloat); | ||||||
|  |     AddFormatQuery(vk::Format::eD16Unorm); | ||||||
|     AddFormatQuery(vk::Format::eD16UnormS8Uint); |     AddFormatQuery(vk::Format::eD16UnormS8Uint); | ||||||
|     AddFormatQuery(vk::Format::eD24UnormS8Uint); |     AddFormatQuery(vk::Format::eD24UnormS8Uint); | ||||||
|     AddFormatQuery(vk::Format::eD32SfloatS8Uint); |     AddFormatQuery(vk::Format::eD32SfloatS8Uint); | ||||||
|  |     AddFormatQuery(vk::Format::eBc1RgbaUnormBlock); | ||||||
|  |     AddFormatQuery(vk::Format::eBc2UnormBlock); | ||||||
|  |     AddFormatQuery(vk::Format::eBc3UnormBlock); | ||||||
|  |     AddFormatQuery(vk::Format::eBc4UnormBlock); | ||||||
| 
 | 
 | ||||||
|     return format_properties; |     return format_properties; | ||||||
| } | } | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 bunnei
						bunnei