mirror of
https://github.com/doitsujin/dxvk.git
synced 2025-03-06 20:58:37 +01:00
Merge 61237b7614
into a08579e555
This commit is contained in:
commit
d6171725cd
17 changed files with 220 additions and 159 deletions
83
dxvk.conf
83
dxvk.conf
|
@ -493,7 +493,8 @@
|
||||||
# Reported shader model
|
# Reported shader model
|
||||||
#
|
#
|
||||||
# The shader model to state that we support in the device
|
# The shader model to state that we support in the device
|
||||||
# capabilities that the applicatation queries.
|
# capabilities that the application queries. Note that
|
||||||
|
# the value will be limited to 1 for D3D8 applications.
|
||||||
#
|
#
|
||||||
# Supported values:
|
# Supported values:
|
||||||
# - 0: Fixed-function only
|
# - 0: Fixed-function only
|
||||||
|
@ -763,3 +764,83 @@
|
||||||
# - True/False
|
# - True/False
|
||||||
|
|
||||||
# d3d9.countLosableResources = True
|
# d3d9.countLosableResources = True
|
||||||
|
|
||||||
|
# Dref scaling for DXS0/FVF
|
||||||
|
#
|
||||||
|
# Some early D3D8 games expect Dref (depth texcoord Z) to be on the range of
|
||||||
|
# [0..2^bitDepth - 1]. This option allows DXSO and fixed vertex function to
|
||||||
|
# scale it back down to [0..1].
|
||||||
|
#
|
||||||
|
# Supported values: Any number representing bitDepth (typically 24).
|
||||||
|
|
||||||
|
# d3d8.drefScaling = 0
|
||||||
|
|
||||||
|
# Shadow perspective divide
|
||||||
|
#
|
||||||
|
# Older applications designed for Nvidia hardware (or ported from XBox)
|
||||||
|
# expect shadow map texture coordinates to be perspective divided, even
|
||||||
|
# though D3DTTFF_PROJECTED is never set for any texture coordinates.
|
||||||
|
# Older Nvidia cards (GeForce 3, GeForce 4 series) performed this
|
||||||
|
# projection directly in hardware.
|
||||||
|
#
|
||||||
|
# This option forces the D3DTTFF_PROJECTED flag for the necessary stages
|
||||||
|
# when a depth texture is bound to slot 0, in order to emulate older
|
||||||
|
# Nvidia hardware behavior.
|
||||||
|
#
|
||||||
|
# Supported values:
|
||||||
|
# - True/False
|
||||||
|
|
||||||
|
# d3d8.shadowPerspectiveDivide = False
|
||||||
|
|
||||||
|
# Force vertex shader declaration
|
||||||
|
#
|
||||||
|
# Some games rely on undefined behavior by using undeclared vertex shader inputs.
|
||||||
|
# The simplest way to fix them is to modify their vertex shader decl.
|
||||||
|
#
|
||||||
|
# This option takes a comma-separated list of colon-separated number pairs, where
|
||||||
|
# the first number is a D3DVSDE_REGISTER value, the second is a D3DVSDT_TYPE value.
|
||||||
|
#
|
||||||
|
# Supported values:
|
||||||
|
# - e.g. "0:2,3:2,7:1" for float3 position : v0, float3 normal : v3, float2 uv : v7.
|
||||||
|
|
||||||
|
# d3d8.forceVsDecl = ""
|
||||||
|
|
||||||
|
# Draw call batching
|
||||||
|
#
|
||||||
|
# Specialized drawcall batcher, typically for games that draw a lot of similar
|
||||||
|
# geometry in separate drawcalls (sometimes even one triangle at a time).
|
||||||
|
#
|
||||||
|
# May hurt performance or introduce graphical artifacts outside of
|
||||||
|
# specific games that are known to benefit from it.
|
||||||
|
#
|
||||||
|
# Supported values:
|
||||||
|
# - True/False
|
||||||
|
|
||||||
|
# d3d8.batching = False
|
||||||
|
|
||||||
|
# P8 texture support workaround
|
||||||
|
#
|
||||||
|
# Early Nvidia GPUs, such as the GeForce 4 generation cards, included and exposed
|
||||||
|
# P8 texture support. However, it was no longer advertised with cards in the FX series
|
||||||
|
# and above. ATI/AMD drivers and hardware were most likely in a similar situation.
|
||||||
|
#
|
||||||
|
# This option will ensure all P8 textures are placed in D3DPOOL_SCRATCH, so that
|
||||||
|
# their creation is guaranteed to succeed even if the format is unsupported.
|
||||||
|
# Can help older titles that don't properly handle the lack of P8 support.
|
||||||
|
#
|
||||||
|
# Supported values:
|
||||||
|
# - True/False
|
||||||
|
|
||||||
|
# d3d8.placeP8InScratch = False
|
||||||
|
|
||||||
|
# Legacy discard buffer behavior
|
||||||
|
#
|
||||||
|
# Older applications may rely on D3DLOCK_DISCARD being ignored for everything
|
||||||
|
# except D3DUSAGE_DYNAMIC + D3DUSAGE_WRITEONLY buffers, however this approach
|
||||||
|
# incurs a performance penalty.
|
||||||
|
#
|
||||||
|
# Supported values:
|
||||||
|
# - True/False
|
||||||
|
|
||||||
|
# d3d8.forceLegacyDiscard = False
|
||||||
|
|
||||||
|
|
|
@ -1815,19 +1815,31 @@ namespace dxvk {
|
||||||
if (unlikely(pDeclaration == nullptr || pHandle == nullptr))
|
if (unlikely(pDeclaration == nullptr || pHandle == nullptr))
|
||||||
return D3DERR_INVALIDCALL;
|
return D3DERR_INVALIDCALL;
|
||||||
|
|
||||||
// Validate VS version for non-FF shaders
|
D3D9VertexShaderCode translatedVS;
|
||||||
|
HRESULT res = TranslateVertexShader8(pDeclaration, pFunction, m_d3d8Options, &translatedVS);
|
||||||
|
if (unlikely(FAILED(res)))
|
||||||
|
return res;
|
||||||
|
|
||||||
|
// Create vertex declaration
|
||||||
|
Com<d3d9::IDirect3DVertexDeclaration9> pVertexDecl;
|
||||||
|
res = GetD3D9()->CreateVertexDeclaration(translatedVS.declaration, &pVertexDecl);
|
||||||
|
if (unlikely(FAILED(res)))
|
||||||
|
return res;
|
||||||
|
|
||||||
|
Com<d3d9::IDirect3DVertexShader9> pVertexShader;
|
||||||
if (pFunction != nullptr) {
|
if (pFunction != nullptr) {
|
||||||
const uint32_t majorVersion = D3DSHADER_VERSION_MAJOR(pFunction[0]);
|
res = GetD3D9()->CreateVertexShader(translatedVS.function.data(), &pVertexShader);
|
||||||
const uint32_t minorVersion = D3DSHADER_VERSION_MINOR(pFunction[0]);
|
} else {
|
||||||
|
// pFunction is NULL: fixed function pipeline
|
||||||
if (unlikely(majorVersion != 1 || minorVersion > 1)) {
|
pVertexShader = nullptr;
|
||||||
Logger::err(str::format("D3D8Device::CreateVertexShader: Unsupported VS version ", majorVersion, ".", minorVersion));
|
|
||||||
return D3DERR_INVALIDCALL;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (likely(SUCCEEDED(res))) {
|
||||||
D3D8VertexShaderInfo& info = m_vertexShaders.emplace_back();
|
D3D8VertexShaderInfo& info = m_vertexShaders.emplace_back();
|
||||||
|
|
||||||
|
info.pVertexDecl = std::move(pVertexDecl);
|
||||||
|
info.pVertexShader = std::move(pVertexShader);
|
||||||
|
|
||||||
// Store D3D8 bytecodes in the shader info
|
// Store D3D8 bytecodes in the shader info
|
||||||
for (UINT i = 0; pDeclaration[i] != D3DVSD_END(); i++)
|
for (UINT i = 0; pDeclaration[i] != D3DVSD_END(); i++)
|
||||||
info.declaration.push_back(pDeclaration[i]);
|
info.declaration.push_back(pDeclaration[i]);
|
||||||
|
@ -1839,21 +1851,6 @@ namespace dxvk {
|
||||||
info.function.push_back(D3DVS_END());
|
info.function.push_back(D3DVS_END());
|
||||||
}
|
}
|
||||||
|
|
||||||
D3D9VertexShaderCode result = TranslateVertexShader8(pDeclaration, pFunction, m_d3d8Options);
|
|
||||||
|
|
||||||
// Create vertex declaration
|
|
||||||
HRESULT res = GetD3D9()->CreateVertexDeclaration(result.declaration, &(info.pVertexDecl));
|
|
||||||
if (unlikely(FAILED(res)))
|
|
||||||
return res;
|
|
||||||
|
|
||||||
if (pFunction != nullptr) {
|
|
||||||
res = GetD3D9()->CreateVertexShader(result.function.data(), &(info.pVertexShader));
|
|
||||||
} else {
|
|
||||||
// pFunction is NULL: fixed function pipeline
|
|
||||||
info.pVertexShader = nullptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (likely(SUCCEEDED(res))) {
|
|
||||||
// Set bit to indicate this is not an FVF
|
// Set bit to indicate this is not an FVF
|
||||||
*pHandle = getShaderHandle(m_vertexShaders.size());
|
*pHandle = getShaderHandle(m_vertexShaders.size());
|
||||||
}
|
}
|
||||||
|
@ -2056,20 +2053,11 @@ namespace dxvk {
|
||||||
if (unlikely(pFunction == nullptr || pHandle == nullptr))
|
if (unlikely(pFunction == nullptr || pHandle == nullptr))
|
||||||
return D3DERR_INVALIDCALL;
|
return D3DERR_INVALIDCALL;
|
||||||
|
|
||||||
const uint32_t majorVersion = D3DSHADER_VERSION_MAJOR(pFunction[0]);
|
Com<d3d9::IDirect3DPixelShader9> pPixelShader;
|
||||||
const uint32_t minorVersion = D3DSHADER_VERSION_MINOR(pFunction[0]);
|
|
||||||
|
|
||||||
if (unlikely(m_isFixedFunctionOnly || majorVersion != 1 || minorVersion > 4)) {
|
|
||||||
Logger::err(str::format("D3D8Device::CreatePixelShader: Unsupported PS version ", majorVersion, ".", minorVersion));
|
|
||||||
return D3DERR_INVALIDCALL;
|
|
||||||
}
|
|
||||||
|
|
||||||
d3d9::IDirect3DPixelShader9* pPixelShader;
|
|
||||||
|
|
||||||
HRESULT res = GetD3D9()->CreatePixelShader(pFunction, &pPixelShader);
|
HRESULT res = GetD3D9()->CreatePixelShader(pFunction, &pPixelShader);
|
||||||
|
|
||||||
if (likely(SUCCEEDED(res))) {
|
if (likely(SUCCEEDED(res))) {
|
||||||
m_pixelShaders.push_back(pPixelShader);
|
m_pixelShaders.push_back(std::move(pPixelShader));
|
||||||
// Still set the shader bit, to prevent conflicts with NULL.
|
// Still set the shader bit, to prevent conflicts with NULL.
|
||||||
*pHandle = getShaderHandle(m_pixelShaders.size());
|
*pHandle = getShaderHandle(m_pixelShaders.size());
|
||||||
}
|
}
|
||||||
|
|
|
@ -15,7 +15,7 @@ namespace dxvk {
|
||||||
throw DxvkError("D3D8Interface: ERROR! Failed to get D3D9 Bridge. d3d9.dll might not be DXVK!");
|
throw DxvkError("D3D8Interface: ERROR! Failed to get D3D9 Bridge. d3d9.dll might not be DXVK!");
|
||||||
}
|
}
|
||||||
|
|
||||||
m_bridge->SetD3D8CompatibilityMode(true);
|
m_bridge->EnableD3D8CompatibilityMode();
|
||||||
|
|
||||||
m_d3d8Options = D3D8Options(*m_bridge->GetConfig());
|
m_d3d8Options = D3D8Options(*m_bridge->GetConfig());
|
||||||
|
|
||||||
|
|
|
@ -8,53 +8,29 @@ namespace dxvk {
|
||||||
|
|
||||||
struct D3D8Options {
|
struct D3D8Options {
|
||||||
|
|
||||||
/// Some games rely on undefined behavior by using undeclared vertex shader inputs.
|
/// Override application vertex shader declarations.
|
||||||
/// The simplest way to fix them is to simply modify their vertex shader decl.
|
|
||||||
///
|
|
||||||
/// This option takes a comma-separated list of colon-separated number pairs, where
|
|
||||||
/// the first number is a D3DVSDE_REGISTER value, the second is a D3DVSDT_TYPE value.
|
|
||||||
/// e.g. "0:2,3:2,7:1" for float3 position : v0, float3 normal : v3, float2 uv : v7
|
|
||||||
std::vector<std::pair<D3DVSDE_REGISTER, D3DVSDT_TYPE>> forceVsDecl;
|
std::vector<std::pair<D3DVSDE_REGISTER, D3DVSDT_TYPE>> forceVsDecl;
|
||||||
|
|
||||||
/// Specialized drawcall batcher, typically for games that draw a lot of similar
|
/// Enable/disable the drawcall batcher.
|
||||||
/// geometry in separate drawcalls (sometimes even one triangle at a time).
|
bool batching;
|
||||||
///
|
|
||||||
/// May hurt performance outside of specifc games that benefit from it.
|
|
||||||
bool batching = false;
|
|
||||||
|
|
||||||
/// The Lord of the Rings: The Fellowship of the Ring tries to create a P8 texture
|
/// Place all P8 textures in D3DPOOL_SCRATCH.
|
||||||
/// in D3DPOOL_MANAGED on Nvidia and Intel, which fails, but has a separate code
|
bool placeP8InScratch;
|
||||||
/// path for ATI/AMD that creates it in D3DPOOL_SCRATCH instead, which works.
|
|
||||||
///
|
|
||||||
/// The internal logic determining this path doesn't seem to be d3d-related, but
|
|
||||||
/// the game works universally if we mimic its own ATI/AMD workaround during P8
|
|
||||||
/// texture creation.
|
|
||||||
///
|
|
||||||
/// Early Nvidia GPUs, such as the GeForce 4 generation cards, included and exposed
|
|
||||||
/// P8 texture support. However, it was no longer advertised with cards in the FX series
|
|
||||||
/// and above. Most likely ATI/AMD drivers never supported P8 in the first place.
|
|
||||||
bool placeP8InScratch = false;
|
|
||||||
|
|
||||||
/// Rayman 3 relies on D3DLOCK_DISCARD being ignored for everything except D3DUSAGE_DYNAMIC +
|
/// Ignore D3DLOCK_DISCARD for everything except D3DUSAGE_DYNAMIC + D3DUSAGE_WRITEONLY buffers.
|
||||||
/// D3DUSAGE_WRITEONLY buffers, however this approach incurs a performance penalty.
|
bool forceLegacyDiscard;
|
||||||
///
|
|
||||||
/// Some titles might abuse this early D3D8 quirk, however at some point in its history
|
|
||||||
/// it was brought in line with standard D3D9 behavior.
|
|
||||||
bool forceLegacyDiscard = false;
|
|
||||||
|
|
||||||
/// Splinter Cell expects shadow map texture coordinates to be perspective divided
|
/// Force D3DTTFF_PROJECTED for the necessary stages when a depth texture is bound to slot 0.
|
||||||
/// even though D3DTTFF_PROJECTED is never set for any texture coordinates. This flag
|
bool shadowPerspectiveDivide;
|
||||||
/// forces that flag for the necessary stages when a depth texture is bound to slot 0
|
|
||||||
bool shadowPerspectiveDivide = false;
|
|
||||||
|
|
||||||
D3D8Options() {}
|
D3D8Options() {}
|
||||||
|
|
||||||
D3D8Options(const Config& config) {
|
D3D8Options(const Config& config) {
|
||||||
auto forceVsDeclStr = config.getOption<std::string>("d3d8.forceVsDecl", "");
|
auto forceVsDeclStr = config.getOption<std::string>("d3d8.forceVsDecl", "");
|
||||||
batching = config.getOption<bool> ("d3d8.batching", batching);
|
batching = config.getOption<bool> ("d3d8.batching", false);
|
||||||
placeP8InScratch = config.getOption<bool> ("d3d8.placeP8InScratch", placeP8InScratch);
|
placeP8InScratch = config.getOption<bool> ("d3d8.placeP8InScratch", false);
|
||||||
forceLegacyDiscard = config.getOption<bool> ("d3d8.forceLegacyDiscard", forceLegacyDiscard);
|
forceLegacyDiscard = config.getOption<bool> ("d3d8.forceLegacyDiscard", false);
|
||||||
shadowPerspectiveDivide = config.getOption<bool> ("d3d8.shadowPerspectiveDivide", shadowPerspectiveDivide);
|
shadowPerspectiveDivide = config.getOption<bool> ("d3d8.shadowPerspectiveDivide", false);
|
||||||
|
|
||||||
parseVsDecl(forceVsDeclStr);
|
parseVsDecl(forceVsDeclStr);
|
||||||
}
|
}
|
||||||
|
|
|
@ -110,26 +110,27 @@ namespace dxvk {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Converts a D3D8 vertex shader + declaration
|
* Validates and converts a D3D8 vertex shader
|
||||||
* to a D3D9 vertex shader + declaration.
|
* + declaration to a D3D9 vertex shader + declaration.
|
||||||
*/
|
*/
|
||||||
D3D9VertexShaderCode TranslateVertexShader8(
|
HRESULT TranslateVertexShader8(
|
||||||
const DWORD* pDeclaration,
|
const DWORD* pDeclaration,
|
||||||
const DWORD* pFunction,
|
const DWORD* pFunction,
|
||||||
const D3D8Options& options) {
|
const D3D8Options& options,
|
||||||
|
D3D9VertexShaderCode* pTranslatedVS) {
|
||||||
using d3d9::D3DDECLTYPE;
|
using d3d9::D3DDECLTYPE;
|
||||||
using d3d9::D3DDECLTYPE_UNUSED;
|
using d3d9::D3DDECLTYPE_UNUSED;
|
||||||
|
|
||||||
D3D9VertexShaderCode result;
|
HRESULT res = D3D_OK;
|
||||||
|
|
||||||
std::vector<DWORD>& tokens = result.function;
|
std::vector<DWORD>& tokens = pTranslatedVS->function;
|
||||||
std::vector<DWORD> defs; // Constant definitions
|
std::vector<DWORD> defs; // Constant definitions
|
||||||
|
|
||||||
// shaderInputRegisters:
|
// shaderInputRegisters:
|
||||||
// set bit N to enable input register vN
|
// set bit N to enable input register vN
|
||||||
DWORD shaderInputRegisters = 0;
|
DWORD shaderInputRegisters = 0;
|
||||||
|
|
||||||
d3d9::D3DVERTEXELEMENT9* vertexElements = result.declaration;
|
d3d9::D3DVERTEXELEMENT9* vertexElements = pTranslatedVS->declaration;
|
||||||
unsigned int elementIdx = 0;
|
unsigned int elementIdx = 0;
|
||||||
|
|
||||||
// These are used for pDeclaration and pFunction
|
// These are used for pDeclaration and pFunction
|
||||||
|
@ -206,6 +207,12 @@ namespace dxvk {
|
||||||
D3DVSDT_TYPE type = D3DVSDT_TYPE(VSD_SHIFT_MASK(token, D3DVSD_DATATYPE));
|
D3DVSDT_TYPE type = D3DVSDT_TYPE(VSD_SHIFT_MASK(token, D3DVSD_DATATYPE));
|
||||||
D3DVSDE_REGISTER reg = D3DVSDE_REGISTER(VSD_SHIFT_MASK(token, D3DVSD_VERTEXREG));
|
D3DVSDE_REGISTER reg = D3DVSDE_REGISTER(VSD_SHIFT_MASK(token, D3DVSD_VERTEXREG));
|
||||||
|
|
||||||
|
// FVF normals are expected to only have 3 components
|
||||||
|
if (unlikely(pFunction == nullptr && reg == D3DVSDE_NORMAL && type != D3DVSDT_FLOAT3)) {
|
||||||
|
Logger::err("D3D8Device::CreateVertexShader: Invalid FVF declaration: D3DVSDE_NORMAL must use D3DVSDT_FLOAT3");
|
||||||
|
return D3DERR_INVALIDCALL;
|
||||||
|
}
|
||||||
|
|
||||||
addVertexElement(reg, type);
|
addVertexElement(reg, type);
|
||||||
|
|
||||||
dbg << "type=" << type << ", register=" << reg;
|
dbg << "type=" << type << ", register=" << reg;
|
||||||
|
@ -332,7 +339,7 @@ namespace dxvk {
|
||||||
} while (token != D3DVS_END());
|
} while (token != D3DVS_END());
|
||||||
}
|
}
|
||||||
|
|
||||||
return result;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,9 +10,10 @@ namespace dxvk {
|
||||||
std::vector<DWORD> function;
|
std::vector<DWORD> function;
|
||||||
};
|
};
|
||||||
|
|
||||||
D3D9VertexShaderCode TranslateVertexShader8(
|
HRESULT TranslateVertexShader8(
|
||||||
const DWORD* pDeclaration,
|
const DWORD* pDeclaration,
|
||||||
const DWORD* pFunction,
|
const DWORD* pFunction,
|
||||||
const D3D8Options& overrides);
|
const D3D8Options& overrides,
|
||||||
|
D3D9VertexShaderCode* pTranslatedVS);
|
||||||
|
|
||||||
}
|
}
|
|
@ -114,7 +114,10 @@ namespace dxvk {
|
||||||
DWORD Usage,
|
DWORD Usage,
|
||||||
D3DRESOURCETYPE RType,
|
D3DRESOURCETYPE RType,
|
||||||
D3D9Format CheckFormat) {
|
D3D9Format CheckFormat) {
|
||||||
if(unlikely(AdapterFormat == D3D9Format::Unknown))
|
if (unlikely(AdapterFormat == D3D9Format::Unknown))
|
||||||
|
return D3DERR_INVALIDCALL;
|
||||||
|
|
||||||
|
if (unlikely(RType == D3DRTYPE_VERTEXBUFFER || RType == D3DRTYPE_INDEXBUFFER))
|
||||||
return D3DERR_INVALIDCALL;
|
return D3DERR_INVALIDCALL;
|
||||||
|
|
||||||
if (!IsSupportedAdapterFormat(AdapterFormat))
|
if (!IsSupportedAdapterFormat(AdapterFormat))
|
||||||
|
@ -168,9 +171,6 @@ namespace dxvk {
|
||||||
if (RType == D3DRTYPE_CUBETEXTURE && mapping.Aspect != VK_IMAGE_ASPECT_COLOR_BIT)
|
if (RType == D3DRTYPE_CUBETEXTURE && mapping.Aspect != VK_IMAGE_ASPECT_COLOR_BIT)
|
||||||
return D3DERR_NOTAVAILABLE;
|
return D3DERR_NOTAVAILABLE;
|
||||||
|
|
||||||
if (RType == D3DRTYPE_VERTEXBUFFER || RType == D3DRTYPE_INDEXBUFFER)
|
|
||||||
return D3D_OK;
|
|
||||||
|
|
||||||
// Let's actually ask Vulkan now that we got some quirks out the way!
|
// Let's actually ask Vulkan now that we got some quirks out the way!
|
||||||
VkFormat format = mapping.FormatColor;
|
VkFormat format = mapping.FormatColor;
|
||||||
if (unlikely(mapping.ConversionFormatInfo.FormatColor != VK_FORMAT_UNDEFINED)) {
|
if (unlikely(mapping.ConversionFormatInfo.FormatColor != VK_FORMAT_UNDEFINED)) {
|
||||||
|
@ -297,6 +297,7 @@ namespace dxvk {
|
||||||
|
|
||||||
auto& options = m_parent->GetOptions();
|
auto& options = m_parent->GetOptions();
|
||||||
|
|
||||||
|
const uint32_t maxShaderModel = m_parent->IsD3D8Compatible() ? std::min(1u, options.shaderModel) : options.shaderModel;
|
||||||
const VkPhysicalDeviceLimits& limits = m_adapter->deviceProperties().limits;
|
const VkPhysicalDeviceLimits& limits = m_adapter->deviceProperties().limits;
|
||||||
|
|
||||||
// TODO: Actually care about what the adapter supports here.
|
// TODO: Actually care about what the adapter supports here.
|
||||||
|
@ -575,8 +576,8 @@ namespace dxvk {
|
||||||
|
|
||||||
// Late fixed-function capable cards, such as the GeForce 4 MX series,
|
// Late fixed-function capable cards, such as the GeForce 4 MX series,
|
||||||
// expose support for VS 1.1, while not advertising any PS support
|
// expose support for VS 1.1, while not advertising any PS support
|
||||||
const uint32_t majorVersionVS = options.shaderModel == 0 ? 1 : options.shaderModel;
|
const uint32_t majorVersionVS = maxShaderModel == 0 ? 1 : maxShaderModel;
|
||||||
const uint32_t majorVersionPS = options.shaderModel;
|
const uint32_t majorVersionPS = maxShaderModel;
|
||||||
// Max supported SM1 is VS 1.1 and PS 1.4
|
// Max supported SM1 is VS 1.1 and PS 1.4
|
||||||
const uint32_t minorVersionVS = majorVersionVS != 1 ? 0 : 1;
|
const uint32_t minorVersionVS = majorVersionVS != 1 ? 0 : 1;
|
||||||
const uint32_t minorVersionPS = majorVersionPS != 1 ? 0 : 4;
|
const uint32_t minorVersionPS = majorVersionPS != 1 ? 0 : 4;
|
||||||
|
@ -588,7 +589,7 @@ namespace dxvk {
|
||||||
// Max Vertex Shader Const
|
// Max Vertex Shader Const
|
||||||
pCaps->MaxVertexShaderConst = MaxFloatConstantsVS;
|
pCaps->MaxVertexShaderConst = MaxFloatConstantsVS;
|
||||||
// Max PS1 Value
|
// Max PS1 Value
|
||||||
pCaps->PixelShader1xMaxValue = options.shaderModel > 0 ? std::numeric_limits<float>::max() : 0.0f;
|
pCaps->PixelShader1xMaxValue = maxShaderModel > 0 ? std::numeric_limits<float>::max() : 0.0f;
|
||||||
// Dev Caps 2
|
// Dev Caps 2
|
||||||
pCaps->DevCaps2 = D3DDEVCAPS2_STREAMOFFSET
|
pCaps->DevCaps2 = D3DDEVCAPS2_STREAMOFFSET
|
||||||
/* | D3DDEVCAPS2_DMAPNPATCH */
|
/* | D3DDEVCAPS2_DMAPNPATCH */
|
||||||
|
@ -635,23 +636,23 @@ namespace dxvk {
|
||||||
/* | D3DPTFILTERCAPS_MAGFPYRAMIDALQUAD */
|
/* | D3DPTFILTERCAPS_MAGFPYRAMIDALQUAD */
|
||||||
/* | D3DPTFILTERCAPS_MAGFGAUSSIANQUAD */;
|
/* | D3DPTFILTERCAPS_MAGFGAUSSIANQUAD */;
|
||||||
|
|
||||||
pCaps->VS20Caps.Caps = options.shaderModel >= 2 ? D3DVS20CAPS_PREDICATION : 0;
|
pCaps->VS20Caps.Caps = maxShaderModel >= 2 ? D3DVS20CAPS_PREDICATION : 0;
|
||||||
pCaps->VS20Caps.DynamicFlowControlDepth = options.shaderModel >= 2 ? D3DVS20_MAX_DYNAMICFLOWCONTROLDEPTH : 0;
|
pCaps->VS20Caps.DynamicFlowControlDepth = maxShaderModel >= 2 ? D3DVS20_MAX_DYNAMICFLOWCONTROLDEPTH : 0;
|
||||||
pCaps->VS20Caps.NumTemps = options.shaderModel >= 2 ? D3DVS20_MAX_NUMTEMPS : 0;
|
pCaps->VS20Caps.NumTemps = maxShaderModel >= 2 ? D3DVS20_MAX_NUMTEMPS : 0;
|
||||||
pCaps->VS20Caps.StaticFlowControlDepth = options.shaderModel >= 2 ? D3DVS20_MAX_STATICFLOWCONTROLDEPTH : 0;
|
pCaps->VS20Caps.StaticFlowControlDepth = maxShaderModel >= 2 ? D3DVS20_MAX_STATICFLOWCONTROLDEPTH : 0;
|
||||||
|
|
||||||
pCaps->PS20Caps.Caps = options.shaderModel >= 2 ? D3DPS20CAPS_ARBITRARYSWIZZLE
|
pCaps->PS20Caps.Caps = maxShaderModel >= 2 ? D3DPS20CAPS_ARBITRARYSWIZZLE
|
||||||
| D3DPS20CAPS_GRADIENTINSTRUCTIONS
|
| D3DPS20CAPS_GRADIENTINSTRUCTIONS
|
||||||
| D3DPS20CAPS_PREDICATION
|
| D3DPS20CAPS_PREDICATION
|
||||||
| D3DPS20CAPS_NODEPENDENTREADLIMIT
|
| D3DPS20CAPS_NODEPENDENTREADLIMIT
|
||||||
| D3DPS20CAPS_NOTEXINSTRUCTIONLIMIT : 0;
|
| D3DPS20CAPS_NOTEXINSTRUCTIONLIMIT : 0;
|
||||||
pCaps->PS20Caps.DynamicFlowControlDepth = options.shaderModel >= 2 ? D3DPS20_MAX_DYNAMICFLOWCONTROLDEPTH : 0;
|
pCaps->PS20Caps.DynamicFlowControlDepth = maxShaderModel >= 2 ? D3DPS20_MAX_DYNAMICFLOWCONTROLDEPTH : 0;
|
||||||
pCaps->PS20Caps.NumTemps = options.shaderModel >= 2 ? D3DPS20_MAX_NUMTEMPS : 0;
|
pCaps->PS20Caps.NumTemps = maxShaderModel >= 2 ? D3DPS20_MAX_NUMTEMPS : 0;
|
||||||
pCaps->PS20Caps.StaticFlowControlDepth = options.shaderModel >= 2 ? D3DPS20_MAX_STATICFLOWCONTROLDEPTH : 0;
|
pCaps->PS20Caps.StaticFlowControlDepth = maxShaderModel >= 2 ? D3DPS20_MAX_STATICFLOWCONTROLDEPTH : 0;
|
||||||
pCaps->PS20Caps.NumInstructionSlots = options.shaderModel >= 2 ? D3DPS20_MAX_NUMINSTRUCTIONSLOTS : 0;
|
pCaps->PS20Caps.NumInstructionSlots = maxShaderModel >= 2 ? D3DPS20_MAX_NUMINSTRUCTIONSLOTS : 0;
|
||||||
|
|
||||||
// Vertex texture samplers are only available as part of SM3, the caps are 0 otherwise.
|
// Vertex texture samplers are only available as part of SM3, the caps are 0 otherwise.
|
||||||
pCaps->VertexTextureFilterCaps = options.shaderModel == 3 ? D3DPTFILTERCAPS_MINFPOINT
|
pCaps->VertexTextureFilterCaps = maxShaderModel == 3 ? D3DPTFILTERCAPS_MINFPOINT
|
||||||
| D3DPTFILTERCAPS_MINFLINEAR
|
| D3DPTFILTERCAPS_MINFLINEAR
|
||||||
/* | D3DPTFILTERCAPS_MINFANISOTROPIC */
|
/* | D3DPTFILTERCAPS_MINFANISOTROPIC */
|
||||||
/* | D3DPTFILTERCAPS_MINFPYRAMIDALQUAD */
|
/* | D3DPTFILTERCAPS_MINFPYRAMIDALQUAD */
|
||||||
|
@ -665,11 +666,11 @@ namespace dxvk {
|
||||||
/* | D3DPTFILTERCAPS_MAGFPYRAMIDALQUAD */
|
/* | D3DPTFILTERCAPS_MAGFPYRAMIDALQUAD */
|
||||||
/* | D3DPTFILTERCAPS_MAGFGAUSSIANQUAD */ : 0;
|
/* | D3DPTFILTERCAPS_MAGFGAUSSIANQUAD */ : 0;
|
||||||
|
|
||||||
pCaps->MaxVShaderInstructionsExecuted = options.shaderModel >= 2 ? 4294967295 : 0;
|
pCaps->MaxVShaderInstructionsExecuted = maxShaderModel >= 2 ? 4294967295 : 0;
|
||||||
pCaps->MaxPShaderInstructionsExecuted = options.shaderModel >= 2 ? 4294967295 : 0;
|
pCaps->MaxPShaderInstructionsExecuted = maxShaderModel >= 2 ? 4294967295 : 0;
|
||||||
|
|
||||||
pCaps->MaxVertexShader30InstructionSlots = options.shaderModel == 3 ? 32768 : 0;
|
pCaps->MaxVertexShader30InstructionSlots = maxShaderModel == 3 ? 32768 : 0;
|
||||||
pCaps->MaxPixelShader30InstructionSlots = options.shaderModel == 3 ? 32768 : 0;
|
pCaps->MaxPixelShader30InstructionSlots = maxShaderModel == 3 ? 32768 : 0;
|
||||||
|
|
||||||
return D3D_OK;
|
return D3D_OK;
|
||||||
}
|
}
|
||||||
|
|
|
@ -114,8 +114,8 @@ namespace dxvk {
|
||||||
return m_interface->QueryInterface(riid, ppvObject);
|
return m_interface->QueryInterface(riid, ppvObject);
|
||||||
}
|
}
|
||||||
|
|
||||||
void DxvkD3D8InterfaceBridge::SetD3D8CompatibilityMode(const bool compatMode) {
|
void DxvkD3D8InterfaceBridge::EnableD3D8CompatibilityMode() {
|
||||||
m_interface->SetD3D8CompatibilityMode(compatMode);
|
m_interface->EnableD3D8CompatibilityMode();
|
||||||
}
|
}
|
||||||
|
|
||||||
const Config* DxvkD3D8InterfaceBridge::GetConfig() const {
|
const Config* DxvkD3D8InterfaceBridge::GetConfig() const {
|
||||||
|
|
|
@ -42,11 +42,9 @@ IDxvkD3D8Bridge : public IUnknown {
|
||||||
MIDL_INTERFACE("D3D9D3D8-A407-773E-18E9-CAFEBEEF3000")
|
MIDL_INTERFACE("D3D9D3D8-A407-773E-18E9-CAFEBEEF3000")
|
||||||
IDxvkD3D8InterfaceBridge : public IUnknown {
|
IDxvkD3D8InterfaceBridge : public IUnknown {
|
||||||
/**
|
/**
|
||||||
* \brief Enables or disables D3D9-specific features and validations
|
* \brief Enforces D3D8-specific features and validations
|
||||||
*
|
|
||||||
* \param [in] compatMode Compatibility state
|
|
||||||
*/
|
*/
|
||||||
virtual void SetD3D8CompatibilityMode(const bool compatMode) = 0;
|
virtual void EnableD3D8CompatibilityMode() = 0;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Retrieves the DXVK configuration
|
* \brief Retrieves the DXVK configuration
|
||||||
|
@ -106,7 +104,7 @@ namespace dxvk {
|
||||||
REFIID riid,
|
REFIID riid,
|
||||||
void** ppvObject);
|
void** ppvObject);
|
||||||
|
|
||||||
void SetD3D8CompatibilityMode(const bool compatMode);
|
void EnableD3D8CompatibilityMode();
|
||||||
|
|
||||||
const Config* GetConfig() const;
|
const Config* GetConfig() const;
|
||||||
|
|
||||||
|
|
|
@ -3329,7 +3329,7 @@ namespace dxvk {
|
||||||
const uint32_t minorVersion = D3DSHADER_VERSION_MINOR(pFunction[0]);
|
const uint32_t minorVersion = D3DSHADER_VERSION_MINOR(pFunction[0]);
|
||||||
|
|
||||||
// Late fixed-function capable hardware exposed support for VS 1.1
|
// Late fixed-function capable hardware exposed support for VS 1.1
|
||||||
const uint32_t shaderModelVS = m_d3d9Options.shaderModel == 0 ? 1 : m_d3d9Options.shaderModel;
|
const uint32_t shaderModelVS = m_isD3D8Compatible ? 1u : std::max(1u, m_d3d9Options.shaderModel);
|
||||||
|
|
||||||
if (unlikely(majorVersion > shaderModelVS
|
if (unlikely(majorVersion > shaderModelVS
|
||||||
|| (majorVersion == 1 && minorVersion > 1)
|
|| (majorVersion == 1 && minorVersion > 1)
|
||||||
|
@ -3352,6 +3352,16 @@ namespace dxvk {
|
||||||
&moduleInfo)))
|
&moduleInfo)))
|
||||||
return D3DERR_INVALIDCALL;
|
return D3DERR_INVALIDCALL;
|
||||||
|
|
||||||
|
|
||||||
|
if (m_isD3D8Compatible && !m_isSWVP) {
|
||||||
|
const uint32_t maxVSConstantIndex = module.GetMaxDefinedConstant();
|
||||||
|
// D3D8 enforces the value advertised in pCaps->MaxVertexShaderConst for HWVP
|
||||||
|
if (unlikely(maxVSConstantIndex > caps::MaxFloatConstantsVS - 1)) {
|
||||||
|
Logger::err(str::format("D3D9DeviceEx::CreateVertexShader: Invalid constant index ", maxVSConstantIndex));
|
||||||
|
return D3DERR_INVALIDCALL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
*ppShader = ref(new D3D9VertexShader(this,
|
*ppShader = ref(new D3D9VertexShader(this,
|
||||||
&m_shaderAllocator,
|
&m_shaderAllocator,
|
||||||
module,
|
module,
|
||||||
|
@ -3706,7 +3716,9 @@ namespace dxvk {
|
||||||
const uint32_t majorVersion = D3DSHADER_VERSION_MAJOR(pFunction[0]);
|
const uint32_t majorVersion = D3DSHADER_VERSION_MAJOR(pFunction[0]);
|
||||||
const uint32_t minorVersion = D3DSHADER_VERSION_MINOR(pFunction[0]);
|
const uint32_t minorVersion = D3DSHADER_VERSION_MINOR(pFunction[0]);
|
||||||
|
|
||||||
if (unlikely(majorVersion > m_d3d9Options.shaderModel
|
const uint32_t shaderModelPS = m_isD3D8Compatible ? std::min(1u, m_d3d9Options.shaderModel) : m_d3d9Options.shaderModel;
|
||||||
|
|
||||||
|
if (unlikely(majorVersion > shaderModelPS
|
||||||
|| (majorVersion == 1 && minorVersion > 4)
|
|| (majorVersion == 1 && minorVersion > 4)
|
||||||
// Skip checking the SM2 minor version, as it has a 2_x mode apparently
|
// Skip checking the SM2 minor version, as it has a 2_x mode apparently
|
||||||
|| (majorVersion == 3 && minorVersion != 0))) {
|
|| (majorVersion == 3 && minorVersion != 0))) {
|
||||||
|
|
|
@ -1160,7 +1160,6 @@ namespace dxvk {
|
||||||
case (DXVK_TSS_TCI_CAMERASPACEPOSITION >> TCIOffset):
|
case (DXVK_TSS_TCI_CAMERASPACEPOSITION >> TCIOffset):
|
||||||
transformed = vtx;
|
transformed = vtx;
|
||||||
if (!applyTransform) {
|
if (!applyTransform) {
|
||||||
Logger::warn(str::format("!applyTransform flags: ", flags, " projidx: ", projIndex));
|
|
||||||
count = 3;
|
count = 3;
|
||||||
projIndex = 4;
|
projIndex = 4;
|
||||||
}
|
}
|
||||||
|
|
|
@ -137,11 +137,9 @@ namespace dxvk {
|
||||||
return m_isD3D8Compatible;
|
return m_isD3D8Compatible;
|
||||||
}
|
}
|
||||||
|
|
||||||
void SetD3D8CompatibilityMode(bool compatMode) {
|
void EnableD3D8CompatibilityMode() {
|
||||||
if (compatMode)
|
m_isD3D8Compatible = true;
|
||||||
Logger::info("The D3D9 interface is now operating in D3D8 compatibility mode.");
|
Logger::info("The D3D9 interface is now operating in D3D8 compatibility mode.");
|
||||||
|
|
||||||
m_isD3D8Compatible = compatMode;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Rc<DxvkInstance> GetInstance() { return m_instance; }
|
Rc<DxvkInstance> GetInstance() { return m_instance; }
|
||||||
|
|
|
@ -44,7 +44,7 @@ namespace dxvk {
|
||||||
this->maxFrameLatency = config.getOption<int32_t> ("d3d9.maxFrameLatency", 0);
|
this->maxFrameLatency = config.getOption<int32_t> ("d3d9.maxFrameLatency", 0);
|
||||||
this->maxFrameRate = config.getOption<int32_t> ("d3d9.maxFrameRate", 0);
|
this->maxFrameRate = config.getOption<int32_t> ("d3d9.maxFrameRate", 0);
|
||||||
this->presentInterval = config.getOption<int32_t> ("d3d9.presentInterval", -1);
|
this->presentInterval = config.getOption<int32_t> ("d3d9.presentInterval", -1);
|
||||||
this->shaderModel = config.getOption<int32_t> ("d3d9.shaderModel", 3);
|
this->shaderModel = config.getOption<int32_t> ("d3d9.shaderModel", 3u);
|
||||||
this->dpiAware = config.getOption<bool> ("d3d9.dpiAware", true);
|
this->dpiAware = config.getOption<bool> ("d3d9.dpiAware", true);
|
||||||
this->strictConstantCopies = config.getOption<bool> ("d3d9.strictConstantCopies", false);
|
this->strictConstantCopies = config.getOption<bool> ("d3d9.strictConstantCopies", false);
|
||||||
this->strictPow = config.getOption<bool> ("d3d9.strictPow", true);
|
this->strictPow = config.getOption<bool> ("d3d9.strictPow", true);
|
||||||
|
|
|
@ -148,9 +148,9 @@ namespace dxvk {
|
||||||
bool isBlockAlignedFormat = blockSize.Width > 0 && blockSize.Height > 0;
|
bool isBlockAlignedFormat = blockSize.Width > 0 && blockSize.Height > 0;
|
||||||
|
|
||||||
// The boundaries of pRect are validated for D3DPOOL_DEFAULT surfaces
|
// The boundaries of pRect are validated for D3DPOOL_DEFAULT surfaces
|
||||||
// with formats which need to be block aligned (mip 0), surfaces created via
|
// with formats which need to be block aligned, surfaces created via
|
||||||
// CreateImageSurface and D3D8 cube textures outside of D3DPOOL_DEFAULT
|
// CreateImageSurface and D3D8 cube textures outside of D3DPOOL_DEFAULT
|
||||||
if ((m_mipLevel == 0 && isBlockAlignedFormat && desc.Pool == D3DPOOL_DEFAULT)
|
if ((isBlockAlignedFormat && desc.Pool == D3DPOOL_DEFAULT)
|
||||||
|| (desc.Pool == D3DPOOL_SYSTEMMEM && type == D3DRTYPE_SURFACE)
|
|| (desc.Pool == D3DPOOL_SYSTEMMEM && type == D3DRTYPE_SURFACE)
|
||||||
|| (m_texture->Device()->IsD3D8Compatible() &&
|
|| (m_texture->Device()->IsD3D8Compatible() &&
|
||||||
desc.Pool != D3DPOOL_DEFAULT && type == D3DRTYPE_CUBETEXTURE)) {
|
desc.Pool != D3DPOOL_DEFAULT && type == D3DRTYPE_CUBETEXTURE)) {
|
||||||
|
@ -161,8 +161,8 @@ namespace dxvk {
|
||||||
|| pRect->right - pRect->left <= 0
|
|| pRect->right - pRect->left <= 0
|
||||||
|| pRect->bottom - pRect->top <= 0
|
|| pRect->bottom - pRect->top <= 0
|
||||||
// Exceeding surface dimensions
|
// Exceeding surface dimensions
|
||||||
|| static_cast<UINT>(pRect->right) > desc.Width
|
|| static_cast<UINT>(pRect->right) > std::max(1u, desc.Width >> m_mipLevel)
|
||||||
|| static_cast<UINT>(pRect->bottom) > desc.Height)
|
|| static_cast<UINT>(pRect->bottom) > std::max(1u, desc.Height >> m_mipLevel))
|
||||||
return D3DERR_INVALIDCALL;
|
return D3DERR_INVALIDCALL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -119,9 +119,9 @@ namespace dxvk {
|
||||||
|| static_cast<LONG>(pBox->Bottom) - static_cast<LONG>(pBox->Top) <= 0
|
|| static_cast<LONG>(pBox->Bottom) - static_cast<LONG>(pBox->Top) <= 0
|
||||||
|| static_cast<LONG>(pBox->Back) - static_cast<LONG>(pBox->Front) <= 0
|
|| static_cast<LONG>(pBox->Back) - static_cast<LONG>(pBox->Front) <= 0
|
||||||
// Exceeding surface dimensions
|
// Exceeding surface dimensions
|
||||||
|| pBox->Right > desc.Width
|
|| pBox->Right > std::max(1u, desc.Width >> m_mipLevel)
|
||||||
|| pBox->Bottom > desc.Height
|
|| pBox->Bottom > std::max(1u, desc.Height >> m_mipLevel)
|
||||||
|| pBox->Back > desc.Depth)
|
|| pBox->Back > std::max(1u, desc.Depth >> m_mipLevel))
|
||||||
return D3DERR_INVALIDCALL;
|
return D3DERR_INVALIDCALL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -481,7 +481,7 @@ namespace dxvk {
|
||||||
std::ifstream ifile = openCacheFileForRead();
|
std::ifstream ifile = openCacheFileForRead();
|
||||||
|
|
||||||
if (!ifile) {
|
if (!ifile) {
|
||||||
Logger::warn("DXVK: No state cache file found");
|
Logger::debug("DXVK: No state cache file found");
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -504,7 +504,7 @@ namespace dxvk {
|
||||||
|
|
||||||
// Notify user about format conversion
|
// Notify user about format conversion
|
||||||
if (curHeader.version != newHeader.version)
|
if (curHeader.version != newHeader.version)
|
||||||
Logger::warn(str::format("DXVK: Updating state cache version to v", newHeader.version));
|
Logger::info(str::format("DXVK: Updating state cache version to v", newHeader.version));
|
||||||
|
|
||||||
// Read actual cache entries from the file.
|
// Read actual cache entries from the file.
|
||||||
// If we encounter invalid entries, we should
|
// If we encounter invalid entries, we should
|
||||||
|
@ -880,7 +880,7 @@ namespace dxvk {
|
||||||
return file;
|
return file;
|
||||||
|
|
||||||
if (recreate) {
|
if (recreate) {
|
||||||
Logger::warn("DXVK: Creating new state cache file");
|
Logger::info("DXVK: Creating new state cache file");
|
||||||
|
|
||||||
// Write header with the current version number
|
// Write header with the current version number
|
||||||
DxvkStateCacheHeader header;
|
DxvkStateCacheHeader header;
|
||||||
|
|
|
@ -703,7 +703,7 @@ namespace dxvk {
|
||||||
{ "d3d9.maxFrameRate", "60" },
|
{ "d3d9.maxFrameRate", "60" },
|
||||||
}} },
|
}} },
|
||||||
/* Escape from Tarkov launcher
|
/* Escape from Tarkov launcher
|
||||||
Same issue as Warhammer: RoR above */
|
Work around partial presentation issues */
|
||||||
{ R"(\\BsgLauncher\.exe$)", {{
|
{ R"(\\BsgLauncher\.exe$)", {{
|
||||||
{ "d3d9.shaderModel", "1" },
|
{ "d3d9.shaderModel", "1" },
|
||||||
}} },
|
}} },
|
||||||
|
|
Loading…
Add table
Reference in a new issue