dxgi.cpp 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610
  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. static const GUID IID_ID3D12CommandQueue = { 0x0ec870a6, 0x5d7e, 0x4c22, { 0x8c, 0xfc, 0x5b, 0xaa, 0xe0, 0x76, 0x16, 0xed } };
  351. HRESULT Dxgi::createSwapChain(IUnknown* _device, const SwapChainDesc& _scd, SwapChainI** _swapChain)
  352. {
  353. HRESULT hr = S_OK;
  354. bool allowTearing = false;
  355. #if BX_PLATFORM_WINDOWS
  356. if (windowsVersionIs(Condition::GreaterEqual, 0x0602) )
  357. {
  358. // BK - CheckFeatureSupport with DXGI_FEATURE_PRESENT_ALLOW_TEARING
  359. // will crash on pre Windows 8. Issue #1356.
  360. hr = m_factory->CheckFeatureSupport(DXGI_FEATURE_PRESENT_ALLOW_TEARING, &allowTearing, sizeof(allowTearing) );
  361. BX_TRACE("Allow tearing is %ssupported.", allowTearing ? "" : "not ");
  362. }
  363. DXGI_SWAP_CHAIN_DESC scd;
  364. scd.BufferDesc.Width = _scd.width;
  365. scd.BufferDesc.Height = _scd.height;
  366. scd.BufferDesc.RefreshRate.Numerator = 1;
  367. scd.BufferDesc.RefreshRate.Denominator = 60;
  368. scd.BufferDesc.Format = _scd.format;
  369. scd.BufferDesc.ScanlineOrdering = DXGI_MODE_SCANLINE_ORDER_UNSPECIFIED;
  370. scd.BufferDesc.Scaling = DXGI_MODE_SCALING_UNSPECIFIED;
  371. scd.SampleDesc = _scd.sampleDesc;
  372. scd.BufferUsage = _scd.bufferUsage;
  373. scd.BufferCount = _scd.bufferCount;
  374. scd.OutputWindow = (HWND)_scd.nwh;
  375. scd.Windowed = _scd.windowed;
  376. scd.SwapEffect = _scd.swapEffect;
  377. scd.Flags = 0
  378. | _scd.flags
  379. | (allowTearing ? DXGI_SWAP_CHAIN_FLAG_ALLOW_TEARING : 0)
  380. ;
  381. if (isType(_device, IID_ID3D12CommandQueue) )
  382. {
  383. scd.SampleDesc.Count = 1;
  384. scd.SampleDesc.Quality = 0;
  385. }
  386. hr = m_factory->CreateSwapChain(
  387. _device
  388. , &scd
  389. , reinterpret_cast<IDXGISwapChain**>(_swapChain)
  390. );
  391. #else
  392. DXGI_SWAP_CHAIN_DESC1 scd;
  393. scd.Width = _scd.width;
  394. scd.Height = _scd.height;
  395. scd.Format = _scd.format;
  396. scd.Stereo = _scd.stereo;
  397. scd.SampleDesc = _scd.sampleDesc;
  398. scd.BufferUsage = _scd.bufferUsage;
  399. scd.BufferCount = _scd.bufferCount;
  400. scd.Scaling = _scd.scaling;
  401. scd.SwapEffect = _scd.swapEffect;
  402. scd.AlphaMode = _scd.alphaMode;
  403. scd.Flags = _scd.flags;
  404. if (isType(_device, IID_ID3D12CommandQueue) )
  405. {
  406. scd.SampleDesc.Count = 1;
  407. scd.SampleDesc.Quality = 0;
  408. }
  409. if (NULL == _scd.ndt)
  410. {
  411. hr = m_factory->CreateSwapChainForCoreWindow(
  412. _device
  413. , (::IUnknown*)_scd.nwh
  414. , &scd
  415. , NULL
  416. , reinterpret_cast<IDXGISwapChain1**>(_swapChain)
  417. );
  418. }
  419. else if (reinterpret_cast<void*>(1) == _scd.ndt)
  420. {
  421. return E_FAIL;
  422. }
  423. else
  424. {
  425. hr = m_factory->CreateSwapChainForComposition(
  426. _device
  427. , &scd
  428. , NULL
  429. , reinterpret_cast<IDXGISwapChain1**>(_swapChain)
  430. );
  431. if (FAILED(hr) )
  432. {
  433. return hr;
  434. }
  435. # if BX_PLATFORM_WINRT
  436. IInspectable *nativeWindow = reinterpret_cast<IInspectable *>(_scd.nwh);
  437. ISwapChainBackgroundPanelNative* panel = NULL;
  438. hr = nativeWindow->QueryInterface(
  439. __uuidof(ISwapChainBackgroundPanelNative)
  440. , (void **)&panel
  441. );
  442. if (FAILED(hr) )
  443. {
  444. return hr;
  445. }
  446. if (NULL != panel)
  447. {
  448. hr = panel->SetSwapChain(*_swapChain);
  449. if (FAILED(hr) )
  450. {
  451. return hr;
  452. }
  453. panel->Release();
  454. }
  455. # endif // BX_PLATFORM_WINRT
  456. }
  457. #endif // BX_PLATFORM_WINDOWS
  458. if (FAILED(hr) )
  459. {
  460. BX_TRACE("Failed to create swap chain.");
  461. return hr;
  462. }
  463. #if BX_PLATFORM_WINDOWS
  464. if (SUCCEEDED(hr) )
  465. {
  466. for (uint32_t ii = 0; ii < BX_COUNTOF(s_dxgiSwapChainIIDs); ++ii)
  467. {
  468. IDXGISwapChain1* swapChain;
  469. hr = (*_swapChain)->QueryInterface(s_dxgiSwapChainIIDs[ii], (void**)&swapChain);
  470. BX_TRACE("DXGI swap chain %d, hr %x", 4-ii, hr);
  471. if (SUCCEEDED(hr) )
  472. {
  473. DX_RELEASE(*_swapChain, 1);
  474. *_swapChain = reinterpret_cast<SwapChainI*>(swapChain);
  475. BX_TRACE("Color space support:");
  476. for (uint32_t jj = 0; jj < BX_COUNTOF(s_colorSpace); ++jj)
  477. {
  478. uint32_t colorSpaceSupport;
  479. reinterpret_cast<IDXGISwapChain3*>(*_swapChain)->CheckColorSpaceSupport(s_colorSpace[jj], &colorSpaceSupport);
  480. BX_TRACE("\t%2d, 0x%08x, %s"
  481. , s_colorSpace[jj]
  482. , colorSpaceSupport
  483. , 0 != (colorSpaceSupport & DXGI_SWAP_CHAIN_COLOR_SPACE_SUPPORT_FLAG_PRESENT)
  484. ? "supported"
  485. : "-"
  486. );
  487. }
  488. break;
  489. }
  490. }
  491. }
  492. {
  493. IDXGIOutput* output;
  494. hr = (*_swapChain)->GetContainingOutput(&output);
  495. if (SUCCEEDED(hr) )
  496. {
  497. IDXGIOutput6* output6;
  498. hr = output->QueryInterface(IID_IDXGIOutput6, (void**)&output6);
  499. if (SUCCEEDED(hr) )
  500. {
  501. DXGI_OUTPUT_DESC1 desc;
  502. hr = output6->GetDesc1(&desc);
  503. if (SUCCEEDED(hr) )
  504. {
  505. BX_TRACE("Display specs:")
  506. BX_TRACE("\t BitsPerColor: %d", desc.BitsPerColor);
  507. BX_TRACE("\t Color space: %s (colorspace, range, gamma, sitting, primaries, transform)"
  508. , s_colorSpaceStr[bx::min<uint32_t>(desc.ColorSpace, kDxgiLastColorSpace+1)]
  509. );
  510. BX_TRACE("\t RedPrimary: %f, %f", desc.RedPrimary[0], desc.RedPrimary[1]);
  511. BX_TRACE("\t GreenPrimary: %f, %f", desc.GreenPrimary[0], desc.GreenPrimary[1]);
  512. BX_TRACE("\t BluePrimary: %f, %f", desc.BluePrimary[0], desc.BluePrimary[1]);
  513. BX_TRACE("\t WhitePoint: %f, %f", desc.WhitePoint[0], desc.WhitePoint[1]);
  514. BX_TRACE("\t MinLuminance: %f", desc.MinLuminance);
  515. BX_TRACE("\t MaxLuminance: %f", desc.MaxLuminance);
  516. BX_TRACE("\tMaxFullFrameLuminance: %f", desc.MaxFullFrameLuminance);
  517. }
  518. DX_RELEASE(output6, 1);
  519. }
  520. DX_RELEASE(output, 0);
  521. }
  522. }
  523. #endif // BX_PLATFORM_WINDOWS
  524. return S_OK;
  525. }
  526. void Dxgi::trim()
  527. {
  528. #if BX_PLATFORM_WINDOWS || BX_PLATFORM_WINRT
  529. IDXGIDevice3* device;
  530. HRESULT hr = m_factory->QueryInterface(IID_IDXGIDevice3, (void**)&device);
  531. if (SUCCEEDED(hr) )
  532. {
  533. device->Trim();
  534. DX_RELEASE(device, 1);
  535. }
  536. #endif // BX_PLATFORM_WINDOWS || BX_PLATFORM_WINRT
  537. }
  538. } // namespace bgfx
  539. #endif // BGFX_CONFIG_RENDERER_DIRECT3D11 || BGFX_CONFIG_RENDERER_DIRECT3D12