SetupDiEnumDeviceInterfaces返回false,错误259 C++范例程序正确,改为C#错误

yylooker2008 2010-09-02 01:23:23
这段代码是用来实现虚拟com口通讯用的。
采用TI公司的MSP430TUSB3410做USB到UMP的转换,驱动都已经打上,范例程序也已通过。

现在在C# winForm下 作通讯程序,为了查找到虚拟的com口(该虚拟com口FriendlyName不确定,可能是COM9,也可能是COM11或者其他),需要枚举系统的COM类设备信息,得到其接口信息。


int dwNeeded = 0; IntPtr pidd = IntPtr.Zero;
Externs.SP_DEVINFO_DATA devinfo = new Externs.SP_DEVINFO_DATA();
IntPtr g_hdi = IntPtr.Zero;

Guid myGuid = new Guid("4D36E978-E325-11CE-BFC1-08002BE10318"); //驱动文件中的ClassGUID
//("86E0D1E0-8089-11D0-9CE4-08003E301F73"); //范例程序中的GUiD

// Open new enumeration if needed 首次查找释放所有该类设备及其内存
if (index == 0)
{
// Close open device interface enum set if any
if (g_hdi != IntPtr.Zero)
Externs.SetupDiDestroyDeviceInfoList(g_hdi);
g_hdi = Externs.SetupDiGetClassDevs(ref myGuid, 0, IntPtr.Zero, Externs.DIGCF_ALLCLASSES | Externs.DIGCF_PRESENT); //C#下得到g_hdi = 17556736
if (g_hdi == IntPtr.Zero)
return Externs.GetLastError();
}
// Get next device interface data
Externs.SP_DEVICE_INTERFACE_DATA did = new Externs.SP_DEVICE_INTERFACE_DATA();
did.cbSize = Marshal.SizeOf(did);
did.InterfaceClassGuid = System.Guid.Empty;
did.Flags = 0;
did.RESERVED = IntPtr.Zero;
if (!Externs.SetupDiEnumDeviceInterfaces(g_hdi,IntPtr.Zero,ref myGuid,(uint)index,ref did))
{
return Externs.GetLastError();
}


/// <summary>
/// 获取一个指定类别或全部类别的所有已安装设备的信息
/// </summary>
/// <param name="gClass"></param>
/// <param name="iEnumerator"></param>
/// <param name="hParent"></param>
/// <param name="nFlags"></param>
/// <returns></returns>
[DllImport("setupapi.dll", SetLastError = true)]
public static extern IntPtr SetupDiGetClassDevs(ref Guid gClass, UInt32 iEnumerator, IntPtr hParent, UInt32 nFlags);

/// <summary>
/// 销毁一个设备信息集合,并且释放所有关联的内存
/// </summary>
/// <param name="lpInfoSet"></param>
/// <returns></returns>
[DllImport("setupapi.dll", SetLastError = true)]
public static extern int SetupDiDestroyDeviceInfoList(IntPtr lpInfoSet);


/// <summary>
/// 枚举一个设备信息集合的设备接口
/// </summary>
/// <param name="lpInfoSet"></param>
/// <param name="ptrDeviceInfoData"></param>
/// <param name="ptrInterfaceClassGuid"></param>
/// <param name="MemberIndex"></param>
/// <param name="SP_DEVICE_INTERFACE_DATA"></param>
[DllImport("setupapi.dll", CharSet = CharSet.Auto, SetLastError = true)]
public static extern bool SetupDiEnumDeviceInterfaces(IntPtr lpInfoSet, IntPtr ptrDeviceInfoData,
ref Guid ptrInterfaceClassGuid, UInt32 MemberIndex, ref SP_DEVICE_INTERFACE_DATA ptrDeviceInterfaceData);

/// <summary>
/// 设备接口信息
/// </summary>
[StructLayout(LayoutKind.Sequential)]
public class SP_DEVICE_INTERFACE_DATA
{
public int cbSize;
public Guid InterfaceClassGuid;
public int Flags;
public IntPtr RESERVED;
}





范例程序中的代码和执行情况:

const TCHAR szHardwareIDUmp[] = "umpport\\VID_0451_BEEF_com";
static HDEVINFO g_hdi = INVALID_HANDLE_VALUE; // Setup API enum handle
SP_DEVICE_INTERFACE_DATA did;

// Open new enumeration if needed
if (dwIndex == 0) // dwIndex为传入的参数,实际上是枚举的该类设备的编号
{
// Close open device interface enum set if any
if (g_hdi != INVALID_HANDLE_VALUE)
SetupDiDestroyDeviceInfoList(g_hdi); //销毁一个设备信息集合,并且释放所有关联的内存

// Enumerate all devices present (DIGCF_PRESENT)
// 使用此函数,需要包含头文件setupapi.h。在project setting中的link页面需要添加setupapi.lib。
// 查询所有设备,并返回与指定参数匹配的所有已安装设备的设备信息结构的句柄
g_hdi = (HDEVINFO)SetupDiGetClassDevs((CONST LPGUID)&GUID_CLASS_COMPORT,
NULL, NULL, DIGCF_PRESENT | DIGCF_DEVICEINTERFACE);
//查看g_hdi值,“CXX0017:ERROR:symbol ‘g_hdi’nod found!”,这是什么?
//这里的GUID_CLASS_COMPORT “86E0D1E0-8089-11D0-9CE4-08003E301F73” 同样是这个,放入c#还是不行

if (g_hdi == INVALID_HANDLE_VALUE)
{
return GetLastError();
}
}

// Get next device interface data ('dwIndex')
did.cbSize = sizeof(did);

if (!SetupDiEnumDeviceInterfaces(g_hdi, NULL,
(CONST LPGUID)&GUID_CLASS_COMPORT, dwIndex, &did))
{
return GetLastError();
}
//返回正确,&did =
//{cbsize = 28}
//{interfaceClassGuid = {86E0D1E0-8089-11D0-9CE4-08003E301F73}}
//{Flags =1}
//{reserved = 1450704}


查询了很多资料,有的说是ClassGuid与SetupDiEnumDeviceInterfaces所需的参数interface GUID不一致,但是按照方法去找只能找到C#中现在用的那个,还是不行。
现在怀疑是SetupDiGetClassDevs得到的句柄不对,在范例程序跟踪 g_hdi ,总是“CXX0017:ERROR:symbol ‘g_hdi’nod found!”,在C#中却能得到一个g_hdi = 17556736,什么都不一致啊!!

求高手解疑!

...全文
1090 8 打赏 收藏 转发到动态 举报
写回复
用AI写文章
8 条回复
切换为时间正序
请发表友善的回复…
发表回复
hello_symbol 2012-11-02
  • 打赏
  • 举报
回复
意见就是: 我也不知道
yylooker2008 2010-09-03
  • 打赏
  • 举报
回复
有人给点意见吗
yylooker2008 2010-09-03
  • 打赏
  • 举报
回复
或者谁帮我检查下 我写的c#程序是不是有错误啊!
yylooker2008 2010-09-03
  • 打赏
  • 举报
回复
等待高手来指点迷津啊!!
zzbinfo 2010-09-02
  • 打赏
  • 举报
回复
到C#那问问比较好,不懂c#的飘过
yylooker2008 2010-09-02
  • 打赏
  • 举报
回复
up
up
yylooker2008 2010-09-02
  • 打赏
  • 举报
回复
为什么同样的 guid 在c++和c#下 执行结果不一样呢
难道是我的SetupDiEnumDeviceInterfaces参数有错误。
但是这些都是从网上的,他们都能够执行啊
yylooker2008 2010-09-02
  • 打赏
  • 举报
回复
都没人来么 。自己up

现在USB设备却很多,因此对USB设备的查找与读写就必不可少了。但是能找到关于USB读写的资料很少。这里使用VC++示范了一些获得USB的信息的方法。 一、枚举USB设备   通过枚举USB控制器->枚举此控制器上的USB HUB->枚举HUB的各个端口->获得设备信息。 枚举控制器: wsprintf(HCName, "\\\\.\\HCD%d", HCNum); hHCDev = CreateFile(HCName, GENERIC_WRITE, FILE_SHARE_WRITE, NULL, OPEN_EXISTING, 0, NULL);   计算机上的USB主控制器以HCD1,HCD2等命名。通过控制器名称,使用CCreateFile 打开它。使用DeviceIoControl即可得到其驱动程序名,以及与它连接的HUB的名称。用CCreateFile打开HUB,获得连接信息。再枚举HUB的各个端口即可获得连接的设备信息。 二、枚举HID设备   HID设备是微软定义的标准人机接口规范。比如USB鼠标,USB游戏手柄等。不用查找具体设备的GUID,使用API HidD_GetHidGuid(&guidHID)即可得到GUID。有了GUID通过API SetupDiEnumDeviceInterfaces可获得是否有设备连接。如果此类设备连接通过SetupDiGetDeviceInterfaceDetail获得它的设备路径信息。使用CCreateFile 打开它,通过HidD_GetAttributes获得其基本属性信息。使用DeviceIoControl可以获得更详细的属性。在本代码中如果计算机上插有USB游戏手柄,可获得其信息。但不知道为什么xp下却不能获得USB鼠标的信息。 三 枚举U盘   先用GetDriveType API获得设备的类型,若类型为REMOVABLE(当然有些大容量U盘可能报告为FIXED,那就需要其他方法来确定了),即可能是U盘。用CCreateFile 打开之后,再用IOCTL_STORAGE_QUERY_PROPERTY为参数的DeviceIoControl来获得其属性。 四、结束语   示例工程在winxp+xpDDK+VC6下编译通过。USB设备种类比较多,也比较特殊,不同厂商的硬件不同,控制软件也不尽相同(我想主要是ICTL码不同,也不容易查到)。使得访问USB口的设备不象串口并口那么方便。这个例程只是展示了访问的基本方法。其中还有些问题还没有解决,发出来希望大家解决之后能通知我或者发表出来。 参考了USBPort,USBview等代码,一并致谢。

110,534

社区成员

发帖
与我相关
我的任务
社区描述
.NET技术 C#
社区管理员
  • C#
  • Web++
  • by_封爱
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告

让您成为最强悍的C#开发者

试试用AI创作助手写篇文章吧