1
0
mirror of https://github.com/doitsujin/dxvk.git synced 2025-01-07 07:46:19 +01:00
dxvk/src/d3d11/d3d11_sampler.cpp

140 lines
4.7 KiB
C++
Raw Normal View History

2017-12-09 20:49:56 +01:00
#include "d3d11_device.h"
#include "d3d11_sampler.h"
#include "d3d11_util.h"
2017-12-09 20:49:56 +01:00
namespace dxvk {
D3D11SamplerState::D3D11SamplerState(
D3D11Device* device,
const D3D11_SAMPLER_DESC& desc)
: m_device(device), m_desc(desc) {
DxvkSamplerCreateInfo info;
2017-12-09 20:49:56 +01:00
// While D3D11_FILTER is technically an enum, its value bits
// can be used to decode the filter properties more efficiently.
const uint32_t filterBits = static_cast<uint32_t>(desc.Filter);
info.magFilter = (filterBits & 0x04) ? VK_FILTER_LINEAR : VK_FILTER_NEAREST;
info.minFilter = (filterBits & 0x10) ? VK_FILTER_LINEAR : VK_FILTER_NEAREST;
info.mipmapMode = (filterBits & 0x01) ? VK_SAMPLER_MIPMAP_MODE_LINEAR : VK_SAMPLER_MIPMAP_MODE_NEAREST;
info.useAnisotropy = (filterBits & 0x40) ? VK_TRUE : VK_FALSE;
info.compareToDepth = (filterBits & 0x80) ? VK_TRUE : VK_FALSE;
// Set up the remaining properties, which are
// stored directly in the sampler description
info.mipmapLodBias = desc.MipLODBias;
info.mipmapLodMin = desc.MinLOD;
info.mipmapLodMax = desc.MaxLOD;
info.maxAnisotropy = static_cast<float>(desc.MaxAnisotropy);
info.addressModeU = DecodeAddressMode(desc.AddressU);
info.addressModeV = DecodeAddressMode(desc.AddressV);
info.addressModeW = DecodeAddressMode(desc.AddressW);
info.compareOp = DecodeCompareOp(desc.ComparisonFunc);
info.borderColor = VK_BORDER_COLOR_FLOAT_TRANSPARENT_BLACK;
info.usePixelCoord = VK_FALSE; // Not supported in D3D11
// Make sure to use a valid anisotropy value
if (desc.MaxAnisotropy < 1) info.maxAnisotropy = 1.0f;
if (desc.MaxAnisotropy > 16) info.maxAnisotropy = 16.0f;
// Try to find a matching border color if clamp to border is enabled
if (info.addressModeU == VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_BORDER
|| info.addressModeV == VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_BORDER
|| info.addressModeW == VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_BORDER)
info.borderColor = DecodeBorderColor(desc.BorderColor);
m_sampler = device->GetDXVKDevice()->createSampler(info);
2017-12-09 20:49:56 +01:00
}
D3D11SamplerState::~D3D11SamplerState() {
}
2017-12-12 12:50:52 +01:00
HRESULT STDMETHODCALLTYPE D3D11SamplerState::QueryInterface(REFIID riid, void** ppvObject) {
*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
Logger::warn("D3D11SamplerState::QueryInterface: Unknown interface query");
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) {
*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;
}
HRESULT D3D11SamplerState::NormalizeDesc(D3D11_SAMPLER_DESC* pDesc) {
const uint32_t filterBits = static_cast<uint32_t>(pDesc->Filter);
if (filterBits & 0xFFFFFF2A) {
Logger::err(str::format(
"D3D11SamplerState: Unhandled filter: ", filterBits));
return E_INVALIDARG;
}
if (filterBits & 0x40 /* anisotropic */) {
if (pDesc->MaxAnisotropy < 1
|| pDesc->MaxAnisotropy > 16)
return E_INVALIDARG;
} else if (pDesc->MaxAnisotropy < 0
|| pDesc->MaxAnisotropy > 16) {
return E_INVALIDARG;
} else {
// 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;
}
return S_OK;
}
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
}