前面我们介绍了枚举系统中的U盘盘符(见《Windows下USB磁盘开发系列一:枚举系统中U盘的盘符》)、以及获取USB设备的信息(见《Windows下USB磁盘开发系列二:枚举系统中所有USB设备》)。有个时候我们不仅仅需要获取U盘盘符(路径),而且需要获取该U盘的硬件信息,比如厂商、friendly name、描述等等。那么我们可以通过前面两个方法,把U盘盘符和设备信息匹配起来吗?答案是肯定的,下面介绍具体的实现方法。
具体方法如下:
1,获取U盘盘符(路径);
2,对U盘路径调用CreateFile()获取U盘句柄;
3,对U盘句柄调用DeviceIoControl()获取其Device Number;
4,调用SetupDiGetClassDevs()/SetupDiEnumDeviceInfo()枚举系统中所有U盘设备;
5,调用SetupDiEnumDeviceInterfaces()/SetupDiGetDeviceInterfaceDetail()获取设备路径;
6,对U盘设备路径调用CreateFile()获取U盘设备句柄;
7,对U盘设备句柄调用DeviceIoControl()获取其Device Number;
8,判断3和7获得的Device Number,两者一致则表示该U盘盘符和设备为同一设备。
具体实现代码如下:
1,获取U盘设备列表
int get_usb_device_list(usb_device_info *usb_list, int list_size) { int usb_device_cnt = 0; char disk_path[5] = {0}; char device_path[10] = {0}; DWORD all_disk = GetLogicalDrives(); int i = 0; DWORD bytes_returned = 0; STORAGE_DEVICE_NUMBER device_num; while (all_disk && usb_device_cnt < list_size) { if ((all_disk & 0x1) == 1) { sprintf_s(disk_path, "%c:", 'A'+i); sprintf_s(device_path, "\\.\%s", disk_path); if (GetDriveTypeA(disk_path) == DRIVE_REMOVABLE) { // get this usb device id HANDLE hDevice = CreateFileA(device_path, 0, FILE_SHARE_READ|FILE_SHARE_WRITE, NULL, OPEN_EXISTING, 0, NULL); if (DeviceIoControl(hDevice, IOCTL_STORAGE_GET_DEVICE_NUMBER, NULL, 0, &device_num, sizeof(device_num), &bytes_returned, (LPOVERLAPPED) NULL)) { usb_list[usb_device_cnt].volume = 'A' + i; usb_list[usb_device_cnt].device_num = device_num.DeviceNumber; usb_device_cnt++; } CloseHandle(hDevice); hDevice = 0; } } all_disk = all_disk >> 1; i++; } return usb_device_cnt; }
2,匹配U盘设备信息
int get_usb_device_friendname(usb_device_info *usb_list, int list_size) { int i = 0; int res = 0; HDEVINFO hDevInfo; SP_DEVINFO_DATA DeviceInfoData = {sizeof(DeviceInfoData)}; // get device class information handle hDevInfo = SetupDiGetClassDevs(&GUID_DEVINTERFACE_DISK,0, 0, DIGCF_PRESENT|DIGCF_DEVICEINTERFACE); if (hDevInfo == INVALID_HANDLE_VALUE) { res = GetLastError(); return res; } // enumerute device information DWORD required_size = 0; for (i = 0; SetupDiEnumDeviceInfo(hDevInfo, i, &DeviceInfoData); i++) { DWORD DataT; char friendly_name[2046] = {0}; DWORD buffersize = 2046; DWORD req_bufsize = 2046; // get device friendly name if (!SetupDiGetDeviceRegistryPropertyA(hDevInfo, &DeviceInfoData, SPDRP_FRIENDLYNAME, &DataT, (LPBYTE)friendly_name, buffersize, &req_bufsize)) { continue; } if (strstr(friendly_name, "USB") == 0) { continue; } int index = 0; SP_DEVICE_INTERFACE_DATA did = {sizeof(did)}; PSP_DEVICE_INTERFACE_DETAIL_DATA pdd = NULL; while(1) { // get device interface data if (!SetupDiEnumDeviceInterfaces(hDevInfo, &DeviceInfoData, &GUID_DEVINTERFACE_DISK, index++, &did)) { res = GetLastError(); if( ERROR_NO_MORE_DEVICES == res || ERROR_NO_MORE_ITEMS == res) break; } // get device interface detail size if (!SetupDiGetDeviceInterfaceDetail(hDevInfo, &did, NULL, 0, &required_size, NULL)) { res = GetLastError(); if(ERROR_INSUFFICIENT_BUFFER == res) { pdd = (PSP_DEVICE_INTERFACE_DETAIL_DATA)LocalAlloc(LPTR, required_size); pdd->cbSize = sizeof(SP_DEVICE_INTERFACE_DETAIL_DATA); } else break; } // get device interface detail if (!SetupDiGetDeviceInterfaceDetail(hDevInfo, &did, pdd, required_size, NULL, NULL)) { res = GetLastError(); LocalFree(pdd); pdd = NULL; break; } // get device number DWORD bytes_returned = 0; STORAGE_DEVICE_NUMBER device_num; HANDLE hDevice = CreateFile(pdd->DevicePath, 0, FILE_SHARE_READ|FILE_SHARE_WRITE, NULL, OPEN_EXISTING, 0, NULL); if (DeviceIoControl(hDevice, IOCTL_STORAGE_GET_DEVICE_NUMBER, NULL, 0, &device_num, sizeof(device_num), &bytes_returned, (LPOVERLAPPED) NULL)) { for (int usb_index = 0; usb_index < list_size; usb_index++) { if (device_num.DeviceNumber == usb_list[usb_index].device_num) { strcpy_s(usb_list[usb_index].friendname, friendly_name); break; } } } CloseHandle(hDevice); LocalFree(pdd); pdd = NULL; } } SetupDiDestroyDeviceInfoList(hDevInfo); return res; }
3,调用代码:
typedef struct usb_device_info_t { char volume; char friendname[256]; int device_num; }usb_device_info;
int _tmain(int argc, _TCHAR* argv[]) { bool bRes = false; usb_device_info usb_list[8]; memset(usb_list, 0, 8*sizeof(usb_device_info)); int usb_cnt = get_usb_device_list(usb_list, 8); printf("System has %d USB disk. ", usb_cnt); if (usb_cnt > 0) { get_usb_device_friendname(usb_list, usb_cnt); { for (int i = 0; i < usb_cnt; i++) { printf("%c: %s ", usb_list[i].volume, usb_list[i].friendname); } } } getchar(); return 1; }
4,运行结果如下: