硬盘序列号(Serial Number)不等于卷标号(Volume Name),后者虽然很容易得到,但是格式化分区后就会重写,不可靠。遗憾的是很多朋友往往分不清这一点。
要得到硬盘的物理序列号,可以通过WMI,也就是Win32_PhysicalMedia.SerialNumber。可惜的是Windows 98/ME的WMI并不支持这个类,访问时会出现异常。
受陆麟的例子的启发,我们还可以通过S.M.A.R.T.接口,直接从RING3调用API DeviceIoControl()来获取硬盘信息,而不需要写VXD或者DRIVER。这样这个问题就解决了,我对它进行了封装,大量使用了P/Invoke技术,一个完整的Library。支持Windows 98-2003。
使用上很简单:
HardDiskInfo hdd = AtapiDevice.GetHddInfo(0); // 第一个硬盘
Console.WriteLine("Module Number: {0}", hdd.ModuleNumber);
Console.WriteLine("Serial Number: {0}", hdd.SerialNumber);
Console.WriteLine("Firmware: {0}", hdd.Firmware);
Console.WriteLine("Capacity: {0} M", hdd.Capacity);
下面是全部代码:
注:
在Windows 98/ME中,S.M.A.R.T并不缺省安装,请将SMARTVSD.VXD拷贝到%SYSTEM%/IOSUBSYS目录下。
在Windows 2000/2003下,需要Administrators组的权限。
不要在装有SCSI硬盘的机器上尝试了,因为SCSI硬盘根本不存在序列号。
最终版权归陆麟所有,任何人不得将此代码占为己有。
--------------- updated on 2005/06/06
http://blog.sunmast.com/Sunmast/archive/2005/06/06/2632.aspx
这里有老外写了更完美的例子,我这次采取了较“安全”的做法,采用C++.NET直接把它包装成托管的dll,这样C#这边不需要用P/Invoke就能使用,C++那边需要改动和增加的代码量都减到了最少,NT下不要求管理员权限,并且支持Win 98/ME系统。
源码(in C++)和例子(in C#)在这里下载。
P.S.
准备用这个东西来保护你的托管程序集不被非法使用?事实上是不可行的。就算有也很弱。除了混淆器以外,唯一能提供有效保护的方法就是使用非托管语言(比如C++)来编写关键代码。在这里,也是C++.NET发挥价值的地方。