在C#中根据HardwareID获取驱动程序信息的实现代码
近日在工作中需要根据设备的hardwareid来获取设备的驱动程序信息,比如驱动程序版本等。经过摸索,得到了两种不同的解决办法,两种办法各有千秋,写出来给大家分享。
1 使用wmi中的win32_pnpsigneddriver类
win32_pnpsigneddriver的详细信息:
使用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。