dxgi.cpp 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596
  1. /*
  2. * Copyright 2011-2018 Branimir Karadzic. All rights reserved.
  3. * License: https://github.com/bkaradzic/bgfx#license-bsd-2-clause
  4. */
  5. #include "bgfx_p.h"
  6. #if BGFX_CONFIG_RENDERER_DIRECT3D11 || BGFX_CONFIG_RENDERER_DIRECT3D12
  7. #include "dxgi.h"
  8. #include "renderer_d3d.h"
  9. #if !BX_PLATFORM_WINDOWS
  10. # include <inspectable.h>
  11. # if BX_PLATFORM_WINRT
  12. # include <windows.ui.xaml.media.dxinterop.h>
  13. # endif // BX_PLATFORM_WINRT
  14. #endif // !BX_PLATFORM_WINDOWS
  15. namespace bgfx
  16. {
  17. BX_PRAGMA_DIAGNOSTIC_PUSH();
  18. BX_PRAGMA_DIAGNOSTIC_IGNORED_GCC("-Wunused-variable");
  19. BX_PRAGMA_DIAGNOSTIC_IGNORED_CLANG("-Wunused-const-variable");
  20. BX_PRAGMA_DIAGNOSTIC_IGNORED_CLANG("-Wunneeded-internal-declaration");
  21. #if BX_PLATFORM_WINDOWS
  22. static PFN_CREATE_DXGI_FACTORY CreateDXGIFactory;
  23. static PFN_GET_DEBUG_INTERFACE DXGIGetDebugInterface;
  24. static PFN_GET_DEBUG_INTERFACE1 DXGIGetDebugInterface1;
  25. #endif // BX_PLATFORM_WINDOWS
  26. static const GUID IID_IDXGIFactory = { 0x7b7166ec, 0x21c7, 0x44ae, { 0xb2, 0x1a, 0xc9, 0xae, 0x32, 0x1a, 0xe3, 0x69 } };
  27. static const GUID IID_IDXGIFactory2 = { 0x50c83a1c, 0xe072, 0x4c48, { 0x87, 0xb0, 0x36, 0x30, 0xfa, 0x36, 0xa6, 0xd0 } };
  28. static const GUID IID_IDXGIFactory3 = { 0x25483823, 0xcd46, 0x4c7d, { 0x86, 0xca, 0x47, 0xaa, 0x95, 0xb8, 0x37, 0xbd } };
  29. static const GUID IID_IDXGIFactory4 = { 0x1bc6ea02, 0xef36, 0x464f, { 0xbf, 0x0c, 0x21, 0xca, 0x39, 0xe5, 0x16, 0x8a } };
  30. static const GUID IID_IDXGIFactory5 = { 0x7632e1f5, 0xee65, 0x4dca, { 0x87, 0xfd, 0x84, 0xcd, 0x75, 0xf8, 0x83, 0x8d } };
  31. static const GUID IID_IDXGIDevice0 = { 0x54ec77fa, 0x1377, 0x44e6, { 0x8c, 0x32, 0x88, 0xfd, 0x5f, 0x44, 0xc8, 0x4c } };
  32. static const GUID IID_IDXGIDevice1 = { 0x77db970f, 0x6276, 0x48ba, { 0xba, 0x28, 0x07, 0x01, 0x43, 0xb4, 0x39, 0x2c } };
  33. static const GUID IID_IDXGIDevice2 = { 0x05008617, 0xfbfd, 0x4051, { 0xa7, 0x90, 0x14, 0x48, 0x84, 0xb4, 0xf6, 0xa9 } };
  34. static const GUID IID_IDXGIDevice3 = { 0x6007896c, 0x3244, 0x4afd, { 0xbf, 0x18, 0xa6, 0xd3, 0xbe, 0xda, 0x50, 0x23 } };
  35. static const GUID IID_IDXGIAdapter = { 0x2411e7e1, 0x12ac, 0x4ccf, { 0xbd, 0x14, 0x97, 0x98, 0xe8, 0x53, 0x4d, 0xc0 } };
  36. static const GUID IID_IDXGIAdapter2 = { 0x0aa1ae0a, 0xfa0e, 0x4b84, { 0x86, 0x44, 0xe0, 0x5f, 0xf8, 0xe5, 0xac, 0xb5 } };
  37. static const GUID IID_IDXGIAdapter3 = { 0x645967a4, 0x1392, 0x4310, { 0xa7, 0x98, 0x80, 0x53, 0xce, 0x3e, 0x93, 0xfd } };
  38. static const GUID IID_IDXGIAdapter4 = { 0x3c8d99d1, 0x4fbf, 0x4181, { 0xa8, 0x2c, 0xaf, 0x66, 0xbf, 0x7b, 0xd2, 0x4e } };
  39. static const GUID IID_IDXGISwapChain3 = { 0x94d99bdb, 0xf1f8, 0x4ab0, { 0xb2, 0x36, 0x7d, 0xa0, 0x17, 0x0e, 0xda, 0xb1 } };
  40. static const GUID IID_IDXGISwapChain4 = { 0x3d585d5a, 0xbd4a, 0x489e, { 0xb1, 0xf4, 0x3d, 0xbc, 0xb6, 0x45, 0x2f, 0xfb } };
  41. static const GUID IID_IDXGIOutput6 = { 0x068346e8, 0xaaec, 0x4b84, { 0xad, 0xd7, 0x13, 0x7f, 0x51, 0x3f, 0x77, 0xa1 } };
  42. BX_PRAGMA_DIAGNOSTIC_POP();
  43. static const DXGI_COLOR_SPACE_TYPE s_colorSpace[] =
  44. {
  45. DXGI_COLOR_SPACE_RGB_FULL_G22_NONE_P709, // gamma 2.2, BT.709
  46. DXGI_COLOR_SPACE_RGB_FULL_G10_NONE_P709, // gamma 1.0, BT.709
  47. DXGI_COLOR_SPACE_RGB_FULL_G2084_NONE_P2020, // gamma 2084, BT.2020
  48. };
  49. static const char* s_colorSpaceStr[] =
  50. {
  51. // https://msdn.microsoft.com/en-us/library/windows/desktop/dn903661(v=vs.85).aspx
  52. "RGB, 0-255, 2.2, Image, BT.709, n/a", // DXGI_COLOR_SPACE_RGB_FULL_G22_NONE_P709
  53. "RGB, 0-255, 1.0, Image, BT.709, n/a", // DXGI_COLOR_SPACE_RGB_FULL_G10_NONE_P709
  54. "RGB, 16-235, 2.2, Image, BT.709, n/a", // DXGI_COLOR_SPACE_RGB_STUDIO_G22_NONE_P709
  55. "RGB, 16-235, 2.2, Image, BT.2020, n/a", // DXGI_COLOR_SPACE_RGB_STUDIO_G22_NONE_P2020
  56. "Reserved", // DXGI_COLOR_SPACE_RESERVED
  57. "YCbCr, 0-255, 2.2, Image, BT.709, BT.601", // DXGI_COLOR_SPACE_YCBCR_FULL_G22_NONE_P709_X601
  58. "YCbCr, 16-235, 2.2, Video, BT.601, n/a", // DXGI_COLOR_SPACE_YCBCR_STUDIO_G22_LEFT_P601
  59. "YCbCr, 0-255, 2.2, Video, BT.601, n/a", // DXGI_COLOR_SPACE_YCBCR_FULL_G22_LEFT_P601
  60. "YCbCr, 16-235, 2.2, Video, BT.709, n/a", // DXGI_COLOR_SPACE_YCBCR_STUDIO_G22_LEFT_P709
  61. "YCbCr, 0-255, 2.2, Video, BT.709, n/a", // DXGI_COLOR_SPACE_YCBCR_FULL_G22_LEFT_P709
  62. "YCbCr, 16-235, 2.2, Video, BT.2020, n/a", // DXGI_COLOR_SPACE_YCBCR_STUDIO_G22_LEFT_P2020
  63. "YCbCr, 0-255, 2.2, Video, BT.2020, n/a", // DXGI_COLOR_SPACE_YCBCR_FULL_G22_LEFT_P2020
  64. "RGB, 0-255, 2084, Image, BT.2020, n/a", // DXGI_COLOR_SPACE_RGB_FULL_G2084_NONE_P2020
  65. "YCbCr, 16-235, 2084, Video, BT.2020, n/a", // DXGI_COLOR_SPACE_YCBCR_STUDIO_G2084_LEFT_P2020
  66. "RGB, 0-255, 2084, Image, BT.2020, n/a", // DXGI_COLOR_SPACE_RGB_STUDIO_G2084_NONE_P2020
  67. "YCbCr, 16-235, 2.2, Video, BT.2020, n/a", // DXGI_COLOR_SPACE_YCBCR_STUDIO_G22_TOPLEFT_P2020
  68. "YCbCr, 16-235, 2084, Video, BT.2020, n/a", // DXGI_COLOR_SPACE_YCBCR_STUDIO_G2084_TOPLEFT_P2020
  69. #if BX_PLATFORM_WINDOWS
  70. "RGB, 0-255, 2.2, Image, BT.2020, n/a", // DXGI_COLOR_SPACE_RGB_FULL_G22_NONE_P2020
  71. "YCbCr, 16-235, HLG, Video, BT.2020, n/a", // DXGI_COLOR_SPACE_YCBCR_STUDIO_GHLG_TOPLEFT_P2020
  72. "YCbCr, 0-255, HLG, Video, BT.2020, n/a", // DXGI_COLOR_SPACE_YCBCR_FULL_GHLG_TOPLEFT_P2020
  73. // "RGB, 16-235, 2.4, Image, BT.709, n/a", // DXGI_COLOR_SPACE_RGB_STUDIO_G24_NONE_P709
  74. // "RGB, 16-235, 2.4, Image, BT.2020, n/a", // DXGI_COLOR_SPACE_RGB_STUDIO_G24_NONE_P2020
  75. // "YCbCr, 16-235, 2.4, Video, BT.709, n/a", // DXGI_COLOR_SPACE_YCBCR_STUDIO_G24_LEFT_P709
  76. // "YCbCr, 16-235, 2.4, Video, BT.2020, n/a", // DXGI_COLOR_SPACE_YCBCR_STUDIO_G24_LEFT_P2020
  77. // "YCbCr, 16-235, 2.4, Video, BT.2020, n/a", // DXGI_COLOR_SPACE_YCBCR_STUDIO_G24_TOPLEFT_P2020
  78. #endif // BX_PLATFORM_WINDOWS
  79. "Custom",
  80. };
  81. static const DXGI_COLOR_SPACE_TYPE kDxgiLastColorSpace =
  82. #if BX_PLATFORM_WINDOWS
  83. DXGI_COLOR_SPACE_YCBCR_FULL_GHLG_TOPLEFT_P2020
  84. #else
  85. DXGI_COLOR_SPACE_YCBCR_STUDIO_G2084_TOPLEFT_P2020
  86. #endif // BX_PLATFORM_WINDOWS
  87. ;
  88. BX_STATIC_ASSERT(BX_COUNTOF(s_colorSpaceStr) == kDxgiLastColorSpace+2, "Colorspace string table mismatch with DXGI_COLOR_SPACE_*.");
  89. static const GUID s_dxgiDeviceIIDs[] =
  90. {
  91. IID_IDXGIDevice3,
  92. IID_IDXGIDevice2,
  93. IID_IDXGIDevice1,
  94. IID_IDXGIDevice0,
  95. };
  96. static const GUID s_dxgiSwapChainIIDs[] =
  97. {
  98. IID_IDXGISwapChain4,
  99. IID_IDXGISwapChain3,
  100. };
  101. DxgiSwapChain::DxgiSwapChain()
  102. {
  103. }
  104. Dxgi::Dxgi()
  105. : m_dxgiDll(NULL)
  106. , m_dxgiDebugDll(NULL)
  107. , m_driverType(D3D_DRIVER_TYPE_NULL)
  108. , m_factory(NULL)
  109. , m_adapter(NULL)
  110. , m_output(NULL)
  111. {
  112. }
  113. bool Dxgi::init(Caps& _caps)
  114. {
  115. #if BX_PLATFORM_WINDOWS
  116. m_dxgiDll = bx::dlopen("dxgi.dll");
  117. if (NULL == m_dxgiDll)
  118. {
  119. BX_TRACE("Init error: Failed to load dxgi.dll.");
  120. return false;
  121. }
  122. m_dxgiDebugDll = bx::dlopen("dxgidebug.dll");
  123. if (NULL != m_dxgiDebugDll)
  124. {
  125. DXGIGetDebugInterface = (PFN_GET_DEBUG_INTERFACE )bx::dlsym(m_dxgiDebugDll, "DXGIGetDebugInterface");
  126. DXGIGetDebugInterface1 = (PFN_GET_DEBUG_INTERFACE1)bx::dlsym(m_dxgiDebugDll, "DXGIGetDebugInterface1");
  127. if (NULL == DXGIGetDebugInterface
  128. && NULL == DXGIGetDebugInterface1)
  129. {
  130. bx::dlclose(m_dxgiDebugDll);
  131. m_dxgiDebugDll = NULL;
  132. }
  133. }
  134. CreateDXGIFactory = (PFN_CREATE_DXGI_FACTORY)bx::dlsym(m_dxgiDll, "CreateDXGIFactory1");
  135. if (NULL == CreateDXGIFactory)
  136. {
  137. CreateDXGIFactory = (PFN_CREATE_DXGI_FACTORY)bx::dlsym(m_dxgiDll, "CreateDXGIFactory");
  138. }
  139. if (NULL == CreateDXGIFactory)
  140. {
  141. BX_TRACE("Init error: Function CreateDXGIFactory not found.");
  142. return false;
  143. }
  144. #endif // BX_PLATFORM_WINDOWS
  145. HRESULT hr = S_OK;
  146. #if BX_PLATFORM_WINRT
  147. // WinRT requires the IDXGIFactory2 interface, which isn't supported on older platforms
  148. hr = CreateDXGIFactory1(IID_IDXGIFactory2, (void**)&m_factory);
  149. #elif BX_PLATFORM_WINDOWS
  150. hr = CreateDXGIFactory(IID_IDXGIFactory, (void**)&m_factory);
  151. #endif // BX_PLATFORM_*
  152. if (FAILED(hr) )
  153. {
  154. BX_TRACE("Init error: Unable to create DXGI factory.");
  155. return false;
  156. }
  157. m_driverType = BGFX_PCI_ID_SOFTWARE_RASTERIZER == _caps.vendorId
  158. ? D3D_DRIVER_TYPE_WARP
  159. : D3D_DRIVER_TYPE_HARDWARE
  160. ;
  161. if (NULL != m_factory)
  162. {
  163. AdapterI* adapter;
  164. for (uint32_t ii = 0
  165. ; DXGI_ERROR_NOT_FOUND != m_factory->EnumAdapters(ii, reinterpret_cast<IDXGIAdapter**>(&adapter) ) && ii < BX_COUNTOF(_caps.gpu)
  166. ; ++ii
  167. )
  168. {
  169. DXGI_ADAPTER_DESC desc;
  170. hr = adapter->GetDesc(&desc);
  171. if (SUCCEEDED(hr) )
  172. {
  173. BX_TRACE("Adapter #%d", ii);
  174. char description[BX_COUNTOF(desc.Description)];
  175. wcstombs(description, desc.Description, BX_COUNTOF(desc.Description) );
  176. BX_TRACE("\tDescription: %s", description);
  177. BX_TRACE("\tVendorId: 0x%08x, DeviceId: 0x%08x, SubSysId: 0x%08x, Revision: 0x%08x"
  178. , desc.VendorId
  179. , desc.DeviceId
  180. , desc.SubSysId
  181. , desc.Revision
  182. );
  183. BX_TRACE("\tMemory: %" PRIi64 " (video), %" PRIi64 " (system), %" PRIi64 " (shared)"
  184. , desc.DedicatedVideoMemory
  185. , desc.DedicatedSystemMemory
  186. , desc.SharedSystemMemory
  187. );
  188. _caps.gpu[ii].vendorId = (uint16_t)desc.VendorId;
  189. _caps.gpu[ii].deviceId = (uint16_t)desc.DeviceId;
  190. ++_caps.numGPUs;
  191. if (NULL == m_adapter)
  192. {
  193. if ( (BGFX_PCI_ID_NONE != _caps.vendorId || 0 != _caps.deviceId)
  194. && (BGFX_PCI_ID_NONE == _caps.vendorId || desc.VendorId == _caps.vendorId)
  195. && ( 0 == _caps.deviceId || desc.DeviceId == _caps.deviceId) )
  196. {
  197. m_adapter = adapter;
  198. m_adapter->AddRef();
  199. m_driverType = D3D_DRIVER_TYPE_UNKNOWN;
  200. }
  201. if (BX_ENABLED(BGFX_CONFIG_DEBUG_PERFHUD)
  202. && 0 != bx::strFind(description, "PerfHUD") )
  203. {
  204. m_adapter = adapter;
  205. m_driverType = D3D_DRIVER_TYPE_REFERENCE;
  206. }
  207. }
  208. }
  209. IDXGIOutput* output;
  210. for (uint32_t jj = 0
  211. ; DXGI_ERROR_NOT_FOUND != adapter->EnumOutputs(jj, &output)
  212. ; ++jj
  213. )
  214. {
  215. DXGI_OUTPUT_DESC outputDesc;
  216. hr = output->GetDesc(&outputDesc);
  217. if (SUCCEEDED(hr))
  218. {
  219. BX_TRACE("\tOutput #%d", jj);
  220. char deviceName[BX_COUNTOF(outputDesc.DeviceName)];
  221. wcstombs(deviceName, outputDesc.DeviceName, BX_COUNTOF(outputDesc.DeviceName));
  222. BX_TRACE("\t\tDeviceName: %s", deviceName);
  223. BX_TRACE("\t\tDesktopCoordinates: %d, %d, %d, %d"
  224. , outputDesc.DesktopCoordinates.left
  225. , outputDesc.DesktopCoordinates.top
  226. , outputDesc.DesktopCoordinates.right
  227. , outputDesc.DesktopCoordinates.bottom
  228. );
  229. BX_TRACE("\t\tAttachedToDesktop: %d", outputDesc.AttachedToDesktop);
  230. BX_TRACE("\t\tRotation: %d", outputDesc.Rotation);
  231. DX_RELEASE(output, 0);
  232. }
  233. }
  234. DX_RELEASE(adapter, adapter == m_adapter ? 1 : 0);
  235. }
  236. if (NULL == m_adapter)
  237. {
  238. hr = m_factory->EnumAdapters(0, reinterpret_cast<IDXGIAdapter**>(&m_adapter) );
  239. BX_WARN(SUCCEEDED(hr), "EnumAdapters failed 0x%08x.", hr);
  240. m_driverType = D3D_DRIVER_TYPE_UNKNOWN;
  241. }
  242. bx::memSet(&m_adapterDesc, 0, sizeof(m_adapterDesc) );
  243. hr = m_adapter->GetDesc(&m_adapterDesc);
  244. BX_WARN(SUCCEEDED(hr), "Adapter GetDesc failed 0x%08x.", hr);
  245. m_adapter->EnumOutputs(0, &m_output);
  246. _caps.vendorId = 0 == m_adapterDesc.VendorId
  247. ? BGFX_PCI_ID_SOFTWARE_RASTERIZER
  248. : (uint16_t)m_adapterDesc.VendorId
  249. ;
  250. _caps.deviceId = (uint16_t)m_adapterDesc.DeviceId;
  251. {
  252. IDXGIDevice* dxgiDevice = NULL;
  253. hr = E_FAIL;
  254. for (uint32_t ii = 0; ii < BX_COUNTOF(s_dxgiDeviceIIDs) && FAILED(hr); ++ii)
  255. {
  256. hr = m_factory->QueryInterface(s_dxgiDeviceIIDs[ii], (void**)&dxgiDevice);
  257. BX_TRACE("DXGI device 11.%d, hr %x", BX_COUNTOF(s_dxgiDeviceIIDs) - 1 - ii, hr);
  258. if (SUCCEEDED(hr))
  259. {
  260. #if BX_COMPILER_MSVC
  261. BX_PRAGMA_DIAGNOSTIC_PUSH();
  262. BX_PRAGMA_DIAGNOSTIC_IGNORED_MSVC(4530) // warning C4530: C++ exception handler used, but unwind semantics are not enabled. Specify /EHsc
  263. try
  264. {
  265. // QueryInterface above can succeed, but getting adapter call might crash on Win7.
  266. hr = dxgiDevice->GetAdapter(reinterpret_cast<IDXGIAdapter**>(&adapter) );
  267. }
  268. catch (...)
  269. {
  270. BX_TRACE("Failed to get adapter for IID_IDXGIDevice%d.", BX_COUNTOF(s_dxgiDeviceIIDs) - 1 - ii);
  271. DX_RELEASE(dxgiDevice, 0);
  272. hr = E_FAIL;
  273. }
  274. BX_PRAGMA_DIAGNOSTIC_POP();
  275. #else
  276. hr = dxgiDevice->GetAdapter(reinterpret_cast<IDXGIAdapter**>(&adapter) );
  277. #endif // BX_COMPILER_MSVC
  278. }
  279. }
  280. }
  281. }
  282. return true;
  283. }
  284. void Dxgi::shutdown()
  285. {
  286. DX_RELEASE(m_output, 0);
  287. DX_RELEASE(m_adapter, 0);
  288. DX_RELEASE(m_factory, 0);
  289. bx::dlclose(m_dxgiDebugDll);
  290. m_dxgiDebugDll = NULL;
  291. bx::dlclose(m_dxgiDll);
  292. m_dxgiDll = NULL;
  293. }
  294. void Dxgi::update(IUnknown* _device)
  295. {
  296. if (NULL == m_factory)
  297. {
  298. IDXGIDevice* dxgiDevice = NULL;
  299. HRESULT hr = E_FAIL;
  300. for (uint32_t ii = 0; ii < BX_COUNTOF(s_dxgiDeviceIIDs) && FAILED(hr); ++ii)
  301. {
  302. hr = _device->QueryInterface(s_dxgiDeviceIIDs[ii], (void**)&dxgiDevice);
  303. BX_TRACE("DXGI device 11.%d, hr %x", BX_COUNTOF(s_dxgiDeviceIIDs) - 1 - ii, hr);
  304. if (SUCCEEDED(hr) )
  305. {
  306. DX_CHECK(dxgiDevice->GetAdapter(reinterpret_cast<IDXGIAdapter**>(&m_adapter) ) );
  307. DX_RELEASE(dxgiDevice, 1);
  308. }
  309. }
  310. bx::memSet(&m_adapterDesc, 0, sizeof(m_adapterDesc) );
  311. hr = m_adapter->GetDesc(&m_adapterDesc);
  312. BX_WARN(SUCCEEDED(hr), "Adapter GetDesc failed 0x%08x.", hr);
  313. DX_CHECK(m_adapter->GetParent(IID_IDXGIFactory2, (void**)&m_factory) );
  314. #if 0
  315. bx::memSet(&m_adapterDesc, 0, sizeof(m_adapterDesc) );
  316. // NOTICE:
  317. // LUID STDMETHODCALLTYPE ID3D12Device::GetAdapterLuid() has a different behaviour in gcc ,
  318. // because gcc64 returns small struct in RAX, but the microsoft implemention of ID3D12Device::GetAdapterLuid() in d3d12.dll
  319. // pass the struct LUID's address as the second parameter.
  320. typedef void (STDMETHODCALLTYPE ID3D12Device::*ID3D12Device_GetAdapterLuid_f)(LUID *);
  321. (m_device->*(ID3D12Device_GetAdapterLuid_f)(&ID3D12Device::GetAdapterLuid))(&luid);
  322. AdapterI* adapter;
  323. for (uint32_t ii = 0; DXGI_ERROR_NOT_FOUND != m_factory->EnumAdapters(ii, reinterpret_cast<IDXGIAdapter**>(&adapter) ); ++ii)
  324. {
  325. adapter->GetDesc(&m_adapterDesc);
  326. if (m_adapterDesc.AdapterLuid.LowPart == luid.LowPart
  327. && m_adapterDesc.AdapterLuid.HighPart == luid.HighPart)
  328. {
  329. if (NULL == m_adapter)
  330. {
  331. m_adapter = adapter;
  332. }
  333. else
  334. {
  335. #if BX_PLATFORM_WINDOWS || BX_PLATFORM_WINRT
  336. DX_RELEASE(adapter, 0);
  337. #else
  338. DX_RELEASE(adapter, 2);
  339. #endif // BX_PLATFORM_WINDOWS || BX_PLATFORM_WINRT
  340. }
  341. break;
  342. }
  343. DX_RELEASE(adapter, 0);
  344. }
  345. g_caps.vendorId = (uint16_t)m_adapterDesc.VendorId;
  346. g_caps.deviceId = (uint16_t)m_adapterDesc.DeviceId;
  347. #endif // 0
  348. }
  349. }
  350. HRESULT Dxgi::createSwapChain(IUnknown* _device, const SwapChainDesc& _scd, SwapChainI** _swapChain)
  351. {
  352. HRESULT hr = S_OK;
  353. bool allowTearing = false;
  354. #if BX_PLATFORM_WINDOWS
  355. if (windowsVersionIs(Condition::GreaterEqual, 0x0602) )
  356. {
  357. // BK - CheckFeatureSupport with DXGI_FEATURE_PRESENT_ALLOW_TEARING
  358. // will crash on pre Windows 8. Issue #1356.
  359. hr = m_factory->CheckFeatureSupport(DXGI_FEATURE_PRESENT_ALLOW_TEARING, &allowTearing, sizeof(allowTearing) );
  360. BX_TRACE("Allow tearing is %ssupported.", allowTearing ? "" : "not ");
  361. }
  362. DXGI_SWAP_CHAIN_DESC scd;
  363. scd.BufferDesc.Width = _scd.width;
  364. scd.BufferDesc.Height = _scd.height;
  365. scd.BufferDesc.RefreshRate.Numerator = 1;
  366. scd.BufferDesc.RefreshRate.Denominator = 60;
  367. scd.BufferDesc.Format = _scd.format;
  368. scd.BufferDesc.ScanlineOrdering = DXGI_MODE_SCANLINE_ORDER_UNSPECIFIED;
  369. scd.BufferDesc.Scaling = DXGI_MODE_SCALING_UNSPECIFIED;
  370. scd.SampleDesc = _scd.sampleDesc;
  371. scd.BufferUsage = _scd.bufferUsage;
  372. scd.BufferCount = _scd.bufferCount;
  373. scd.OutputWindow = (HWND)_scd.nwh;
  374. scd.Windowed = _scd.windowed;
  375. scd.SwapEffect = _scd.swapEffect;
  376. scd.Flags = 0
  377. | _scd.flags
  378. | (allowTearing ? DXGI_SWAP_CHAIN_FLAG_ALLOW_TEARING : 0)
  379. ;
  380. hr = m_factory->CreateSwapChain(
  381. _device
  382. , &scd
  383. , reinterpret_cast<IDXGISwapChain**>(_swapChain)
  384. );
  385. #else
  386. DXGI_SWAP_CHAIN_DESC1 scd;
  387. scd.Width = _scd.width;
  388. scd.Height = _scd.height;
  389. scd.Format = _scd.format;
  390. scd.Stereo = _scd.stereo;
  391. scd.SampleDesc = _scd.sampleDesc;
  392. scd.BufferUsage = _scd.bufferUsage;
  393. scd.BufferCount = _scd.bufferCount;
  394. scd.Scaling = _scd.scaling;
  395. scd.SwapEffect = _scd.swapEffect;
  396. scd.AlphaMode = _scd.alphaMode;
  397. scd.Flags = _scd.flags;
  398. if (NULL == _scd.ndt)
  399. {
  400. hr = m_factory->CreateSwapChainForCoreWindow(
  401. _device
  402. , (::IUnknown*)_scd.nwh
  403. , &scd
  404. , NULL
  405. , reinterpret_cast<IDXGISwapChain1**>(_swapChain)
  406. );
  407. }
  408. else if (reinterpret_cast<void*>(1) == _scd.ndt)
  409. {
  410. return E_FAIL;
  411. }
  412. else
  413. {
  414. hr = m_factory->CreateSwapChainForComposition(
  415. _device
  416. , &scd
  417. , NULL
  418. , reinterpret_cast<IDXGISwapChain1**>(_swapChain)
  419. );
  420. if (FAILED(hr) )
  421. {
  422. return hr;
  423. }
  424. # if BX_PLATFORM_WINRT
  425. IInspectable *nativeWindow = reinterpret_cast<IInspectable *>(_scd.nwh);
  426. ISwapChainBackgroundPanelNative* panel = NULL;
  427. hr = nativeWindow->QueryInterface(
  428. __uuidof(ISwapChainBackgroundPanelNative)
  429. , (void **)&panel
  430. );
  431. if (FAILED(hr) )
  432. {
  433. return hr;
  434. }
  435. if (NULL != panel)
  436. {
  437. hr = panel->SetSwapChain(*_swapChain);
  438. if (FAILED(hr) )
  439. {
  440. return hr;
  441. }
  442. panel->Release();
  443. }
  444. # endif // BX_PLATFORM_WINRT
  445. }
  446. #endif // BX_PLATFORM_WINDOWS
  447. if (FAILED(hr) )
  448. {
  449. BX_TRACE("Failed to create swap chain.");
  450. return hr;
  451. }
  452. #if BX_PLATFORM_WINDOWS
  453. if (SUCCEEDED(hr) )
  454. {
  455. for (uint32_t ii = 0; ii < BX_COUNTOF(s_dxgiSwapChainIIDs); ++ii)
  456. {
  457. IDXGISwapChain1* swapChain;
  458. hr = (*_swapChain)->QueryInterface(s_dxgiSwapChainIIDs[ii], (void**)&swapChain);
  459. BX_TRACE("DXGI swap chain %d, hr %x", 4-ii, hr);
  460. if (SUCCEEDED(hr) )
  461. {
  462. DX_RELEASE(*_swapChain, 1);
  463. *_swapChain = reinterpret_cast<SwapChainI*>(swapChain);
  464. BX_TRACE("Color space support:");
  465. for (uint32_t jj = 0; jj < BX_COUNTOF(s_colorSpace); ++jj)
  466. {
  467. uint32_t colorSpaceSupport;
  468. reinterpret_cast<IDXGISwapChain3*>(*_swapChain)->CheckColorSpaceSupport(s_colorSpace[jj], &colorSpaceSupport);
  469. BX_TRACE("\t%2d, 0x%08x, %s"
  470. , s_colorSpace[jj]
  471. , colorSpaceSupport
  472. , 0 != (colorSpaceSupport & DXGI_SWAP_CHAIN_COLOR_SPACE_SUPPORT_FLAG_PRESENT)
  473. ? "supported"
  474. : "-"
  475. );
  476. }
  477. break;
  478. }
  479. }
  480. }
  481. {
  482. IDXGIOutput* output;
  483. hr = (*_swapChain)->GetContainingOutput(&output);
  484. if (SUCCEEDED(hr) )
  485. {
  486. IDXGIOutput6* output6;
  487. hr = output->QueryInterface(IID_IDXGIOutput6, (void**)&output6);
  488. if (SUCCEEDED(hr) )
  489. {
  490. DXGI_OUTPUT_DESC1 desc;
  491. hr = output6->GetDesc1(&desc);
  492. if (SUCCEEDED(hr) )
  493. {
  494. BX_TRACE("Display specs:")
  495. BX_TRACE("\t BitsPerColor: %d", desc.BitsPerColor);
  496. BX_TRACE("\t Color space: %s (colorspace, range, gamma, sitting, primaries, transform)"
  497. , s_colorSpaceStr[bx::min<uint32_t>(desc.ColorSpace, kDxgiLastColorSpace+1)]
  498. );
  499. BX_TRACE("\t RedPrimary: %f, %f", desc.RedPrimary[0], desc.RedPrimary[1]);
  500. BX_TRACE("\t GreenPrimary: %f, %f", desc.GreenPrimary[0], desc.GreenPrimary[1]);
  501. BX_TRACE("\t BluePrimary: %f, %f", desc.BluePrimary[0], desc.BluePrimary[1]);
  502. BX_TRACE("\t WhitePoint: %f, %f", desc.WhitePoint[0], desc.WhitePoint[1]);
  503. BX_TRACE("\t MinLuminance: %f", desc.MinLuminance);
  504. BX_TRACE("\t MaxLuminance: %f", desc.MaxLuminance);
  505. BX_TRACE("\tMaxFullFrameLuminance: %f", desc.MaxFullFrameLuminance);
  506. }
  507. DX_RELEASE(output6, 1);
  508. }
  509. DX_RELEASE(output, 0);
  510. }
  511. }
  512. #endif // BX_PLATFORM_WINDOWS
  513. return S_OK;
  514. }
  515. void Dxgi::trim()
  516. {
  517. #if BX_PLATFORM_WINDOWS || BX_PLATFORM_WINRT
  518. IDXGIDevice3* device;
  519. HRESULT hr = m_factory->QueryInterface(IID_IDXGIDevice3, (void**)&device);
  520. if (SUCCEEDED(hr) )
  521. {
  522. device->Trim();
  523. DX_RELEASE(device, 1);
  524. }
  525. #endif // BX_PLATFORM_WINDOWS || BX_PLATFORM_WINRT
  526. }
  527. } // namespace bgfx
  528. #endif // BGFX_CONFIG_RENDERER_DIRECT3D11 || BGFX_CONFIG_RENDERER_DIRECT3D12