2019-12-16 03:28:01 +00:00
|
|
|
#pragma once
|
|
|
|
|
2022-07-30 23:58:40 +01:00
|
|
|
#include <array>
|
|
|
|
|
2019-12-16 03:28:01 +00:00
|
|
|
#include <cstdint>
|
|
|
|
|
2022-07-30 23:58:40 +01:00
|
|
|
#include "../spirv/spirv_module.h"
|
|
|
|
|
|
|
|
class D3D9DeviceEx;
|
|
|
|
|
2019-12-16 03:28:01 +00:00
|
|
|
namespace dxvk {
|
|
|
|
|
|
|
|
enum D3D9SpecConstantId : uint32_t {
|
2022-07-30 23:58:40 +01:00
|
|
|
SpecSamplerType, // 2 bits for 16 PS samplers | Bits: 32
|
|
|
|
|
|
|
|
SpecSamplerDepthMode, // 1 bit for 20 VS + PS samplers | Bits: 20
|
|
|
|
SpecAlphaCompareOp, // Range: 0 -> 7 | Bits: 3
|
|
|
|
SpecPointMode, // Range: 0 -> 3 | Bits: 2
|
|
|
|
SpecVertexFogMode, // Range: 0 -> 3 | Bits: 2
|
|
|
|
SpecPixelFogMode, // Range: 0 -> 3 | Bits: 2
|
|
|
|
SpecFogEnabled, // Range: 0 -> 1 | Bits: 1
|
|
|
|
|
|
|
|
SpecSamplerNull, // 1 bit for 20 samplers | Bits: 20
|
|
|
|
SpecProjectionType, // 1 bit for 6 PS 1.x samplers | Bits: 6
|
|
|
|
|
|
|
|
SpecVertexShaderBools, // 16 bools | Bits: 16
|
|
|
|
SpecPixelShaderBools, // 16 bools | Bits: 16
|
|
|
|
|
|
|
|
SpecFetch4, // 1 bit for 16 PS samplers | Bits: 16
|
|
|
|
|
|
|
|
SpecConstantCount,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct BitfieldPosition {
|
|
|
|
constexpr uint32_t mask() const {
|
|
|
|
return uint32_t((1ull << sizeInBits) - 1) << bitOffset;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t dwordOffset;
|
|
|
|
uint32_t bitOffset;
|
|
|
|
uint32_t sizeInBits;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct D3D9SpecializationInfo {
|
|
|
|
static constexpr uint32_t MaxSpecDwords = 5;
|
|
|
|
|
|
|
|
static constexpr std::array<BitfieldPosition, SpecConstantCount> Layout{{
|
|
|
|
{ 0, 0, 32 }, // SamplerType
|
|
|
|
|
|
|
|
{ 1, 0, 20 }, // SamplerDepthMode
|
|
|
|
{ 1, 20, 3 }, // AlphaCompareOp
|
|
|
|
{ 1, 23, 2 }, // PointMode
|
|
|
|
{ 1, 25, 2 }, // VertexFogMode
|
|
|
|
{ 1, 27, 2 }, // PixelFogMode
|
|
|
|
{ 1, 29, 1 }, // FogEnabled
|
|
|
|
|
|
|
|
{ 2, 0, 20 }, // SamplerNull
|
|
|
|
{ 2, 20, 6 }, // ProjectionType
|
|
|
|
|
|
|
|
{ 3, 0, 16 }, // VertexShaderBools
|
|
|
|
{ 3, 16, 16 }, // PixelShaderBools
|
|
|
|
|
|
|
|
{ 4, 0, 16 }, // Fetch4
|
|
|
|
}};
|
|
|
|
|
|
|
|
template <D3D9SpecConstantId Id, typename T>
|
|
|
|
bool set(const T& value) {
|
|
|
|
const uint32_t x = uint32_t(value);
|
|
|
|
if (get<Id>() == x)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
constexpr auto& layout = Layout[Id];
|
|
|
|
|
|
|
|
data[layout.dwordOffset] &= ~layout.mask();
|
|
|
|
data[layout.dwordOffset] |= (x << layout.bitOffset) & layout.mask();
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <D3D9SpecConstantId Id>
|
|
|
|
uint32_t get() const {
|
|
|
|
constexpr auto& layout = Layout[Id];
|
|
|
|
|
|
|
|
return (data[layout.dwordOffset] & layout.mask()) >> layout.bitOffset;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::array<uint32_t, MaxSpecDwords> data = {};
|
|
|
|
};
|
|
|
|
|
|
|
|
class D3D9ShaderSpecConstantManager {
|
|
|
|
public:
|
|
|
|
uint32_t get(SpirvModule &module, D3D9SpecConstantId id) {
|
|
|
|
const auto &layout = D3D9SpecializationInfo::Layout[id];
|
|
|
|
|
|
|
|
uint32_t val = getSpecConstDword(module, layout.dwordOffset);
|
|
|
|
if (layout.sizeInBits == 32)
|
|
|
|
return val;
|
|
|
|
|
|
|
|
return module.opBitFieldUExtract(
|
|
|
|
module.defIntType(32, 0),
|
|
|
|
val,
|
|
|
|
module.consti32(layout.bitOffset),
|
|
|
|
module.consti32(layout.sizeInBits));
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
uint32_t getSpecConstDword(SpirvModule &module, uint32_t idx) {
|
|
|
|
if (!m_specConstantIds[idx]) {
|
|
|
|
m_specConstantIds[idx] = module.specConst32(module.defIntType(32, 0), 0);
|
|
|
|
module.decorateSpecId(m_specConstantIds[idx], idx);
|
|
|
|
}
|
|
|
|
|
|
|
|
return m_specConstantIds[idx];
|
|
|
|
}
|
|
|
|
|
|
|
|
std::array<uint32_t, D3D9SpecializationInfo::MaxSpecDwords> m_specConstantIds = {};
|
2019-12-16 03:28:01 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|