mirror of
https://github.com/MaskRay/ccls.git
synced 2025-04-03 15:32:09 +00:00
488 lines
17 KiB
C++
488 lines
17 KiB
C++
/* Copyright 2017-2018 ccls Authors
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
==============================================================================*/
|
|
|
|
#include "clang_complete.hh"
|
|
#include "filesystem.hh"
|
|
#include "include_complete.h"
|
|
#include "log.hh"
|
|
#include "message_handler.hh"
|
|
#include "pipeline.hh"
|
|
#include "platform.h"
|
|
#include "project.hh"
|
|
#include "serializers/json.h"
|
|
#include "working_files.h"
|
|
|
|
#include <llvm/ADT/Twine.h>
|
|
#include <llvm/Support/Threading.h>
|
|
|
|
#include <stdlib.h>
|
|
#include <stdexcept>
|
|
#include <thread>
|
|
|
|
namespace ccls {
|
|
using namespace llvm;
|
|
|
|
// TODO Cleanup global variables
|
|
extern std::string g_init_options;
|
|
|
|
namespace {
|
|
|
|
// Code Lens options.
|
|
struct lsCodeLensOptions {
|
|
// Code lens has a resolve provider as well.
|
|
bool resolveProvider = false;
|
|
};
|
|
MAKE_REFLECT_STRUCT(lsCodeLensOptions, resolveProvider);
|
|
|
|
// Completion options.
|
|
struct lsCompletionOptions {
|
|
// The server provides support to resolve additional
|
|
// information for a completion item.
|
|
bool resolveProvider = false;
|
|
|
|
// The characters that trigger completion automatically.
|
|
// vscode doesn't support trigger character sequences, so we use ':'
|
|
// for
|
|
// '::' and '>' for '->'. See
|
|
// https://github.com/Microsoft/language-server-protocol/issues/138.
|
|
std::vector<std::string> triggerCharacters = {".", ":", ">", "#",
|
|
"<", "\"", "/"};
|
|
};
|
|
MAKE_REFLECT_STRUCT(lsCompletionOptions, resolveProvider, triggerCharacters);
|
|
|
|
// Format document on type options
|
|
struct lsDocumentOnTypeFormattingOptions {
|
|
// A character on which formatting should be triggered, like `}`.
|
|
std::string firstTriggerCharacter = "}";
|
|
|
|
// More trigger characters.
|
|
std::vector<std::string> moreTriggerCharacter;
|
|
};
|
|
MAKE_REFLECT_STRUCT(lsDocumentOnTypeFormattingOptions, firstTriggerCharacter,
|
|
moreTriggerCharacter);
|
|
|
|
|
|
// Save options.
|
|
struct lsSaveOptions {
|
|
// The client is supposed to include the content on save.
|
|
bool includeText = false;
|
|
};
|
|
MAKE_REFLECT_STRUCT(lsSaveOptions, includeText);
|
|
|
|
// Signature help options.
|
|
struct lsSignatureHelpOptions {
|
|
// The characters that trigger signature help automatically.
|
|
// NOTE: If updating signature help tokens make sure to also update
|
|
// WorkingFile::FindClosestCallNameInBuffer.
|
|
std::vector<std::string> triggerCharacters = {"(", ","};
|
|
};
|
|
MAKE_REFLECT_STRUCT(lsSignatureHelpOptions, triggerCharacters);
|
|
|
|
// Defines how the host (editor) should sync document changes to the language
|
|
// server.
|
|
enum class lsTextDocumentSyncKind {
|
|
// Documents should not be synced at all.
|
|
None = 0,
|
|
|
|
// Documents are synced by always sending the full content
|
|
// of the document.
|
|
Full = 1,
|
|
|
|
// Documents are synced by sending the full content on open.
|
|
// After that only incremental updates to the document are
|
|
// send.
|
|
Incremental = 2
|
|
};
|
|
MAKE_REFLECT_TYPE_PROXY(lsTextDocumentSyncKind)
|
|
|
|
struct lsTextDocumentSyncOptions {
|
|
// Open and close notifications are sent to the server.
|
|
bool openClose = false;
|
|
// Change notificatins are sent to the server. See TextDocumentSyncKind.None,
|
|
// TextDocumentSyncKind.Full and TextDocumentSyncKindIncremental.
|
|
lsTextDocumentSyncKind change = lsTextDocumentSyncKind::Incremental;
|
|
// Will save notifications are sent to the server.
|
|
std::optional<bool> willSave;
|
|
// Will save wait until requests are sent to the server.
|
|
std::optional<bool> willSaveWaitUntil;
|
|
// Save notifications are sent to the server.
|
|
std::optional<lsSaveOptions> save;
|
|
};
|
|
MAKE_REFLECT_STRUCT(lsTextDocumentSyncOptions, openClose, change, willSave,
|
|
willSaveWaitUntil, save);
|
|
|
|
struct lsServerCapabilities {
|
|
// Defines how text documents are synced. Is either a detailed structure
|
|
// defining each notification or for backwards compatibility the
|
|
// TextDocumentSyncKind number.
|
|
// TODO: It seems like the new API is broken and doesn't work.
|
|
// std::optional<lsTextDocumentSyncOptions> textDocumentSync;
|
|
lsTextDocumentSyncKind textDocumentSync = lsTextDocumentSyncKind::Incremental;
|
|
|
|
// The server provides hover support.
|
|
bool hoverProvider = true;
|
|
// The server provides completion support.
|
|
lsCompletionOptions completionProvider;
|
|
// The server provides signature help support.
|
|
lsSignatureHelpOptions signatureHelpProvider;
|
|
bool definitionProvider = true;
|
|
bool typeDefinitionProvider = true;
|
|
bool implementationProvider = true;
|
|
bool referencesProvider = true;
|
|
bool documentHighlightProvider = true;
|
|
bool documentSymbolProvider = true;
|
|
bool workspaceSymbolProvider = true;
|
|
bool codeActionProvider = true;
|
|
lsCodeLensOptions codeLensProvider;
|
|
bool documentFormattingProvider = true;
|
|
bool documentRangeFormattingProvider = true;
|
|
lsDocumentOnTypeFormattingOptions documentOnTypeFormattingProvider;
|
|
bool renameProvider = true;
|
|
struct DocumentLinkOptions {
|
|
bool resolveProvider = true;
|
|
} documentLinkProvider;
|
|
bool foldingRangeProvider = true;
|
|
// The server provides execute command support.
|
|
struct ExecuteCommandOptions {
|
|
std::vector<std::string> commands{std::string(ccls_xref)};
|
|
} executeCommandProvider;
|
|
struct Workspace {
|
|
struct WorkspaceFolders {
|
|
bool supported = true;
|
|
bool changeNotifications = true;
|
|
} workspaceFolders;
|
|
} workspace;
|
|
};
|
|
MAKE_REFLECT_STRUCT(lsServerCapabilities::DocumentLinkOptions, resolveProvider);
|
|
MAKE_REFLECT_STRUCT(lsServerCapabilities::ExecuteCommandOptions, commands);
|
|
MAKE_REFLECT_STRUCT(lsServerCapabilities::Workspace::WorkspaceFolders,
|
|
supported, changeNotifications);
|
|
MAKE_REFLECT_STRUCT(lsServerCapabilities::Workspace, workspaceFolders);
|
|
MAKE_REFLECT_STRUCT(lsServerCapabilities, textDocumentSync, hoverProvider,
|
|
completionProvider, signatureHelpProvider,
|
|
definitionProvider, implementationProvider,
|
|
typeDefinitionProvider, referencesProvider,
|
|
documentHighlightProvider, documentSymbolProvider,
|
|
workspaceSymbolProvider, codeActionProvider,
|
|
codeLensProvider, documentFormattingProvider,
|
|
documentRangeFormattingProvider,
|
|
documentOnTypeFormattingProvider, renameProvider,
|
|
documentLinkProvider, foldingRangeProvider,
|
|
executeCommandProvider, workspace);
|
|
|
|
// Workspace specific client capabilities.
|
|
struct lsWorkspaceClientCapabilites {
|
|
// The client supports applying batch edits to the workspace.
|
|
std::optional<bool> applyEdit;
|
|
|
|
struct lsWorkspaceEdit {
|
|
// The client supports versioned document changes in `WorkspaceEdit`s
|
|
std::optional<bool> documentChanges;
|
|
};
|
|
|
|
// Capabilities specific to `WorkspaceEdit`s
|
|
std::optional<lsWorkspaceEdit> workspaceEdit;
|
|
|
|
struct lsGenericDynamicReg {
|
|
// Did foo notification supports dynamic registration.
|
|
std::optional<bool> dynamicRegistration;
|
|
};
|
|
|
|
// Capabilities specific to the `workspace/didChangeConfiguration`
|
|
// notification.
|
|
std::optional<lsGenericDynamicReg> didChangeConfiguration;
|
|
|
|
// Capabilities specific to the `workspace/didChangeWatchedFiles`
|
|
// notification.
|
|
std::optional<lsGenericDynamicReg> didChangeWatchedFiles;
|
|
|
|
// Capabilities specific to the `workspace/symbol` request.
|
|
std::optional<lsGenericDynamicReg> symbol;
|
|
|
|
// Capabilities specific to the `workspace/executeCommand` request.
|
|
std::optional<lsGenericDynamicReg> executeCommand;
|
|
};
|
|
|
|
MAKE_REFLECT_STRUCT(lsWorkspaceClientCapabilites::lsWorkspaceEdit,
|
|
documentChanges);
|
|
MAKE_REFLECT_STRUCT(lsWorkspaceClientCapabilites::lsGenericDynamicReg,
|
|
dynamicRegistration);
|
|
MAKE_REFLECT_STRUCT(lsWorkspaceClientCapabilites, applyEdit, workspaceEdit,
|
|
didChangeConfiguration, didChangeWatchedFiles, symbol,
|
|
executeCommand);
|
|
|
|
// Text document specific client capabilities.
|
|
struct lsTextDocumentClientCapabilities {
|
|
struct lsSynchronization {
|
|
// Whether text document synchronization supports dynamic registration.
|
|
std::optional<bool> dynamicRegistration;
|
|
|
|
// The client supports sending will save notifications.
|
|
std::optional<bool> willSave;
|
|
|
|
// The client supports sending a will save request and
|
|
// waits for a response providing text edits which will
|
|
// be applied to the document before it is saved.
|
|
std::optional<bool> willSaveWaitUntil;
|
|
|
|
// The client supports did save notifications.
|
|
std::optional<bool> didSave;
|
|
};
|
|
|
|
lsSynchronization synchronization;
|
|
|
|
struct lsCompletion {
|
|
// Whether completion supports dynamic registration.
|
|
std::optional<bool> dynamicRegistration;
|
|
|
|
struct lsCompletionItem {
|
|
// Client supports snippets as insert text.
|
|
//
|
|
// A snippet can define tab stops and placeholders with `$1`, `$2`
|
|
// and `${3:foo}`. `$0` defines the final tab stop, it defaults to
|
|
// the end of the snippet. Placeholders with equal identifiers are linked,
|
|
// that is typing in one will update others too.
|
|
bool snippetSupport = false;
|
|
} completionItem;
|
|
} completion;
|
|
|
|
struct lsDocumentSymbol {
|
|
bool hierarchicalDocumentSymbolSupport = false;
|
|
} documentSymbol;
|
|
|
|
struct lsGenericDynamicReg {
|
|
// Whether foo supports dynamic registration.
|
|
std::optional<bool> dynamicRegistration;
|
|
};
|
|
|
|
struct CodeLensRegistrationOptions : public lsGenericDynamicReg {
|
|
// Code lens has a resolve provider as well.
|
|
bool resolveProvider;
|
|
};
|
|
|
|
// Capabilities specific to the `textDocument/codeLens`
|
|
std::optional<CodeLensRegistrationOptions> codeLens;
|
|
|
|
// Capabilities specific to the `textDocument/rename`
|
|
std::optional<lsGenericDynamicReg> rename;
|
|
};
|
|
|
|
MAKE_REFLECT_STRUCT(lsTextDocumentClientCapabilities::lsSynchronization,
|
|
dynamicRegistration, willSave, willSaveWaitUntil, didSave);
|
|
MAKE_REFLECT_STRUCT(lsTextDocumentClientCapabilities::lsCompletion,
|
|
dynamicRegistration, completionItem);
|
|
MAKE_REFLECT_STRUCT(lsTextDocumentClientCapabilities::lsDocumentSymbol,
|
|
hierarchicalDocumentSymbolSupport);
|
|
MAKE_REFLECT_STRUCT(
|
|
lsTextDocumentClientCapabilities::lsCompletion::lsCompletionItem,
|
|
snippetSupport);
|
|
MAKE_REFLECT_STRUCT(lsTextDocumentClientCapabilities::lsGenericDynamicReg,
|
|
dynamicRegistration);
|
|
MAKE_REFLECT_STRUCT(
|
|
lsTextDocumentClientCapabilities::CodeLensRegistrationOptions,
|
|
dynamicRegistration, resolveProvider);
|
|
MAKE_REFLECT_STRUCT(lsTextDocumentClientCapabilities, completion,
|
|
documentSymbol, rename, synchronization);
|
|
|
|
struct lsClientCapabilities {
|
|
// Workspace specific client capabilities.
|
|
lsWorkspaceClientCapabilites workspace;
|
|
|
|
// Text document specific client capabilities.
|
|
lsTextDocumentClientCapabilities textDocument;
|
|
};
|
|
MAKE_REFLECT_STRUCT(lsClientCapabilities, workspace, textDocument);
|
|
|
|
struct lsInitializeParams {
|
|
// The rootUri of the workspace. Is null if no
|
|
// folder is open. If both `rootPath` and `rootUri` are set
|
|
// `rootUri` wins.
|
|
std::optional<lsDocumentUri> rootUri;
|
|
|
|
// User provided initialization options.
|
|
Config initializationOptions;
|
|
|
|
// The capabilities provided by the client (editor or tool)
|
|
lsClientCapabilities capabilities;
|
|
|
|
enum class lsTrace {
|
|
// NOTE: serialized as a string, one of 'off' | 'messages' | 'verbose';
|
|
Off, // off
|
|
Messages, // messages
|
|
Verbose // verbose
|
|
};
|
|
|
|
// The initial trace setting. If omitted trace is disabled ('off').
|
|
lsTrace trace = lsTrace::Off;
|
|
|
|
std::vector<WorkspaceFolder> workspaceFolders;
|
|
};
|
|
|
|
void Reflect(Reader &reader, lsInitializeParams::lsTrace &value) {
|
|
if (!reader.IsString()) {
|
|
value = lsInitializeParams::lsTrace::Off;
|
|
return;
|
|
}
|
|
std::string v = reader.GetString();
|
|
if (v == "off")
|
|
value = lsInitializeParams::lsTrace::Off;
|
|
else if (v == "messages")
|
|
value = lsInitializeParams::lsTrace::Messages;
|
|
else if (v == "verbose")
|
|
value = lsInitializeParams::lsTrace::Verbose;
|
|
}
|
|
|
|
MAKE_REFLECT_STRUCT(lsInitializeParams, rootUri, initializationOptions,
|
|
capabilities, trace, workspaceFolders);
|
|
|
|
struct lsInitializeResult {
|
|
lsServerCapabilities capabilities;
|
|
};
|
|
MAKE_REFLECT_STRUCT(lsInitializeResult, capabilities);
|
|
|
|
void *Indexer(void *arg_) {
|
|
MessageHandler *h;
|
|
int idx;
|
|
auto *arg = static_cast<std::pair<MessageHandler *, int> *>(arg_);
|
|
std::tie(h, idx) = *arg;
|
|
delete arg;
|
|
std::string name = "indexer" + std::to_string(idx);
|
|
set_thread_name(name.c_str());
|
|
pipeline::Indexer_Main(h->clang_complete, h->vfs, h->project,
|
|
h->working_files);
|
|
return nullptr;
|
|
}
|
|
} // namespace
|
|
|
|
void Initialize(MessageHandler *m, lsInitializeParams ¶m, ReplyOnce &reply) {
|
|
std::string project_path = NormalizePath(param.rootUri->GetPath());
|
|
LOG_S(INFO) << "initialize in directory " << project_path << " with uri "
|
|
<< param.rootUri->raw_uri;
|
|
|
|
{
|
|
g_config = new Config(param.initializationOptions);
|
|
rapidjson::Document reader;
|
|
reader.Parse(g_init_options.c_str());
|
|
if (!reader.HasParseError()) {
|
|
JsonReader json_reader{&reader};
|
|
try {
|
|
Reflect(json_reader, *g_config);
|
|
} catch (std::invalid_argument &) {
|
|
// This will not trigger because parse error is handled in
|
|
// MessageRegistry::Parse in lsp.cc
|
|
}
|
|
}
|
|
|
|
rapidjson::StringBuffer output;
|
|
rapidjson::Writer<rapidjson::StringBuffer> writer(output);
|
|
JsonWriter json_writer(&writer);
|
|
Reflect(json_writer, *g_config);
|
|
LOG_S(INFO) << "initializationOptions: " << output.GetString();
|
|
|
|
if (g_config->cacheDirectory.size()) {
|
|
g_config->cacheDirectory = NormalizePath(g_config->cacheDirectory);
|
|
EnsureEndsInSlash(g_config->cacheDirectory);
|
|
}
|
|
}
|
|
|
|
// Client capabilities
|
|
const auto &capabilities = param.capabilities;
|
|
g_config->client.snippetSupport &=
|
|
capabilities.textDocument.completion.completionItem.snippetSupport;
|
|
g_config->client.hierarchicalDocumentSymbolSupport &=
|
|
capabilities.textDocument.documentSymbol
|
|
.hierarchicalDocumentSymbolSupport;
|
|
|
|
// Ensure there is a resource directory.
|
|
if (g_config->clang.resourceDir.empty())
|
|
g_config->clang.resourceDir = GetDefaultResourceDirectory();
|
|
DoPathMapping(g_config->clang.resourceDir);
|
|
LOG_S(INFO) << "use -resource-dir=" << g_config->clang.resourceDir;
|
|
|
|
// Send initialization before starting indexers, so we don't send a
|
|
// status update too early.
|
|
{
|
|
lsInitializeResult result;
|
|
reply(result);
|
|
}
|
|
|
|
// Set project root.
|
|
EnsureEndsInSlash(project_path);
|
|
g_config->fallbackFolder = project_path;
|
|
for (const WorkspaceFolder &wf : param.workspaceFolders) {
|
|
std::string path = wf.uri.GetPath();
|
|
EnsureEndsInSlash(path);
|
|
g_config->workspaceFolders.push_back(path);
|
|
LOG_S(INFO) << "add workspace folder " << wf.name << ": " << path;
|
|
}
|
|
if (param.workspaceFolders.empty())
|
|
g_config->workspaceFolders.push_back(project_path);
|
|
if (g_config->cacheDirectory.size())
|
|
for (const std::string &folder : g_config->workspaceFolders) {
|
|
// Create two cache directories for files inside and outside of the
|
|
// project.
|
|
std::string escaped = EscapeFileName(folder.substr(0, folder.size() - 1));
|
|
sys::fs::create_directories(g_config->cacheDirectory + escaped);
|
|
sys::fs::create_directories(g_config->cacheDirectory + '@' + escaped);
|
|
}
|
|
|
|
idx::Init();
|
|
for (const std::string &folder : g_config->workspaceFolders)
|
|
m->project->Load(folder);
|
|
|
|
// Start indexer threads. Start this after loading the project, as that
|
|
// may take a long time. Indexer threads will emit status/progress
|
|
// reports.
|
|
if (g_config->index.threads == 0)
|
|
g_config->index.threads = std::thread::hardware_concurrency();
|
|
|
|
LOG_S(INFO) << "start " << g_config->index.threads << " indexers";
|
|
for (int i = 0; i < g_config->index.threads; i++)
|
|
SpawnThread(Indexer, new std::pair<MessageHandler *, int>{m, i});
|
|
|
|
// Start scanning include directories before dispatching project
|
|
// files, because that takes a long time.
|
|
m->include_complete->Rescan();
|
|
|
|
LOG_S(INFO) << "dispatch initial index requests";
|
|
m->project->Index(m->working_files, reply.id);
|
|
}
|
|
|
|
void MessageHandler::initialize(Reader &reader, ReplyOnce &reply) {
|
|
lsInitializeParams param;
|
|
Reflect(reader, param);
|
|
if (!param.rootUri)
|
|
return;
|
|
Initialize(this, param, reply);
|
|
}
|
|
|
|
void StandaloneInitialize(MessageHandler &handler, const std::string &root) {
|
|
lsInitializeParams param;
|
|
param.rootUri = lsDocumentUri::FromPath(root);
|
|
ReplyOnce reply;
|
|
Initialize(&handler, param, reply);
|
|
}
|
|
|
|
void MessageHandler::shutdown(EmptyParam &, ReplyOnce &reply) {
|
|
JsonNull result;
|
|
reply(result);
|
|
}
|
|
|
|
void MessageHandler::exit(EmptyParam &) {
|
|
// FIXME cancel threads
|
|
::exit(0);
|
|
}
|
|
} // namespace ccls
|