缺少 Apple Clang 13 C++20 模块支持

Apple Clang 13 C++20 Module Support Missing

根据 official documentation,Clang 13 通过使用 -fmodules 命令行参数支持 C++20 模块。

我什至无法在基于 Intel 或 M1 的 mac 上使用 Clang 13 (macOS Monterey) 编译基本模块。

假设文件 module.cpp 的文本内容如下:

export module a;

export int f(int a, int b) {
    return a + b;
}

运行 以下:

$ clang++ --version
Apple clang version 13.0.0 (clang-1300.0.29.3)
Target: x86_64-apple-darwin21.1.0

$ clang++ -std=c++20 -stdlib=libc++ -fmodules -fbuiltin-module-map -c module.cpp 
module.cpp:1:8: error: expected template
export module a;
       ^
module.cpp:1:8: error: unknown type name 'module'
module.cpp:3:8: error: expected template
export int f(int a, int b) {
       ^
3 errors generated.

在 ARM M1 芯片上测试结果相同:

$ clang++ --version
Apple clang version 13.0.0 (clang-1300.0.29.3)
Target: arm64-apple-darwin21.1.0

是否有其他方法让模块正常工作,或者 Apple Clang 13 是否存在一些未记录的 C++20 模块限制?

注意:使用实验性 -fmodules-ts 标志进行编译。

原因

感谢您的评论 - 以上错误表明 Clang 版本是在没有模块支持的情况下构建的。这就是 Xcode 附带的,即在终端中通过 运行 xcode-select --install

解决方案

正如所建议的那样,解决方案是通过 HomeBrew 安装 Clang,操作如下(在 macOS Monterey 上测试):

brew install llvm

Clang 安装到 /opt/homebrew/opt/llvm/bin/clang++。确认运行版本如下图:

% /opt/homebrew/opt/llvm/bin/clang++ --version
Homebrew clang version 13.0.0
Target: arm64-apple-darwin21.1.0
Thread model: posix
InstalledDir: /opt/homebrew/opt/llvm/bin

与 Xcode 系统范围默认版本不同的版本:

% clang++ --version
Apple clang version 13.0.0 (clang-1300.0.29.3)
Target: arm64-apple-darwin21.1.0
Thread model: posix
InstalledDir: /Library/Developer/CommandLineTools/usr/bin

工作示例

根据@alexpanter 发布的 repo 查看工作示例的步骤:

main.cpp

import <iostream>;
import mathlib;

using namespace std;

int main() {
    cout << "Modules, baby!" << endl;
    cout << "2 plus 3 makes " << add(2, 3) << " says module 'mathlib'" << endl;
}

mathlib.cpp

export module mathlib;

export int add(int a, int b)
{
    return a + b;
}

通过在与上述文件相同的目录中的终端中运行来构建:

/opt/homebrew/opt/llvm/bin/clang++ -std=c++20 -c -Xclang -emit-module-interface mathlib.cpp -o mathlib.pcm
/opt/homebrew/opt/llvm/bin/clang++ -std=c++20 -fmodules -c -fprebuilt-module-path=. main.cpp -o main.o
/opt/homebrew/opt/llvm/bin/clang++ -std=c++2a -fmodules -o main main.o *.pcm

测试基于模块的可执行文件:

./main

预期输出:

Modules, baby!
2 plus 3 makes 5 says module 'mathlib'