forked from OSchip/llvm-project
326 lines
14 KiB
C++
326 lines
14 KiB
C++
//===--- ClangdLSPServer.h - LSP server --------------------------*- 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_CLANGDLSPSERVER_H
|
|
#define LLVM_CLANG_TOOLS_EXTRA_CLANGD_CLANGDLSPSERVER_H
|
|
|
|
#include "ClangdServer.h"
|
|
#include "DraftStore.h"
|
|
#include "Features.inc"
|
|
#include "FindSymbols.h"
|
|
#include "GlobalCompilationDatabase.h"
|
|
#include "Protocol.h"
|
|
#include "Transport.h"
|
|
#include "support/Context.h"
|
|
#include "support/MemoryTree.h"
|
|
#include "support/Path.h"
|
|
#include "support/Threading.h"
|
|
#include "clang/Tooling/Core/Replacement.h"
|
|
#include "llvm/ADT/Optional.h"
|
|
#include "llvm/ADT/StringSet.h"
|
|
#include "llvm/Support/JSON.h"
|
|
#include <chrono>
|
|
#include <cstddef>
|
|
#include <memory>
|
|
|
|
namespace clang {
|
|
namespace clangd {
|
|
|
|
class SymbolIndex;
|
|
|
|
/// This class exposes ClangdServer's capabilities via Language Server Protocol.
|
|
///
|
|
/// MessageHandler binds the implemented LSP methods (e.g. onInitialize) to
|
|
/// corresponding JSON-RPC methods ("initialize").
|
|
/// The server also supports $/cancelRequest (MessageHandler provides this).
|
|
class ClangdLSPServer : private ClangdServer::Callbacks {
|
|
public:
|
|
struct Options : ClangdServer::Options {
|
|
/// Supplies configuration (overrides ClangdServer::ContextProvider).
|
|
config::Provider *ConfigProvider = nullptr;
|
|
/// Look for compilation databases, rather than using compile commands
|
|
/// set via LSP (extensions) only.
|
|
bool UseDirBasedCDB = true;
|
|
/// A fixed directory to search for a compilation database in.
|
|
/// If not set, we search upward from the source file.
|
|
llvm::Optional<Path> CompileCommandsDir;
|
|
/// The offset-encoding to use, or None to negotiate it over LSP.
|
|
llvm::Optional<OffsetEncoding> Encoding;
|
|
/// If set, periodically called to release memory.
|
|
/// Consider malloc_trim(3)
|
|
std::function<void()> MemoryCleanup = nullptr;
|
|
|
|
/// Per-feature options. Generally ClangdServer lets these vary
|
|
/// per-request, but LSP allows limited/no customizations.
|
|
clangd::CodeCompleteOptions CodeComplete;
|
|
clangd::RenameOptions Rename;
|
|
/// Returns true if the tweak should be enabled.
|
|
std::function<bool(const Tweak &)> TweakFilter = [](const Tweak &T) {
|
|
return !T.hidden(); // only enable non-hidden tweaks.
|
|
};
|
|
};
|
|
|
|
ClangdLSPServer(Transport &Transp, const ThreadsafeFS &TFS,
|
|
const ClangdLSPServer::Options &Opts);
|
|
/// The destructor blocks on any outstanding background tasks.
|
|
~ClangdLSPServer();
|
|
|
|
/// Run LSP server loop, communicating with the Transport provided in the
|
|
/// constructor. This method must not be executed more than once.
|
|
///
|
|
/// \return Whether we shut down cleanly with a 'shutdown' -> 'exit' sequence.
|
|
bool run();
|
|
|
|
/// Profiles resource-usage.
|
|
void profile(MemoryTree &MT) const;
|
|
|
|
private:
|
|
// Implement ClangdServer::Callbacks.
|
|
void onDiagnosticsReady(PathRef File, llvm::StringRef Version,
|
|
std::vector<Diag> Diagnostics) override;
|
|
void onFileUpdated(PathRef File, const TUStatus &Status) override;
|
|
void
|
|
onHighlightingsReady(PathRef File, llvm::StringRef Version,
|
|
std::vector<HighlightingToken> Highlightings) override;
|
|
void onBackgroundIndexProgress(const BackgroundQueue::Stats &Stats) override;
|
|
|
|
// LSP methods. Notifications have signature void(const Params&).
|
|
// Calls have signature void(const Params&, Callback<Response>).
|
|
void onInitialize(const InitializeParams &, Callback<llvm::json::Value>);
|
|
void onInitialized(const InitializedParams &);
|
|
void onShutdown(Callback<std::nullptr_t>);
|
|
void onSync(Callback<std::nullptr_t>);
|
|
void onDocumentDidOpen(const DidOpenTextDocumentParams &);
|
|
void onDocumentDidChange(const DidChangeTextDocumentParams &);
|
|
void onDocumentDidClose(const DidCloseTextDocumentParams &);
|
|
void onDocumentDidSave(const DidSaveTextDocumentParams &);
|
|
void onAST(const ASTParams &, Callback<llvm::Optional<ASTNode>>);
|
|
void onDocumentOnTypeFormatting(const DocumentOnTypeFormattingParams &,
|
|
Callback<std::vector<TextEdit>>);
|
|
void onDocumentRangeFormatting(const DocumentRangeFormattingParams &,
|
|
Callback<std::vector<TextEdit>>);
|
|
void onDocumentFormatting(const DocumentFormattingParams &,
|
|
Callback<std::vector<TextEdit>>);
|
|
// The results are serialized 'vector<DocumentSymbol>' if
|
|
// SupportsHierarchicalDocumentSymbol is true and 'vector<SymbolInformation>'
|
|
// otherwise.
|
|
void onDocumentSymbol(const DocumentSymbolParams &,
|
|
Callback<llvm::json::Value>);
|
|
void onFoldingRange(const FoldingRangeParams &,
|
|
Callback<std::vector<FoldingRange>>);
|
|
void onCodeAction(const CodeActionParams &, Callback<llvm::json::Value>);
|
|
void onCompletion(const CompletionParams &, Callback<CompletionList>);
|
|
void onSignatureHelp(const TextDocumentPositionParams &,
|
|
Callback<SignatureHelp>);
|
|
void onGoToDeclaration(const TextDocumentPositionParams &,
|
|
Callback<std::vector<Location>>);
|
|
void onGoToDefinition(const TextDocumentPositionParams &,
|
|
Callback<std::vector<Location>>);
|
|
void onGoToImplementation(const TextDocumentPositionParams &,
|
|
Callback<std::vector<Location>>);
|
|
void onReference(const ReferenceParams &, Callback<std::vector<Location>>);
|
|
void onSwitchSourceHeader(const TextDocumentIdentifier &,
|
|
Callback<llvm::Optional<URIForFile>>);
|
|
void onDocumentHighlight(const TextDocumentPositionParams &,
|
|
Callback<std::vector<DocumentHighlight>>);
|
|
void onFileEvent(const DidChangeWatchedFilesParams &);
|
|
void onCommand(const ExecuteCommandParams &, Callback<llvm::json::Value>);
|
|
void onWorkspaceSymbol(const WorkspaceSymbolParams &,
|
|
Callback<std::vector<SymbolInformation>>);
|
|
void onPrepareRename(const TextDocumentPositionParams &,
|
|
Callback<llvm::Optional<Range>>);
|
|
void onRename(const RenameParams &, Callback<WorkspaceEdit>);
|
|
void onHover(const TextDocumentPositionParams &,
|
|
Callback<llvm::Optional<Hover>>);
|
|
void onTypeHierarchy(const TypeHierarchyParams &,
|
|
Callback<llvm::Optional<TypeHierarchyItem>>);
|
|
void onResolveTypeHierarchy(const ResolveTypeHierarchyItemParams &,
|
|
Callback<llvm::Optional<TypeHierarchyItem>>);
|
|
void onPrepareCallHierarchy(const CallHierarchyPrepareParams &,
|
|
Callback<std::vector<CallHierarchyItem>>);
|
|
void onCallHierarchyIncomingCalls(
|
|
const CallHierarchyIncomingCallsParams &,
|
|
Callback<std::vector<CallHierarchyIncomingCall>>);
|
|
void onCallHierarchyOutgoingCalls(
|
|
const CallHierarchyOutgoingCallsParams &,
|
|
Callback<std::vector<CallHierarchyOutgoingCall>>);
|
|
void onChangeConfiguration(const DidChangeConfigurationParams &);
|
|
void onSymbolInfo(const TextDocumentPositionParams &,
|
|
Callback<std::vector<SymbolDetails>>);
|
|
void onSelectionRange(const SelectionRangeParams &,
|
|
Callback<std::vector<SelectionRange>>);
|
|
void onDocumentLink(const DocumentLinkParams &,
|
|
Callback<std::vector<DocumentLink>>);
|
|
void onSemanticTokens(const SemanticTokensParams &, Callback<SemanticTokens>);
|
|
void onSemanticTokensDelta(const SemanticTokensDeltaParams &,
|
|
Callback<SemanticTokensOrDelta>);
|
|
/// This is a clangd extension. Provides a json tree representing memory usage
|
|
/// hierarchy.
|
|
void onMemoryUsage(Callback<MemoryTree>);
|
|
|
|
std::vector<Fix> getFixes(StringRef File, const clangd::Diagnostic &D);
|
|
|
|
/// Checks if completion request should be ignored. We need this due to the
|
|
/// limitation of the LSP. Per LSP, a client sends requests for all "trigger
|
|
/// character" we specify, but for '>' and ':' we need to check they actually
|
|
/// produce '->' and '::', respectively.
|
|
bool shouldRunCompletion(const CompletionParams &Params) const;
|
|
|
|
/// Requests a reparse of currently opened files using their latest source.
|
|
/// This will typically only rebuild if something other than the source has
|
|
/// changed (e.g. the CDB yields different flags, or files included in the
|
|
/// preamble have been modified).
|
|
void reparseOpenFilesIfNeeded(
|
|
llvm::function_ref<bool(llvm::StringRef File)> Filter);
|
|
void applyConfiguration(const ConfigurationSettings &Settings);
|
|
|
|
/// Sends a "publishSemanticHighlighting" notification to the LSP client.
|
|
void
|
|
publishTheiaSemanticHighlighting(const TheiaSemanticHighlightingParams &);
|
|
|
|
/// Sends a "publishDiagnostics" notification to the LSP client.
|
|
void publishDiagnostics(const PublishDiagnosticsParams &);
|
|
|
|
/// Runs profiling and exports memory usage metrics if tracing is enabled and
|
|
/// profiling hasn't happened recently.
|
|
void maybeExportMemoryProfile();
|
|
PeriodicThrottler ShouldProfile;
|
|
|
|
/// Run the MemoryCleanup callback if it's time.
|
|
/// This method is thread safe.
|
|
void maybeCleanupMemory();
|
|
PeriodicThrottler ShouldCleanupMemory;
|
|
|
|
/// Since initialization of CDBs and ClangdServer is done lazily, the
|
|
/// following context captures the one used while creating ClangdLSPServer and
|
|
/// passes it to above mentioned object instances to make sure they share the
|
|
/// same state.
|
|
Context BackgroundContext;
|
|
|
|
/// Used to indicate that the 'shutdown' request was received from the
|
|
/// Language Server client.
|
|
bool ShutdownRequestReceived = false;
|
|
|
|
/// Used to indicate the ClangdLSPServer is being destroyed.
|
|
std::atomic<bool> IsBeingDestroyed = {false};
|
|
|
|
std::mutex FixItsMutex;
|
|
typedef std::map<clangd::Diagnostic, std::vector<Fix>, LSPDiagnosticCompare>
|
|
DiagnosticToReplacementMap;
|
|
/// Caches FixIts per file and diagnostics
|
|
llvm::StringMap<DiagnosticToReplacementMap> FixItsMap;
|
|
std::mutex HighlightingsMutex;
|
|
llvm::StringMap<std::vector<HighlightingToken>> FileToHighlightings;
|
|
// Last semantic-tokens response, for incremental requests.
|
|
std::mutex SemanticTokensMutex;
|
|
llvm::StringMap<SemanticTokens> LastSemanticTokens;
|
|
|
|
// Most code should not deal with Transport directly.
|
|
// MessageHandler deals with incoming messages, use call() etc for outgoing.
|
|
clangd::Transport &Transp;
|
|
class MessageHandler;
|
|
std::unique_ptr<MessageHandler> MsgHandler;
|
|
std::mutex TranspWriter;
|
|
|
|
template <typename T>
|
|
static Expected<T> parse(const llvm::json::Value &Raw,
|
|
llvm::StringRef PayloadName,
|
|
llvm::StringRef PayloadKind) {
|
|
T Result;
|
|
llvm::json::Path::Root Root;
|
|
if (!fromJSON(Raw, Result, Root)) {
|
|
elog("Failed to decode {0} {1}: {2}", PayloadName, PayloadKind,
|
|
Root.getError());
|
|
// Dump the relevant parts of the broken message.
|
|
std::string Context;
|
|
llvm::raw_string_ostream OS(Context);
|
|
Root.printErrorContext(Raw, OS);
|
|
vlog("{0}", OS.str());
|
|
// Report the error (e.g. to the client).
|
|
return llvm::make_error<LSPError>(
|
|
llvm::formatv("failed to decode {0} {1}: {2}", PayloadName,
|
|
PayloadKind, fmt_consume(Root.getError())),
|
|
ErrorCode::InvalidParams);
|
|
}
|
|
return std::move(Result);
|
|
}
|
|
|
|
template <typename Response>
|
|
void call(StringRef Method, llvm::json::Value Params, Callback<Response> CB) {
|
|
// Wrap the callback with LSP conversion and error-handling.
|
|
auto HandleReply =
|
|
[CB = std::move(CB), Ctx = Context::current().clone(),
|
|
Method = Method.str()](
|
|
llvm::Expected<llvm::json::Value> RawResponse) mutable {
|
|
if (!RawResponse)
|
|
return CB(RawResponse.takeError());
|
|
CB(parse<Response>(*RawResponse, Method, "response"));
|
|
};
|
|
callRaw(Method, std::move(Params), std::move(HandleReply));
|
|
}
|
|
void callRaw(StringRef Method, llvm::json::Value Params,
|
|
Callback<llvm::json::Value> CB);
|
|
void notify(StringRef Method, llvm::json::Value Params);
|
|
template <typename T> void progress(const llvm::json::Value &Token, T Value) {
|
|
ProgressParams<T> Params;
|
|
Params.token = Token;
|
|
Params.value = std::move(Value);
|
|
notify("$/progress", Params);
|
|
}
|
|
|
|
const ThreadsafeFS &TFS;
|
|
/// Options used for diagnostics.
|
|
ClangdDiagnosticOptions DiagOpts;
|
|
/// The supported kinds of the client.
|
|
SymbolKindBitset SupportedSymbolKinds;
|
|
/// The supported completion item kinds of the client.
|
|
CompletionItemKindBitset SupportedCompletionItemKinds;
|
|
/// Whether the client supports CodeAction response objects.
|
|
bool SupportsCodeAction = false;
|
|
/// From capabilities of textDocument/documentSymbol.
|
|
bool SupportsHierarchicalDocumentSymbol = false;
|
|
/// Whether the client supports showing file status.
|
|
bool SupportFileStatus = false;
|
|
/// Which kind of markup should we use in textDocument/hover responses.
|
|
MarkupKind HoverContentFormat = MarkupKind::PlainText;
|
|
/// Whether the client supports offsets for parameter info labels.
|
|
bool SupportsOffsetsInSignatureHelp = false;
|
|
std::mutex BackgroundIndexProgressMutex;
|
|
enum class BackgroundIndexProgress {
|
|
// Client doesn't support reporting progress. No transitions possible.
|
|
Unsupported,
|
|
// The queue is idle, and the client has no progress bar.
|
|
// Can transition to Creating when we have some activity.
|
|
Empty,
|
|
// We've requested the client to create a progress bar.
|
|
// Meanwhile, the state is buffered in PendingBackgroundIndexProgress.
|
|
Creating,
|
|
// The client has a progress bar, and we can send it updates immediately.
|
|
Live,
|
|
} BackgroundIndexProgressState = BackgroundIndexProgress::Unsupported;
|
|
// The progress to send when the progress bar is created.
|
|
// Only valid in state Creating.
|
|
BackgroundQueue::Stats PendingBackgroundIndexProgress;
|
|
/// LSP extension: skip WorkDoneProgressCreate, just send progress streams.
|
|
bool BackgroundIndexSkipCreate = false;
|
|
// Store of the current versions of the open documents.
|
|
DraftStore DraftMgr;
|
|
|
|
Options Opts;
|
|
// The CDB is created by the "initialize" LSP method.
|
|
std::unique_ptr<GlobalCompilationDatabase> BaseCDB;
|
|
// CDB is BaseCDB plus any commands overridden via LSP extensions.
|
|
llvm::Optional<OverlayCDB> CDB;
|
|
// The ClangdServer is created by the "initialize" LSP method.
|
|
llvm::Optional<ClangdServer> Server;
|
|
};
|
|
} // namespace clangd
|
|
} // namespace clang
|
|
|
|
#endif // LLVM_CLANG_TOOLS_EXTRA_CLANGD_CLANGDLSPSERVER_H
|