gpt4 book ai didi

c# - 从 C++ 调用 C++ DLL 可以,但不能从 C# 调用

转载 作者:可可西里 更新时间:2023-11-01 17:59:42 36 4
gpt4 key购买 nike

我有一个名为 tccdvc.dll 的 DLL,它是此处提供的 SDK 的一部分:

http://www.commell.com.tw/Download/Driver/Industrial%20Peripheral/Driver/MPX-885/MPX-885%20SDK%20(1.2)/SetupCOMMELL%20MPX-885_20100627.rar

DLL 是用 C++ 编写的,检查 DLL 显示它与链接器版本 6.0 链接,因此我假设它是用 VC++ 6.0 编写的。 DLL 不附带源代码,只有一个.lib 文件和一个.h 文件。所有导出的函数都声明为 extern "C"(因此没有 C++ 名称修改)并使用 APIENTRY(因此 __stdcall)。

我已经在 Windows XP SP3(32 位)上的 Visual Studio 2010 中编写了一个 C++(不是 .NET)程序来访问这个 tccdvc.dll。这在使用提供的 .lib 文件和使用 LoadLibrary/GetProcAddress 时都很好用。我还编写了一个使用 tccdvc.dll 的 C++ DLL(我们称它为 mywrapper.dll),同样,有两个版本,一个使用 .lib 文件,另一个使用 LoadLibrary/GetProcAddress。同样,这工作正常。此 mywrapper.dll 使用 __cdecl 调用约定。它包含一个名为 InitMyWrapperDLL() 的函数,该函数加载 tccdvc.dll。使用 LoadLibrary/GetProcAddress 的 mywrapper.dll 版本具有如下代码:

typedef int (APIENTRY *TCCPROCTYPE01)();

HMODULE TCCmodule;
TCCPROCTYPE01 Proc_TCC_DVCOpen;

extern "C" __declspec(dllexport) void InitMyWrapperDLL ()
{ TCCmodule = LoadLibrary("tccdvc.dll");
Proc_TCC_DVCOpen = (TCCPROCTYPE01)GetProcAddress(TCCmodule, "TCC_DVCOpen");
...
}

同样,使用 C++ 前端,效果很好。但是,当从 C#(在同一台机器上)调用它时,LoadLibrary("tccdvc.dll") 调用返回 NULL。在 C# 中,我使用:

[DllImport("mywrapper.dll", CallingConvention=CallingConvention.Cdecl, CharSet=CharSet.Ansi, ExactSpelling=true, EntryPoint="InitMyWrapperDLL")]
private static extern void InitMyWrapperDLL ();

...

InitMyWrapperDLL();

当使用提供的 tccdvc.lib 文件编译 mywrapper.dll 时,它也失败了,错误代码为 0x8007045a(也称为 1114),这意味着 DLL 初始化失败,它给出 mywrapper.dll 作为动态链接库。原来失败是因为tccdvc.dll,它是通过mywrapper.dll加载的。

在 C# 中使用以下内容也会失败:

[DllImport("tcc.dll", CallingConvention=CallingConvention.StdCall, CharSet=CharSet.Ansi, ExactSpelling=true, EntryPoint="TCC_DVCOpen")]
private static extern Int32 TCC_DVCOpen ();

...

TCC_DVCOpen();

我也在声明中使用了“不安全”,但这没有任何区别。可以预见,因为 LoadLibrary() 失败,所以它甚至没有到达 TCC_DVCOpen()。

为了查明问题所在,我再次使用了 mywrapper.dll 的 LoadLibrary/GetProcAddress 版本,并将以下代码放入我的 C# 程序中:

[DllImport("kernel32.dll")]
private static extern IntPtr LoadLibrary (string lpLibFileName);

[DllImport("kernel32.dll")]
private static extern Int32 GetLastError ();

...

IntPtr hdll1 = LoadLibrary("mywrapper.dll");
IntPtr hdll2 = LoadLibrary("tccdvc.dll");
Int32 errcode = GetLastError();

在此之后,hdll1 具有有效值,但 hdll2 为 0。使用 .NET 3.5 Framework 时,GetLastError() 再次返回 0x8007045a,但使用 .NET 4.0 时,GetLastError() 返回 0 (ERROR_SUCCESS)。

我使用 Sysinternals 的 Process Monitor 获取更多信息,我可以看到 tccdvc.dll 正在被成功读取和映射。 Process Monitor 显示的任何内容都没有提示我为什么在使用 C# 时会失败,但在使用 C++ 时却不会。

有什么想法吗?谢谢!

最佳答案

我有一些建议给你:

  • 您可以创建一个 C++/CLI 类库项目,然后在您的 C# 项目中引用它。
  • 在我的案例中,我发现 UnmanagedFunctionPointerAttribute 对调用至关重要,
  • 有一个案例,无论我做什么,对 .DLL 的调用都无法从 C# 调用,只有 .LIB 对我有用(这意味着实现我的第一个建议)。故障排除让我知道“DLL 空间”不适合那个特定的库。

(关于最后一句话:我绝不是 C++ 专家,实际上那是我迄今为止所做的唯一项目。这当然值得更多详细信息,但我不知道也没有找到问题根源的知识,但它得到修复是因为我只是需要它。感谢指出任何错误/更好的解释。)

以下是适用于我的问题的一些解决方案:

How to use a C callback from C#?有指向 .DLL 的链接,如果您想查看我的所有代码,请询问。

另外,一些在这个领域对我有帮助的工具:

愿原力与你同在:-)

关于c# - 从 C++ 调用 C++ DLL 可以,但不能从 C# 调用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11345724/

36 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com