VitaNuo created this revision.
Herald added subscribers: kadircet, arphaman.
Herald added a project: All.
VitaNuo requested review of this revision.
Herald added subscribers: cfe-commits, MaskRay, ilya-biryukov.
Herald added a project: clang-tools-extra.

Repository:
  rG LLVM Github Monorepo

https://reviews.llvm.org/D143663

Files:
  clang-tools-extra/clangd/Hover.cpp
  clang-tools-extra/clangd/IncludeCleaner.cpp
  clang-tools-extra/clangd/IncludeCleaner.h

Index: clang-tools-extra/clangd/IncludeCleaner.h
===================================================================
--- clang-tools-extra/clangd/IncludeCleaner.h
+++ clang-tools-extra/clangd/IncludeCleaner.h
@@ -21,9 +21,11 @@
 #include "Config.h"
 #include "Headers.h"
 #include "ParsedAST.h"
+#include "clang-include-cleaner/Types.h"
 #include "index/CanonicalIncludes.h"
 #include "clang/Basic/SourceLocation.h"
 #include "clang/Tooling/Inclusions/StandardLibrary.h"
+#include "llvm/ADT/DenseMapInfo.h"
 #include "llvm/ADT/DenseSet.h"
 #include "llvm/ADT/STLFunctionalExtras.h"
 #include "llvm/ADT/StringSet.h"
@@ -38,6 +40,11 @@
   llvm::DenseSet<tooling::stdlib::Symbol> Stdlib;
 };
 
+struct UsedSymbol {
+  std::string Name;
+  SourceLocation Location;
+};
+
 class CachedIncludeCleaner {
 
 public:
@@ -50,6 +57,7 @@
   void run();
   std::vector<std::pair<std::string, unsigned>> computeMissingIncludes();
   std::vector<const Inclusion *> computeUnusedIncludesExperimental();
+  llvm::DenseSet<UsedSymbol> computeUsedSymbols(const Inclusion &Inclusion);
 
 private:
   ParsedAST &AST;
@@ -143,4 +151,29 @@
 } // namespace clangd
 } // namespace clang
 
+namespace llvm {
+template <> struct llvm::DenseMapInfo<clang::clangd::UsedSymbol> {
+  static clang::clangd::UsedSymbol getEmptyKey() {
+    constexpr clang::SourceLocation::UIntTy Zero = 0;
+    return clang::clangd::UsedSymbol{
+        "", clang::SourceLocation::getFromRawEncoding(~Zero)};
+  }
+
+  static clang::clangd::UsedSymbol getTombstoneKey() {
+    constexpr clang::SourceLocation::UIntTy Zero = 0;
+    return clang::clangd::UsedSymbol{
+        "", clang::SourceLocation::getFromRawEncoding(~Zero - 1)};
+  }
+
+  static unsigned getHashValue(clang::clangd::UsedSymbol Sym) {
+    return std::hash<std::string>{}(Sym.Name);
+  }
+
+  static bool isEqual(clang::clangd::UsedSymbol LHS,
+                      clang::clangd::UsedSymbol RHS) {
+    return LHS.Name == RHS.Name;
+  }
+};
+} // namespace llvm
+
 #endif // LLVM_CLANG_TOOLS_EXTRA_CLANGD_INCLUDECLEANER_H
Index: clang-tools-extra/clangd/IncludeCleaner.cpp
===================================================================
--- clang-tools-extra/clangd/IncludeCleaner.cpp
+++ clang-tools-extra/clangd/IncludeCleaner.cpp
@@ -27,6 +27,7 @@
 #include "clang/Lex/Preprocessor.h"
 #include "clang/Tooling/Syntax/Tokens.h"
 #include "llvm/ADT/ArrayRef.h"
+#include "llvm/ADT/DenseSet.h"
 #include "llvm/ADT/STLFunctionalExtras.h"
 #include "llvm/ADT/SmallString.h"
 #include "llvm/ADT/SmallVector.h"
@@ -212,6 +213,56 @@
   return getUnused(AST, Used, /*ReferencedPublicHeaders*/ {});
 }
 
+// TODO(bakalova) Add unit test
+llvm::DenseSet<UsedSymbol>
+CachedIncludeCleaner::computeUsedSymbols(const Inclusion &Inclusion) {
+  if (Refs.empty()) {
+    run();
+  }
+  llvm::DenseSet<UsedSymbol> UsedSymbols;
+  // TODO(bakalova) Avoid double loop by adding a cache by header?
+  for (unsigned I = 0; I < Refs.size(); ++I) {
+    include_cleaner::SymbolReference Ref{Refs[I]};
+    std::string Name;
+    SourceLocation Location{Ref.RefLocation};
+    // TODO(bakalova) Name is not enough. Can I get more context? Check other hovers to see how
+    // they get more stuff (from a Decl?)
+    switch(Ref.Target.kind()) {
+       case include_cleaner::Symbol::Declaration:
+        Name = cast<NamedDecl>(Ref.Target.declaration())
+            .getDeclName()
+            .getAsString();   
+        break;
+      case include_cleaner::Symbol::Macro:
+        Name = Ref.Target.macro().Name->getName(); 
+        break;
+    }
+    llvm::SmallVector<include_cleaner::Header> Providers = ProviderRefs[I];
+    for (const include_cleaner::Header &H : Providers) {
+      switch (H.kind()) {
+      case include_cleaner::Header::Physical:
+        if (H.physical()->tryGetRealPathName().contains(
+                llvm::StringRef(Inclusion.Written).trim("\"<>"))) {
+          UsedSymbols.insert({Name, Location});
+        }
+        break;
+      case include_cleaner::Header::Standard:
+        if (Inclusion.Written == H.standard().name()) {
+          UsedSymbols.insert({Name, Location});
+        }
+        break;
+      // TODO(bakalova) does the verbatim thing include quotes or not?
+      case include_cleaner::Header::Verbatim:
+        if (Inclusion.Written == H.verbatim()) {
+          UsedSymbols.insert({Name, Location});
+        }
+        break;
+      }
+    }
+  }
+  return UsedSymbols;
+}
+
 include_cleaner::Includes
 CachedIncludeCleaner::convertIncludes(const SourceManager &SM,
                                       std::vector<Inclusion> MainFileIncludes) {
Index: clang-tools-extra/clangd/Hover.cpp
===================================================================
--- clang-tools-extra/clangd/Hover.cpp
+++ clang-tools-extra/clangd/Hover.cpp
@@ -12,9 +12,11 @@
 #include "CodeCompletionStrings.h"
 #include "Config.h"
 #include "FindTarget.h"
+#include "IncludeCleaner.h"
 #include "ParsedAST.h"
 #include "Selection.h"
 #include "SourceCode.h"
+#include "clang-include-cleaner/Types.h"
 #include "index/SymbolCollector.h"
 #include "support/Markup.h"
 #include "clang/AST/ASTContext.h"
@@ -1105,6 +1107,21 @@
     HI.Definition =
         URIForFile::canonicalize(Inc.Resolved, AST.tuPath()).file().str();
     HI.DefinitionLanguage = "";
+
+    // TODO(bakalova) Continue here. Identify a good place to insert symbol refs.
+    // Definition is not a good place.
+    std::vector<include_cleaner::SymbolReference> MacroReferences =
+        collectMacroReferences(AST);
+    CachedIncludeCleaner IncludeCleaner(AST, MacroReferences);
+    llvm::DenseSet<UsedSymbol> SymbolRefs =
+        IncludeCleaner.computeUsedSymbols(Inc);
+    if (!SymbolRefs.empty()) {
+      HI.Definition.append("\nUsed symbols:\n");
+    }
+    for (UsedSymbol Ref : SymbolRefs) {
+        HI.Definition.append(Ref.Name);
+        HI.Definition.append("\n");
+    }
     return HI;
   }
 
_______________________________________________
cfe-commits mailing list
cfe-commits@lists.llvm.org
https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits

Reply via email to