2017-12-07 10:12:48 +01:00
|
|
|
#include "d3d11_device.h"
|
2017-12-06 14:16:14 +01:00
|
|
|
#include "d3d11_shader.h"
|
|
|
|
|
|
|
|
namespace dxvk {
|
|
|
|
|
2017-12-06 18:54:01 +01:00
|
|
|
static std::string GetEnvVar(LPCWSTR name) {
|
|
|
|
DWORD len = ::GetEnvironmentVariableW(name, nullptr, 0);
|
|
|
|
|
|
|
|
std::wstring result;
|
|
|
|
|
|
|
|
while (len > result.size()) {
|
|
|
|
result.resize(len);
|
|
|
|
len = ::GetEnvironmentVariableW(
|
|
|
|
name, result.data(), result.size());
|
|
|
|
}
|
|
|
|
|
|
|
|
result.resize(len);
|
|
|
|
return str::fromws(result);
|
|
|
|
}
|
2017-12-06 14:16:14 +01:00
|
|
|
|
|
|
|
|
2017-12-06 18:54:01 +01:00
|
|
|
D3D11ShaderModule:: D3D11ShaderModule() { }
|
|
|
|
D3D11ShaderModule::~D3D11ShaderModule() { }
|
|
|
|
|
|
|
|
|
|
|
|
D3D11ShaderModule::D3D11ShaderModule(
|
2017-12-07 10:12:48 +01:00
|
|
|
D3D11Device* pDevice,
|
|
|
|
const void* pShaderBytecode,
|
|
|
|
size_t BytecodeLength) {
|
2017-12-06 18:54:01 +01:00
|
|
|
|
|
|
|
DxbcReader reader(
|
|
|
|
reinterpret_cast<const char*>(pShaderBytecode),
|
|
|
|
BytecodeLength);
|
|
|
|
|
|
|
|
DxbcModule module(reader);
|
2017-12-07 10:12:48 +01:00
|
|
|
|
|
|
|
SpirvCodeBuffer spirvCode = module.compile();
|
2017-12-06 18:54:01 +01:00
|
|
|
|
|
|
|
// TODO pre-process shader bindings
|
2017-12-07 10:12:48 +01:00
|
|
|
std::vector<DxvkResourceSlot> resourceSlots;
|
2017-12-06 18:54:01 +01:00
|
|
|
|
|
|
|
// If requested by the user, dump both the raw DXBC
|
|
|
|
// shader and the compiled SPIR-V module to a file.
|
|
|
|
const std::string dumpPath = GetEnvVar(L"DXVK_SHADER_DUMP_PATH");
|
2017-12-07 18:51:41 +01:00
|
|
|
const std::string readPath = GetEnvVar(L"DXVK_SHADER_READ_PATH");
|
2017-12-06 18:54:01 +01:00
|
|
|
|
|
|
|
if (dumpPath.size() != 0) {
|
|
|
|
const std::string baseName = str::format(dumpPath, "/",
|
|
|
|
ConstructFileName(ComputeShaderHash(pShaderBytecode, BytecodeLength),
|
|
|
|
module.version().type()));
|
|
|
|
|
|
|
|
reader.store(std::ofstream(str::format(baseName, ".dxbc"),
|
|
|
|
std::ios_base::binary | std::ios_base::trunc));
|
|
|
|
|
2017-12-07 10:12:48 +01:00
|
|
|
spirvCode.store(std::ofstream(str::format(baseName, ".spv"),
|
2017-12-06 18:54:01 +01:00
|
|
|
std::ios_base::binary | std::ios_base::trunc));
|
|
|
|
}
|
2017-12-07 10:12:48 +01:00
|
|
|
|
2017-12-07 18:51:41 +01:00
|
|
|
if (readPath.size() != 0) {
|
|
|
|
const std::string baseName = str::format(readPath, "/",
|
|
|
|
ConstructFileName(ComputeShaderHash(pShaderBytecode, BytecodeLength),
|
|
|
|
module.version().type()));
|
|
|
|
|
|
|
|
spirvCode = SpirvCodeBuffer(std::ifstream(
|
|
|
|
str::format(baseName, ".spv"),
|
|
|
|
std::ios_base::binary));
|
|
|
|
}
|
|
|
|
|
2017-12-07 10:12:48 +01:00
|
|
|
m_shader = pDevice->GetDXVKDevice()->createShader(
|
|
|
|
module.version().shaderStage(),
|
|
|
|
resourceSlots.size(),
|
|
|
|
resourceSlots.data(),
|
|
|
|
spirvCode);
|
2017-12-06 18:54:01 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Sha1Hash D3D11ShaderModule::ComputeShaderHash(
|
|
|
|
const void* pShaderBytecode,
|
|
|
|
size_t BytecodeLength) const {
|
|
|
|
return Sha1Hash::compute(
|
|
|
|
reinterpret_cast<const uint8_t*>(pShaderBytecode),
|
|
|
|
BytecodeLength);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
std::string D3D11ShaderModule::ConstructFileName(
|
|
|
|
const Sha1Hash& hash,
|
|
|
|
const DxbcProgramType& type) const {
|
|
|
|
|
|
|
|
std::string typeStr;
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
case DxbcProgramType::PixelShader: typeStr = "PS_"; break;
|
|
|
|
case DxbcProgramType::VertexShader: typeStr = "VS_"; break;
|
|
|
|
case DxbcProgramType::GeometryShader: typeStr = "GS_"; break;
|
|
|
|
case DxbcProgramType::HullShader: typeStr = "HS_"; break;
|
|
|
|
case DxbcProgramType::DomainShader: typeStr = "DS_"; break;
|
|
|
|
case DxbcProgramType::ComputeShader: typeStr = "CS_"; break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return str::format(typeStr, hash.toString());
|
|
|
|
}
|
|
|
|
|
2017-12-06 14:16:14 +01:00
|
|
|
}
|