.NET - 深入系统编程 - 第一部分






4.79/5 (24投票s)
使用 C# 进行设备配置(第一部分)
引言
在之前的文章中,我展示了系统编程的应用。 我记得很多年前,我喜欢深入研究系统表或验证系统设施。 当然,那并不是一台真正的个人电脑,而是像 IBM360 或 PDP11 或 microVAX 这样的计算机,但那感觉很棒。 现在,多年以后,我仍然喜欢做 COM 接口、DLL 库、硬件配置等事情。 现在对我来说和以前一样有趣……嗯……系统编程……我永远爱你。
目标
普遍认为 C# 是一种“儿童语言”。 我不这么认为。 我认为这完全是不正确的。 为了证明这一点,我决定编写三个应用程序,显示有关硬件设备配置的信息。 我将演示 C# 如何使用 Win32 API 函数(甚至 DDK 中的函数)来访问有关配置的信息。 那些使用 C# 的人知道,这可以通过 P/Invoke 实现。 C# 成功解决的另一个问题是从非结构化的非托管数据(通常是执行 Win32 API 函数的结果)到托管结构的数据转换。 最后,可以使用此处讨论的技术与 Windows Forms 一起使用,使应用程序看起来很漂亮。
设备类
系统中的所有设备都加入到设备类中。 正如您在下图中所看到的,该类具有名称和 Guid(因此可以在注册表中找到)。 该类还可以具有描述。 例如,对于类“端口”,描述是“端口(COM 和 LPT)”。 类还具有存在于配置中的设备。
 
 
系统设备管理器提供有关(包括隐藏的)存在于 PC 上的类的信息
 
 
下面是 C# 中的代码,它将枚举 PC 上所有存在的设备类。 正如我所说,此代码使用 P/Invoke 来访问 DDK (cfgmgr32.dll) 和 SDK (setupapi.dll) DLL。
using System;
using System.Runtime.InteropServices;
using System.Text;
namespace DevClasses
{
 /// <summary>
 /// Summary description for Class.
 /// </summary>
 class DeviceClasses
 {
  /// <summary>
  /// The main entry point for the application.
  /// </summary>
  public const int MAX_NAME_PORTS=7;
  public const int RegDisposition_OpenExisting=(0x00000001); 
    // open key only if exists
  public const int CM_REGISTRY_HARDWARE=(0x00000000);
  public const int CR_SUCCESS = (0x00000000);
  public const int CR_NO_SUCH_VALUE = (0x00000025);
  public const int CR_INVALID_DATA = (0x0000001F);
  public const int DIGCF_PRESENT = (0x00000002);
  public const int DIOCR_INSTALLER = (0x00000001);
// MaximumAllowed access type to Reg.
  public const int MAXIMUM_ALLOWED = (0x02000000);
[StructLayout(LayoutKind.Sequential)]
 public class SP_DEVINFO_DATA
 {
 public int cbSize;
 public Guid ClassGuid;
 public int DevInst; // DEVINST handle
 public ulong Reserved;
 };
  [DllImport("cfgmgr32.dll")]
  public static extern UInt32
  CM_Open_DevNode_Key(IntPtr dnDevNode, UInt32 samDesired, 
         UInt32 ulHardwareProfile,
         UInt32 Disposition,IntPtr phkDevice, UInt32 ulFlags);
  [DllImport("cfgmgr32.dll")]
  public static extern UInt32
  CM_Enumerate_Classes(UInt32 ClassIndex,ref Guid ClassGuid, UInt32 Params);
  [DllImport("setupapi.dll")]//
  public static extern Boolean
   SetupDiClassNameFromGuidA(ref Guid ClassGuid,
            StringBuilder ClassName, //char * ?
   UInt32 ClassNameSize, ref UInt32 RequiredSize);
  [DllImport("setupapi.dll")]
  public static extern IntPtr
   SetupDiGetClassDevsA(ref Guid ClassGuid, UInt32 Enumerator,
   IntPtr  hwndParent, UInt32 Flags);
  [DllImport("setupapi.dll")]
  public static extern Boolean
   SetupDiEnumDeviceInfo(IntPtr DeviceInfoSet, UInt32 MemberIndex,
   ref SP_DEVINFO_DATA  DeviceInfoData);
  [DllImport("setupapi.dll")]
  public static extern Boolean
   SetupDiDestroyDeviceInfoList(IntPtr DeviceInfoSet);
  [DllImport("setupapi.dll")]
  public static extern IntPtr
   SetupDiGetClassDevsA(ref Guid ClassGuid, UInt32 samDesired,
   UInt32 Flags, ref string hwndParent, IntPtr Reserved);
  [DllImport("setupapi.dll")]
  public static extern IntPtr
  SetupDiOpenClassRegKeyExA(
  ref Guid ClassGuid, UInt32 samDesired, int Flags, IntPtr MachineName,
  UInt32 Reserved);
  [DllImport("advapi32.dll")]
  public static extern UInt32
  RegQueryValueA(IntPtr KeyClass,UInt32 SubKey,
         StringBuilder ClassDescription,ref UInt32 sizeB);
  [DllImport("user32.dll")]
  public static extern Boolean
  CharToOem(String lpszSrc, StringBuilder lpszDst);
  public static int EnumerateClasses(UInt32 ClassIndex, 
   ref StringBuilder ClassName, StringBuilder ClassDescription, 
        ref bool DevicePresent)
  {
   Guid ClassGuid=Guid.Empty;
   IntPtr NewDeviceInfoSet;
   SP_DEVINFO_DATA DeviceInfoData;
   UInt32 result;
   StringBuilder name=new StringBuilder("");
   bool resNam=false;
   UInt32 RequiredSize=0;
   IntPtr ptr;
   result = CM_Enumerate_Classes(ClassIndex, ref ClassGuid,0);
    ClassName=new StringBuilder("");
    DevicePresent=false;
   //incorrect device class:
   if(result == CR_INVALID_DATA)
   {
    return -2;
   }
  //device class is absent
   if(result == CR_NO_SUCH_VALUE)
   {
    return -1;
   }
  //bad param. - fatal error
   if(result != CR_SUCCESS)
   {
    return -3;
   }
   name.Capacity=0;
   resNam=SetupDiClassNameFromGuidA(ref ClassGuid,name,RequiredSize,
         ref RequiredSize);
   if(RequiredSize > 0)
    {
    name.Capacity=(int)RequiredSize;
    resNam=SetupDiClassNameFromGuidA(ref ClassGuid,name,
           RequiredSize,ref RequiredSize);
    }
   NewDeviceInfoSet=SetupDiGetClassDevsA(
    ref ClassGuid,
    0,
    IntPtr.Zero,
    DIGCF_PRESENT);
   if(NewDeviceInfoSet.ToInt32() == -1)
    {  DevicePresent=false;
      ClassName=name;
      return 0;}
   IntPtr KeyClass=SetupDiOpenClassRegKeyExA(
    ref ClassGuid, MAXIMUM_ALLOWED, DIOCR_INSTALLER,IntPtr.Zero,0);
   if(KeyClass.ToInt32() == -1)
    {  DevicePresent=false;
      ClassName=name;
      return 0;}
   UInt32 sizeB=1000;
   String abcd="";
   StringBuilder CD=new StringBuilder("");
   ClassDescription.Capacity=1000;
  
   UInt32 res=RegQueryValueA(KeyClass,0,ClassDescription,ref sizeB);
   if(res != 0)ClassDescription=new StringBuilder("");
   SetupDiDestroyDeviceInfoList(NewDeviceInfoSet);
    ClassName=name;
    DevicePresent=true;
   return 0;
  }
  [STAThread]
  static void Main(string[] args)
  {
   StringBuilder classes=new StringBuilder("");
   StringBuilder classesDescr=new StringBuilder("");
   StringBuilder classesDescrOEM=new StringBuilder("");
   classesDescrOEM.Capacity=1000;
   Boolean DevExist=false;
   UInt32 i=0;
   while(true)
   {
   int res=EnumerateClasses(i,ref classes,classesDescr,ref DevExist);
   if(res == -1)break;
   ++i;
   if(res < -1 || !DevExist)continue;
   Console.WriteLine("ClassName={0}, Description={1}",classes,classesDescr);
   }
   return;
  }
 }
}
运行应用程序后,您将看到 PC 上的所有设备类(托管版本;我已在 Win2000 上测试过)。



