2013-04-17 23:29:41 -04:00
// Copyright 2013 Dolphin Emulator Project
// Licensed under GPLv2
// Refer to the license.txt file included.
2010-06-13 19:50:06 +00:00
2011-01-25 16:43:08 +00:00
# include "StringUtil.h"
# include "VideoConfig.h"
2010-06-13 19:50:06 +00:00
# include "D3DBase.h"
# include "D3DTexture.h"
2011-01-25 16:43:08 +00:00
# include "GfxState.h"
2010-06-13 19:50:06 +00:00
2011-01-29 20:16:51 +00:00
namespace DX11
{
2011-02-26 23:41:02 +00:00
HINSTANCE hD3DCompilerDll = NULL ;
D3DREFLECT PD3DReflect = NULL ;
int d3dcompiler_dll_ref = 0 ;
2010-06-27 14:04:49 +00:00
HINSTANCE hD3DXDll = NULL ;
2010-06-27 16:08:54 +00:00
D3DX11COMPILEFROMMEMORYTYPE PD3DX11CompileFromMemory = NULL ;
2010-06-27 14:04:49 +00:00
D3DX11FILTERTEXTURETYPE PD3DX11FilterTexture = NULL ;
D3DX11SAVETEXTURETOFILEATYPE PD3DX11SaveTextureToFileA = NULL ;
D3DX11SAVETEXTURETOFILEWTYPE PD3DX11SaveTextureToFileW = NULL ;
2010-12-29 13:07:00 +00:00
int d3dx_dll_ref = 0 ;
2010-06-27 14:04:49 +00:00
2010-11-21 15:34:04 +00:00
CREATEDXGIFACTORY PCreateDXGIFactory = NULL ;
HINSTANCE hDXGIDll = NULL ;
2010-12-29 13:07:00 +00:00
int dxgi_dll_ref = 0 ;
2010-11-21 15:34:04 +00:00
typedef HRESULT ( WINAPI * D3D11CREATEDEVICEANDSWAPCHAIN ) ( IDXGIAdapter * , D3D_DRIVER_TYPE , HMODULE , UINT , CONST D3D_FEATURE_LEVEL * , UINT , UINT , CONST DXGI_SWAP_CHAIN_DESC * , IDXGISwapChain * * , ID3D11Device * * , D3D_FEATURE_LEVEL * , ID3D11DeviceContext * * ) ;
2010-11-23 19:58:02 +00:00
D3D11CREATEDEVICE PD3D11CreateDevice = NULL ;
2010-11-21 15:34:04 +00:00
D3D11CREATEDEVICEANDSWAPCHAIN PD3D11CreateDeviceAndSwapChain = NULL ;
2011-06-11 19:37:21 +00:00
HINSTANCE hD3DDll = NULL ;
2010-12-29 13:07:00 +00:00
int d3d_dll_ref = 0 ;
2010-11-21 15:34:04 +00:00
2010-06-13 19:50:06 +00:00
namespace D3D
{
2011-06-11 19:37:21 +00:00
ID3D11Device * device = NULL ;
ID3D11DeviceContext * context = NULL ;
IDXGISwapChain * swapchain = NULL ;
2010-06-13 19:50:06 +00:00
D3D_FEATURE_LEVEL featlevel ;
2011-06-11 19:37:21 +00:00
D3DTexture2D * backbuf = NULL ;
2010-06-13 19:50:06 +00:00
HWND hWnd ;
2011-06-11 19:37:21 +00:00
std : : vector < DXGI_SAMPLE_DESC > aa_modes ; // supported AA modes of the current adapter
2010-11-23 19:58:02 +00:00
2010-06-19 01:02:43 +00:00
bool bgra_textures_supported ;
2010-06-14 22:38:47 +00:00
# define NUM_SUPPORTED_FEATURE_LEVELS 3
const D3D_FEATURE_LEVEL supported_feature_levels [ NUM_SUPPORTED_FEATURE_LEVELS ] = {
D3D_FEATURE_LEVEL_11_0 ,
D3D_FEATURE_LEVEL_10_1 ,
D3D_FEATURE_LEVEL_10_0
} ;
2010-06-13 19:50:06 +00:00
unsigned int xres , yres ;
bool bFrameInProgress = false ;
2010-11-23 19:58:02 +00:00
HRESULT LoadDXGI ( )
2010-06-13 19:50:06 +00:00
{
2010-12-29 13:07:00 +00:00
if ( dxgi_dll_ref + + > 0 ) return S_OK ;
2010-11-23 19:58:02 +00:00
if ( hDXGIDll ) return S_OK ;
2010-11-21 15:34:04 +00:00
hDXGIDll = LoadLibraryA ( " dxgi.dll " ) ;
if ( ! hDXGIDll )
{
MessageBoxA ( NULL , " Failed to load dxgi.dll " , " Critical error " , MB_OK | MB_ICONERROR ) ;
2010-12-29 13:07:00 +00:00
- - dxgi_dll_ref ;
2010-11-21 15:34:04 +00:00
return E_FAIL ;
}
PCreateDXGIFactory = ( CREATEDXGIFACTORY ) GetProcAddress ( hDXGIDll , " CreateDXGIFactory " ) ;
if ( PCreateDXGIFactory = = NULL ) MessageBoxA ( NULL , " GetProcAddress failed for CreateDXGIFactory! " , " Critical error " , MB_OK | MB_ICONERROR ) ;
2010-06-13 19:50:06 +00:00
2010-11-21 15:34:04 +00:00
return S_OK ;
}
2010-11-23 19:58:02 +00:00
HRESULT LoadD3D ( )
2010-11-21 15:34:04 +00:00
{
2011-06-11 19:37:21 +00:00
if ( d3d_dll_ref + + > 0 ) return S_OK ;
2010-12-29 13:07:00 +00:00
2011-06-11 19:37:21 +00:00
if ( hD3DDll ) return S_OK ;
hD3DDll = LoadLibraryA ( " d3d11.dll " ) ;
if ( ! hD3DDll )
2010-11-23 19:58:02 +00:00
{
MessageBoxA ( NULL , " Failed to load d3d11.dll " , " Critical error " , MB_OK | MB_ICONERROR ) ;
2010-12-29 13:07:00 +00:00
- - d3d_dll_ref ;
2010-11-23 19:58:02 +00:00
return E_FAIL ;
}
2011-06-11 19:37:21 +00:00
PD3D11CreateDevice = ( D3D11CREATEDEVICE ) GetProcAddress ( hD3DDll , " D3D11CreateDevice " ) ;
if ( PD3D11CreateDevice = = NULL ) MessageBoxA ( NULL , " GetProcAddress failed for D3D11CreateDevice! " , " Critical error " , MB_OK | MB_ICONERROR ) ;
2010-11-23 19:58:02 +00:00
2011-06-11 19:37:21 +00:00
PD3D11CreateDeviceAndSwapChain = ( D3D11CREATEDEVICEANDSWAPCHAIN ) GetProcAddress ( hD3DDll , " D3D11CreateDeviceAndSwapChain " ) ;
if ( PD3D11CreateDeviceAndSwapChain = = NULL ) MessageBoxA ( NULL , " GetProcAddress failed for D3D11CreateDeviceAndSwapChain! " , " Critical error " , MB_OK | MB_ICONERROR ) ;
2010-11-23 19:58:02 +00:00
return S_OK ;
2010-11-21 15:34:04 +00:00
}
2010-06-13 19:50:06 +00:00
2010-11-23 19:58:02 +00:00
HRESULT LoadD3DX ( )
2010-11-21 15:34:04 +00:00
{
2010-12-29 13:07:00 +00:00
if ( d3dx_dll_ref + + > 0 ) return S_OK ;
2010-11-23 19:58:02 +00:00
if ( hD3DXDll ) return S_OK ;
2010-06-27 14:04:49 +00:00
// try to load D3DX11 first to check whether we have proper runtime support
2011-02-14 02:18:03 +00:00
// try to use the dll the backend was compiled against first - don't bother about debug runtimes
2011-02-26 23:41:02 +00:00
hD3DXDll = LoadLibraryA ( D3DX11_DLL_A ) ;
2010-06-27 14:04:49 +00:00
if ( ! hD3DXDll )
{
// if that fails, use the dll which should be available in every SDK which officially supports DX11.
hD3DXDll = LoadLibraryA ( " d3dx11_42.dll " ) ;
if ( ! hD3DXDll )
{
MessageBoxA ( NULL , " Failed to load d3dx11_42.dll, update your DX11 runtime, please " , " Critical error " , MB_OK | MB_ICONERROR ) ;
return E_FAIL ;
}
else
{
NOTICE_LOG ( VIDEO , " Successfully loaded d3dx11_42.dll. If you're having trouble, try updating your DX runtime first. " ) ;
}
}
2010-06-27 16:08:54 +00:00
PD3DX11CompileFromMemory = ( D3DX11COMPILEFROMMEMORYTYPE ) GetProcAddress ( hD3DXDll , " D3DX11CompileFromMemory " ) ;
if ( PD3DX11CompileFromMemory = = NULL ) MessageBoxA ( NULL , " GetProcAddress failed for D3DX11CompileFromMemory! " , " Critical error " , MB_OK | MB_ICONERROR ) ;
2010-06-27 14:04:49 +00:00
PD3DX11FilterTexture = ( D3DX11FILTERTEXTURETYPE ) GetProcAddress ( hD3DXDll , " D3DX11FilterTexture " ) ;
2010-06-27 16:08:54 +00:00
if ( PD3DX11FilterTexture = = NULL ) MessageBoxA ( NULL , " GetProcAddress failed for D3DX11FilterTexture! " , " Critical error " , MB_OK | MB_ICONERROR ) ;
2010-06-27 14:04:49 +00:00
PD3DX11SaveTextureToFileA = ( D3DX11SAVETEXTURETOFILEATYPE ) GetProcAddress ( hD3DXDll , " D3DX11SaveTextureToFileA " ) ;
2010-06-27 16:08:54 +00:00
if ( PD3DX11SaveTextureToFileA = = NULL ) MessageBoxA ( NULL , " GetProcAddress failed for D3DX11SaveTextureToFileA! " , " Critical error " , MB_OK | MB_ICONERROR ) ;
2010-06-27 14:04:49 +00:00
PD3DX11SaveTextureToFileW = ( D3DX11SAVETEXTURETOFILEWTYPE ) GetProcAddress ( hD3DXDll , " D3DX11SaveTextureToFileW " ) ;
2010-06-27 16:08:54 +00:00
if ( PD3DX11SaveTextureToFileW = = NULL ) MessageBoxA ( NULL , " GetProcAddress failed for D3DX11SaveTextureToFileW! " , " Critical error " , MB_OK | MB_ICONERROR ) ;
2010-06-27 14:04:49 +00:00
2010-11-23 19:58:02 +00:00
return S_OK ;
}
2010-11-21 15:34:04 +00:00
2011-02-26 23:41:02 +00:00
HRESULT LoadD3DCompiler ( )
{
if ( d3dcompiler_dll_ref + + > 0 ) return S_OK ;
if ( hD3DCompilerDll ) return S_OK ;
// try to load D3DCompiler first to check whether we have proper runtime support
// try to use the dll the backend was compiled against first - don't bother about debug runtimes
hD3DCompilerDll = LoadLibraryA ( D3DCOMPILER_DLL_A ) ;
if ( ! hD3DCompilerDll )
{
// if that fails, use the dll which should be available in every SDK which officially supports DX11.
hD3DCompilerDll = LoadLibraryA ( " D3DCompiler_42.dll " ) ;
if ( ! hD3DCompilerDll )
{
MessageBoxA ( NULL , " Failed to load D3DCompiler_42.dll, update your DX11 runtime, please " , " Critical error " , MB_OK | MB_ICONERROR ) ;
return E_FAIL ;
}
else
{
NOTICE_LOG ( VIDEO , " Successfully loaded D3DCompiler_42.dll. If you're having trouble, try updating your DX runtime first. " ) ;
}
}
PD3DReflect = ( D3DREFLECT ) GetProcAddress ( hD3DCompilerDll , " D3DReflect " ) ;
if ( PD3DReflect = = NULL ) MessageBoxA ( NULL , " GetProcAddress failed for D3DReflect! " , " Critical error " , MB_OK | MB_ICONERROR ) ;
return S_OK ;
}
2010-11-23 19:58:02 +00:00
void UnloadDXGI ( )
{
2010-12-29 13:07:00 +00:00
if ( ! dxgi_dll_ref ) return ;
if ( - - dxgi_dll_ref ! = 0 ) return ;
2010-11-23 19:58:02 +00:00
if ( hDXGIDll ) FreeLibrary ( hDXGIDll ) ;
hDXGIDll = NULL ;
PCreateDXGIFactory = NULL ;
}
void UnloadD3DX ( )
{
2010-12-29 13:07:00 +00:00
if ( ! d3dx_dll_ref ) return ;
if ( - - d3dx_dll_ref ! = 0 ) return ;
2010-11-23 19:58:02 +00:00
if ( hD3DXDll ) FreeLibrary ( hD3DXDll ) ;
hD3DXDll = NULL ;
PD3DX11FilterTexture = NULL ;
PD3DX11SaveTextureToFileA = NULL ;
PD3DX11SaveTextureToFileW = NULL ;
}
void UnloadD3D ( )
{
2011-06-11 19:37:21 +00:00
if ( ! d3d_dll_ref ) return ;
if ( - - d3d_dll_ref ! = 0 ) return ;
2010-12-29 13:07:00 +00:00
2011-06-11 19:37:21 +00:00
if ( hD3DDll ) FreeLibrary ( hD3DDll ) ;
hD3DDll = NULL ;
2010-11-23 19:58:02 +00:00
PD3D11CreateDevice = NULL ;
PD3D11CreateDeviceAndSwapChain = NULL ;
}
2011-02-26 23:41:02 +00:00
void UnloadD3DCompiler ( )
{
if ( ! d3dcompiler_dll_ref ) return ;
if ( - - d3dcompiler_dll_ref ! = 0 ) return ;
if ( hD3DCompilerDll ) FreeLibrary ( hD3DCompilerDll ) ;
hD3DCompilerDll = NULL ;
PD3DReflect = NULL ;
}
2011-03-30 07:17:23 +00:00
std : : vector < DXGI_SAMPLE_DESC > EnumAAModes ( IDXGIAdapter * adapter )
2010-11-23 19:58:02 +00:00
{
2011-06-11 19:37:21 +00:00
std : : vector < DXGI_SAMPLE_DESC > aa_modes ;
2010-11-28 15:23:51 +00:00
// NOTE: D3D 10.0 doesn't support multisampled resources which are bound as depth buffers AND shader resources.
// Thus, we can't have MSAA with 10.0 level hardware.
2011-06-11 19:37:21 +00:00
ID3D11Device * device ;
ID3D11DeviceContext * context ;
2010-11-23 19:58:02 +00:00
D3D_FEATURE_LEVEL feat_level ;
2011-06-11 19:37:21 +00:00
HRESULT hr = PD3D11CreateDevice ( adapter , D3D_DRIVER_TYPE_UNKNOWN , NULL , D3D11_CREATE_DEVICE_SINGLETHREADED , supported_feature_levels , NUM_SUPPORTED_FEATURE_LEVELS , D3D11_SDK_VERSION , & device , & feat_level , & context ) ;
2010-11-28 15:23:51 +00:00
if ( FAILED ( hr ) | | feat_level = = D3D_FEATURE_LEVEL_10_0 )
{
2010-11-28 17:25:19 +00:00
DXGI_SAMPLE_DESC desc ;
desc . Count = 1 ;
desc . Quality = 0 ;
aa_modes . push_back ( desc ) ;
2011-06-11 19:37:21 +00:00
SAFE_RELEASE ( context ) ;
SAFE_RELEASE ( device ) ;
2010-11-28 15:23:51 +00:00
}
2011-03-30 07:17:23 +00:00
else
2010-11-21 15:34:04 +00:00
{
2011-06-11 19:37:21 +00:00
for ( int samples = 0 ; samples < D3D11_MAX_MULTISAMPLE_SAMPLE_COUNT ; + + samples )
2011-03-30 07:17:23 +00:00
{
UINT quality_levels = 0 ;
device - > CheckMultisampleQualityLevels ( DXGI_FORMAT_R8G8B8A8_UNORM , samples , & quality_levels ) ;
2011-06-11 19:37:21 +00:00
if ( quality_levels > 0 ) {
2011-03-30 07:17:23 +00:00
DXGI_SAMPLE_DESC desc ;
desc . Count = samples ;
2011-06-11 19:37:21 +00:00
for ( desc . Quality = 0 ; desc . Quality < quality_levels ; + + desc . Quality )
2011-03-30 07:17:23 +00:00
aa_modes . push_back ( desc ) ;
}
2010-11-23 19:58:02 +00:00
}
2011-06-11 19:37:21 +00:00
context - > Release ( ) ;
device - > Release ( ) ;
2010-11-21 15:34:04 +00:00
}
2011-03-30 07:17:23 +00:00
return aa_modes ;
2010-11-23 19:58:02 +00:00
}
DXGI_SAMPLE_DESC GetAAMode ( int index )
{
2011-06-11 19:37:21 +00:00
return aa_modes [ index ] ;
2010-11-21 15:34:04 +00:00
}
HRESULT Create ( HWND wnd )
{
hWnd = wnd ;
HRESULT hr ;
RECT client ;
GetClientRect ( hWnd , & client ) ;
xres = client . right - client . left ;
yres = client . bottom - client . top ;
2010-11-23 19:58:02 +00:00
hr = LoadDXGI ( ) ;
if ( SUCCEEDED ( hr ) ) hr = LoadD3D ( ) ;
if ( SUCCEEDED ( hr ) ) hr = LoadD3DX ( ) ;
2011-02-26 23:41:02 +00:00
if ( SUCCEEDED ( hr ) ) hr = LoadD3DCompiler ( ) ;
2010-11-23 19:58:02 +00:00
if ( FAILED ( hr ) )
{
UnloadDXGI ( ) ;
UnloadD3D ( ) ;
2011-02-26 23:41:02 +00:00
UnloadD3DX ( ) ;
UnloadD3DCompiler ( ) ;
2010-11-23 19:58:02 +00:00
return hr ;
}
2010-11-21 15:34:04 +00:00
2010-06-14 19:20:41 +00:00
IDXGIFactory * factory ;
2011-06-11 19:37:21 +00:00
IDXGIAdapter * adapter ;
IDXGIOutput * output ;
2010-11-21 15:34:04 +00:00
hr = PCreateDXGIFactory ( __uuidof ( IDXGIFactory ) , ( void * * ) & factory ) ;
2011-06-11 19:37:21 +00:00
if ( FAILED ( hr ) ) MessageBox ( wnd , _T ( " Failed to create IDXGIFactory object " ) , _T ( " Dolphin Direct3D 11 backend " ) , MB_OK | MB_ICONERROR ) ;
2010-06-14 19:20:41 +00:00
hr = factory - > EnumAdapters ( g_ActiveConfig . iAdapter , & adapter ) ;
if ( FAILED ( hr ) )
{
// try using the first one
hr = factory - > EnumAdapters ( 0 , & adapter ) ;
2011-06-11 19:37:21 +00:00
if ( FAILED ( hr ) ) MessageBox ( wnd , _T ( " Failed to enumerate adapters " ) , _T ( " Dolphin Direct3D 11 backend " ) , MB_OK | MB_ICONERROR ) ;
2010-06-14 19:20:41 +00:00
}
// TODO: Make this configurable
hr = adapter - > EnumOutputs ( 0 , & output ) ;
if ( FAILED ( hr ) )
{
// try using the first one
hr = adapter - > EnumOutputs ( 0 , & output ) ;
2011-06-11 19:37:21 +00:00
if ( FAILED ( hr ) ) MessageBox ( wnd , _T ( " Failed to enumerate outputs " ) , _T ( " Dolphin Direct3D 11 backend " ) , MB_OK | MB_ICONERROR ) ;
2010-06-14 19:20:41 +00:00
}
2010-11-23 19:58:02 +00:00
// get supported AA modes
2011-06-11 19:37:21 +00:00
aa_modes = EnumAAModes ( adapter ) ;
if ( g_Config . iMultisampleMode > = ( int ) aa_modes . size ( ) )
2010-11-28 15:23:51 +00:00
{
g_Config . iMultisampleMode = 0 ;
UpdateActiveConfig ( ) ;
}
2010-11-23 19:58:02 +00:00
2010-06-14 19:20:41 +00:00
DXGI_SWAP_CHAIN_DESC swap_chain_desc ;
memset ( & swap_chain_desc , 0 , sizeof ( swap_chain_desc ) ) ;
swap_chain_desc . BufferCount = 1 ;
swap_chain_desc . BufferUsage = DXGI_USAGE_RENDER_TARGET_OUTPUT ;
swap_chain_desc . OutputWindow = wnd ;
swap_chain_desc . SampleDesc . Count = 1 ;
swap_chain_desc . SampleDesc . Quality = 0 ;
swap_chain_desc . Windowed = TRUE ;
DXGI_MODE_DESC mode_desc ;
memset ( & mode_desc , 0 , sizeof ( mode_desc ) ) ;
mode_desc . Width = xres ;
mode_desc . Height = yres ;
mode_desc . Format = DXGI_FORMAT_R8G8B8A8_UNORM ;
mode_desc . Scaling = DXGI_MODE_SCALING_UNSPECIFIED ;
hr = output - > FindClosestMatchingMode ( & mode_desc , & swap_chain_desc . BufferDesc , NULL ) ;
2011-06-11 19:37:21 +00:00
if ( FAILED ( hr ) ) MessageBox ( wnd , _T ( " Failed to find a supported video mode " ) , _T ( " Dolphin Direct3D 11 backend " ) , MB_OK | MB_ICONERROR ) ;
2010-06-14 21:31:23 +00:00
// forcing buffer resolution to xres and yres.. TODO: The new video mode might not actually be supported!
swap_chain_desc . BufferDesc . Width = xres ;
swap_chain_desc . BufferDesc . Height = yres ;
2010-06-13 19:50:06 +00:00
# if defined(_DEBUG) || defined(DEBUGFAST)
2011-06-11 19:37:21 +00:00
D3D11_CREATE_DEVICE_FLAG device_flags = ( D3D11_CREATE_DEVICE_FLAG ) ( D3D11_CREATE_DEVICE_DEBUG | D3D11_CREATE_DEVICE_SINGLETHREADED ) ;
2010-06-13 19:50:06 +00:00
# else
2011-06-11 19:37:21 +00:00
D3D11_CREATE_DEVICE_FLAG device_flags = D3D11_CREATE_DEVICE_SINGLETHREADED ;
2010-06-13 19:50:06 +00:00
# endif
2011-06-11 19:37:21 +00:00
hr = PD3D11CreateDeviceAndSwapChain ( adapter , D3D_DRIVER_TYPE_UNKNOWN , NULL , device_flags ,
supported_feature_levels , NUM_SUPPORTED_FEATURE_LEVELS ,
D3D11_SDK_VERSION , & swap_chain_desc , & swapchain , & device ,
& featlevel , & context ) ;
if ( FAILED ( hr ) | | ! device | | ! context | | ! swapchain )
2010-06-13 19:50:06 +00:00
{
2011-06-11 19:37:21 +00:00
MessageBox ( wnd , _T ( " Failed to initialize Direct3D. \n Make sure your video card supports at least D3D 10.0 " ) , _T ( " Dolphin Direct3D 11 backend " ) , MB_OK | MB_ICONERROR ) ;
SAFE_RELEASE ( device ) ;
SAFE_RELEASE ( context ) ;
SAFE_RELEASE ( swapchain ) ;
2010-06-13 19:50:06 +00:00
return E_FAIL ;
}
2011-06-11 19:37:21 +00:00
SetDebugObjectName ( ( ID3D11DeviceChild * ) context , " device context " ) ;
2010-07-17 11:42:28 +00:00
SAFE_RELEASE ( factory ) ;
SAFE_RELEASE ( output ) ;
SAFE_RELEASE ( adapter ) ;
2010-06-13 19:50:06 +00:00
2011-06-11 19:37:21 +00:00
ID3D11Texture2D * buf ;
hr = swapchain - > GetBuffer ( 0 , IID_ID3D11Texture2D , ( void * * ) & buf ) ;
if ( FAILED ( hr ) )
2010-06-13 19:50:06 +00:00
{
2011-02-14 02:18:03 +00:00
MessageBox ( wnd , _T ( " Failed to get swapchain buffer " ) , _T ( " Dolphin Direct3D 11 backend " ) , MB_OK | MB_ICONERROR ) ;
2011-06-11 19:37:21 +00:00
SAFE_RELEASE ( device ) ;
SAFE_RELEASE ( context ) ;
SAFE_RELEASE ( swapchain ) ;
2010-06-13 19:50:06 +00:00
return E_FAIL ;
}
2011-06-11 19:37:21 +00:00
backbuf = new D3DTexture2D ( buf , D3D11_BIND_RENDER_TARGET ) ;
SAFE_RELEASE ( buf ) ;
2010-06-18 18:40:58 +00:00
CHECK ( backbuf ! = NULL , " Create back buffer texture " ) ;
2011-06-11 19:37:21 +00:00
SetDebugObjectName ( ( ID3D11DeviceChild * ) backbuf - > GetTex ( ) , " backbuffer texture " ) ;
SetDebugObjectName ( ( ID3D11DeviceChild * ) backbuf - > GetRTV ( ) , " backbuffer render target view " ) ;
2010-06-13 19:50:06 +00:00
context - > OMSetRenderTargets ( 1 , & backbuf - > GetRTV ( ) , NULL ) ;
2010-06-19 01:02:43 +00:00
// BGRA textures are easier to deal with in TextureCache, but might not be supported by the hardware
UINT format_support ;
device - > CheckFormatSupport ( DXGI_FORMAT_B8G8R8A8_UNORM , & format_support ) ;
bgra_textures_supported = ( format_support & D3D11_FORMAT_SUPPORT_TEXTURE2D ) ! = 0 ;
2011-06-11 19:37:21 +00:00
stateman = new StateManager ;
2010-06-13 19:50:06 +00:00
return S_OK ;
}
void Close ( )
{
// release all bound resources
2011-06-11 19:37:21 +00:00
context - > ClearState ( ) ;
SAFE_RELEASE ( backbuf ) ;
SAFE_RELEASE ( swapchain ) ;
SAFE_DELETE ( stateman ) ;
context - > Flush ( ) ; // immediately destroy device objects
SAFE_RELEASE ( context ) ;
ULONG references = device - > Release ( ) ;
if ( references )
{
ERROR_LOG ( VIDEO , " Unreleased references: %i. " , references ) ;
}
else
{
NOTICE_LOG ( VIDEO , " Successfully released all device references! " ) ;
}
device = NULL ;
2010-11-21 15:34:04 +00:00
// unload DLLs
2010-11-23 19:58:02 +00:00
UnloadD3DX ( ) ;
UnloadD3D ( ) ;
2010-11-21 15:34:04 +00:00
UnloadDXGI ( ) ;
2010-06-13 19:50:06 +00:00
}
2010-11-27 11:11:05 +00:00
const char * VertexShaderVersionString ( )
{
if ( featlevel = = D3D_FEATURE_LEVEL_11_0 ) return " vs_5_0 " ;
else if ( featlevel = = D3D_FEATURE_LEVEL_10_1 ) return " vs_4_1 " ;
else /*if(featlevel == D3D_FEATURE_LEVEL_10_0)*/ return " vs_4_0 " ;
}
2011-03-14 09:38:29 +00:00
const char * GeometryShaderVersionString ( )
{
if ( featlevel = = D3D_FEATURE_LEVEL_11_0 ) return " gs_5_0 " ;
else if ( featlevel = = D3D_FEATURE_LEVEL_10_1 ) return " gs_4_1 " ;
else /*if(featlevel == D3D_FEATURE_LEVEL_10_0)*/ return " gs_4_0 " ;
}
2010-11-27 11:11:05 +00:00
const char * PixelShaderVersionString ( )
{
if ( featlevel = = D3D_FEATURE_LEVEL_11_0 ) return " ps_5_0 " ;
else if ( featlevel = = D3D_FEATURE_LEVEL_10_1 ) return " ps_4_1 " ;
else /*if(featlevel == D3D_FEATURE_LEVEL_10_0)*/ return " ps_4_0 " ;
}
2010-06-13 19:50:06 +00:00
2011-06-11 19:37:21 +00:00
D3DTexture2D * & GetBackBuffer ( ) { return backbuf ; }
2010-06-13 19:50:06 +00:00
unsigned int GetBackBufferWidth ( ) { return xres ; }
unsigned int GetBackBufferHeight ( ) { return yres ; }
2010-06-19 01:02:43 +00:00
bool BGRATexturesSupported ( ) { return bgra_textures_supported ; }
2010-07-11 16:26:46 +00:00
// Returns the maximum width/height of a texture. This value only depends upon the feature level in DX11
unsigned int GetMaxTextureSize ( )
{
switch ( featlevel )
{
case D3D_FEATURE_LEVEL_11_0 :
2010-11-27 11:11:05 +00:00
return D3D11_REQ_TEXTURE2D_U_OR_V_DIMENSION ;
2010-07-11 16:26:46 +00:00
case D3D_FEATURE_LEVEL_10_1 :
case D3D_FEATURE_LEVEL_10_0 :
2010-11-27 11:11:05 +00:00
return D3D10_REQ_TEXTURE2D_U_OR_V_DIMENSION ;
2010-07-11 16:26:46 +00:00
case D3D_FEATURE_LEVEL_9_3 :
return 4096 ;
case D3D_FEATURE_LEVEL_9_2 :
case D3D_FEATURE_LEVEL_9_1 :
return 2048 ;
2010-07-17 11:42:28 +00:00
default :
return 0 ;
2010-07-11 16:26:46 +00:00
}
}
2010-06-13 19:50:06 +00:00
void Reset ( )
{
2010-06-17 10:42:57 +00:00
// release all back buffer references
2011-06-11 19:37:21 +00:00
SAFE_RELEASE ( backbuf ) ;
2010-06-17 10:42:57 +00:00
// resize swapchain buffers
RECT client ;
GetClientRect ( hWnd , & client ) ;
xres = client . right - client . left ;
yres = client . bottom - client . top ;
2011-06-11 19:37:21 +00:00
D3D : : swapchain - > ResizeBuffers ( 1 , xres , yres , DXGI_FORMAT_R8G8B8A8_UNORM , 0 ) ;
2010-06-17 10:42:57 +00:00
// recreate back buffer texture
2011-06-11 19:37:21 +00:00
ID3D11Texture2D * buf ;
HRESULT hr = swapchain - > GetBuffer ( 0 , IID_ID3D11Texture2D , ( void * * ) & buf ) ;
if ( FAILED ( hr ) )
2010-06-17 10:42:57 +00:00
{
2011-02-14 02:18:03 +00:00
MessageBox ( hWnd , _T ( " Failed to get swapchain buffer " ) , _T ( " Dolphin Direct3D 11 backend " ) , MB_OK | MB_ICONERROR ) ;
2011-06-11 19:37:21 +00:00
SAFE_RELEASE ( device ) ;
SAFE_RELEASE ( context ) ;
SAFE_RELEASE ( swapchain ) ;
2010-06-17 10:42:57 +00:00
return ;
}
2011-06-11 19:37:21 +00:00
backbuf = new D3DTexture2D ( buf , D3D11_BIND_RENDER_TARGET ) ;
SAFE_RELEASE ( buf ) ;
2010-06-18 18:40:58 +00:00
CHECK ( backbuf ! = NULL , " Create back buffer texture " ) ;
2011-06-11 19:37:21 +00:00
SetDebugObjectName ( ( ID3D11DeviceChild * ) backbuf - > GetTex ( ) , " backbuffer texture " ) ;
SetDebugObjectName ( ( ID3D11DeviceChild * ) backbuf - > GetRTV ( ) , " backbuffer render target view " ) ;
2010-06-13 19:50:06 +00:00
}
bool BeginFrame ( )
{
if ( bFrameInProgress )
{
PanicAlert ( " BeginFrame called although a frame is already in progress " ) ;
return false ;
}
bFrameInProgress = true ;
2011-06-11 19:37:21 +00:00
return ( device ! = NULL ) ;
2010-06-13 19:50:06 +00:00
}
void EndFrame ( )
{
if ( ! bFrameInProgress )
{
PanicAlert ( " EndFrame called although no frame is in progress " ) ;
return ;
}
bFrameInProgress = false ;
}
void Present ( )
{
// TODO: Is 1 the correct value for vsyncing?
2013-03-18 20:41:45 -04:00
swapchain - > Present ( ( UINT ) g_ActiveConfig . IsVSync ( ) , 0 ) ;
2010-06-13 19:50:06 +00:00
}
2011-01-29 20:16:51 +00:00
} // namespace D3D
2011-02-14 02:18:03 +00:00
} // namespace DX11