前面我们介绍了枚举系统中的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.\n", 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\n", usb_list[i].volume, usb_list[i].friendname);
}
}
} getchar();
return 1;
}

4,运行结果如下:

Windows下USB磁盘开发系列三:枚举系统中U盘、并获取其设备信息的更多相关文章

  1. Windows下USB磁盘开发系列二:枚举系统中所有USB设备

    上篇 <Windows下USB磁盘开发系列一:枚举系统中U盘的盘符>介绍了很简单的获取系统U盘盘符的办法,现在介绍下如何枚举系统中所有USB设备(不光是U盘). 主要调用的API如下: 1 ...

  2. Windows下USB磁盘开发系列一:枚举系统中U盘的盘符

    个时候我们需要区分系统磁盘中,哪些是U盘,这样我们在访问的时候可以区别对待.具体方法如下: 1,调用GetLogicalDrives()返回系统盘符标记位 API GetLogicalDrives() ...

  3. windows下部署.netcore+docker系列三 (unbuntu 18.4 下安装ftp)

    // 先更新下系统sudo apt-get update//安装ftpsudo apt-get install vsftpd// 启动 服务sudo service vsftpd start//ftp ...

  4. S5PV210开发系列三_简易Bootloader的实现

    S5PV210开发系列三 简易Bootloader的实现 象棋小子          1048272975 Bootloader是嵌入式系统上电后第一段运行的代码.对于功能简单的处理器,可能并没有Bo ...

  5. windows下VC界面 DIY系列1----写给想要写界面的C++程序猿的话

    非常早就想写关于C++ UI开发的一系列博文,博客专栏刚审核通过,就立即開始刷博文,不能辜负自己的一番热血,我并非写界面的高手,仅仅想通过写博文提高我自己的技术积累,也顺便帮助大家解决界面开发的瓶颈. ...

  6. Windows下C,C++开发环境搭建指南

    Windows下C,C++开发环境搭建指南 前情提要 基于近一段时间很多网友发邮件反馈,说一些项目编译出现问题,诸如此类的情况. 就觉得很有必要写一篇C,C++开发环境的小指南,统一回复. 1.君欲善 ...

  7. Windows下搭建Git开发环境

    Windows下搭建Git开发环境主要有以下三种方法: 1,VS,vs2013和vs2015中已经集成了git插件了 2,msysGit+TortoiseGit 3,msysGit+SourceTre ...

  8. [转]windows下安装Object-C开发环境

    本文转自:http://hi.baidu.com/jeremylai/item/f40b9116cb3c5d582b3e22f5 在Windows下搭建Objective C开发环境,需要到GNUst ...

  9. 转:Windows下的PHP开发环境搭建——PHP线程安全与非线程安全、Apache版本选择,及详解五种运行模式。

    原文来自于:http://www.ituring.com.cn/article/128439 Windows下的PHP开发环境搭建——PHP线程安全与非线程安全.Apache版本选择,及详解五种运行模 ...

随机推荐

  1. DHT(Distributed Hash Table) Translator

    DHT(Distributed Hash Table) Translator What is DHT? DHT is the real core of how GlusterFS aggregates ...

  2. Group GridView:用于.Net的分组显示的GridView

    我的项目需要一个可以分组显示的GridView,我不会写,上网找了一圈,最终在国外的网站上找到的这个,比较符合我的要求,但它的分页得重写,它写了能分页,但我发现它的分页功能事实上并没有实现,也不知道是 ...

  3. synchronized原理

    http://www.cnblogs.com/YDDMAX/p/5658607.html http://www.cnblogs.com/YDDMAX/p/5658668.html synzhroniz ...

  4. C# base和this

    • 是否可以在静态方法中使用base和this,为什么? • base常用于哪些方面?this常用于哪些方面? • 可以base访问基类的一切成员吗? • 如果有三层或者更多继承,那么最下级派生类的b ...

  5. notepad++必读

    #预备知识Ctrl: control 控制键 Ctrl + TabAlt:alter 改变键(换码键) Alt + TabShift: shift 换挡键(切换键) Ctrl + Shift + Ta ...

  6. C# Albert工程阅读关键字解析

    // 摘要: // 为强类型集合提供 abstract 基类. [Serializable] [ComVisible(true)] public abstract class CollectionBa ...

  7. gcc常用

    gcc选项:-I指定头文件搜索路径.-D编译时定义宏-L链接时指定库文件搜索路径-l指定库文件名称-pipe使用管道,一个程序的输出作为输入直接送给另外一个程序, 而且还可以一直连续下去,不需要临时文 ...

  8. makefile中引用其他makefile方法

    在Makefile中引用其他Makefile文件的方法是,使用inclue   filename.mk

  9. 第七篇 Replication:合并复制-订阅

    本篇文章是SQL Server Replication系列的第七篇,详细内容请参考原文. 订阅服务器就是复制发布项目的所有变更将传送到的服务器.每一个发布需要至少一个订阅,但是一个发布可以有多个订阅. ...

  10. python_类

    1. 对象的概念 对象包括特性和方法.特性只是作为对象的一部分的变量,方法则是存储在对象内的函数.对象中的方法和其他函数的区别在于方法总是将对象作为自己的第一个参数,这个参数一般称为self. 2. ...