diff --git a/clang-tools-extra/clangd/CMakeLists.txt b/clang-tools-extra/clangd/CMakeLists.txt --- a/clang-tools-extra/clangd/CMakeLists.txt +++ b/clang-tools-extra/clangd/CMakeLists.txt @@ -96,6 +96,7 @@ index/IndexAction.cpp index/MemIndex.cpp index/Merge.cpp + index/ProjectAware.cpp index/Ref.cpp index/Relation.cpp index/Serialization.cpp diff --git a/clang-tools-extra/clangd/Config.h b/clang-tools-extra/clangd/Config.h --- a/clang-tools-extra/clangd/Config.h +++ b/clang-tools-extra/clangd/Config.h @@ -85,6 +85,17 @@ } Style; }; +inline bool operator<(const Config::ExternalIndexSpec &LHS, + const Config::ExternalIndexSpec &RHS) { + return std::tie(LHS.Kind, LHS.Location, LHS.MountPoint) < + std::tie(RHS.Kind, RHS.Location, RHS.MountPoint); +} + +inline bool operator==(const Config::ExternalIndexSpec &LHS, + const Config::ExternalIndexSpec &RHS) { + return std::tie(LHS.Kind, LHS.Location, LHS.MountPoint) == + std::tie(RHS.Kind, RHS.Location, RHS.MountPoint); +} } // namespace clangd } // namespace clang diff --git a/clang-tools-extra/clangd/index/ProjectAware.h b/clang-tools-extra/clangd/index/ProjectAware.h new file mode 100644 --- /dev/null +++ b/clang-tools-extra/clangd/index/ProjectAware.h @@ -0,0 +1,30 @@ +//===--- ProjectAware.h ------------------------------------------*- C++-*-===// +// +// 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 +// +//===----------------------------------------------------------------------===// + +#ifndef LLVM_CLANG_TOOLS_EXTRA_CLANGD_INDEX_PROJECT_AWARE_H +#define LLVM_CLANG_TOOLS_EXTRA_CLANGD_INDEX_PROJECT_AWARE_H + +#include "Config.h" +#include "index/Index.h" +#include "support/Threading.h" +#include +#include +namespace clang { +namespace clangd { + +/// A functor to create an index for an external index specification. +using IndexGenerator = std::function( + const Config::ExternalIndexSpec &, AsyncTaskRunner &)>; + +/// Returns an index that answers queries using external indices. IndexGenerator +/// can be used to customize how to generate an index from an external source. +std::unique_ptr createProjectAwareIndex(IndexGenerator = nullptr); +} // namespace clangd +} // namespace clang + +#endif diff --git a/clang-tools-extra/clangd/index/ProjectAware.cpp b/clang-tools-extra/clangd/index/ProjectAware.cpp new file mode 100644 --- /dev/null +++ b/clang-tools-extra/clangd/index/ProjectAware.cpp @@ -0,0 +1,163 @@ +//===--- ProjectAware.h ------------------------------------------*- C++-*-===// +// +// 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 "ProjectAware.h" +#include "Config.h" +#include "index/Index.h" +#include "index/MemIndex.h" +#include "index/Merge.h" +#include "index/Ref.h" +#include "index/Serialization.h" +#include "index/Symbol.h" +#include "index/SymbolID.h" +#include "index/remote/Client.h" +#include "support/Logger.h" +#include "support/Threading.h" +#include "support/Trace.h" +#include "llvm/ADT/ArrayRef.h" +#include "llvm/ADT/STLExtras.h" +#include "llvm/ADT/StringRef.h" +#include "llvm/Support/ErrorHandling.h" +#include +#include +#include +#include + +namespace clang { +namespace clangd { +namespace { +class ProjectAwareIndex : public SymbolIndex { +public: + size_t estimateMemoryUsage() const override; + + /// Only queries the associated index with the current context. + void lookup(const LookupRequest &Req, + llvm::function_ref Callback) const override; + + /// Query all indexes while prioritizing the associated one (if any). + bool refs(const RefsRequest &Req, + llvm::function_ref Callback) const override; + + /// Queries only the associates index when Req.RestrictForCodeCompletion is + /// set, otherwise queries all. + bool + fuzzyFind(const FuzzyFindRequest &Req, + llvm::function_ref Callback) const override; + + /// Query all indexes while prioritizing the associated one (if any). + void relations(const RelationsRequest &Req, + llvm::function_ref + Callback) const override; + + ProjectAwareIndex(IndexGenerator Gen) : Gen(std::move(Gen)) {} + +private: + // Returns the index associated with current context, if any. + SymbolIndex *getIndex() const; + // Adds the Idx into IndexStorage and adjusts the MergedIdx view. Returns a + // pointer to the Idx. + SymbolIndex *addIndex(std::unique_ptr Idx) const; + + // Storage for all the external indexes. + Memoize> IndexForProject; + mutable std::mutex Mu; + mutable std::vector> IndexStorage; + mutable AsyncTaskRunner Tasks; + + const IndexGenerator Gen; +}; + +size_t ProjectAwareIndex::estimateMemoryUsage() const { + size_t Total = 0; + std::lock_guard Lock(Mu); + for (auto &Idx : IndexStorage) + Total += Idx->estimateMemoryUsage(); + return Total; +} + +void ProjectAwareIndex::lookup( + const LookupRequest &Req, + llvm::function_ref Callback) const { + trace::Span Tracer("ProjectAwareIndex::lookup"); + if (auto *Idx = getIndex()) + Idx->lookup(Req, Callback); +} + +bool ProjectAwareIndex::refs( + const RefsRequest &Req, + llvm::function_ref Callback) const { + trace::Span Tracer("ProjectAwareIndex::refs"); + if (auto *Idx = getIndex()) + return Idx->refs(Req, Callback); + return false; +} + +bool ProjectAwareIndex::fuzzyFind( + const FuzzyFindRequest &Req, + llvm::function_ref Callback) const { + trace::Span Tracer("ProjectAwareIndex::fuzzyFind"); + if (auto *Idx = getIndex()) + return Idx->fuzzyFind(Req, Callback); + return false; +} + +void ProjectAwareIndex::relations( + const RelationsRequest &Req, + llvm::function_ref Callback) const { + trace::Span Tracer("ProjectAwareIndex::relations"); + if (auto *Idx = getIndex()) + return Idx->relations(Req, Callback); +} + +SymbolIndex *ProjectAwareIndex::getIndex() const { + const auto &C = Config::current(); + if (!C.Index.External) + return nullptr; + const auto &External = *C.Index.External; + return IndexForProject.get( + External, [&, this] { return addIndex(Gen(External, Tasks)); }); +} + +SymbolIndex * +ProjectAwareIndex::addIndex(std::unique_ptr Idx) const { + SymbolIndex *NewIdx = Idx.get(); + std::lock_guard Lock(Mu); + IndexStorage.emplace_back(std::move(Idx)); + return NewIdx; +} +} // namespace + +std::unique_ptr createProjectAwareIndex(IndexGenerator Gen) { + if (!Gen) { + Gen = [](const Config::ExternalIndexSpec &External, + AsyncTaskRunner &Tasks) { + switch (External.Kind) { + case Config::ExternalIndexSpec::Server: + log("Associating {0} with remote index at {1}.", External.MountPoint, + External.Location); + return remote::getClient(External.Location, External.MountPoint); + case Config::ExternalIndexSpec::File: + log("Associating {0} with monolithic index at {1}.", + External.MountPoint, External.Location); + auto NewIndex = + std::make_unique(std::make_unique()); + Tasks.runAsync( + "Load-index:" + External.Location, + [File = External.Location, PlaceHolder = NewIndex.get()] { + if (auto Idx = loadIndex(File, /*UseDex=*/true)) + PlaceHolder->reset(std::move(Idx)); + }); + return std::unique_ptr(std::move(NewIndex)); + } + llvm_unreachable("Invalid ExternalIndexKind."); + }; + } + return std::make_unique(std::move(Gen)); +} +} // namespace clangd +} // namespace clang diff --git a/clang-tools-extra/clangd/unittests/CMakeLists.txt b/clang-tools-extra/clangd/unittests/CMakeLists.txt --- a/clang-tools-extra/clangd/unittests/CMakeLists.txt +++ b/clang-tools-extra/clangd/unittests/CMakeLists.txt @@ -73,6 +73,7 @@ PathMappingTests.cpp PreambleTests.cpp PrintASTTests.cpp + ProjectAwareIndexTests.cpp QualityTests.cpp RenameTests.cpp RIFFTests.cpp diff --git a/clang-tools-extra/clangd/unittests/ProjectAwareIndexTests.cpp b/clang-tools-extra/clangd/unittests/ProjectAwareIndexTests.cpp new file mode 100644 --- /dev/null +++ b/clang-tools-extra/clangd/unittests/ProjectAwareIndexTests.cpp @@ -0,0 +1,84 @@ +//===-- ProjectAwareIndexTests.cpp -------------------*- C++ -*-----------===// +// +// 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 "Config.h" +#include "TestIndex.h" +#include "index/Index.h" +#include "index/MemIndex.h" +#include "index/ProjectAware.h" +#include "index/Ref.h" +#include "index/Relation.h" +#include "support/Context.h" +#include "support/Threading.h" +#include "llvm/ADT/StringRef.h" +#include "gmock/gmock.h" +#include "gtest/gtest.h" +#include +#include + +namespace clang { +namespace clangd { +using testing::ElementsAre; +using testing::IsEmpty; + +std::unique_ptr createIndex() { + std::vector Symbols = {symbol("1")}; + return std::make_unique(std::move(Symbols), RefSlab(), + RelationSlab()); +} + +TEST(ProjectAware, Test) { + IndexGenerator Gen = [](const Config::ExternalIndexSpec &, + AsyncTaskRunner &) { return createIndex(); }; + + auto Idx = createProjectAwareIndex(std::move(Gen)); + FuzzyFindRequest Req; + Req.Query = "1"; + Req.AnyScope = true; + + EXPECT_THAT(match(*Idx, Req), IsEmpty()); + + Config C; + C.Index.External.emplace(); + WithContextValue With(Config::Key, std::move(C)); + EXPECT_THAT(match(*Idx, Req), ElementsAre("1")); + return; +} + +TEST(ProjectAware, CreatedOnce) { + unsigned InvocationCount = 0; + IndexGenerator Gen = [&InvocationCount](const Config::ExternalIndexSpec &, + AsyncTaskRunner &) { + ++InvocationCount; + return createIndex(); + }; + + auto Idx = createProjectAwareIndex(std::move(Gen)); + // No invocation at start. + EXPECT_EQ(InvocationCount, 0U); + FuzzyFindRequest Req; + Req.Query = "1"; + Req.AnyScope = true; + + // Cannot invoke without proper config. + match(*Idx, Req); + EXPECT_EQ(InvocationCount, 0U); + + Config C; + C.Index.External.emplace(); + WithContextValue With(Config::Key, std::move(C)); + match(*Idx, Req); + // Now it should be created. + EXPECT_EQ(InvocationCount, 1U); + match(*Idx, Req); + // It is cached afterwards. + EXPECT_EQ(InvocationCount, 1U); + return; +} +} // namespace clangd +} // namespace clang