如何在 CLang C++ 下编译/使用头单元模块?

问题描述

文档中说 CLang 中的模块支持是部分的。我在最近发布的 LLVM 12.0 的 Windows 64 位下使用 CLang。

我成功地使用了常规模块(您通过 import modulename; 导入)。

但我还没有设法创建和使用标题单元模块,即您通过 import "header.hpp"; 导入的模块。你能用例子来建议如何做到这一点吗?

为了尝试标题单元,我创建了下一个玩具文件

hello.hpp

#include <vector>

use.cpp

import "hello.hpp";

int main() {
    std::vector<int> v(123);
}

然后我成功(我希望)将头单元 hello.hpp 编译成 PCM 文件

clang++ -std=c++20 -Xclang -emit-header-module -I. hello.hpp -o hello.pcm

命令运行没有错误并产生 hello.pcm。如果您在没有 -o 标志的情况下运行上面的命令,则文件 hello.hpp.gch 已创建。

然后我尝试编译 use.cpp,但没有成功,不知何故它无法识别我的标题单元和/或找不到相应的 hello.pcm。我想我错过了一些特殊的标志,这些标志表明编译器是头单元。使用了下一个命令:

clang++ -std=c++20 -fprebuilt-module-path=. -fmodule-file=hello.hpp=hello.pcm -I. use.cpp

出现编译错误

use.cpp:1:8: error: header file "hello.hpp" (aka './hello.hpp') cannot be imported because it is not kNown to be a header unit
import "hello.hpp";
       ^

在 MSVC 下,我成功地使用了常规模块和标题单元模块。但不是在 Clang 中。你能帮我解决这个问题吗?或者告诉我可能还不支持 CLang 标头单元。

解决方法

最后我设法解决了上面的几乎任务。

以下说明适用于 Windows 64 位、来自 LLVM 12.0 版本的最新 CLang(您可以获得 here)和最新的 MSVC Community Build Tools 2019 v16.9.4(您可以从 {{3 }}).

我解决的任务不是完全针对标头单元,而是针对标头模块,它们的行为几乎相同,它们的用法没有区别。

玩具示例文件如下:

module.modulemap

module mod {
    requires cplusplus17
    header "mod.hpp"
    export *
}

mod.hpp

#include <iostream>

use.cpp

import mod;

int main() {
    std::cout << "Hello,world!" << std::endl;
}

我使用了接下来的 3 个命令:

clang++.exe -cc1 module.modulemap -o prebuilt/mod.pcm -emit-module -fmodules -fmodule-name=mod -std=c++20 ^
    -internal-isystem "d:\\bin2\\Microsoft Visual Studio\\2019\\BuildTools\\VC\\Tools\\MSVC\\14.28.29910\\include" ^
    -internal-isystem "C:\\Program Files (x86)\\Windows Kits\\10\\Include\\10.0.19041.0\\ucrt" ^
    -debug-info-kind=limited -fms-extensions -fms-compatibility -fms-compatibility-version=19.28 -xc++ ^
    -fmath-errno -fexceptions -fcxx-exceptions -triple x86_64-pc-windows-msvc || exit /b
    
clang++.exe -cc1 -emit-obj use.cpp -fmodule-file=prebuilt/mod.pcm -std=c++20 ^
    -internal-isystem "d:\\bin2\\Microsoft Visual Studio\\2019\\BuildTools\\VC\\Tools\\MSVC\\14.28.29910\\include" ^
    -internal-isystem "C:\\Program Files (x86)\\Windows Kits\\10\\Include\\10.0.19041.0\\ucrt" ^
    -debug-info-kind=limited -fms-extensions -fms-compatibility -fms-compatibility-version=19.28 -xc++ ^
    -fmath-errno -fexceptions -fcxx-exceptions -triple x86_64-pc-windows-msvc || exit /b
    
clang++.exe use.o -o use.exe || exit /b

这一切都没有错误。您可以看到包含标准库目录的完整路径,这些路径特定于我的系统。这是必需的,因为在命令中我使用了 -cc1 选项,它允许使用低级 CLang 前端而不是简化的驱动程序,这个前端需要很多低级选项才能工作。

您只需执行 clang++ -### use.cpp 即可获得所有选项,这将转储到控制台您系统所需的所有选项。

以上命令只能用于 -cc1 前端,驱动不支持模块映射文件。

其实上面第二条命令中的3条命令可以简化,编译目标文件不需要低级前端。但只有在第一个命令有clang -###命令获得的默认参数的情况下才能简化,然后第二个命令可以简写为clang++ use.cpp -o use.o -c -std=c++20 -fmodule-file=prebuilt/mod.pcm

这些命令的结果是在几分之一秒内编译 use.o。众所周知,iostream 需要很多时间来编译。 use.o 的快速编译意味着我们正确使用了模块并提高了速度。

为什么我首先需要标题单元?为了能够升级我的旧代码,只需将常规旧式包含自动替换为导入,即可大大缩短编译时间。这种更换仅适用于集管单元或集管模块。据我所知,常规模块无法导出其他完整标题。

有关模块的更多说明,请参阅 CLang 的 hereModules Doc

相关问答

Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其...
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。...
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbc...