SetupDiGetClassDevs是否与文档中的设备实例ID一起使用?

时间:2009-06-05 15:55:52

标签: windows winapi device hardware-interface setupapi

根据MSDN文档,SetupDiGetClassDevs可以通过device instance ID获取特定设备的device information set

  

要仅返回特定设备,请进行设置   DIFCF_DEVICEINTERFACE标志并使用   提供的Enumerator参数   设备的设备实例ID。

我通过解析WM_DEVICECHANGE消息DBT_DEVICEARRIVAL事件中的符号名称获取设备实例ID,并通过将结果ID与SetupDiGetDeviceInstanceId返回的ID进行比较来验证结果ID。即使传递操作系统提供的设备实例ID也不起作用(即SetupDiGetClassDevs调用失败并显示ERROR_INVALID_PARAMETER)。

我目前为新到设备获取SP_DEVINFO_DATA结构的解决方法是枚举同一类中的所有设备,并将SetupDiGetDeviceInstanceId的结果与符号名称进行比较。但是,根据文件......我不明白为什么这是必要的。

有没有人让SetupDiGetClassDevs以这种方式工作?有没有更好的方法可以使用DBT_DEVICEARRIVAL事件中的数据获取设备的更多信息?

2 个答案:

答案 0 :(得分:9)

您似乎必须指定DIGCF_ALLCLASSES标志才能找到与给定设备实例ID匹配的所有类,否则请指定ClassGuid并使用DIGCF_DEFAULT标志。

这对我有用:

void error(DWORD err)
{
    WCHAR buf[0x200];
    FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM, NULL, err, 0, buf, 0x200, NULL);
    wprintf(L"%x: %s\n", err,  buf);
}


int _tmain(int argc, _TCHAR* argv[])
{
    PCWSTR devinst = L"HID\\VID_413C&PID_2105\\6&22CE0F66&0&0000";
    HDEVINFO hinfo = SetupDiGetClassDevs(NULL, devinst, NULL, DIGCF_DEVICEINTERFACE | DIGCF_ALLCLASSES);
    if (hinfo == INVALID_HANDLE_VALUE)
    {
        error(GetLastError());
        return 1;
    }

    SP_DEVINFO_DATA dinfo;
    dinfo.cbSize = sizeof(dinfo);
    int ix = 0;
    while (SetupDiEnumDeviceInfo(hinfo, ix++, &dinfo))
    {
        wprintf(L"Match\n");
    }

    error(GetLastError());

    SetupDiDestroyDeviceInfoList(hinfo);
    return 0;
}

输出:

Match
103: No more data is available.

答案 1 :(得分:4)

您似乎误解了DBT_DEVICEARRIVAL

有一些不同类型的DBT_DEVICEARRIVAL消息 - 用于卷,用于句柄,用于设备接口。我猜你在谈论DBT_DEVTYP_DEVICEINTERFACE变种。在这种情况下,dbcc_name结构的DEV_BROADCAST_DEVICEINTERFACE字段将包含“设备接口路径”。

“设备接口路径”与“设备实例ID”不同。

如果您想了解有关此设备的更多信息,您应该通过此设备接口GUID枚举所有设备接口(通过带有DIGCF_DEVICEINTERFACE的SetupDiGetClassDevs),并将dbcc_name与{{检索到的字符串进行比较1}}。