Как использовать модули C++ в Clang?
Модули являются альтернативой #include. Clang имеет полную реализацию для C++. Как бы я поступил, если бы сейчас хотел использовать модули, использующие Clang?
С помощью
import std.io;
в исходном файле C++ пока не работает (не компилируется), так как спецификация для модулей (которая включает синтаксис) не является окончательной.
Документация Clang гласит, что при прохождении -fmodules
флаг, #include будет переписан для соответствующего импорта. Однако проверка препроцессора предполагает иное (test.cpp содержит только #include <stdio.h>
и пустая основная)
$ clang++-3.5 -fmodules -E test.cpp -o test
$ grep " printf " test
extern int printf (const char *__restrict __format, ...);
Кроме того, компиляция этого тестового файла с -fmodules
vs без флагов вообще создает один и тот же объектный файл.
Что я делаю неправильно?
2 ответа
Как вы упомянули, у clang еще нет синтаксиса C++ для импорта, поэтому я сомневаюсь, что #include
Директивы будут буквально переписаны как импорт при предварительной обработке файла, так что это может быть не лучшим способом проверить, работают ли модули так, как задумано.
Однако, если вы установите -fmodules-cache-path=<path>
в явном виде вы можете наблюдать, как clang заполняет его предварительно скомпилированными файлами модулей (*.pcm) во время сборки - если есть какие-либо задействованные модули.
Вам нужно будет использовать libC++ (который, похоже, поставляется с module.modulemap начиная с версии 3.7.0), если вы хотите использовать стандартную библиотеку с поддержкой модулей прямо сейчас - хотя, по моему опыту, пока это не совсем работает. (Компилятор Visual Studio 2015 C++ также должен получить некоторую поддержку модулей с обновлением 1 в ноябре)
Независимо от stdlib, вы все равно можете использовать модули в своем собственном коде. Документы clang содержат подробное описание языка карты модулей, но я также создал небольшой пример проекта (с использованием cmake), который должен сгенерировать cache
каталог с некоторыми модулями при сборке.
Начиная с этого коммита, Clang имеет экспериментальную поддержку модулей TS.
Давайте возьмем те же файлы примеров (с небольшим изменением), что и в блоге VS о поддержке экспериментального модуля.
Сначала определите файл интерфейса модуля. По умолчанию Clang распознает файлы с cppm
расширение (и некоторые другие) в виде файлов интерфейса модуля C++.
// file: foo.cppm
export module M;
export int f(int x)
{
return 2 + x;
}
export double g(double y, int z)
{
return y * z;
}
Обратите внимание, что объявление интерфейса модуля должно быть export module M;
и не только module M;
как в блоге VS.
Затем используйте модуль следующим образом:
// file: bar.cpp
import M;
int main()
{
f(5);
g(0.0, 1);
return 0;
}
Теперь предварительно скомпилируйте модуль foo.cppm
с
clang++ -fmodules-ts --precompile foo.cppm -o M.pcm
или, если расширение интерфейса модуля отличается от cppm
(скажем ixx
, как и в случае с VS), вы можете использовать:
clang++ -fmodules-ts --precompile -x c++-module foo.ixx -o M.pcm
Затем соберите программу с
clang++ -fmodules-ts -c M.pcm -o M.o
clang++ -fmodules-ts -fprebuilt-module-path=. M.o bar.cpp
или, если имя файла PCM не совпадает с именем модуля, вы должны будете использовать:
clang++ -fmodules-ts -fmodule-file=M.pcm bar.cpp
Я протестировал эти команды в Windows, используя сборку r303050 (15 мая 2017 г.).
Примечание: при использовании -fprebuilt-module-path=.
вариант, я получаю предупреждение:
clang ++. exe: warning: аргумент не используется во время компиляции: '-fprebuilt-module-path=.' [-Wunused-командная строка аргументы]
который представляется неправильным, потому что без этой опции, модуль M
не найден.