2

我有一个使用 SetupAPI 在 Windows XP 上枚举 USB 设备的代码:

    HDEVINFO hDevInfo = SetupDiGetClassDevs( &_DEVINTERFACE_USB_DEVICE, 0, 0, DIGCF_DEVICEINTERFACE | DIGCF_PRESENT);

    for (DWORD i = 0; ; ++i)
    {
        SP_DEVINFO_DATA devInfo;
        devInfo.cbSize = sizeof(SP_DEVINFO_DATA);
        BOOL succ = SetupDiEnumDeviceInfo(hDevInfo, i, &devInfo);
        if (GetLastError() == ERROR_NO_MORE_ITEMS)
            break;
        if (!succ) continue;

        DWORD devClassPropRequiredSize = 0;
        succ = SetupDiGetDeviceRegistryProperty(hDevInfo, &devInfo, SPDRP_COMPATIBLEIDS, NULL, NULL, 0, &devClassPropRequiredSize);
        if (!succ)
        {
           // This shouldn't happen!
           continue;
        }
    }

它曾经工作多年,但现在我得到FALSESetupDiGetDeviceRegistryProperty,最后一个错误是“传递给系统调用的数据区域太小”。看来我的调用参数对应于这个函数的文档:http: //msdn.microsoft.com/en-us/library/windows/hardware/ff551967 (v=vs.85).aspx

有什么想法有什么问题吗?

4

1 回答 1

3

问题出在您的原始代码中:当所需的属性不存在(或者当其数据无效时,是的,他们一直懒惰选择正确的错误代码)时,SetupDiGetDeviceRegistryProperty函数可能会返回FALSE(并将最后一个错误设置为),因此您应该始终检查ERROR_INSUFFICIENT_BUFFER作为ERROR_INSUFFICIENT_BUFFER(不是)特殊情况:

DWORD devClassPropRequiredSize = 0;
succ = SetupDiGetDeviceRegistryProperty(
    hDevInfo,
    &devInfo,
    SPDRP_COMPATIBLEIDS,
    NULL, 
    NULL, 
    0,
    &devClassPropRequiredSize);

if (!succ) {
    if (ERROR_INSUFFICIENT_BUFFER == GetLastError() {
        // I may ignore this property or I may simply
        // go on, required size has been set in devClassPropRequiredSize
        // so next call should work as expected (or fail in a managed way).
    } else {
        continue; // Cannot read property size
    }
}

通常,当您读取属性大小时,您可能会简单地忽略此错误(如果devClassPropRequiredSize仍然为零,您可以将其默认为适当的常量以获得最大允许长度)。如果无法读取属性,则下一次调用SetupDiGetDeviceRegistryProperty将失败(您将在那里管理错误)但通常您能够读取值并且您的代码将顺利运行。

于 2014-01-15T09:34:46.497 回答