这是编组输出字节数组的正确方法吗?

时间:2013-10-24 20:48:33

标签: c# pinvoke marshalling

我从C dll导出了以下函数:

// C
BOOL WINAPI GetAttributeValue(
        IN     TAG                      * psTag, 
        IN     DWORD                      dwEltIdx,
        IN     DWORD                      dwAttrIdx,
        OUT    BYTE                     * pbBuffer,
        IN OUT DWORD                    * pdwLen )

// C#
[DllImport(Simulator.ASSEMBLY, SetLastError = true, CallingConvention = CallingConvention.StdCall)]
public extern static int GetAttributeValue(
        IntPtr tag_id,
        int element_index,
        int attribute_index,
        [In, Out, MarshalAs(UnmanagedType.LPArray, SizeParamIndex=4)]
        byte[] data_buffer,
        [In, Out]
        ref int data_length
    );

这就是我尝试使用它的方法,基于SO的几个答案:

int result = -1;
byte[] buffer = new byte[2048];
int length = buffer.Length;

result = Simulator.GetAttributeValue(
        tag.NativeId,
        element_index,
        attribute_index,
        buffer,
        ref length
    );

int[] output = new int[length];

for (int i = 0; i < length; i++)
{
    output[i] = buffer[i];
}

return output;

我尝试的另一件事是这个,也基于SO上的答案:

[DllImport(Simulator.ASSEMBLY, SetLastError = true, CallingConvention = CallingConvention.StdCall)]
public extern static int GetAttributeValue(
        IntPtr tag_id,
        int element_index,
        int attribute_index,
        IntPtr data_buffer, // changed this
        [In, Out]
        ref int data_length
    );

// snip

GCHandle pinned_array = GCHandle.Alloc(buffer, GCHandleType.Pinned);
IntPtr pointer = pinned_array.AddrOfPinnedObject();

result = Simulator.GetAttributeValue(
        tag.NativeId,
        element_index,
        attribute_index,
        pointer,
        ref length
    );

// snip, copying stuff to output

pinned_array.Free();
return output;

现在,在这两种情况下,我的length似乎都已正确填写,但buffer始终为空。我不太熟悉P / Invoke和编组,所以我不确定这是否正确。有更好的方法吗?

1 个答案:

答案 0 :(得分:1)

两个版本都很好,很难猜到什么是错的。嘎嘎叫像“模拟器”,没有正确模拟。标签id的IntPtr是奇数。您应该使用 result 做一些合理的事情,例如在收到错误代码时抛出异常。

要求您传递缓冲区的C函数通常很麻烦,您必须猜测正确的缓冲区大小。挑选2048是一个希望 - 它足够大的猜测,当你猜到太低时确实会出错。这种功能的通用协议是你必须调用它两次。首先是data_length的故意低值,如0.然后该函数返回错误代码并将data_length设置为所需的缓冲区大小。然后使用正确大小的缓冲区再次调用它。这只是猜测,它确实适合您的问题。