我从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和编组,所以我不确定这是否正确。有更好的方法吗?
答案 0 :(得分:1)
两个版本都很好,很难猜到什么是错的。嘎嘎叫像“模拟器”,没有正确模拟。标签id的IntPtr是奇数。您应该使用 result 做一些合理的事情,例如在收到错误代码时抛出异常。
要求您传递缓冲区的C函数通常很麻烦,您必须猜测正确的缓冲区大小。挑选2048是一个希望 - 它足够大的猜测,当你猜到太低时确实会出错。这种功能的通用协议是你必须调用它两次。首先是data_length
的故意低值,如0.然后该函数返回错误代码并将data_length
设置为所需的缓冲区大小。然后使用正确大小的缓冲区再次调用它。这只是猜测,它确实适合您的问题。