mirror of
https://github.com/MaskRay/ccls.git
synced 2025-04-03 15:32:09 +00:00
Delete virtual bases Reader & Writer Delete unused MAKE_REFLECT_STRUCT_WRITER_AS_ARRAY Merge serializers/{json,binary}.hh into serializer.{hh,cc} MAKE_REFLECT_STRUCT => REFLECT_STRUCT MAKE_REFLECT_TYPE_PROXY => REFLECT_UNDERLYING
359 lines
12 KiB
C++
359 lines
12 KiB
C++
// Copyright 2017-2018 ccls Authors
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
#include "sema_manager.hh"
|
|
#include "filesystem.hh"
|
|
#include "include_complete.hh"
|
|
#include "log.hh"
|
|
#include "message_handler.hh"
|
|
#include "pipeline.hh"
|
|
#include "platform.hh"
|
|
#include "project.hh"
|
|
#include "working_files.hh"
|
|
|
|
#include <llvm/ADT/Twine.h>
|
|
#include <llvm/Support/Threading.h>
|
|
|
|
#include <rapidjson/document.h>
|
|
#include <rapidjson/writer.h>
|
|
|
|
#include <stdlib.h>
|
|
#include <stdexcept>
|
|
#include <thread>
|
|
|
|
namespace ccls {
|
|
using namespace llvm;
|
|
|
|
// TODO Cleanup global variables
|
|
extern std::string g_init_options;
|
|
|
|
namespace {
|
|
enum class TextDocumentSyncKind { None = 0, Full = 1, Incremental = 2 };
|
|
REFLECT_UNDERLYING(TextDocumentSyncKind)
|
|
|
|
struct ServerCap {
|
|
struct SaveOptions {
|
|
bool includeText = false;
|
|
};
|
|
struct TextDocumentSyncOptions {
|
|
bool openClose = true;
|
|
TextDocumentSyncKind change = TextDocumentSyncKind::Incremental;
|
|
bool willSave = false;
|
|
bool willSaveWaitUntil = false;
|
|
SaveOptions save;
|
|
} textDocumentSync;
|
|
|
|
// The server provides hover support.
|
|
bool hoverProvider = true;
|
|
struct CompletionOptions {
|
|
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<const char *> triggerCharacters = {".", ":", ">", "#",
|
|
"<", "\"", "/"};
|
|
} completionProvider;
|
|
struct SignatureHelpOptions {
|
|
std::vector<const char *> triggerCharacters = {"(", ","};
|
|
} signatureHelpProvider;
|
|
bool definitionProvider = true;
|
|
bool typeDefinitionProvider = true;
|
|
bool implementationProvider = true;
|
|
bool referencesProvider = true;
|
|
bool documentHighlightProvider = true;
|
|
bool documentSymbolProvider = true;
|
|
bool workspaceSymbolProvider = true;
|
|
struct CodeActionOptions {
|
|
std::vector<const char *> codeActionKinds = {"quickfix"};
|
|
} codeActionProvider;
|
|
struct CodeLensOptions {
|
|
bool resolveProvider = false;
|
|
} codeLensProvider;
|
|
bool documentFormattingProvider = true;
|
|
bool documentRangeFormattingProvider = true;
|
|
struct DocumentOnTypeFormattingOptions {
|
|
std::string firstTriggerCharacter = "}";
|
|
std::vector<const char *> moreTriggerCharacter;
|
|
} documentOnTypeFormattingProvider;
|
|
bool renameProvider = true;
|
|
struct DocumentLinkOptions {
|
|
bool resolveProvider = true;
|
|
} documentLinkProvider;
|
|
bool foldingRangeProvider = true;
|
|
// The server provides execute command support.
|
|
struct ExecuteCommandOptions {
|
|
std::vector<const char *> commands = {ccls_xref};
|
|
} executeCommandProvider;
|
|
struct Workspace {
|
|
struct WorkspaceFolders {
|
|
bool supported = true;
|
|
bool changeNotifications = true;
|
|
} workspaceFolders;
|
|
} workspace;
|
|
};
|
|
REFLECT_STRUCT(ServerCap::CodeActionOptions, codeActionKinds);
|
|
REFLECT_STRUCT(ServerCap::CodeLensOptions, resolveProvider);
|
|
REFLECT_STRUCT(ServerCap::CompletionOptions, resolveProvider,
|
|
triggerCharacters);
|
|
REFLECT_STRUCT(ServerCap::DocumentLinkOptions, resolveProvider);
|
|
REFLECT_STRUCT(ServerCap::DocumentOnTypeFormattingOptions,
|
|
firstTriggerCharacter, moreTriggerCharacter);
|
|
REFLECT_STRUCT(ServerCap::ExecuteCommandOptions, commands);
|
|
REFLECT_STRUCT(ServerCap::SaveOptions, includeText);
|
|
REFLECT_STRUCT(ServerCap::SignatureHelpOptions, triggerCharacters);
|
|
REFLECT_STRUCT(ServerCap::TextDocumentSyncOptions, openClose, change, willSave,
|
|
willSaveWaitUntil, save);
|
|
REFLECT_STRUCT(ServerCap::Workspace::WorkspaceFolders, supported,
|
|
changeNotifications);
|
|
REFLECT_STRUCT(ServerCap::Workspace, workspaceFolders);
|
|
REFLECT_STRUCT(ServerCap, textDocumentSync, hoverProvider, completionProvider,
|
|
signatureHelpProvider, definitionProvider,
|
|
implementationProvider, typeDefinitionProvider,
|
|
referencesProvider, documentHighlightProvider,
|
|
documentSymbolProvider, workspaceSymbolProvider,
|
|
codeActionProvider, codeLensProvider, documentFormattingProvider,
|
|
documentRangeFormattingProvider,
|
|
documentOnTypeFormattingProvider, renameProvider,
|
|
documentLinkProvider, foldingRangeProvider,
|
|
executeCommandProvider, workspace);
|
|
|
|
struct DynamicReg {
|
|
bool dynamicRegistration = false;
|
|
};
|
|
REFLECT_STRUCT(DynamicReg, dynamicRegistration);
|
|
|
|
// Workspace specific client capabilities.
|
|
struct WorkspaceClientCap {
|
|
// The client supports applying batch edits to the workspace.
|
|
std::optional<bool> applyEdit;
|
|
|
|
struct WorkspaceEdit {
|
|
// The client supports versioned document changes in `WorkspaceEdit`s
|
|
std::optional<bool> documentChanges;
|
|
};
|
|
|
|
// Capabilities specific to `WorkspaceEdit`s
|
|
std::optional<WorkspaceEdit> workspaceEdit;
|
|
DynamicReg didChangeConfiguration;
|
|
DynamicReg didChangeWatchedFiles;
|
|
DynamicReg symbol;
|
|
DynamicReg executeCommand;
|
|
};
|
|
|
|
REFLECT_STRUCT(WorkspaceClientCap::WorkspaceEdit, documentChanges);
|
|
REFLECT_STRUCT(WorkspaceClientCap, applyEdit, workspaceEdit,
|
|
didChangeConfiguration, didChangeWatchedFiles, symbol,
|
|
executeCommand);
|
|
|
|
// Text document specific client capabilities.
|
|
struct TextDocumentClientCap {
|
|
struct Completion {
|
|
struct CompletionItem {
|
|
// 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 DocumentSymbol {
|
|
bool hierarchicalDocumentSymbolSupport = false;
|
|
} documentSymbol;
|
|
};
|
|
|
|
REFLECT_STRUCT(TextDocumentClientCap::Completion::CompletionItem,
|
|
snippetSupport);
|
|
REFLECT_STRUCT(TextDocumentClientCap::Completion, completionItem);
|
|
REFLECT_STRUCT(TextDocumentClientCap::DocumentSymbol,
|
|
hierarchicalDocumentSymbolSupport);
|
|
REFLECT_STRUCT(TextDocumentClientCap, completion, documentSymbol);
|
|
|
|
struct ClientCap {
|
|
WorkspaceClientCap workspace;
|
|
TextDocumentClientCap textDocument;
|
|
};
|
|
REFLECT_STRUCT(ClientCap, workspace, textDocument);
|
|
|
|
struct InitializeParam {
|
|
// The rootUri of the workspace. Is null if no
|
|
// folder is open. If both `rootPath` and `rootUri` are set
|
|
// `rootUri` wins.
|
|
std::optional<DocumentUri> rootUri;
|
|
|
|
Config initializationOptions;
|
|
ClientCap capabilities;
|
|
|
|
enum class Trace {
|
|
// NOTE: serialized as a string, one of 'off' | 'messages' | 'verbose';
|
|
Off, // off
|
|
Messages, // messages
|
|
Verbose // verbose
|
|
};
|
|
Trace trace = Trace::Off;
|
|
|
|
std::vector<WorkspaceFolder> workspaceFolders;
|
|
};
|
|
|
|
void Reflect(JsonReader &reader, InitializeParam::Trace &value) {
|
|
if (!reader.m->IsString()) {
|
|
value = InitializeParam::Trace::Off;
|
|
return;
|
|
}
|
|
std::string v = reader.m->GetString();
|
|
if (v == "off")
|
|
value = InitializeParam::Trace::Off;
|
|
else if (v == "messages")
|
|
value = InitializeParam::Trace::Messages;
|
|
else if (v == "verbose")
|
|
value = InitializeParam::Trace::Verbose;
|
|
}
|
|
|
|
REFLECT_STRUCT(InitializeParam, rootUri, initializationOptions, capabilities,
|
|
trace, workspaceFolders);
|
|
|
|
struct InitializeResult {
|
|
ServerCap capabilities;
|
|
};
|
|
REFLECT_STRUCT(InitializeResult, 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->manager, h->vfs, h->project, h->wfiles);
|
|
return nullptr;
|
|
}
|
|
} // namespace
|
|
|
|
void Initialize(MessageHandler *m, InitializeParam ¶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()) {
|
|
SmallString<256> Path(g_config->cacheDirectory);
|
|
sys::fs::make_absolute(Path);
|
|
g_config->cacheDirectory = Path.str();
|
|
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.
|
|
reply(InitializeResult{});
|
|
|
|
// 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->wfiles, reply.id);
|
|
|
|
m->manager->sessions.SetCapacity(g_config->session.maxNum);
|
|
}
|
|
|
|
void MessageHandler::initialize(JsonReader &reader, ReplyOnce &reply) {
|
|
InitializeParam param;
|
|
Reflect(reader, param);
|
|
if (!param.rootUri) {
|
|
reply.Error(ErrorCode::InvalidRequest, "expected rootUri");
|
|
return;
|
|
}
|
|
Initialize(this, param, reply);
|
|
}
|
|
|
|
void StandaloneInitialize(MessageHandler &handler, const std::string &root) {
|
|
InitializeParam param;
|
|
param.rootUri = DocumentUri::FromPath(root);
|
|
ReplyOnce reply;
|
|
Initialize(&handler, param, reply);
|
|
}
|
|
|
|
void MessageHandler::shutdown(EmptyParam &, ReplyOnce &reply) {
|
|
reply(JsonNull{});
|
|
}
|
|
|
|
void MessageHandler::exit(EmptyParam &) {
|
|
// FIXME cancel threads
|
|
::exit(0);
|
|
}
|
|
} // namespace ccls
|