C++ から C# に構造体の配列を渡す必要があります。次のコードは、最初に一時構造を作成し、次にmemcpy
C# 側の構造のアドレスに対して実行します。要素の同じ順序で両側の構造を定義しました。
デバッグすると、一時構造体が正しく埋められていることがわかります。temp_buf (ターゲット変数のアドレス) は、反復ごとに構造体のサイズがインクリメントされ、memcpy
エラーは返されません。
ただし、配列の最初の項目のみが C# 側で設定されます。
C# 側の定義は次のとおりです。
[DllImport(@"MyAwesomeDLL.dll", CallingConvention = CallingConvention.Cdecl, EntryPoint = "?GetDevices@@YAHPAXIPAIPAUDEVICE_S@@@Z", CharSet = CharSet.Auto)]
public static extern Int32 GetCoolDevices(out UInt32 p_deviceCount, out Device_s p_devicesBuf);
[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi, Pack = 1)]
public struct Device_s
{
[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 51)]
public String DeviceName;
public UInt32 DeviceID;
}
C++ 側の定義は次のとおりです。
#pragma pack (1)
typedef struct
{
TCHAR device_name [51];
UINT32 device_rid;
} DEVICE_S;
int GetDevices (UINT32 *device_count, DEVICE_S *devices_buf)
{
....
while(....)
{
DEVICE_S tmp_device;
memset(&tmp_device, 0, sizeof(DEVICE_S));
tmp_device.device_id = 5;
sprintf( tmp_device.device_name, "a string");
DEVICE_S *temp_buf = &(devices_buf[i]);
size_t mysize = sizeof(DEVICE_S);
memcpy(temp_buf, &tmp_device, mysize);
i++;
getNextDevice();
}
.....
}
そして、これが私がそれを呼ぶ方法です:
UInt32 MyDecDeviceCnt = 0;
Device_s[] MyDecDevices = new Device_s[10];
int ret = GetCoolDevices(out MyDecDeviceCnt, out MyDecDevices[0]);
どんな提案でも大歓迎です!