You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			196 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			196 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
|   | //===--- ClangdMain.cpp - clangd server loop ------------------------------===//
 | ||
|  | //
 | ||
|  | //                     The LLVM Compiler Infrastructure
 | ||
|  | //
 | ||
|  | // This file is distributed under the University of Illinois Open Source
 | ||
|  | // License. See LICENSE.TXT for details.
 | ||
|  | //
 | ||
|  | //===----------------------------------------------------------------------===//
 | ||
|  | 
 | ||
|  | #include "ClangdLSPServer.h"
 | ||
|  | #include "JSONRPCDispatcher.h"
 | ||
|  | #include "Path.h"
 | ||
|  | #include "Trace.h"
 | ||
|  | #include "llvm/Support/CommandLine.h"
 | ||
|  | #include "llvm/Support/FileSystem.h"
 | ||
|  | #include "llvm/Support/Path.h"
 | ||
|  | #include "llvm/Support/Program.h"
 | ||
|  | #include "llvm/Support/raw_ostream.h"
 | ||
|  | #include <iostream>
 | ||
|  | #include <memory>
 | ||
|  | #include <string>
 | ||
|  | #include <thread>
 | ||
|  | 
 | ||
|  | using namespace clang; | ||
|  | using namespace clang::clangd; | ||
|  | 
 | ||
|  | namespace { | ||
|  | enum class PCHStorageFlag { Disk, Memory }; | ||
|  | } | ||
|  | 
 | ||
|  | static llvm::cl::opt<Path> CompileCommandsDir( | ||
|  |     "compile-commands-dir", | ||
|  |     llvm::cl::desc("Specify a path to look for compile_commands.json. If path " | ||
|  |                    "is invalid, clangd will look in the current directory and " | ||
|  |                    "parent paths of each source file.")); | ||
|  | 
 | ||
|  | static llvm::cl::opt<unsigned> | ||
|  |     WorkerThreadsCount("j", | ||
|  |                        llvm::cl::desc("Number of async workers used by clangd"), | ||
|  |                        llvm::cl::init(getDefaultAsyncThreadsCount())); | ||
|  | 
 | ||
|  | static llvm::cl::opt<bool> EnableSnippets( | ||
|  |     "enable-snippets", | ||
|  |     llvm::cl::desc( | ||
|  |         "Present snippet completions instead of plaintext completions. " | ||
|  |         "This also enables code pattern results." /* FIXME: should it? */), | ||
|  |     llvm::cl::init(clangd::CodeCompleteOptions().EnableSnippets)); | ||
|  | 
 | ||
|  | // FIXME: Flags are the wrong mechanism for user preferences.
 | ||
|  | // We should probably read a dotfile or similar.
 | ||
|  | static llvm::cl::opt<bool> IncludeIneligibleResults( | ||
|  |     "include-ineligible-results", | ||
|  |     llvm::cl::desc( | ||
|  |         "Include ineligible completion results (e.g. private members)"), | ||
|  |     llvm::cl::init(clangd::CodeCompleteOptions().IncludeIneligibleResults), | ||
|  |     llvm::cl::Hidden); | ||
|  | 
 | ||
|  | static llvm::cl::opt<bool> | ||
|  |     PrettyPrint("pretty", llvm::cl::desc("Pretty-print JSON output"), | ||
|  |                 llvm::cl::init(false)); | ||
|  | 
 | ||
|  | static llvm::cl::opt<PCHStorageFlag> PCHStorage( | ||
|  |     "pch-storage", | ||
|  |     llvm::cl::desc("Storing PCHs in memory increases memory usages, but may " | ||
|  |                    "improve performance"), | ||
|  |     llvm::cl::values( | ||
|  |         clEnumValN(PCHStorageFlag::Disk, "disk", "store PCHs on disk"), | ||
|  |         clEnumValN(PCHStorageFlag::Memory, "memory", "store PCHs in memory")), | ||
|  |     llvm::cl::init(PCHStorageFlag::Disk)); | ||
|  | 
 | ||
|  | static llvm::cl::opt<bool> RunSynchronously( | ||
|  |     "run-synchronously", | ||
|  |     llvm::cl::desc("Parse on main thread. If set, -j is ignored"), | ||
|  |     llvm::cl::init(false), llvm::cl::Hidden); | ||
|  | 
 | ||
|  | static llvm::cl::opt<Path> | ||
|  |     ResourceDir("resource-dir", | ||
|  |                 llvm::cl::desc("Directory for system clang headers"), | ||
|  |                 llvm::cl::init(""), llvm::cl::Hidden); | ||
|  | 
 | ||
|  | static llvm::cl::opt<Path> InputMirrorFile( | ||
|  |     "input-mirror-file", | ||
|  |     llvm::cl::desc( | ||
|  |         "Mirror all LSP input to the specified file. Useful for debugging."), | ||
|  |     llvm::cl::init(""), llvm::cl::Hidden); | ||
|  | 
 | ||
|  | static llvm::cl::opt<Path> TraceFile( | ||
|  |     "trace", | ||
|  |     llvm::cl::desc( | ||
|  |         "Trace internal events and timestamps in chrome://tracing JSON format"), | ||
|  |     llvm::cl::init(""), llvm::cl::Hidden); | ||
|  | 
 | ||
|  | static llvm::cl::opt<bool> EnableIndexBasedCompletion( | ||
|  |     "enable-index-based-completion", | ||
|  |     llvm::cl::desc( | ||
|  |         "Enable index-based global code completion (experimental). Clangd will " | ||
|  |         "use index built from symbols in opened files"), | ||
|  |     llvm::cl::init(false), llvm::cl::Hidden); | ||
|  | 
 | ||
|  | int main(int argc, char *argv[]) { | ||
|  |   llvm::cl::ParseCommandLineOptions(argc, argv, "clangd"); | ||
|  | 
 | ||
|  |   if (!RunSynchronously && WorkerThreadsCount == 0) { | ||
|  |     llvm::errs() << "A number of worker threads cannot be 0. Did you mean to " | ||
|  |                     "specify -run-synchronously?"; | ||
|  |     return 1; | ||
|  |   } | ||
|  | 
 | ||
|  |   // Ignore -j option if -run-synchonously is used.
 | ||
|  |   // FIXME: a warning should be shown here.
 | ||
|  |   if (RunSynchronously) | ||
|  |     WorkerThreadsCount = 0; | ||
|  | 
 | ||
|  |   // Validate command line arguments.
 | ||
|  |   llvm::Optional<llvm::raw_fd_ostream> InputMirrorStream; | ||
|  |   if (!InputMirrorFile.empty()) { | ||
|  |     std::error_code EC; | ||
|  |     InputMirrorStream.emplace(InputMirrorFile, /*ref*/ EC, llvm::sys::fs::F_RW); | ||
|  |     if (EC) { | ||
|  |       InputMirrorStream.reset(); | ||
|  |       llvm::errs() << "Error while opening an input mirror file: " | ||
|  |                    << EC.message(); | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   // Setup tracing facilities.
 | ||
|  |   llvm::Optional<llvm::raw_fd_ostream> TraceStream; | ||
|  |   std::unique_ptr<trace::EventTracer> Tracer; | ||
|  |   if (!TraceFile.empty()) { | ||
|  |     std::error_code EC; | ||
|  |     TraceStream.emplace(TraceFile, /*ref*/ EC, llvm::sys::fs::F_RW); | ||
|  |     if (EC) { | ||
|  |       TraceFile.reset(); | ||
|  |       llvm::errs() << "Error while opening trace file: " << EC.message(); | ||
|  |     } else { | ||
|  |       Tracer = trace::createJSONTracer(*TraceStream, PrettyPrint); | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   llvm::Optional<trace::Session> TracingSession; | ||
|  |   if (Tracer) | ||
|  |     TracingSession.emplace(*Tracer); | ||
|  | 
 | ||
|  |   llvm::raw_ostream &Outs = llvm::outs(); | ||
|  |   llvm::raw_ostream &Logs = llvm::errs(); | ||
|  |   JSONOutput Out(Outs, Logs, | ||
|  |                  InputMirrorStream ? InputMirrorStream.getPointer() : nullptr, | ||
|  |                  PrettyPrint); | ||
|  | 
 | ||
|  |   clangd::LoggingSession LoggingSession(Out); | ||
|  | 
 | ||
|  |   // If --compile-commands-dir arg was invoked, check value and override default
 | ||
|  |   // path.
 | ||
|  |   llvm::Optional<Path> CompileCommandsDirPath; | ||
|  | 
 | ||
|  |   if (CompileCommandsDir.empty()) { | ||
|  |     CompileCommandsDirPath = llvm::None; | ||
|  |   } else if (!llvm::sys::path::is_absolute(CompileCommandsDir) || | ||
|  |              !llvm::sys::fs::exists(CompileCommandsDir)) { | ||
|  |     llvm::errs() << "Path specified by --compile-commands-dir either does not " | ||
|  |                     "exist or is not an absolute " | ||
|  |                     "path. The argument will be ignored.\n"; | ||
|  |     CompileCommandsDirPath = llvm::None; | ||
|  |   } else { | ||
|  |     CompileCommandsDirPath = CompileCommandsDir; | ||
|  |   } | ||
|  | 
 | ||
|  |   bool StorePreamblesInMemory; | ||
|  |   switch (PCHStorage) { | ||
|  |   case PCHStorageFlag::Memory: | ||
|  |     StorePreamblesInMemory = true; | ||
|  |     break; | ||
|  |   case PCHStorageFlag::Disk: | ||
|  |     StorePreamblesInMemory = false; | ||
|  |     break; | ||
|  |   } | ||
|  | 
 | ||
|  |   llvm::Optional<StringRef> ResourceDirRef = None; | ||
|  |   if (!ResourceDir.empty()) | ||
|  |     ResourceDirRef = ResourceDir; | ||
|  | 
 | ||
|  |   // Change stdin to binary to not lose \r\n on windows.
 | ||
|  |   llvm::sys::ChangeStdinToBinary(); | ||
|  | 
 | ||
|  |   clangd::CodeCompleteOptions CCOpts; | ||
|  |   CCOpts.EnableSnippets = EnableSnippets; | ||
|  |   CCOpts.IncludeIneligibleResults = IncludeIneligibleResults; | ||
|  |   // Initialize and run ClangdLSPServer.
 | ||
|  |   ClangdLSPServer LSPServer(Out, WorkerThreadsCount, StorePreamblesInMemory, | ||
|  |                             CCOpts, ResourceDirRef, CompileCommandsDirPath, | ||
|  |                             EnableIndexBasedCompletion); | ||
|  |   constexpr int NoShutdownRequestErrorCode = 1; | ||
|  |   llvm::set_thread_name("clangd.main"); | ||
|  |   return LSPServer.run(std::cin) ? 0 : NoShutdownRequestErrorCode; | ||
|  | } |