[英]Unresolved Externals in C++: Visual C++ mangles method signature differently from mangled method in dll
我正在使用托管C ++的Visual Studio 2012在第三方SDK和我们用C#编写的系统之间架起一座桥梁。 我已经成功包装并使用了上述SDK中的几个功能。 除了一个,只会导致一个未解决的外部错误。
SDK的头文件定义了函数的签名:
#if defined WIN32
#if defined BUILD_ADS_SHARED_LIB
#define ADS_LINK_SPEC __declspec (dllexport)
#define ADS_CALLING_CONVENTION __stdcall
#elif defined USE_ADS_SHARED_LIB
#define ADS_LINK_SPEC __declspec (dllimport)
#define ADS_CALLING_CONVENTION __stdcall
#else
#define ADS_LINK_SPEC
#define ADS_CALLING_CONVENTION
#endif
#else
#define ADS_LINK_SPEC
#define ADS_CALLING_CONVENTION
#endif
DatabaseResult ADS_LINK_SPEC ADS_CALLING_CONVENTION
createDatabase(
const Settings& settings, Artec::SdkDatabase::iDatabase *& instance);
错误提示:
Error 10 error LNK2028: unresolved token (0A000089) "enum Artec::SdkDatabase::DatabaseResult __cdecl Artec::SdkDatabase::createDatabase(class Artec::SdkDatabase::Settings const &,class Artec::SdkDatabase::iDatabase * &)" (?createDatabase@SdkDatabase@Artec@@$$FYA?AW4DatabaseResult@12@ABVSettings@12@AAPAViDatabase@12@@Z) referenced in function "private: static enum Artec::SdkDatabase::DatabaseResult __clrcall Broadway3dWrapper::Broadway3dWrapper::GetConn(wchar_t const *,wchar_t const *,wchar_t const *,wchar_t const *,char const *,class Artec::SdkDatabase::iDatabase * &)" (?GetConn@Broadway3dWrapper@1@$$FCM?AW4DatabaseResult@SdkDatabase@Artec@@PB_W000PBDAAPAViDatabase@34@@Z) C:\bioap\tfs\Identitum\Dev\src\BA.Identitum.Devices.Broadway3d\Broadway3dWrapper.obj BA.Identitum.Devices.Brodway3D
因此,它正在寻找名称混乱的名称:
?createDatabase@SdkDatabase@Artec@@$$FYA?AW4DatabaseResult@12@ABVSettings@12@AAPAViDatabase@12@@Z
在引用的dll上做一些dumpbin,我发现实际上有一个名为export的函数,东西的名字有点不同:
?createDatabase@SdkDatabase@Artec@@YG?AW4DatabaseResult@12@ABVSettings@12@AAPAViDatabase@12@@Z
有人能帮我一下吗? 我无法联系SDK供应商,我在这里完全迷路了。
两者之间的区别在于调用约定部分。
createDatabase@SdkDatabase@Artec@@YG?AW4DatabaseResult@12@ABVSettings@12@AAPAViDatabase@12@@Z
是stdcall: enum Artec::SdkDatabase::DatabaseResult __stdcall Artec::SdkDatabase::createDatabase(class Artec::SdkDatabase::Settings const &,class Artec::SdkDatabase::iDatabase * &)
我使用的 ?createDatabase@SdkDatabase@Artec@@$$FYA?AW4DatabaseResult@12@ABVSettings@12@AAPAViDatabase@12@@Z
器无法理解?createDatabase@SdkDatabase@Artec@@$$FYA?AW4DatabaseResult@12@ABVSettings@12@AAPAViDatabase@12@@Z
,但是它们的区别( @@$$FYA?
vs @@YG?
)是调用约定(如果我将YG
更改为YF
,则调用约定会更改,而没有其他操作)。
将函数的声明更改为return-type __stdcall function-name[(argument-list)]
。
当包含头文件时,您是在显式还是在编译器命令行上#define USE_ADS_SHARED_LIB
? 您是否要定位32位窗口?
使用undname.exe实用程序取消修饰名称。 它正在寻找:
enum Artec::SdkDatabase::DatabaseResult
__cdecl
Artec::SdkDatabase::createDatabase(
class Artec::SdkDatabase::Settings const &,
class Artec::SdkDatabase::iDatabase * &
)
您找到的是:
enum Artec::SdkDatabase::DatabaseResult
__stdcall
Artec::SdkDatabase::createDatabase(
class Artec::SdkDatabase::Settings const &,
class Artec::SdkDatabase::iDatabase * &
)
除调用约定__cdecl与__stdcall外,其他所有内容都匹配。 请注意,SDK标头是如何允许这种情况发生的,当BUILD_ADS_SHARED_LIB和USE_ADS_SHARED_LIB都未定义时,它不会发臭。 这将产生一个__cdecl函数。 坏主意顺便说一句。
您非常容易忘记定义USE_ADS_SHARED_LIB。 项目+属性,C / C ++,预处理器,预处理器定义设置。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.