在C#中根据HardwareID获取驱动程序信息的实现代码

作者:mdxy-dxy 时间:2023-08-04 08:15:31 

近日在工作中需要根据设备的HardwareID来获取设备的驱动程序信息,比如驱动程序版本等。经过摸索,得到了两种不同的解决办法,两种办法各有千秋,写出来给大家分享。

1 使用WMI中的Win32_PnPSignedDriver类

Win32_PnPSignedDriver的详细信息:http://msdn2.microsoft.com/en-us/library/aa394354.aspx
使用WMI(Windows Management Instrumentation)是最为方便的方法。可以根据下面的程序片段来得到我们所需要的DriverVersion。


private string GetDriverVersion( string hardwareID )
{
 string queryString = "SELECT HardwareID, DriverVersion FROM Win32_PnPSignedDriver";
 SelectQuery selectQuery = new SelectQuery( queryString );
 ManagementObjectSearcher searcher = new ManagementObjectSearcher(selectQuery);

foreach (ManagementObject mo in searcher.Get())
 {
   object tempID = mo["HardwareID"];
   if( tempID!=null && tempID.ToString().ToUpper() == hardwareID.Trim().ToUpper() )
   {
     return mo["DriverVersion"].ToString();
   }
 }

return "UnknownVersion";
}

这样取得驱动程序的方式是非常简洁的,但是有一个非常严重的问题就是效率问题。平均说来,每执行一次查询,得到一个DriverVersion需要大约3秒的时间。对于我们的应用来说,这个时间是不可以接受的。也许你会说,为什么不用更多的限定符号来进一步减少查询的次数呢?

如果我们把连接字符串改成:


string queryString = "SELECT HardwareID, DriverVersion FROM Win32_PnPSignedDriver WHERE HardwareID='somehardware'";

程序的效率并没有明显的改进。而且还发现一个问题,如果我们somehardware里面含有一个'\'(也就是HardwareID='some\\hardware'),那么一定会得到一个“Invalid Query”异常。但是在WMITOOLS里面查询又是正常的,希望达人出来指点一下。最后根据MSDN的描述,只有Windows Vista,Windows XP和Windows 2003支持这个类。由于我们的程序需要跑在2000下,因此这种方法是行不通的了。

2 使用PInvoke

由于无法使用WMI,因此就想到了使用PInvoke的方式调用Windows API。通过查询MSDN,知道可以使用SetupDixxxx这种函数来实现我们的功能。基本的思路如下:
(1)利用SetupDiGetClassDevs这个函数得到一个含有所有设备信息的类。
(2)利用SetupDiEnumDeviceInfo得到某个具体设备的信息,保存在一个名为SP_DEVINFO_DATA的结构中。
(3)利用SetupDiGetDeviceRegistryProperty得到设备的HardwareID,和输入的HardwareID比较
(4)如果两个HardwareID是一样的,那么就利用SetupDiBuildDriverInfoList得到这个设备的驱动程序信息列表
(5)利用SetupDiEnumDriverInfo遍历驱动程序信息列表,得到所有需要的信息,保存在一个名为SP_DRVINFO_DATA的结构中
(6)从SP_DRVINFO_DATA中就可以得到驱动程序的版本。是一个DWORDLONG类型的数,需要转换成x.x.x.x的结构

要值得注意的是上述函数都封装在setupapi.dll中,要使用这些函数,需要安装Windows DDK。

在C#中,我们利用pInvoke的方式来调用Windows API的时候,需要注意类型的对应和结构对齐。比如上面的SP_DEVINFO_DATA结构需要按照如下方式声明


[StructLayout(LayoutKind.Sequential, Pack = 4, CharSet = CharSet.Auto)]
public struct SP_DEVINFO_DATA
{
  public int cbSize;
  public Guid ClassGuid;
  public IntPtr DevInst;
  public IntPtr Reserved;
}

要注意的是LayoutKind.Sequential, Pack = 4 和 public IntPtr Reserved。如果不按照这样声明,无法调用成功。
SP_DRVINFO_DATA也可以按照一样的方式进行声明。


[StructLayout(LayoutKind.Sequential, Pack = 4, CharSet = CharSet.Auto)]
public struct SP_DRVINFO_DATA
{
 public int cbSize;
 public int DriverType;
 public IntPtr Reserved;
 [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 256)]
 public string Description;
 [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 256)]
 public string MfgName;
 [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 256)]
 public string ProviderName;
 public FILETIME DriverDate;
 public ulong DriverVersion;
}

对于最后的从DWORDLONG转换成x.x.x.x的版本,可以按照下面的方式转换。DWORDLONG是8字节的无符号整数,x.x.x.x中的x是从0到65536的无符号整数,占2个字节。因此可以直接把8字节的整数分成4个2字节的整数,最后合起来就是版本号了。假设版本version = 1407379348914176,将version转换成2进制数为:
101 00000000 00000001 00001010 00101000 00000000 00000000
--- --------------------- ---------------------- ---------------------
5 1 2600 0
因此,可以得到版本是5.1.2600.0。

可以用下面这个示例函数来得到版本信息


//version = 1407379348914176,转换后的版本为5.1.2600.0
private string GetVersionFromLong( ulong version )
{
 ulong baseNumber = 0xFFFF;
 StringBuilder sb = new StringBuilder();
 ulong temp = 0L;

for( int offset = 48; offset >= 0; offset -= 16 )
 {
   temp = (version >> offset) & baseNumber;
   sb.Append( temp.ToString() + "." );
 }

return sb.ToString();
}

通过调用API这种方式,速度得到了很大的提高,1秒之内就可以完成一次查询。而且适合于Win2000,Win XP,Win2003和Vista。

标签:HardwareID,驱动程序信息
0
投稿

猜你喜欢

  • Android实现捕获TextView超链接的方法

    2021-07-03 08:44:04
  • mybatis 映射文件中if标签判断字符串相等的两种方式

    2023-08-02 16:06:44
  • Android 文件操作方法

    2023-06-02 12:51:17
  • Maven的生命周期与自定义插件实现方法

    2022-04-06 20:13:48
  • Android实现View滑动效果的6种方法

    2023-01-29 15:12:19
  • 详解WPF中用户控件和自定义控件的使用

    2023-07-25 12:20:26
  • Java线程的生命周期的详解

    2022-05-18 00:32:08
  • Android 6.0动态权限申请教程

    2023-09-26 16:43:56
  • 浅析Android Service中实现弹出对话框的坑

    2023-04-20 16:17:13
  • Winform实现抓取web页面内容的方法

    2022-03-07 21:15:53
  • C#实现类似新浪微博长URL转短地址的方法

    2023-06-02 12:59:28
  • javac final变量未赋值检测案例讲解

    2023-09-29 04:25:17
  • 排序算法图解之Java冒泡排序及优化

    2022-07-16 01:28:38
  • 使用注解解决ShardingJdbc不支持复杂SQL方法

    2022-08-03 02:15:12
  • C#使用Tesseract进行Ocr识别的方法实现

    2022-12-15 06:54:23
  • Android Studio ADB网络调试汇总

    2023-07-13 22:10:07
  • 解决idea默认带的equals和hashcode引起的bug

    2023-12-22 19:02:43
  • java接口返回参数按照请求参数进行排序方式

    2023-02-19 07:30:05
  • 详解IntelliJ IDEA中TortoiseSVN修改服务器地址的方法

    2023-11-25 04:51:04
  • 使用android隐藏api实现亮度调节的方法

    2022-10-16 23:37:36
  • asp之家 软件编程 m.aspxhome.com