| // Copyright (c) 2010 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. |
| |
| #include "base/win/wmi.h" |
| |
| #include <windows.h> |
| |
| #include <objbase.h> |
| #include <stdint.h> |
| #include <utility> |
| |
| #include "base/win/scoped_bstr.h" |
| #include "base/win/scoped_variant.h" |
| |
| using Microsoft::WRL::ComPtr; |
| |
| namespace base { |
| namespace win { |
| |
| bool CreateLocalWmiConnection(bool set_blanket, |
| ComPtr<IWbemServices>* wmi_services) { |
| ComPtr<IWbemLocator> wmi_locator; |
| HRESULT hr = |
| ::CoCreateInstance(CLSID_WbemLocator, nullptr, CLSCTX_INPROC_SERVER, |
| IID_PPV_ARGS(&wmi_locator)); |
| if (FAILED(hr)) |
| return false; |
| |
| ComPtr<IWbemServices> wmi_services_r; |
| hr = wmi_locator->ConnectServer(ScopedBstr(L"ROOT\\CIMV2"), nullptr, nullptr, |
| nullptr, 0, nullptr, nullptr, |
| wmi_services_r.GetAddressOf()); |
| if (FAILED(hr)) |
| return false; |
| |
| if (set_blanket) { |
| hr = ::CoSetProxyBlanket(wmi_services_r.Get(), RPC_C_AUTHN_WINNT, |
| RPC_C_AUTHZ_NONE, nullptr, RPC_C_AUTHN_LEVEL_CALL, |
| RPC_C_IMP_LEVEL_IMPERSONATE, nullptr, EOAC_NONE); |
| if (FAILED(hr)) |
| return false; |
| } |
| |
| *wmi_services = std::move(wmi_services_r); |
| return true; |
| } |
| |
| bool CreateWmiClassMethodObject(IWbemServices* wmi_services, |
| const StringPiece16& class_name, |
| const StringPiece16& method_name, |
| ComPtr<IWbemClassObject>* class_instance) { |
| // We attempt to instantiate a COM object that represents a WMI object plus |
| // a method rolled into one entity. |
| ScopedBstr b_class_name(class_name); |
| ScopedBstr b_method_name(method_name); |
| ComPtr<IWbemClassObject> class_object; |
| HRESULT hr; |
| hr = wmi_services->GetObject(b_class_name, 0, nullptr, |
| class_object.GetAddressOf(), nullptr); |
| if (FAILED(hr)) |
| return false; |
| |
| ComPtr<IWbemClassObject> params_def; |
| hr = class_object->GetMethod(b_method_name, 0, params_def.GetAddressOf(), |
| nullptr); |
| if (FAILED(hr)) |
| return false; |
| |
| if (!params_def.Get()) { |
| // You hit this special case if the WMI class is not a CIM class. MSDN |
| // sometimes tells you this. Welcome to WMI hell. |
| return false; |
| } |
| |
| hr = params_def->SpawnInstance(0, class_instance->GetAddressOf()); |
| return SUCCEEDED(hr); |
| } |
| |
| bool SetWmiClassMethodParameter(IWbemClassObject* class_method, |
| const StringPiece16& parameter_name, |
| VARIANT* parameter) { |
| HRESULT hr = class_method->Put(parameter_name.data(), 0, parameter, 0); |
| return SUCCEEDED(hr); |
| } |
| |
| // The code in Launch() basically calls the Create Method of the Win32_Process |
| // CIM class is documented here: |
| // http://msdn2.microsoft.com/en-us/library/aa389388(VS.85).aspx |
| // NOTE: The documentation for the Create method suggests that the ProcessId |
| // parameter and return value are of type uint32_t, but when we call the method |
| // the values in the returned out_params, are VT_I4, which is int32_t. |
| bool WmiLaunchProcess(const string16& command_line, int* process_id) { |
| ComPtr<IWbemServices> wmi_local; |
| if (!CreateLocalWmiConnection(true, &wmi_local)) |
| return false; |
| |
| static constexpr wchar_t class_name[] = L"Win32_Process"; |
| static constexpr wchar_t method_name[] = L"Create"; |
| ComPtr<IWbemClassObject> process_create; |
| if (!CreateWmiClassMethodObject(wmi_local.Get(), class_name, method_name, |
| &process_create)) { |
| return false; |
| } |
| |
| ScopedVariant b_command_line(command_line.c_str()); |
| |
| if (!SetWmiClassMethodParameter(process_create.Get(), L"CommandLine", |
| b_command_line.AsInput())) { |
| return false; |
| } |
| |
| ComPtr<IWbemClassObject> out_params; |
| HRESULT hr = wmi_local->ExecMethod( |
| ScopedBstr(class_name), ScopedBstr(method_name), 0, nullptr, |
| process_create.Get(), out_params.GetAddressOf(), nullptr); |
| if (FAILED(hr)) |
| return false; |
| |
| // We're only expecting int32_t or uint32_t values, so no need for |
| // ScopedVariant. |
| VARIANT ret_value = {{{VT_EMPTY}}}; |
| hr = out_params->Get(L"ReturnValue", 0, &ret_value, nullptr, 0); |
| if (FAILED(hr) || V_I4(&ret_value) != 0) |
| return false; |
| |
| VARIANT pid = {{{VT_EMPTY}}}; |
| hr = out_params->Get(L"ProcessId", 0, &pid, nullptr, 0); |
| if (FAILED(hr) || V_I4(&pid) == 0) |
| return false; |
| |
| if (process_id) |
| *process_id = V_I4(&pid); |
| |
| return true; |
| } |
| |
| WmiComputerSystemInfo WmiComputerSystemInfo::Get() { |
| ComPtr<IWbemServices> services; |
| if (!CreateLocalWmiConnection(true, &services)) |
| return WmiComputerSystemInfo(); |
| |
| ScopedBstr query_language(L"WQL"); |
| ScopedBstr query(L"SELECT * FROM Win32_ComputerSystem"); |
| ComPtr<IEnumWbemClassObject> enumerator; |
| HRESULT hr = |
| services->ExecQuery(query_language, query, |
| WBEM_FLAG_FORWARD_ONLY | WBEM_FLAG_RETURN_IMMEDIATELY, |
| nullptr, enumerator.GetAddressOf()); |
| if (FAILED(hr) || !enumerator.Get()) |
| return WmiComputerSystemInfo(); |
| |
| ComPtr<IWbemClassObject> class_object; |
| ULONG items_returned = 0; |
| hr = enumerator->Next(WBEM_INFINITE, 1, class_object.GetAddressOf(), |
| &items_returned); |
| if (!items_returned) |
| return WmiComputerSystemInfo(); |
| |
| ScopedVariant manufacturer; |
| class_object->Get(L"Manufacturer", 0, manufacturer.Receive(), 0, 0); |
| ScopedVariant model; |
| class_object->Get(L"Model", 0, model.Receive(), 0, 0); |
| |
| WmiComputerSystemInfo info; |
| if (manufacturer.type() == VT_BSTR) |
| info.manufacturer_ = V_BSTR(manufacturer.ptr()); |
| |
| if (model.type() == VT_BSTR) |
| info.model_ = V_BSTR(model.ptr()); |
| |
| return info; |
| } |
| |
| } // namespace win |
| } // namespace base |