diff --git a/clang-tools-extra/clangd/XRefs.cpp b/clang-tools-extra/clangd/XRefs.cpp --- a/clang-tools-extra/clangd/XRefs.cpp +++ b/clang-tools-extra/clangd/XRefs.cpp @@ -191,10 +191,10 @@ // Macros are simple: there's no declaration/definition distinction. // As a consequence, there's no need to look them up in the index either. - SourceLocation MaybeMacroLocation = SM.getMacroArgExpandedLocation( + SourceLocation IdentStartLoc = SM.getMacroArgExpandedLocation( getBeginningOfIdentifier(Pos, AST.getSourceManager(), AST.getLangOpts())); std::vector Result; - if (auto M = locateMacroAt(MaybeMacroLocation, AST.getPreprocessor())) { + if (auto M = locateMacroAt(IdentStartLoc, AST.getPreprocessor())) { if (auto Loc = makeLocation(AST.getASTContext(), M->Info->getDefinitionLoc(), *MainFilePath)) { LocatedSymbol Macro; @@ -234,6 +234,17 @@ for (const Decl *D : getDeclAtPosition(AST, SourceLoc, Relations)) { const Decl *Def = getDefinition(D); const Decl *Preferred = Def ? Def : D; + + // If we're at the point of declaration of a template specialization, + // it's more useful to navigate to the template declaration. + if (Preferred->getLocation() == IdentStartLoc) { + if (auto *CTSD = dyn_cast(Preferred)) { + D = CTSD->getSpecializedTemplate(); + Def = getDefinition(D); + Preferred = Def ? Def : D; + } + } + auto Loc = makeLocation(AST.getASTContext(), spellingLocIfSpelled(findName(Preferred), SM), *MainFilePath); 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 @@ -450,7 +450,22 @@ +^+x; } )cpp", - }; + + R"cpp(// Declaration of explicit template specialization + template + struct $decl[[Foo]] {}; + + template <> + struct Fo^o {}; + )cpp", + + R"cpp(// Declaration of partial template specialization + template + struct $decl[[Foo]] {}; + + template + struct Fo^o {}; + )cpp"}; for (const char *Test : Tests) { Annotations T(Test); llvm::Optional WantDecl;