nridge created this revision.
nridge added a reviewer: sammccall.
Herald added subscribers: cfe-commits, usaxena95, kadircet, arphaman, jkorous, 
MaskRay, ilya-biryukov.
Herald added a project: clang.
nridge marked an inline comment as done.
nridge added inline comments.


================
Comment at: clang-tools-extra/clangd/XRefs.cpp:234
                   ParsedAST &AST, llvm::StringRef MainFilePath,
-                  const SymbolIndex *Index) {
+                  const SymbolIndex *Index, bool *IsDependentName) {
   const SourceManager &SM = AST.getSourceManager();
----------------
In the future, I can envision expanding this to return additional information, 
possibly the AST node itself. For now, I just started with a boolean.


Repository:
  rG LLVM Github Monorepo

https://reviews.llvm.org/D76451

Files:
  clang-tools-extra/clangd/XRefs.cpp
  clang-tools-extra/clangd/XRefs.h
  clang-tools-extra/clangd/unittests/XRefsTests.cpp

Index: clang-tools-extra/clangd/unittests/XRefsTests.cpp
===================================================================
--- clang-tools-extra/clangd/unittests/XRefsTests.cpp
+++ clang-tools-extra/clangd/unittests/XRefsTests.cpp
@@ -647,14 +647,40 @@
         int myFunction(int);
         // Not triggered for token which survived preprocessing.
         int var = m^yFunction();
-      )cpp",
+      )cpp"};
+
+  for (const char *Test : Tests) {
+    Annotations T(Test);
+    llvm::Optional<Range> WantDecl;
+    if (!T.ranges().empty())
+      WantDecl = T.range();
+
+    auto TU = TestTU::withCode(T.code());
+
+    auto AST = TU.build();
+    auto Index = TU.index();
+    auto Results = locateSymbolNamedTextuallyAt(
+        AST, Index.get(),
+        cantFail(sourceLocationInMainFile(AST.getSourceManager(), T.point())),
+        testPath(TU.Filename), /*IsDependent=*/false);
+
+    if (!WantDecl) {
+      EXPECT_THAT(Results, IsEmpty()) << Test;
+    } else {
+      ASSERT_THAT(Results, ::testing::SizeIs(1)) << Test;
+      EXPECT_EQ(Results[0].PreferredDeclaration.range, *WantDecl) << Test;
+    }
+  }
+} // namespace
+
+TEST(LocateSymbol, TextualDependent) {
+  const char *Tests[] = {
       R"cpp(// Dependent type
         struct Foo {
-          void uniqueMethodName();
+          void [[uniqueMethodName]]();
         };
         template <typename T>
         void f(T t) {
-          // Not triggered for token which survived preprocessing.
           t->u^niqueMethodName();
         }
       )cpp"};
@@ -669,10 +695,10 @@
 
     auto AST = TU.build();
     auto Index = TU.index();
-    auto Results = locateSymbolNamedTextuallyAt(
-        AST, Index.get(),
-        cantFail(sourceLocationInMainFile(AST.getSourceManager(), T.point())),
-        testPath(TU.Filename));
+    // Need to use locateSymbolAt() since we are testing an
+    // interaction between locateASTReferent() and
+    // locateSymbolNamedTextuallyAt().
+    auto Results = locateSymbolAt(AST, T.point(), Index.get());
 
     if (!WantDecl) {
       EXPECT_THAT(Results, IsEmpty()) << Test;
@@ -765,17 +791,18 @@
         struct Bar {
           void $BarLoc[[uniqueMethodName]]();
         };
-        // Will call u^niqueMethodName() on t.
         template <typename T>
-        void f(T t);
+        void f(T t) {
+          t.u^niqueMethodName();
+        }
       )cpp");
   auto TU = TestTU::withCode(T.code());
   auto AST = TU.build();
   auto Index = TU.index();
-  auto Results = locateSymbolNamedTextuallyAt(
-      AST, Index.get(),
-      cantFail(sourceLocationInMainFile(AST.getSourceManager(), T.point())),
-      testPath(TU.Filename));
+  // Need to use locateSymbolAt() since we are testing an
+  // interaction between locateASTReferent() and
+  // locateSymbolNamedTextuallyAt().
+  auto Results = locateSymbolAt(AST, T.point(), Index.get());
   EXPECT_THAT(Results,
               UnorderedElementsAre(Sym("uniqueMethodName", T.range("FooLoc")),
                                    Sym("uniqueMethodName", T.range("BarLoc"))));
Index: clang-tools-extra/clangd/XRefs.h
===================================================================
--- clang-tools-extra/clangd/XRefs.h
+++ clang-tools-extra/clangd/XRefs.h
@@ -62,7 +62,7 @@
 std::vector<LocatedSymbol>
 locateSymbolNamedTextuallyAt(ParsedAST &AST, const SymbolIndex *Index,
                              SourceLocation Loc,
-                             const std::string &MainFilePath);
+                             const std::string &MainFilePath, bool IsDependent);
 
 /// Get all document links
 std::vector<DocumentLink> getDocumentLinks(ParsedAST &AST);
Index: clang-tools-extra/clangd/XRefs.cpp
===================================================================
--- clang-tools-extra/clangd/XRefs.cpp
+++ clang-tools-extra/clangd/XRefs.cpp
@@ -22,6 +22,7 @@
 #include "index/Relation.h"
 #include "index/SymbolLocation.h"
 #include "clang/AST/ASTContext.h"
+#include "clang/AST/ASTTypeTraits.h"
 #include "clang/AST/Attr.h"
 #include "clang/AST/Attrs.inc"
 #include "clang/AST/Decl.h"
@@ -136,19 +137,31 @@
   return Merged.CanonicalDeclaration;
 }
 
-std::vector<const NamedDecl *> getDeclAtPosition(ParsedAST &AST,
-                                                 SourceLocation Pos,
-                                                 DeclRelationSet Relations) {
+bool isDependentName(const DynTypedNode &N) {
+  if (const Stmt *S = N.get<Stmt>()) {
+    return llvm::isa<OverloadExpr>(S) ||
+           llvm::isa<CXXDependentScopeMemberExpr>(S) ||
+           llvm::isa<DependentScopeDeclRefExpr>(S);
+  }
+  return false;
+}
+
+std::vector<const NamedDecl *>
+getDeclAtPosition(ParsedAST &AST, SourceLocation Pos, DeclRelationSet Relations,
+                  bool *IsDependentName = nullptr) {
   unsigned Offset = AST.getSourceManager().getDecomposedSpellingLoc(Pos).second;
   std::vector<const NamedDecl *> Result;
-  SelectionTree::createEach(AST.getASTContext(), AST.getTokens(), Offset,
-                            Offset, [&](SelectionTree ST) {
-                              if (const SelectionTree::Node *N =
-                                      ST.commonAncestor())
-                                llvm::copy(targetDecl(N->ASTNode, Relations),
-                                           std::back_inserter(Result));
-                              return !Result.empty();
-                            });
+  SelectionTree::createEach(
+      AST.getASTContext(), AST.getTokens(), Offset, Offset,
+      [&](SelectionTree ST) {
+        if (const SelectionTree::Node *N = ST.commonAncestor()) {
+          if (IsDependentName && isDependentName(N->ASTNode))
+            *IsDependentName = true;
+          llvm::copy(targetDecl(N->ASTNode, Relations),
+                     std::back_inserter(Result));
+        }
+        return !Result.empty();
+      });
   return Result;
 }
 
@@ -218,7 +231,7 @@
 std::vector<LocatedSymbol>
 locateASTReferent(SourceLocation CurLoc, const syntax::Token *TouchedIdentifier,
                   ParsedAST &AST, llvm::StringRef MainFilePath,
-                  const SymbolIndex *Index) {
+                  const SymbolIndex *Index, bool *IsDependentName) {
   const SourceManager &SM = AST.getSourceManager();
   // Results follow the order of Symbols.Decls.
   std::vector<LocatedSymbol> Result;
@@ -249,7 +262,8 @@
   // Emit all symbol locations (declaration or definition) from AST.
   DeclRelationSet Relations =
       DeclRelation::TemplatePattern | DeclRelation::Alias;
-  for (const NamedDecl *D : getDeclAtPosition(AST, CurLoc, Relations)) {
+  for (const NamedDecl *D :
+       getDeclAtPosition(AST, CurLoc, Relations, IsDependentName)) {
     // Special case: void foo() ^override: jump to the overridden method.
     if (const auto *CMD = llvm::dyn_cast<CXXMethodDecl>(D)) {
       const InheritableAttr *Attr = D->getAttr<OverrideAttr>();
@@ -357,10 +371,9 @@
 
 } // namespace
 
-std::vector<LocatedSymbol>
-locateSymbolNamedTextuallyAt(ParsedAST &AST, const SymbolIndex *Index,
-                             SourceLocation Loc,
-                             const std::string &MainFilePath) {
+std::vector<LocatedSymbol> locateSymbolNamedTextuallyAt(
+    ParsedAST &AST, const SymbolIndex *Index, SourceLocation Loc,
+    const std::string &MainFilePath, bool IsDependent) {
   const auto &SM = AST.getSourceManager();
 
   // Get the raw word at the specified location.
@@ -388,8 +401,9 @@
   // Do not consider tokens that survived preprocessing.
   // We are erring on the safe side here, as a user may expect to get
   // accurate (as opposed to textual-heuristic) results for such tokens.
-  // FIXME: Relax this for dependent code.
-  if (tokenSurvivedPreprocessing(WordStart, AST.getTokens()))
+  // For dependent code, we relax this as a textual heuristic allows us
+  // to find results in cases where AST heuristics fall down.
+  if (!IsDependent && tokenSurvivedPreprocessing(WordStart, AST.getTokens()))
     return {};
 
   // Additionally filter for signals that the word is likely to be an
@@ -511,12 +525,14 @@
       // expansion.)
       return {*std::move(Macro)};
 
-  auto ASTResults =
-      locateASTReferent(*CurLoc, TouchedIdentifier, AST, *MainFilePath, Index);
+  bool IsDependentName = false;
+  auto ASTResults = locateASTReferent(*CurLoc, TouchedIdentifier, AST,
+                                      *MainFilePath, Index, &IsDependentName);
   if (!ASTResults.empty())
     return ASTResults;
 
-  return locateSymbolNamedTextuallyAt(AST, Index, *CurLoc, *MainFilePath);
+  return locateSymbolNamedTextuallyAt(AST, Index, *CurLoc, *MainFilePath,
+                                      IsDependentName);
 }
 
 std::vector<DocumentLink> getDocumentLinks(ParsedAST &AST) {
_______________________________________________
cfe-commits mailing list
cfe-commits@lists.llvm.org
https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits

Reply via email to