1

我正在开发一个 .NET DLL 来访问给我的 C++ 库。在 C++ 库中,有这个结构:

typedef struct FOO
{
    DWORD DataSize;
    BYTE  *pData;
}

因此,我在 C# 中重新创建了它:

[StructLayout(LayoutKind.Sequential)]
public struct FOO
{
    public uint DataSize;
    public byte[] pData;
}

接下来是我从 C++ DLL 的导入。我也将包含 C++ 端的标题。C++ 中的方法需要一个指向我传入的结构的指针,因此从我收集到的内容中传入的引用将在这种情况下起作用:

// C++ Header
HRESULT CallFoo(FOO * pFoo);

[DllImport("SomeLibrary.DLL", EntryPoint = "CallFoo")]
private static extern uint CallFoo(ref FOO rFoo);

当我在 C++ 端进入我的代码时,我得到了结构,但 pData 中的值是一个内存地址。这似乎污染了 C++ 库中的代码,但我无法理解它返回给我的 HRESULT(我已经向 C++ 库的所有者提出了一个关于错误消息是什么的问题)。

我根据这个问题的答案采取的另一种方法是尝试传递 IntPtr 而不是字节数组。我修改了结构:

[StructLayout(LayoutKind.Sequential)]
public struct FOO
{
    public uint DataSize;
    public IntPtr pData;
}

并称之为:

FOO fooParm = new Foo();
var ptr = IntPtr.Zero;
byte[] bArr = MethodThatReturnsAByteArray();

ptr = Marshal.AllocHGlobal(bArr.Length);
Marshal.Copy(bArr, 0, ptr, bArr.Length);

fooParm.pData = ptr;
fooParm.DataSize = bArr.Length;

uint i = CallFoo(fooParm);

不幸的是,这也不起作用。我得到与原始方法相同的错误代码。

4

2 回答 2

1

你的整个方法不好。关于托管结构/类的布局有一些信息,但最好不要使用它。尝试这样做:

[DllImport("SomeLibrary.DLL", EntryPoint = "CallFoo")]
private static extern uint CallFoo(uint DataSize, IntPtr pData);

您来回传递的参数越简单,结果就会越好。

于 2012-10-12T06:37:17.987 回答
1

我面前没有 C++ 库,所以我觉得很难准确地回答这个问题,但听起来你得到了一个指针,而库对此有问题。是否可以使用Marshal::PtrToStructure将该byte[]指针映射到正确的struct?

这是我正在谈论的特定方法的链接。靠近底部的第二个代码示例(在 C++ 中)可能适用:http: //msdn.microsoft.com/en-US/library/4ca6d5z7.aspx

于 2012-10-12T06:42:29.187 回答