VC++实现USB 硬盘等硬件插拔检测

本文介绍了一种在Windows环境下使用MFC和Win32 API监听USB设备插拔的方法,包括如何注册设备通知、处理不同类型的USB事件以及遍历硬件设备来获取USB盘符。


首先捕捉WM_DEVICECHANGE消息,该消息在usb插拔时均有触发。

MFC下  

添加消息处理函数afx_msg BOOL OnDeviceChange( UINT nEventType, DWORD dwData );

添加消息映射BEGIN_MESSAGE_MAP()

...

ON_WM_DEVICECHANGE()

END_MESSAGE_MAP()

Win32 在消息回调处理即可


MFC下  nEventType 在没有注册设备的情况下均为7,无法区分,需要设备GUID去注册,方法如下:

USB Raw Device/USB设备

{a5dcbf10-6530-11d2   -901f-00c04fb951ed}

Disk Device/磁盘设备

{53f56307-b6bf-11d0   -94f2-00a0c91efb8b}

Network Card/网卡

{ad498944-762f-11d0   -8dcb-00c04fc3358c}

Human Interface Device (HID)/人机界面设备

{4d1e55b2-f16f-11cf    -88cb-001111000030}

Palm/手持设备

{784126bf-4190-11d4   - b5c2-00c04f687a67}


HDEVNOTIFY hDevNotify;
DEV_BROADCAST_DEVICEINTERFACE NotificationFilter;
ZeroMemory( &NotificationFilter, sizeof(NotificationFilter) );
NotificationFilter.dbcc_size = sizeof(DEV_BROADCAST_DEVICEINTERFACE);
NotificationFilter.dbcc_devicetype = DBT_DEVTYP_DEVICEINTERFACE;

for(int i=0; i<sizeof(GUID_DEVINTERFACE_LIST)/sizeof(GUID); i++) 
{
NotificationFilter.dbcc_classguid = GUID_DEVINTERFACE_LIST[i];
hDevNotify = RegisterDeviceNotification(this->GetSafeHwnd(), &NotificationFilter, DEVICE_NOTIFY_WINDOW_HANDLE);
if( !hDevNotify ) 
{
return FALSE;
}
}

注册代码请在程序初始化时进行,如MFC的OnInitDialog

在消息响应中
switch(nEventType)
{
case DBT_DEVICEARRIVAL://设备可用
{
// PDEV_BROADCAST_DEVICEINTERFACE
if(lpdb->dbch_devicetype==DBT_DEVTYP_VOLUME)
{

PDEV_BROADCAST_VOLUME lpdbv =  (PDEV_BROADCAST_VOLUME)lpdb;
switch(lpdbv->dbcv_flags)
{
dosomething
}
}
else if(lpdb->dbch_devicetype==DBT_DEVTYP_DEVICEINTERFACE
{
}
else
{
}
}
以上方法较为复杂,需要处理DBT_DEVTYP_VOLUMEDBT_DEVTYP_DEVICEINTERFACE等,现提供一种简单的方法,在DBT_DEVICEARRIVAL有新设备进入时,通过遍历硬件设备变相处理:


  1. //以找usb设备为例,取得的usb盘符放在UsbRoot里面,len为UsbRoot字符数组的长度  
  2. BOOL GetUsbRoot(CHAR* UsbRoot,SHORT len)  
  3. {  
  4. DWORD dwDriveStrLen = 0;  
  5. CHAR *pDriveName = NULL;  
  6. UINT Drive = 0;  
  7. if (UsbRoot == NULL || len <= 0)  
  8. {  
  9. return FALSE;  
  10. }  
  11. dwDriveStrLen = ::GetLogicalDriveStrings(0,NULL);//取得你计算机上盘符数目  
  12. if (0 == dwDriveStrLen)  
  13. {  
  14. return FALSE;  
  15. }  
  16. CHAR *szDriveNameBuff = (CHAR*)malloc(dwDriveStrLen);//根据你机器上的磁盘数目分配内存  
  17. if (NULL == szDriveNameBuff)  
  18. {  
  19. return FALSE;  
  20. }  
  21. memset(szDriveNameBuff,0,sizeof(szDriveNameBuff));  
  22. ::GetLogicalDriveStrings(dwDriveStrLen,szDriveNameBuff);//取得你计算机上所有盘符  
  23. pDriveName = szDriveNameBuff;  
  24. while (*pDriveName != NULL)//遍历所有盘符,找到你需要的设备  
  25. {  
  26. Drive = ::GetDriveType(pDriveName);  
  27. switch (Drive)  
  28. {  
  29. case DRIVE_UNKNOWN://未知设备  
  30. break;  
  31. case DRIVE_NO_ROOT_DIR:  
  32. break;  
  33. case DRIVE_REMOVABLE://usb设备  
  34. //如果为usb,在这里进行相关处理  
  35. //break;  
  36. case DRIVE_FIXED://硬盘  
  37. break;  
  38. case DRIVE_REMOTE://网络硬盘,如:局域网服务器上的盘  
  39. break;  
  40. case DRIVE_CDROM://光驱  
  41. break;  
  42. case DRIVE_RAMDISK://RAM 盘  
  43. break;  
  44. default:  
  45. break;  
  46. }  
  47. pDriveName += strlen(pDriveName) + 1;  
  48. }  
  49. if (szDriveNameBuff != NULL)  
  50. {  
  51. free(szDriveNameBuff);  
  52. szDriveNameBuff = NULL;  
  53. }  
  54. return FALSE;  
  55. }  
在遍历到你关心的部分时,做逻辑处理即可
现在USB设备却很多,因此对USB设备的查找与读写就必不可少了。但是能找到关于USB读写的资料很少。这里使用VC++示范了一些获得USB的信息的方法。 一、枚举USB设备   通过枚举USB控制器->枚举此控制器上的USB HUB->枚举HUB的各个端口->获得设备信息。 枚举控制器: wsprintf(HCName, "\\\\.\\HCD%d", HCNum); hHCDev = CreateFile(HCName, GENERIC_WRITE, FILE_SHARE_WRITE, NULL, OPEN_EXISTING, 0, NULL);   计算机上的USB主控制器以HCD1,HCD2等命名。通过控制器名称,使用CCreateFile 打开它。使用DeviceIoControl即可得到其驱动程序名,以及与它连接的HUB的名称。用CCreateFile打开HUB,获得连接信息。再枚举HUB的各个端口即可获得连接的设备信息。 二、枚举HID设备   HID设备是微软定义的标准人机接口规范。比如USB鼠标,USB游戏手柄等。不用查找具体设备的GUID,使用API HidD_GetHidGuid(&guidHID)即可得到GUID。有了GUID通过API SetupDiEnumDeviceInterfaces可获得是否有设备连接。如果此类设备连接通过SetupDiGetDeviceInterfaceDetail获得它的设备路径信息。使用CCreateFile 打开它,通过HidD_GetAttributes获得其基本属性信息。使用DeviceIoControl可以获得更详细的属性。在本代码中如果计算机上插有USB游戏手柄,可获得其信息。但不知道为什么xp下却不能获得USB鼠标的信息。 三 枚举U盘   先用GetDriveType API获得设备的类型,若类型为REMOVABLE(当然有些大容量U盘可能报告为FIXED,那就需要其他方法来确定了),即可能是U盘。用CCreateFile 打开之后,再用IOCTL_STORAGE_QUERY_PROPERTY为参数的DeviceIoControl来获得其属性。 四、结束语   示例工程在winxp+xpDDK+VC6下编译通过。USB设备种类比较多,也比较特殊,不同厂商的硬件不同,控制软件也不尽相同(我想主要是ICTL码不同,也不容易查到)。使得访问USB口的设备不象串口并口那么方便。这个例程只是展示了访问的基本方法。其中还有些问题还没有解决,发出来希望大家解决之后能通知我或者发表出来。 参考了USBPort,USBview等代码,一并致谢。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值