mirror of
https://github.com/doitsujin/dxvk.git
synced 2025-01-23 17:52:10 +01:00
423 lines
14 KiB
C++
423 lines
14 KiB
C++
#include "d3d9_util.h"
|
|
|
|
#include "../util/util_win32_compat.h"
|
|
|
|
namespace dxvk {
|
|
|
|
typedef HRESULT (STDMETHODCALLTYPE *D3DXDisassembleShader) (
|
|
const void* pShader,
|
|
BOOL EnableColorCode,
|
|
char* pComments,
|
|
ID3DBlob** ppDisassembly); // ppDisassembly is actually a D3DXBUFFER, but it has the exact same vtable as a ID3DBlob at the start.
|
|
|
|
D3DXDisassembleShader g_pfnDisassembleShader = nullptr;
|
|
|
|
HRESULT DisassembleShader(
|
|
const void* pShader,
|
|
BOOL EnableColorCode,
|
|
char* pComments,
|
|
ID3DBlob** ppDisassembly) {
|
|
if (g_pfnDisassembleShader == nullptr) {
|
|
HMODULE d3d9x = LoadLibraryA("d3dx9.dll");
|
|
|
|
if (d3d9x == nullptr)
|
|
d3d9x = LoadLibraryA("d3dx9_43.dll");
|
|
|
|
g_pfnDisassembleShader =
|
|
reinterpret_cast<D3DXDisassembleShader>(GetProcAddress(d3d9x, "D3DXDisassembleShader"));
|
|
}
|
|
|
|
if (g_pfnDisassembleShader == nullptr)
|
|
return D3DERR_INVALIDCALL;
|
|
|
|
return g_pfnDisassembleShader(
|
|
pShader,
|
|
EnableColorCode,
|
|
pComments,
|
|
ppDisassembly);
|
|
}
|
|
|
|
|
|
HRESULT DecodeMultiSampleType(
|
|
const Rc<DxvkDevice>& pDevice,
|
|
D3DMULTISAMPLE_TYPE MultiSample,
|
|
DWORD MultisampleQuality,
|
|
VkSampleCountFlagBits* pSampleCount) {
|
|
uint32_t sampleCount = std::max<uint32_t>(MultiSample, 1u);
|
|
|
|
// Check if this is a power of two...
|
|
if (sampleCount & (sampleCount - 1))
|
|
return D3DERR_INVALIDCALL;
|
|
|
|
if (MultiSample == D3DMULTISAMPLE_NONMASKABLE)
|
|
sampleCount = 1u << MultisampleQuality;
|
|
|
|
const auto& limits = pDevice->properties().core.properties.limits;
|
|
VkSampleCountFlags supportedSampleCounts = limits.framebufferColorSampleCounts & limits.framebufferDepthSampleCounts;
|
|
|
|
while (sampleCount > supportedSampleCounts)
|
|
sampleCount >>= 1;
|
|
|
|
if (pSampleCount)
|
|
*pSampleCount = VkSampleCountFlagBits(sampleCount);
|
|
|
|
return D3D_OK;
|
|
}
|
|
|
|
|
|
VkFormat GetPackedDepthStencilFormat(D3D9Format Format) {
|
|
switch (Format) {
|
|
case D3D9Format::D15S1:
|
|
return VK_FORMAT_D16_UNORM_S8_UINT; // This should never happen!
|
|
|
|
case D3D9Format::D16:
|
|
case D3D9Format::D16_LOCKABLE:
|
|
case D3D9Format::DF16:
|
|
return VK_FORMAT_D16_UNORM;
|
|
|
|
case D3D9Format::D24X8:
|
|
case D3D9Format::DF24:
|
|
return VK_FORMAT_X8_D24_UNORM_PACK32;
|
|
|
|
case D3D9Format::D24X4S4:
|
|
case D3D9Format::D24FS8:
|
|
case D3D9Format::D24S8:
|
|
case D3D9Format::INTZ:
|
|
return VK_FORMAT_D24_UNORM_S8_UINT;
|
|
|
|
case D3D9Format::D32:
|
|
case D3D9Format::D32_LOCKABLE:
|
|
case D3D9Format::D32F_LOCKABLE:
|
|
return VK_FORMAT_D32_SFLOAT;
|
|
|
|
case D3D9Format::S8_LOCKABLE:
|
|
return VK_FORMAT_S8_UINT;
|
|
|
|
default:
|
|
return VK_FORMAT_UNDEFINED;
|
|
}
|
|
}
|
|
|
|
|
|
VkFormatFeatureFlags2 GetImageFormatFeatures(DWORD Usage) {
|
|
VkFormatFeatureFlags2 features = VK_FORMAT_FEATURE_2_SAMPLED_IMAGE_BIT;
|
|
|
|
if (Usage & D3DUSAGE_DEPTHSTENCIL)
|
|
features |= VK_FORMAT_FEATURE_2_DEPTH_STENCIL_ATTACHMENT_BIT;
|
|
|
|
if (Usage & D3DUSAGE_RENDERTARGET)
|
|
features |= VK_FORMAT_FEATURE_2_COLOR_ATTACHMENT_BIT;
|
|
|
|
return features;
|
|
}
|
|
|
|
|
|
VkImageUsageFlags GetImageUsageFlags(DWORD Usage) {
|
|
VkImageUsageFlags usage = VK_IMAGE_USAGE_SAMPLED_BIT;
|
|
|
|
if (Usage & D3DUSAGE_DEPTHSTENCIL)
|
|
usage |= VK_IMAGE_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT;
|
|
|
|
if (Usage & D3DUSAGE_RENDERTARGET)
|
|
usage |= VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT;
|
|
|
|
return usage;
|
|
}
|
|
|
|
|
|
uint32_t GetVertexCount(D3DPRIMITIVETYPE type, UINT count) {
|
|
switch (type) {
|
|
default:
|
|
case D3DPT_TRIANGLELIST: return count * 3;
|
|
case D3DPT_POINTLIST: return count;
|
|
case D3DPT_LINELIST: return count * 2;
|
|
case D3DPT_LINESTRIP: return count + 1;
|
|
case D3DPT_TRIANGLESTRIP: return count + 2;
|
|
case D3DPT_TRIANGLEFAN: return count + 2;
|
|
}
|
|
}
|
|
|
|
|
|
DxvkInputAssemblyState DecodeInputAssemblyState(D3DPRIMITIVETYPE type) {
|
|
switch (type) {
|
|
default:
|
|
case D3DPT_TRIANGLELIST:
|
|
return { VK_PRIMITIVE_TOPOLOGY_TRIANGLE_LIST, VK_FALSE, 0 };
|
|
|
|
case D3DPT_POINTLIST:
|
|
return { VK_PRIMITIVE_TOPOLOGY_POINT_LIST, VK_FALSE, 0 };
|
|
|
|
case D3DPT_LINELIST:
|
|
return { VK_PRIMITIVE_TOPOLOGY_LINE_LIST, VK_FALSE, 0 };
|
|
|
|
case D3DPT_LINESTRIP:
|
|
return { VK_PRIMITIVE_TOPOLOGY_LINE_STRIP, VK_FALSE, 0 };
|
|
|
|
case D3DPT_TRIANGLESTRIP:
|
|
return { VK_PRIMITIVE_TOPOLOGY_TRIANGLE_STRIP, VK_FALSE, 0 };
|
|
|
|
case D3DPT_TRIANGLEFAN:
|
|
return { VK_PRIMITIVE_TOPOLOGY_TRIANGLE_FAN, VK_FALSE, 0 };
|
|
}
|
|
}
|
|
|
|
|
|
VkBlendFactor DecodeBlendFactor(D3DBLEND BlendFactor, bool IsAlpha) {
|
|
switch (BlendFactor) {
|
|
default:
|
|
case D3DBLEND_ZERO: return VK_BLEND_FACTOR_ZERO;
|
|
case D3DBLEND_ONE: return VK_BLEND_FACTOR_ONE;
|
|
case D3DBLEND_SRCCOLOR: return VK_BLEND_FACTOR_SRC_COLOR;
|
|
case D3DBLEND_INVSRCCOLOR: return VK_BLEND_FACTOR_ONE_MINUS_SRC_COLOR;
|
|
case D3DBLEND_SRCALPHA: return VK_BLEND_FACTOR_SRC_ALPHA;
|
|
case D3DBLEND_INVSRCALPHA: return VK_BLEND_FACTOR_ONE_MINUS_SRC_ALPHA;
|
|
case D3DBLEND_DESTALPHA: return VK_BLEND_FACTOR_DST_ALPHA;
|
|
case D3DBLEND_INVDESTALPHA: return VK_BLEND_FACTOR_ONE_MINUS_DST_ALPHA;
|
|
case D3DBLEND_DESTCOLOR: return VK_BLEND_FACTOR_DST_COLOR;
|
|
case D3DBLEND_INVDESTCOLOR: return VK_BLEND_FACTOR_ONE_MINUS_DST_COLOR;
|
|
case D3DBLEND_SRCALPHASAT: return VK_BLEND_FACTOR_SRC_ALPHA_SATURATE;
|
|
case D3DBLEND_BOTHSRCALPHA: return VK_BLEND_FACTOR_SRC_ALPHA;
|
|
case D3DBLEND_BOTHINVSRCALPHA: return VK_BLEND_FACTOR_ONE_MINUS_SRC_ALPHA;
|
|
case D3DBLEND_BLENDFACTOR: return IsAlpha ? VK_BLEND_FACTOR_CONSTANT_ALPHA : VK_BLEND_FACTOR_CONSTANT_COLOR;
|
|
case D3DBLEND_INVBLENDFACTOR: return IsAlpha ? VK_BLEND_FACTOR_ONE_MINUS_CONSTANT_ALPHA : VK_BLEND_FACTOR_ONE_MINUS_CONSTANT_COLOR;
|
|
case D3DBLEND_SRCCOLOR2: return VK_BLEND_FACTOR_SRC1_COLOR;
|
|
case D3DBLEND_INVSRCCOLOR2: return VK_BLEND_FACTOR_ONE_MINUS_SRC1_COLOR;
|
|
}
|
|
}
|
|
|
|
|
|
VkBlendOp DecodeBlendOp(D3DBLENDOP BlendOp) {
|
|
switch (BlendOp) {
|
|
default:
|
|
case D3DBLENDOP_ADD: return VK_BLEND_OP_ADD;
|
|
case D3DBLENDOP_SUBTRACT: return VK_BLEND_OP_SUBTRACT;
|
|
case D3DBLENDOP_REVSUBTRACT: return VK_BLEND_OP_REVERSE_SUBTRACT;
|
|
case D3DBLENDOP_MIN: return VK_BLEND_OP_MIN;
|
|
case D3DBLENDOP_MAX: return VK_BLEND_OP_MAX;
|
|
}
|
|
}
|
|
|
|
|
|
VkFilter DecodeFilter(D3DTEXTUREFILTERTYPE Filter) {
|
|
switch (Filter) {
|
|
case D3DTEXF_NONE:
|
|
case D3DTEXF_POINT:
|
|
return VK_FILTER_NEAREST;
|
|
default:
|
|
return VK_FILTER_LINEAR;
|
|
}
|
|
}
|
|
|
|
|
|
D3D9MipFilter DecodeMipFilter(D3DTEXTUREFILTERTYPE Filter) {
|
|
D3D9MipFilter filter;
|
|
filter.MipsEnabled = Filter != D3DTEXF_NONE;
|
|
|
|
switch (Filter) {
|
|
case D3DTEXF_POINT:
|
|
case D3DTEXF_NONE:
|
|
filter.MipFilter = VK_SAMPLER_MIPMAP_MODE_NEAREST; break;
|
|
default:
|
|
filter.MipFilter = VK_SAMPLER_MIPMAP_MODE_LINEAR; break;
|
|
}
|
|
|
|
return filter;
|
|
}
|
|
|
|
|
|
bool IsAnisotropic(D3DTEXTUREFILTERTYPE Filter) {
|
|
return Filter == D3DTEXF_ANISOTROPIC;
|
|
}
|
|
|
|
|
|
VkSamplerAddressMode DecodeAddressMode(D3DTEXTUREADDRESS Mode) {
|
|
switch (Mode) {
|
|
default:
|
|
case D3DTADDRESS_WRAP:
|
|
return VK_SAMPLER_ADDRESS_MODE_REPEAT;
|
|
case D3DTADDRESS_MIRROR:
|
|
return VK_SAMPLER_ADDRESS_MODE_MIRRORED_REPEAT;
|
|
case D3DTADDRESS_CLAMP:
|
|
return VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE;
|
|
case D3DTADDRESS_BORDER:
|
|
return VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_BORDER;
|
|
case D3DTADDRESS_MIRRORONCE:
|
|
return VK_SAMPLER_ADDRESS_MODE_MIRROR_CLAMP_TO_EDGE;
|
|
}
|
|
}
|
|
|
|
|
|
VkCompareOp DecodeCompareOp(D3DCMPFUNC Func) {
|
|
switch (Func) {
|
|
default:
|
|
case D3DCMP_NEVER: return VK_COMPARE_OP_NEVER;
|
|
case D3DCMP_LESS: return VK_COMPARE_OP_LESS;
|
|
case D3DCMP_EQUAL: return VK_COMPARE_OP_EQUAL;
|
|
case D3DCMP_LESSEQUAL: return VK_COMPARE_OP_LESS_OR_EQUAL;
|
|
case D3DCMP_GREATER: return VK_COMPARE_OP_GREATER;
|
|
case D3DCMP_NOTEQUAL: return VK_COMPARE_OP_NOT_EQUAL;
|
|
case D3DCMP_GREATEREQUAL: return VK_COMPARE_OP_GREATER_OR_EQUAL;
|
|
case D3DCMP_ALWAYS: return VK_COMPARE_OP_ALWAYS;
|
|
}
|
|
}
|
|
|
|
|
|
VkStencilOp DecodeStencilOp(D3DSTENCILOP Op) {
|
|
switch (Op) {
|
|
default:
|
|
case D3DSTENCILOP_KEEP: return VK_STENCIL_OP_KEEP;
|
|
case D3DSTENCILOP_ZERO: return VK_STENCIL_OP_ZERO;
|
|
case D3DSTENCILOP_REPLACE: return VK_STENCIL_OP_REPLACE;
|
|
case D3DSTENCILOP_INCRSAT: return VK_STENCIL_OP_INCREMENT_AND_CLAMP;
|
|
case D3DSTENCILOP_DECRSAT: return VK_STENCIL_OP_DECREMENT_AND_CLAMP;
|
|
case D3DSTENCILOP_INVERT: return VK_STENCIL_OP_INVERT;
|
|
case D3DSTENCILOP_INCR: return VK_STENCIL_OP_INCREMENT_AND_WRAP;
|
|
case D3DSTENCILOP_DECR: return VK_STENCIL_OP_DECREMENT_AND_WRAP;
|
|
}
|
|
}
|
|
|
|
|
|
VkCullModeFlags DecodeCullMode(D3DCULL Mode) {
|
|
switch (Mode) {
|
|
case D3DCULL_CW: return VK_CULL_MODE_FRONT_BIT;
|
|
case D3DCULL_CCW: return VK_CULL_MODE_BACK_BIT;
|
|
default:
|
|
case D3DCULL_NONE: return VK_CULL_MODE_NONE;
|
|
}
|
|
}
|
|
|
|
|
|
VkPolygonMode DecodeFillMode(D3DFILLMODE Mode) {
|
|
switch (Mode) {
|
|
case D3DFILL_POINT: return VK_POLYGON_MODE_POINT;
|
|
case D3DFILL_WIREFRAME: return VK_POLYGON_MODE_LINE;
|
|
default:
|
|
case D3DFILL_SOLID: return VK_POLYGON_MODE_FILL;
|
|
}
|
|
}
|
|
|
|
|
|
VkIndexType DecodeIndexType(D3D9Format Format) {
|
|
return Format == D3D9Format::INDEX16
|
|
? VK_INDEX_TYPE_UINT16
|
|
: VK_INDEX_TYPE_UINT32;
|
|
}
|
|
|
|
|
|
VkFormat DecodeDecltype(D3DDECLTYPE Type) {
|
|
switch (Type) {
|
|
case D3DDECLTYPE_FLOAT1: return VK_FORMAT_R32_SFLOAT;
|
|
case D3DDECLTYPE_FLOAT2: return VK_FORMAT_R32G32_SFLOAT;
|
|
case D3DDECLTYPE_FLOAT3: return VK_FORMAT_R32G32B32_SFLOAT;
|
|
case D3DDECLTYPE_FLOAT4: return VK_FORMAT_R32G32B32A32_SFLOAT;
|
|
case D3DDECLTYPE_D3DCOLOR: return VK_FORMAT_B8G8R8A8_UNORM;
|
|
case D3DDECLTYPE_UBYTE4: return VK_FORMAT_R8G8B8A8_USCALED;
|
|
case D3DDECLTYPE_SHORT2: return VK_FORMAT_R16G16_SSCALED;
|
|
case D3DDECLTYPE_SHORT4: return VK_FORMAT_R16G16B16A16_SSCALED;
|
|
case D3DDECLTYPE_UBYTE4N: return VK_FORMAT_R8G8B8A8_UNORM;
|
|
case D3DDECLTYPE_SHORT2N: return VK_FORMAT_R16G16_SNORM;
|
|
case D3DDECLTYPE_SHORT4N: return VK_FORMAT_R16G16B16A16_SNORM;
|
|
case D3DDECLTYPE_USHORT2N: return VK_FORMAT_R16G16_UNORM;
|
|
case D3DDECLTYPE_USHORT4N: return VK_FORMAT_R16G16B16A16_UNORM;
|
|
case D3DDECLTYPE_UDEC3: return VK_FORMAT_A2B10G10R10_USCALED_PACK32;
|
|
case D3DDECLTYPE_FLOAT16_2: return VK_FORMAT_R16G16_SFLOAT;
|
|
case D3DDECLTYPE_FLOAT16_4: return VK_FORMAT_R16G16B16A16_SFLOAT;
|
|
case D3DDECLTYPE_DEC3N: return VK_FORMAT_A2B10G10R10_SNORM_PACK32;
|
|
case D3DDECLTYPE_UNUSED:
|
|
default: return VK_FORMAT_UNDEFINED;
|
|
}
|
|
}
|
|
|
|
void ConvertBox(D3DBOX box, VkOffset3D& offset, VkExtent3D& extent) {
|
|
offset.x = box.Left;
|
|
offset.y = box.Top;
|
|
offset.z = box.Front;
|
|
|
|
extent.width = box.Right - box.Left;
|
|
extent.height = box.Bottom - box.Top;
|
|
extent.depth = box.Back - box.Front;
|
|
}
|
|
|
|
void ConvertRect(RECT rect, VkOffset3D& offset, VkExtent3D& extent) {
|
|
offset.x = rect.left;
|
|
offset.y = rect.top;
|
|
offset.z = 0;
|
|
|
|
extent.width = rect.right - rect.left;
|
|
extent.height = rect.bottom - rect.top;
|
|
extent.depth = 1;
|
|
}
|
|
|
|
void ConvertRect(RECT rect, VkOffset2D& offset, VkExtent2D& extent) {
|
|
offset.x = rect.left;
|
|
offset.y = rect.top;
|
|
|
|
extent.width = rect.right - rect.left;
|
|
extent.height = rect.bottom - rect.top;
|
|
}
|
|
|
|
uint32_t GetDecltypeSize(D3DDECLTYPE Type) {
|
|
switch (Type) {
|
|
case D3DDECLTYPE_FLOAT1: return 1 * sizeof(float);
|
|
case D3DDECLTYPE_FLOAT2: return 2 * sizeof(float);
|
|
case D3DDECLTYPE_FLOAT3: return 3 * sizeof(float);
|
|
case D3DDECLTYPE_FLOAT4: return 4 * sizeof(float);
|
|
case D3DDECLTYPE_D3DCOLOR: return 1 * sizeof(DWORD);
|
|
case D3DDECLTYPE_UBYTE4: return 4 * sizeof(BYTE);
|
|
case D3DDECLTYPE_SHORT2: return 2 * sizeof(short);
|
|
case D3DDECLTYPE_SHORT4: return 4 * sizeof(short);
|
|
case D3DDECLTYPE_UBYTE4N: return 4 * sizeof(BYTE);
|
|
case D3DDECLTYPE_SHORT2N: return 2 * sizeof(short);
|
|
case D3DDECLTYPE_SHORT4N: return 4 * sizeof(short);
|
|
case D3DDECLTYPE_USHORT2N: return 2 * sizeof(short);
|
|
case D3DDECLTYPE_USHORT4N: return 4 * sizeof(short);
|
|
case D3DDECLTYPE_UDEC3: return 4;
|
|
case D3DDECLTYPE_DEC3N: return 4;
|
|
case D3DDECLTYPE_FLOAT16_2: return 2 * 2;
|
|
case D3DDECLTYPE_FLOAT16_4: return 4 * 2;
|
|
default: return 0;
|
|
}
|
|
}
|
|
|
|
|
|
uint32_t GetDecltypeCount(D3DDECLTYPE Type) {
|
|
switch (Type) {
|
|
case D3DDECLTYPE_FLOAT1: return 1;
|
|
case D3DDECLTYPE_FLOAT2: return 2;
|
|
case D3DDECLTYPE_FLOAT3: return 3;
|
|
case D3DDECLTYPE_FLOAT4: return 4;
|
|
case D3DDECLTYPE_D3DCOLOR: return 4;
|
|
case D3DDECLTYPE_UBYTE4: return 4;
|
|
case D3DDECLTYPE_SHORT2: return 2;
|
|
case D3DDECLTYPE_SHORT4: return 4;
|
|
case D3DDECLTYPE_UBYTE4N: return 4;
|
|
case D3DDECLTYPE_SHORT2N: return 2;
|
|
case D3DDECLTYPE_SHORT4N: return 4;
|
|
case D3DDECLTYPE_USHORT2N: return 2;
|
|
case D3DDECLTYPE_USHORT4N: return 4;
|
|
case D3DDECLTYPE_UDEC3: return 3;
|
|
case D3DDECLTYPE_DEC3N: return 3;
|
|
case D3DDECLTYPE_FLOAT16_2: return 2;
|
|
case D3DDECLTYPE_FLOAT16_4: return 4;
|
|
default: return 0;
|
|
}
|
|
}
|
|
|
|
|
|
bool IsDepthFormat(D3D9Format Format) {
|
|
return Format == D3D9Format::D16_LOCKABLE
|
|
|| Format == D3D9Format::D32
|
|
|| Format == D3D9Format::D15S1
|
|
|| Format == D3D9Format::D24S8
|
|
|| Format == D3D9Format::D24X8
|
|
|| Format == D3D9Format::D24X4S4
|
|
|| Format == D3D9Format::D16
|
|
|| Format == D3D9Format::D32F_LOCKABLE
|
|
|| Format == D3D9Format::D24FS8
|
|
|| Format == D3D9Format::D32_LOCKABLE
|
|
|| Format == D3D9Format::DF16
|
|
|| Format == D3D9Format::DF24
|
|
|| Format == D3D9Format::INTZ;
|
|
}
|
|
|
|
} |