双重编辑:仅在使用 libstdc++ 时,由于可能的错误和 Clang 段错误而导致 GCC 段错误。起初我不知道这是因为编译器段错误不是特别有用,但是使用标志 -stdlib=libc++,你可以让 clang 编译这段代码就好了。使用 libstdc++ 时的 Clang 段错误是一个单独的问题,这个问题特别关注 GCC。
可以在其相关的错误报告 #101184 中找到进一步简化的 GCC 示例。如果您对解决方案感兴趣,我建议您跟上错误报告。
在下面的代码墙中,有一些小的 c++20 模块在使用包含的命令构建时会导致GCC-11编译器段错误。
我试图在下面的文字中解释这个问题,但这个问题有点挑剔,我发现理解它的最好方法是稍微尝试一下。
其中两个模块,“filler1”和“filler2”不导出任何东西,只在导出模块指令之前包含几个标准库。因为这两个实际上没有导出任何内容,所以我希望它不会对程序行为产生任何影响,但由于某种原因,如果两者都包括在内,则 GCC 段错误。如果以任何顺序删除其中任何一个,程序都可以正常编译。此外,这些填充模块中包含的标准库是特殊的,如果删除了“功能”或“内存”,即使它们没有在整个程序的任何地方使用,它也可以正常编译。
还有 testmod1 模块,它只包含“向量”。但与填充模块不同的是,要发生段错误,必须在 export 模块指令之后定义一个可能导出也可能不导出的任意类型的向量。
最后一个也可能是最奇怪的部分是 testmodp 模块。除了存在所有前面的微粒之外,还必须定义一个模板结构:template<typename...Ts>和一个成员向量:std::vectorstd::variant<Ts...>,然后由 test_impl 模块实例化. 如果我从结构中删除模板并将成员更改为:“ std::vector<std::variant<int, float>> ”即使它与模板类型实例化的类型相同,它也可以正常工作反而。谁能帮我理解这里发生了什么?
注意:因为这个错误非常特殊,所以我可以在不改变行为的情况下将它提取到最小的 5 个文件示例,我也不允许包含链接,所以如果你想测试它,请复制所有带有对应名称的内容。如果这里有问题,请随时在评论中联系(:
谢谢阅读!
文件“filler1.cc”
module;
#include <memory>
export module filler1;
文件“filler2.cc”
module;
#include <memory>
#include <functional>
export module filler2;
文件“testmodp.cc”
module;
#include <functional>
#include <vector>
#include <variant>
export module testmodp;
export template<typename... Ts> // must be a template with args used in meta_variant for error to occur
struct testmodp{
typedef std::variant<Ts...> meta_variant;
std::vector<meta_variant> meta_gs;
};
文件“testmod1.cc”
module;
#include <vector>
export module testmod1;
export struct testasdf{
std::vector<int> testvec;
};
文件“test_impl.cc”
module;
#include <memory>
import filler1;
import filler2;
import testmodp;
import testmod1;
export module test_impl;
export namespace test_impl {
void test_init(){
new testmodp<int,float>();
}
};
GCC命令运行
mkdir build
g++ -std=c++2a -fmodules-ts -c testmod1.cc -o build/testmod1.pcm
g++ -std=c++2a -fmodules-ts -c filler1.cc -o build/filler1.pcm
g++ -std=c++2a -fmodules-ts -c filler2.cc -o build/filler2.pcm
g++ -std=c++2a -fmodules-ts -c testmodp.cc -o build/testmodp.pcm
g++ -std=c++2a -fmodules-ts -c test_impl.cc -o build/test_impl.pcm
GCC内部编译器错误:
module_test/test3/test_impl.cc:11:8: internal compiler error: in write_location, at cp/module.cc:15605
11 | export module test_impl;
| ^~~~~~
0x1797368 internal_error(char const*, ...)
???:0
0x67f8f9 fancy_abort(char const*, int, char const*)
???:0
0x7652d9 trees_out::core_vals(tree_node*)
???:0
0x768e78 trees_out::tree_value(tree_node*)
???:0
0x761e9c trees_out::tree_node(tree_node*)
???:0
0x763cd6 trees_out::write_var_def(tree_node*)
???:0
0x764fd5 module_state::write_cluster(elf_out*, depset**, unsigned int, depset::hash&, unsigned int*, unsigned int*)
???:0
0x767ec9 module_state::write(elf_out*, cpp_reader*)
???:0
0x768d86 finish_module_processing(cpp_reader*)
???:0
0x713d1b c_parse_final_cleanups()
???:0
Please submit a full bug report,
Please include the complete backtrace with any bug report.