как получить идентификатор поставщика и идентификатор продукта подключенного usb-устройства в windows
Я использую Qt на платформе windows.
Я хочу получить и отобразить идентификатор поставщика и идентификатор продукта подключенного usb-устройства из моей локальной системы.
Ниже приведен мой полный исходный код, чтобы получить идентификатор поставщика и идентификатор продукта с usb-устройства.
когда я запускаю приложение my qt, оно не выдает мне никаких ошибок .
поэтому я подключаю usb-устройство к системе.
но моя инструкция печати отображает результат как ниже
qDebug ()<<pDetData->DevicePath;
Я получаю результат как 0x4
есть ли у меня какие-либо ошибки реализации в исходном коде ?
если это так, пожалуйста, руководство мне, что я делаю неправильно..
я пропустил какие-то другие функции ?
можно ли получить идентификатор поставщика и идентификатор продукта с usb-устройства на основе моего исходного кода .(моя реализация кодекса)?
пожалуйста, найдите мой исходный код ниже
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: %sn"),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 & pid_0111#5&1ba5a77f&0 & 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 ответов
после этой строки:
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()
. Сделайте это, есть много функций, которые работают так, с указателями на структуры переменного размера.
-------- отредактировано для добавления: --------
ты действительно идешь по неправильному пути. Я смотрите, вы следуете совету, который вы получили здесь, и это привело вас по неправильному пути. idVendor
и idProduct
можно найти только в USB_DEVICE_DESCRIPTOR
(MSDN).
похоже, вы уже знаете, как получить дескриптор устройства (используя 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->idVendor
и udd->idProduct
должно быть то, что вы хотите.
Microsoft использовала образец кода для всего этого в DDK и, вероятно, все еще делает, но у меня нет доступа к одному.
---------- отредактировано для добавления: ----------
Даниил к пишет, что код действительно должен быть:
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);
см. комментарии более подробная информация.
альтернативой является, чтобы получить hardwareidне, которая включает VID и PID.
вызов SetupDiGetDeviceRegistryProperty с помощью SPDRP_HARDWAREID следующим образом:
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.
вы перечисляете устройство "интерфейс". Интерфейсы не имеют экземпляров VID или PID-устройств. Я не уверен, перечисляете ли вы интерфейсы, чтобы сузить интересующие вас устройства, потому что это ошибка.
Если вы просто перечислите экземпляры устройства, то вы можете вызвать SetupDiGetDeviceProperty с DEVPKEY_Device_HardwareIds, а затем grep результирующий идентификатор оборудования для VID и PID.
Если вы используете устройство интерфейсы специально, затем вам нужно вызвать SetupDiGetDeviceInterfaceDetail один раз с нулевым параметром PSP_DEVICE_INTERFACE_DETAIL и допустимым указателем requiredSize, чтобы получить необходимый размер памяти для выделения, выделения этой памяти, а затем снова вызвать функцию. В этом вызове последним параметром является структура SP_DEVINFO_DATA, которую после извлечения можно использовать в вызове SetupDiGetDeviceProperty, как я упоминал выше.