根据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事件中的数据获取设备的更多信息?
答案 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}}。