2015-05-24 06:55:12 +02:00
|
|
|
// Copyright 2010 Dolphin Emulator Project
|
2015-05-18 01:08:10 +02:00
|
|
|
// Licensed under GPLv2+
|
2013-04-17 23:29:41 -04:00
|
|
|
// Refer to the license.txt file included.
|
2010-06-13 19:50:06 +00:00
|
|
|
|
2019-06-01 07:55:09 -04:00
|
|
|
#include "VideoBackends/D3D/D3DBase.h"
|
|
|
|
|
2015-12-12 13:00:08 +01:00
|
|
|
#include <algorithm>
|
2019-06-01 07:55:09 -04:00
|
|
|
#include <array>
|
2015-12-12 13:00:08 +01:00
|
|
|
|
2016-01-17 16:54:31 -05:00
|
|
|
#include "Common/CommonTypes.h"
|
2019-03-09 23:31:36 +10:00
|
|
|
#include "Common/DynamicLibrary.h"
|
2016-06-24 10:43:46 +02:00
|
|
|
#include "Common/Logging/Log.h"
|
2016-01-17 16:54:31 -05:00
|
|
|
#include "Common/MsgHandler.h"
|
2017-05-18 13:59:38 +01:00
|
|
|
#include "Core/Config/GraphicsSettings.h"
|
2016-11-09 01:41:38 +01:00
|
|
|
#include "Core/ConfigManager.h"
|
2014-06-23 08:11:07 +02:00
|
|
|
#include "VideoBackends/D3D/D3DState.h"
|
2019-02-15 11:59:50 +10:00
|
|
|
#include "VideoBackends/D3D/DXTexture.h"
|
2019-03-09 23:31:36 +10:00
|
|
|
#include "VideoBackends/D3DCommon/Common.h"
|
2014-02-19 12:14:09 +01:00
|
|
|
#include "VideoCommon/VideoConfig.h"
|
2010-06-13 19:50:06 +00:00
|
|
|
|
2011-01-29 20:16:51 +00:00
|
|
|
namespace DX11
|
|
|
|
{
|
2019-03-09 23:31:36 +10:00
|
|
|
static Common::DynamicLibrary s_d3d11_library;
|
2010-06-13 19:50:06 +00:00
|
|
|
namespace D3D
|
|
|
|
{
|
2019-05-12 14:42:16 +10:00
|
|
|
ComPtr<IDXGIFactory> dxgi_factory;
|
2019-03-09 23:31:36 +10:00
|
|
|
ComPtr<ID3D11Device> device;
|
|
|
|
ComPtr<ID3D11Device1> device1;
|
|
|
|
ComPtr<ID3D11DeviceContext> context;
|
|
|
|
D3D_FEATURE_LEVEL feature_level;
|
2010-06-19 01:02:43 +00:00
|
|
|
|
2019-03-09 23:31:36 +10:00
|
|
|
static ComPtr<ID3D11Debug> s_debug;
|
2017-11-19 13:23:56 -05:00
|
|
|
|
2019-06-01 07:55:09 -04:00
|
|
|
constexpr std::array<D3D_FEATURE_LEVEL, 3> s_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
|
|
|
|
2019-03-09 23:31:36 +10:00
|
|
|
bool Create(u32 adapter_index, bool enable_debug_layer)
|
2010-11-21 15:34:04 +00:00
|
|
|
{
|
2019-03-09 23:31:36 +10:00
|
|
|
PFN_D3D11_CREATE_DEVICE d3d11_create_device;
|
|
|
|
if (!s_d3d11_library.Open("d3d11.dll") ||
|
|
|
|
!s_d3d11_library.GetSymbol("D3D11CreateDevice", &d3d11_create_device))
|
2019-02-15 11:59:50 +10:00
|
|
|
{
|
2019-03-09 23:31:36 +10:00
|
|
|
PanicAlertT("Failed to load d3d11.dll");
|
|
|
|
s_d3d11_library.Close();
|
2019-02-15 11:59:50 +10:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-03-09 23:31:36 +10:00
|
|
|
if (!D3DCommon::LoadLibraries())
|
2018-01-26 16:23:24 +10:00
|
|
|
{
|
2019-03-09 23:31:36 +10:00
|
|
|
s_d3d11_library.Close();
|
2018-01-26 16:23:24 +10:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-03-09 23:31:36 +10:00
|
|
|
dxgi_factory = D3DCommon::CreateDXGIFactory(enable_debug_layer);
|
|
|
|
if (!dxgi_factory)
|
2018-01-26 16:23:24 +10:00
|
|
|
{
|
2019-03-09 23:31:36 +10:00
|
|
|
PanicAlertT("Failed to create DXGI factory");
|
|
|
|
D3DCommon::UnloadLibraries();
|
|
|
|
s_d3d11_library.Close();
|
2018-01-26 16:23:24 +10:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-03-09 23:31:36 +10:00
|
|
|
ComPtr<IDXGIAdapter> adapter;
|
|
|
|
HRESULT hr = dxgi_factory->EnumAdapters(adapter_index, &adapter);
|
2016-06-24 10:43:46 +02:00
|
|
|
if (FAILED(hr))
|
|
|
|
{
|
2019-03-09 23:31:36 +10:00
|
|
|
WARN_LOG(VIDEO, "Adapter %u not found, using default", adapter_index);
|
|
|
|
adapter = nullptr;
|
2016-06-24 10:43:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Creating debug devices can sometimes fail if the user doesn't have the correct
|
|
|
|
// version of the DirectX SDK. If it does, simply fallback to a non-debug device.
|
2019-03-09 23:31:36 +10:00
|
|
|
if (enable_debug_layer)
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
2019-03-09 23:31:36 +10:00
|
|
|
hr = d3d11_create_device(adapter.Get(), D3D_DRIVER_TYPE_UNKNOWN, nullptr,
|
2019-06-01 07:55:09 -04:00
|
|
|
D3D11_CREATE_DEVICE_DEBUG, s_supported_feature_levels.data(),
|
|
|
|
static_cast<UINT>(s_supported_feature_levels.size()),
|
2019-03-09 23:31:36 +10:00
|
|
|
D3D11_SDK_VERSION, &device, &feature_level, &context);
|
2017-06-25 20:57:46 +02:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
// Debugbreak on D3D error
|
2019-07-21 00:17:15 +02:00
|
|
|
if (SUCCEEDED(hr) && SUCCEEDED(device.As(&s_debug)))
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
2019-03-09 23:31:36 +10:00
|
|
|
ComPtr<ID3D11InfoQueue> info_queue;
|
2019-07-21 00:17:15 +02:00
|
|
|
if (SUCCEEDED(s_debug.As(&info_queue)))
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
2019-03-09 23:31:36 +10:00
|
|
|
info_queue->SetBreakOnSeverity(D3D11_MESSAGE_SEVERITY_CORRUPTION, true);
|
|
|
|
info_queue->SetBreakOnSeverity(D3D11_MESSAGE_SEVERITY_ERROR, true);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
|
|
|
D3D11_MESSAGE_ID hide[] = {D3D11_MESSAGE_ID_SETPRIVATEDATA_CHANGINGPARAMS};
|
|
|
|
|
|
|
|
D3D11_INFO_QUEUE_FILTER filter = {};
|
|
|
|
filter.DenyList.NumIDs = sizeof(hide) / sizeof(D3D11_MESSAGE_ID);
|
|
|
|
filter.DenyList.pIDList = hide;
|
2019-03-09 23:31:36 +10:00
|
|
|
info_queue->AddStorageFilterEntries(&filter);
|
2016-06-24 10:43:46 +02:00
|
|
|
}
|
|
|
|
}
|
2019-03-09 23:31:36 +10:00
|
|
|
else
|
|
|
|
{
|
|
|
|
WARN_LOG(VIDEO, "Debug layer requested but not available.");
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
}
|
|
|
|
|
2019-03-09 23:31:36 +10:00
|
|
|
if (!enable_debug_layer || FAILED(hr))
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
2019-03-09 23:31:36 +10:00
|
|
|
hr = d3d11_create_device(adapter.Get(), D3D_DRIVER_TYPE_UNKNOWN, nullptr, 0,
|
2019-06-01 07:55:09 -04:00
|
|
|
s_supported_feature_levels.data(),
|
|
|
|
static_cast<UINT>(s_supported_feature_levels.size()),
|
2019-03-09 23:31:36 +10:00
|
|
|
D3D11_SDK_VERSION, &device, &feature_level, &context);
|
2017-07-10 01:02:29 +02:00
|
|
|
}
|
2017-06-25 20:57:46 +02:00
|
|
|
|
2019-03-09 23:31:36 +10:00
|
|
|
if (FAILED(hr))
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
2019-03-09 23:31:36 +10:00
|
|
|
PanicAlertT(
|
|
|
|
"Failed to initialize Direct3D.\nMake sure your video card supports at least D3D 10.0");
|
2019-05-26 19:59:29 -07:00
|
|
|
dxgi_factory.Reset();
|
2019-03-09 23:31:36 +10:00
|
|
|
D3DCommon::UnloadLibraries();
|
2019-05-26 19:59:29 -07:00
|
|
|
s_d3d11_library.Close();
|
2019-03-09 23:31:36 +10:00
|
|
|
return false;
|
2016-06-24 10:43:46 +02:00
|
|
|
}
|
|
|
|
|
2019-07-21 00:17:15 +02:00
|
|
|
hr = device.As(&device1);
|
2017-09-03 16:33:47 +10:00
|
|
|
if (FAILED(hr))
|
2018-05-26 00:04:18 +10:00
|
|
|
{
|
2017-09-03 16:33:47 +10:00
|
|
|
WARN_LOG(VIDEO, "Missing Direct3D 11.1 support. Logical operations will not be supported.");
|
2018-05-26 00:04:18 +10:00
|
|
|
g_Config.backend_info.bSupportsLogicOp = false;
|
|
|
|
}
|
2017-09-03 16:33:47 +10:00
|
|
|
|
2019-03-29 19:55:00 +10:00
|
|
|
stateman = std::make_unique<StateManager>();
|
2019-03-09 23:31:36 +10:00
|
|
|
return true;
|
2010-06-13 19:50:06 +00:00
|
|
|
}
|
|
|
|
|
2019-03-09 23:31:36 +10:00
|
|
|
void Destroy()
|
2010-06-13 19:50:06 +00:00
|
|
|
{
|
2019-03-29 19:55:00 +10:00
|
|
|
stateman.reset();
|
2014-06-20 03:45:42 +02:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
context->ClearState();
|
2019-03-09 23:31:36 +10:00
|
|
|
context->Flush();
|
2011-06-11 19:37:21 +00:00
|
|
|
|
2019-03-09 23:31:36 +10:00
|
|
|
context.Reset();
|
|
|
|
device1.Reset();
|
2015-02-09 13:00:42 +01:00
|
|
|
|
2019-03-09 23:31:36 +10:00
|
|
|
auto remaining_references = device.Reset();
|
2017-11-19 13:23:56 -05:00
|
|
|
if (s_debug)
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
2019-03-09 23:31:36 +10:00
|
|
|
--remaining_references; // the debug interface increases the refcount of the device, subtract
|
|
|
|
// that.
|
|
|
|
if (remaining_references)
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
|
|
|
// print out alive objects, but only if we actually have pending references
|
|
|
|
// note this will also print out internal live objects to the debug console
|
2017-11-19 13:23:56 -05:00
|
|
|
s_debug->ReportLiveDeviceObjects(D3D11_RLDO_SUMMARY | D3D11_RLDO_DETAIL);
|
2016-06-24 10:43:46 +02:00
|
|
|
}
|
2019-03-09 23:31:36 +10:00
|
|
|
s_debug.Reset();
|
2016-06-24 10:43:46 +02:00
|
|
|
}
|
2015-02-09 13:00:42 +01:00
|
|
|
|
2019-03-09 23:31:36 +10:00
|
|
|
if (remaining_references)
|
|
|
|
ERROR_LOG(VIDEO, "Unreleased references: %i.", remaining_references);
|
2016-06-24 10:43:46 +02:00
|
|
|
else
|
|
|
|
NOTICE_LOG(VIDEO, "Successfully released all device references!");
|
2010-06-19 01:02:43 +00:00
|
|
|
|
2019-05-26 19:59:29 -07:00
|
|
|
dxgi_factory.Reset();
|
2019-03-09 23:31:36 +10:00
|
|
|
D3DCommon::UnloadLibraries();
|
|
|
|
s_d3d11_library.Close();
|
2017-09-03 17:33:12 +02:00
|
|
|
}
|
|
|
|
|
2019-03-09 23:31:36 +10:00
|
|
|
std::vector<u32> GetAAModes(u32 adapter_index)
|
2010-07-11 16:26:46 +00:00
|
|
|
{
|
2019-03-09 23:31:36 +10:00
|
|
|
// Use temporary device if we don't have one already.
|
|
|
|
Common::DynamicLibrary temp_lib;
|
|
|
|
ComPtr<ID3D11Device> temp_device = device;
|
|
|
|
if (!temp_device)
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
2019-05-12 14:42:16 +10:00
|
|
|
ComPtr<IDXGIFactory> temp_dxgi_factory = D3DCommon::CreateDXGIFactory(false);
|
2019-03-09 23:31:36 +10:00
|
|
|
if (!temp_dxgi_factory)
|
|
|
|
return {};
|
2010-07-11 16:26:46 +00:00
|
|
|
|
2019-03-09 23:31:36 +10:00
|
|
|
ComPtr<IDXGIAdapter> adapter;
|
|
|
|
temp_dxgi_factory->EnumAdapters(adapter_index, &adapter);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-03-09 23:31:36 +10:00
|
|
|
PFN_D3D11_CREATE_DEVICE d3d11_create_device;
|
|
|
|
if (!temp_lib.Open("d3d11.dll") ||
|
|
|
|
!temp_lib.GetSymbol("D3D11CreateDevice", &d3d11_create_device))
|
|
|
|
{
|
|
|
|
return {};
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-03-09 23:31:36 +10:00
|
|
|
HRESULT hr = d3d11_create_device(nullptr, D3D_DRIVER_TYPE_HARDWARE, nullptr, 0,
|
2019-06-01 07:55:09 -04:00
|
|
|
s_supported_feature_levels.data(),
|
|
|
|
static_cast<UINT>(s_supported_feature_levels.size()),
|
2019-03-09 23:31:36 +10:00
|
|
|
D3D11_SDK_VERSION, &temp_device, nullptr, nullptr);
|
|
|
|
if (FAILED(hr))
|
|
|
|
return {};
|
2016-06-24 10:43:46 +02:00
|
|
|
}
|
2010-06-13 19:50:06 +00:00
|
|
|
|
2019-03-09 23:31:36 +10: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.
|
|
|
|
if (temp_device->GetFeatureLevel() == D3D_FEATURE_LEVEL_10_0)
|
|
|
|
return {};
|
2017-09-03 17:33:12 +02:00
|
|
|
|
2019-03-09 23:31:36 +10:00
|
|
|
std::vector<u32> aa_modes;
|
|
|
|
for (u32 samples = 1; samples < D3D11_MAX_MULTISAMPLE_SAMPLE_COUNT; ++samples)
|
2017-11-19 13:23:56 -05:00
|
|
|
{
|
2019-03-09 23:31:36 +10:00
|
|
|
UINT quality_levels = 0;
|
|
|
|
if (SUCCEEDED(temp_device->CheckMultisampleQualityLevels(DXGI_FORMAT_R8G8B8A8_UNORM, samples,
|
|
|
|
&quality_levels)) &&
|
|
|
|
quality_levels > 0)
|
|
|
|
{
|
|
|
|
aa_modes.push_back(samples);
|
|
|
|
}
|
2017-11-19 13:23:56 -05:00
|
|
|
}
|
2017-06-26 21:47:34 +02:00
|
|
|
|
2019-03-09 23:31:36 +10:00
|
|
|
return aa_modes;
|
2017-09-02 14:22:18 -04:00
|
|
|
}
|
2019-04-28 15:26:46 +10:00
|
|
|
|
|
|
|
bool SupportsTextureFormat(DXGI_FORMAT format)
|
|
|
|
{
|
|
|
|
UINT support;
|
|
|
|
if (FAILED(device->CheckFormatSupport(format, &support)))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return (support & D3D11_FORMAT_SUPPORT_TEXTURE2D) != 0;
|
|
|
|
}
|
|
|
|
|
2011-01-29 20:16:51 +00:00
|
|
|
} // namespace D3D
|
|
|
|
|
2011-02-14 02:18:03 +00:00
|
|
|
} // namespace DX11
|