144

我在 C:\cryptopp 中下载并提取了 Crypto++。我使用 Visual Studio Express 2012 构建了里面的所有项目(按照自述文件中的说明),一切都构建成功。然后我在其他文件夹中创建了一个测试项目,并添加了 cryptolib 作为依赖项。之后,我添加了包含路径,以便我可以轻松地包含所有标题。当我尝试编译时,我收到一个关于未解析符号的错误。

为了解决这个问题,我添加C:\cryptopp\Win32\Output\Debug\cryptlib.lib了链接其他依赖项。现在我得到这个错误:

Error   1   error LNK2038: mismatch detected for 'RuntimeLibrary': value 'MTd_StaticDebug' doesn't match value 'MDd_DynamicDebug' in program.obj    C:\Data\Work\C++ VS\CryptoTest\CryptoTest\cryptlib.lib(cryptlib.obj)    CryptoTest
Error   2   error LNK2038: mismatch detected for 'RuntimeLibrary': value 'MTd_StaticDebug' doesn't match value 'MDd_DynamicDebug' in program.obj    C:\Data\Work\C++ VS\CryptoTest\CryptoTest\cryptlib.lib(iterhash.obj)    CryptoTest
Error   3   error LNK2038: mismatch detected for 'RuntimeLibrary': value 'MTd_StaticDebug' doesn't match value 'MDd_DynamicDebug' in program.obj    C:\Data\Work\C++ VS\CryptoTest\CryptoTest\cryptlib.lib(sha.obj) CryptoTest
Error   4   error LNK2038: mismatch detected for 'RuntimeLibrary': value 'MTd_StaticDebug' doesn't match value 'MDd_DynamicDebug' in program.obj    C:\Data\Work\C++ VS\CryptoTest\CryptoTest\cryptlib.lib(pch.obj) CryptoTest
Error   5   error LNK2038: mismatch detected for 'RuntimeLibrary': value 'MTd_StaticDebug' doesn't match value 'MDd_DynamicDebug' in program.obj    C:\Data\Work\C++ VS\CryptoTest\CryptoTest\cryptlib.lib(misc.obj)    CryptoTest
Error   6   error LNK2038: mismatch detected for 'RuntimeLibrary': value 'MTd_StaticDebug' doesn't match value 'MDd_DynamicDebug' in program.obj    C:\Data\Work\C++ VS\CryptoTest\CryptoTest\cryptlib.lib(queue.obj)   CryptoTest
Error   7   error LNK2038: mismatch detected for 'RuntimeLibrary': value 'MTd_StaticDebug' doesn't match value 'MDd_DynamicDebug' in program.obj    C:\Data\Work\C++ VS\CryptoTest\CryptoTest\cryptlib.lib(algparam.obj)    CryptoTest
Error   8   error LNK2038: mismatch detected for 'RuntimeLibrary': value 'MTd_StaticDebug' doesn't match value 'MDd_DynamicDebug' in program.obj    C:\Data\Work\C++ VS\CryptoTest\CryptoTest\cryptlib.lib(filters.obj) CryptoTest
Error   9   error LNK2038: mismatch detected for 'RuntimeLibrary': value 'MTd_StaticDebug' doesn't match value 'MDd_DynamicDebug' in program.obj    C:\Data\Work\C++ VS\CryptoTest\CryptoTest\cryptlib.lib(fips140.obj) CryptoTest
Error   10  error LNK2038: mismatch detected for 'RuntimeLibrary': value 'MTd_StaticDebug' doesn't match value 'MDd_DynamicDebug' in program.obj    C:\Data\Work\C++ VS\CryptoTest\CryptoTest\cryptlib.lib(cpu.obj) CryptoTest
Error   11  error LNK2038: mismatch detected for 'RuntimeLibrary': value 'MTd_StaticDebug' doesn't match value 'MDd_DynamicDebug' in program.obj    C:\Data\Work\C++ VS\CryptoTest\CryptoTest\cryptlib.lib(mqueue.obj)  CryptoTest

我也得到:

Error   12  error LNK2005: "public: __thiscall std::_Container_base12::_Container_base12(void)" (??0_Container_base12@std@@QAE@XZ) already defined in cryptlib.lib(cryptlib.obj)    C:\Data\Work\C++ VS\CryptoTest\CryptoTest\msvcprtd.lib(MSVCP110D.dll)   CryptoTest
Error   13  error LNK2005: "public: __thiscall std::_Container_base12::~_Container_base12(void)" (??1_Container_base12@std@@QAE@XZ) already defined in cryptlib.lib(cryptlib.obj)   C:\Data\Work\C++ VS\CryptoTest\CryptoTest\msvcprtd.lib(MSVCP110D.dll)   CryptoTest
Error   14  error LNK2005: "public: void __thiscall std::_Container_base12::_Orphan_all(void)" (?_Orphan_all@_Container_base12@std@@QAEXXZ) already defined in cryptlib.lib(cryptlib.obj)   C:\Data\Work\C++ VS\CryptoTest\CryptoTest\msvcprtd.lib(MSVCP110D.dll)   CryptoTest
Error   15  error LNK2005: "public: __thiscall std::locale::id::id(unsigned int)" (??0id@locale@std@@QAE@I@Z) already defined in cryptlib.lib(iterhash.obj) C:\Data\Work\C++ VS\CryptoTest\CryptoTest\msvcprtd.lib(MSVCP110D.dll)   CryptoTest
Warning 16  warning LNK4098: defaultlib 'LIBCMTD' conflicts with use of other libs; use /NODEFAULTLIB:library   C:\Data\Work\C++ VS\CryptoTest\CryptoTest\LINK  CryptoTest
Error   17  error LNK1169: one or more multiply defined symbols found   C:\Data\Work\C++ VS\CryptoTest\Debug\CryptoTest.exe 1   1   CryptoTest

我试图编译的代码很简单(我从另一个站点得到的):

#include <iostream>
#include <string>
#include "sha.h"
#include "hex.h"
using namespace std;

string SHA256(string data) {
    byte const* pbData = (byte*) data.data();
    unsigned int nDataLen = data.size();
    byte abDigest[32];

    CryptoPP::SHA256().CalculateDigest(abDigest, pbData, nDataLen);

    return string((char*)abDigest);
}

int main(void) {

    return 0;
}

任何想法如何解决这一问题?我现在真的只需要 SHA-256,没有别的。我用的是Windows 7 64位,今天下载了VS C++,应该是最新版本。

4

4 回答 4

270

(这已经在评论中得到了回答,但由于它缺乏实际的答案,我正在写这个。)

这个问题出现在新版本的 Visual C++ 中(旧版本通常只是默默地链接程序,它会在运行时崩溃和烧毁。)这意味着您正在与程序链接的一些库(甚至是一些源程序本身内的文件)正在使用不同版本的 CRT(C 运行时库。)

要更正此错误,您需要进入您的Project Properties(和/或您正在使用的库),然后进入C/C++,然后Code Generation检查 ; 的值Runtime Library。对于您链接在一起的所有文件和库,这应该完全相同。(与 DLL 链接的规则稍微宽松一些,但我不打算在这里讨论“为什么”和更多细节。)

此设置目前有四个选项:

  1. 多线程调试
  2. 多线程调试 DLL
  3. 多线程发布
  4. 多线程发布 DLL

您的特定问题似乎源于您将使用“多线程调试”(即静态多线程调试 CRT)构建的库与使用“多线程调试DLL ”设置(即动态多线程调试 CRT)构建的程序链接。您应该更改这个设置要么在库中,要么在你的程序中。现在,我建议在你的程序中改变它。

请注意,由于 Visual Studio 项目对调试和发布版本(以及 32/64 位版本)使用不同的项目设置集,因此您应该确保所有这些项目配置中的设置匹配。

有关(一些)更多信息,您可以查看这些(从上面的评论链接):

  1. MSDN 上的链接器工具警告 LNK4098
  2. MSDN 上的 /MD、/ML、/MT、/LD(使用运行时库)
  3. 使用 VC11 Beta 构建错误 - 将 MTd 库与 MDd exe 混合无法链接到 Bugzilla@Mozilla

更新:(这是对评论的回应,该评论询问了为什么必须如此小心。)

如果我们链接在一起的两段代码本身链接并使用标准库,那么它们的标准库必须相同,除非非常注意我们的两个代码段如何交互和传递数据。一般来说,我会说,对于几乎所有情况,只需使用完全相同版本的标准库运行时(关于调试/发布、线程,显然是 Visual C++ 的版本,以及迭代器调试等)

问题的最重要部分是:对函数调用两侧的对象大小有相同的想法

例如,考虑上述两段代码被称为AB。A 是针对标准库的一个版本编译的,而 B 是针对另一个版本的。在 A 看来,标准函数返回给它的一些随机对象(例如内存块或迭代器或FILE对象或其他)具有特定的大小和布局(请记住,结构布局在 C/C/ C++。)出于多种原因,B 对相同对象的大小/布局的想法是不同的(这可能是因为额外的调试信息、数据结构随时间的自然演变等)

现在,如果 A 调用标准库并取回一个对象,然后将该对象传递给 B,并且 B 以任何方式接触该对象,则 B 很可能会弄乱该对象(例如,写入错误的字段,或超过结尾等等)

以上不是唯一可能发生的问题。标准库中的内部全局或静态对象也可能导致问题。还有更多晦涩难懂的问题。

当使用 DLL(动态运行时库)而不是 libs(静态运行时库)时,这一切在某些方面变得更加奇怪。

这种情况可以适用于两个协同工作的代码使用的任何库,但是标准库被大多数(如果不是几乎所有)程序使用,这增加了冲突的机会。

如果您混合库版本,我所描述的显然是等待您的实际混乱的淡化和简化版本。我希望它能让您了解为什么不应该这样做!

于 2013-09-05T12:00:09.600 回答
8

我遇到了这个问题以及 ITERATOR_DEBUG_LEVEL 中的不匹配。作为一个周日晚上的问题,毕竟看起来还不错,很好,我被放了一段时间。在 de VS2017 IDE(解决方案资源管理器)中工作我最近从另一个项目添加/复制了对我的项目(ctrl-drag)的源文件引用。查看属性 - > C/C++/预处理器 -在源文件级别,而不是项目级别- 我注意到在发布配置中指定了 _DEBUG 而不是 NDEBUG 用于此源文件。这就是摆脱问题所需的所有改变。

于 2018-07-22T19:03:34.207 回答
3

我在 C:\cryptopp 中下载并提取了 Crypto++。我使用 Visual Studio Express 2012 构建了里面的所有项目(按照自述文件中的说明),一切都构建成功。然后我在其他文件夹中创建了一个测试项目,并添加了 cryptolib 作为依赖项。

转换可能不成功。唯一成功的是 VCUpgrade 的运行。实际的转换本身失败了,但在您遇到所看到的错误之前您不知道。有关一些详细信息,请参阅Crypto++ wiki 上的Visual Studio 。


任何想法如何解决这一问题?

要解决您的问题,vs2010.zip如果您想要静态 C/C++ 运行时链接(/MT/MTd),或者vs2010-dynamic.zip如果您想要动态 C/C++ 运行时链接(/MT/MTd),您应该下载。两者都修复了 VCUpgrade 产生的潜在的、无声的故障。


vs2010.zipvs2010-dynamic.zip并且vs2005-dynamic.zip是从最新的 GitHub 源构建的。在撰写本文时(2016 年 6 月 1 日),这实际上是 Crypto++ 5.6.4 之前的版本。如果您使用较低级别的 Crypto++(如 5.6.2 或 5.6.3)的 ZIP 文件,那么您会遇到一些小问题。

我知道有两个小问题。首先是bench.cpptobench1.cpp的重命名。它的错误是:

  • C1083: Cannot open source file: 'bench1.cpp': No such file or directory
  • LNK2001: unresolved external symbol "void __cdecl OutputResultOperations(char const *,char const *,bool,unsigned long,double)" (?OutputResultOperations@@YAXPBD0_NKN@Z)

解决方法是 (1)cryptest.vcxproj在记事本中打开,找到bench1.cpp,然后将其重命名为bench.cpp. 或 (2) 在文件系统上重命名bench.cppbench1.cpp。请不要删除此文件。

第二个问题有点棘手,因为它是一个移动的目标。下级版本,如 5.6.2 或 5.6.3,缺少GitHub中可用的最新类。缺失的类文件包括HKDF(5.6.3)、RDRAND(5.6.3)、RDSEED(5.6.3)、ChaCha(5.6.4)、BLAKE2(5.6.4)、Poly1305(5.6.4)等。

修复方法是从 Visual Studio 项目文件中删除缺少的源文件,因为它们对于下级版本不存在。

另一种选择是从最新来源添加缺少的类文件,但可能会出现复杂情况。例如,许多来源巧妙地依赖于最新的config.h,cpu.hcpu.cpp. “微妙”是你不会意识到你正在上一个表现不佳的课程。

表现不佳的一个例子是 BLAKE2。config.h添加编译时 ARM-32 和 ARM-64 检测。cpu.hcpu.cpp添加了运行时 ARM 指令检测,这取决于编译时检测。如果您在没有其他文件的情况下添加 BLAKE2,则不会发生任何检测,您将获得直接的 C/C++ 实现。您可能不会意识到您错过了 NEON 机会,它每字节运行大约 9 到 12 个周期,而普通 C/C++ 大约每字节 40 个周期。

于 2015-10-24T02:06:51.053 回答
1

可以通过在链接器库中添加 msvcrtd.lib 的 CRT 来解决问题。因为 cryptlib.lib 使用了 CRT 版本的调试。

于 2019-06-20T09:30:50.430 回答