203 lines
5.7 KiB
C++
203 lines
5.7 KiB
C++
#include "Test/Tester.h"
|
|
|
|
namespace clice::testing {
|
|
|
|
namespace {
|
|
|
|
suite<"Directive"> directive = [] {
|
|
Tester tester;
|
|
std::vector<Include> includes;
|
|
std::vector<HasInclude> has_includes;
|
|
std::vector<Condition> conditions;
|
|
std::vector<MacroRef> macros;
|
|
std::vector<Pragma> pragmas;
|
|
|
|
using u32 = std::uint32_t;
|
|
|
|
auto run = [&](llvm::StringRef code) {
|
|
tester.clear();
|
|
tester.add_files("main.cpp", code);
|
|
tester.compile("-std=c++23");
|
|
auto fid = tester.unit->interested_file();
|
|
includes = tester.unit->directives()[fid].includes;
|
|
has_includes = tester.unit->directives()[fid].has_includes;
|
|
conditions = tester.unit->directives()[fid].conditions;
|
|
macros = tester.unit->directives()[fid].macros;
|
|
pragmas = tester.unit->directives()[fid].pragmas;
|
|
};
|
|
|
|
auto expect_include = [&](u32 index, llvm::StringRef position, llvm::StringRef path) {
|
|
auto& include = includes[index];
|
|
auto [_, offset] = tester.unit->decompose_location(include.location);
|
|
expect(that % offset == tester.point(position));
|
|
|
|
/// FIXME: Implicit relative path ...
|
|
llvm::SmallString<64> target = include.skipped ? "" : tester.unit->file_path(include.fid);
|
|
path::remove_dots(target);
|
|
|
|
expect(that % target == path);
|
|
};
|
|
|
|
auto expect_has_inl = [&](u32 index, llvm::StringRef position, llvm::StringRef path) {
|
|
auto& has_include = has_includes[index];
|
|
auto [_, offset] = tester.unit->decompose_location(has_include.location);
|
|
expect(that % offset == tester.point(position));
|
|
|
|
/// FIXME:
|
|
llvm::SmallString<64> target =
|
|
has_include.fid.isValid() ? tester.unit->file_path(has_include.fid) : "";
|
|
path::remove_dots(target);
|
|
|
|
expect(that % target == path);
|
|
};
|
|
|
|
auto expect_con = [&](u32 index, Condition::BranchKind kind, llvm::StringRef pos) {
|
|
auto& condition = conditions[index];
|
|
auto [_, offset] = tester.unit->decompose_location(condition.loc);
|
|
expect(that % int(condition.kind) == int(kind));
|
|
expect(that % offset == tester.point(pos));
|
|
};
|
|
|
|
auto expect_macro = [&](u32 index, MacroRef::Kind kind, llvm::StringRef position) {
|
|
auto& macro = macros[index];
|
|
auto [_, offset] = tester.unit->decompose_location(macro.loc);
|
|
expect(that % int(macro.kind) == int(kind));
|
|
expect(that % offset == tester.point(position));
|
|
};
|
|
|
|
auto expect_pragma =
|
|
[&](u32 index, Pragma::Kind kind, llvm::StringRef pos, llvm::StringRef text) {
|
|
auto& pragma = pragmas[index];
|
|
auto [_, offset] = tester.unit->decompose_location(pragma.loc);
|
|
expect(that % int(pragma.kind) == int(kind));
|
|
expect(that % pragma.stmt == text);
|
|
expect(that % offset == tester.point(pos));
|
|
};
|
|
|
|
test("Include") = [&] {
|
|
run(R"cpp(
|
|
#[test.h]
|
|
|
|
#[pragma_once.h]
|
|
#pragma once
|
|
|
|
#[guard_macro.h]
|
|
#ifndef TEST3_H
|
|
#define TEST3_H
|
|
#endif
|
|
|
|
#[main.cpp]
|
|
#$(0)include "test.h"
|
|
#$(1)include "test.h"
|
|
#$(2)include "pragma_once.h"
|
|
#$(3)include "pragma_once.h"
|
|
#$(4)include "guard_macro.h"
|
|
#$(5)include "guard_macro.h"
|
|
)cpp");
|
|
|
|
expect(that % includes.size() == 6);
|
|
expect_include(0, "0", "test.h");
|
|
expect_include(1, "1", "test.h");
|
|
expect_include(2, "2", "pragma_once.h");
|
|
expect_include(3, "3", "");
|
|
expect_include(4, "4", "guard_macro.h");
|
|
expect_include(5, "5", "");
|
|
|
|
/// TODO: test include source range.
|
|
};
|
|
|
|
test("HasInclude") = [&] {
|
|
run(R"cpp(
|
|
#[test.h]
|
|
|
|
#[main.cpp]
|
|
#include "test.h"
|
|
#if __has_include($(0)"test.h")
|
|
#endif
|
|
|
|
#if __has_include($(1)"test2.h")
|
|
#endif
|
|
)cpp");
|
|
|
|
expect(that % has_includes.size() == 2);
|
|
expect_has_inl(0, "0", "test.h");
|
|
expect_has_inl(1, "1", "");
|
|
};
|
|
|
|
test("Condition") = [&] {
|
|
run(R"cpp(
|
|
#[main.cpp]
|
|
#$(0)if 0
|
|
#$(1)elif 1
|
|
#$(2)else
|
|
#$(3)endif
|
|
|
|
#$(4)ifdef name
|
|
#$(5)elifdef name
|
|
#$(6)else
|
|
#$(7)endif
|
|
)cpp");
|
|
|
|
expect(that % conditions.size() == 8);
|
|
expect_con(0, Condition::BranchKind::If, "0");
|
|
expect_con(1, Condition::BranchKind::Elif, "1");
|
|
expect_con(2, Condition::BranchKind::Else, "2");
|
|
expect_con(3, Condition::BranchKind::EndIf, "3");
|
|
expect_con(4, Condition::BranchKind::Ifdef, "4");
|
|
expect_con(5, Condition::BranchKind::Elifdef, "5");
|
|
expect_con(6, Condition::BranchKind::Else, "6");
|
|
expect_con(7, Condition::BranchKind::EndIf, "7");
|
|
};
|
|
|
|
test("Macro") = [&] {
|
|
run(R"cpp(
|
|
#[main.cpp]
|
|
#define $(0)expr(v) v
|
|
|
|
#ifdef $(1)expr
|
|
int x = $(2)expr(1);
|
|
#endif
|
|
|
|
#undef $(3)expr
|
|
|
|
#define $(4)expr(v) v
|
|
|
|
#ifdef $(5)expr
|
|
int y = $(6)expr($(7)expr(1));
|
|
#endif
|
|
|
|
#undef $(8)expr
|
|
|
|
)cpp");
|
|
|
|
expect(that % macros.size() == 9);
|
|
expect_macro(0, MacroRef::Kind::Def, "0");
|
|
expect_macro(1, MacroRef::Kind::Ref, "1");
|
|
expect_macro(2, MacroRef::Kind::Ref, "2");
|
|
expect_macro(3, MacroRef::Kind::Undef, "3");
|
|
expect_macro(4, MacroRef::Kind::Def, "4");
|
|
expect_macro(5, MacroRef::Kind::Ref, "5");
|
|
expect_macro(6, MacroRef::Kind::Ref, "6");
|
|
expect_macro(7, MacroRef::Kind::Ref, "7");
|
|
expect_macro(8, MacroRef::Kind::Undef, "8");
|
|
};
|
|
|
|
test("Pragma") = [&] {
|
|
run(R"cpp(
|
|
#[main.cpp]
|
|
$(0)#pragma GCC poison printf sprintf fprintf
|
|
$(1)#pragma region
|
|
$(2)#pragma endregion
|
|
)cpp");
|
|
|
|
expect(that % pragmas.size() == 3);
|
|
expect_pragma(0, Pragma::Kind::Other, "0", "#pragma GCC poison printf sprintf fprintf");
|
|
expect_pragma(1, Pragma::Kind::Region, "1", "#pragma region");
|
|
expect_pragma(2, Pragma::Kind::EndRegion, "2", "#pragma endregion");
|
|
};
|
|
};
|
|
|
|
} // namespace
|
|
|
|
} // namespace clice::testing
|