如何在Windows上获取插入的USB设备的供应商ID和产品ID

时间:2010-05-29 13:12:29

标签: c++ windows qt winapi usb

我在Windows平台上使用Qt。

我希望从本地系统获取并显示插入的USB设备的供应商ID和产品ID。

以下是我从USB设备获取供应商ID和产品ID的完整源代码。

当我运行我的qt应用程序时,它不会给我任何错误。

所以我将usb设备插入系统。

但我的print语句显示结果如下

qDebug ()<<pDetData->DevicePath;

我得到的结果为0x4

我的源代码中是否有任何实现错误?

如果是的话请指导我做错了什么..

我错过了其他任何功能吗?

是否可以根据我的源代码从usb设备获取供应商ID和产品ID。(我的代码实现)?

请在下面找到我的源代码

static GUID GUID_DEVINTERFACE_USB_DEVICE = { 0xA5DCBF10L, 0x6530, 0x11D2, 
    { 0x90, 0x1F, 0x00, 0xC0, 0x4F, 0xB9, 0x51, 0xED } };

HANDLE hInfo = SetupDiGetClassDevs(&GUID_DEVINTERFACE_USB_DEVICE,NULL,NULL,
    DIGCF_PRESENT | DIGCF_INTERFACEDEVICE);

if ( hInfo == INVALID_HANDLE_VALUE )    
{    
    qDebug ()<<"invalid";   
}    
else    
{        
    qDebug ()<<"valid handle";    

    SP_DEVINFO_DATA DeviceInfoData;
    DeviceInfoData.cbSize = sizeof(SP_DEVINFO_DATA);

    SP_INTERFACE_DEVICE_DATA Interface_Info;    
    Interface_Info.cbSize = sizeof(Interface_Info);

    BYTE Buf[1024];
    DWORD i;
    DWORD InterfaceNumber= 0;

    PSP_DEVICE_INTERFACE_DETAIL_DATA pspdidd = 
        (PSP_DEVICE_INTERFACE_DETAIL_DATA) Buf;

    for (i=0;SetupDiEnumDeviceInfo(hInfo,i,&DeviceInfoData);i++)
    {
        DWORD DataT;
        LPTSTR buffer = NULL;
        DWORD buffersize = 0;

        while (!SetupDiGetDeviceRegistryProperty( hInfo,
            &DeviceInfoData,
            SPDRP_DEVICEDESC,
            &DataT,
            (PBYTE)buffer,
            buffersize,
            &buffersize))    
         {
            if (GetLastError() == ERROR_INSUFFICIENT_BUFFER)
            {
                // Change the buffer size.
                if (buffer) LocalFree(buffer);
                buffer = (LPTSTR)LocalAlloc(LPTR,buffersize);
            }
            else
            {
                // Insert error handling here.
                break;
            }

            qDebug ()<<(TEXT("Device Number %i is: %s\n"),i, buffer);

            if (buffer) LocalFree(buffer);

            if ( GetLastError() != NO_ERROR 
                     && GetLastError() != ERROR_NO_MORE_ITEMS )    
            {
                // Insert error handling here.
                qDebug ()<<"return false";
            }

            InterfaceNumber = 0; // this just returns the first one, you can iterate on this

            if (SetupDiEnumDeviceInterfaces(hInfo,
                                   NULL, 
                                   &GUID_DEVINTERFACE_USB_DEVICE,
                                   InterfaceNumber,
                                   &Interface_Info))
            {
                printf("Got interface");
                DWORD needed;
                pspdidd->cbSize = sizeof(*pspdidd);    
                SP_DEVICE_INTERFACE_DETAIL_DATA *pDetData = NULL;
                DWORD dwDetDataSize = sizeof(SP_DEVICE_INTERFACE_DETAIL_DATA) 
                                      + 256;

                SetupDiGetDeviceInterfaceDetail(hInfo, 
                    &Interface_Info, pDetData,dwDetDataSize, NULL,
                    &DeviceInfoData);

                qDebug ()<<pDetData->DevicePath;
                //qDebug ()<<QString::fromWCharArray(pDetData->DevicePath);
            }
            else
            {    
                printf("\nNo interface");

                //ErrorExit((LPTSTR) "SetupDiEnumDeviceInterfaces");

                if ( GetLastError() == ERROR_NO_MORE_ITEMS) 
                    printf(", since there are no more items found.");
                else 
                    printf(", unknown reason.");

            }
            // Cleanup

            SetupDiDestroyDeviceInfoList(hInfo);
            qDebug ()<<"return true";
        }
    }
}

---------------编辑添加:-----------------

嗨......应用程序出现并打印出来

  

\?\ usb#vid_04f2&amp; pid_0111#5&amp; 1ba5a77f&amp; 0&amp; 2#{a5dcbf1 0-6530-11d2-901f-00c04fb951ed}

再次进入while循环....这里它在else语句中被破坏了......

Qt代码:

if (GetLastError() == ERROR_INSUFFICIENT_BUFFER) { 
    // Change the buffer size. 
    if (buffer) LocalFree(buffer); 
    buffer = (LPTSTR)LocalAlloc(LPTR,buffersize); 
} else { 
    qDebug ()<<"Here it quits the application"; 
    // Insert error handling here. break; 
} 

这个想法......

3 个答案:

答案 0 :(得分:8)

这一行之后:

SP_DEVICE_INTERFACE_DETAIL_DATA *pDetData = NULL;

添加:

DWORD dwDetDataSize = sizeof (SP_DEVICE_INTERFACE_DETAIL_DATA) + 256;
pDetData = (_SP_DEVICE_INTERFACE_DETAIL_DATA_A*) malloc (dwDetDataSize);
pDetData->cbSize = sizeof (SP_DEVICE_INTERFACE_DETAIL_DATA);

这一行之后:

qDebug ()<<pDetData->DevicePath;

添加:

free(pDetData);

但最终你将不得不阅读SetupDiGetDeviceInterfaceDetail()的文档。这样做,有很多函数可以像这样工作,并指向可变大小的结构。

--------编辑添加:--------

你真的是以错误的方式解决这个问题。我看到你正在遵循你所得到的建议here,这是你走错了路。 idVendoridProduct只能在USB_DEVICE_DESCRIPTORMSDN)中找到。

看起来您已经知道如何获取设备句柄(使用CreateFile())。之后,您拨打WinUsb_Initialize()MSDN)。这会让你WINUSB_INTERFACE_HANDLE

完成此操作后,您需要调用WinUsb_GetDescriptor()MSDN),DescriptorType设置为URB_FUNCTION_GET_DESCRIPTOR_FROM_DEVICE。我现在无法测试代码,但它看起来像这样:

USB_DEVICE_DESCRIPTOR udd;
memset(&udd, 0, sizeof(udd));
ULONG LengthTransferred = 0;

WinUsb_GetDescriptor(
    winusb_interface_handle, // returned by WinUsbInitialize
    URB_FUNCTION_GET_DESCRIPTOR_FROM_DEVICE,
    0,     // not sure if we need this
    0x409, // not sure if we need this
    &udd,
    sizeof(udd),
    &LengthTransferred);

之后,udd->idVendorudd->idProduct应该拥有您想要的内容。

Microsoft过去常常在DDK中为所有这些提供示例代码,但可能仍然如此,但我无权访问。

----------编辑添加:----------

Daniel K写道,代码应该是:

USB_DEVICE_DESCRIPTOR udd;
memset(&udd, 0, sizeof(udd));
ULONG LengthTransferred = 0;

WinUsb_GetDescriptor(
    winusb_interface_handle,    // returned by WinUsbInitialize
    USB_DEVICE_DESCRIPTOR_TYPE, // Daniel K's suggestion
    0,
    0x409,     // asks for English
    &udd,
    sizeof(udd),
    &LengthTransferred);

有关详细信息,请参阅评论。

答案 1 :(得分:1)

另一种方法是获取包含VID和PID的hardwareID。

使用SPDRP_HARDWAREID调用SetupDiGetDeviceRegistryProperty,如下所示:

wchar_t *hardwareID;

// First get requiredLength
SetupDiGetDeviceRegistryProperty(deviceInfoList, &deviceInfoData, SPDRP_HARDWAREID, NULL, NULL, 0, &requiredLength);

hardwareID = (wchar_t*)(new char[requiredLength]());

// Second call to populate hardwareID
SetupDiGetDeviceRegistryProperty(deviceInfoList, &deviceInfoData, SPDRP_HARDWAREID, NULL, (PBYTE)hardwareID, requiredLength, NULL);

// Display the string
qDebug() << "hardwareID =" << QString::fromWCharArray(hardwareID);

这将为您提供一个可以解析的USB\ROOT_HUB20&VID1002&PID4396&REV0000字符串。

*注意:并非所有设备都具有VID和PID,例如非USB设备。

答案 2 :(得分:0)

您正在枚举设备“界面”。接口没有VID或PID - 设备实例。我不确定您是否在枚举接口以缩小您感兴趣的设备范围,因为这是一个错误。

如果您只是枚举设备实例,那么可以使用DEVPKEY_Device_HardwareIds调用SetupDiGetDeviceProperty,然后grep生成VID和PID的硬件ID。

如果您正在使用设备接口,则需要使用NULL PSP_DEVICE_INTERFACE_DETAIL参数和有效的requiredSize指针调用SetupDiGetDeviceInterfaceDetail一次,以获取要分配的所需内存大小,分配该内存,然后再次调用该函数。在该调用中,最后一个参数是SP_DEVINFO_DATA结构,一旦检索到,您就可以使用上面提到的SetupDiGetDeviceProperty调用。