2017-12-09 20:49:56 +01:00
|
|
|
#include "d3d11_device.h"
|
|
|
|
#include "d3d11_sampler.h"
|
2018-03-05 02:21:34 +01:00
|
|
|
#include "d3d11_util.h"
|
2017-12-09 20:49:56 +01:00
|
|
|
|
|
|
|
namespace dxvk {
|
|
|
|
|
|
|
|
D3D11SamplerState::D3D11SamplerState(
|
|
|
|
D3D11Device* device,
|
2018-03-05 02:21:34 +01:00
|
|
|
const D3D11_SAMPLER_DESC& desc)
|
2018-08-11 20:42:28 +02:00
|
|
|
: m_device(device), m_desc(desc), m_d3d10(this) {
|
2018-03-05 02:21:34 +01:00
|
|
|
DxvkSamplerCreateInfo info;
|
2017-12-09 20:49:56 +01:00
|
|
|
|
2018-03-05 02:21:34 +01:00
|
|
|
// While D3D11_FILTER is technically an enum, its value bits
|
|
|
|
// can be used to decode the filter properties more efficiently.
|
2018-05-09 11:55:05 +02:00
|
|
|
const uint32_t filterBits = uint32_t(desc.Filter);
|
2018-03-05 02:21:34 +01:00
|
|
|
info.magFilter = (filterBits & 0x04) ? VK_FILTER_LINEAR : VK_FILTER_NEAREST;
|
|
|
|
info.minFilter = (filterBits & 0x10) ? VK_FILTER_LINEAR : VK_FILTER_NEAREST;
|
2018-05-09 11:55:05 +02:00
|
|
|
|
|
|
|
// Set up the remaining properties, which are
|
|
|
|
// stored directly in the sampler description
|
2018-03-05 02:21:34 +01:00
|
|
|
info.mipmapMode = (filterBits & 0x01) ? VK_SAMPLER_MIPMAP_MODE_LINEAR : VK_SAMPLER_MIPMAP_MODE_NEAREST;
|
2018-05-09 11:55:05 +02:00
|
|
|
info.mipmapLodBias = desc.MipLODBias;
|
|
|
|
info.mipmapLodMin = desc.MinLOD;
|
|
|
|
info.mipmapLodMax = desc.MaxLOD;
|
|
|
|
|
2018-03-05 02:21:34 +01:00
|
|
|
info.useAnisotropy = (filterBits & 0x40) ? VK_TRUE : VK_FALSE;
|
2018-05-09 11:55:05 +02:00
|
|
|
info.maxAnisotropy = float(desc.MaxAnisotropy);
|
|
|
|
|
|
|
|
info.addressModeU = DecodeAddressMode(desc.AddressU);
|
|
|
|
info.addressModeV = DecodeAddressMode(desc.AddressV);
|
|
|
|
info.addressModeW = DecodeAddressMode(desc.AddressW);
|
|
|
|
|
2018-03-05 02:21:34 +01:00
|
|
|
info.compareToDepth = (filterBits & 0x80) ? VK_TRUE : VK_FALSE;
|
2018-05-09 11:55:05 +02:00
|
|
|
info.compareOp = DecodeCompareOp(desc.ComparisonFunc);
|
2018-03-05 02:21:34 +01:00
|
|
|
|
2018-10-31 21:51:23 +01:00
|
|
|
for (uint32_t i = 0; i < 4; i++)
|
|
|
|
info.borderColor.float32[i] = desc.BorderColor[i];
|
|
|
|
|
2018-05-09 11:55:05 +02:00
|
|
|
info.usePixelCoord = VK_FALSE; // Not supported in D3D11
|
2018-03-05 02:21:34 +01:00
|
|
|
|
2018-03-09 15:24:28 +01:00
|
|
|
// Make sure to use a valid anisotropy value
|
|
|
|
if (desc.MaxAnisotropy < 1) info.maxAnisotropy = 1.0f;
|
|
|
|
if (desc.MaxAnisotropy > 16) info.maxAnisotropy = 16.0f;
|
|
|
|
|
2018-09-10 15:42:55 +02:00
|
|
|
// Enforce anisotropy specified in the device options
|
|
|
|
int32_t samplerAnisotropyOption = device->GetOptions()->samplerAnisotropy;
|
|
|
|
|
|
|
|
if (samplerAnisotropyOption >= 0) {
|
|
|
|
info.useAnisotropy = samplerAnisotropyOption > 0;
|
|
|
|
info.maxAnisotropy = float(samplerAnisotropyOption);
|
|
|
|
}
|
|
|
|
|
2018-03-05 02:21:34 +01:00
|
|
|
m_sampler = device->GetDXVKDevice()->createSampler(info);
|
2017-12-09 20:49:56 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
D3D11SamplerState::~D3D11SamplerState() {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-10-14 01:27:59 +02:00
|
|
|
ULONG STDMETHODCALLTYPE D3D11SamplerState::AddRef() {
|
|
|
|
ULONG refCount = m_refCount++;
|
|
|
|
if (!refCount)
|
|
|
|
m_device->AddRef();
|
2019-10-25 21:36:08 +02:00
|
|
|
return refCount + 1;
|
2019-10-14 01:27:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
ULONG STDMETHODCALLTYPE D3D11SamplerState::Release() {
|
|
|
|
ULONG refCount = --m_refCount;
|
|
|
|
if (!refCount)
|
|
|
|
m_device->Release();
|
|
|
|
return refCount;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-12-12 12:50:52 +01:00
|
|
|
HRESULT STDMETHODCALLTYPE D3D11SamplerState::QueryInterface(REFIID riid, void** ppvObject) {
|
2019-02-10 08:01:01 +01:00
|
|
|
if (ppvObject == nullptr)
|
|
|
|
return E_POINTER;
|
|
|
|
|
2018-04-02 12:52:02 +02:00
|
|
|
*ppvObject = nullptr;
|
|
|
|
|
|
|
|
if (riid == __uuidof(IUnknown)
|
|
|
|
|| riid == __uuidof(ID3D11DeviceChild)
|
|
|
|
|| riid == __uuidof(ID3D11SamplerState)) {
|
|
|
|
*ppvObject = ref(this);
|
|
|
|
return S_OK;
|
|
|
|
}
|
2017-12-09 20:49:56 +01:00
|
|
|
|
2018-08-11 20:42:28 +02:00
|
|
|
if (riid == __uuidof(ID3D10DeviceChild)
|
|
|
|
|| riid == __uuidof(ID3D10SamplerState)) {
|
|
|
|
*ppvObject = ref(&m_d3d10);
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
2017-12-09 20:49:56 +01:00
|
|
|
Logger::warn("D3D11SamplerState::QueryInterface: Unknown interface query");
|
2018-03-12 12:05:43 +01:00
|
|
|
Logger::warn(str::format(riid));
|
2017-12-09 20:49:56 +01:00
|
|
|
return E_NOINTERFACE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-12-12 12:50:52 +01:00
|
|
|
void STDMETHODCALLTYPE D3D11SamplerState::GetDevice(ID3D11Device** ppDevice) {
|
2018-03-18 14:57:14 +01:00
|
|
|
*ppDevice = ref(m_device);
|
2017-12-09 20:49:56 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-12-12 12:50:52 +01:00
|
|
|
void STDMETHODCALLTYPE D3D11SamplerState::GetDesc(D3D11_SAMPLER_DESC* pDesc) {
|
2017-12-09 20:49:56 +01:00
|
|
|
*pDesc = m_desc;
|
|
|
|
}
|
|
|
|
|
2018-03-05 02:21:34 +01:00
|
|
|
|
2018-03-18 23:35:40 +01:00
|
|
|
HRESULT D3D11SamplerState::NormalizeDesc(D3D11_SAMPLER_DESC* pDesc) {
|
2018-05-09 11:55:05 +02:00
|
|
|
const uint32_t filterBits = uint32_t(pDesc->Filter);
|
2018-03-05 02:21:34 +01:00
|
|
|
|
|
|
|
if (filterBits & 0xFFFFFF2A) {
|
2018-04-30 10:41:57 +02:00
|
|
|
Logger::err(str::format(
|
|
|
|
"D3D11SamplerState: Unhandled filter: ", filterBits));
|
2018-03-05 02:21:34 +01:00
|
|
|
return E_INVALIDARG;
|
|
|
|
}
|
|
|
|
|
2018-04-30 19:36:42 +02:00
|
|
|
if (pDesc->MaxAnisotropy < 0
|
|
|
|
|| pDesc->MaxAnisotropy > 16) {
|
|
|
|
return E_INVALIDARG;
|
|
|
|
} else if ((filterBits & 0x40) == 0 /* not anisotropic */) {
|
2018-04-30 10:41:57 +02:00
|
|
|
// Reset anisotropy if it is not used
|
|
|
|
pDesc->MaxAnisotropy = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (filterBits & 0x80 /* compare-to-depth */) {
|
|
|
|
if (!ValidateComparisonFunc(pDesc->ComparisonFunc))
|
|
|
|
return E_INVALIDARG;
|
|
|
|
} else {
|
|
|
|
// Reset compare func if it is not used
|
|
|
|
pDesc->ComparisonFunc = D3D11_COMPARISON_NEVER;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!ValidateAddressMode(pDesc->AddressU)
|
|
|
|
|| !ValidateAddressMode(pDesc->AddressV)
|
|
|
|
|| !ValidateAddressMode(pDesc->AddressW))
|
|
|
|
return E_INVALIDARG;
|
|
|
|
|
|
|
|
// Clear BorderColor to 0 if none of the address
|
|
|
|
// modes are D3D11_TEXTURE_ADDRESS_BORDER
|
|
|
|
if (pDesc->AddressU != D3D11_TEXTURE_ADDRESS_BORDER
|
|
|
|
&& pDesc->AddressV != D3D11_TEXTURE_ADDRESS_BORDER
|
|
|
|
&& pDesc->AddressW != D3D11_TEXTURE_ADDRESS_BORDER) {
|
|
|
|
for (int i = 0; i < 4; i++)
|
|
|
|
pDesc->BorderColor[i] = 0.0f;
|
|
|
|
}
|
|
|
|
|
2018-03-05 02:21:34 +01:00
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
2018-04-30 10:41:57 +02:00
|
|
|
|
|
|
|
bool D3D11SamplerState::ValidateAddressMode(D3D11_TEXTURE_ADDRESS_MODE Mode) {
|
|
|
|
return Mode >= D3D11_TEXTURE_ADDRESS_WRAP
|
|
|
|
&& Mode <= D3D11_TEXTURE_ADDRESS_MIRROR_ONCE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool D3D11SamplerState::ValidateComparisonFunc(D3D11_COMPARISON_FUNC Comparison) {
|
|
|
|
return Comparison >= D3D11_COMPARISON_NEVER
|
|
|
|
&& Comparison <= D3D11_COMPARISON_ALWAYS;
|
|
|
|
}
|
|
|
|
|
2017-12-09 20:49:56 +01:00
|
|
|
}
|