/* 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 #include #include #include #include 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 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 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 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 willSave; // Will save wait until requests are sent to the server. std::optional willSaveWaitUntil; // Save notifications are sent to the server. std::optional 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 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 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 applyEdit; struct lsWorkspaceEdit { // The client supports versioned document changes in `WorkspaceEdit`s std::optional documentChanges; }; // Capabilities specific to `WorkspaceEdit`s std::optional workspaceEdit; struct lsGenericDynamicReg { // Did foo notification supports dynamic registration. std::optional dynamicRegistration; }; // Capabilities specific to the `workspace/didChangeConfiguration` // notification. std::optional didChangeConfiguration; // Capabilities specific to the `workspace/didChangeWatchedFiles` // notification. std::optional didChangeWatchedFiles; // Capabilities specific to the `workspace/symbol` request. std::optional symbol; // Capabilities specific to the `workspace/executeCommand` request. std::optional 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 dynamicRegistration; // The client supports sending will save notifications. std::optional 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 willSaveWaitUntil; // The client supports did save notifications. std::optional didSave; }; lsSynchronization synchronization; struct lsCompletion { // Whether completion supports dynamic registration. std::optional 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 dynamicRegistration; }; struct CodeLensRegistrationOptions : public lsGenericDynamicReg { // Code lens has a resolve provider as well. bool resolveProvider; }; // Capabilities specific to the `textDocument/codeLens` std::optional codeLens; // Capabilities specific to the `textDocument/rename` std::optional 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 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 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 *>(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 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{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