diff --git a/clang-tools-extra/clangd/FindTarget.cpp b/clang-tools-extra/clangd/FindTarget.cpp --- a/clang-tools-extra/clangd/FindTarget.cpp +++ b/clang-tools-extra/clangd/FindTarget.cpp @@ -57,6 +57,30 @@ return S; } +// This is similar to CXXRecordDecl::lookupDependentName(), but extended +// to handle certain other kinds of decls (currently, EnumDecl) as well. +std::vector +lookupDependentName(TagDecl *TD, DeclarationName Name, + llvm::function_ref Filter) { + if (TagDecl *Def = TD->getDefinition()) { + TD = Def; + } + + if (auto *RD = dyn_cast_or_null(TD)) { + return RD->lookupDependentName(Name, Filter); + } + if (auto *ED = dyn_cast_or_null(TD)) { + std::vector Result; + for (const auto *Decl : ED->lookup(Name)) { + if (Filter(Decl)) { + Result.push_back(Decl); + } + } + return Result; + } + return {}; +} + // Given a dependent type and a member name, heuristically resolve the // name to one or more declarations. // The current heuristic is simply to look up the name in the primary @@ -76,22 +100,24 @@ bool IsNonstaticMember) { if (!T) return {}; - if (auto *ICNT = T->getAs()) { - T = ICNT->getInjectedSpecializationType().getTypePtrOrNull(); + TagDecl *LookupInDecl = nullptr; + if (auto *ET = T->getAs()) { + LookupInDecl = ET->getDecl(); + } else { + if (auto *ICNT = T->getAs()) { + T = ICNT->getInjectedSpecializationType().getTypePtrOrNull(); + } + auto *TST = T->getAs(); + if (!TST) + return {}; + const ClassTemplateDecl *TempD = dyn_cast_or_null( + TST->getTemplateName().getAsTemplateDecl()); + if (!TempD) + return {}; + LookupInDecl = TempD->getTemplatedDecl(); } - auto *TST = T->getAs(); - if (!TST) - return {}; - const ClassTemplateDecl *TD = dyn_cast_or_null( - TST->getTemplateName().getAsTemplateDecl()); - if (!TD) - return {}; - CXXRecordDecl *RD = TD->getTemplatedDecl(); - if (!RD->hasDefinition()) - return {}; - RD = RD->getDefinition(); - DeclarationName Name = NameFactory(RD->getASTContext()); - return RD->lookupDependentName(Name, [=](const NamedDecl *D) { + DeclarationName Name = NameFactory(LookupInDecl->getASTContext()); + return lookupDependentName(LookupInDecl, Name, [=](const NamedDecl *D) { return IsNonstaticMember ? D->isCXXInstanceMember() : !D->isCXXInstanceMember(); }); diff --git a/clang-tools-extra/clangd/unittests/XRefsTests.cpp b/clang-tools-extra/clangd/unittests/XRefsTests.cpp --- a/clang-tools-extra/clangd/unittests/XRefsTests.cpp +++ b/clang-tools-extra/clangd/unittests/XRefsTests.cpp @@ -520,6 +520,14 @@ void test(unique_ptr>& V) { V->fo^o(); } + )cpp", + + R"cpp(// Heuristic resolution of dependent enumerator + template + struct Foo { + enum class E { [[A]], B }; + E e = E::A^; + }; )cpp"}; for (const char *Test : Tests) { Annotations T(Test);