具有枚举成员的元帅非托管结构到c#

时间:2014-01-01 08:27:29

标签: c# c++ pinvoke marshalling

我有一个unmanaged C++ code,我想将其转换为managed C# code,非托管代码如下所示,我搜索过但我找不到答案...我想知道编组下面的代码的正确方法,我不知道如何编组枚举然后在结构中引用它。MMTPConxNack结构是另一个结构内的联合的成员。 层次结构如下所示:

typedef enum
{
    MMTPCnxNckRsn_NoAnswer=-2,            
    MMTPCnxNckRsn_SendError=-1,         
    MMTPCnxNckRsn_Ok=0,                  
    MMTPCnxNckRsn_InvalidMember,         
    MMTPCnxNckRsn_HubNotReady,           
    MMTPCnxNckRsn_UnknownMember,         
    MMTPCnxNckRsn_LastCnxTooRecent,       
    MMTPCnxNckRsn_InvalidVersion,        
    MMTPCnxNckRsn_InvalidOptions,       
    MMTPCnxNckRsn_TooManyCnx             
} MMTPCnxNckRsn;

typedef struct
{
    MMTPCnxNckRsn Reason;
} MMTPConxNack;


typedef struct
{
    long Length;
    short Type;
    union
    {
        MMTPConxReq ConxReq;
        MMTPConxAck ConxAck;
        MMTPConxNack ConxNack; // the structure with an enum inside
        MMTPErrInd ErrInd;      
    } Data;  
} MMTPMsg;

实际上我想编组MMTPConxNack结构......我使用FieldOffset来定义大小。提前谢谢。

2 个答案:

答案 0 :(得分:1)

您需要一个C#枚举和结构来匹配原生的:

public enum MMTPCnxNckRsn
{
    MMTPCnxNckRsn_NoAnswer,            
    MMTPCnxNckRsn_SendError,         
    MMTPCnxNckRsn_Ok,                  
    MMTPCnxNckRsn_InvalidMember,         
    MMTPCnxNckRsn_HubNotReady,           
    MMTPCnxNckRsn_UnknownMember,         
    MMTPCnxNckRsn_LastCnxTooRecent,       
    MMTPCnxNckRsn_InvalidVersion,        
    MMTPCnxNckRsn_InvalidOptions,       
    MMTPCnxNckRsn_TooManyCnx             
}

public struct MMTPConxNack
{
    public readonly MMTPCnxNckRsn Reason;
}

你像这样编组:

var managedItem = (MMTPConxNack)Marshal.PtrToStructure(pointer,typeof(MMTPConxNack));

答案 1 :(得分:1)

枚举看起来像这样:

public enum MMTPCnxNckRsn {
    MMTPCnxNckRsn_NoAnswer = -2, 
    MMTPCnxNckRsn_SendError = -1, 
    MMTPCnxNckRsn_Ok = 0, 
    MMTPCnxNckRsn_InvalidMember,         
    MMTPCnxNckRsn_HubNotReady,           
    MMTPCnxNckRsn_UnknownMember,         
    MMTPCnxNckRsn_LastCnxTooRecent,       
    MMTPCnxNckRsn_InvalidVersion,        
    MMTPCnxNckRsn_InvalidOptions,       
    MMTPCnxNckRsn_TooManyCnx             
}

包含的结构是:

public struct MMTPConxNack {
    public MMTPCnxNckRsn Reason; 
}

工会是:

[StructLayout(LayoutKind.Explicit)]
public struct MMTPMsgDataUnion
{
    [FieldOffset(0)]
    public MMTPConxReq ConxReq;
    [FieldOffset(0)]
    public MMTPConxAck ConxAck;
    [FieldOffset(0)]
    public MMTPConxNack ConxNack;
    [FieldOffset(0)]
    public MMTPErrInd ErrInd;
}

这是棘手的部分。您使用LayoutKind.ExplicitFieldOffset指定C ++联合的所有成员彼此重叠。显然,您需要为此union中包含的其他3种类型定义,这些定义在问题的C ++代码中无法看到。我认为你已经知道如何定义它们。

一旦声明了union,最终的结构很简单:

public struct MMTPMMsg
{
    public int Length;
    public short Type;
    public MMTPMsgDataUnion Data;
}