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
|
||||
#
|
||||
# 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:
|
||||
# - 0: Fixed-function only
|
||||
|
@ -763,3 +764,83 @@
|
|||
# - True/False
|
||||
|
||||
# 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))
|
||||
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) {
|
||||
const uint32_t majorVersion = D3DSHADER_VERSION_MAJOR(pFunction[0]);
|
||||
const uint32_t minorVersion = D3DSHADER_VERSION_MINOR(pFunction[0]);
|
||||
|
||||
if (unlikely(majorVersion != 1 || minorVersion > 1)) {
|
||||
Logger::err(str::format("D3D8Device::CreateVertexShader: Unsupported VS version ", majorVersion, ".", minorVersion));
|
||||
return D3DERR_INVALIDCALL;
|
||||
}
|
||||
res = GetD3D9()->CreateVertexShader(translatedVS.function.data(), &pVertexShader);
|
||||
} else {
|
||||
// pFunction is NULL: fixed function pipeline
|
||||
pVertexShader = nullptr;
|
||||
}
|
||||
|
||||
if (likely(SUCCEEDED(res))) {
|
||||
D3D8VertexShaderInfo& info = m_vertexShaders.emplace_back();
|
||||
|
||||
info.pVertexDecl = std::move(pVertexDecl);
|
||||
info.pVertexShader = std::move(pVertexShader);
|
||||
|
||||
// Store D3D8 bytecodes in the shader info
|
||||
for (UINT i = 0; pDeclaration[i] != D3DVSD_END(); i++)
|
||||
info.declaration.push_back(pDeclaration[i]);
|
||||
|
@ -1839,21 +1851,6 @@ namespace dxvk {
|
|||
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
|
||||
*pHandle = getShaderHandle(m_vertexShaders.size());
|
||||
}
|
||||
|
@ -2056,20 +2053,11 @@ namespace dxvk {
|
|||
if (unlikely(pFunction == nullptr || pHandle == nullptr))
|
||||
return D3DERR_INVALIDCALL;
|
||||
|
||||
const uint32_t majorVersion = D3DSHADER_VERSION_MAJOR(pFunction[0]);
|
||||
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;
|
||||
|
||||
Com<d3d9::IDirect3DPixelShader9> pPixelShader;
|
||||
HRESULT res = GetD3D9()->CreatePixelShader(pFunction, &pPixelShader);
|
||||
|
||||
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.
|
||||
*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!");
|
||||
}
|
||||
|
||||
m_bridge->SetD3D8CompatibilityMode(true);
|
||||
m_bridge->EnableD3D8CompatibilityMode();
|
||||
|
||||
m_d3d8Options = D3D8Options(*m_bridge->GetConfig());
|
||||
|
||||
|
|
|
@ -8,53 +8,29 @@ namespace dxvk {
|
|||
|
||||
struct D3D8Options {
|
||||
|
||||
/// Some games rely on undefined behavior by using undeclared vertex shader inputs.
|
||||
/// 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
|
||||
/// Override application vertex shader declarations.
|
||||
std::vector<std::pair<D3DVSDE_REGISTER, D3DVSDT_TYPE>> forceVsDecl;
|
||||
|
||||
/// 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 outside of specifc games that benefit from it.
|
||||
bool batching = false;
|
||||
/// Enable/disable the drawcall batcher.
|
||||
bool batching;
|
||||
|
||||
/// The Lord of the Rings: The Fellowship of the Ring tries to create a P8 texture
|
||||
/// in D3DPOOL_MANAGED on Nvidia and Intel, which fails, but has a separate code
|
||||
/// 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;
|
||||
/// Place all P8 textures in D3DPOOL_SCRATCH.
|
||||
bool placeP8InScratch;
|
||||
|
||||
/// Rayman 3 relies on D3DLOCK_DISCARD being ignored for everything except D3DUSAGE_DYNAMIC +
|
||||
/// D3DUSAGE_WRITEONLY buffers, however this approach incurs a performance penalty.
|
||||
///
|
||||
/// 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;
|
||||
/// Ignore D3DLOCK_DISCARD for everything except D3DUSAGE_DYNAMIC + D3DUSAGE_WRITEONLY buffers.
|
||||
bool forceLegacyDiscard;
|
||||
|
||||
/// Splinter Cell expects shadow map texture coordinates to be perspective divided
|
||||
/// even though D3DTTFF_PROJECTED is never set for any texture coordinates. This flag
|
||||
/// forces that flag for the necessary stages when a depth texture is bound to slot 0
|
||||
bool shadowPerspectiveDivide = false;
|
||||
/// Force D3DTTFF_PROJECTED for the necessary stages when a depth texture is bound to slot 0.
|
||||
bool shadowPerspectiveDivide;
|
||||
|
||||
D3D8Options() {}
|
||||
|
||||
D3D8Options(const Config& config) {
|
||||
auto forceVsDeclStr = config.getOption<std::string>("d3d8.forceVsDecl", "");
|
||||
batching = config.getOption<bool> ("d3d8.batching", batching);
|
||||
placeP8InScratch = config.getOption<bool> ("d3d8.placeP8InScratch", placeP8InScratch);
|
||||
forceLegacyDiscard = config.getOption<bool> ("d3d8.forceLegacyDiscard", forceLegacyDiscard);
|
||||
shadowPerspectiveDivide = config.getOption<bool> ("d3d8.shadowPerspectiveDivide", shadowPerspectiveDivide);
|
||||
batching = config.getOption<bool> ("d3d8.batching", false);
|
||||
placeP8InScratch = config.getOption<bool> ("d3d8.placeP8InScratch", false);
|
||||
forceLegacyDiscard = config.getOption<bool> ("d3d8.forceLegacyDiscard", false);
|
||||
shadowPerspectiveDivide = config.getOption<bool> ("d3d8.shadowPerspectiveDivide", false);
|
||||
|
||||
parseVsDecl(forceVsDeclStr);
|
||||
}
|
||||
|
|
|
@ -110,26 +110,27 @@ namespace dxvk {
|
|||
}
|
||||
|
||||
/**
|
||||
* Converts a D3D8 vertex shader + declaration
|
||||
* to a D3D9 vertex shader + declaration.
|
||||
* Validates and converts a D3D8 vertex shader
|
||||
* + declaration to a D3D9 vertex shader + declaration.
|
||||
*/
|
||||
D3D9VertexShaderCode TranslateVertexShader8(
|
||||
HRESULT TranslateVertexShader8(
|
||||
const DWORD* pDeclaration,
|
||||
const DWORD* pFunction,
|
||||
const D3D8Options& options) {
|
||||
const D3D8Options& options,
|
||||
D3D9VertexShaderCode* pTranslatedVS) {
|
||||
using d3d9::D3DDECLTYPE;
|
||||
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
|
||||
|
||||
// shaderInputRegisters:
|
||||
// set bit N to enable input register vN
|
||||
DWORD shaderInputRegisters = 0;
|
||||
|
||||
d3d9::D3DVERTEXELEMENT9* vertexElements = result.declaration;
|
||||
d3d9::D3DVERTEXELEMENT9* vertexElements = pTranslatedVS->declaration;
|
||||
unsigned int elementIdx = 0;
|
||||
|
||||
// These are used for pDeclaration and pFunction
|
||||
|
@ -206,6 +207,12 @@ namespace dxvk {
|
|||
D3DVSDT_TYPE type = D3DVSDT_TYPE(VSD_SHIFT_MASK(token, D3DVSD_DATATYPE));
|
||||
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);
|
||||
|
||||
dbg << "type=" << type << ", register=" << reg;
|
||||
|
@ -332,7 +339,7 @@ namespace dxvk {
|
|||
} while (token != D3DVS_END());
|
||||
}
|
||||
|
||||
return result;
|
||||
return res;
|
||||
}
|
||||
|
||||
}
|
||||
|
|
|
@ -10,9 +10,10 @@ namespace dxvk {
|
|||
std::vector<DWORD> function;
|
||||
};
|
||||
|
||||
D3D9VertexShaderCode TranslateVertexShader8(
|
||||
HRESULT TranslateVertexShader8(
|
||||
const DWORD* pDeclaration,
|
||||
const DWORD* pFunction,
|
||||
const D3D8Options& overrides);
|
||||
const D3D8Options& overrides,
|
||||
D3D9VertexShaderCode* pTranslatedVS);
|
||||
|
||||
}
|
|
@ -117,6 +117,9 @@ namespace dxvk {
|
|||
if (unlikely(AdapterFormat == D3D9Format::Unknown))
|
||||
return D3DERR_INVALIDCALL;
|
||||
|
||||
if (unlikely(RType == D3DRTYPE_VERTEXBUFFER || RType == D3DRTYPE_INDEXBUFFER))
|
||||
return D3DERR_INVALIDCALL;
|
||||
|
||||
if (!IsSupportedAdapterFormat(AdapterFormat))
|
||||
return D3DERR_NOTAVAILABLE;
|
||||
|
||||
|
@ -168,9 +171,6 @@ namespace dxvk {
|
|||
if (RType == D3DRTYPE_CUBETEXTURE && mapping.Aspect != VK_IMAGE_ASPECT_COLOR_BIT)
|
||||
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!
|
||||
VkFormat format = mapping.FormatColor;
|
||||
if (unlikely(mapping.ConversionFormatInfo.FormatColor != VK_FORMAT_UNDEFINED)) {
|
||||
|
@ -297,6 +297,7 @@ namespace dxvk {
|
|||
|
||||
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;
|
||||
|
||||
// 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,
|
||||
// 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 majorVersionPS = options.shaderModel;
|
||||
const uint32_t majorVersionVS = maxShaderModel == 0 ? 1 : maxShaderModel;
|
||||
const uint32_t majorVersionPS = maxShaderModel;
|
||||
// Max supported SM1 is VS 1.1 and PS 1.4
|
||||
const uint32_t minorVersionVS = majorVersionVS != 1 ? 0 : 1;
|
||||
const uint32_t minorVersionPS = majorVersionPS != 1 ? 0 : 4;
|
||||
|
@ -588,7 +589,7 @@ namespace dxvk {
|
|||
// Max Vertex Shader Const
|
||||
pCaps->MaxVertexShaderConst = MaxFloatConstantsVS;
|
||||
// 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
|
||||
pCaps->DevCaps2 = D3DDEVCAPS2_STREAMOFFSET
|
||||
/* | D3DDEVCAPS2_DMAPNPATCH */
|
||||
|
@ -635,23 +636,23 @@ namespace dxvk {
|
|||
/* | D3DPTFILTERCAPS_MAGFPYRAMIDALQUAD */
|
||||
/* | D3DPTFILTERCAPS_MAGFGAUSSIANQUAD */;
|
||||
|
||||
pCaps->VS20Caps.Caps = options.shaderModel >= 2 ? D3DVS20CAPS_PREDICATION : 0;
|
||||
pCaps->VS20Caps.DynamicFlowControlDepth = options.shaderModel >= 2 ? D3DVS20_MAX_DYNAMICFLOWCONTROLDEPTH : 0;
|
||||
pCaps->VS20Caps.NumTemps = options.shaderModel >= 2 ? D3DVS20_MAX_NUMTEMPS : 0;
|
||||
pCaps->VS20Caps.StaticFlowControlDepth = options.shaderModel >= 2 ? D3DVS20_MAX_STATICFLOWCONTROLDEPTH : 0;
|
||||
pCaps->VS20Caps.Caps = maxShaderModel >= 2 ? D3DVS20CAPS_PREDICATION : 0;
|
||||
pCaps->VS20Caps.DynamicFlowControlDepth = maxShaderModel >= 2 ? D3DVS20_MAX_DYNAMICFLOWCONTROLDEPTH : 0;
|
||||
pCaps->VS20Caps.NumTemps = maxShaderModel >= 2 ? D3DVS20_MAX_NUMTEMPS : 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_PREDICATION
|
||||
| D3DPS20CAPS_NODEPENDENTREADLIMIT
|
||||
| D3DPS20CAPS_NOTEXINSTRUCTIONLIMIT : 0;
|
||||
pCaps->PS20Caps.DynamicFlowControlDepth = options.shaderModel >= 2 ? D3DPS20_MAX_DYNAMICFLOWCONTROLDEPTH : 0;
|
||||
pCaps->PS20Caps.NumTemps = options.shaderModel >= 2 ? D3DPS20_MAX_NUMTEMPS : 0;
|
||||
pCaps->PS20Caps.StaticFlowControlDepth = options.shaderModel >= 2 ? D3DPS20_MAX_STATICFLOWCONTROLDEPTH : 0;
|
||||
pCaps->PS20Caps.NumInstructionSlots = options.shaderModel >= 2 ? D3DPS20_MAX_NUMINSTRUCTIONSLOTS : 0;
|
||||
pCaps->PS20Caps.DynamicFlowControlDepth = maxShaderModel >= 2 ? D3DPS20_MAX_DYNAMICFLOWCONTROLDEPTH : 0;
|
||||
pCaps->PS20Caps.NumTemps = maxShaderModel >= 2 ? D3DPS20_MAX_NUMTEMPS : 0;
|
||||
pCaps->PS20Caps.StaticFlowControlDepth = maxShaderModel >= 2 ? D3DPS20_MAX_STATICFLOWCONTROLDEPTH : 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.
|
||||
pCaps->VertexTextureFilterCaps = options.shaderModel == 3 ? D3DPTFILTERCAPS_MINFPOINT
|
||||
pCaps->VertexTextureFilterCaps = maxShaderModel == 3 ? D3DPTFILTERCAPS_MINFPOINT
|
||||
| D3DPTFILTERCAPS_MINFLINEAR
|
||||
/* | D3DPTFILTERCAPS_MINFANISOTROPIC */
|
||||
/* | D3DPTFILTERCAPS_MINFPYRAMIDALQUAD */
|
||||
|
@ -665,11 +666,11 @@ namespace dxvk {
|
|||
/* | D3DPTFILTERCAPS_MAGFPYRAMIDALQUAD */
|
||||
/* | D3DPTFILTERCAPS_MAGFGAUSSIANQUAD */ : 0;
|
||||
|
||||
pCaps->MaxVShaderInstructionsExecuted = options.shaderModel >= 2 ? 4294967295 : 0;
|
||||
pCaps->MaxPShaderInstructionsExecuted = options.shaderModel >= 2 ? 4294967295 : 0;
|
||||
pCaps->MaxVShaderInstructionsExecuted = maxShaderModel >= 2 ? 4294967295 : 0;
|
||||
pCaps->MaxPShaderInstructionsExecuted = maxShaderModel >= 2 ? 4294967295 : 0;
|
||||
|
||||
pCaps->MaxVertexShader30InstructionSlots = options.shaderModel == 3 ? 32768 : 0;
|
||||
pCaps->MaxPixelShader30InstructionSlots = options.shaderModel == 3 ? 32768 : 0;
|
||||
pCaps->MaxVertexShader30InstructionSlots = maxShaderModel == 3 ? 32768 : 0;
|
||||
pCaps->MaxPixelShader30InstructionSlots = maxShaderModel == 3 ? 32768 : 0;
|
||||
|
||||
return D3D_OK;
|
||||
}
|
||||
|
|
|
@ -114,8 +114,8 @@ namespace dxvk {
|
|||
return m_interface->QueryInterface(riid, ppvObject);
|
||||
}
|
||||
|
||||
void DxvkD3D8InterfaceBridge::SetD3D8CompatibilityMode(const bool compatMode) {
|
||||
m_interface->SetD3D8CompatibilityMode(compatMode);
|
||||
void DxvkD3D8InterfaceBridge::EnableD3D8CompatibilityMode() {
|
||||
m_interface->EnableD3D8CompatibilityMode();
|
||||
}
|
||||
|
||||
const Config* DxvkD3D8InterfaceBridge::GetConfig() const {
|
||||
|
|
|
@ -42,11 +42,9 @@ IDxvkD3D8Bridge : public IUnknown {
|
|||
MIDL_INTERFACE("D3D9D3D8-A407-773E-18E9-CAFEBEEF3000")
|
||||
IDxvkD3D8InterfaceBridge : public IUnknown {
|
||||
/**
|
||||
* \brief Enables or disables D3D9-specific features and validations
|
||||
*
|
||||
* \param [in] compatMode Compatibility state
|
||||
* \brief Enforces D3D8-specific features and validations
|
||||
*/
|
||||
virtual void SetD3D8CompatibilityMode(const bool compatMode) = 0;
|
||||
virtual void EnableD3D8CompatibilityMode() = 0;
|
||||
|
||||
/**
|
||||
* \brief Retrieves the DXVK configuration
|
||||
|
@ -106,7 +104,7 @@ namespace dxvk {
|
|||
REFIID riid,
|
||||
void** ppvObject);
|
||||
|
||||
void SetD3D8CompatibilityMode(const bool compatMode);
|
||||
void EnableD3D8CompatibilityMode();
|
||||
|
||||
const Config* GetConfig() const;
|
||||
|
||||
|
|
|
@ -3329,7 +3329,7 @@ namespace dxvk {
|
|||
const uint32_t minorVersion = D3DSHADER_VERSION_MINOR(pFunction[0]);
|
||||
|
||||
// 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
|
||||
|| (majorVersion == 1 && minorVersion > 1)
|
||||
|
@ -3352,6 +3352,16 @@ namespace dxvk {
|
|||
&moduleInfo)))
|
||||
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,
|
||||
&m_shaderAllocator,
|
||||
module,
|
||||
|
@ -3706,7 +3716,9 @@ namespace dxvk {
|
|||
const uint32_t majorVersion = D3DSHADER_VERSION_MAJOR(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)
|
||||
// Skip checking the SM2 minor version, as it has a 2_x mode apparently
|
||||
|| (majorVersion == 3 && minorVersion != 0))) {
|
||||
|
|
|
@ -1160,7 +1160,6 @@ namespace dxvk {
|
|||
case (DXVK_TSS_TCI_CAMERASPACEPOSITION >> TCIOffset):
|
||||
transformed = vtx;
|
||||
if (!applyTransform) {
|
||||
Logger::warn(str::format("!applyTransform flags: ", flags, " projidx: ", projIndex));
|
||||
count = 3;
|
||||
projIndex = 4;
|
||||
}
|
||||
|
|
|
@ -137,11 +137,9 @@ namespace dxvk {
|
|||
return m_isD3D8Compatible;
|
||||
}
|
||||
|
||||
void SetD3D8CompatibilityMode(bool compatMode) {
|
||||
if (compatMode)
|
||||
void EnableD3D8CompatibilityMode() {
|
||||
m_isD3D8Compatible = true;
|
||||
Logger::info("The D3D9 interface is now operating in D3D8 compatibility mode.");
|
||||
|
||||
m_isD3D8Compatible = compatMode;
|
||||
}
|
||||
|
||||
Rc<DxvkInstance> GetInstance() { return m_instance; }
|
||||
|
|
|
@ -44,7 +44,7 @@ namespace dxvk {
|
|||
this->maxFrameLatency = config.getOption<int32_t> ("d3d9.maxFrameLatency", 0);
|
||||
this->maxFrameRate = config.getOption<int32_t> ("d3d9.maxFrameRate", 0);
|
||||
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->strictConstantCopies = config.getOption<bool> ("d3d9.strictConstantCopies", false);
|
||||
this->strictPow = config.getOption<bool> ("d3d9.strictPow", true);
|
||||
|
|
|
@ -148,9 +148,9 @@ namespace dxvk {
|
|||
bool isBlockAlignedFormat = blockSize.Width > 0 && blockSize.Height > 0;
|
||||
|
||||
// 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
|
||||
if ((m_mipLevel == 0 && isBlockAlignedFormat && desc.Pool == D3DPOOL_DEFAULT)
|
||||
if ((isBlockAlignedFormat && desc.Pool == D3DPOOL_DEFAULT)
|
||||
|| (desc.Pool == D3DPOOL_SYSTEMMEM && type == D3DRTYPE_SURFACE)
|
||||
|| (m_texture->Device()->IsD3D8Compatible() &&
|
||||
desc.Pool != D3DPOOL_DEFAULT && type == D3DRTYPE_CUBETEXTURE)) {
|
||||
|
@ -161,8 +161,8 @@ namespace dxvk {
|
|||
|| pRect->right - pRect->left <= 0
|
||||
|| pRect->bottom - pRect->top <= 0
|
||||
// Exceeding surface dimensions
|
||||
|| static_cast<UINT>(pRect->right) > desc.Width
|
||||
|| static_cast<UINT>(pRect->bottom) > desc.Height)
|
||||
|| static_cast<UINT>(pRect->right) > std::max(1u, desc.Width >> m_mipLevel)
|
||||
|| static_cast<UINT>(pRect->bottom) > std::max(1u, desc.Height >> m_mipLevel))
|
||||
return D3DERR_INVALIDCALL;
|
||||
}
|
||||
|
||||
|
|
|
@ -119,9 +119,9 @@ namespace dxvk {
|
|||
|| static_cast<LONG>(pBox->Bottom) - static_cast<LONG>(pBox->Top) <= 0
|
||||
|| static_cast<LONG>(pBox->Back) - static_cast<LONG>(pBox->Front) <= 0
|
||||
// Exceeding surface dimensions
|
||||
|| pBox->Right > desc.Width
|
||||
|| pBox->Bottom > desc.Height
|
||||
|| pBox->Back > desc.Depth)
|
||||
|| pBox->Right > std::max(1u, desc.Width >> m_mipLevel)
|
||||
|| pBox->Bottom > std::max(1u, desc.Height >> m_mipLevel)
|
||||
|| pBox->Back > std::max(1u, desc.Depth >> m_mipLevel))
|
||||
return D3DERR_INVALIDCALL;
|
||||
}
|
||||
|
||||
|
|
|
@ -481,7 +481,7 @@ namespace dxvk {
|
|||
std::ifstream ifile = openCacheFileForRead();
|
||||
|
||||
if (!ifile) {
|
||||
Logger::warn("DXVK: No state cache file found");
|
||||
Logger::debug("DXVK: No state cache file found");
|
||||
return true;
|
||||
}
|
||||
|
||||
|
@ -504,7 +504,7 @@ namespace dxvk {
|
|||
|
||||
// Notify user about format conversion
|
||||
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.
|
||||
// If we encounter invalid entries, we should
|
||||
|
@ -880,7 +880,7 @@ namespace dxvk {
|
|||
return file;
|
||||
|
||||
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
|
||||
DxvkStateCacheHeader header;
|
||||
|
|
|
@ -703,7 +703,7 @@ namespace dxvk {
|
|||
{ "d3d9.maxFrameRate", "60" },
|
||||
}} },
|
||||
/* Escape from Tarkov launcher
|
||||
Same issue as Warhammer: RoR above */
|
||||
Work around partial presentation issues */
|
||||
{ R"(\\BsgLauncher\.exe$)", {{
|
||||
{ "d3d9.shaderModel", "1" },
|
||||
}} },
|
||||
|
|
Loading…
Add table
Reference in a new issue