【问题标题】:Trouble Mapping DeviceIoControl Call Return Value Structs from C to C#将 DeviceIoControl 调用返回值结构从 C 映射到 C# 时遇到问题
【发布时间】:2016-10-02 05:37:33
【问题描述】:

我正在修改一些查询设备树并报告设备功能的代码,特别是 USB 端口及其速度。我从 DeviceIoControl 调用中取回了 _USB_NODE_CONNECTION_INFORMATION_EX 对象,但它错误地将 USB 3.0 报告为高速。进一步的研究表明,我需要从 DeviceIoControl 获取 _USB_NODE_CONNECTION_INFORMATION_EX_V2 结构以适应 USB 3.0。我正在尝试将新结构从 C 映射到 C#,并且对如何处理作为新(ish)_USB_NODE_CONNECTION_INFORMATION_EX_V2 结构成员存在的两个 C 联合有点困惑。

我在大学里学过 C 哦,比如说,差不多 20 年前。我的大部分职业生涯都在 Powerbuilder(别笑那是很久以前)、Java(别笑那是很久以前),但主要是 C#。任何 C 大师都可以帮助我将这些带有联合的结构映射到 C# 结构中,并可能给我一个关于如何编组它们的线索吗?请注意,我已经映射了 _USB_NODE_CONNECTION_INFORMATION_EX_V2 但并不真正知道如何处理 _USB_PROTOCOLS 和 _USB_NODE_CONNECTION_INFORMATION_EX_V2_FLAGS 结构。最好将 _USB_PROTOCOLS 结构映射到 C# Flags 枚举中,这样我也可以按位使用 and's 和 or's。

下面的代码显示了注释掉的 C typedef 和我在 C# 中的相应结构(未注释和......好吧,未完成:()。提前谢谢大家!

//typedef struct _USB_NODE_CONNECTION_INFORMATION_EX_V2
    //{
    //    ULONG ConnectionIndex;
    //    ULONG Length;
    //    USB_PROTOCOLS SupportedUsbProtocols;
    //    USB_NODE_CONNECTION_INFORMATION_EX_V2_FLAGS Flags;
    //}
    //USB_NODE_CONNECTION_INFORMATION_EX_V2, *PUSB_NODE_CONNECTION_INFORMATION_EX_V2;

    struct USB_NODE_CONNECTION_INFORMATION_EX_V2
    {
        public int ConnectionIndex;
        public int Length;
        public USB_PROTOCOLS SupportedUsbProtocols;
        public USB_NODE_CONNECTION_INFORMATION_EX_V2_FLAGS Flags;
    }

    //typedef union _USB_NODE_CONNECTION_INFORMATION_EX_V2_FLAGS {
    //  ULONG ul;
    //        struct {
    //    ULONG DeviceIsOperatingAtSuperSpeedOrHigher  :1;
    //        ULONG DeviceIsSuperSpeedCapableOrHigher  :1;
    //        ULONG ReservedMBZ  :30;
    //    };
    //}
    //USB_NODE_CONNECTION_INFORMATION_EX_V2_FLAGS, * PUSB_NODE_CONNECTION_INFORMATION_EX_V2_FLAGS;
    struct USB_NODE_CONNECTION_INFORMATION_EX_V2_FLAGS
    {

    }

    //typedef union _USB_PROTOCOLS {
    //  ULONG ul;
    //        struct {
    //        ULONG Usb110  :1;
    //        ULONG Usb200  :1;
    //        ULONG Usb300  :1;
    //        ULONG ReservedMBZ  :29;
    //    };
    //}
    //USB_PROTOCOLS, * PUSB_PROTOCOLS;
    [StructLayout(LayoutKind.Sequential, Pack = 1)]
    struct USB_PROTOCOLS
    {

    }

【问题讨论】:

    标签: c# c winapi


    【解决方案1】:

    USB_PROTOCOLS 是一个联合结构体,它只是包装了一个 32 位整数,其中第 0 位是 Usb110,第 1 位是 Usb200,第 2 位是 Usb300。像这样定义它:

    public struct USB_PROTOCOLS
    {
        UInt32 protocols;
    
        public bool Usb110 { get { return (this.protocols & 0x01) == 0x01; } }
        public bool Usb200 { get { return (this.protocols & 0x02) == 0x02; } }
        public bool Usb300 { get { return (this.protocols & 0x04) == 0x04; } }
    
    }
    

    USB_NODE_CONNECTION_INFORMATION_EX_V2_FLAGS 的故事大致相同,但为了完整性:

    public struct USB_NODE_CONNECTION_INFORMATION_EX_V2_FLAGS
    {
        UInt32 flags;
    
        public bool DeviceIsOperatingAtSuperSpeedOrHigher 
        { 
            get { return (this.flags & 0x01) == 0x01; } 
        }
        public bool DeviceIsSuperSpeedCapableOrHigher
        {
            get { return (this.flags & 0x02) == 0x02; }
        }
    }
    

    或者,您可以使用Flags 属性将这两个都定义为枚举。

    [Flags]
    public enum USB_PROTOCOLS : uint
    {
        Usb110 = 0x01,
        Usb200 = 0x02,
        Usb300 = 0x04,
    }
    
    [Flags]
    public enum USB_NODE_CONNECTION_INFORMATION_EX_V2_FLAGS : uint // nice short type name
    {
        DeviceIsOperatingAtSuperSpeedOrHigher = 0x01,
        DeviceIsSuperSpeedCapableOrHigher = 0x02,
    }
    

    或者因为三个选项优于两个,您也可以将第一个结构的定义更改为:

    struct USB_NODE_CONNECTION_INFORMATION_EX_V2
    {
        public int ConnectionIndex;
        public int Length;
        public uint SupportedUsbProtocols;
        public uint Flags;
    }
    

    并自己进行位操作。

    我是怎么做到的?我在 Visual Studio 中打开了一个 C++ 控制台应用程序,并编写了以下内容,然后在调试器中单步执行:

    #include <usbioctl.h>
    
    int size = sizeof(USB_PROTOCOLS); // size = 4
    USB_PROTOCOLS proto = { 0 }; 
    proto.Usb110 = 1;
    unsigned int val = *((unsigned int*)(&proto)); // val = 1
    proto.Usb110 = 0;
    proto.Usb200 = 1;
    val = *((unsigned int*)(&proto)); // val = 2
    proto.Usb200 = 0;
    proto.Usb300 = 1;
    val = *((unsigned int*)(&proto)); // val = 4
    
    size = sizeof(USB_NODE_CONNECTION_INFORMATION_EX_V2_FLAGS); // size = 4
    USB_NODE_CONNECTION_INFORMATION_EX_V2_FLAGS flags = { 0 };
    flags.DeviceIsOperatingAtSuperSpeedOrHigher = 1;
    val = *((unsigned int*)(&flags)); // val = 1
    flags.DeviceIsOperatingAtSuperSpeedOrHigher = 0;
    flags.DeviceIsSuperSpeedCapableOrHigher = 1;
    val = *((unsigned int*)(&flags)); // val = 2
    

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多