#include "Test/Tester.h" namespace clice::testing { namespace { suite<"Directive"> directive = [] { Tester tester; std::vector includes; std::vector has_includes; std::vector conditions; std::vector macros; std::vector 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