c2e349eb56
- Add Sampler base class - Add ColorAttachment, DepthAttachment classes - Add function name to all traces - Add Framebuffer resizing
156 lines
5.9 KiB
C++
156 lines
5.9 KiB
C++
#pragma once
|
|
|
|
#include <shaderc/shaderc.hpp>
|
|
|
|
#include "FileSystem.h"
|
|
#include "Common.h"
|
|
#include "Instance.h"
|
|
|
|
namespace Copium
|
|
{
|
|
class Shader
|
|
{
|
|
CP_DELETE_COPY_AND_MOVE_CTOR(Shader);
|
|
public:
|
|
enum class Type
|
|
{
|
|
GlslFile, GlslCode, SpvFile, SpvCode
|
|
};
|
|
|
|
private:
|
|
Instance& instance;
|
|
|
|
VkShaderModule vertShaderModule;
|
|
VkShaderModule fragShaderModule;
|
|
std::vector<VkPipelineShaderStageCreateInfo> shaderStages;
|
|
public:
|
|
Shader(Instance& instance, Type type, const std::string& vertexInput, const std::string& fragmentInput)
|
|
: instance{instance}
|
|
{
|
|
switch (type)
|
|
{
|
|
case Type::GlslCode:
|
|
vertShaderModule = InitializeShaderModuleFromGlslCode(vertexInput, shaderc_vertex_shader);
|
|
fragShaderModule = InitializeShaderModuleFromGlslCode(fragmentInput, shaderc_fragment_shader);
|
|
break;
|
|
case Type::GlslFile:
|
|
vertShaderModule = InitializeShaderModuleFromGlslFile(vertexInput, shaderc_vertex_shader);
|
|
fragShaderModule = InitializeShaderModuleFromGlslFile(fragmentInput, shaderc_fragment_shader);
|
|
break;
|
|
case Type::SpvCode:
|
|
vertShaderModule = InitializeShaderModule(vertexInput);
|
|
fragShaderModule = InitializeShaderModule(fragmentInput);
|
|
break;
|
|
case Type::SpvFile:
|
|
vertShaderModule = InitializeShaderModule(FileSystem::ReadFile(vertexInput));
|
|
fragShaderModule = InitializeShaderModule(FileSystem::ReadFile(fragmentInput));
|
|
break;
|
|
default:
|
|
CP_ASSERT(false, "Shader : Unreachable switch case %d", (int)type);
|
|
}
|
|
|
|
shaderStages.resize(2);
|
|
shaderStages[0] = {};
|
|
shaderStages[0].sType = VK_STRUCTURE_TYPE_PIPELINE_SHADER_STAGE_CREATE_INFO;
|
|
shaderStages[0].stage = VK_SHADER_STAGE_VERTEX_BIT;
|
|
shaderStages[0].module = vertShaderModule;
|
|
shaderStages[0].pName = "main";
|
|
|
|
shaderStages[1] = {};
|
|
shaderStages[1].sType = VK_STRUCTURE_TYPE_PIPELINE_SHADER_STAGE_CREATE_INFO;
|
|
shaderStages[1].stage = VK_SHADER_STAGE_FRAGMENT_BIT;
|
|
shaderStages[1].module = fragShaderModule;
|
|
shaderStages[1].pName = "main";
|
|
}
|
|
|
|
~Shader()
|
|
{
|
|
vkDestroyShaderModule(instance.GetDevice(), vertShaderModule, nullptr);
|
|
vkDestroyShaderModule(instance.GetDevice(), fragShaderModule, nullptr);
|
|
}
|
|
|
|
const std::vector<VkPipelineShaderStageCreateInfo> GetShaderStages() const
|
|
{
|
|
return shaderStages;
|
|
}
|
|
|
|
private:
|
|
VkShaderModule InitializeShaderModule(const std::vector<uint32_t>& codeSpv)
|
|
{
|
|
return InitializeShaderModule(codeSpv.data(), codeSpv.size() * sizeof(uint32_t));
|
|
}
|
|
|
|
VkShaderModule InitializeShaderModule(const std::string& codeSpv)
|
|
{
|
|
return InitializeShaderModule(reinterpret_cast<const uint32_t*>(codeSpv.data()), codeSpv.size());
|
|
}
|
|
|
|
VkShaderModule InitializeShaderModule(const std::vector<char>& codeSpv)
|
|
{
|
|
return InitializeShaderModule(reinterpret_cast<const uint32_t*>(codeSpv.data()), codeSpv.size());
|
|
}
|
|
|
|
VkShaderModule InitializeShaderModuleFromGlslFile(const std::string& filename, shaderc_shader_kind type)
|
|
{
|
|
std::string spvFilename = ".cache/" + filename + ".spv";
|
|
try
|
|
{
|
|
if (FileSystem::FileExists(spvFilename))
|
|
{
|
|
if (FileSystem::DateModified(filename) < FileSystem::DateModified(spvFilename))
|
|
{
|
|
CP_DEBUG("InitializeShaderModuleFromGlslFile : Loading cached shader file: %s", filename.c_str());
|
|
std::vector<char> data = FileSystem::ReadFile(spvFilename);
|
|
CP_ASSERT(data.size() % 4 == 0, "Spv data size is not a factor of 4");
|
|
return InitializeShaderModule((const uint32_t*)data.data(), data.size());
|
|
}
|
|
}
|
|
}
|
|
catch (const std::runtime_error& e)
|
|
{
|
|
CP_WARN("InitializeShaderModuleFromGlslFile : Cached shader file is invalid, recreating it");
|
|
}
|
|
CP_DEBUG("InitializeShaderModuleFromGlslFile : Compiling shader file: %s", filename.c_str());
|
|
shaderc::Compiler compiler;
|
|
shaderc::CompileOptions options;
|
|
|
|
options.SetOptimizationLevel(shaderc_optimization_level_size);
|
|
|
|
std::vector<char> glslCode = FileSystem::ReadFile(filename);
|
|
shaderc::SpvCompilationResult result = compiler.CompileGlslToSpv(glslCode.data(), glslCode.size(), type, filename.c_str(), options);
|
|
CP_ASSERT(result.GetCompilationStatus() == shaderc_compilation_status_success, "InitializeShaderModuleFromGlslFile : Failed to compile shader: %s\n%s", filename.c_str(), result.GetErrorMessage().c_str());
|
|
|
|
std::vector<uint32_t> data{result.cbegin(), result.cend()};
|
|
FileSystem::WriteFile(spvFilename, (const char*)data.data(), data.size() * sizeof(uint32_t));
|
|
return InitializeShaderModule(data.data(), data.size() * sizeof(uint32_t));
|
|
}
|
|
|
|
VkShaderModule InitializeShaderModuleFromGlslCode(const std::string& code, shaderc_shader_kind type)
|
|
{
|
|
shaderc::Compiler compiler;
|
|
shaderc::CompileOptions options;
|
|
|
|
options.SetOptimizationLevel(shaderc_optimization_level_size);
|
|
|
|
shaderc::SpvCompilationResult result = compiler.CompileGlslToSpv(code.data(), type, "inline_shader_code", options);
|
|
CP_ASSERT(result.GetCompilationStatus() == shaderc_compilation_status_success, "InitializeShaderModuleFromGlslCode : Failed to compile inline shader code: %s", result.GetErrorMessage());
|
|
|
|
std::vector<uint32_t> data{result.cbegin(), result.cend()};
|
|
return InitializeShaderModule(data.data(), data.size() * sizeof(uint32_t));
|
|
}
|
|
|
|
VkShaderModule InitializeShaderModule(const uint32_t* data, size_t size)
|
|
{
|
|
VkShaderModuleCreateInfo createInfo{};
|
|
createInfo.sType = VK_STRUCTURE_TYPE_SHADER_MODULE_CREATE_INFO;
|
|
createInfo.codeSize = size;
|
|
createInfo.pCode = data;
|
|
|
|
VkShaderModule shaderModule;
|
|
CP_VK_ASSERT(vkCreateShaderModule(instance.GetDevice(), &createInfo, nullptr, &shaderModule), "InitializeShaderModule : Failed to initialize shader module");
|
|
|
|
return shaderModule;
|
|
}
|
|
};
|
|
}
|