// Copyright 2025 Dolphin Emulator Project // SPDX-License-Identifier: GPL-2.0-or-later #ifdef _WIN32 #include "Common/WindowsDevice.h" #include #include "Hidclass.h" #include "Common/CommonFuncs.h" #include "Common/Logging/Log.h" namespace Common { std::optional GetPropertyHelper(auto function, auto dev, const DEVPROPKEY* requested_property, DEVPROPTYPE expected_type) { DEVPROPTYPE type{}; ULONG buffer_size{}; if (const auto result = function(dev, requested_property, &type, nullptr, &buffer_size, 0); result != CR_SUCCESS && result != CR_BUFFER_SMALL) { if (result != CR_NO_SUCH_VALUE) WARN_LOG_FMT(COMMON, "CM_Get_DevNode_Property returned: {}", result); return std::nullopt; } if (type != expected_type) { WARN_LOG_FMT(COMMON, "CM_Get_DevNode_Property unexpected type: 0x{:x}", type); return std::nullopt; } std::optional property; // FYI: It's legal to write the null terminator at data()[size()] of std::basic_string. property.emplace(buffer_size / sizeof(WCHAR) - 1, L'\0'); if (const auto result = function(dev, requested_property, &type, reinterpret_cast(property->data()), &buffer_size, 0); result != CR_SUCCESS) { ERROR_LOG_FMT(COMMON, "CM_Get_DevNode_Property returned: {}", result); return std::nullopt; } return property; } std::optional GetDevNodeStringProperty(DEVINST dev, const DEVPROPKEY* requested_property) { return GetPropertyHelper(CM_Get_DevNode_Property, dev, requested_property, DEVPROP_TYPE_STRING); } std::optional GetDeviceInterfaceStringProperty(LPCWSTR iface, const DEVPROPKEY* requested_property) { return GetPropertyHelper(CM_Get_Device_Interface_Property, iface, requested_property, DEVPROP_TYPE_STRING); } NullTerminatedStringList GetDeviceInterfaceList(LPGUID iface_class_guid, DEVINSTID device_id, ULONG flags) { while (true) { ULONG list_size = 0; const auto size_result = CM_Get_Device_Interface_List_Size(&list_size, iface_class_guid, device_id, flags); if (size_result != CR_SUCCESS || list_size == 0) list_size = 1; auto buffer = std::make_unique_for_overwrite(list_size); const auto list_result = CM_Get_Device_Interface_List(iface_class_guid, device_id, buffer.get(), list_size, flags); // "A new device can be added to the system causing the size returned to no longer be valid." // Microsoft recommends trying again in a loop. if (list_result == CR_BUFFER_SMALL) continue; if (list_result != CR_SUCCESS) { ERROR_LOG_FMT(COMMON, "CM_Get_Device_Interface_List: {}", list_result); buffer[0] = 0; } return {std::move(buffer)}; } } } // namespace Common #endif