15

我正在尝试在 g++ 中启用链接时间优化。-flto没有该选项,我的程序编译得很好。当我将它添加到我的 Makefile 时,目标文件编译时没有错误,例如

g++ main.cpp -I ../includes -std=c++0x -fopenmp -Wall -pedantic -Wno-vla -flto -D INFO_ -c -o .obj/main.o

但是当涉及到链接程序时:

g++ -fwhole-program -I ../includes -std=c++0x -fopenmp -Wall -pedantic -Wno-vla -flto -D INFO_ .obj/main.o .obj/atom.o .obj/bee.o .obj/colony.o ../includes/.obj/error.o ../includes/.obj/CmdLine.o ../includes/boost_lib_deb/libboost_program_options.a ../includes/gmp_lib_deb/lib/libgmpxx.a ../includes/gmp_lib_deb/lib/libgmp.a -o BeeBench

我收到很多这样的错误:

includes/gmp_lib_deb/lib/libgmpxx.a ../includes/gmp_lib_deb/lib/libgmp.a -o BeeBench
`typeinfo for boost::program_options::too_many_positional_options_error' referenced in section `.rodata._ZTVN5boost15program_options33too_many_positional_options_errorE[vtable for boost::program_options::too_many_positional_options_error]' of ../includes/boost_lib_deb/libboost_program_options.a(cmdline.o): defined in discarded section `.gnu.linkonce.t._ZTIN5boost15program_options33too_many_positional_options_errorE' of .obj/main.o (symbol from plugin)

`typeinfo for boost::program_options::too_many_positional_options_error' referenced in section `.rodata._ZTIN5boost16exception_detail19error_info_injectorINS_15program_options33too_many_positional_options_errorEEE[typeinfo for boost::exception_detail::error_info_injector<boost::program_options::too_many_positional_options_error>]' of ../includes/boost_lib_deb/libboost_program_options.a(cmdline.o): defined in discarded section `.gnu.linkonce.t._ZTIN5boost15program_options33too_many_positional_options_errorE' of .obj/main.o (symbol from plugin)
`typeinfo for boost::program_options::invalid_command_line_style' referenced in section `.rodata._ZTVN5boost15program_options26invalid_command_line_styleE[vtable for boost::program_options::invalid_command_line_style]' of ../includes/boost_lib_deb/libboost_program_options.a(cmdline.o): defined in discarded section `.gnu.linkonce.t._ZTIN5boost15program_options26invalid_command_line_styleE' of .obj/main.o (symbol from plugin)

我不知道出了什么问题。我使用-flto. 这些库,即 Boost 和 GMP,是在没有-flto选项的情况下编译的。这是导致错误吗?gcc 手册说可以混合使用 & 不带-flto选项编译的目标文件。还是我错过了其他东西,例如错误所说的这个插件是什么?

我在 Debian Wheezy 上使用 G++ 4.6.3。

更新:

正如评论中的建议,我做了一个最小的例子。我的测试程序的代码只有这样:

#include "boost/program_options.hpp"

int main ( int argC, char* argV[] )
{
    return 0;
}

当我使用以下方法编译它时:

g++ -o test -I ../includes -Wall -std=c++0x test.cpp -flto -fwhole-program -static

它给出了与上述类似的错误。如果我省略 -static、-flto 或 std=c++0x 选项,它编译时不会出错。-fwhole-program 选项不会改变结果。我现在也用 G++ 4.7 进行了测试,同样的错误。

有什么建议么?这真的是编译器错误,还是我仍然做错了什么?

4

3 回答 3

13

由于我没有发现任何证据表明我的代码有问题,因此我发布了Boost错误报告。它也被其他 boost 用户复制,因此我认为它实际上是 boost 或 g++ 中的一个错误。到目前为止,Boost 维护者没有任何回应。有的话我会更新这篇文章。

更新

似乎是 g++ 链接器插件导致了问题(我仍然不知道为什么)。因此,一种可能的解决方法是通过使用禁用链接器插件-fno-use-linker-plugin

于 2012-06-25T13:33:53.563 回答
3

库中的某些内容引用了该特定类的类型信息(通常是针对该特定异常的“catch”语句或“dynamic_cast”)。因此,您的错误消息中的“在部分中引用”。

但是,要生成 typeinfo,非内联非纯虚函数必须存在于其中一个编译单元中。如果函数是在类定义中定义的,则不计算在内(它被内联并被视为内联链接,即使您传递了“-fno-default-inline”)。

因此,dynamic_cast 或 catch 语句可能无法按作者的预期工作;直到在标题上尝试 LTO 之前,这个问题才被注意到。

因此,我将其称为 BOOST 错误和/或 g++ 的缺点。

于 2012-07-20T15:57:14.447 回答
1

-fltoflag 必须同时出现在编译和链接器命令行中才能工作。-fwhole-program,另一方面,根本不需要。顺便说一下,LTO 不能与未在 LTO 支持下编译的翻译单元一起使用。

于 2012-04-06T23:03:31.317 回答