2021-07-24 21:11:47 -07:00

798 lines
28 KiB
C++

//===== Copyright (c) 1996-2005, Valve Corporation, All rights reserved. ======//
//
// Purpose:
//
// $NoKeywords: $
//
//=============================================================================//
////#define APIENTRY
//#if defined(_WIN32) && !defined(_X360)
////#define WIN32_LEAN_AND_MEAN
////#define _WIN32_WINNT 0x0500
////#include <windows.h>
////#include <assert.h>
//#endif
//#include <GL/gl.h>
//#include "tier0/platform.h"
////#include <GL/glext.h>
////#include "tier0/dynfunction.h"
#define DISABLE_PROTECTED_THINGS
#include "togl/rendermechanism.h"
#include "locald3dtypes.h"
#include "colorformatdx8.h"
#include "shaderapidx8_global.h"
#include "bitmap/imageformat.h"
#include "shaderapi/ishaderutil.h"
#include "tier0/dbg.h"
#include "tier1/strtools.h"
#include "shaderdevicedx8.h"
// Must be last
#include "tier0/memdbgon.h"
//-----------------------------------------------------------------------------
// Figures out what texture formats we support
//-----------------------------------------------------------------------------
//-----------------------------------------------------------------------------
// globals
//-----------------------------------------------------------------------------
// Texture formats supported by DX driver[vertextexture][render target][non filterable]
static D3DFORMAT g_D3DColorFormat[NUM_IMAGE_FORMATS][2][2][2];
static UINT g_DisplayAdapter;
static D3DDEVTYPE g_DeviceType;
static ImageFormat g_DeviceFormat;
static bool g_bSupportsD24S8;
static bool g_bSupportsD24X8;
static bool g_bSupportsD16;
static bool g_bSupportsD24X4S4;
static bool g_bSupportsD15S1;
//-----------------------------------------------------------------------------
// Determines what formats we actually *do* support
//-----------------------------------------------------------------------------
static bool TestTextureFormat( D3DFORMAT format, bool bIsRenderTarget,
bool bIsVertexTexture, bool bIsFilterableRequired )
{
int nUsage = bIsRenderTarget ? D3DUSAGE_RENDERTARGET : 0;
if ( bIsVertexTexture )
{
// vertex textures never need filtering
nUsage |= D3DUSAGE_QUERY_VERTEXTEXTURE;
}
if ( bIsFilterableRequired )
{
nUsage |= D3DUSAGE_QUERY_FILTER;
}
HRESULT hr;
// IHV depth texture formats require a slightly different check...
if ( !IsGameConsole() && bIsRenderTarget && ( ( format == NVFMT_RAWZ ) || ( format == NVFMT_INTZ ) ||
( format == D3DFMT_D16 ) || ( format == D3DFMT_D24S8 ) ||
( format == ATIFMT_D16 ) || ( format == ATIFMT_D24S8 ) ) )
{
hr = D3D()->CheckDeviceFormat(
g_DisplayAdapter, g_DeviceType, ImageLoader::ImageFormatToD3DFormat( g_DeviceFormat ),
D3DUSAGE_DEPTHSTENCIL, D3DRTYPE_TEXTURE, format );
}
else if ( IsX360() && bIsRenderTarget )
{
// 360 can only validate render targets as surface display format
hr = D3D()->CheckDeviceFormat( g_DisplayAdapter, g_DeviceType, format, 0, D3DRTYPE_SURFACE, format );
}
else // general case:
{
// See if we can do it!
hr = D3D()->CheckDeviceFormat(
g_DisplayAdapter, g_DeviceType, ImageLoader::ImageFormatToD3DFormat( g_DeviceFormat ),
nUsage, D3DRTYPE_TEXTURE, format );
}
#ifdef _PS3
// We have a conflict with the SUCCEEDED macro
return hr >= 0;
#else // _PS3
return SUCCEEDED( hr );
#endif // !_PS3
}
D3DFORMAT GetNearestD3DColorFormat( ImageFormat fmt,
bool isRenderTarget, bool bIsVertexTexture,
bool bIsFilterableRequired)
{
switch(fmt)
{
case IMAGE_FORMAT_RGBA8888:
case IMAGE_FORMAT_ABGR8888:
case IMAGE_FORMAT_ARGB8888:
case IMAGE_FORMAT_BGRA8888:
if (TestTextureFormat(D3DFMT_A8R8G8B8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_A8R8G8B8;
if (TestTextureFormat(D3DFMT_A4R4G4B4, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_A4R4G4B4;
break;
#if defined( _X360 )
case IMAGE_FORMAT_LINEAR_RGBA8888:
case IMAGE_FORMAT_LINEAR_ABGR8888:
case IMAGE_FORMAT_LINEAR_ARGB8888:
case IMAGE_FORMAT_LINEAR_BGRA8888:
// same as above - all xxxx8888 RGBA ordering funnels to d3d a8r8g8b8
if ( TestTextureFormat( D3DFMT_LIN_A8R8G8B8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired ) )
return D3DFMT_LIN_A8R8G8B8;
break;
#endif
#if defined( _X360 )
case IMAGE_FORMAT_LINEAR_BGRX8888:
if ( TestTextureFormat( D3DFMT_LIN_X8R8G8B8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired ) )
return D3DFMT_LIN_X8R8G8B8;
break;
#endif
case IMAGE_FORMAT_BGRX8888:
// We want this format to return exactly it's equivalent so that
// when we create render targets to blit to from the framebuffer,
// the CopyRect won't fail due to format mismatches.
if (TestTextureFormat(D3DFMT_X8R8G8B8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_X8R8G8B8;
// fall through. . . .
case IMAGE_FORMAT_RGB888:
case IMAGE_FORMAT_BGR888:
#if !defined( _GAMECONSOLE )
if (TestTextureFormat(D3DFMT_R8G8B8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_R8G8B8;
#endif
if (TestTextureFormat(D3DFMT_X8R8G8B8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_X8R8G8B8;
if (TestTextureFormat(D3DFMT_A8R8G8B8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_A8R8G8B8;
if (TestTextureFormat(D3DFMT_R5G6B5, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_R5G6B5;
if (TestTextureFormat(D3DFMT_X1R5G5B5, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_X1R5G5B5;
if (TestTextureFormat(D3DFMT_A1R5G5B5, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_A1R5G5B5;
break;
case IMAGE_FORMAT_BGR565:
case IMAGE_FORMAT_RGB565:
if (TestTextureFormat(D3DFMT_R5G6B5, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_R5G6B5;
if (TestTextureFormat(D3DFMT_X1R5G5B5, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_X1R5G5B5;
if (TestTextureFormat(D3DFMT_A1R5G5B5, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_A1R5G5B5;
#if !defined( _GAMECONSOLE )
if (TestTextureFormat(D3DFMT_R8G8B8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_R8G8B8;
#endif
if (TestTextureFormat(D3DFMT_X8R8G8B8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_X8R8G8B8;
if (TestTextureFormat(D3DFMT_A8R8G8B8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_A8R8G8B8;
break;
case IMAGE_FORMAT_BGRX5551:
if (TestTextureFormat(D3DFMT_X1R5G5B5, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_X1R5G5B5;
if (TestTextureFormat(D3DFMT_A1R5G5B5, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_A1R5G5B5;
if (TestTextureFormat(D3DFMT_R5G6B5, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_R5G6B5;
#if !defined( _GAMECONSOLE )
if (TestTextureFormat(D3DFMT_R8G8B8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_R8G8B8;
#endif
if (TestTextureFormat(D3DFMT_X8R8G8B8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_X8R8G8B8;
if (TestTextureFormat(D3DFMT_A8R8G8B8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_A8R8G8B8;
break;
#if defined( _X360 )
case IMAGE_FORMAT_LINEAR_BGRX5551:
if ( TestTextureFormat( D3DFMT_LIN_X1R5G5B5, isRenderTarget, bIsVertexTexture, bIsFilterableRequired ) )
return D3DFMT_LIN_X1R5G5B5;
break;
#endif
case IMAGE_FORMAT_BGRA5551:
if (TestTextureFormat(D3DFMT_A1R5G5B5, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_A1R5G5B5;
if (TestTextureFormat(D3DFMT_A4R4G4B4, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_A4R4G4B4;
if (TestTextureFormat(D3DFMT_A8R8G8B8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_A8R8G8B8;
break;
case IMAGE_FORMAT_BGRA4444:
if (TestTextureFormat(D3DFMT_A4R4G4B4, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_A4R4G4B4;
if (TestTextureFormat(D3DFMT_A8R8G8B8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_A8R8G8B8;
break;
case IMAGE_FORMAT_I8:
if (TestTextureFormat(D3DFMT_L8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_L8;
if (TestTextureFormat(D3DFMT_A8R8G8B8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_A8R8G8B8;
break;
#if defined( _X360 )
case IMAGE_FORMAT_LINEAR_I8:
if ( TestTextureFormat( D3DFMT_LIN_L8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired ) )
return D3DFMT_LIN_L8;
break;
#endif
case IMAGE_FORMAT_IA88:
if (TestTextureFormat(D3DFMT_A8L8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_A8L8;
if (TestTextureFormat(D3DFMT_A8R8G8B8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_A8R8G8B8;
break;
case IMAGE_FORMAT_A8:
if (TestTextureFormat(D3DFMT_A8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_A8;
if (TestTextureFormat(D3DFMT_A8R8G8B8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_A8R8G8B8;
break;
#if defined( _X360 )
case IMAGE_FORMAT_LINEAR_A8:
if (TestTextureFormat(D3DFMT_LIN_A8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_LIN_A8;
break;
#endif
case IMAGE_FORMAT_DXT1:
case IMAGE_FORMAT_DXT1_ONEBITALPHA:
case IMAGE_FORMAT_DXT1_RUNTIME:
if (TestTextureFormat(D3DFMT_DXT1, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_DXT1;
break;
#if defined( _X360 )
case IMAGE_FORMAT_LINEAR_DXT1:
if (TestTextureFormat(D3DFMT_LIN_DXT1, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_LIN_DXT1;
break;
#endif
case IMAGE_FORMAT_DXT3:
if (TestTextureFormat(D3DFMT_DXT3, isRenderTarget, bIsVertexTexture, bIsFilterableRequired ))
return D3DFMT_DXT3;
break;
#if defined( _X360 )
case IMAGE_FORMAT_LINEAR_DXT3:
if (TestTextureFormat(D3DFMT_LIN_DXT3, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_LIN_DXT3;
break;
#endif
case IMAGE_FORMAT_DXT5:
case IMAGE_FORMAT_DXT5_RUNTIME:
if (TestTextureFormat(D3DFMT_DXT5, isRenderTarget, bIsVertexTexture, bIsFilterableRequired ))
return D3DFMT_DXT5;
break;
#if defined( _X360 )
case IMAGE_FORMAT_LINEAR_DXT5:
if (TestTextureFormat(D3DFMT_LIN_DXT5, isRenderTarget, bIsVertexTexture, bIsFilterableRequired))
return D3DFMT_LIN_DXT5;
break;
#endif
case IMAGE_FORMAT_UV88:
if (TestTextureFormat(D3DFMT_V8U8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired ))
return D3DFMT_V8U8;
break;
case IMAGE_FORMAT_UVWQ8888:
if (TestTextureFormat(D3DFMT_Q8W8V8U8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired ))
return D3DFMT_Q8W8V8U8;
break;
case IMAGE_FORMAT_UVLX8888:
if (TestTextureFormat(D3DFMT_X8L8V8U8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired ))
return D3DFMT_X8L8V8U8;
break;
case IMAGE_FORMAT_RGBA16161616F:
if ( TestTextureFormat( D3DFMT_A16B16G16R16F, isRenderTarget, bIsVertexTexture, bIsFilterableRequired ) )
return D3DFMT_A16B16G16R16F;
if ( TestTextureFormat( D3DFMT_A16B16G16R16, isRenderTarget, bIsVertexTexture, bIsFilterableRequired ) )
return D3DFMT_A16B16G16R16;
break;
case IMAGE_FORMAT_RGBA16161616:
if ( TestTextureFormat( D3DFMT_A16B16G16R16, isRenderTarget, bIsVertexTexture, bIsFilterableRequired ) )
return D3DFMT_A16B16G16R16;
if ( TestTextureFormat( D3DFMT_A16B16G16R16F, isRenderTarget, bIsVertexTexture, bIsFilterableRequired ) )
return D3DFMT_A16B16G16R16F;
break;
#if defined( _X360 )
case IMAGE_FORMAT_LINEAR_RGBA16161616:
if ( TestTextureFormat( D3DFMT_LIN_A16B16G16R16, isRenderTarget, bIsVertexTexture, bIsFilterableRequired ) )
return D3DFMT_LIN_A16B16G16R16;
break;
#endif
case IMAGE_FORMAT_R32F:
if ( TestTextureFormat( D3DFMT_R32F, isRenderTarget, bIsVertexTexture, bIsFilterableRequired ) )
return D3DFMT_R32F;
break;
case IMAGE_FORMAT_RGBA32323232F:
if ( TestTextureFormat( D3DFMT_A32B32G32R32F, isRenderTarget, bIsVertexTexture, bIsFilterableRequired ) )
return D3DFMT_A32B32G32R32F;
break;
#if defined( _X360 ) || defined( _PS3 )
case IMAGE_FORMAT_D16:
return D3DFMT_D16;
case IMAGE_FORMAT_D24S8:
return D3DFMT_D24S8;
#endif // _X360 || _PS3
#if defined( _X360 )
case IMAGE_FORMAT_D24FS8:
return D3DFMT_D24FS8;
case IMAGE_FORMAT_LE_BGRX8888:
return D3DFMT_LE_X8R8G8B8;
case IMAGE_FORMAT_LE_BGRA8888:
return D3DFMT_LE_A8R8G8B8;
#endif // _X360
#if defined( DX_TO_GL_ABSTRACTION )
case IMAGE_FORMAT_D24S8:
return D3DFMT_D24S8;
case IMAGE_FORMAT_D24X8:
return D3DFMT_D24X8;
#endif
// nVidia overloads DST formats as texture formats
case IMAGE_FORMAT_D16:
case IMAGE_FORMAT_D16_SHADOW:
if ( IsOpenGL() )
{
return D3DFMT_D16;
}
// Only try ATIFMT_D16 on non-DX10 capable ATI cards (where we use fetch4). On DX10 capable ATI cards we use hardware PCF.
if ( g_pHardwareConfig->Caps().m_VendorID == VENDORID_ATI && ( CommandLine()->CheckParm( "-forceatifetch4" ) || !g_pHardwareConfig->Caps().m_bDX10Card ) )
{
if ( TestTextureFormat( ATIFMT_D16, isRenderTarget, bIsVertexTexture, false ) )
return ATIFMT_D16;
}
else
{
if ( TestTextureFormat( D3DFMT_D16, isRenderTarget, bIsVertexTexture, bIsFilterableRequired ) )
return D3DFMT_D16;
}
break;
case IMAGE_FORMAT_D24X8_SHADOW:
if ( IsOSXOpenGL() )
{
return D3DFMT_D24X8;
}
// Only try ATIFMT_D24S8 on non-DX10 capable ATI cards (where we use fetch4). On DX10 capable ATI cards we use hardware PCF.
if ( g_pHardwareConfig->Caps().m_VendorID == VENDORID_ATI && ( CommandLine()->CheckParm( "-forceatifetch4" ) || !g_pHardwareConfig->Caps().m_bDX10Card ) )
{
if ( TestTextureFormat( ATIFMT_D24S8, isRenderTarget, bIsVertexTexture, false ) )
return ATIFMT_D24S8;
}
else
{
if ( TestTextureFormat( D3DFMT_D24S8, isRenderTarget, bIsVertexTexture, bIsFilterableRequired ) )
return D3DFMT_D24S8;
}
break;
case IMAGE_FORMAT_NULL:
if ( TestTextureFormat( NVFMT_NULL, isRenderTarget, bIsVertexTexture, false ) )
return NVFMT_NULL;
break;
case IMAGE_FORMAT_ATI2N:
if ( TestTextureFormat( ATIFMT_ATI2N, isRenderTarget, bIsVertexTexture, bIsFilterableRequired ) )
return ATIFMT_ATI2N;
break;
case IMAGE_FORMAT_ATI1N:
if ( TestTextureFormat( ATIFMT_ATI1N, isRenderTarget, bIsVertexTexture, bIsFilterableRequired ) )
return ATIFMT_ATI1N;
break;
case IMAGE_FORMAT_INTZ:
if ( TestTextureFormat( NVFMT_INTZ, isRenderTarget, bIsVertexTexture, bIsFilterableRequired ) )
return NVFMT_INTZ;
break;
}
return D3DFMT_UNKNOWN;
}
void InitializeColorInformation( UINT displayAdapter, D3DDEVTYPE deviceType,
ImageFormat displayFormat )
{
g_DisplayAdapter = displayAdapter;
g_DeviceType = deviceType;
g_DeviceFormat = displayFormat;
int fmt = 0;
while ( fmt < NUM_IMAGE_FORMATS )
{
for ( int nVertexTexture = 0; nVertexTexture <= 1; ++nVertexTexture )
{
for ( int nRenderTarget = 0; nRenderTarget <= 1; ++nRenderTarget )
{
for ( int nFilterable = 0; nFilterable <= 1; ++nFilterable )
{
g_D3DColorFormat[fmt][nVertexTexture][nRenderTarget][nFilterable] =
GetNearestD3DColorFormat( (ImageFormat)fmt, nRenderTarget != 0, nVertexTexture != 0, nFilterable != 0 );
}
}
}
++fmt;
}
// Check the depth formats
HRESULT hr = D3D()->CheckDeviceFormat(
g_DisplayAdapter, g_DeviceType, ImageLoader::ImageFormatToD3DFormat( g_DeviceFormat ),
D3DUSAGE_DEPTHSTENCIL, D3DRTYPE_SURFACE, D3DFMT_D24S8 );
g_bSupportsD24S8 = !FAILED(hr);
hr = D3D()->CheckDeviceFormat(
g_DisplayAdapter, g_DeviceType, ImageLoader::ImageFormatToD3DFormat( g_DeviceFormat ),
D3DUSAGE_DEPTHSTENCIL, D3DRTYPE_SURFACE, D3DFMT_D24X8 );
g_bSupportsD24X8 = !FAILED(hr);
hr = D3D()->CheckDeviceFormat(
g_DisplayAdapter, g_DeviceType, ImageLoader::ImageFormatToD3DFormat( g_DeviceFormat ),
D3DUSAGE_DEPTHSTENCIL, D3DRTYPE_SURFACE, D3DFMT_D16 );
g_bSupportsD16 = !FAILED(hr);
#if !defined( _X360 )
hr = D3D()->CheckDeviceFormat(
g_DisplayAdapter, g_DeviceType, ImageLoader::ImageFormatToD3DFormat( g_DeviceFormat ),
D3DUSAGE_DEPTHSTENCIL, D3DRTYPE_SURFACE, D3DFMT_D24X4S4 );
g_bSupportsD24X4S4 = !FAILED(hr);
#else
g_bSupportsD24X4S4 = false;
#endif
#if !defined( _X360 )
hr = D3D()->CheckDeviceFormat(
g_DisplayAdapter, g_DeviceType, ImageLoader::ImageFormatToD3DFormat( g_DeviceFormat ),
D3DUSAGE_DEPTHSTENCIL, D3DRTYPE_SURFACE, D3DFMT_D15S1 );
g_bSupportsD15S1 = !FAILED(hr);
#else
g_bSupportsD15S1 = false;
#endif
}
//-----------------------------------------------------------------------------
// Returns true if compressed textures are supported
//-----------------------------------------------------------------------------
bool D3DSupportsCompressedTextures()
{
return (g_D3DColorFormat[IMAGE_FORMAT_DXT1][0][0][0] != D3DFMT_UNKNOWN) &&
(g_D3DColorFormat[IMAGE_FORMAT_DXT3][0][0][0] != D3DFMT_UNKNOWN) &&
(g_D3DColorFormat[IMAGE_FORMAT_DXT5][0][0][0] != D3DFMT_UNKNOWN);
}
//-----------------------------------------------------------------------------
// Returns closest supported format
//-----------------------------------------------------------------------------
D3DFORMAT FindNearestSupportedFormat( ImageFormat format, bool bIsVertexTexture, bool bIsRenderTarget, bool bFilterableRequired )
{
return g_D3DColorFormat[format][bIsVertexTexture][bIsRenderTarget][bFilterableRequired];
}
//-----------------------------------------------------------------------------
// Returns true if compressed textures are supported
//-----------------------------------------------------------------------------
bool D3DSupportsDepthTexture(D3DFORMAT format)
{
// See if we can do it!
HRESULT hr = D3D()->CheckDeviceFormat(
g_DisplayAdapter, g_DeviceType, ImageLoader::ImageFormatToD3DFormat(g_DeviceFormat),
D3DUSAGE_DEPTHSTENCIL, D3DRTYPE_TEXTURE, format);
return !FAILED(hr);
}
//-----------------------------------------------------------------------------
// Returns true if the depth format is compatible with the display
//-----------------------------------------------------------------------------
static inline bool IsDepthFormatCompatible( int nAdapter, ImageFormat displayFormat, ImageFormat renderTargetFormat, D3DFORMAT depthFormat )
{
D3DFORMAT d3dDisplayFormat = ImageLoader::ImageFormatToD3DFormat( displayFormat );
D3DFORMAT d3dRenderTargetFormat = ImageLoader::ImageFormatToD3DFormat( renderTargetFormat );
// Verify that the depth format is compatible.
HRESULT hr = D3D()->CheckDepthStencilMatch( nAdapter, DX8_DEVTYPE,
d3dDisplayFormat, d3dRenderTargetFormat, depthFormat);
return !FAILED(hr);
}
//-----------------------------------------------------------------------------
// Finds the nearest supported depth buffer format
//-----------------------------------------------------------------------------
D3DFORMAT FindNearestSupportedDepthFormat( int nAdapter, ImageFormat displayFormat, ImageFormat renderTargetFormat, D3DFORMAT depthFormat )
{
// This is the default case, used for rendering to the main render target
Assert( displayFormat != IMAGE_FORMAT_UNKNOWN && renderTargetFormat != IMAGE_FORMAT_UNKNOWN );
switch (depthFormat)
{
#if defined( _X360 )
case D3DFMT_D24FS8:
return D3DFMT_D24FS8;
case D3DFMT_LIN_D24S8:
if ( g_bSupportsD24S8 && IsDepthFormatCompatible( nAdapter, displayFormat, renderTargetFormat, D3DFMT_LIN_D24S8 ) )
return D3DFMT_LIN_D24S8;
#endif
case D3DFMT_D24S8:
if ( g_bSupportsD24S8 && IsDepthFormatCompatible( nAdapter, displayFormat, renderTargetFormat, D3DFMT_D24S8 ) )
return D3DFMT_D24S8;
#if !defined( _X360 )
if ( g_bSupportsD24X4S4 && IsDepthFormatCompatible( nAdapter, displayFormat, renderTargetFormat, D3DFMT_D24X4S4 ) )
return D3DFMT_D24X4S4;
if ( g_bSupportsD15S1 && IsDepthFormatCompatible( nAdapter, displayFormat, renderTargetFormat, D3DFMT_D15S1 ) )
return D3DFMT_D15S1;
#endif
if ( g_bSupportsD24X8 && IsDepthFormatCompatible( nAdapter, displayFormat, renderTargetFormat, D3DFMT_D24X8 ) )
return D3DFMT_D24X8;
if ( g_bSupportsD16 && IsDepthFormatCompatible( nAdapter, displayFormat, renderTargetFormat, D3DFMT_D16 ) )
return D3DFMT_D16;
break;
case D3DFMT_D24X8:
if ( g_bSupportsD24X8 && IsDepthFormatCompatible( nAdapter, displayFormat, renderTargetFormat, D3DFMT_D24X8 ) )
return D3DFMT_D24X8;
if ( g_bSupportsD24S8 && IsDepthFormatCompatible( nAdapter, displayFormat, renderTargetFormat, D3DFMT_D24S8 ) )
return D3DFMT_D24S8;
#if !defined( _X360 )
if ( g_bSupportsD24X4S4 && IsDepthFormatCompatible( nAdapter, displayFormat, renderTargetFormat, D3DFMT_D24X4S4 ) )
return D3DFMT_D24X4S4;
#endif
if ( g_bSupportsD16 && IsDepthFormatCompatible( nAdapter, displayFormat, renderTargetFormat, D3DFMT_D16 ) )
return D3DFMT_D16;
#if !defined( _X360 )
if ( g_bSupportsD15S1 && IsDepthFormatCompatible( nAdapter, displayFormat, renderTargetFormat, D3DFMT_D15S1 ) )
return D3DFMT_D15S1;
#endif
break;
case D3DFMT_D16:
if ( g_bSupportsD16 && IsDepthFormatCompatible( nAdapter, displayFormat, renderTargetFormat, D3DFMT_D16 ) )
return D3DFMT_D16;
#if !defined( _X360 )
if ( g_bSupportsD15S1 && IsDepthFormatCompatible( nAdapter, displayFormat, renderTargetFormat, D3DFMT_D15S1 ) )
return D3DFMT_D15S1;
#endif
if ( g_bSupportsD24X8 && IsDepthFormatCompatible( nAdapter, displayFormat, renderTargetFormat, D3DFMT_D24X8 ) )
return D3DFMT_D24X8;
if ( g_bSupportsD24S8 && IsDepthFormatCompatible( nAdapter, displayFormat, renderTargetFormat, D3DFMT_D24S8 ) )
return D3DFMT_D24S8;
#if !defined( _X360 )
if ( g_bSupportsD24X4S4 && IsDepthFormatCompatible( nAdapter, displayFormat, renderTargetFormat, D3DFMT_D24X4S4 ) )
return D3DFMT_D24X4S4;
#endif
break;
}
Assert( 0 );
return D3DFMT_D16;
}
//-----------------------------------------------------------------------------
// Is a display buffer valid?
//-----------------------------------------------------------------------------
static inline bool IsFrameBufferFormatValid( UINT displayAdapter, D3DDEVTYPE deviceType,
D3DFORMAT displayFormat, D3DFORMAT backBufferFormat, bool bIsWindowed )
{
HRESULT hr = D3D()->CheckDeviceType( displayAdapter, deviceType, displayFormat,
backBufferFormat, bIsWindowed );
return !FAILED(hr);
}
//-----------------------------------------------------------------------------
// Finds the nearest supported frame buffer format
//-----------------------------------------------------------------------------
ImageFormat FindNearestSupportedBackBufferFormat( UINT displayAdapter,
D3DDEVTYPE deviceType, ImageFormat displayFormat, ImageFormat backBufferFormat, bool bIsWindowed )
{
D3DFORMAT d3dDisplayFormat = ImageLoader::ImageFormatToD3DFormat( displayFormat );
switch (backBufferFormat)
{
case IMAGE_FORMAT_RGBA8888:
case IMAGE_FORMAT_ABGR8888:
case IMAGE_FORMAT_ARGB8888:
case IMAGE_FORMAT_BGRA8888:
case IMAGE_FORMAT_BGRA4444: // This is not supported ever; bump up to 32 bit
if (IsFrameBufferFormatValid( displayAdapter, deviceType, d3dDisplayFormat, D3DFMT_A8R8G8B8, bIsWindowed ))
return IMAGE_FORMAT_BGRA8888;
// Bye, bye dest alpha
if (IsFrameBufferFormatValid( displayAdapter, deviceType, d3dDisplayFormat, D3DFMT_X8R8G8B8, bIsWindowed ))
return IMAGE_FORMAT_BGRX8888;
if (IsFrameBufferFormatValid( displayAdapter, deviceType, d3dDisplayFormat, D3DFMT_R5G6B5, bIsWindowed ))
return IMAGE_FORMAT_BGR565;
if (IsFrameBufferFormatValid( displayAdapter, deviceType, d3dDisplayFormat, D3DFMT_A1R5G5B5, bIsWindowed ))
return IMAGE_FORMAT_BGRA5551;
if (IsFrameBufferFormatValid( displayAdapter, deviceType, d3dDisplayFormat, D3DFMT_X1R5G5B5, bIsWindowed ))
return IMAGE_FORMAT_BGRX5551;
return IMAGE_FORMAT_UNKNOWN;
case IMAGE_FORMAT_RGB888:
case IMAGE_FORMAT_BGR888:
case IMAGE_FORMAT_RGB888_BLUESCREEN:
case IMAGE_FORMAT_BGRX8888:
if (IsFrameBufferFormatValid( displayAdapter, deviceType, d3dDisplayFormat, D3DFMT_X8R8G8B8, bIsWindowed ))
return IMAGE_FORMAT_BGRX8888;
if (IsFrameBufferFormatValid( displayAdapter, deviceType, d3dDisplayFormat, D3DFMT_A8R8G8B8, bIsWindowed ))
return IMAGE_FORMAT_BGRA8888;
if (IsFrameBufferFormatValid( displayAdapter, deviceType, d3dDisplayFormat, D3DFMT_R5G6B5, bIsWindowed ))
return IMAGE_FORMAT_BGR565;
if (IsFrameBufferFormatValid( displayAdapter, deviceType, d3dDisplayFormat, D3DFMT_A1R5G5B5, bIsWindowed ))
return IMAGE_FORMAT_BGRA5551;
if (IsFrameBufferFormatValid( displayAdapter, deviceType, d3dDisplayFormat, D3DFMT_X1R5G5B5, bIsWindowed ))
return IMAGE_FORMAT_BGRX5551;
return IMAGE_FORMAT_UNKNOWN;
case IMAGE_FORMAT_RGB565:
case IMAGE_FORMAT_BGR565:
if (IsFrameBufferFormatValid( displayAdapter, deviceType, d3dDisplayFormat, D3DFMT_R5G6B5, bIsWindowed ))
return IMAGE_FORMAT_BGR565;
if (IsFrameBufferFormatValid( displayAdapter, deviceType, d3dDisplayFormat, D3DFMT_A1R5G5B5, bIsWindowed ))
return IMAGE_FORMAT_BGRA5551;
if (IsFrameBufferFormatValid( displayAdapter, deviceType, d3dDisplayFormat, D3DFMT_X1R5G5B5, bIsWindowed ))
return IMAGE_FORMAT_BGRX5551;
if (IsFrameBufferFormatValid( displayAdapter, deviceType, d3dDisplayFormat, D3DFMT_X8R8G8B8, bIsWindowed ))
return IMAGE_FORMAT_BGRX8888;
if (IsFrameBufferFormatValid( displayAdapter, deviceType, d3dDisplayFormat, D3DFMT_A8R8G8B8, bIsWindowed ))
return IMAGE_FORMAT_BGRA8888;
return IMAGE_FORMAT_UNKNOWN;
case IMAGE_FORMAT_BGRX5551:
if (IsFrameBufferFormatValid( displayAdapter, deviceType, d3dDisplayFormat, D3DFMT_X1R5G5B5, bIsWindowed ))
return IMAGE_FORMAT_BGRX5551;
if (IsFrameBufferFormatValid( displayAdapter, deviceType, d3dDisplayFormat, D3DFMT_A1R5G5B5, bIsWindowed ))
return IMAGE_FORMAT_BGRA5551;
if (IsFrameBufferFormatValid( displayAdapter, deviceType, d3dDisplayFormat, D3DFMT_R5G6B5, bIsWindowed ))
return IMAGE_FORMAT_BGR565;
if (IsFrameBufferFormatValid( displayAdapter, deviceType, d3dDisplayFormat, D3DFMT_X8R8G8B8, bIsWindowed ))
return IMAGE_FORMAT_BGRX8888;
if (IsFrameBufferFormatValid( displayAdapter, deviceType, d3dDisplayFormat, D3DFMT_A8R8G8B8, bIsWindowed ))
return IMAGE_FORMAT_BGRA8888;
return IMAGE_FORMAT_UNKNOWN;
case IMAGE_FORMAT_BGRA5551:
if (IsFrameBufferFormatValid( displayAdapter, deviceType, d3dDisplayFormat, D3DFMT_A1R5G5B5, bIsWindowed ))
return IMAGE_FORMAT_BGRA5551;
if (IsFrameBufferFormatValid( displayAdapter, deviceType, d3dDisplayFormat, D3DFMT_X1R5G5B5, bIsWindowed ))
return IMAGE_FORMAT_BGRX5551;
if (IsFrameBufferFormatValid( displayAdapter, deviceType, d3dDisplayFormat, D3DFMT_R5G6B5, bIsWindowed ))
return IMAGE_FORMAT_BGR565;
if (IsFrameBufferFormatValid( displayAdapter, deviceType, d3dDisplayFormat, D3DFMT_A8R8G8B8, bIsWindowed ))
return IMAGE_FORMAT_BGRA8888;
if (IsFrameBufferFormatValid( displayAdapter, deviceType, d3dDisplayFormat, D3DFMT_X8R8G8B8, bIsWindowed ))
return IMAGE_FORMAT_BGRX8888;
return IMAGE_FORMAT_UNKNOWN;
}
return IMAGE_FORMAT_UNKNOWN;
}
#if defined( _X360 )
const char *D3DFormatName( D3DFORMAT d3dFormat )
{
if ( IS_D3DFORMAT_SRGB( d3dFormat ) )
{
// sanitize the format from possible sRGB state for comparison purposes
d3dFormat = MAKE_NON_SRGB_FMT( d3dFormat );
}
switch ( d3dFormat )
{
case D3DFMT_A8R8G8B8:
return "D3DFMT_A8R8G8B8";
case D3DFMT_LIN_A8R8G8B8:
return "D3DFMT_LIN_A8R8G8B8";
case D3DFMT_X8R8G8B8:
return "D3DFMT_X8R8G8B8";
case D3DFMT_LIN_X8R8G8B8:
return "D3DFMT_LIN_X8R8G8B8";
case D3DFMT_R5G6B5:
return "D3DFMT_R5G6B5";
case D3DFMT_X1R5G5B5:
return "D3DFMT_X1R5G5B5";
case D3DFMT_A1R5G5B5:
return "D3DFMT_A1R5G5B5";
case D3DFMT_A4R4G4B4:
return "D3DFMT_A4R4G4B4";
case D3DFMT_L8:
return "D3DFMT_L8";
case D3DFMT_A8L8:
return "D3DFMT_A8L8";
case D3DFMT_A8:
return "D3DFMT_A8";
case D3DFMT_DXT1:
return "D3DFMT_DXT1";
case D3DFMT_DXT3:
return "D3DFMT_DXT3";
case D3DFMT_DXT5:
return "D3DFMT_DXT5";
case D3DFMT_V8U8:
return "D3DFMT_V8U8";
case D3DFMT_Q8W8V8U8:
return "D3DFMT_Q8W8V8U8";
case D3DFMT_D16:
return "D3DFMT_D16";
case D3DFMT_D24S8:
return "D3DFMT_D24S8";
case D3DFMT_D24FS8:
return "D3DFMT_D24FS8";
case D3DFMT_LIN_D24S8:
return "D3DFMT_LIN_D24S8";
case D3DFMT_A16B16G16R16:
return "D3DFMT_A16B16G16R16";
case D3DFMT_LIN_A16B16G16R16:
return "D3DFMT_LIN_A16B16G16R16";
}
return "???";
}
#endif