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

277 lines
7.2 KiB
C++

#include "Test/Test.h"
#include "Compiler/Preamble.h"
#include "Compiler/Compilation.h"
#include "Test/Annotation.h"
namespace clice::testing {
namespace {
suite<"Preamble"> preamble = [] {
auto expect_bounds = [](std::vector<llvm::StringRef> marks, llvm::StringRef content) {
auto annotation = AnnotatedSource::from(content);
auto bounds = compute_preamble_bounds(annotation.content);
expect(that % bounds.size() == marks.size());
for(std::uint32_t i = 0; i < bounds.size(); i++) {
expect(that % bounds[i] == annotation.offsets[marks[i]]);
}
};
auto expect_build_pch = [](llvm::StringRef main_file,
llvm::StringRef test_contents,
llvm::StringRef preamble = "") {
auto tmp = fs::createTemporaryFile("clice", "pch");
expect(that % tmp);
std::string output_path = std::move(*tmp);
AnnotatedSources sources;
sources.add_sources(test_contents);
auto& files = sources.all_files;
if(!preamble.empty()) {
files.try_emplace("preamble.h", AnnotatedSource{.content = preamble.str()});
}
expect(that % files.contains(main_file));
std::string content = files[main_file].content;
files.erase(main_file);
CompilationParams params;
params.output_file = output_path;
auto bound = compute_preamble_bound(content);
params.add_remapped_file(main_file, content, bound);
params.arguments = {
"clang++",
"-xc++",
"-std=c++20",
};
if(!preamble.empty()) {
params.arguments.emplace_back("--include=preamble.h");
}
std::string buffer = main_file.str();
params.arguments.emplace_back(buffer.c_str());
for(auto& [path, source]: files) {
params.add_remapped_file(path::join(".", path), source.content);
}
/// Build PCH.
PCHInfo info;
{
/// NOTE: PCH file is written when CompilerInstance is destructed.
auto unit = compile(params, info);
expect(that % unit.has_value());
expect(that % info.path == output_path);
/// expect(that % info.command == params.arguments);
/// TODO: expect(that % info.deps == deps);
}
/// Build AST with PCH.
for(auto& [path, source]: files) {
params.add_remapped_file(path::join(".", path), source.content);
}
params.add_remapped_file(main_file, content);
params.pch = {info.path, info.preamble.size()};
auto unit = compile(params);
expect(that % unit.has_value());
};
test("Bounds") = [&] {
expect_bounds({}, "int main(){}");
expect_bounds({"0"}, "#include <iostream>$(0)");
expect_bounds({"0"}, "#include <iostream>$(0)\n");
expect_bounds({"0", "1", "2", "3"},
R"cpp(
#ifdef TEST$(0)
#include <iostream>$(1)
#define 1$(2)
#endif$(3)
)cpp");
expect_bounds({"0"},
R"cpp(
#include <iostream>$(0)
int x = 1;
)cpp");
expect_bounds({"0", "1"}, R"cpp(
module;$(0)
#include <iostream>$(1)
export module test;
)cpp");
};
test("TranslationUnit") = [&] {
expect_build_pch("main.cpp",
R"cpp(
#[test.h]
int foo();
#[main.cpp]
#include "test.h"
int x = foo();
)cpp");
};
test("Module") = [&] {
expect_build_pch("main.cpp",
R"cpp(
#[test.h]
int foo();
#[main.cpp]
module;
#include "test.h"
export module test;
export int x = foo();
)cpp");
};
test("Header") = [&] {
llvm::StringRef test_contents = R"cpp(
#[test.h]
int bar();
#[test1.h]
#include "test.h"
Point x = {foo(), bar()};
#[test2.h]
struct Point {
int x;
int y;
};
#include "test1.h"
#[test3.h]
int foo();
#[main.cpp]
#include "test3.h"
#include "test2.h"
)cpp";
AnnotatedSources sources;
sources.add_sources(test_contents);
auto& files = sources.all_files;
expect(that % files.contains("main.cpp"));
std::string content = files["main.cpp"].content;
files.erase("main.cpp");
std::string preamble;
/// Compute implicit include.
{
CompilationParams params;
params.add_remapped_file("main.cpp", content);
params.arguments = {"clang++", "-std=c++20", "main.cpp"};
for(auto& [path, source]: files) {
params.add_remapped_file(path::join(".", path), source.content);
}
auto unit = preprocess(params);
expect(that % unit.has_value());
auto path = path::join(".", "test1.h");
auto fid = unit->file_id(path);
expect(that % fid.isValid());
while(fid.isValid()) {
auto location = unit->include_location(fid);
auto [fid2, offset] = unit->decompose_location(location);
auto content = unit->file_content(fid2).substr(0, offset);
/// Remove incomplete include.
content = content.substr(0, content.rfind("\n"));
preamble += content;
fid = fid2;
}
}
expect_build_pch("test1.h", test_contents, preamble);
};
test("Chain") = [&] {
llvm::StringRef test_contents = R"cpp(
#[test.h]
int bar();
#[test2.h]
int foo();
#[main.cpp]
#include "test.h"
#include "test2.h"
int x = bar();
int y = foo();
)cpp";
AnnotatedSources sources;
sources.add_sources(test_contents);
auto& files = sources.all_files;
expect(that % files.contains("main.cpp"));
std::string content = files["main.cpp"].content;
files.erase("main.cpp");
auto bounds = compute_preamble_bounds(content);
CompilationParams params;
params.arguments = {"clang++", "-std=c++20", "main.cpp"};
PCHInfo info;
std::uint32_t last_bound = 0;
for(auto bound: bounds) {
auto tmp = fs::createTemporaryFile("clice", "pch");
expect(that % tmp);
std::string outPath = std::move(*tmp);
params.add_remapped_file("main.cpp", content, bound);
if(params.output_file.empty()) {
params.pch = {params.output_file.str().str(), last_bound};
}
params.output_file = outPath;
last_bound = bound;
for(auto& [path, source]: files) {
params.add_remapped_file(path::join(".", path), source.content);
}
{
auto unit = compile(params, info);
expect(that % unit.has_value());
expect(that % info.path == outPath);
/// expect(that % info.command == params.arguments);
}
}
/// Build AST with PCH.
for(auto& [path, source]: files) {
params.add_remapped_file(path::join(".", path), source.content);
}
params.add_remapped_file("main.cpp", content);
params.pch = {info.path, last_bound};
auto unit = compile(params);
expect(that % unit.has_value());
};
};
} // namespace
} // namespace clice::testing