从C#调用非托管(C++)函数时发生PInvoke错误

PInvoke error when calling unmanaged (C++) function from C#

本文关键字:函数 错误 PInvoke C++ 调用      更新时间:2023-10-16

我有一个未经管理的C++dll,我已经编写并测试过了。当在未经管理的控制台应用程序中构建和运行时,未经管理代码会非常出色。函数声明如下所示。

#ifndef IMPORT_MYLIB
#    define MYLIB_API __declspec(dllexport)
#else
#    define MYLIB_API __declspec(dllimport)
#endif
namespace gsod_data_parsing {
extern "C"
{
  MYLIB_API int parse_raw_gsod_file_nocb(
    const char *path,
    int temp_threshold
  );
}
}

我正试图从托管应用程序调用此。我在C#文件中这样声明函数:

 [DllImport("GSODDLL.dll")]
 public static extern int parse_raw_gsod_file_nocb(
   [MarshalAs(UnmanagedType.LPStr)] string path,
   int temp_threshold
 );

这些功能随后在两个并行任务上执行,如下所示:

// Start a task - this runs on the background thread...
task1 = Task.Factory.StartNew(() => 
{
  int t1 = parse_raw_gsod_file_nocb(filePaths[i], this._frostTemp);
  return (t1 == 0);
}, this._tokenSource.Token);

它一开始运行得很好,但后来(我相信当函数执行完毕时(我出现了以下错误。

对PInvoke函数"数据库"的调用创造者!数据库创建者。表单1::parse_raw_gsod_file_nocb'具有使堆栈不平衡。这可能是因为托管PInvoke签名与非托管目标签名不匹配。检查一下PInvoke签名匹配的调用约定和参数目标非托管签名。

我在这里看到了一个类似的问题,但我不太理解公认的答案。

有人有什么想法吗?感谢

所以我的问题是由于调用约定不匹配。根据本文档,windows中C/C++程序的默认调用约定是Cdecl。此外,根据本文档,PInvoke的默认调用约定为StdCall。我最初没有指定任何调用约定,所以它默认为StdCall。由于这些约定指定了在函数调用后如何清理堆栈,因此在函数执行结束时抛出错误是有道理的。

将我的PInvoke声明更改为这个解决了我的问题:

[DllImport("GSODDLL.dll", CallingConvention = CallingConvention.Cdecl)]
public static extern int parse_raw_gsod_file_nocb(
  [MarshalAs(UnmanagedType.LPStr)] string path,
  int temp_threshold
);