forked from OSchip/llvm-project
1445 lines
44 KiB
C++
1445 lines
44 KiB
C++
//===-- HoverTests.cpp ----------------------------------------------------===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "Annotations.h"
|
|
#include "Hover.h"
|
|
#include "TestIndex.h"
|
|
#include "TestTU.h"
|
|
#include "index/MemIndex.h"
|
|
#include "clang/Index/IndexSymbol.h"
|
|
#include "llvm/ADT/None.h"
|
|
#include "llvm/ADT/StringRef.h"
|
|
|
|
#include "gmock/gmock.h"
|
|
#include "gtest/gtest.h"
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
namespace clang {
|
|
namespace clangd {
|
|
namespace {
|
|
|
|
TEST(Hover, Structured) {
|
|
struct {
|
|
const char *const Code;
|
|
const std::function<void(HoverInfo &)> ExpectedBuilder;
|
|
} Cases[] = {
|
|
// Global scope.
|
|
{R"cpp(
|
|
// Best foo ever.
|
|
void [[fo^o]]() {}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.NamespaceScope = "";
|
|
HI.Name = "foo";
|
|
HI.Kind = index::SymbolKind::Function;
|
|
HI.Documentation = "Best foo ever.";
|
|
HI.Definition = "void foo()";
|
|
HI.ReturnType = "void";
|
|
HI.Type = "void ()";
|
|
HI.Parameters.emplace();
|
|
}},
|
|
// Inside namespace
|
|
{R"cpp(
|
|
namespace ns1 { namespace ns2 {
|
|
/// Best foo ever.
|
|
void [[fo^o]]() {}
|
|
}}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.NamespaceScope = "ns1::ns2::";
|
|
HI.Name = "foo";
|
|
HI.Kind = index::SymbolKind::Function;
|
|
HI.Documentation = "Best foo ever.";
|
|
HI.Definition = "void foo()";
|
|
HI.ReturnType = "void";
|
|
HI.Type = "void ()";
|
|
HI.Parameters.emplace();
|
|
}},
|
|
// Field
|
|
{R"cpp(
|
|
namespace ns1 { namespace ns2 {
|
|
struct Foo {
|
|
int [[b^ar]];
|
|
};
|
|
}}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.NamespaceScope = "ns1::ns2::";
|
|
HI.LocalScope = "Foo::";
|
|
HI.Name = "bar";
|
|
HI.Kind = index::SymbolKind::Field;
|
|
HI.Definition = "int bar";
|
|
HI.Type = "int";
|
|
}},
|
|
// Local to class method.
|
|
{R"cpp(
|
|
namespace ns1 { namespace ns2 {
|
|
struct Foo {
|
|
void foo() {
|
|
int [[b^ar]];
|
|
}
|
|
};
|
|
}}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.NamespaceScope = "ns1::ns2::";
|
|
HI.LocalScope = "Foo::foo::";
|
|
HI.Name = "bar";
|
|
HI.Kind = index::SymbolKind::Variable;
|
|
HI.Definition = "int bar";
|
|
HI.Type = "int";
|
|
}},
|
|
// Anon namespace and local scope.
|
|
{R"cpp(
|
|
namespace ns1 { namespace {
|
|
struct {
|
|
int [[b^ar]];
|
|
} T;
|
|
}}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.NamespaceScope = "ns1::(anonymous)::";
|
|
HI.LocalScope = "(anonymous struct)::";
|
|
HI.Name = "bar";
|
|
HI.Kind = index::SymbolKind::Field;
|
|
HI.Definition = "int bar";
|
|
HI.Type = "int";
|
|
}},
|
|
// Variable with template type
|
|
{R"cpp(
|
|
template <typename T, class... Ts> class Foo { public: Foo(int); };
|
|
Foo<int, char, bool> [[fo^o]] = Foo<int, char, bool>(5);
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.NamespaceScope = "";
|
|
HI.Name = "foo";
|
|
HI.Kind = index::SymbolKind::Variable;
|
|
HI.Definition = "Foo<int, char, bool> foo = Foo<int, char, bool>(5)";
|
|
HI.Type = "Foo<int, char, bool>";
|
|
}},
|
|
// Implicit template instantiation
|
|
{R"cpp(
|
|
template <typename T> class vector{};
|
|
[[vec^tor]]<int> foo;
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.NamespaceScope = "";
|
|
HI.Name = "vector";
|
|
HI.Kind = index::SymbolKind::Class;
|
|
HI.Definition = "template <typename T> class vector {}";
|
|
HI.TemplateParameters = {
|
|
{std::string("typename"), std::string("T"), llvm::None},
|
|
};
|
|
}},
|
|
// Class template
|
|
{R"cpp(
|
|
template <template<typename, bool...> class C,
|
|
typename = char,
|
|
int = 0,
|
|
bool Q = false,
|
|
class... Ts> class Foo {};
|
|
template <template<typename, bool...> class T>
|
|
[[F^oo]]<T> foo;
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.NamespaceScope = "";
|
|
HI.Name = "Foo";
|
|
HI.Kind = index::SymbolKind::Class;
|
|
HI.Definition =
|
|
R"cpp(template <template <typename, bool...> class C, typename = char, int = 0,
|
|
bool Q = false, class... Ts>
|
|
class Foo {})cpp";
|
|
HI.TemplateParameters = {
|
|
{std::string("template <typename, bool...> class"),
|
|
std::string("C"), llvm::None},
|
|
{std::string("typename"), llvm::None, std::string("char")},
|
|
{std::string("int"), llvm::None, std::string("0")},
|
|
{std::string("bool"), std::string("Q"), std::string("false")},
|
|
{std::string("class..."), std::string("Ts"), llvm::None},
|
|
};
|
|
}},
|
|
// Function template
|
|
{R"cpp(
|
|
template <template<typename, bool...> class C,
|
|
typename = char,
|
|
int = 0,
|
|
bool Q = false,
|
|
class... Ts> void foo();
|
|
template<typename, bool...> class Foo;
|
|
|
|
void bar() {
|
|
[[fo^o]]<Foo>();
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.NamespaceScope = "";
|
|
HI.Name = "foo";
|
|
HI.Kind = index::SymbolKind::Function;
|
|
HI.Definition =
|
|
R"cpp(template <template <typename, bool...> class C, typename = char, int = 0,
|
|
bool Q = false, class... Ts>
|
|
void foo())cpp";
|
|
HI.ReturnType = "void";
|
|
HI.Type = "void ()";
|
|
HI.Parameters.emplace();
|
|
HI.TemplateParameters = {
|
|
{std::string("template <typename, bool...> class"),
|
|
std::string("C"), llvm::None},
|
|
{std::string("typename"), llvm::None, std::string("char")},
|
|
{std::string("int"), llvm::None, std::string("0")},
|
|
{std::string("bool"), std::string("Q"), std::string("false")},
|
|
{std::string("class..."), std::string("Ts"), llvm::None},
|
|
};
|
|
}},
|
|
// Function decl
|
|
{R"cpp(
|
|
template<typename, bool...> class Foo {};
|
|
Foo<bool, true, false> foo(int, bool T = false);
|
|
|
|
void bar() {
|
|
[[fo^o]](3);
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.NamespaceScope = "";
|
|
HI.Name = "foo";
|
|
HI.Kind = index::SymbolKind::Function;
|
|
HI.Definition = "Foo<bool, true, false> foo(int, bool T = false)";
|
|
HI.ReturnType = "Foo<bool, true, false>";
|
|
HI.Type = "Foo<bool, true, false> (int, bool)";
|
|
HI.Parameters = {
|
|
{std::string("int"), llvm::None, llvm::None},
|
|
{std::string("bool"), std::string("T"), std::string("false")},
|
|
};
|
|
}},
|
|
// Pointers to lambdas
|
|
{R"cpp(
|
|
void foo() {
|
|
auto lamb = [](int T, bool B) -> bool { return T && B; };
|
|
auto *b = &lamb;
|
|
auto *[[^c]] = &b;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.NamespaceScope = "";
|
|
HI.LocalScope = "foo::";
|
|
HI.Name = "c";
|
|
HI.Kind = index::SymbolKind::Variable;
|
|
HI.Definition = "auto *c = &b";
|
|
HI.Type = "class (lambda) **";
|
|
HI.ReturnType = "bool";
|
|
HI.Parameters = {
|
|
{std::string("int"), std::string("T"), llvm::None},
|
|
{std::string("bool"), std::string("B"), llvm::None},
|
|
};
|
|
return HI;
|
|
}},
|
|
// Lambda parameter with decltype reference
|
|
{R"cpp(
|
|
auto lamb = [](int T, bool B) -> bool { return T && B; };
|
|
void foo(decltype(lamb)& bar) {
|
|
[[ba^r]](0, false);
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.NamespaceScope = "";
|
|
HI.LocalScope = "foo::";
|
|
HI.Name = "bar";
|
|
HI.Kind = index::SymbolKind::Parameter;
|
|
HI.Definition = "decltype(lamb) &bar";
|
|
HI.Type = "decltype(lamb) &";
|
|
HI.ReturnType = "bool";
|
|
HI.Parameters = {
|
|
{std::string("int"), std::string("T"), llvm::None},
|
|
{std::string("bool"), std::string("B"), llvm::None},
|
|
};
|
|
return HI;
|
|
}},
|
|
// Lambda parameter with decltype
|
|
{R"cpp(
|
|
auto lamb = [](int T, bool B) -> bool { return T && B; };
|
|
void foo(decltype(lamb) bar) {
|
|
[[ba^r]](0, false);
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.NamespaceScope = "";
|
|
HI.LocalScope = "foo::";
|
|
HI.Name = "bar";
|
|
HI.Kind = index::SymbolKind::Parameter;
|
|
HI.Definition = "decltype(lamb) bar";
|
|
HI.Type = "class (lambda)";
|
|
HI.ReturnType = "bool";
|
|
HI.Parameters = {
|
|
{std::string("int"), std::string("T"), llvm::None},
|
|
{std::string("bool"), std::string("B"), llvm::None},
|
|
};
|
|
HI.Value = "false";
|
|
return HI;
|
|
}},
|
|
// Lambda variable
|
|
{R"cpp(
|
|
void foo() {
|
|
int bar = 5;
|
|
auto lamb = [&bar](int T, bool B) -> bool { return T && B && bar; };
|
|
bool res = [[lam^b]](bar, false);
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.NamespaceScope = "";
|
|
HI.LocalScope = "foo::";
|
|
HI.Name = "lamb";
|
|
HI.Kind = index::SymbolKind::Variable;
|
|
HI.Definition = "auto lamb = [&bar](int T, bool B) -> bool {}";
|
|
HI.Type = "class (lambda)";
|
|
HI.ReturnType = "bool";
|
|
HI.Parameters = {
|
|
{std::string("int"), std::string("T"), llvm::None},
|
|
{std::string("bool"), std::string("B"), llvm::None},
|
|
};
|
|
return HI;
|
|
}},
|
|
// Local variable in lambda
|
|
{R"cpp(
|
|
void foo() {
|
|
auto lamb = []{int [[te^st]];};
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.NamespaceScope = "";
|
|
HI.LocalScope = "foo::(anonymous class)::operator()::";
|
|
HI.Name = "test";
|
|
HI.Kind = index::SymbolKind::Variable;
|
|
HI.Definition = "int test";
|
|
HI.Type = "int";
|
|
}},
|
|
// Partially-specialized class template. (formerly type-parameter-0-0)
|
|
{R"cpp(
|
|
template <typename T> class X;
|
|
template <typename T> class [[^X]]<T*> {};
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "X<T *>";
|
|
HI.NamespaceScope = "";
|
|
HI.Kind = index::SymbolKind::Class;
|
|
HI.Definition = "template <typename T> class X<T *> {}";
|
|
}},
|
|
// Constructor of partially-specialized class template
|
|
{R"cpp(
|
|
template<typename, typename=void> struct X;
|
|
template<typename T> struct X<T*>{ [[^X]](); };
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.NamespaceScope = "";
|
|
HI.Name = "X";
|
|
HI.LocalScope = "X<T *>::"; // FIXME: X<T *, void>::
|
|
HI.Kind = index::SymbolKind::Constructor;
|
|
HI.ReturnType = "X<T *>";
|
|
HI.Definition = "X()";
|
|
HI.Parameters.emplace();
|
|
}},
|
|
{"class X { [[^~]]X(); };", // FIXME: Should be [[~X]]()
|
|
[](HoverInfo &HI) {
|
|
HI.NamespaceScope = "";
|
|
HI.Name = "~X";
|
|
HI.LocalScope = "X::";
|
|
HI.Kind = index::SymbolKind::Destructor;
|
|
HI.ReturnType = "void";
|
|
HI.Definition = "~X()";
|
|
HI.Parameters.emplace();
|
|
}},
|
|
|
|
// auto on lambda
|
|
{R"cpp(
|
|
void foo() {
|
|
[[au^to]] lamb = []{};
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "class (lambda)";
|
|
HI.Kind = index::SymbolKind::Class;
|
|
}},
|
|
// auto on template instantiation
|
|
{R"cpp(
|
|
template<typename T> class Foo{};
|
|
void foo() {
|
|
[[au^to]] x = Foo<int>();
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "class Foo<int>";
|
|
HI.Kind = index::SymbolKind::Class;
|
|
}},
|
|
// auto on specialized template
|
|
{R"cpp(
|
|
template<typename T> class Foo{};
|
|
template<> class Foo<int>{};
|
|
void foo() {
|
|
[[au^to]] x = Foo<int>();
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "class Foo<int>";
|
|
HI.Kind = index::SymbolKind::Class;
|
|
}},
|
|
|
|
// macro
|
|
{R"cpp(
|
|
// Best MACRO ever.
|
|
#define MACRO(x,y,z) void foo(x, y, z);
|
|
[[MAC^RO]](int, double d, bool z = false);
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "MACRO", HI.Kind = index::SymbolKind::Macro,
|
|
HI.Definition = "#define MACRO(x, y, z) void foo(x, y, z);";
|
|
}},
|
|
|
|
// constexprs
|
|
{R"cpp(
|
|
constexpr int add(int a, int b) { return a + b; }
|
|
int [[b^ar]] = add(1, 2);
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "bar";
|
|
HI.Definition = "int bar = add(1, 2)";
|
|
HI.Kind = index::SymbolKind::Variable;
|
|
HI.Type = "int";
|
|
HI.NamespaceScope = "";
|
|
HI.Value = "3";
|
|
}},
|
|
{R"cpp(
|
|
int [[b^ar]] = sizeof(char);
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "bar";
|
|
HI.Definition = "int bar = sizeof(char)";
|
|
HI.Kind = index::SymbolKind::Variable;
|
|
HI.Type = "int";
|
|
HI.NamespaceScope = "";
|
|
HI.Value = "1";
|
|
}},
|
|
{R"cpp(
|
|
template<int a, int b> struct Add {
|
|
static constexpr int result = a + b;
|
|
};
|
|
int [[ba^r]] = Add<1, 2>::result;
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "bar";
|
|
HI.Definition = "int bar = Add<1, 2>::result";
|
|
HI.Kind = index::SymbolKind::Variable;
|
|
HI.Type = "int";
|
|
HI.NamespaceScope = "";
|
|
HI.Value = "3";
|
|
}},
|
|
{R"cpp(
|
|
enum Color { RED, GREEN, };
|
|
Color x = [[GR^EEN]];
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "GREEN";
|
|
HI.NamespaceScope = "";
|
|
HI.LocalScope = "Color::";
|
|
HI.Definition = "GREEN";
|
|
HI.Kind = index::SymbolKind::EnumConstant;
|
|
HI.Type = "enum Color";
|
|
HI.Value = "1"; // Numeric when hovering on the enumerator name.
|
|
}},
|
|
{R"cpp(
|
|
enum Color { RED, GREEN, };
|
|
Color x = GREEN;
|
|
Color y = [[^x]];
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "x";
|
|
HI.NamespaceScope = "";
|
|
HI.Definition = "enum Color x = GREEN";
|
|
HI.Kind = index::SymbolKind::Variable;
|
|
HI.Type = "enum Color";
|
|
HI.Value = "GREEN (1)"; // Symbolic when hovering on an expression.
|
|
}},
|
|
// FIXME: We should use the Decl referenced, even if from an implicit
|
|
// instantiation. Then the scope would be Add<1, 2>.
|
|
{R"cpp(
|
|
template<int a, int b> struct Add {
|
|
static constexpr int result = a + b;
|
|
};
|
|
int bar = Add<1, 2>::[[resu^lt]];
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "result";
|
|
HI.Definition = "static constexpr int result = a + b";
|
|
HI.Kind = index::SymbolKind::StaticProperty;
|
|
HI.Type = "const int";
|
|
HI.NamespaceScope = "";
|
|
HI.LocalScope = "Add<a, b>::";
|
|
HI.Value = "3";
|
|
}},
|
|
{R"cpp(
|
|
constexpr int answer() { return 40 + 2; }
|
|
int x = [[ans^wer]]();
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "answer";
|
|
HI.Definition = "constexpr int answer()";
|
|
HI.Kind = index::SymbolKind::Function;
|
|
HI.Type = "int ()";
|
|
HI.ReturnType = "int";
|
|
HI.Parameters.emplace();
|
|
HI.NamespaceScope = "";
|
|
HI.Value = "42";
|
|
}},
|
|
{R"cpp(
|
|
const char *[[ba^r]] = "1234";
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "bar";
|
|
HI.Definition = "const char *bar = \"1234\"";
|
|
HI.Kind = index::SymbolKind::Variable;
|
|
HI.Type = "const char *";
|
|
HI.NamespaceScope = "";
|
|
HI.Value = "&\"1234\"[0]";
|
|
}},
|
|
{R"cpp(// Should not crash
|
|
template <typename T>
|
|
struct Tmpl {
|
|
Tmpl(int name);
|
|
};
|
|
|
|
template <typename A>
|
|
void boom(int name) {
|
|
new Tmpl<A>([[na^me]]);
|
|
})cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "name";
|
|
HI.Definition = "int name";
|
|
HI.Kind = index::SymbolKind::Parameter;
|
|
HI.Type = "int";
|
|
HI.NamespaceScope = "";
|
|
HI.LocalScope = "boom::";
|
|
}},
|
|
};
|
|
for (const auto &Case : Cases) {
|
|
SCOPED_TRACE(Case.Code);
|
|
|
|
Annotations T(Case.Code);
|
|
TestTU TU = TestTU::withCode(T.code());
|
|
TU.ExtraArgs.push_back("-std=c++17");
|
|
TU.ExtraArgs.push_back("-fno-delayed-template-parsing");
|
|
auto AST = TU.build();
|
|
ASSERT_TRUE(AST.getDiagnostics().empty());
|
|
|
|
auto H = getHover(AST, T.point(), format::getLLVMStyle(), nullptr);
|
|
ASSERT_TRUE(H);
|
|
HoverInfo Expected;
|
|
Expected.SymRange = T.range();
|
|
Case.ExpectedBuilder(Expected);
|
|
|
|
EXPECT_EQ(H->NamespaceScope, Expected.NamespaceScope);
|
|
EXPECT_EQ(H->LocalScope, Expected.LocalScope);
|
|
EXPECT_EQ(H->Name, Expected.Name);
|
|
EXPECT_EQ(H->Kind, Expected.Kind);
|
|
EXPECT_EQ(H->Documentation, Expected.Documentation);
|
|
EXPECT_EQ(H->Definition, Expected.Definition);
|
|
EXPECT_EQ(H->Type, Expected.Type);
|
|
EXPECT_EQ(H->ReturnType, Expected.ReturnType);
|
|
EXPECT_EQ(H->Parameters, Expected.Parameters);
|
|
EXPECT_EQ(H->TemplateParameters, Expected.TemplateParameters);
|
|
EXPECT_EQ(H->SymRange, Expected.SymRange);
|
|
EXPECT_EQ(H->Value, Expected.Value);
|
|
}
|
|
}
|
|
|
|
TEST(Hover, NoHover) {
|
|
llvm::StringRef Tests[] = {
|
|
"^int main() {}",
|
|
"void foo() {^}",
|
|
R"cpp(// structured binding. Not supported yet
|
|
struct Bar {};
|
|
void foo() {
|
|
Bar a[2];
|
|
^auto [x,y] = a;
|
|
}
|
|
)cpp",
|
|
R"cpp(// Template auto parameter. Nothing (Not useful).
|
|
template<a^uto T>
|
|
void func() {
|
|
}
|
|
void foo() {
|
|
func<1>();
|
|
}
|
|
)cpp",
|
|
};
|
|
|
|
for (const auto &Test : Tests) {
|
|
SCOPED_TRACE(Test);
|
|
|
|
Annotations T(Test);
|
|
TestTU TU = TestTU::withCode(T.code());
|
|
TU.ExtraArgs.push_back("-std=c++17");
|
|
auto AST = TU.build();
|
|
ASSERT_TRUE(AST.getDiagnostics().empty());
|
|
|
|
auto H = getHover(AST, T.point(), format::getLLVMStyle(), nullptr);
|
|
ASSERT_FALSE(H);
|
|
}
|
|
}
|
|
|
|
TEST(Hover, All) {
|
|
struct {
|
|
const char *const Code;
|
|
const std::function<void(HoverInfo &)> ExpectedBuilder;
|
|
} Cases[] = {
|
|
{
|
|
R"cpp(// Local variable
|
|
int main() {
|
|
int bonjour;
|
|
^[[bonjour]] = 2;
|
|
int test1 = bonjour;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "bonjour";
|
|
HI.Kind = index::SymbolKind::Variable;
|
|
HI.NamespaceScope = "";
|
|
HI.LocalScope = "main::";
|
|
HI.Type = "int";
|
|
HI.Definition = "int bonjour";
|
|
}},
|
|
{
|
|
R"cpp(// Local variable in method
|
|
struct s {
|
|
void method() {
|
|
int bonjour;
|
|
^[[bonjour]] = 2;
|
|
}
|
|
};
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "bonjour";
|
|
HI.Kind = index::SymbolKind::Variable;
|
|
HI.NamespaceScope = "";
|
|
HI.LocalScope = "s::method::";
|
|
HI.Type = "int";
|
|
HI.Definition = "int bonjour";
|
|
}},
|
|
{
|
|
R"cpp(// Struct
|
|
namespace ns1 {
|
|
struct MyClass {};
|
|
} // namespace ns1
|
|
int main() {
|
|
ns1::[[My^Class]]* Params;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "MyClass";
|
|
HI.Kind = index::SymbolKind::Struct;
|
|
HI.NamespaceScope = "ns1::";
|
|
HI.Definition = "struct MyClass {}";
|
|
}},
|
|
{
|
|
R"cpp(// Class
|
|
namespace ns1 {
|
|
class MyClass {};
|
|
} // namespace ns1
|
|
int main() {
|
|
ns1::[[My^Class]]* Params;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "MyClass";
|
|
HI.Kind = index::SymbolKind::Class;
|
|
HI.NamespaceScope = "ns1::";
|
|
HI.Definition = "class MyClass {}";
|
|
}},
|
|
{
|
|
R"cpp(// Union
|
|
namespace ns1 {
|
|
union MyUnion { int x; int y; };
|
|
} // namespace ns1
|
|
int main() {
|
|
ns1::[[My^Union]] Params;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "MyUnion";
|
|
HI.Kind = index::SymbolKind::Union;
|
|
HI.NamespaceScope = "ns1::";
|
|
HI.Definition = "union MyUnion {}";
|
|
}},
|
|
{
|
|
R"cpp(// Function definition via pointer
|
|
void foo(int) {}
|
|
int main() {
|
|
auto *X = &^[[foo]];
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "foo";
|
|
HI.Kind = index::SymbolKind::Function;
|
|
HI.NamespaceScope = "";
|
|
HI.Type = "void (int)";
|
|
HI.Definition = "void foo(int)";
|
|
HI.Documentation = "Function definition via pointer";
|
|
HI.ReturnType = "void";
|
|
HI.Parameters = {
|
|
{std::string("int"), llvm::None, llvm::None},
|
|
};
|
|
}},
|
|
{
|
|
R"cpp(// Function declaration via call
|
|
int foo(int);
|
|
int main() {
|
|
return ^[[foo]](42);
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "foo";
|
|
HI.Kind = index::SymbolKind::Function;
|
|
HI.NamespaceScope = "";
|
|
HI.Type = "int (int)";
|
|
HI.Definition = "int foo(int)";
|
|
HI.Documentation = "Function declaration via call";
|
|
HI.ReturnType = "int";
|
|
HI.Parameters = {
|
|
{std::string("int"), llvm::None, llvm::None},
|
|
};
|
|
}},
|
|
{
|
|
R"cpp(// Field
|
|
struct Foo { int x; };
|
|
int main() {
|
|
Foo bar;
|
|
(void)bar.^[[x]];
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "x";
|
|
HI.Kind = index::SymbolKind::Field;
|
|
HI.NamespaceScope = "";
|
|
HI.LocalScope = "Foo::";
|
|
HI.Type = "int";
|
|
HI.Definition = "int x";
|
|
}},
|
|
{
|
|
R"cpp(// Field with initialization
|
|
struct Foo { int x = 5; };
|
|
int main() {
|
|
Foo bar;
|
|
(void)bar.^[[x]];
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "x";
|
|
HI.Kind = index::SymbolKind::Field;
|
|
HI.NamespaceScope = "";
|
|
HI.LocalScope = "Foo::";
|
|
HI.Type = "int";
|
|
HI.Definition = "int x = 5";
|
|
}},
|
|
{
|
|
R"cpp(// Static field
|
|
struct Foo { static int x; };
|
|
int main() {
|
|
(void)Foo::^[[x]];
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "x";
|
|
HI.Kind = index::SymbolKind::StaticProperty;
|
|
HI.NamespaceScope = "";
|
|
HI.LocalScope = "Foo::";
|
|
HI.Type = "int";
|
|
HI.Definition = "static int x";
|
|
}},
|
|
{
|
|
R"cpp(// Field, member initializer
|
|
struct Foo {
|
|
int x;
|
|
Foo() : ^[[x]](0) {}
|
|
};
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "x";
|
|
HI.Kind = index::SymbolKind::Field;
|
|
HI.NamespaceScope = "";
|
|
HI.LocalScope = "Foo::";
|
|
HI.Type = "int";
|
|
HI.Definition = "int x";
|
|
}},
|
|
{
|
|
R"cpp(// Field, GNU old-style field designator
|
|
struct Foo { int x; };
|
|
int main() {
|
|
Foo bar = { ^[[x]] : 1 };
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "x";
|
|
HI.Kind = index::SymbolKind::Field;
|
|
HI.NamespaceScope = "";
|
|
HI.LocalScope = "Foo::";
|
|
HI.Type = "int";
|
|
HI.Definition = "int x";
|
|
HI.Value = "{1}";
|
|
}},
|
|
{
|
|
R"cpp(// Field, field designator
|
|
struct Foo { int x; };
|
|
int main() {
|
|
Foo bar = { .^[[x]] = 2 };
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "x";
|
|
HI.Kind = index::SymbolKind::Field;
|
|
HI.NamespaceScope = "";
|
|
HI.LocalScope = "Foo::";
|
|
HI.Type = "int";
|
|
HI.Definition = "int x";
|
|
HI.Value = "{2}";
|
|
}},
|
|
{
|
|
R"cpp(// Method call
|
|
struct Foo { int x(); };
|
|
int main() {
|
|
Foo bar;
|
|
bar.^[[x]]();
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "x";
|
|
HI.Kind = index::SymbolKind::InstanceMethod;
|
|
HI.NamespaceScope = "";
|
|
HI.LocalScope = "Foo::";
|
|
HI.Type = "int ()";
|
|
HI.Definition = "int x()";
|
|
HI.ReturnType = "int";
|
|
HI.Parameters = std::vector<HoverInfo::Param>{};
|
|
}},
|
|
{
|
|
R"cpp(// Static method call
|
|
struct Foo { static int x(); };
|
|
int main() {
|
|
Foo::^[[x]]();
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "x";
|
|
HI.Kind = index::SymbolKind::StaticMethod;
|
|
HI.NamespaceScope = "";
|
|
HI.LocalScope = "Foo::";
|
|
HI.Type = "int ()";
|
|
HI.Definition = "static int x()";
|
|
HI.ReturnType = "int";
|
|
HI.Parameters = std::vector<HoverInfo::Param>{};
|
|
}},
|
|
{
|
|
R"cpp(// Typedef
|
|
typedef int Foo;
|
|
int main() {
|
|
^[[Foo]] bar;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "Foo";
|
|
HI.Kind = index::SymbolKind::TypeAlias;
|
|
HI.NamespaceScope = "";
|
|
HI.Definition = "typedef int Foo";
|
|
HI.Documentation = "Typedef";
|
|
// FIXME: Maybe put underlying type into HI.Type for aliases?
|
|
}},
|
|
{
|
|
R"cpp(// Typedef with embedded definition
|
|
typedef struct Bar {} Foo;
|
|
int main() {
|
|
^[[Foo]] bar;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "Foo";
|
|
HI.Kind = index::SymbolKind::TypeAlias;
|
|
HI.NamespaceScope = "";
|
|
HI.Definition = "typedef struct Bar Foo";
|
|
HI.Documentation = "Typedef with embedded definition";
|
|
}},
|
|
{
|
|
R"cpp(// Namespace
|
|
namespace ns {
|
|
struct Foo { static void bar(); };
|
|
} // namespace ns
|
|
int main() { ^[[ns]]::Foo::bar(); }
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "ns";
|
|
HI.Kind = index::SymbolKind::Namespace;
|
|
HI.NamespaceScope = "";
|
|
HI.Definition = "namespace ns {}";
|
|
}},
|
|
{
|
|
R"cpp(// Anonymous namespace
|
|
namespace ns {
|
|
namespace {
|
|
int foo;
|
|
} // anonymous namespace
|
|
} // namespace ns
|
|
int main() { ns::[[f^oo]]++; }
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "foo";
|
|
HI.Kind = index::SymbolKind::Variable;
|
|
HI.NamespaceScope = "ns::(anonymous)::";
|
|
HI.Type = "int";
|
|
HI.Definition = "int foo";
|
|
}},
|
|
{
|
|
R"cpp(// Macro
|
|
#define MACRO 0
|
|
int main() { return ^[[MACRO]]; }
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "MACRO";
|
|
HI.Kind = index::SymbolKind::Macro;
|
|
HI.Definition = "#define MACRO 0";
|
|
}},
|
|
{
|
|
R"cpp(// Macro
|
|
#define MACRO 0
|
|
#define MACRO2 ^[[MACRO]]
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "MACRO";
|
|
HI.Kind = index::SymbolKind::Macro;
|
|
HI.Definition = "#define MACRO 0";
|
|
}},
|
|
{
|
|
R"cpp(// Macro
|
|
#define MACRO {\
|
|
return 0;\
|
|
}
|
|
int main() ^[[MACRO]]
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "MACRO";
|
|
HI.Kind = index::SymbolKind::Macro;
|
|
HI.Definition =
|
|
R"cpp(#define MACRO \
|
|
{ return 0; })cpp";
|
|
}},
|
|
{
|
|
R"cpp(// Forward class declaration
|
|
class Foo;
|
|
class Foo {};
|
|
[[F^oo]]* foo();
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "Foo";
|
|
HI.Kind = index::SymbolKind::Class;
|
|
HI.NamespaceScope = "";
|
|
HI.Definition = "class Foo {}";
|
|
HI.Documentation = "Forward class declaration";
|
|
}},
|
|
{
|
|
R"cpp(// Function declaration
|
|
void foo();
|
|
void g() { [[f^oo]](); }
|
|
void foo() {}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "foo";
|
|
HI.Kind = index::SymbolKind::Function;
|
|
HI.NamespaceScope = "";
|
|
HI.Type = "void ()";
|
|
HI.Definition = "void foo()";
|
|
HI.Documentation = "Function declaration";
|
|
HI.ReturnType = "void";
|
|
HI.Parameters = std::vector<HoverInfo::Param>{};
|
|
}},
|
|
{
|
|
R"cpp(// Enum declaration
|
|
enum Hello {
|
|
ONE, TWO, THREE,
|
|
};
|
|
void foo() {
|
|
[[Hel^lo]] hello = ONE;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "Hello";
|
|
HI.Kind = index::SymbolKind::Enum;
|
|
HI.NamespaceScope = "";
|
|
HI.Definition = "enum Hello {}";
|
|
HI.Documentation = "Enum declaration";
|
|
}},
|
|
{
|
|
R"cpp(// Enumerator
|
|
enum Hello {
|
|
ONE, TWO, THREE,
|
|
};
|
|
void foo() {
|
|
Hello hello = [[O^NE]];
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "ONE";
|
|
HI.Kind = index::SymbolKind::EnumConstant;
|
|
HI.NamespaceScope = "";
|
|
HI.LocalScope = "Hello::";
|
|
HI.Type = "enum Hello";
|
|
HI.Definition = "ONE";
|
|
HI.Value = "0";
|
|
}},
|
|
{
|
|
R"cpp(// Enumerator in anonymous enum
|
|
enum {
|
|
ONE, TWO, THREE,
|
|
};
|
|
void foo() {
|
|
int hello = [[O^NE]];
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "ONE";
|
|
HI.Kind = index::SymbolKind::EnumConstant;
|
|
HI.NamespaceScope = "";
|
|
// FIXME: This should be `(anon enum)::`
|
|
HI.LocalScope = "";
|
|
HI.Type = "enum (anonymous)";
|
|
HI.Definition = "ONE";
|
|
HI.Value = "0";
|
|
}},
|
|
{
|
|
R"cpp(// Global variable
|
|
static int hey = 10;
|
|
void foo() {
|
|
[[he^y]]++;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "hey";
|
|
HI.Kind = index::SymbolKind::Variable;
|
|
HI.NamespaceScope = "";
|
|
HI.Type = "int";
|
|
HI.Definition = "static int hey = 10";
|
|
HI.Documentation = "Global variable";
|
|
// FIXME: Value shouldn't be set in this case
|
|
HI.Value = "10";
|
|
}},
|
|
{
|
|
R"cpp(// Global variable in namespace
|
|
namespace ns1 {
|
|
static int hey = 10;
|
|
}
|
|
void foo() {
|
|
ns1::[[he^y]]++;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "hey";
|
|
HI.Kind = index::SymbolKind::Variable;
|
|
HI.NamespaceScope = "ns1::";
|
|
HI.Type = "int";
|
|
HI.Definition = "static int hey = 10";
|
|
HI.Value = "10";
|
|
}},
|
|
{
|
|
R"cpp(// Field in anonymous struct
|
|
static struct {
|
|
int hello;
|
|
} s;
|
|
void foo() {
|
|
s.[[he^llo]]++;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "hello";
|
|
HI.Kind = index::SymbolKind::Field;
|
|
HI.NamespaceScope = "";
|
|
HI.LocalScope = "(anonymous struct)::";
|
|
HI.Type = "int";
|
|
HI.Definition = "int hello";
|
|
}},
|
|
{
|
|
R"cpp(// Templated function
|
|
template <typename T>
|
|
T foo() {
|
|
return 17;
|
|
}
|
|
void g() { auto x = [[f^oo]]<int>(); }
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "foo";
|
|
HI.Kind = index::SymbolKind::Function;
|
|
HI.NamespaceScope = "";
|
|
HI.Type = "T ()";
|
|
HI.Definition = "template <typename T> T foo()";
|
|
HI.Documentation = "Templated function";
|
|
HI.ReturnType = "T";
|
|
HI.Parameters = std::vector<HoverInfo::Param>{};
|
|
HI.TemplateParameters = {
|
|
{std::string("typename"), std::string("T"), llvm::None},
|
|
};
|
|
}},
|
|
{
|
|
R"cpp(// Anonymous union
|
|
struct outer {
|
|
union {
|
|
int abc, def;
|
|
} v;
|
|
};
|
|
void g() { struct outer o; o.v.[[d^ef]]++; }
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "def";
|
|
HI.Kind = index::SymbolKind::Field;
|
|
HI.NamespaceScope = "";
|
|
HI.LocalScope = "outer::(anonymous union)::";
|
|
HI.Type = "int";
|
|
HI.Definition = "int def";
|
|
}},
|
|
{
|
|
R"cpp(// documentation from index
|
|
int nextSymbolIsAForwardDeclFromIndexWithNoLocalDocs;
|
|
void indexSymbol();
|
|
void g() { [[ind^exSymbol]](); }
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "indexSymbol";
|
|
HI.Kind = index::SymbolKind::Function;
|
|
HI.NamespaceScope = "";
|
|
HI.Type = "void ()";
|
|
HI.Definition = "void indexSymbol()";
|
|
HI.ReturnType = "void";
|
|
HI.Parameters = std::vector<HoverInfo::Param>{};
|
|
HI.Documentation = "comment from index";
|
|
}},
|
|
{
|
|
R"cpp(// Simple initialization with auto
|
|
void foo() {
|
|
^[[auto]] i = 1;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "int";
|
|
// FIXME: Should be Builtin/Integral.
|
|
HI.Kind = index::SymbolKind::Unknown;
|
|
}},
|
|
{
|
|
R"cpp(// Simple initialization with const auto
|
|
void foo() {
|
|
const ^[[auto]] i = 1;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) { HI.Name = "int"; }},
|
|
{
|
|
R"cpp(// Simple initialization with const auto&
|
|
void foo() {
|
|
const ^[[auto]]& i = 1;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) { HI.Name = "int"; }},
|
|
{
|
|
R"cpp(// Simple initialization with auto&
|
|
void foo() {
|
|
int x;
|
|
^[[auto]]& i = x;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) { HI.Name = "int"; }},
|
|
{
|
|
R"cpp(// Simple initialization with auto*
|
|
void foo() {
|
|
int a = 1;
|
|
^[[auto]]* i = &a;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) { HI.Name = "int"; }},
|
|
{
|
|
R"cpp(// Auto with initializer list.
|
|
namespace std
|
|
{
|
|
template<class _E>
|
|
class initializer_list {};
|
|
}
|
|
void foo() {
|
|
^[[auto]] i = {1,2};
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "class std::initializer_list<int>";
|
|
HI.Kind = index::SymbolKind::Class;
|
|
}},
|
|
{
|
|
R"cpp(// User defined conversion to auto
|
|
struct Bar {
|
|
operator ^[[auto]]() const { return 10; }
|
|
};
|
|
)cpp",
|
|
[](HoverInfo &HI) { HI.Name = "int"; }},
|
|
{
|
|
R"cpp(// Simple initialization with decltype(auto)
|
|
void foo() {
|
|
^[[decltype]](auto) i = 1;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) { HI.Name = "int"; }},
|
|
{
|
|
R"cpp(// Simple initialization with const decltype(auto)
|
|
void foo() {
|
|
const int j = 0;
|
|
^[[decltype]](auto) i = j;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) { HI.Name = "const int"; }},
|
|
{
|
|
R"cpp(// Simple initialization with const& decltype(auto)
|
|
void foo() {
|
|
int k = 0;
|
|
const int& j = k;
|
|
^[[decltype]](auto) i = j;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) { HI.Name = "const int &"; }},
|
|
{
|
|
R"cpp(// Simple initialization with & decltype(auto)
|
|
void foo() {
|
|
int k = 0;
|
|
int& j = k;
|
|
^[[decltype]](auto) i = j;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) { HI.Name = "int &"; }},
|
|
{
|
|
R"cpp(// simple trailing return type
|
|
^[[auto]] main() -> int {
|
|
return 0;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) { HI.Name = "int"; }},
|
|
{
|
|
R"cpp(// auto function return with trailing type
|
|
struct Bar {};
|
|
^[[auto]] test() -> decltype(Bar()) {
|
|
return Bar();
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "struct Bar";
|
|
HI.Kind = index::SymbolKind::Struct;
|
|
}},
|
|
{
|
|
R"cpp(// trailing return type
|
|
struct Bar {};
|
|
auto test() -> ^[[decltype]](Bar()) {
|
|
return Bar();
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "struct Bar";
|
|
HI.Kind = index::SymbolKind::Struct;
|
|
}},
|
|
{
|
|
R"cpp(// auto in function return
|
|
struct Bar {};
|
|
^[[auto]] test() {
|
|
return Bar();
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "struct Bar";
|
|
HI.Kind = index::SymbolKind::Struct;
|
|
}},
|
|
{
|
|
R"cpp(// auto& in function return
|
|
struct Bar {};
|
|
^[[auto]]& test() {
|
|
static Bar x;
|
|
return x;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "struct Bar";
|
|
HI.Kind = index::SymbolKind::Struct;
|
|
}},
|
|
{
|
|
R"cpp(// auto* in function return
|
|
struct Bar {};
|
|
^[[auto]]* test() {
|
|
Bar* bar;
|
|
return bar;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "struct Bar";
|
|
HI.Kind = index::SymbolKind::Struct;
|
|
}},
|
|
{
|
|
R"cpp(// const auto& in function return
|
|
struct Bar {};
|
|
const ^[[auto]]& test() {
|
|
static Bar x;
|
|
return x;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "struct Bar";
|
|
HI.Kind = index::SymbolKind::Struct;
|
|
}},
|
|
{
|
|
R"cpp(// decltype(auto) in function return
|
|
struct Bar {};
|
|
^[[decltype]](auto) test() {
|
|
return Bar();
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "struct Bar";
|
|
HI.Kind = index::SymbolKind::Struct;
|
|
}},
|
|
{
|
|
R"cpp(// decltype(auto) reference in function return
|
|
^[[decltype]](auto) test() {
|
|
static int a;
|
|
return (a);
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) { HI.Name = "int &"; }},
|
|
{
|
|
R"cpp(// decltype lvalue reference
|
|
void foo() {
|
|
int I = 0;
|
|
^[[decltype]](I) J = I;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) { HI.Name = "int"; }},
|
|
{
|
|
R"cpp(// decltype lvalue reference
|
|
void foo() {
|
|
int I= 0;
|
|
int &K = I;
|
|
^[[decltype]](K) J = I;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) { HI.Name = "int &"; }},
|
|
{
|
|
R"cpp(// decltype lvalue reference parenthesis
|
|
void foo() {
|
|
int I = 0;
|
|
^[[decltype]]((I)) J = I;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) { HI.Name = "int &"; }},
|
|
{
|
|
R"cpp(// decltype rvalue reference
|
|
void foo() {
|
|
int I = 0;
|
|
^[[decltype]](static_cast<int&&>(I)) J = static_cast<int&&>(I);
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) { HI.Name = "int &&"; }},
|
|
{
|
|
R"cpp(// decltype rvalue reference function call
|
|
int && bar();
|
|
void foo() {
|
|
int I = 0;
|
|
^[[decltype]](bar()) J = bar();
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) { HI.Name = "int &&"; }},
|
|
{
|
|
R"cpp(// decltype of function with trailing return type.
|
|
struct Bar {};
|
|
auto test() -> decltype(Bar()) {
|
|
return Bar();
|
|
}
|
|
void foo() {
|
|
^[[decltype]](test()) i = test();
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "struct Bar";
|
|
HI.Kind = index::SymbolKind::Struct;
|
|
}},
|
|
{
|
|
R"cpp(// decltype of var with decltype.
|
|
void foo() {
|
|
int I = 0;
|
|
decltype(I) J = I;
|
|
^[[decltype]](J) K = J;
|
|
}
|
|
)cpp",
|
|
[](HoverInfo &HI) { HI.Name = "int"; }},
|
|
{
|
|
R"cpp(// More compilcated structured types.
|
|
int bar();
|
|
^[[auto]] (*foo)() = bar;
|
|
)cpp",
|
|
[](HoverInfo &HI) { HI.Name = "int"; }},
|
|
{
|
|
R"cpp(// Should not crash when evaluating the initializer.
|
|
struct Test {};
|
|
void test() { Test && [[te^st]] = {}; }
|
|
)cpp",
|
|
[](HoverInfo &HI) {
|
|
HI.Name = "test";
|
|
HI.Kind = index::SymbolKind::Variable;
|
|
HI.NamespaceScope = "";
|
|
HI.LocalScope = "test::";
|
|
HI.Type = "struct Test &&";
|
|
HI.Definition = "struct Test &&test = {}";
|
|
HI.Value = "{}";
|
|
}},
|
|
};
|
|
|
|
// Create a tiny index, so tests above can verify documentation is fetched.
|
|
Symbol IndexSym = func("indexSymbol");
|
|
IndexSym.Documentation = "comment from index";
|
|
SymbolSlab::Builder Symbols;
|
|
Symbols.insert(IndexSym);
|
|
auto Index =
|
|
MemIndex::build(std::move(Symbols).build(), RefSlab(), RelationSlab());
|
|
|
|
for (const auto &Case : Cases) {
|
|
SCOPED_TRACE(Case.Code);
|
|
|
|
Annotations T(Case.Code);
|
|
TestTU TU = TestTU::withCode(T.code());
|
|
TU.ExtraArgs.push_back("-std=c++17");
|
|
TU.ExtraArgs.push_back("-Wno-gnu-designator");
|
|
auto AST = TU.build();
|
|
for (const auto &D : AST.getDiagnostics())
|
|
ADD_FAILURE() << D;
|
|
ASSERT_TRUE(AST.getDiagnostics().empty());
|
|
|
|
auto H = getHover(AST, T.point(), format::getLLVMStyle(), Index.get());
|
|
ASSERT_TRUE(H);
|
|
HoverInfo Expected;
|
|
Expected.SymRange = T.range();
|
|
Case.ExpectedBuilder(Expected);
|
|
|
|
EXPECT_EQ(H->NamespaceScope, Expected.NamespaceScope);
|
|
EXPECT_EQ(H->LocalScope, Expected.LocalScope);
|
|
EXPECT_EQ(H->Name, Expected.Name);
|
|
EXPECT_EQ(H->Kind, Expected.Kind);
|
|
EXPECT_EQ(H->Documentation, Expected.Documentation);
|
|
EXPECT_EQ(H->Definition, Expected.Definition);
|
|
EXPECT_EQ(H->Type, Expected.Type);
|
|
EXPECT_EQ(H->ReturnType, Expected.ReturnType);
|
|
EXPECT_EQ(H->Parameters, Expected.Parameters);
|
|
EXPECT_EQ(H->TemplateParameters, Expected.TemplateParameters);
|
|
EXPECT_EQ(H->SymRange, Expected.SymRange);
|
|
EXPECT_EQ(H->Value, Expected.Value);
|
|
}
|
|
}
|
|
|
|
} // namespace
|
|
} // namespace clangd
|
|
} // namespace clang
|