From 2d6a3396dd3e000e6d55f10f4faaa1ba06196caf Mon Sep 17 00:00:00 2001 From: Philip Rebohle Date: Wed, 23 Oct 2019 16:36:52 +0200 Subject: [PATCH] [dxbc] Declare length of shader input array late Otherwise, we generate out-of-bounds array reads and writes in broken ENB shaders. --- src/dxbc/dxbc_compiler.cpp | 19 ++++++++++++------- src/dxbc/dxbc_compiler.h | 3 +++ 2 files changed, 15 insertions(+), 7 deletions(-) diff --git a/src/dxbc/dxbc_compiler.cpp b/src/dxbc/dxbc_compiler.cpp index c3029a637..7db161ffa 100644 --- a/src/dxbc/dxbc_compiler.cpp +++ b/src/dxbc/dxbc_compiler.cpp @@ -674,9 +674,10 @@ namespace dxvk { m_module.enableCapability(spv::CapabilitySampleRateShading); m_module.decorate(varId, spv::DecorationSample); } - + // Declare the input slot as defined m_interfaceSlots.inputSlots |= 1u << regIdx; + m_vArrayLength = std::max(m_vArrayLength, regIdx + 1); } else if (sv != DxbcSystemValue::None) { // Add a new system value mapping if needed bool skipSv = sv == DxbcSystemValue::ClipDistance @@ -5682,6 +5683,8 @@ namespace dxvk { void DxbcCompiler::emitInputSetup() { + m_module.setLateConst(m_vArrayLengthId, &m_vArrayLength); + // Copy all defined v# registers into the input array const uint32_t vecTypeId = m_module.defVectorType(m_module.defFloatType(32), 4); const uint32_t ptrTypeId = m_module.defPointerType(vecTypeId, spv::StorageClassPrivate); @@ -5727,6 +5730,8 @@ namespace dxvk { void DxbcCompiler::emitInputSetup(uint32_t vertexCount) { + m_module.setLateConst(m_vArrayLengthId, &m_vArrayLength); + // Copy all defined v# registers into the input array. Note // that the outer index of the array is the vertex index. const uint32_t vecTypeId = m_module.defVectorType(m_module.defFloatType(32), 4); @@ -7027,18 +7032,18 @@ namespace dxvk { void DxbcCompiler::emitDclInputArray(uint32_t vertexCount) { - DxbcArrayType info; + DxbcVectorType info; info.ctype = DxbcScalarType::Float32; info.ccount = 4; - info.alength = m_isgn != nullptr ? m_isgn->maxRegisterCount() : 0; - if (info.alength == 0) - return; - // Define the array type. This will be two-dimensional // in some shaders, with the outer index representing // the vertex ID within an invocation. - uint32_t arrayTypeId = getArrayTypeId(info); + m_vArrayLength = m_isgn != nullptr ? m_isgn->maxRegisterCount() : 1; + m_vArrayLengthId = m_module.lateConst32(getScalarTypeId(DxbcScalarType::Uint32)); + + uint32_t vectorTypeId = getVectorTypeId(info); + uint32_t arrayTypeId = m_module.defArrayType(vectorTypeId, m_vArrayLengthId); if (vertexCount != 0) { arrayTypeId = m_module.defArrayType( diff --git a/src/dxbc/dxbc_compiler.h b/src/dxbc/dxbc_compiler.h index 07df1bffe..03b143c88 100644 --- a/src/dxbc/dxbc_compiler.h +++ b/src/dxbc/dxbc_compiler.h @@ -476,6 +476,9 @@ namespace dxvk { /////////////////////////////////////////////////////////// // Array of input values. Since v# registers are indexable // in DXBC, we need to copy them into an array first. + uint32_t m_vArrayLength = 0; + uint32_t m_vArrayLengthId = 0; + uint32_t m_vArray = 0; ////////////////////////////////////////////////////