Basic support C++20 named module. (#12)

This commit is contained in:
ykiko
2024-12-14 13:40:13 +08:00
committed by GitHub
parent 7d42b4e1ec
commit 5326480cd6
30 changed files with 1082 additions and 593 deletions

View File

@@ -0,0 +1,113 @@
#include "gtest/gtest.h"
#include "Compiler/Compiler.h"
#include "llvm/Support/ToolOutputFile.h"
namespace clice {
namespace {
PCMInfo buildPCM(llvm::StringRef file, llvm::StringRef code) {
llvm::SmallString<128> outPath;
fs::createUniquePath(llvm::Twine(file) + "%%%%%%.pcm", outPath, true);
CompilationParams params;
params.content = code;
params.srcPath = file;
params.outPath = outPath;
params.command = "clang++ -std=c++20 -x c++ " + file.str();
params.remappedFiles.emplace_back("./test.h", "export int foo2();");
PCMInfo pcm;
if(!compile(params, pcm)) {
llvm::errs() << "Failed to build PCM\n";
std::terminate();
}
return pcm;
}
ModuleInfo scan(llvm::StringRef content) {
CompilationParams params;
params.content = content;
params.srcPath = "main.ixx";
params.command = "clang++ -std=c++20 -x c++ main.ixx";
params.remappedFiles.emplace_back("./test.h", "export module A");
auto info = scanModule(params);
if(!info) {
llvm::errs() << "Failed to scan module\n";
std::terminate();
}
return std::move(*info);
}
TEST(Module, Scan) {
/// Simple case.
const char* content = R"(
export module A;
import B;
)";
auto info = scan(content);
ASSERT_EQ(info.isInterfaceUnit, true);
ASSERT_EQ(info.name, "A");
ASSERT_EQ(info.mods.size(), 1);
ASSERT_EQ(info.mods[0], "B");
/// With global module fragment and private module fragment.
content = R"(
module;
#include <iostream>
export module A;
import B;
import C;
module : private;
)";
info = scan(content);
ASSERT_EQ(info.isInterfaceUnit, true);
ASSERT_EQ(info.name, "A");
ASSERT_EQ(info.mods.size(), 2);
ASSERT_EQ(info.mods[0], "B");
ASSERT_EQ(info.mods[1], "C");
/// With module partition.
content = R"(
module;
#include <iostream>
export module A:B;
import B;
import C;
module : private;
)";
info = scan(content);
ASSERT_EQ(info.isInterfaceUnit, true);
ASSERT_EQ(info.name, "A:B");
ASSERT_EQ(info.mods.size(), 2);
ASSERT_EQ(info.mods[0], "B");
ASSERT_EQ(info.mods[1], "C");
content = R"(
module A;
import B;
import C;
)";
info = scan(content);
ASSERT_EQ(info.isInterfaceUnit, false);
ASSERT_EQ(info.name, "A");
ASSERT_EQ(info.mods.size(), 2);
ASSERT_EQ(info.mods[0], "B");
ASSERT_EQ(info.mods[1], "C");
}
TEST(Module, Normal) {
const char* content = R"(
export module A;
)";
auto pcm = buildPCM("A.ixx", content);
// ASSERT_EQ(pcm.isInterfaceUnit, true);
// ASSERT_EQ(pcm.name, "A");
// ASSERT_EQ(pcm.mods.size(), 0);
}
} // namespace
} // namespace clice