ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

监控USB设备插拔

2021-03-05 18:33:38  阅读:267  来源: 互联网

标签:插拔 USB WMI 监控 Device new DBT


最近做了UKey加密中设计到USB设备. 因UKEy是用来加密和执行PC与项目间通信加密的介质.从作用范围来讲不是传统意义上U盘作为存储介质来使用.其实熟悉网银驱动DR应该了解.在网银系统安全上一个最基本需求就是动态即时监控通信PC驱动以及设备列表通信变化.当然包括我们加密存储介质在PC上USB插拔.

思路一.在WinFrom中通过拦截Windows 消息机制来实现. 类似定义MEssageForm窗体. 假如用鼠标左击一下窗体, 系统会收到一条 WM_LBUTTONDOWN 消息;当鼠标抬起, 系统又会收到 WM_LBUTTONUP 消息.系统收到消息后, 会告诉窗体发生的事情, 然后窗体再做出反应; 当然窗体能否做出反应要看窗体是否有相应的响应代码. 同样也可以把USB设备插拔事件通过重写窗体WndProc(Ref Message m)方法拦截并处理.

首先引入命名空间:

<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   1:  </span><span style="color:#0000ff">using</span> System.Management;</span></span>
   2:  using System.Threading;
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   3:  </span><span style="color:#0000ff">using</span> System.Security.Permissions;</span></span>

定义Device Management Event的枚举:

<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   1:  </span>  <span style="color:#0000ff">public</span> <span style="color:#0000ff">enum</span> DeviceEvent : <span style="color:#0000ff">int</span></span></span>
   2:          {
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   3:  </span>            DBT_CONFIGCHANGECANCELED = 0x0019,</span></span>
   4:              DBT_CONFIGCHANGED=0x0018,
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   5:  </span>            DBT_CUSTOMEVENT=0x8006,</span></span>
   6:              DBT_DEVICEARRIVAL=0x8000,//USB Insert DEvice Statu
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   7:  </span>            DBT_DEVICEQUERYREMOVE=0x8001,</span></span>
   8:              DBT_DEVICEQUERYREMOVEFAILED=0x8002,
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   9:  </span>            DBT_DEVICEREMOVEPENDING=0x8003,<span style="color:#008000">//USB Revoing.</span></span></span>
  10:              DBT_DEVICEREMOVECOMPLETE=0x8004,//USB Remove Completed
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  11:  </span>            DBT_DEVICETYPESPECIFIC=0x8005,</span></span>
  12:              DBT_DEVNODES_CHANGED=0x0007,//Device List _Changed
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  13:  </span>            DBT_QUERYCHANGECONFIG=0x0017,</span></span>
  14:              DBT_USERDEFINED=0xFFFF
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  15:  </span>        }</span></span>

其中涉及到USB设备插拔的是DEVICEREMOVEPENDING/DEVICEREMOVECOMPLETE[删除]  DEVICEARRIVAL[插入设备]  重写WinFProc实现窗体上对Windows MEssage进行拦截并重新处理:

<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   1:  </span>      [PermissionSet(SecurityAction.Demand, Name = <span style="color:#006080">"FullTrust"</span>)]</span></span>
   2:          protected override void WndProc(ref Message m)
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   3:  </span>        {</span></span>
   4:              base.WndProc(ref m);
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   5:  </span>            DeviceEvent lEvent;</span></span>
   6:      
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   7:  </span>            lEvent = (DeviceEvent)m.WParam.ToInt32();</span></span>
   8:              switch (lEvent)
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   9:  </span>            {</span></span>
  10:                  case DeviceEvent.DBT_DEVICEARRIVAL://[Insert]
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  11:  </span>                    <span style="color:#0000ff">this</span>.CheckDeviceStatus_Lable.BackColor = Color.Green;</span></span>
  12:                      this.CheckDeviceStatus_Lable.Text = "----Connection Device!----";
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  13:  </span>                    MessageBox.Show(<span style="color:#006080">"Just Insert At Moment !"</span>, <span style="color:#006080">"Insert"</span>);</span></span>
  14:                      break;
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  15:  </span>                <span style="color:#0000ff">case</span> DeviceEvent.DBT_DEVICEREMOVECOMPLETE:<span style="color:#008000">//[REmove]</span></span></span>
  16:                      this.CheckDeviceStatus_Lable.BackColor = Color.Red;
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  17:  </span>                    <span style="color:#0000ff">this</span>.CheckDeviceStatus_Lable.Text = <span style="color:#006080">"------No Connection------"</span>;</span></span>
  18:                      MessageBox.Show("Remove Complete At Moment!", "Remove");
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  19:  </span>                    <span style="color:#0000ff">break</span>;</span></span>
  20:                  case DeviceEvent.DBT_DEVNODES_CHANGED://[Device List Have Changed]
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  21:  </span>                    MessageBox.Show(<span style="color:#006080">"Device List have been Changed!"</span>);</span></span>
  22:                      break;
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  23:  </span>                <span style="color:#0000ff">default</span>:</span></span>
  24:                      break;
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  25:  </span>            }</span></span>
  26:          }

首先USB在进行插拔即时如果没有存储介质即Disk或是无驱动的方式可能不能触发DeviceEvent.DBT_DEVICEREMOVECOMPLETE 和 DeviceEvent.DBT_DEVICEARRIVAL插把事件. 但是一点是可以确认的是.PC端只要接受USB设备.PC识别之后设备列表肯定会发生变化. 如果不是加密Key. u盘的 盘符方式存在Disk存储介质.则没有问题.

当然需要移植这种基于From窗体系统Message拦截方法时发现这种可重用性就不高.换一种思路采用Windows底层方式WMI实现.

WMI以CIMOM为基础,CIMOM即公共信息模型对象管理器[Common Information Model Object Manager],是一个描述操作系统构成单元的对象数据库,为MMC和脚本程序提供了一个访问操作系统构成单元的公共接口。有了WMI,工具软件和脚本程序访问操作系统的不同部分时不需要使用不同的API;相反,操作系统的不同部分都可以插入WMI,工具软件和WMI可以方便地读写WMI.

WMI 可以产生的系统级事件的一些更有用。 每当创建 WMI 类的新实例 ; 例如对于激发称为 __instancecreationevent 事件__instancedeletionevent 时将触发一个实例将被删除.  当然USB插拔时同样获得这个系统事件:

<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   1:  </span>  <span style="color:#0000ff">public</span> <span style="color:#0000ff">void</span> ControlUSBConnectionStatu()</span></span>
   2:          {
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   3:  </span>            ManagementEventWatcher getEventWatcher = <span style="color:#0000ff">null</span>;</span></span>
   4:              WqlEventQuery getEventQuery = null;
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   5:  </span> </span></span>
   6:              ManagementOperationObserver getObserver = new ManagementOperationObserver();
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   7:  </span> </span></span>
   8:              //Bind to Loacl Machine and Watch the PortConnection
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   9:  </span>            ManagementScope getScope = <span style="color:#0000ff">new</span> ManagementScope(<span style="color:#006080">"root\\CIMV2"</span>);</span></span>
  10:              getScope.Options.EnablePrivileges = true;//set requeired
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  11:  </span> </span></span>
  12:              try
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  13:  </span>            {</span></span>
  14:                  getEventQuery = new WqlEventQuery();
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  15:  </span>                getEventQuery.EventClassName = <span style="color:#006080">"__InstanceOperationEvent"</span>;</span></span>
  16:                  getEventQuery.WithinInterval = new TimeSpan(0, 0, 0, 1);
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  17:  </span>                getEventQuery.Condition = <span style="color:#006080">@"TargetInstance ISA 'Win32_DiskDrive' "</span>;</span></span>
<span style="color:#000000"><span style="color:#000000"><span style="color:#008000">                       //[Disk must have DiskDrive fuck ]</span></span></span>
 
  20:                  //Event Watcher [Test Event and semd informatio to this message and create new informtion .]
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  21:  </span>                getEventWatcher = <span style="color:#0000ff">new</span> ManagementEventWatcher(getEventQuery);</span></span>
  22:                  getEventWatcher.EventArrived += new EventArrivedEventHandler(getEventWatcher_EventArrived);
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  23:  </span>                getEventWatcher.Start();<span style="color:#008000">//Start Watch Event</span></span></span>
 
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  25:  </span>            }</span></span>
  26:              catch (Exception se)
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  27:  </span>            { }</span></span>
  28:              finally
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  29:  </span>            {</span></span>
  30:                  // getEventWatcher.Stop();
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  31:  </span>            }</span></span>
  32:          }

当发生USB插拔并成功监听到事件时处理方法:

<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   1:  </span> <span style="color:#0000ff">void</span> getEventWatcher_EventArrived(<span style="color:#0000ff">object</span> sender, EventArrivedEventArgs e)</span></span>
   2:          {
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   3:  </span>            ManagementBaseObject getBaseObject = (ManagementBaseObject)e.NewEvent;</span></span>
   4:              if ((getBaseObject.ClassPath.ClassName == "__InstanceCreationEvent"))
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   5:  </span>            {</span></span>
   6:                  //Usb Inserted
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   7:  </span>                MessageBox.Show(<span style="color:#006080">"USB Disk Inserted!"</span>);</span></span>
   8:              }
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   9:  </span>            <span style="color:#0000ff">else</span></span></span>
  10:              {
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  11:  </span>                <span style="color:#008000">//Usb Removed</span></span></span>
  12:                  MessageBox.Show("USB Device Removed!");
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  13:  </span>            }</span></span>
  14:          }

如上在定义时设置一个Condition[条件]:存在Disk Driver 磁盘驱动.如果类似某些UKey不存在磁盘驱动同时也无存储介质 经过测试你会发现.加密的UKey插入 如上的WMI监听事件并不能扑捉到USB插拔. 但是针对这种方式我们还有一种更为彻底的方式就是USB插拔唯一可以确定在PC识别必然变化的因素是系统设备列表发生更新.

WMI处理监听:

<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   1:  </span>       <span style="color:#008000">/// <summary></span></span></span>
   2:          /// 监听USB Device设备插拔事件 完整操作.
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   3:  </span>        <span style="color:#008000">/// WMI Handle Event Change Device List chenkai</span></span></span>
   4:          /// </summary>
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   5:  </span>        <span style="color:#0000ff">public</span> <span style="color:#0000ff">void</span> RegisterDeviceWMIEventStatu()</span></span>
   6:          {
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   7:  </span>            <span style="color:#0000ff">try</span> </span></span>
   8:              {
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   9:  </span>                <span style="color:#008000">//Device List HAve Changed  And Send Message </span></span></span>
  10:                  WqlEventQuery query = new WqlEventQuery("SELECT * FROM Win32_DeviceChangeEvent");
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  11:  </span>                ManagementEventWatcher watcher = <span style="color:#0000ff">new</span> ManagementEventWatcher(query);</span></span>
  12:                  watcher.EventArrived += new EventArrivedEventHandler(watcher_EventArrived);
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  13:  </span>                watcher.Start();  <span style="color:#008000">// Start listening for events</span></span></span>
  14:              }
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  15:  </span>            <span style="color:#0000ff">catch</span> (Exception se)</span></span>
  16:              { }
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  17:  </span>        }</span></span>

WMI监听处理函数:

<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   1:  </span> <span style="color:#0000ff">void</span> watcher_EventArrived(<span style="color:#0000ff">object</span> sender, EventArrivedEventArgs e)</span></span>
   2:          {
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   3:  </span>            <span style="color:#0000ff">string</span> geteventtype = e.NewEvent.GetPropertyValue(<span style="color:#006080">"EventType"</span>).ToString();</span></span>
   4:              ManagementBaseObject getEventObject = (ManagementBaseObject)e.NewEvent;
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   5:  </span> </span></span>
   6:              if (getEventObject != null)
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   7:  </span>            {</span></span>
   8:                  //close Operator 
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   9:  </span>                <span style="color:#0000ff">this</span>.CloseDeviceEqument();</span></span>
  10:              }
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  11:  </span>        }</span></span>

其实这依然还是一种折中办法.因为导致PC端设备列表发生变化的因素有很多.如何判定是USB口发生设备变化.WMI依然没有让我们失望:

<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   1:  </span>  <span style="color:#008000">//WMI Control The USB Device Change Statu And Send Message to When IT‘s Changed/</span></span></span>
   2:          public void ControlUSBDeviceSTatu()
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   3:  </span>        {</span></span>
   4:              try
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   5:  </span>            {</span></span>
   6:                  WqlEventQuery query = new WqlEventQuery("select * from Win32_VolumeChangeEvent");
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   7:  </span>                ManagementEventWatcher getwatcher = <span style="color:#0000ff">new</span> ManagementEventWatcher(query);</span></span>
   8:                  getwatcher.EventArrived += new EventArrivedEventHandler(getwatcher_EventArrived);
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   9:  </span>                getwatcher.Start();</span></span>
  10:              }
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  11:  </span>            <span style="color:#0000ff">catch</span> (Exception se)</span></span>
  12:              { }
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  13:  </span>        }</span></span>

WMI在USB事情处理函数:

   2:          void getwatcher_EventArrived(object sender, EventArrivedEventArgs e)
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   3:  </span>        {</span></span>
   4:              MessageBox.Show(e.NewEvent.GetText(TextFormat.Mof).ToString());
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   5:  </span>        }</span></span>

这样来一来就可以清晰判定USB在插拔时所发生在系统PC端的变化通知给应用程序来进行处理.如何获得当前PC端 USB Driver列表.经过几番测试找到一种很好的方式获取全部的USB Driver信息:

<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   1:  </span> <span style="color:#008000">//Get ALL USB DRiver And Driver Property Fuck this shit。chenkai</span></span></span>
   2:          public static string[] AllInformation()
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   3:  </span>        {</span></span>
   4:              StringCollection propNames = new StringCollection();
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   5:  </span>            ManagementClass driveClass = <span style="color:#0000ff">new</span> ManagementClass(<span style="color:#006080">"Win32_USBController"</span>);</span></span>
   6:              PropertyDataCollection props = driveClass.Properties;
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">   7:  </span>            <span style="color:#0000ff">foreach</span> (PropertyData driveProperty <span style="color:#0000ff">in</span> props)</span></span>
   8:                  propNames.Add(driveProperty.Name);
 
  12:              int idx = 0;
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  13:  </span>            ManagementObjectCollection drives = driveClass.GetInstances();</span></span>
  14:              string _s = string.Empty;
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  15:  </span>            List<<span style="color:#0000ff">string</span>> harddisk = <span style="color:#0000ff">new</span> List<<span style="color:#0000ff">string</span>>();</span></span>
  16:   
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  17:  </span>            <span style="color:#0000ff">foreach</span> (ManagementObject drv <span style="color:#0000ff">in</span> drives)</span></span>
  18:              {
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  19:  </span>                idx++;</span></span>
  20:                  _s = string.Format(" USB Driver({0}) Properties ", idx);
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  21:  </span>                harddisk.Add(_s);</span></span>
  22:                  foreach (string strProp in propNames)
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  23:  </span>                {</span></span>
  24:                      _s = string.Format("Property: {0}, Value: {1}", strProp, drv[strProp]);
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  25:  </span>                    harddisk.Add(_s);</span></span>
  26:                  }
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  27:  </span>            }</span></span>
  28:              string[] _ss = harddisk.ToArray();
<span style="color:#000000"><span style="color:#000000"><span style="color:#606060">  29:  </span>            <span style="color:#0000ff">return</span> _ss;</span></span>
  30:          }

WMI的更多体现是对Windows 交互中进一步封装和管理.

Can考资料:

Device ManageMent Events

System.Form.control.WinProc[]Method

System.Form.Control.Message[]Method

标签:插拔,USB,WMI,监控,Device,new,DBT
来源: https://blog.csdn.net/daocaokafei/article/details/114407272

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有