// 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 #include #include #include #include #include #include namespace ccls { using namespace llvm; extern std::vector g_init_options; namespace { enum class TextDocumentSyncKind { None = 0, Full = 1, Incremental = 2 }; REFLECT_UNDERLYING(TextDocumentSyncKind) bool didChangeWatchedFiles; 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 triggerCharacters = {".", ":", ">", "#", "<", "\"", "/"}; } completionProvider; struct SignatureHelpOptions { std::vector triggerCharacters = {"(", ","}; } signatureHelpProvider; bool declarationProvider = true; 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 codeActionKinds = {"quickfix"}; } codeActionProvider; struct CodeLensOptions { bool resolveProvider = false; } codeLensProvider; bool documentFormattingProvider = true; bool documentRangeFormattingProvider = true; Config::ServerCap::DocumentOnTypeFormattingOptions documentOnTypeFormattingProvider; bool renameProvider = true; struct DocumentLinkOptions { bool resolveProvider = true; } documentLinkProvider; bool foldingRangeProvider = true; // The server provides execute command support. struct ExecuteCommandOptions { std::vector commands = {ccls_xref}; } executeCommandProvider; Config::ServerCap::Workspace 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::ExecuteCommandOptions, commands); REFLECT_STRUCT(ServerCap::SaveOptions, includeText); REFLECT_STRUCT(ServerCap::SignatureHelpOptions, triggerCharacters); REFLECT_STRUCT(ServerCap::TextDocumentSyncOptions, openClose, change, willSave, willSaveWaitUntil, save); REFLECT_STRUCT(ServerCap, textDocumentSync, hoverProvider, completionProvider, signatureHelpProvider, declarationProvider, 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 applyEdit; struct WorkspaceEdit { // The client supports versioned document changes in `WorkspaceEdit`s std::optional documentChanges; }; // Capabilities specific to `WorkspaceEdit`s std::optional 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; // Ignore declaration, implementation, typeDefinition struct LinkSupport { bool linkSupport = false; } definition; 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::LinkSupport, linkSupport); REFLECT_STRUCT(TextDocumentClientCap, completion, definition, 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 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 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); struct FileSystemWatcher { std::string globPattern = "**/*"; }; struct DidChangeWatchedFilesRegistration { std::string id = "didChangeWatchedFiles"; std::string method = "workspace/didChangeWatchedFiles"; struct Option { std::vector watchers = {{}}; } registerOptions; }; struct RegistrationParam { std::vector registrations = {{}}; }; REFLECT_STRUCT(FileSystemWatcher, globPattern); REFLECT_STRUCT(DidChangeWatchedFilesRegistration::Option, watchers); REFLECT_STRUCT(DidChangeWatchedFilesRegistration, id, method, registerOptions); REFLECT_STRUCT(RegistrationParam, registrations); 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->manager, h->vfs, h->project, h->wfiles); pipeline::ThreadLeave(); 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; for (const std::string &str : g_init_options) { reader.Parse(str.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()) { SmallString<256> Path(g_config->cacheDirectory); sys::fs::make_absolute(project_path, Path); // Use upper case for the Driver letter on Windows. g_config->cacheDirectory = NormalizePath(Path.str()); EnsureEndsInSlash(g_config->cacheDirectory); } } // Client capabilities const auto &capabilities = param.capabilities; g_config->client.hierarchicalDocumentSymbolSupport &= capabilities.textDocument.documentSymbol .hierarchicalDocumentSymbolSupport; g_config->client.linkSupport &= capabilities.textDocument.definition.linkSupport; g_config->client.snippetSupport &= capabilities.textDocument.completion.completionItem.snippetSupport; didChangeWatchedFiles = capabilities.workspace.didChangeWatchedFiles.dynamicRegistration; // 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. { InitializeResult result; auto &c = result.capabilities; c.documentOnTypeFormattingProvider = g_config->capabilities.documentOnTypeFormattingProvider; c.foldingRangeProvider = g_config->capabilities.foldingRangeProvider; c.workspace = g_config->capabilities.workspace; 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->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{handler}; Initialize(&handler, param, reply); } void MessageHandler::initialized(EmptyParam &) { if (didChangeWatchedFiles) { RegistrationParam param; pipeline::Request("client/registerCapability", param); } } void MessageHandler::shutdown(EmptyParam &, ReplyOnce &reply) { reply(JsonNull{}); } void MessageHandler::exit(EmptyParam &) { pipeline::quit.store(true, std::memory_order_relaxed); } } // namespace ccls