【发布时间】:2012-07-05 22:33:20
【问题描述】:
我有一个名为 tccdvc.dll 的 DLL,它是此处提供的 SDK 的一部分:
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++ 时却没有。
有什么想法吗? 谢谢!
【问题讨论】:
-
tccdvc.dll 是否有任何其他可通过依赖查看的依赖项?想也许那些没有被 .net 应用程序找到
-
Using Dependency Walker from link 显示了 5 个直接依赖项:kernel32.dll、user32.dll、advapi32.dll、ole32.dll、oleaut32.dll,它们都存在于系统中。许多级别嵌套在 advapi32.dll 中,这表明延迟加载依赖项 ieshims.dll 和 wer.dll 不存在。但是,根据 Process Monitor,它从不尝试加载这些。
-
tccdvc.dll在加载过程中是否产生任何额外的线程? -
如果您没有该 DLL 的源代码,则需要联系供应商寻求支持。
-
P/调用 GetLastError 并不可靠。您需要使用Marshal.GetLastWin32Error。
标签: c# c++ dll loadlibrary