2017-12-06 14:16:14 +01:00
|
|
|
#pragma once
|
|
|
|
|
2018-04-06 17:54:02 +02:00
|
|
|
#include <mutex>
|
|
|
|
#include <unordered_map>
|
|
|
|
|
|
|
|
#include "../dxbc/dxbc_module.h"
|
|
|
|
#include "../dxvk/dxvk_device.h"
|
2017-12-06 14:16:14 +01:00
|
|
|
|
2018-08-12 00:52:39 +02:00
|
|
|
#include "../d3d10/d3d10_shader.h"
|
|
|
|
|
2017-12-06 18:54:01 +01:00
|
|
|
#include "../util/sha1/sha1_util.h"
|
|
|
|
|
2017-12-08 01:32:02 +01:00
|
|
|
#include "../util/util_env.h"
|
|
|
|
|
2017-12-06 14:16:14 +01:00
|
|
|
#include "d3d11_device_child.h"
|
|
|
|
#include "d3d11_interfaces.h"
|
|
|
|
|
|
|
|
namespace dxvk {
|
|
|
|
|
|
|
|
class D3D11Device;
|
|
|
|
|
2017-12-06 18:54:01 +01:00
|
|
|
/**
|
2018-04-06 17:54:02 +02:00
|
|
|
* \brief Shader key
|
2017-12-06 18:54:01 +01:00
|
|
|
*
|
2018-04-06 17:54:02 +02:00
|
|
|
* A unique identifier for a shader consisting
|
|
|
|
* of the program type and the SHA-1 hash of
|
|
|
|
* the shader's original bytecode.
|
|
|
|
*/
|
|
|
|
class D3D11ShaderKey {
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
D3D11ShaderKey(
|
|
|
|
DxbcProgramType ProgramType,
|
|
|
|
const void* pShaderBytecode,
|
|
|
|
size_t BytecodeLength);
|
|
|
|
|
|
|
|
std::string GetName() const;
|
|
|
|
|
|
|
|
size_t GetHash() const;
|
|
|
|
|
|
|
|
bool operator == (const D3D11ShaderKey& other) const {
|
|
|
|
return m_type == other.m_type
|
|
|
|
&& m_hash == other.m_hash;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
DxbcProgramType m_type;
|
|
|
|
Sha1Hash m_hash;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
struct D3D11ShaderKeyHash {
|
|
|
|
size_t operator () (const D3D11ShaderKey& a) const {
|
|
|
|
return a.GetHash();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
2018-07-30 19:34:48 +02:00
|
|
|
* \brief Common shader object
|
2017-12-06 18:54:01 +01:00
|
|
|
*
|
2018-04-06 17:54:02 +02:00
|
|
|
* Stores the compiled SPIR-V shader and the SHA-1
|
|
|
|
* hash of the original DXBC shader, which can be
|
|
|
|
* used to identify the shader.
|
2017-12-06 18:54:01 +01:00
|
|
|
*/
|
2018-07-30 19:34:48 +02:00
|
|
|
class D3D11CommonShader {
|
2017-12-06 18:54:01 +01:00
|
|
|
|
|
|
|
public:
|
|
|
|
|
2018-07-30 19:34:48 +02:00
|
|
|
D3D11CommonShader();
|
|
|
|
D3D11CommonShader(
|
2018-07-30 20:15:19 +02:00
|
|
|
D3D11Device* pDevice,
|
2018-04-06 17:54:02 +02:00
|
|
|
const D3D11ShaderKey* pShaderKey,
|
2018-06-23 17:14:35 +02:00
|
|
|
const DxbcModuleInfo* pDxbcModuleInfo,
|
2018-04-06 17:54:02 +02:00
|
|
|
const void* pShaderBytecode,
|
|
|
|
size_t BytecodeLength);
|
2018-07-30 19:34:48 +02:00
|
|
|
~D3D11CommonShader();
|
|
|
|
|
2017-12-07 10:12:48 +01:00
|
|
|
Rc<DxvkShader> GetShader() const {
|
|
|
|
return m_shader;
|
|
|
|
}
|
2018-07-30 20:15:19 +02:00
|
|
|
|
|
|
|
Rc<DxvkBuffer> GetIcb() const {
|
|
|
|
return m_buffer;
|
|
|
|
}
|
2017-12-06 18:54:01 +01:00
|
|
|
|
2018-04-06 17:54:02 +02:00
|
|
|
std::string GetName() const {
|
2018-01-12 00:43:19 +01:00
|
|
|
return m_name;
|
|
|
|
}
|
|
|
|
|
2017-12-07 10:12:48 +01:00
|
|
|
private:
|
2017-12-06 18:54:01 +01:00
|
|
|
|
2018-01-12 00:06:54 +01:00
|
|
|
std::string m_name;
|
2017-12-07 10:12:48 +01:00
|
|
|
Rc<DxvkShader> m_shader;
|
2018-07-30 20:15:19 +02:00
|
|
|
Rc<DxvkBuffer> m_buffer;
|
2017-12-06 18:54:01 +01:00
|
|
|
|
|
|
|
};
|
|
|
|
|
2017-12-06 14:16:14 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Common shader interface
|
|
|
|
*
|
|
|
|
* Implements methods for all D3D11*Shader
|
|
|
|
* interfaces and stores the actual shader
|
|
|
|
* module object.
|
|
|
|
*/
|
2018-08-12 00:52:39 +02:00
|
|
|
template<typename D3D11Interface, typename D3D10Interface>
|
|
|
|
class D3D11Shader : public D3D11DeviceChild<D3D11Interface> {
|
|
|
|
using D3D10ShaderClass = D3D10Shader<D3D10Interface, D3D11Interface>;
|
2017-12-06 14:16:14 +01:00
|
|
|
public:
|
|
|
|
|
2018-07-30 19:34:48 +02:00
|
|
|
D3D11Shader(D3D11Device* device, const D3D11CommonShader& shader)
|
2018-08-12 00:52:39 +02:00
|
|
|
: m_device(device), m_shader(shader), m_d3d10(this) { }
|
2017-12-06 14:16:14 +01:00
|
|
|
|
|
|
|
~D3D11Shader() { }
|
|
|
|
|
2017-12-12 12:50:52 +01:00
|
|
|
HRESULT STDMETHODCALLTYPE QueryInterface(REFIID riid, void** ppvObject) final {
|
2018-04-02 12:52:02 +02:00
|
|
|
*ppvObject = nullptr;
|
|
|
|
|
|
|
|
if (riid == __uuidof(IUnknown)
|
|
|
|
|| riid == __uuidof(ID3D11DeviceChild)
|
2018-08-12 00:52:39 +02:00
|
|
|
|| riid == __uuidof(D3D11Interface)) {
|
2018-04-02 12:52:02 +02:00
|
|
|
*ppvObject = ref(this);
|
|
|
|
return S_OK;
|
|
|
|
}
|
2017-12-06 14:16:14 +01:00
|
|
|
|
2018-08-12 00:52:39 +02:00
|
|
|
if (riid == __uuidof(IUnknown)
|
|
|
|
|| riid == __uuidof(ID3D10DeviceChild)
|
|
|
|
|| riid == __uuidof(D3D10Interface)) {
|
|
|
|
*ppvObject = ref(&m_d3d10);
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
2017-12-06 14:16:14 +01:00
|
|
|
Logger::warn("D3D11Shader::QueryInterface: Unknown interface query");
|
|
|
|
return E_NOINTERFACE;
|
|
|
|
}
|
|
|
|
|
2017-12-12 12:50:52 +01:00
|
|
|
void STDMETHODCALLTYPE GetDevice(ID3D11Device **ppDevice) final {
|
2017-12-09 15:57:05 +01:00
|
|
|
*ppDevice = m_device.ref();
|
2017-12-06 14:16:14 +01:00
|
|
|
}
|
|
|
|
|
2018-07-30 19:34:48 +02:00
|
|
|
const D3D11CommonShader* GetCommonShader() const {
|
|
|
|
return &m_shader;
|
2018-01-12 00:43:19 +01:00
|
|
|
}
|
2018-08-12 00:52:39 +02:00
|
|
|
|
|
|
|
D3D10ShaderClass* GetD3D10Iface() {
|
|
|
|
return &m_d3d10;
|
|
|
|
}
|
|
|
|
|
2017-12-06 14:16:14 +01:00
|
|
|
private:
|
|
|
|
|
2017-12-09 15:57:05 +01:00
|
|
|
Com<D3D11Device> m_device;
|
2018-07-30 19:34:48 +02:00
|
|
|
D3D11CommonShader m_shader;
|
2018-08-12 00:52:39 +02:00
|
|
|
D3D10ShaderClass m_d3d10;
|
2017-12-06 14:16:14 +01:00
|
|
|
|
|
|
|
};
|
|
|
|
|
2018-08-12 00:52:39 +02:00
|
|
|
using D3D11VertexShader = D3D11Shader<ID3D11VertexShader, ID3D10VertexShader>;
|
|
|
|
using D3D11HullShader = D3D11Shader<ID3D11HullShader, ID3D10DeviceChild>;
|
|
|
|
using D3D11DomainShader = D3D11Shader<ID3D11DomainShader, ID3D10DeviceChild>;
|
|
|
|
using D3D11GeometryShader = D3D11Shader<ID3D11GeometryShader, ID3D10GeometryShader>;
|
|
|
|
using D3D11PixelShader = D3D11Shader<ID3D11PixelShader, ID3D10PixelShader>;
|
|
|
|
using D3D11ComputeShader = D3D11Shader<ID3D11ComputeShader, ID3D10DeviceChild>;
|
2017-12-06 14:16:14 +01:00
|
|
|
|
2018-04-06 17:54:02 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Shader module set
|
|
|
|
*
|
|
|
|
* Some applications may compile the same shader multiple
|
|
|
|
* times, so we should cache the resulting shader modules
|
|
|
|
* and reuse them rather than creating new ones. This
|
|
|
|
* class is thread-safe.
|
|
|
|
*/
|
|
|
|
class D3D11ShaderModuleSet {
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
D3D11ShaderModuleSet();
|
|
|
|
~D3D11ShaderModuleSet();
|
|
|
|
|
2018-07-30 19:34:48 +02:00
|
|
|
D3D11CommonShader GetShaderModule(
|
2018-07-30 20:15:19 +02:00
|
|
|
D3D11Device* pDevice,
|
2018-06-23 17:14:35 +02:00
|
|
|
const DxbcModuleInfo* pDxbcModuleInfo,
|
2018-04-06 17:54:02 +02:00
|
|
|
const void* pShaderBytecode,
|
|
|
|
size_t BytecodeLength,
|
|
|
|
DxbcProgramType ProgramType);
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
std::mutex m_mutex;
|
|
|
|
|
|
|
|
std::unordered_map<
|
|
|
|
D3D11ShaderKey,
|
2018-07-30 19:34:48 +02:00
|
|
|
D3D11CommonShader,
|
2018-04-06 17:54:02 +02:00
|
|
|
D3D11ShaderKeyHash> m_modules;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
2017-12-06 14:16:14 +01:00
|
|
|
}
|