2024-04-05 05:28:33 +00:00
|
|
|
// SPDX-FileCopyrightText: 2019-2024 Connor McLaughlin <stenzek@gmail.com>
|
2023-08-13 03:42:02 +00:00
|
|
|
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
|
|
|
|
|
|
|
|
#include "vulkan_pipeline.h"
|
|
|
|
#include "vulkan_builders.h"
|
|
|
|
#include "vulkan_device.h"
|
|
|
|
|
|
|
|
#include "common/assert.h"
|
|
|
|
#include "common/log.h"
|
|
|
|
|
2024-04-05 05:28:33 +00:00
|
|
|
#include "shaderc/shaderc.hpp"
|
|
|
|
|
2023-08-13 03:42:02 +00:00
|
|
|
Log_SetChannel(VulkanDevice);
|
|
|
|
|
2024-04-05 05:28:33 +00:00
|
|
|
static std::unique_ptr<shaderc::Compiler> s_shaderc_compiler;
|
|
|
|
|
2023-08-13 03:42:02 +00:00
|
|
|
VulkanShader::VulkanShader(GPUShaderStage stage, VkShaderModule mod) : GPUShader(stage), m_module(mod)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
VulkanShader::~VulkanShader()
|
|
|
|
{
|
|
|
|
vkDestroyShaderModule(VulkanDevice::GetInstance().GetVulkanDevice(), m_module, nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void VulkanShader::SetDebugName(const std::string_view& name)
|
|
|
|
{
|
|
|
|
Vulkan::SetObjectName(VulkanDevice::GetInstance().GetVulkanDevice(), m_module, name);
|
|
|
|
}
|
|
|
|
|
2023-09-05 11:07:20 +00:00
|
|
|
std::unique_ptr<GPUShader> VulkanDevice::CreateShaderFromBinary(GPUShaderStage stage, std::span<const u8> data)
|
2023-08-13 03:42:02 +00:00
|
|
|
{
|
|
|
|
VkShaderModule mod;
|
|
|
|
|
|
|
|
const VkShaderModuleCreateInfo ci = {VK_STRUCTURE_TYPE_SHADER_MODULE_CREATE_INFO, nullptr, 0, data.size(),
|
|
|
|
reinterpret_cast<const u32*>(data.data())};
|
|
|
|
VkResult res = vkCreateShaderModule(m_device, &ci, nullptr, &mod);
|
|
|
|
if (res != VK_SUCCESS)
|
|
|
|
{
|
|
|
|
LOG_VULKAN_ERROR(res, "vkCreateShaderModule() failed: ");
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
return std::unique_ptr<GPUShader>(new VulkanShader(stage, mod));
|
|
|
|
}
|
|
|
|
|
|
|
|
std::unique_ptr<GPUShader> VulkanDevice::CreateShaderFromSource(GPUShaderStage stage, const std::string_view& source,
|
2023-09-02 12:09:20 +00:00
|
|
|
const char* entry_point,
|
|
|
|
DynamicHeapArray<u8>* out_binary)
|
2023-08-13 03:42:02 +00:00
|
|
|
{
|
2024-04-05 05:28:33 +00:00
|
|
|
static constexpr const std::array<shaderc_shader_kind, static_cast<size_t>(GPUShaderStage::MaxCount)> stage_kinds = {{
|
|
|
|
shaderc_glsl_vertex_shader,
|
|
|
|
shaderc_glsl_fragment_shader,
|
|
|
|
shaderc_glsl_geometry_shader,
|
|
|
|
shaderc_glsl_compute_shader,
|
|
|
|
}};
|
|
|
|
|
|
|
|
// TODO: NOT thread safe, yet.
|
|
|
|
if (!s_shaderc_compiler)
|
|
|
|
s_shaderc_compiler = std::make_unique<shaderc::Compiler>();
|
|
|
|
|
|
|
|
shaderc::CompileOptions options;
|
|
|
|
options.SetSourceLanguage(shaderc_source_language_glsl);
|
|
|
|
options.SetTargetEnvironment(shaderc_target_env_vulkan, 0);
|
|
|
|
|
|
|
|
if (m_debug_device)
|
2023-08-13 03:42:02 +00:00
|
|
|
{
|
2024-04-05 05:28:33 +00:00
|
|
|
options.SetOptimizationLevel(shaderc_optimization_level_zero);
|
|
|
|
options.SetGenerateDebugInfo();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
options.SetOptimizationLevel(shaderc_optimization_level_performance);
|
2023-08-13 03:42:02 +00:00
|
|
|
}
|
|
|
|
|
2024-04-05 05:28:33 +00:00
|
|
|
const shaderc::SpvCompilationResult result = s_shaderc_compiler->CompileGlslToSpv(
|
|
|
|
source.data(), source.length(), stage_kinds[static_cast<size_t>(stage)], "source", entry_point, options);
|
|
|
|
if (result.GetCompilationStatus() != shaderc_compilation_status_success)
|
2023-08-13 03:42:02 +00:00
|
|
|
{
|
2024-04-05 05:28:33 +00:00
|
|
|
const std::string errors = result.GetErrorMessage();
|
|
|
|
DumpBadShader(source, errors);
|
|
|
|
Log_ErrorFmt("Failed to compile shader to SPIR-V:\n{}", errors);
|
2023-08-13 03:42:02 +00:00
|
|
|
return {};
|
|
|
|
}
|
2024-04-05 05:28:33 +00:00
|
|
|
else if (result.GetNumWarnings() > 0)
|
|
|
|
{
|
|
|
|
Log_WarningFmt("Shader compiled with warnings:\n{}", result.GetErrorMessage());
|
|
|
|
}
|
2023-08-13 03:42:02 +00:00
|
|
|
|
2024-04-05 05:28:33 +00:00
|
|
|
const size_t spirv_size = std::distance(result.cbegin(), result.cend()) * sizeof(*result.cbegin());
|
|
|
|
DebugAssert(spirv_size > 0);
|
2023-08-13 03:42:02 +00:00
|
|
|
if (out_binary)
|
|
|
|
{
|
|
|
|
out_binary->resize(spirv_size);
|
2024-04-05 05:28:33 +00:00
|
|
|
std::copy(result.cbegin(), result.cend(), reinterpret_cast<uint32_t*>(out_binary->data()));
|
2023-08-13 03:42:02 +00:00
|
|
|
}
|
|
|
|
|
2024-04-05 05:28:33 +00:00
|
|
|
return CreateShaderFromBinary(stage, std::span<const u8>(reinterpret_cast<const u8*>(result.cbegin()), spirv_size));
|
2023-08-13 03:42:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2024-03-08 07:55:02 +00:00
|
|
|
VulkanPipeline::VulkanPipeline(VkPipeline pipeline, Layout layout, u8 vertices_per_primitive,
|
|
|
|
RenderPassFlag render_pass_flags)
|
|
|
|
: GPUPipeline(), m_pipeline(pipeline), m_layout(layout), m_vertices_per_primitive(vertices_per_primitive),
|
|
|
|
m_render_pass_flags(render_pass_flags)
|
2023-08-13 03:42:02 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
VulkanPipeline::~VulkanPipeline()
|
|
|
|
{
|
|
|
|
VulkanDevice::GetInstance().DeferPipelineDestruction(m_pipeline);
|
|
|
|
}
|
|
|
|
|
|
|
|
void VulkanPipeline::SetDebugName(const std::string_view& name)
|
|
|
|
{
|
|
|
|
Vulkan::SetObjectName(VulkanDevice::GetInstance().GetVulkanDevice(), m_pipeline, name);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::unique_ptr<GPUPipeline> VulkanDevice::CreatePipeline(const GPUPipeline::GraphicsConfig& config)
|
|
|
|
{
|
2024-03-08 07:55:02 +00:00
|
|
|
static constexpr std::array<std::pair<VkPrimitiveTopology, u32>, static_cast<u32>(GPUPipeline::Primitive::MaxCount)>
|
|
|
|
primitives = {{
|
|
|
|
{VK_PRIMITIVE_TOPOLOGY_POINT_LIST, 1}, // Points
|
|
|
|
{VK_PRIMITIVE_TOPOLOGY_LINE_LIST, 2}, // Lines
|
|
|
|
{VK_PRIMITIVE_TOPOLOGY_TRIANGLE_LIST, 3}, // Triangles
|
|
|
|
{VK_PRIMITIVE_TOPOLOGY_TRIANGLE_STRIP, 3}, // TriangleStrips
|
|
|
|
}};
|
2023-08-13 03:42:02 +00:00
|
|
|
|
|
|
|
static constexpr u32 MAX_COMPONENTS = 4;
|
|
|
|
static constexpr const VkFormat format_mapping[static_cast<u8>(
|
|
|
|
GPUPipeline::VertexAttribute::Type::MaxCount)][MAX_COMPONENTS] = {
|
|
|
|
{VK_FORMAT_R32_SFLOAT, VK_FORMAT_R32G32_SFLOAT, VK_FORMAT_R32G32B32_SFLOAT, VK_FORMAT_R32G32B32A32_SFLOAT}, // Float
|
|
|
|
{VK_FORMAT_R8_UINT, VK_FORMAT_R8G8_UINT, VK_FORMAT_R8G8B8_UINT, VK_FORMAT_R8G8B8A8_UINT}, // UInt8
|
|
|
|
{VK_FORMAT_R8_SINT, VK_FORMAT_R8G8_SINT, VK_FORMAT_R8G8B8_SINT, VK_FORMAT_R8G8B8A8_SINT}, // SInt8
|
|
|
|
{VK_FORMAT_R8_UNORM, VK_FORMAT_R8G8_UNORM, VK_FORMAT_R8G8B8_UNORM, VK_FORMAT_R8G8B8A8_UNORM}, // UNorm8
|
|
|
|
{VK_FORMAT_R16_UINT, VK_FORMAT_R16G16_UINT, VK_FORMAT_R16G16B16_UINT, VK_FORMAT_R16G16B16A16_UINT}, // UInt16
|
|
|
|
{VK_FORMAT_R16_SINT, VK_FORMAT_R16G16_SINT, VK_FORMAT_R16G16B16_SINT, VK_FORMAT_R16G16B16A16_SINT}, // SInt16
|
|
|
|
{VK_FORMAT_R16_UNORM, VK_FORMAT_R16G16_UNORM, VK_FORMAT_R16G16B16_UNORM, VK_FORMAT_R16G16B16A16_UNORM}, // UNorm16
|
|
|
|
{VK_FORMAT_R32_UINT, VK_FORMAT_R32G32_UINT, VK_FORMAT_R32G32B32_UINT, VK_FORMAT_R32G32B32A32_UINT}, // UInt32
|
|
|
|
{VK_FORMAT_R32_SINT, VK_FORMAT_R32G32_SINT, VK_FORMAT_R32G32B32_SINT, VK_FORMAT_R32G32B32A32_SINT}, // SInt32
|
|
|
|
};
|
|
|
|
|
|
|
|
static constexpr std::array<VkCullModeFlagBits, static_cast<u32>(GPUPipeline::CullMode::MaxCount)> cull_mapping = {{
|
|
|
|
VK_CULL_MODE_NONE, // None
|
|
|
|
VK_CULL_MODE_FRONT_BIT, // Front
|
|
|
|
VK_CULL_MODE_BACK_BIT, // Back
|
|
|
|
}};
|
|
|
|
|
|
|
|
static constexpr std::array<VkCompareOp, static_cast<u32>(GPUPipeline::DepthFunc::MaxCount)> compare_mapping = {{
|
|
|
|
VK_COMPARE_OP_NEVER, // Never
|
|
|
|
VK_COMPARE_OP_ALWAYS, // Always
|
|
|
|
VK_COMPARE_OP_LESS, // Less
|
|
|
|
VK_COMPARE_OP_LESS_OR_EQUAL, // LessEqual
|
|
|
|
VK_COMPARE_OP_GREATER, // Greater
|
|
|
|
VK_COMPARE_OP_GREATER_OR_EQUAL, // GreaterEqual
|
|
|
|
VK_COMPARE_OP_EQUAL, // Equal
|
|
|
|
}};
|
|
|
|
|
|
|
|
static constexpr std::array<VkBlendFactor, static_cast<u32>(GPUPipeline::BlendFunc::MaxCount)> blend_mapping = {{
|
|
|
|
VK_BLEND_FACTOR_ZERO, // Zero
|
|
|
|
VK_BLEND_FACTOR_ONE, // One
|
|
|
|
VK_BLEND_FACTOR_SRC_COLOR, // SrcColor
|
|
|
|
VK_BLEND_FACTOR_ONE_MINUS_SRC_COLOR, // InvSrcColor
|
|
|
|
VK_BLEND_FACTOR_DST_COLOR, // DstColor
|
|
|
|
VK_BLEND_FACTOR_ONE_MINUS_DST_COLOR, // InvDstColor
|
|
|
|
VK_BLEND_FACTOR_SRC_ALPHA, // SrcAlpha
|
|
|
|
VK_BLEND_FACTOR_ONE_MINUS_SRC_ALPHA, // InvSrcAlpha
|
|
|
|
VK_BLEND_FACTOR_SRC1_ALPHA, // SrcAlpha1
|
|
|
|
VK_BLEND_FACTOR_ONE_MINUS_SRC1_ALPHA, // InvSrcAlpha1
|
|
|
|
VK_BLEND_FACTOR_DST_ALPHA, // DstAlpha
|
|
|
|
VK_BLEND_FACTOR_ONE_MINUS_DST_ALPHA, // InvDstAlpha
|
2023-11-28 04:08:29 +00:00
|
|
|
VK_BLEND_FACTOR_CONSTANT_COLOR, // ConstantAlpha
|
|
|
|
VK_BLEND_FACTOR_ONE_MINUS_CONSTANT_COLOR, // InvConstantAlpha
|
2023-08-13 03:42:02 +00:00
|
|
|
}};
|
|
|
|
|
|
|
|
static constexpr std::array<VkBlendOp, static_cast<u32>(GPUPipeline::BlendOp::MaxCount)> op_mapping = {{
|
|
|
|
VK_BLEND_OP_ADD, // Add
|
|
|
|
VK_BLEND_OP_SUBTRACT, // Subtract
|
|
|
|
VK_BLEND_OP_REVERSE_SUBTRACT, // ReverseSubtract
|
|
|
|
VK_BLEND_OP_MIN, // Min
|
|
|
|
VK_BLEND_OP_MAX, // Max
|
|
|
|
}};
|
|
|
|
|
|
|
|
Vulkan::GraphicsPipelineBuilder gpb;
|
|
|
|
gpb.SetVertexShader(static_cast<const VulkanShader*>(config.vertex_shader)->GetModule());
|
|
|
|
gpb.SetFragmentShader(static_cast<const VulkanShader*>(config.fragment_shader)->GetModule());
|
|
|
|
|
2023-09-02 12:09:20 +00:00
|
|
|
if (config.geometry_shader)
|
|
|
|
gpb.SetGeometryShader(static_cast<const VulkanShader*>(config.geometry_shader)->GetModule());
|
|
|
|
|
2023-08-13 03:42:02 +00:00
|
|
|
if (!config.input_layout.vertex_attributes.empty())
|
|
|
|
{
|
|
|
|
gpb.AddVertexBuffer(0, config.input_layout.vertex_stride);
|
|
|
|
for (u32 i = 0; i < static_cast<u32>(config.input_layout.vertex_attributes.size()); i++)
|
|
|
|
{
|
|
|
|
const GPUPipeline::VertexAttribute& va = config.input_layout.vertex_attributes[i];
|
|
|
|
DebugAssert(va.components > 0 && va.components <= MAX_COMPONENTS);
|
|
|
|
gpb.AddVertexAttribute(
|
|
|
|
i, 0, format_mapping[static_cast<u8>(va.type.GetValue())][static_cast<u8>(va.components.GetValue() - 1)],
|
|
|
|
va.offset);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-08 07:55:02 +00:00
|
|
|
const auto [vk_topology, vertices_per_primitive] = primitives[static_cast<u8>(config.primitive)];
|
|
|
|
gpb.SetPrimitiveTopology(vk_topology);
|
2023-08-29 16:03:22 +00:00
|
|
|
|
2023-08-13 03:42:02 +00:00
|
|
|
// Line width?
|
|
|
|
|
|
|
|
gpb.SetRasterizationState(VK_POLYGON_MODE_FILL,
|
|
|
|
cull_mapping[static_cast<u8>(config.rasterization.cull_mode.GetValue())],
|
|
|
|
VK_FRONT_FACE_CLOCKWISE);
|
|
|
|
if (config.samples > 1)
|
|
|
|
gpb.SetMultisamples(config.samples, config.per_sample_shading);
|
|
|
|
gpb.SetDepthState(config.depth.depth_test != GPUPipeline::DepthFunc::Always || config.depth.depth_write,
|
|
|
|
config.depth.depth_write, compare_mapping[static_cast<u8>(config.depth.depth_test.GetValue())]);
|
|
|
|
gpb.SetNoStencilState();
|
|
|
|
|
2024-03-24 11:01:03 +00:00
|
|
|
for (u32 i = 0; i < MAX_RENDER_TARGETS; i++)
|
|
|
|
{
|
|
|
|
if (config.color_formats[i] == GPUTexture::Format::Unknown)
|
|
|
|
break;
|
|
|
|
|
|
|
|
gpb.SetBlendAttachment(i, config.blend.enable, blend_mapping[static_cast<u8>(config.blend.src_blend.GetValue())],
|
|
|
|
blend_mapping[static_cast<u8>(config.blend.dst_blend.GetValue())],
|
|
|
|
op_mapping[static_cast<u8>(config.blend.blend_op.GetValue())],
|
|
|
|
blend_mapping[static_cast<u8>(config.blend.src_alpha_blend.GetValue())],
|
|
|
|
blend_mapping[static_cast<u8>(config.blend.dst_alpha_blend.GetValue())],
|
|
|
|
op_mapping[static_cast<u8>(config.blend.alpha_blend_op.GetValue())],
|
|
|
|
config.blend.write_mask);
|
|
|
|
}
|
2023-08-13 03:42:02 +00:00
|
|
|
|
|
|
|
const auto blend_constants = config.blend.GetConstantFloatColor();
|
|
|
|
gpb.SetBlendConstants(blend_constants[0], blend_constants[1], blend_constants[2], blend_constants[3]);
|
|
|
|
|
|
|
|
gpb.AddDynamicState(VK_DYNAMIC_STATE_VIEWPORT);
|
|
|
|
gpb.AddDynamicState(VK_DYNAMIC_STATE_SCISSOR);
|
|
|
|
|
|
|
|
gpb.SetPipelineLayout(m_pipeline_layouts[static_cast<u8>(config.layout)]);
|
|
|
|
|
2024-03-08 07:55:02 +00:00
|
|
|
if (m_optional_extensions.vk_khr_dynamic_rendering && (m_optional_extensions.vk_khr_dynamic_rendering_local_read ||
|
|
|
|
!(config.render_pass_flags & GPUPipeline::ColorFeedbackLoop)))
|
2023-12-04 05:47:18 +00:00
|
|
|
{
|
|
|
|
gpb.SetDynamicRendering();
|
|
|
|
|
|
|
|
for (u32 i = 0; i < MAX_RENDER_TARGETS; i++)
|
|
|
|
{
|
|
|
|
if (config.color_formats[i] == GPUTexture::Format::Unknown)
|
|
|
|
break;
|
|
|
|
|
|
|
|
gpb.AddDynamicRenderingColorAttachment(
|
|
|
|
VulkanDevice::TEXTURE_FORMAT_MAPPING[static_cast<u8>(config.color_formats[i])]);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (config.depth_format != GPUTexture::Format::Unknown)
|
|
|
|
{
|
|
|
|
gpb.SetDynamicRenderingDepthAttachment(VulkanDevice::TEXTURE_FORMAT_MAPPING[static_cast<u8>(config.depth_format)],
|
|
|
|
VK_FORMAT_UNDEFINED);
|
|
|
|
}
|
2024-03-08 07:55:02 +00:00
|
|
|
|
|
|
|
if (config.render_pass_flags & GPUPipeline::ColorFeedbackLoop)
|
|
|
|
{
|
|
|
|
DebugAssert(m_optional_extensions.vk_khr_dynamic_rendering_local_read &&
|
|
|
|
config.color_formats[0] != GPUTexture::Format::Unknown);
|
|
|
|
gpb.AddDynamicRenderingInputAttachment(0);
|
|
|
|
}
|
2023-12-04 05:47:18 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
const VkRenderPass render_pass = GetRenderPass(config);
|
|
|
|
DebugAssert(render_pass != VK_NULL_HANDLE);
|
|
|
|
gpb.SetRenderPass(render_pass, 0);
|
|
|
|
}
|
2023-08-13 03:42:02 +00:00
|
|
|
|
|
|
|
const VkPipeline pipeline = gpb.Create(m_device, m_pipeline_cache, false);
|
|
|
|
if (!pipeline)
|
|
|
|
return {};
|
|
|
|
|
2024-03-08 07:55:02 +00:00
|
|
|
return std::unique_ptr<GPUPipeline>(
|
|
|
|
new VulkanPipeline(pipeline, config.layout, static_cast<u8>(vertices_per_primitive), config.render_pass_flags));
|
2023-08-13 03:42:02 +00:00
|
|
|
}
|