blob: f63615c00959534c5c9998bd04f6d1c311cb2f28 [file] [log] [blame]
Krasimir Georgiev95ef1712017-04-12 17:13:08 +00001//===--- ClangdMain.cpp - clangd server loop ------------------------------===//
Benjamin Kramerbb1cdb62017-02-07 10:28:20 +00002//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
Ilya Biryukov38d79772017-05-16 09:38:59 +000010#include "ClangdLSPServer.h"
Ilya Biryukovafb55542017-05-16 14:40:30 +000011#include "JSONRPCDispatcher.h"
Ilya Biryukove6dbb582017-10-10 09:08:47 +000012#include "Path.h"
Sam McCall8567cb32017-11-02 09:21:51 +000013#include "Trace.h"
Haojian Wuba28e9a2018-01-10 14:44:34 +000014#include "index/SymbolYAML.h"
Benjamin Kramerf0af3e62017-03-01 16:16:29 +000015#include "llvm/Support/CommandLine.h"
Benjamin Kramerbb1cdb62017-02-07 10:28:20 +000016#include "llvm/Support/FileSystem.h"
Ilya Biryukov0c1ca6b2017-10-02 15:13:20 +000017#include "llvm/Support/Path.h"
Benjamin Kramer6a3d74e2017-02-07 12:40:59 +000018#include "llvm/Support/Program.h"
Eric Liuc5105f92018-02-16 14:15:55 +000019#include "llvm/Support/Signals.h"
Ilya Biryukove6dbb582017-10-10 09:08:47 +000020#include "llvm/Support/raw_ostream.h"
Sam McCalle72d0972018-06-29 13:24:20 +000021#include "clang/Basic/Version.h"
Sam McCalled2717a2018-02-14 03:20:07 +000022#include <cstdlib>
Benjamin Kramerbb1cdb62017-02-07 10:28:20 +000023#include <iostream>
Ilya Biryukov38d79772017-05-16 09:38:59 +000024#include <memory>
Benjamin Kramerbb1cdb62017-02-07 10:28:20 +000025#include <string>
Ilya Biryukovdb8b2d72017-08-14 08:45:47 +000026#include <thread>
Ilya Biryukov38d79772017-05-16 09:38:59 +000027
28using namespace clang;
Benjamin Kramerbb1cdb62017-02-07 10:28:20 +000029using namespace clang::clangd;
30
Ilya Biryukove9eb7f02017-11-16 16:25:18 +000031namespace {
32enum class PCHStorageFlag { Disk, Memory };
Haojian Wuba28e9a2018-01-10 14:44:34 +000033
34// Build an in-memory static index for global symbols from a YAML-format file.
35// The size of global symbols should be relatively small, so that all symbols
36// can be managed in memory.
Kirill Bobyrev5a267ed2018-05-29 11:50:51 +000037std::unique_ptr<SymbolIndex> buildStaticIndex(llvm::StringRef YamlSymbolFile) {
Haojian Wuba28e9a2018-01-10 14:44:34 +000038 auto Buffer = llvm::MemoryBuffer::getFile(YamlSymbolFile);
39 if (!Buffer) {
40 llvm::errs() << "Can't open " << YamlSymbolFile << "\n";
41 return nullptr;
42 }
Sam McCall60039512018-02-09 14:42:01 +000043 auto Slab = SymbolsFromYAML(Buffer.get()->getBuffer());
Haojian Wuba28e9a2018-01-10 14:44:34 +000044 SymbolSlab::Builder SymsBuilder;
45 for (auto Sym : Slab)
46 SymsBuilder.insert(Sym);
47
48 return MemIndex::build(std::move(SymsBuilder).build());
Ilya Biryukove9eb7f02017-11-16 16:25:18 +000049}
Haojian Wuba28e9a2018-01-10 14:44:34 +000050} // namespace
Ilya Biryukove9eb7f02017-11-16 16:25:18 +000051
Ilya Biryukov0c1ca6b2017-10-02 15:13:20 +000052static llvm::cl::opt<Path> CompileCommandsDir(
53 "compile-commands-dir",
54 llvm::cl::desc("Specify a path to look for compile_commands.json. If path "
55 "is invalid, clangd will look in the current directory and "
56 "parent paths of each source file."));
57
Ilya Biryukovdb8b2d72017-08-14 08:45:47 +000058static llvm::cl::opt<unsigned>
59 WorkerThreadsCount("j",
60 llvm::cl::desc("Number of async workers used by clangd"),
61 llvm::cl::init(getDefaultAsyncThreadsCount()));
62
Sam McCallc18c2802018-06-15 11:06:29 +000063// FIXME: also support "plain" style where signatures are always omitted.
64enum CompletionStyleFlag {
65 Detailed,
66 Bundled,
67};
68static llvm::cl::opt<CompletionStyleFlag> CompletionStyle(
69 "completion-style",
70 llvm::cl::desc("Granularity of code completion suggestions"),
71 llvm::cl::values(
72 clEnumValN(Detailed, "detailed",
73 "One completion item for each semantically distinct "
74 "completion, with full type information."),
75 clEnumValN(Bundled, "bundled",
76 "Similar completion items (e.g. function overloads) are "
77 "combined. Type information shown where possible.")),
78 llvm::cl::init(Detailed));
79
Sam McCalladccab62017-11-23 16:58:22 +000080// FIXME: Flags are the wrong mechanism for user preferences.
81// We should probably read a dotfile or similar.
82static llvm::cl::opt<bool> IncludeIneligibleResults(
83 "include-ineligible-results",
84 llvm::cl::desc(
85 "Include ineligible completion results (e.g. private members)"),
86 llvm::cl::init(clangd::CodeCompleteOptions().IncludeIneligibleResults),
87 llvm::cl::Hidden);
Ilya Biryukovb33c1572017-09-12 13:57:14 +000088
Sam McCall5ed599e2018-02-06 10:47:30 +000089static llvm::cl::opt<JSONStreamStyle> InputStyle(
90 "input-style", llvm::cl::desc("Input JSON stream encoding"),
91 llvm::cl::values(
92 clEnumValN(JSONStreamStyle::Standard, "standard", "usual LSP protocol"),
93 clEnumValN(JSONStreamStyle::Delimited, "delimited",
94 "messages delimited by --- lines, with # comment support")),
95 llvm::cl::init(JSONStreamStyle::Standard));
96
Sam McCalldd0566b2017-11-06 15:40:30 +000097static llvm::cl::opt<bool>
98 PrettyPrint("pretty", llvm::cl::desc("Pretty-print JSON output"),
99 llvm::cl::init(false));
100
Sam McCall5ed599e2018-02-06 10:47:30 +0000101static llvm::cl::opt<bool> Test(
102 "lit-test",
103 llvm::cl::desc(
104 "Abbreviation for -input-style=delimited -pretty -run-synchronously. "
105 "Intended to simplify lit tests."),
106 llvm::cl::init(false), llvm::cl::Hidden);
107
Ilya Biryukove9eb7f02017-11-16 16:25:18 +0000108static llvm::cl::opt<PCHStorageFlag> PCHStorage(
109 "pch-storage",
110 llvm::cl::desc("Storing PCHs in memory increases memory usages, but may "
111 "improve performance"),
112 llvm::cl::values(
113 clEnumValN(PCHStorageFlag::Disk, "disk", "store PCHs on disk"),
114 clEnumValN(PCHStorageFlag::Memory, "memory", "store PCHs in memory")),
115 llvm::cl::init(PCHStorageFlag::Disk));
116
Marc-Andre Laperleb387b6e2018-04-23 20:00:52 +0000117static llvm::cl::opt<int> LimitResults(
118 "limit-results",
119 llvm::cl::desc("Limit the number of results returned by clangd. "
Haojian Wu48b48652018-01-25 09:20:09 +0000120 "0 means no limit."),
Sam McCallea283c72018-01-30 09:21:30 +0000121 llvm::cl::init(100));
Haojian Wu48b48652018-01-25 09:20:09 +0000122
Ilya Biryukovdb8b2d72017-08-14 08:45:47 +0000123static llvm::cl::opt<bool> RunSynchronously(
124 "run-synchronously",
125 llvm::cl::desc("Parse on main thread. If set, -j is ignored"),
126 llvm::cl::init(false), llvm::cl::Hidden);
Benjamin Kramerf0af3e62017-03-01 16:16:29 +0000127
Ilya Biryukove6dbb582017-10-10 09:08:47 +0000128static llvm::cl::opt<Path>
Krasimir Georgiev0dcb48e2017-07-19 15:43:35 +0000129 ResourceDir("resource-dir",
Ilya Biryukov4ca7d852017-08-02 08:53:48 +0000130 llvm::cl::desc("Directory for system clang headers"),
Krasimir Georgiev0dcb48e2017-07-19 15:43:35 +0000131 llvm::cl::init(""), llvm::cl::Hidden);
132
Ilya Biryukove6dbb582017-10-10 09:08:47 +0000133static llvm::cl::opt<Path> InputMirrorFile(
134 "input-mirror-file",
135 llvm::cl::desc(
136 "Mirror all LSP input to the specified file. Useful for debugging."),
137 llvm::cl::init(""), llvm::cl::Hidden);
138
Marc-Andre Laperleb387b6e2018-04-23 20:00:52 +0000139static llvm::cl::opt<bool> EnableIndex(
140 "index",
141 llvm::cl::desc("Enable index-based features such as global code completion "
142 "and searching for symbols."
143 "Clang uses an index built from symbols in opened files"),
Sam McCallea283c72018-01-30 09:21:30 +0000144 llvm::cl::init(true));
Eric Liubfac8f72017-12-19 18:00:37 +0000145
Haojian Wuba28e9a2018-01-10 14:44:34 +0000146static llvm::cl::opt<Path> YamlSymbolFile(
147 "yaml-symbol-file",
148 llvm::cl::desc(
149 "YAML-format global symbol file to build the static index. Clangd will "
150 "use the static index for global code completion.\n"
151 "WARNING: This option is experimental only, and will be removed "
152 "eventually. Don't rely on it."),
153 llvm::cl::init(""), llvm::cl::Hidden);
154
Benjamin Kramerbb1cdb62017-02-07 10:28:20 +0000155int main(int argc, char *argv[]) {
Sam McCall3ebf7602018-02-20 11:46:39 +0000156 llvm::sys::PrintStackTraceOnErrorSignal(argv[0]);
Sam McCalle72d0972018-06-29 13:24:20 +0000157 llvm::cl::SetVersionPrinter([](llvm::raw_ostream &OS) {
158 OS << clang::getClangToolFullVersion("clangd") << "\n";
159 });
160 llvm::cl::ParseCommandLineOptions(
161 argc, argv,
162 "clangd is a language server that provides IDE-like features to editors. "
163 "\n\nIt should be used via an editor plugin rather than invoked directly."
164 "For more information, see:"
165 "\n\thttps://clang.llvm.org/extra/clangd.html"
166 "\n\thttps://microsoft.github.io/language-server-protocol/");
Sam McCall5ed599e2018-02-06 10:47:30 +0000167 if (Test) {
168 RunSynchronously = true;
169 InputStyle = JSONStreamStyle::Delimited;
170 PrettyPrint = true;
171 }
Ilya Biryukovafb55542017-05-16 14:40:30 +0000172
Ilya Biryukovdb8b2d72017-08-14 08:45:47 +0000173 if (!RunSynchronously && WorkerThreadsCount == 0) {
174 llvm::errs() << "A number of worker threads cannot be 0. Did you mean to "
175 "specify -run-synchronously?";
176 return 1;
177 }
178
Kirill Bobyrevbcaf3802018-02-25 07:21:16 +0000179 if (RunSynchronously) {
180 if (WorkerThreadsCount.getNumOccurrences())
181 llvm::errs() << "Ignoring -j because -run-synchronously is set.\n";
Ilya Biryukovdb8b2d72017-08-14 08:45:47 +0000182 WorkerThreadsCount = 0;
Kirill Bobyrevbcaf3802018-02-25 07:21:16 +0000183 }
Ilya Biryukovdb8b2d72017-08-14 08:45:47 +0000184
Benjamin Kramer74a18952017-10-26 10:07:04 +0000185 // Validate command line arguments.
Ilya Biryukove6dbb582017-10-10 09:08:47 +0000186 llvm::Optional<llvm::raw_fd_ostream> InputMirrorStream;
187 if (!InputMirrorFile.empty()) {
188 std::error_code EC;
Zachary Turner1f67a3c2018-06-07 19:58:58 +0000189 InputMirrorStream.emplace(InputMirrorFile, /*ref*/ EC,
190 llvm::sys::fs::FA_Read | llvm::sys::fs::FA_Write);
Ilya Biryukove6dbb582017-10-10 09:08:47 +0000191 if (EC) {
192 InputMirrorStream.reset();
193 llvm::errs() << "Error while opening an input mirror file: "
194 << EC.message();
195 }
196 }
Ilya Biryukovee27d2e2017-12-14 15:04:59 +0000197
Sam McCalled2717a2018-02-14 03:20:07 +0000198 // Setup tracing facilities if CLANGD_TRACE is set. In practice enabling a
199 // trace flag in your editor's config is annoying, launching with
200 // `CLANGD_TRACE=trace.json vim` is easier.
Sam McCall8567cb32017-11-02 09:21:51 +0000201 llvm::Optional<llvm::raw_fd_ostream> TraceStream;
Ilya Biryukovee27d2e2017-12-14 15:04:59 +0000202 std::unique_ptr<trace::EventTracer> Tracer;
Sam McCalled2717a2018-02-14 03:20:07 +0000203 if (auto *TraceFile = getenv("CLANGD_TRACE")) {
Sam McCall8567cb32017-11-02 09:21:51 +0000204 std::error_code EC;
Zachary Turner1f67a3c2018-06-07 19:58:58 +0000205 TraceStream.emplace(TraceFile, /*ref*/ EC,
206 llvm::sys::fs::FA_Read | llvm::sys::fs::FA_Write);
Sam McCall8567cb32017-11-02 09:21:51 +0000207 if (EC) {
Sam McCalled2717a2018-02-14 03:20:07 +0000208 TraceStream.reset();
209 llvm::errs() << "Error while opening trace file " << TraceFile << ": "
210 << EC.message();
Sam McCall8567cb32017-11-02 09:21:51 +0000211 } else {
Ilya Biryukovee27d2e2017-12-14 15:04:59 +0000212 Tracer = trace::createJSONTracer(*TraceStream, PrettyPrint);
Sam McCall8567cb32017-11-02 09:21:51 +0000213 }
214 }
Ilya Biryukove6dbb582017-10-10 09:08:47 +0000215
Ilya Biryukovee27d2e2017-12-14 15:04:59 +0000216 llvm::Optional<trace::Session> TracingSession;
217 if (Tracer)
218 TracingSession.emplace(*Tracer);
219
Sam McCall7363a2f2018-03-05 17:28:54 +0000220 JSONOutput Out(llvm::outs(), llvm::errs(),
Sam McCalldd0566b2017-11-06 15:40:30 +0000221 InputMirrorStream ? InputMirrorStream.getPointer() : nullptr,
222 PrettyPrint);
Benjamin Kramerbb1cdb62017-02-07 10:28:20 +0000223
Ilya Biryukov940901e2017-12-13 12:51:22 +0000224 clangd::LoggingSession LoggingSession(Out);
225
Ilya Biryukov0c1ca6b2017-10-02 15:13:20 +0000226 // If --compile-commands-dir arg was invoked, check value and override default
227 // path.
Ilya Biryukov0c1ca6b2017-10-02 15:13:20 +0000228 llvm::Optional<Path> CompileCommandsDirPath;
Ilya Biryukov0c1ca6b2017-10-02 15:13:20 +0000229 if (CompileCommandsDir.empty()) {
230 CompileCommandsDirPath = llvm::None;
231 } else if (!llvm::sys::path::is_absolute(CompileCommandsDir) ||
232 !llvm::sys::fs::exists(CompileCommandsDir)) {
233 llvm::errs() << "Path specified by --compile-commands-dir either does not "
234 "exist or is not an absolute "
235 "path. The argument will be ignored.\n";
236 CompileCommandsDirPath = llvm::None;
237 } else {
238 CompileCommandsDirPath = CompileCommandsDir;
239 }
Benjamin Kramer6a3d74e2017-02-07 12:40:59 +0000240
Sam McCall7363a2f2018-03-05 17:28:54 +0000241 ClangdServer::Options Opts;
Ilya Biryukove9eb7f02017-11-16 16:25:18 +0000242 switch (PCHStorage) {
243 case PCHStorageFlag::Memory:
Sam McCall7363a2f2018-03-05 17:28:54 +0000244 Opts.StorePreamblesInMemory = true;
Ilya Biryukove9eb7f02017-11-16 16:25:18 +0000245 break;
246 case PCHStorageFlag::Disk:
Sam McCall7363a2f2018-03-05 17:28:54 +0000247 Opts.StorePreamblesInMemory = false;
Ilya Biryukove9eb7f02017-11-16 16:25:18 +0000248 break;
249 }
Krasimir Georgiev0dcb48e2017-07-19 15:43:35 +0000250 if (!ResourceDir.empty())
Sam McCall7363a2f2018-03-05 17:28:54 +0000251 Opts.ResourceDir = ResourceDir;
Marc-Andre Laperleb387b6e2018-04-23 20:00:52 +0000252 Opts.BuildDynamicSymbolIndex = EnableIndex;
Haojian Wuba28e9a2018-01-10 14:44:34 +0000253 std::unique_ptr<SymbolIndex> StaticIdx;
Marc-Andre Laperleb387b6e2018-04-23 20:00:52 +0000254 if (EnableIndex && !YamlSymbolFile.empty()) {
Kirill Bobyrev5a267ed2018-05-29 11:50:51 +0000255 StaticIdx = buildStaticIndex(YamlSymbolFile);
Sam McCall7363a2f2018-03-05 17:28:54 +0000256 Opts.StaticIndex = StaticIdx.get();
257 }
258 Opts.AsyncThreadsCount = WorkerThreadsCount;
259
Sam McCalladccab62017-11-23 16:58:22 +0000260 clangd::CodeCompleteOptions CCOpts;
Sam McCalladccab62017-11-23 16:58:22 +0000261 CCOpts.IncludeIneligibleResults = IncludeIneligibleResults;
Marc-Andre Laperleb387b6e2018-04-23 20:00:52 +0000262 CCOpts.Limit = LimitResults;
Sam McCallc18c2802018-06-15 11:06:29 +0000263 CCOpts.BundleOverloads = CompletionStyle != Detailed;
Sam McCall7363a2f2018-03-05 17:28:54 +0000264
Benjamin Kramer74a18952017-10-26 10:07:04 +0000265 // Initialize and run ClangdLSPServer.
Sam McCall7363a2f2018-03-05 17:28:54 +0000266 ClangdLSPServer LSPServer(Out, CCOpts, CompileCommandsDirPath, Opts);
Ilya Biryukov0d9b8a32017-10-25 08:45:41 +0000267 constexpr int NoShutdownRequestErrorCode = 1;
Sam McCall8567cb32017-11-02 09:21:51 +0000268 llvm::set_thread_name("clangd.main");
Sam McCall7363a2f2018-03-05 17:28:54 +0000269 // Change stdin to binary to not lose \r\n on windows.
270 llvm::sys::ChangeStdinToBinary();
Sam McCall27a07cf2018-06-05 09:34:46 +0000271 return LSPServer.run(stdin, InputStyle) ? 0 : NoShutdownRequestErrorCode;
Benjamin Kramerbb1cdb62017-02-07 10:28:20 +0000272}