2017-10-11 23:28:06 +02:00
|
|
|
#include "dxvk_shader.h"
|
|
|
|
|
|
|
|
namespace dxvk {
|
|
|
|
|
2017-12-07 09:38:31 +01:00
|
|
|
DxvkShaderModule::DxvkShaderModule(
|
2017-11-20 14:03:00 +01:00
|
|
|
const Rc<vk::DeviceFn>& vkd,
|
2017-10-14 23:52:47 +02:00
|
|
|
VkShaderStageFlagBits stage,
|
2018-02-07 16:44:30 +01:00
|
|
|
const SpirvCodeBuffer& code,
|
|
|
|
const std::string& name)
|
|
|
|
: m_vkd(vkd), m_stage(stage), m_debugName(name) {
|
2017-11-20 14:03:00 +01:00
|
|
|
VkShaderModuleCreateInfo info;
|
|
|
|
info.sType = VK_STRUCTURE_TYPE_SHADER_MODULE_CREATE_INFO;
|
|
|
|
info.pNext = nullptr;
|
|
|
|
info.flags = 0;
|
|
|
|
info.codeSize = code.size();
|
|
|
|
info.pCode = code.data();
|
|
|
|
|
|
|
|
if (m_vkd->vkCreateShaderModule(m_vkd->device(),
|
|
|
|
&info, nullptr, &m_module) != VK_SUCCESS)
|
|
|
|
throw DxvkError("DxvkComputePipeline::DxvkComputePipeline: Failed to create shader module");
|
2017-10-11 23:28:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-12-07 09:38:31 +01:00
|
|
|
DxvkShaderModule::~DxvkShaderModule() {
|
2017-11-20 14:03:00 +01:00
|
|
|
m_vkd->vkDestroyShaderModule(
|
|
|
|
m_vkd->device(), m_module, nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-01-10 11:44:40 +01:00
|
|
|
VkPipelineShaderStageCreateInfo DxvkShaderModule::stageInfo(const VkSpecializationInfo* specInfo) const {
|
2017-11-20 14:03:00 +01:00
|
|
|
VkPipelineShaderStageCreateInfo info;
|
|
|
|
info.sType = VK_STRUCTURE_TYPE_PIPELINE_SHADER_STAGE_CREATE_INFO;
|
|
|
|
info.pNext = nullptr;
|
|
|
|
info.flags = 0;
|
|
|
|
info.stage = m_stage;
|
|
|
|
info.module = m_module;
|
|
|
|
info.pName = "main";
|
2018-01-10 11:44:40 +01:00
|
|
|
info.pSpecializationInfo = specInfo;
|
2017-11-20 14:03:00 +01:00
|
|
|
return info;
|
2017-10-14 23:52:47 +02:00
|
|
|
}
|
|
|
|
|
2017-12-07 09:38:31 +01:00
|
|
|
|
|
|
|
DxvkShader::DxvkShader(
|
2017-12-08 18:14:05 +01:00
|
|
|
VkShaderStageFlagBits stage,
|
|
|
|
uint32_t slotCount,
|
|
|
|
const DxvkResourceSlot* slotInfos,
|
2018-01-12 14:25:26 +01:00
|
|
|
const DxvkInterfaceSlots& iface,
|
2017-12-08 18:14:05 +01:00
|
|
|
const SpirvCodeBuffer& code)
|
2018-01-12 14:25:26 +01:00
|
|
|
: m_stage(stage), m_code(code), m_interface(iface) {
|
2017-12-07 09:38:31 +01:00
|
|
|
for (uint32_t i = 0; i < slotCount; i++)
|
|
|
|
m_slots.push_back(slotInfos[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
DxvkShader::~DxvkShader() {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void DxvkShader::defineResourceSlots(
|
|
|
|
DxvkDescriptorSlotMapping& mapping) const {
|
|
|
|
for (const auto& slot : m_slots)
|
2018-01-09 20:35:29 +01:00
|
|
|
mapping.defineSlot(slot.slot, slot.type, slot.view, m_stage);
|
2017-12-07 09:38:31 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Rc<DxvkShaderModule> DxvkShader::createShaderModule(
|
2017-12-08 18:14:05 +01:00
|
|
|
const Rc<vk::DeviceFn>& vkd,
|
2017-12-07 09:38:31 +01:00
|
|
|
const DxvkDescriptorSlotMapping& mapping) const {
|
2017-12-08 22:30:41 +01:00
|
|
|
// Iterate over the code and replace every resource slot
|
|
|
|
// index with the corresponding mapped binding index.
|
|
|
|
SpirvCodeBuffer spirvCode = m_code;
|
|
|
|
|
|
|
|
for (auto ins : spirvCode) {
|
|
|
|
if (ins.opCode() == spv::OpDecorate
|
2018-01-10 13:44:04 +01:00
|
|
|
&& ((ins.arg(2) == spv::DecorationBinding)
|
|
|
|
|| (ins.arg(2) == spv::DecorationSpecId))) {
|
2017-12-08 22:30:41 +01:00
|
|
|
|
|
|
|
const uint32_t oldBinding = ins.arg(3);
|
|
|
|
const uint32_t newBinding = mapping.getBindingId(oldBinding);
|
|
|
|
ins.setArg(3, newBinding);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-07 16:44:30 +01:00
|
|
|
return new DxvkShaderModule(vkd, m_stage, spirvCode, m_debugName);
|
2017-12-08 18:14:05 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void DxvkShader::dump(std::ostream&& outputStream) const {
|
|
|
|
m_code.store(std::move(outputStream));
|
2017-12-07 09:38:31 +01:00
|
|
|
}
|
|
|
|
|
2017-12-10 12:21:33 +01:00
|
|
|
|
|
|
|
void DxvkShader::read(std::istream&& inputStream) {
|
|
|
|
m_code = SpirvCodeBuffer(std::move(inputStream));
|
|
|
|
}
|
|
|
|
|
2017-10-11 23:28:06 +02:00
|
|
|
}
|