调用DLL方法时,方法的类型签名不兼容PInvoke

时间:2012-12-09 09:21:41

标签: c# c++ c dll dllimport

我有一个带接口的DLL

struct modeegPackage
{
    uint8_t     version;    // = 2
    uint8_t     count;      // packet counter. Increases by 1 each packet
    uint16_t    data[6];    // 10-bit sample (= 0 - 1023) in big endian (Motorola) format
    uint8_t     switches;   // State of PD5 to PD2, in bits 3 to 0
};

__declspec(dllexport) void __cdecl initSerial();

__declspec(dllexport) void __cdecl closeSerialPort();

__declspec(dllexport) struct modeegPackage __cdecl getPackage();

和C#adapter

class EEGCommunication
{
    [StructLayout(LayoutKind.Sequential)]
    public struct modeegPackage
    {

        /// unsigned char
        public byte version;

        /// unsigned char
        public byte count;

        /// unsigned int[6]
        [MarshalAs(UnmanagedType.ByValArray, SizeConst = 6, ArraySubType = UnmanagedType.U2)]
        public UInt16[] data;

        /// unsigned char
        public byte switches;
    }

    private const string DLL = "libneureader-lib.dll";

    [DllImport(DLL, EntryPoint = "_Z10initSerialv")]
    public static extern void InitSerial();

    [DllImport(DLL, EntryPoint = "_Z15closeSerialPortv")]
    internal static extern void CloseSerialPort();

    [DllImport(DLL, EntryPoint = "_Z10getPackagev", CallingConvention = CallingConvention.Cdecl)]
    public static extern modeegPackage GetPackage();
}

但是当我尝试调用GetPackage方法时,收到错误Method's type signature is not PInvoke compatible.

我的代码出了什么问题?

更新:代码已更新

2 个答案:

答案 0 :(得分:5)

在我之前标记为“答案”的答案并不正确,已经有1。5年了。

OP得到错误的原因确实正是错误描述所说的“方法的类型签名不是PInvoke兼容的”

如果你有一个C / C ++函数,如下面声明的那个,

    __declspec(dllexport) struct modeegPackage __cdecl getPackage();

由于函数返回的struct值大于任何寄存器可以容纳的值,GCC编译器将尝试对其进行优化(返回值优化),因此实际实现如下所示,

    __declspec(dllexport) void __cdecl getPackage(struct* modeegPackage);

所以你的P / Invoke声明应该是,

    [DllImport(DLL, EntryPoint = "_Z10getPackagev", CallingConvention = CallingConvention.Cdecl)]
    public static extern GetPackage(out modeegPackage);

我希望我的回答可以帮助其他可能在将来遇到类似问题的开发人员。

答案 1 :(得分:0)

导致问题的是数组。 pinvoke marshaller不喜欢在通过值返回结构作为函数返回值的特定情况下处理它。这通常很麻烦,这样做的方式高度依赖于编译器。很有可能你会遇到麻烦,因为它听起来像你正在使用GCC。调用者为堆栈上的返回值分配空间并将指针传递给它,通常完成。

一个原始但有效的技巧是自己扩展阵列,足够实用,因为它只有6个元素。像这样替换数组:

        /// unsigned int[6]
        public short data0;
        public short data1;
        //...
        public short data5;

哪个会解决异常。您是否正确获取数据还有待观察,如果没有,那么您可能需要切换到MSVC。