247 lines
7.8 KiB
C++
247 lines
7.8 KiB
C++
#include "Test/Test.h"
|
|
#include "Test/Tester.h"
|
|
#include "Index/TUIndex.h"
|
|
|
|
namespace clice::testing {
|
|
namespace {
|
|
|
|
TEST_SUITE(TUIndex) {
|
|
|
|
Tester tester;
|
|
index::TUIndex tu_index;
|
|
|
|
void build_index(llvm::StringRef code,
|
|
std::source_location location = std::source_location::current()) {
|
|
tester.clear();
|
|
tester.add_main("main.cpp", code);
|
|
ASSERT_TRUE(tester.compile());
|
|
|
|
tu_index = index::TUIndex::build(*tester.unit);
|
|
}
|
|
|
|
auto select(llvm::StringRef pos, llvm::StringRef file = "") -> std::vector<index::Occurrence> {
|
|
auto offset = tester.point(pos, file);
|
|
auto fid = file.empty() ? tester.unit->interested_file() : tester.unit->file_id(file);
|
|
auto& index = fid == tester.unit->interested_file() ? tu_index.main_file_index
|
|
: tu_index.file_indices[fid];
|
|
|
|
auto it =
|
|
std::ranges::lower_bound(index.occurrences, offset, {}, [](index::Occurrence& occurrence) {
|
|
return occurrence.range.end;
|
|
});
|
|
|
|
std::vector<index::Occurrence> occurrences;
|
|
while(it != index.occurrences.end()) {
|
|
if(it->range.contains(offset)) {
|
|
occurrences.emplace_back(*it);
|
|
it++;
|
|
continue;
|
|
}
|
|
|
|
break;
|
|
}
|
|
return occurrences;
|
|
}
|
|
|
|
void expect_select(llvm::StringRef pos,
|
|
llvm::StringRef expect_range,
|
|
llvm::StringRef file = "",
|
|
std::source_location location = std::source_location::current()) {
|
|
auto offset = tester.point(pos, file);
|
|
auto range = tester.range(expect_range, file);
|
|
auto occurrences = select(pos, file);
|
|
|
|
ASSERT_FALSE(occurrences.empty());
|
|
/// << std::format("Fail to find symbol for offset: {}, target range: {}", offset, dump(range));
|
|
|
|
/// FIXME: Make eq pretty print reflectable struct.
|
|
ASSERT_EQ(dump(occurrences.front().range), dump(range));
|
|
};
|
|
|
|
void go_to_definition(llvm::StringRef pos,
|
|
llvm::StringRef definition,
|
|
llvm::StringRef file = "",
|
|
std::source_location location = std::source_location::current()) {
|
|
auto offset = tester.point(pos, file);
|
|
auto range = tester.range(definition, file);
|
|
auto occurrences = select(pos, file);
|
|
|
|
ASSERT_EQ(occurrences.size(), 1U);
|
|
/// << std::format("Fail to find symbol for offset: {}, target range: {}", offset, dump(range));
|
|
|
|
auto fid = file.empty() ? tester.unit->interested_file() : tester.unit->file_id(file);
|
|
auto& index = fid == tester.unit->interested_file() ? tu_index.main_file_index
|
|
: tu_index.file_indices[fid];
|
|
|
|
auto it = index.relations.find(occurrences.front().target);
|
|
ASSERT_TRUE(it != index.relations.end());
|
|
///<< std::format("Cannot find target: {}", occurrences.front().target);
|
|
|
|
auto& relations = it->second;
|
|
auto target = std::ranges::find(relations, RelationKind::Definition, &index::Relation::kind);
|
|
|
|
ASSERT_TRUE(target != relations.end());
|
|
/// << std::format("Fail to find definition in {}", dump(relations));
|
|
ASSERT_EQ(dump(target->range), dump(range));
|
|
}
|
|
|
|
TEST_CASE(Basic) {
|
|
build_index(R"(
|
|
int @1[f$(1)oo]();
|
|
|
|
int @2[b$(2)ar]() {
|
|
return @3[fo$(3)o]() + 1;
|
|
}
|
|
)");
|
|
|
|
auto& index = tu_index.main_file_index;
|
|
ASSERT_EQ(index.relations.size(), 2U);
|
|
ASSERT_EQ(index.occurrences.size(), 3U);
|
|
|
|
expect_select("1", "1");
|
|
expect_select("2", "2");
|
|
expect_select("3", "3");
|
|
}
|
|
|
|
TEST_CASE(ClassTemplate) {
|
|
build_index(R"(
|
|
template <typename T, typename U>
|
|
struct $(primary_decl)foo;
|
|
|
|
/// using type = $(forward_full)foo<int, int>;
|
|
|
|
template <typename T, typename U>
|
|
struct @primary[foo] {};
|
|
|
|
template <typename T>
|
|
struct $(partial_spec_decl)foo<T, T>;
|
|
|
|
template <typename T>
|
|
struct @partial_spec[foo]<T, T> {};
|
|
|
|
template <>
|
|
struct $(full_spec_decl)foo<int, int>;
|
|
|
|
template <>
|
|
struct @full_spec[foo]<int, int> {};
|
|
|
|
template struct $(explicit_primary)foo<char, int>;
|
|
|
|
template struct $(explicit_partial)foo<char, char>;
|
|
|
|
$(implicit_primary_1)foo<int, char> b;
|
|
$(implicit_primary_2)foo<char, int> c;
|
|
$(implicit_partial)foo<char, char> d;
|
|
$(implicit_full)foo<int, int> a;
|
|
)");
|
|
|
|
go_to_definition("primary_decl", "primary");
|
|
go_to_definition("explicit_primary", "primary");
|
|
go_to_definition("implicit_primary_1", "primary");
|
|
go_to_definition("implicit_primary_2", "primary");
|
|
go_to_definition("partial_spec_decl", "partial_spec");
|
|
go_to_definition("explicit_partial", "partial_spec");
|
|
go_to_definition("implicit_partial", "partial_spec");
|
|
/// FIXME: Figure forward template declaration.
|
|
/// go_to_definition("forward_full", "full_spec");
|
|
go_to_definition("full_spec_decl", "full_spec");
|
|
go_to_definition("implicit_full", "full_spec");
|
|
}
|
|
|
|
TEST_CASE(FunctionTemplate) {
|
|
build_index(R"(
|
|
template <typename T> void $(primary_decl)foo();
|
|
|
|
template <typename T> void @primary[foo]() {}
|
|
|
|
template <> void $(spec_decl)foo<int>();
|
|
|
|
template <> void @spec[foo]<int>() {}
|
|
|
|
template void $(explicit_primary)foo<char>();
|
|
|
|
int main() {
|
|
$(implicit_primary)foo<char>();
|
|
$(implicit_spec)foo<int>();
|
|
}
|
|
)");
|
|
|
|
go_to_definition("primary_decl", "primary");
|
|
/// FIXME: clang doen't record location info of explicit function instantiation/
|
|
/// See https://github.com/llvm/llvm-project/issues/115418.
|
|
/// go_to_definition("explicit_primary", "primary");
|
|
go_to_definition("implicit_primary", "primary");
|
|
go_to_definition("spec_decl", "spec");
|
|
go_to_definition("implicit_spec", "spec");
|
|
}
|
|
|
|
TEST_CASE(AliasTemplate) {
|
|
build_index(R"(
|
|
template <typename T>
|
|
using @primary[foo] = T;
|
|
|
|
$(implicit_primary)foo<int> a;
|
|
)");
|
|
|
|
go_to_definition("implicit_primary", "primary");
|
|
}
|
|
|
|
TEST_CASE(VarTemplate) {
|
|
build_index(R"(
|
|
template <typename T, typename U>
|
|
extern int $(primary_decl)foo;
|
|
|
|
template <typename T, typename U>
|
|
int @primary[foo] = 1;
|
|
|
|
template <typename T>
|
|
extern int $(partial_spec_decl)foo<T, T>;
|
|
|
|
template <typename T>
|
|
int @partial_spec[foo]<T, T> = 2;
|
|
|
|
template <>
|
|
float @full_spec[foo]<int, int> = 1.0f;
|
|
|
|
template int $(explicit_primary)foo<char, int>;
|
|
|
|
template int $(explicit_partial)foo<char, char>;
|
|
|
|
int main() {
|
|
$(implicit_primary_1)foo<int, char> = 1;
|
|
$(implicit_primary_2)foo<char, int> = 2;
|
|
$(implicit_partial)foo<char, char> = 3;
|
|
$(implicit_full)foo<int, int> = 4;
|
|
return 0;
|
|
}
|
|
)");
|
|
|
|
go_to_definition("primary_decl", "primary");
|
|
/// go_to_definition("explicit_primary", "primary");
|
|
go_to_definition("implicit_primary_1", "primary");
|
|
go_to_definition("implicit_primary_2", "primary");
|
|
go_to_definition("partial_spec_decl", "partial_spec");
|
|
/// tester.GotoDefinition("explicit_partial", "partial_spec");
|
|
go_to_definition("implicit_partial", "partial_spec");
|
|
go_to_definition("implicit_full", "full_spec");
|
|
}
|
|
|
|
TEST_CASE(Concept) {
|
|
build_index(R"(
|
|
template <typename T>
|
|
concept @primary[$(primary)foo] = true;
|
|
|
|
static_assert($(implicit)foo<int>);
|
|
|
|
$(implicit2)foo auto bar = 1;
|
|
)");
|
|
|
|
go_to_definition("primary", "primary");
|
|
go_to_definition("implicit", "primary");
|
|
go_to_definition("implicit2", "primary");
|
|
}
|
|
|
|
}; // TEST_SUITE(TUIndex)
|
|
} // namespace
|
|
} // namespace clice::testing
|