| // Copyright (c) 2011 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| // |
| // Functions to enumerate the Dx Diagnostic Tool hierarchy and build up |
| // a tree of nodes with name / value properties. |
| |
| #define INITGUID |
| #include <dxdiag.h> |
| #include <windows.h> |
| |
| #include "base/strings/string_number_conversions.h" |
| #include "base/strings/utf_string_conversions.h" |
| #include "base/win/scoped_com_initializer.h" |
| #include "gpu/config/gpu_info_collector.h" |
| |
| namespace gpu { |
| |
| namespace { |
| |
| // Traverses the IDxDiagContainer tree and populates a tree of DxDiagNode |
| // structures that contains property name / value pairs and subtrees of DirectX |
| // diagnostic information. |
| void RecurseDiagnosticTree(DxDiagNode* output, |
| IDxDiagContainer* container, |
| int depth) { |
| HRESULT hr; |
| |
| VARIANT variant; |
| VariantInit(&variant); |
| |
| DWORD prop_count; |
| hr = container->GetNumberOfProps(&prop_count); |
| if (SUCCEEDED(hr)) { |
| for (DWORD i = 0; i < prop_count; i++) { |
| WCHAR prop_name16[256]; |
| hr = container->EnumPropNames(i, prop_name16, arraysize(prop_name16)); |
| if (SUCCEEDED(hr)) { |
| std::string prop_name8 = WideToUTF8(prop_name16); |
| |
| hr = container->GetProp(prop_name16, &variant); |
| if (SUCCEEDED(hr)) { |
| switch (variant.vt) { |
| case VT_UI4: |
| output->values[prop_name8] = base::UintToString(variant.ulVal); |
| break; |
| case VT_I4: |
| output->values[prop_name8] = base::IntToString(variant.lVal); |
| break; |
| case VT_BOOL: |
| output->values[prop_name8] = variant.boolVal ? "true" : "false"; |
| break; |
| case VT_BSTR: |
| output->values[prop_name8] = WideToUTF8(variant.bstrVal); |
| break; |
| default: |
| break; |
| } |
| |
| // Clear the variant (this is needed to free BSTR memory). |
| VariantClear(&variant); |
| } |
| } |
| } |
| } |
| |
| if (depth > 0) { |
| DWORD child_count; |
| hr = container->GetNumberOfChildContainers(&child_count); |
| if (SUCCEEDED(hr)) { |
| for (DWORD i = 0; i < child_count; i++) { |
| WCHAR child_name16[256]; |
| hr = container->EnumChildContainerNames(i, |
| child_name16, |
| arraysize(child_name16)); |
| if (SUCCEEDED(hr)) { |
| std::string child_name8 = WideToUTF8(child_name16); |
| DxDiagNode* output_child = &output->children[child_name8]; |
| |
| IDxDiagContainer* child_container = NULL; |
| hr = container->GetChildContainer(child_name16, &child_container); |
| if (SUCCEEDED(hr)) { |
| RecurseDiagnosticTree(output_child, child_container, depth - 1); |
| |
| child_container->Release(); |
| } |
| } |
| } |
| } |
| } |
| } |
| } // namespace anonymous |
| |
| bool GetDxDiagnostics(DxDiagNode* output) { |
| HRESULT hr; |
| bool success = false; |
| base::win::ScopedCOMInitializer com_initializer; |
| |
| IDxDiagProvider* provider = NULL; |
| hr = CoCreateInstance(CLSID_DxDiagProvider, |
| NULL, |
| CLSCTX_INPROC_SERVER, |
| IID_IDxDiagProvider, |
| reinterpret_cast<void**>(&provider)); |
| if (SUCCEEDED(hr)) { |
| DXDIAG_INIT_PARAMS params = { sizeof(params) }; |
| params.dwDxDiagHeaderVersion = DXDIAG_DX9_SDK_VERSION; |
| params.bAllowWHQLChecks = FALSE; |
| params.pReserved = NULL; |
| |
| hr = provider->Initialize(¶ms); |
| if (SUCCEEDED(hr)) { |
| IDxDiagContainer* root = NULL; |
| hr = provider->GetRootContainer(&root); |
| if (SUCCEEDED(hr)) { |
| // Limit to the DisplayDevices subtree. The tree in its entirity is |
| // enormous and only this branch contains useful information. |
| IDxDiagContainer* display_devices = NULL; |
| hr = root->GetChildContainer(L"DxDiag_DisplayDevices", |
| &display_devices); |
| if (SUCCEEDED(hr)) { |
| RecurseDiagnosticTree(output, display_devices, 1); |
| success = true; |
| display_devices->Release(); |
| } |
| |
| root->Release(); |
| } |
| } |
| provider->Release(); |
| } |
| |
| return success; |
| } |
| } // namespace gpu |