Files
clice/tests/unit/Compiler/Module.cpp
2025-09-07 23:27:39 +08:00

174 lines
4.6 KiB
C++

#include "Test/Test.h"
#include "Compiler/Compilation.h"
#include "llvm/Support/ToolOutputFile.h"
namespace clice::testing {
namespace {
using namespace std::literals;
auto buildPCM = [](llvm::StringRef file, llvm::StringRef code) {
llvm::SmallString<128> outPath;
fs::createUniquePath(llvm::Twine(file) + "%%%%%%.pcm", outPath, true);
std::string path = file.str();
CompilationParams params;
params.output_file = outPath;
params.arguments = {
"clang++",
"-std=c++20",
"-xc++",
path.c_str(),
};
params.add_remapped_file(file, code);
params.add_remapped_file("./test.h", "export int foo2();");
PCMInfo pcm;
if(!compile(params, pcm)) {
llvm::errs() << "Failed to build PCM\n";
std::abort();
}
return pcm;
};
auto scan = [](llvm::StringRef content) {
CompilationParams params;
params.arguments = {
"clang++",
"-std=c++20",
"-xc++",
"main.ixx",
};
params.add_remapped_file("main.ixx", content);
params.add_remapped_file("./test.h", "export module A");
auto info = scanModule(params);
if(!info) {
/// std::println("Fail to scan module: {}", info.error());
std::abort();
}
return std::move(*info);
};
suite<"Module"> module = [] {
test("Scan") = [&] {
/// Simple case.
const char* content = R"(
export module A;
import B;
)";
auto info = scan(content);
expect(that % info.isInterfaceUnit == true);
expect(that % info.name == "A"sv);
expect(that % info.mods.size() == 1);
expect(that % info.mods[0] == "B"sv);
/// FIXME: Fix standard library search path(resource dir).
/// 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);
/// expect(that % info.isInterfaceUnit == true);
/// expect(that % info.name == "A");
/// expect(that % info.mods.size() == 2);
/// expect(that % info.mods[0] == "B");
/// expect(that % 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);
/// expect(that % info.isInterfaceUnit == true);
/// expect(that % info.name == "A:B");
/// expect(that % info.mods.size() == 2);
/// expect(that % info.mods[0] == "B");
/// expect(that % info.mods[1] == "C");
content = R"(
module A;
import B;
import C;
)";
info = scan(content);
expect(that % info.isInterfaceUnit == false);
expect(that % info.name == "A"sv);
expect(that % info.mods.size() == 2);
expect(that % info.mods[0] == "B"sv);
expect(that % info.mods[1] == "C"sv);
};
test("Normal") = [&] {
const char* content = R"(
export module A;
)";
auto pcm = buildPCM("A.ixx", content);
// expect(that % pcm.isInterfaceUnit == true);
// expect(that % pcm.name == "A");
// expect(that % pcm.mods.size() == 0);
};
};
// TEST(Module, ScanModuleName) {
// CompilationParams params;
//
// /// Test module name not in condition directive.
// params.content = "export module A;";
// ASSERT_EQ(scanModuleName(params), "A");
//
// params.content = "export module A.B.C.D;";
// ASSERT_EQ(scanModuleName(params), "A.B.C.D");
//
// params.content = "export module A:B;";
// ASSERT_EQ(scanModuleName(params), "A:B");
//
// params.content = R"(
// module;
// #ifdef TEST
// #include <iostream>
// #endif
// export module A;
//)";
// ASSERT_EQ(scanModuleName(params), "A");
//
// /// Test non-module interface unit.
// params.content = "module A;";
// ASSERT_EQ(scanModuleName(params), "");
//
// params.content = "";
// ASSERT_EQ(scanModuleName(params), "");
//
// /// Test module name in condition directive.
// params.content = R"(
// #ifdef TEST
// export module A;
// #else
// export module B;
// #endif
//)";
// params.srcPath = "main.cppm";
// params.command = "clang++ -std=c++20 -x c++ main.cppm -DTEST";
// ASSERT_EQ(scanModuleName(params), "A");
//
// params.command = "clang++ -std=c++20 -x c++ main.cppm";
// ASSERT_EQ(scanModuleName(params), "B");
// }
} // namespace
} // namespace clice::testing