mirror of
https://github.com/MaskRay/ccls.git
synced 2025-01-19 12:05:50 +00:00
Introduce MessageHandler abstraction. Mainly just code reorg.
Only the initialize request uses it so far, but this will enable pulling quite a bit of code out of command_line.cc.
This commit is contained in:
parent
8b5d9d33ab
commit
3599a831b1
36
src/cache_loader.cc
Normal file
36
src/cache_loader.cc
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
#include "cache_loader.h"
|
||||||
|
|
||||||
|
#include "cache.h"
|
||||||
|
#include "indexer.h"
|
||||||
|
|
||||||
|
CacheLoader::CacheLoader(Config* config) : config_(config) {}
|
||||||
|
|
||||||
|
IndexFile* CacheLoader::TryLoad(const std::string& path) {
|
||||||
|
auto it = caches.find(path);
|
||||||
|
if (it != caches.end())
|
||||||
|
return it->second.get();
|
||||||
|
|
||||||
|
std::unique_ptr<IndexFile> cache = LoadCachedIndex(config_, path);
|
||||||
|
if (!cache)
|
||||||
|
return nullptr;
|
||||||
|
|
||||||
|
caches[path] = std::move(cache);
|
||||||
|
return caches[path].get();
|
||||||
|
}
|
||||||
|
|
||||||
|
std::unique_ptr<IndexFile> CacheLoader::TryTakeOrLoad(const std::string& path) {
|
||||||
|
auto it = caches.find(path);
|
||||||
|
if (it != caches.end()) {
|
||||||
|
auto result = std::move(it->second);
|
||||||
|
caches.erase(it);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
return LoadCachedIndex(config_, path);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::unique_ptr<IndexFile> CacheLoader::TakeOrLoad(const std::string& path) {
|
||||||
|
auto result = TryTakeOrLoad(path);
|
||||||
|
assert(result);
|
||||||
|
return result;
|
||||||
|
}
|
23
src/cache_loader.h
Normal file
23
src/cache_loader.h
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include <unordered_map>
|
||||||
|
|
||||||
|
// Manages loading caches from file paths for the indexer process.
|
||||||
|
struct CacheLoader {
|
||||||
|
explicit CacheLoader(Config* config);
|
||||||
|
|
||||||
|
IndexFile* TryLoad(const std::string& path);
|
||||||
|
|
||||||
|
// Takes the existing cache or loads the cache at |path|. May return nullptr
|
||||||
|
// if the cache does not exist.
|
||||||
|
std::unique_ptr<IndexFile> TryTakeOrLoad(const std::string& path);
|
||||||
|
|
||||||
|
// Takes the existing cache or loads the cache at |path|. Asserts the cache
|
||||||
|
// exists.
|
||||||
|
std::unique_ptr<IndexFile> TakeOrLoad(const std::string& path);
|
||||||
|
|
||||||
|
std::unordered_map<std::string, std::unique_ptr<IndexFile>> caches;
|
||||||
|
Config* config_;
|
||||||
|
};
|
@ -1,3 +1,5 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
#include "atomic_object.h"
|
#include "atomic_object.h"
|
||||||
#include "clang_index.h"
|
#include "clang_index.h"
|
||||||
#include "clang_translation_unit.h"
|
#include "clang_translation_unit.h"
|
||||||
|
12
src/code_complete_cache.cc
Normal file
12
src/code_complete_cache.cc
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
#include "code_complete_cache.h"
|
||||||
|
|
||||||
|
void CodeCompleteCache::WithLock(std::function<void()> action) {
|
||||||
|
std::lock_guard<std::mutex> lock(mutex_);
|
||||||
|
action();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool CodeCompleteCache::IsCacheValid(lsTextDocumentPositionParams position) {
|
||||||
|
std::lock_guard<std::mutex> lock(mutex_);
|
||||||
|
return cached_path_ == position.textDocument.uri.GetPath() &&
|
||||||
|
cached_completion_position_ == position.position;
|
||||||
|
}
|
25
src/code_complete_cache.h
Normal file
25
src/code_complete_cache.h
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "language_server_api.h"
|
||||||
|
|
||||||
|
#include <optional.h>
|
||||||
|
|
||||||
|
#include <mutex>
|
||||||
|
|
||||||
|
using namespace std::experimental;
|
||||||
|
using std::experimental::nullopt;
|
||||||
|
|
||||||
|
// Cached completion information, so we can give fast completion results when
|
||||||
|
// the user erases a character. vscode will resend the completion request if
|
||||||
|
// that happens.
|
||||||
|
struct CodeCompleteCache {
|
||||||
|
// NOTE: Make sure to access these variables under |WithLock|.
|
||||||
|
optional<std::string> cached_path_;
|
||||||
|
optional<lsPosition> cached_completion_position_;
|
||||||
|
NonElidedVector<lsCompletionItem> cached_results_;
|
||||||
|
|
||||||
|
std::mutex mutex_;
|
||||||
|
|
||||||
|
void WithLock(std::function<void()> action);
|
||||||
|
bool IsCacheValid(lsTextDocumentPositionParams position);
|
||||||
|
};
|
@ -1,5 +1,6 @@
|
|||||||
// TODO: cleanup includes
|
// TODO: cleanup includes
|
||||||
#include "cache.h"
|
#include "cache.h"
|
||||||
|
#include "cache_loader.h"
|
||||||
#include "clang_complete.h"
|
#include "clang_complete.h"
|
||||||
#include "file_consumer.h"
|
#include "file_consumer.h"
|
||||||
#include "include_complete.h"
|
#include "include_complete.h"
|
||||||
@ -9,6 +10,7 @@
|
|||||||
#include "lex_utils.h"
|
#include "lex_utils.h"
|
||||||
#include "lru_cache.h"
|
#include "lru_cache.h"
|
||||||
#include "match.h"
|
#include "match.h"
|
||||||
|
#include "message_handler.h"
|
||||||
#include "options.h"
|
#include "options.h"
|
||||||
#include "platform.h"
|
#include "platform.h"
|
||||||
#include "project.h"
|
#include "project.h"
|
||||||
@ -19,6 +21,7 @@
|
|||||||
#include "test.h"
|
#include "test.h"
|
||||||
#include "threaded_queue.h"
|
#include "threaded_queue.h"
|
||||||
#include "timer.h"
|
#include "timer.h"
|
||||||
|
#include "timestamp_manager.h"
|
||||||
#include "work_thread.h"
|
#include "work_thread.h"
|
||||||
#include "working_files.h"
|
#include "working_files.h"
|
||||||
|
|
||||||
@ -50,35 +53,9 @@ namespace {
|
|||||||
|
|
||||||
std::vector<std::string> kEmptyArgs;
|
std::vector<std::string> kEmptyArgs;
|
||||||
|
|
||||||
// Expected client version. We show an error if this doesn't match.
|
|
||||||
const int kExpectedClientVersion = 3;
|
|
||||||
|
|
||||||
// If true stdout will be printed to stderr.
|
// If true stdout will be printed to stderr.
|
||||||
bool g_log_stdin_stdout_to_stderr = false;
|
bool g_log_stdin_stdout_to_stderr = false;
|
||||||
|
|
||||||
// Cached completion information, so we can give fast completion results when
|
|
||||||
// the user erases a character. vscode will resend the completion request if
|
|
||||||
// that happens.
|
|
||||||
struct CodeCompleteCache {
|
|
||||||
// NOTE: Make sure to access these variables under |WithLock|.
|
|
||||||
optional<std::string> cached_path_;
|
|
||||||
optional<lsPosition> cached_completion_position_;
|
|
||||||
NonElidedVector<lsCompletionItem> cached_results_;
|
|
||||||
|
|
||||||
std::mutex mutex_;
|
|
||||||
|
|
||||||
void WithLock(std::function<void()> action) {
|
|
||||||
std::lock_guard<std::mutex> lock(mutex_);
|
|
||||||
action();
|
|
||||||
}
|
|
||||||
|
|
||||||
bool IsCacheValid(lsTextDocumentPositionParams position) {
|
|
||||||
std::lock_guard<std::mutex> lock(mutex_);
|
|
||||||
return cached_path_ == position.textDocument.uri.GetPath() &&
|
|
||||||
cached_completion_position_ == position.position;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// This function returns true if e2e timing should be displayed for the given
|
// This function returns true if e2e timing should be displayed for the given
|
||||||
// IpcId.
|
// IpcId.
|
||||||
bool ShouldDisplayIpcTiming(IpcId id) {
|
bool ShouldDisplayIpcTiming(IpcId id) {
|
||||||
@ -604,103 +581,8 @@ void FilterCompletionResponse(Out_TextDocumentComplete* complete_response,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct Index_Request {
|
|
||||||
std::string path;
|
|
||||||
// TODO: make |args| a string that is parsed lazily.
|
|
||||||
std::vector<std::string> args;
|
|
||||||
bool is_interactive;
|
|
||||||
optional<std::string> contents; // Preloaded contents. Useful for tests.
|
|
||||||
|
|
||||||
Index_Request(const std::string& path,
|
|
||||||
const std::vector<std::string>& args,
|
|
||||||
bool is_interactive,
|
|
||||||
optional<std::string> contents)
|
|
||||||
: path(path),
|
|
||||||
args(args),
|
|
||||||
is_interactive(is_interactive),
|
|
||||||
contents(contents) {}
|
|
||||||
};
|
|
||||||
|
|
||||||
struct Index_DoIdMap {
|
|
||||||
std::unique_ptr<IndexFile> current;
|
|
||||||
std::unique_ptr<IndexFile> previous;
|
|
||||||
|
|
||||||
PerformanceImportFile perf;
|
|
||||||
bool is_interactive = false;
|
|
||||||
bool write_to_disk = false;
|
|
||||||
bool load_previous = false;
|
|
||||||
|
|
||||||
Index_DoIdMap(std::unique_ptr<IndexFile> current,
|
|
||||||
PerformanceImportFile perf,
|
|
||||||
bool is_interactive,
|
|
||||||
bool write_to_disk)
|
|
||||||
: current(std::move(current)),
|
|
||||||
perf(perf),
|
|
||||||
is_interactive(is_interactive),
|
|
||||||
write_to_disk(write_to_disk) {
|
|
||||||
assert(this->current);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
struct Index_OnIdMapped {
|
|
||||||
struct File {
|
|
||||||
std::unique_ptr<IndexFile> file;
|
|
||||||
std::unique_ptr<IdMap> ids;
|
|
||||||
|
|
||||||
File(std::unique_ptr<IndexFile> file, std::unique_ptr<IdMap> ids)
|
|
||||||
: file(std::move(file)), ids(std::move(ids)) {}
|
|
||||||
};
|
|
||||||
|
|
||||||
std::unique_ptr<File> previous;
|
|
||||||
std::unique_ptr<File> current;
|
|
||||||
|
|
||||||
PerformanceImportFile perf;
|
|
||||||
bool is_interactive;
|
|
||||||
bool write_to_disk;
|
|
||||||
|
|
||||||
Index_OnIdMapped(PerformanceImportFile perf,
|
|
||||||
bool is_interactive,
|
|
||||||
bool write_to_disk)
|
|
||||||
: perf(perf),
|
|
||||||
is_interactive(is_interactive),
|
|
||||||
write_to_disk(write_to_disk) {}
|
|
||||||
};
|
|
||||||
|
|
||||||
struct Index_OnIndexed {
|
|
||||||
IndexUpdate update;
|
|
||||||
PerformanceImportFile perf;
|
|
||||||
|
|
||||||
Index_OnIndexed(IndexUpdate& update, PerformanceImportFile perf)
|
|
||||||
: update(update), perf(perf) {}
|
|
||||||
};
|
|
||||||
|
|
||||||
struct QueueManager {
|
|
||||||
using Index_RequestQueue = ThreadedQueue<Index_Request>;
|
|
||||||
using Index_DoIdMapQueue = ThreadedQueue<Index_DoIdMap>;
|
|
||||||
using Index_OnIdMappedQueue = ThreadedQueue<Index_OnIdMapped>;
|
|
||||||
using Index_OnIndexedQueue = ThreadedQueue<Index_OnIndexed>;
|
|
||||||
|
|
||||||
Index_RequestQueue index_request;
|
|
||||||
Index_DoIdMapQueue do_id_map;
|
|
||||||
Index_DoIdMapQueue load_previous_index;
|
|
||||||
Index_OnIdMappedQueue on_id_mapped;
|
|
||||||
Index_OnIndexedQueue on_indexed;
|
|
||||||
|
|
||||||
QueueManager(MultiQueueWaiter* waiter)
|
|
||||||
: index_request(waiter),
|
|
||||||
do_id_map(waiter),
|
|
||||||
load_previous_index(waiter),
|
|
||||||
on_id_mapped(waiter),
|
|
||||||
on_indexed(waiter) {}
|
|
||||||
|
|
||||||
bool HasWork() {
|
|
||||||
return !index_request.IsEmpty() || !do_id_map.IsEmpty() ||
|
|
||||||
!load_previous_index.IsEmpty() || !on_id_mapped.IsEmpty() ||
|
|
||||||
!on_indexed.IsEmpty();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
void RegisterMessageTypes() {
|
void RegisterMessageTypes() {
|
||||||
|
// TODO: use automatic registration similar to MessageHandler.
|
||||||
MessageRegistry::instance()->Register<Ipc_CancelRequest>();
|
MessageRegistry::instance()->Register<Ipc_CancelRequest>();
|
||||||
MessageRegistry::instance()->Register<Ipc_InitializeRequest>();
|
MessageRegistry::instance()->Register<Ipc_InitializeRequest>();
|
||||||
MessageRegistry::instance()->Register<Ipc_InitializedNotification>();
|
MessageRegistry::instance()->Register<Ipc_InitializedNotification>();
|
||||||
@ -736,136 +618,6 @@ void RegisterMessageTypes() {
|
|||||||
MessageRegistry::instance()->Register<Ipc_CqueryExitWhenIdle>();
|
MessageRegistry::instance()->Register<Ipc_CqueryExitWhenIdle>();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Manages files inside of the indexing pipeline so we don't have the same file
|
|
||||||
// being imported multiple times.
|
|
||||||
//
|
|
||||||
// NOTE: This is not thread safe and should only be used on the querydb thread.
|
|
||||||
struct ImportManager {
|
|
||||||
// Try to mark the given dependency as imported. A dependency can only ever be
|
|
||||||
// imported once.
|
|
||||||
bool TryMarkDependencyImported(const std::string& path) {
|
|
||||||
std::lock_guard<std::mutex> lock(depdency_mutex_);
|
|
||||||
return depdency_imported_.insert(path).second;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Try to import the given file into querydb. We should only ever be
|
|
||||||
// importing a file into querydb once per file. Returns true if the file
|
|
||||||
// can be imported.
|
|
||||||
bool StartQueryDbImport(const std::string& path) {
|
|
||||||
return querydb_processing_.insert(path).second;
|
|
||||||
}
|
|
||||||
|
|
||||||
// The file has been fully imported and can be imported again later on.
|
|
||||||
void DoneQueryDbImport(const std::string& path) {
|
|
||||||
querydb_processing_.erase(path);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Returns true if there any any files currently being imported.
|
|
||||||
bool HasActiveQuerydbImports() { return !querydb_processing_.empty(); }
|
|
||||||
|
|
||||||
std::unordered_set<std::string> querydb_processing_;
|
|
||||||
|
|
||||||
// TODO: use std::shared_mutex so we can have multiple readers.
|
|
||||||
std::mutex depdency_mutex_;
|
|
||||||
std::unordered_set<std::string> depdency_imported_;
|
|
||||||
};
|
|
||||||
|
|
||||||
// Manages loading caches from file paths for the indexer process.
|
|
||||||
struct CacheLoader {
|
|
||||||
explicit CacheLoader(Config* config) : config_(config) {}
|
|
||||||
|
|
||||||
IndexFile* TryLoad(const std::string& path) {
|
|
||||||
auto it = caches.find(path);
|
|
||||||
if (it != caches.end())
|
|
||||||
return it->second.get();
|
|
||||||
|
|
||||||
std::unique_ptr<IndexFile> cache = LoadCachedIndex(config_, path);
|
|
||||||
if (!cache)
|
|
||||||
return nullptr;
|
|
||||||
|
|
||||||
caches[path] = std::move(cache);
|
|
||||||
return caches[path].get();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Takes the existing cache or loads the cache at |path|. May return nullptr
|
|
||||||
// if the cache does not exist.
|
|
||||||
std::unique_ptr<IndexFile> TryTakeOrLoad(const std::string& path) {
|
|
||||||
auto it = caches.find(path);
|
|
||||||
if (it != caches.end()) {
|
|
||||||
auto result = std::move(it->second);
|
|
||||||
caches.erase(it);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
return LoadCachedIndex(config_, path);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Takes the existing cache or loads the cache at |path|. Asserts the cache
|
|
||||||
// exists.
|
|
||||||
std::unique_ptr<IndexFile> TakeOrLoad(const std::string& path) {
|
|
||||||
auto result = TryTakeOrLoad(path);
|
|
||||||
assert(result);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::unordered_map<std::string, std::unique_ptr<IndexFile>> caches;
|
|
||||||
Config* config_;
|
|
||||||
};
|
|
||||||
|
|
||||||
// Caches timestamps of cc files so we can avoid a filesystem reads. This is
|
|
||||||
// important for import perf, as during dependency checking the same files are
|
|
||||||
// checked over and over again if they are common headers.
|
|
||||||
struct TimestampManager {
|
|
||||||
optional<int64_t> GetLastCachedModificationTime(CacheLoader* cache_loader,
|
|
||||||
const std::string& path) {
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> guard(mutex_);
|
|
||||||
auto it = timestamps_.find(path);
|
|
||||||
if (it != timestamps_.end())
|
|
||||||
return it->second;
|
|
||||||
}
|
|
||||||
IndexFile* file = cache_loader->TryLoad(path);
|
|
||||||
if (!file)
|
|
||||||
return nullopt;
|
|
||||||
|
|
||||||
UpdateCachedModificationTime(path, file->last_modification_time);
|
|
||||||
return file->last_modification_time;
|
|
||||||
}
|
|
||||||
|
|
||||||
void UpdateCachedModificationTime(const std::string& path,
|
|
||||||
int64_t timestamp) {
|
|
||||||
std::lock_guard<std::mutex> guard(mutex_);
|
|
||||||
timestamps_[path] = timestamp;
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO: use std::shared_mutex so we can have multiple readers.
|
|
||||||
std::mutex mutex_;
|
|
||||||
std::unordered_map<std::string, int64_t> timestamps_;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct IndexManager {
|
|
||||||
std::unordered_set<std::string> files_being_indexed_;
|
|
||||||
std::mutex mutex_;
|
|
||||||
|
|
||||||
// Marks a file as being indexed. Returns true if the file is not already
|
|
||||||
// being indexed.
|
|
||||||
bool MarkIndex(const std::string& path) {
|
|
||||||
std::lock_guard<std::mutex> lock(mutex_);
|
|
||||||
|
|
||||||
return files_being_indexed_.insert(path).second;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Unmarks a file as being indexed, so it can get indexed again in the
|
|
||||||
// future.
|
|
||||||
void ClearIndex(const std::string& path) {
|
|
||||||
std::lock_guard<std::mutex> lock(mutex_);
|
|
||||||
|
|
||||||
auto it = files_being_indexed_.find(path);
|
|
||||||
assert(it != files_being_indexed_.end());
|
|
||||||
files_being_indexed_.erase(it);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// Send indexing progress to client if reporting is enabled.
|
// Send indexing progress to client if reporting is enabled.
|
||||||
void EmitProgress(Config* config, QueueManager* queue) {
|
void EmitProgress(Config* config, QueueManager* queue) {
|
||||||
if (config->enableProgressReports) {
|
if (config->enableProgressReports) {
|
||||||
@ -1487,183 +1239,17 @@ bool QueryDbMainLoop(Config* config,
|
|||||||
for (auto& message : messages) {
|
for (auto& message : messages) {
|
||||||
did_work = true;
|
did_work = true;
|
||||||
|
|
||||||
switch (message->method_id) {
|
for (MessageHandler* handler : *MessageHandler::message_handlers) {
|
||||||
case IpcId::Initialize: {
|
if (handler->GetId() == message->method_id) {
|
||||||
auto request = message->As<Ipc_InitializeRequest>();
|
handler->Run(std::move(message));
|
||||||
|
|
||||||
// Log initialization parameters.
|
|
||||||
rapidjson::StringBuffer output;
|
|
||||||
Writer writer(output);
|
|
||||||
Reflect(writer, request->params.initializationOptions);
|
|
||||||
LOG_S(INFO) << "Init parameters: " << output.GetString();
|
|
||||||
|
|
||||||
if (request->params.rootUri) {
|
|
||||||
std::string project_path = request->params.rootUri->GetPath();
|
|
||||||
LOG_S(INFO) << "[querydb] Initialize in directory " << project_path
|
|
||||||
<< " with uri " << request->params.rootUri->raw_uri;
|
|
||||||
|
|
||||||
if (!request->params.initializationOptions) {
|
|
||||||
LOG_S(FATAL) << "Initialization parameters (particularily "
|
|
||||||
"cacheDirectory) are required";
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
*config = *request->params.initializationOptions;
|
|
||||||
|
|
||||||
// Check client version.
|
|
||||||
if (config->clientVersion.has_value() &&
|
|
||||||
*config->clientVersion != kExpectedClientVersion) {
|
|
||||||
Out_ShowLogMessage out;
|
|
||||||
out.display_type = Out_ShowLogMessage::DisplayType::Show;
|
|
||||||
out.params.type = lsMessageType::Error;
|
|
||||||
out.params.message =
|
|
||||||
"cquery client (v" + std::to_string(*config->clientVersion) +
|
|
||||||
") and server (v" + std::to_string(kExpectedClientVersion) +
|
|
||||||
") version mismatch. Please update ";
|
|
||||||
if (config->clientVersion > kExpectedClientVersion)
|
|
||||||
out.params.message += "the cquery binary.";
|
|
||||||
else
|
|
||||||
out.params.message +=
|
|
||||||
"your extension client (VSIX file). Make sure to uninstall "
|
|
||||||
"the cquery extension and restart vscode before "
|
|
||||||
"reinstalling.";
|
|
||||||
out.Write(std::cout);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Make sure cache directory is valid.
|
|
||||||
if (config->cacheDirectory.empty()) {
|
|
||||||
LOG_S(FATAL) << "Exiting; no cache directory";
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
config->cacheDirectory = NormalizePath(config->cacheDirectory);
|
|
||||||
EnsureEndsInSlash(config->cacheDirectory);
|
|
||||||
|
|
||||||
// Ensure there is a resource directory.
|
|
||||||
if (config->resourceDirectory.empty()) {
|
|
||||||
config->resourceDirectory = GetWorkingDirectory();
|
|
||||||
#if defined(_WIN32)
|
|
||||||
config->resourceDirectory +=
|
|
||||||
std::string("../../clang_resource_dir/");
|
|
||||||
#else
|
|
||||||
config->resourceDirectory += std::string("../clang_resource_dir/");
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
config->resourceDirectory = NormalizePath(config->resourceDirectory);
|
|
||||||
LOG_S(INFO) << "Using -resource-dir=" << config->resourceDirectory;
|
|
||||||
|
|
||||||
// Send initialization before starting indexers, so we don't send a
|
|
||||||
// status update too early.
|
|
||||||
// TODO: query request->params.capabilities.textDocument and support
|
|
||||||
// only things the client supports.
|
|
||||||
|
|
||||||
Out_InitializeResponse out;
|
|
||||||
out.id = request->id;
|
|
||||||
|
|
||||||
// out.result.capabilities.textDocumentSync =
|
|
||||||
// lsTextDocumentSyncOptions();
|
|
||||||
// out.result.capabilities.textDocumentSync->openClose = true;
|
|
||||||
// out.result.capabilities.textDocumentSync->change =
|
|
||||||
// lsTextDocumentSyncKind::Full;
|
|
||||||
// out.result.capabilities.textDocumentSync->willSave = true;
|
|
||||||
// out.result.capabilities.textDocumentSync->willSaveWaitUntil =
|
|
||||||
// true;
|
|
||||||
out.result.capabilities.textDocumentSync =
|
|
||||||
lsTextDocumentSyncKind::Incremental;
|
|
||||||
|
|
||||||
out.result.capabilities.renameProvider = true;
|
|
||||||
|
|
||||||
out.result.capabilities.completionProvider = lsCompletionOptions();
|
|
||||||
out.result.capabilities.completionProvider->resolveProvider = false;
|
|
||||||
// vscode doesn't support trigger character sequences, so we use ':'
|
|
||||||
// for
|
|
||||||
// '::' and '>' for '->'. See
|
|
||||||
// https://github.com/Microsoft/language-server-protocol/issues/138.
|
|
||||||
out.result.capabilities.completionProvider->triggerCharacters = {
|
|
||||||
".", ":", ">", "#"};
|
|
||||||
|
|
||||||
out.result.capabilities.signatureHelpProvider =
|
|
||||||
lsSignatureHelpOptions();
|
|
||||||
// NOTE: If updating signature help tokens make sure to also update
|
|
||||||
// WorkingFile::FindClosestCallNameInBuffer.
|
|
||||||
out.result.capabilities.signatureHelpProvider->triggerCharacters = {
|
|
||||||
"(", ","};
|
|
||||||
|
|
||||||
out.result.capabilities.codeLensProvider = lsCodeLensOptions();
|
|
||||||
out.result.capabilities.codeLensProvider->resolveProvider = false;
|
|
||||||
|
|
||||||
out.result.capabilities.definitionProvider = true;
|
|
||||||
out.result.capabilities.documentHighlightProvider = true;
|
|
||||||
out.result.capabilities.hoverProvider = true;
|
|
||||||
out.result.capabilities.referencesProvider = true;
|
|
||||||
|
|
||||||
out.result.capabilities.codeActionProvider = true;
|
|
||||||
|
|
||||||
out.result.capabilities.documentSymbolProvider = true;
|
|
||||||
out.result.capabilities.workspaceSymbolProvider = true;
|
|
||||||
|
|
||||||
out.result.capabilities.documentLinkProvider =
|
|
||||||
lsDocumentLinkOptions();
|
|
||||||
out.result.capabilities.documentLinkProvider->resolveProvider = false;
|
|
||||||
|
|
||||||
IpcManager::WriteStdout(IpcId::Initialize, out);
|
|
||||||
|
|
||||||
// Set project root.
|
|
||||||
config->projectRoot =
|
|
||||||
NormalizePath(request->params.rootUri->GetPath());
|
|
||||||
EnsureEndsInSlash(config->projectRoot);
|
|
||||||
MakeDirectoryRecursive(config->cacheDirectory +
|
|
||||||
EscapeFileName(config->projectRoot));
|
|
||||||
|
|
||||||
// Start indexer threads.
|
|
||||||
if (config->indexerCount == 0) {
|
|
||||||
// If the user has not specified how many indexers to run, try to
|
|
||||||
// guess an appropriate value. Default to 80% utilization.
|
|
||||||
const float kDefaultTargetUtilization = 0.8;
|
|
||||||
config->indexerCount =
|
|
||||||
std::thread::hardware_concurrency() * kDefaultTargetUtilization;
|
|
||||||
if (config->indexerCount <= 0)
|
|
||||||
config->indexerCount = 1;
|
|
||||||
}
|
|
||||||
LOG_S(INFO) << "Starting " << config->indexerCount << " indexers";
|
|
||||||
for (int i = 0; i < config->indexerCount; ++i) {
|
|
||||||
WorkThread::StartThread("indexer" + std::to_string(i), [=]() {
|
|
||||||
return IndexMain(config, file_consumer_shared, timestamp_manager,
|
|
||||||
import_manager, project, working_files, waiter,
|
|
||||||
queue);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
Timer time;
|
|
||||||
|
|
||||||
// Open up / load the project.
|
|
||||||
project->Load(config->extraClangArguments,
|
|
||||||
config->compilationDatabaseDirectory, project_path,
|
|
||||||
config->resourceDirectory);
|
|
||||||
time.ResetAndPrint("[perf] Loaded compilation entries (" +
|
|
||||||
std::to_string(project->entries.size()) +
|
|
||||||
" files)");
|
|
||||||
|
|
||||||
// Start scanning include directories before dispatching project
|
|
||||||
// files, because that takes a long time.
|
|
||||||
include_complete->Rescan();
|
|
||||||
|
|
||||||
time.Reset();
|
|
||||||
project->ForAllFilteredFiles(
|
|
||||||
config, [&](int i, const Project::Entry& entry) {
|
|
||||||
bool is_interactive =
|
|
||||||
working_files->GetFileByFilename(entry.filename) != nullptr;
|
|
||||||
queue->index_request.Enqueue(Index_Request(
|
|
||||||
entry.filename, entry.args, is_interactive, nullopt));
|
|
||||||
});
|
|
||||||
|
|
||||||
// We need to support multiple concurrent index processes.
|
|
||||||
time.ResetAndPrint("[perf] Dispatched initial index requests");
|
|
||||||
}
|
|
||||||
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
if (!message)
|
||||||
|
continue;
|
||||||
|
// FIXME: assert(!message), ie, verify that a handler was run.
|
||||||
|
|
||||||
|
switch (message->method_id) {
|
||||||
case IpcId::Exit: {
|
case IpcId::Exit: {
|
||||||
LOG_S(INFO) << "Exiting; got IpcId::Exit";
|
LOG_S(INFO) << "Exiting; got IpcId::Exit";
|
||||||
exit(0);
|
exit(0);
|
||||||
@ -3036,10 +2622,30 @@ void RunQueryDbThread(const std::string& bin_name,
|
|||||||
auto signature_cache = MakeUnique<CodeCompleteCache>();
|
auto signature_cache = MakeUnique<CodeCompleteCache>();
|
||||||
ImportManager import_manager;
|
ImportManager import_manager;
|
||||||
TimestampManager timestamp_manager;
|
TimestampManager timestamp_manager;
|
||||||
|
QueryDatabase db;
|
||||||
|
|
||||||
|
// Setup shared references.
|
||||||
|
for (MessageHandler* handler : *MessageHandler::message_handlers) {
|
||||||
|
handler->config = config;
|
||||||
|
handler->db = &db;
|
||||||
|
handler->exit_when_idle = &exit_when_idle;
|
||||||
|
handler->waiter = waiter;
|
||||||
|
handler->queue = queue;
|
||||||
|
handler->project = &project;
|
||||||
|
handler->file_consumer_shared = &file_consumer_shared;
|
||||||
|
handler->import_manager = &import_manager;
|
||||||
|
handler->timestamp_manager = ×tamp_manager;
|
||||||
|
handler->working_files = &working_files;
|
||||||
|
handler->clang_complete = &clang_complete;
|
||||||
|
handler->include_complete = &include_complete;
|
||||||
|
handler->global_code_complete_cache = global_code_complete_cache.get();
|
||||||
|
handler->non_global_code_complete_cache =
|
||||||
|
non_global_code_complete_cache.get();
|
||||||
|
handler->signature_cache = signature_cache.get();
|
||||||
|
}
|
||||||
|
|
||||||
// Run query db main loop.
|
// Run query db main loop.
|
||||||
SetCurrentThreadName("querydb");
|
SetCurrentThreadName("querydb");
|
||||||
QueryDatabase db;
|
|
||||||
while (true) {
|
while (true) {
|
||||||
bool did_work = QueryDbMainLoop(
|
bool did_work = QueryDbMainLoop(
|
||||||
config, &db, &exit_when_idle, waiter, queue, &project,
|
config, &db, &exit_when_idle, waiter, queue, &project,
|
||||||
|
@ -107,3 +107,6 @@ MAKE_REFLECT_STRUCT(Config,
|
|||||||
|
|
||||||
clientVersion,
|
clientVersion,
|
||||||
enableSnippetInsertion);
|
enableSnippetInsertion);
|
||||||
|
|
||||||
|
// Expected client version. We show an error if this doesn't match.
|
||||||
|
constexpr const int kExpectedClientVersion = 3;
|
22
src/entry_points.h
Normal file
22
src/entry_points.h
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
#include "file_consumer.h"
|
||||||
|
#include "import_manager.h"
|
||||||
|
#include "ipc_manager.h"
|
||||||
|
#include "project.h"
|
||||||
|
#include "threaded_queue.h"
|
||||||
|
#include "timestamp_manager.h"
|
||||||
|
#include "work_thread.h"
|
||||||
|
#include "working_files.h"
|
||||||
|
|
||||||
|
// Contains declarations for some of the thread-main functions.
|
||||||
|
|
||||||
|
WorkThread::Result IndexMain(Config* config,
|
||||||
|
FileConsumer::SharedState* file_consumer_shared,
|
||||||
|
TimestampManager* timestamp_manager,
|
||||||
|
ImportManager* import_manager,
|
||||||
|
Project* project,
|
||||||
|
WorkingFiles* working_files,
|
||||||
|
MultiQueueWaiter* waiter,
|
||||||
|
QueueManager* queue);
|
18
src/import_manager.cc
Normal file
18
src/import_manager.cc
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
#include "import_manager.h"
|
||||||
|
|
||||||
|
bool ImportManager::TryMarkDependencyImported(const std::string& path) {
|
||||||
|
std::lock_guard<std::mutex> lock(depdency_mutex_);
|
||||||
|
return depdency_imported_.insert(path).second;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ImportManager::StartQueryDbImport(const std::string& path) {
|
||||||
|
return querydb_processing_.insert(path).second;
|
||||||
|
}
|
||||||
|
|
||||||
|
void ImportManager::DoneQueryDbImport(const std::string& path) {
|
||||||
|
querydb_processing_.erase(path);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ImportManager::HasActiveQuerydbImports() {
|
||||||
|
return !querydb_processing_.empty();
|
||||||
|
}
|
32
src/import_manager.h
Normal file
32
src/import_manager.h
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <mutex>
|
||||||
|
#include <string>
|
||||||
|
#include <unordered_set>
|
||||||
|
|
||||||
|
// Manages files inside of the indexing pipeline so we don't have the same file
|
||||||
|
// being imported multiple times.
|
||||||
|
//
|
||||||
|
// NOTE: This is not thread safe and should only be used on the querydb thread.
|
||||||
|
struct ImportManager {
|
||||||
|
// Try to mark the given dependency as imported. A dependency can only ever be
|
||||||
|
// imported once.
|
||||||
|
bool TryMarkDependencyImported(const std::string& path);
|
||||||
|
|
||||||
|
// Try to import the given file into querydb. We should only ever be
|
||||||
|
// importing a file into querydb once per file. Returns true if the file
|
||||||
|
// can be imported.
|
||||||
|
bool StartQueryDbImport(const std::string& path);
|
||||||
|
|
||||||
|
// The file has been fully imported and can be imported again later on.
|
||||||
|
void DoneQueryDbImport(const std::string& path);
|
||||||
|
|
||||||
|
// Returns true if there any any files currently being imported.
|
||||||
|
bool HasActiveQuerydbImports();
|
||||||
|
|
||||||
|
std::unordered_set<std::string> querydb_processing_;
|
||||||
|
|
||||||
|
// TODO: use std::shared_mutex so we can have multiple readers.
|
||||||
|
std::mutex depdency_mutex_;
|
||||||
|
std::unordered_set<std::string> depdency_imported_;
|
||||||
|
};
|
@ -1,6 +1,7 @@
|
|||||||
#include "ipc_manager.h"
|
#include "ipc_manager.h"
|
||||||
|
|
||||||
#include "language_server_api.h"
|
#include "language_server_api.h"
|
||||||
|
#include "query.h"
|
||||||
|
|
||||||
#include <sstream>
|
#include <sstream>
|
||||||
|
|
||||||
@ -29,3 +30,51 @@ void IpcManager::WriteStdout(IpcId id, lsBaseOutMessage& response) {
|
|||||||
|
|
||||||
IpcManager::IpcManager(MultiQueueWaiter* waiter)
|
IpcManager::IpcManager(MultiQueueWaiter* waiter)
|
||||||
: for_stdout(waiter), for_querydb(waiter) {}
|
: for_stdout(waiter), for_querydb(waiter) {}
|
||||||
|
|
||||||
|
Index_Request::Index_Request(const std::string& path,
|
||||||
|
const std::vector<std::string>& args,
|
||||||
|
bool is_interactive,
|
||||||
|
optional<std::string> contents)
|
||||||
|
: path(path),
|
||||||
|
args(args),
|
||||||
|
is_interactive(is_interactive),
|
||||||
|
contents(contents) {}
|
||||||
|
|
||||||
|
Index_DoIdMap::Index_DoIdMap(std::unique_ptr<IndexFile> current,
|
||||||
|
PerformanceImportFile perf,
|
||||||
|
bool is_interactive,
|
||||||
|
bool write_to_disk)
|
||||||
|
: current(std::move(current)),
|
||||||
|
perf(perf),
|
||||||
|
is_interactive(is_interactive),
|
||||||
|
write_to_disk(write_to_disk) {
|
||||||
|
assert(this->current);
|
||||||
|
}
|
||||||
|
|
||||||
|
Index_OnIdMapped::File::File(std::unique_ptr<IndexFile> file,
|
||||||
|
std::unique_ptr<IdMap> ids)
|
||||||
|
: file(std::move(file)), ids(std::move(ids)) {}
|
||||||
|
|
||||||
|
Index_OnIdMapped::Index_OnIdMapped(PerformanceImportFile perf,
|
||||||
|
bool is_interactive,
|
||||||
|
bool write_to_disk)
|
||||||
|
: perf(perf),
|
||||||
|
is_interactive(is_interactive),
|
||||||
|
write_to_disk(write_to_disk) {}
|
||||||
|
|
||||||
|
Index_OnIndexed::Index_OnIndexed(IndexUpdate& update,
|
||||||
|
PerformanceImportFile perf)
|
||||||
|
: update(update), perf(perf) {}
|
||||||
|
|
||||||
|
QueueManager::QueueManager(MultiQueueWaiter* waiter)
|
||||||
|
: index_request(waiter),
|
||||||
|
do_id_map(waiter),
|
||||||
|
load_previous_index(waiter),
|
||||||
|
on_id_mapped(waiter),
|
||||||
|
on_indexed(waiter) {}
|
||||||
|
|
||||||
|
bool QueueManager::HasWork() {
|
||||||
|
return !index_request.IsEmpty() || !do_id_map.IsEmpty() ||
|
||||||
|
!load_previous_index.IsEmpty() || !on_id_mapped.IsEmpty() ||
|
||||||
|
!on_indexed.IsEmpty();
|
||||||
|
}
|
||||||
|
@ -1,10 +1,14 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "ipc.h"
|
#include "ipc.h"
|
||||||
|
#include "performance.h"
|
||||||
|
#include "query.h"
|
||||||
#include "threaded_queue.h"
|
#include "threaded_queue.h"
|
||||||
|
|
||||||
#include <memory>
|
#include <memory>
|
||||||
|
|
||||||
|
// TODO/FIXME: Merge IpcManager and QueueManager.
|
||||||
|
|
||||||
struct lsBaseOutMessage;
|
struct lsBaseOutMessage;
|
||||||
|
|
||||||
struct IpcManager {
|
struct IpcManager {
|
||||||
@ -26,3 +30,75 @@ struct IpcManager {
|
|||||||
|
|
||||||
static IpcManager* instance_;
|
static IpcManager* instance_;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct Index_Request {
|
||||||
|
std::string path;
|
||||||
|
// TODO: make |args| a string that is parsed lazily.
|
||||||
|
std::vector<std::string> args;
|
||||||
|
bool is_interactive;
|
||||||
|
optional<std::string> contents; // Preloaded contents. Useful for tests.
|
||||||
|
|
||||||
|
Index_Request(const std::string& path,
|
||||||
|
const std::vector<std::string>& args,
|
||||||
|
bool is_interactive,
|
||||||
|
optional<std::string> contents);
|
||||||
|
};
|
||||||
|
|
||||||
|
struct Index_DoIdMap {
|
||||||
|
std::unique_ptr<IndexFile> current;
|
||||||
|
std::unique_ptr<IndexFile> previous;
|
||||||
|
|
||||||
|
PerformanceImportFile perf;
|
||||||
|
bool is_interactive = false;
|
||||||
|
bool write_to_disk = false;
|
||||||
|
bool load_previous = false;
|
||||||
|
|
||||||
|
Index_DoIdMap(std::unique_ptr<IndexFile> current,
|
||||||
|
PerformanceImportFile perf,
|
||||||
|
bool is_interactive,
|
||||||
|
bool write_to_disk);
|
||||||
|
};
|
||||||
|
|
||||||
|
struct Index_OnIdMapped {
|
||||||
|
struct File {
|
||||||
|
std::unique_ptr<IndexFile> file;
|
||||||
|
std::unique_ptr<IdMap> ids;
|
||||||
|
|
||||||
|
File(std::unique_ptr<IndexFile> file, std::unique_ptr<IdMap> ids);
|
||||||
|
};
|
||||||
|
|
||||||
|
std::unique_ptr<File> previous;
|
||||||
|
std::unique_ptr<File> current;
|
||||||
|
|
||||||
|
PerformanceImportFile perf;
|
||||||
|
bool is_interactive;
|
||||||
|
bool write_to_disk;
|
||||||
|
|
||||||
|
Index_OnIdMapped(PerformanceImportFile perf,
|
||||||
|
bool is_interactive,
|
||||||
|
bool write_to_disk);
|
||||||
|
};
|
||||||
|
|
||||||
|
struct Index_OnIndexed {
|
||||||
|
IndexUpdate update;
|
||||||
|
PerformanceImportFile perf;
|
||||||
|
|
||||||
|
Index_OnIndexed(IndexUpdate& update, PerformanceImportFile perf);
|
||||||
|
};
|
||||||
|
|
||||||
|
struct QueueManager {
|
||||||
|
using Index_RequestQueue = ThreadedQueue<Index_Request>;
|
||||||
|
using Index_DoIdMapQueue = ThreadedQueue<Index_DoIdMap>;
|
||||||
|
using Index_OnIdMappedQueue = ThreadedQueue<Index_OnIdMapped>;
|
||||||
|
using Index_OnIndexedQueue = ThreadedQueue<Index_OnIndexed>;
|
||||||
|
|
||||||
|
Index_RequestQueue index_request;
|
||||||
|
Index_DoIdMapQueue do_id_map;
|
||||||
|
Index_DoIdMapQueue load_previous_index;
|
||||||
|
Index_OnIdMappedQueue on_id_mapped;
|
||||||
|
Index_OnIndexedQueue on_indexed;
|
||||||
|
|
||||||
|
QueueManager(MultiQueueWaiter* waiter);
|
||||||
|
|
||||||
|
bool HasWork();
|
||||||
|
};
|
||||||
|
12
src/message_handler.cc
Normal file
12
src/message_handler.cc
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
#include "message_handler.h"
|
||||||
|
|
||||||
|
MessageHandler::MessageHandler() {
|
||||||
|
// Dynamically allocate |message_handlers|, otherwise there will be static
|
||||||
|
// initialization order races.
|
||||||
|
if (!message_handlers)
|
||||||
|
message_handlers = new std::vector<MessageHandler*>();
|
||||||
|
message_handlers->push_back(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
// static
|
||||||
|
std::vector<MessageHandler*>* MessageHandler::message_handlers = nullptr;
|
64
src/message_handler.h
Normal file
64
src/message_handler.h
Normal file
@ -0,0 +1,64 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "cache_loader.h"
|
||||||
|
#include "clang_complete.h"
|
||||||
|
#include "code_complete_cache.h"
|
||||||
|
#include "config.h"
|
||||||
|
#include "import_manager.h"
|
||||||
|
#include "include_complete.h"
|
||||||
|
#include "ipc_manager.h"
|
||||||
|
#include "project.h"
|
||||||
|
#include "query.h"
|
||||||
|
#include "threaded_queue.h"
|
||||||
|
#include "timestamp_manager.h"
|
||||||
|
#include "working_files.h"
|
||||||
|
|
||||||
|
// Usage:
|
||||||
|
//
|
||||||
|
// struct FooHandler : MessageHandler {
|
||||||
|
// // ...
|
||||||
|
// };
|
||||||
|
// REGISTER_MESSAGE_HANDLER(FooHandler);
|
||||||
|
//
|
||||||
|
// Then there will be a global FooHandler instance in
|
||||||
|
// |MessageHandler::message_handlers|.
|
||||||
|
|
||||||
|
#define REGISTER_MESSAGE_HANDLER(type) \
|
||||||
|
static type type##message_handler_instance_;
|
||||||
|
|
||||||
|
struct MessageHandler {
|
||||||
|
Config* config = nullptr;
|
||||||
|
QueryDatabase* db = nullptr;
|
||||||
|
bool* exit_when_idle = nullptr;
|
||||||
|
MultiQueueWaiter* waiter = nullptr;
|
||||||
|
QueueManager* queue = nullptr;
|
||||||
|
Project* project = nullptr;
|
||||||
|
FileConsumer::SharedState* file_consumer_shared = nullptr;
|
||||||
|
ImportManager* import_manager = nullptr;
|
||||||
|
TimestampManager* timestamp_manager = nullptr;
|
||||||
|
WorkingFiles* working_files = nullptr;
|
||||||
|
ClangCompleteManager* clang_complete = nullptr;
|
||||||
|
IncludeComplete* include_complete = nullptr;
|
||||||
|
CodeCompleteCache* global_code_complete_cache = nullptr;
|
||||||
|
CodeCompleteCache* non_global_code_complete_cache = nullptr;
|
||||||
|
CodeCompleteCache* signature_cache = nullptr;
|
||||||
|
|
||||||
|
virtual IpcId GetId() const = 0;
|
||||||
|
virtual void Run(std::unique_ptr<BaseIpcMessage> message) = 0;
|
||||||
|
|
||||||
|
static std::vector<MessageHandler*>* message_handlers;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
MessageHandler();
|
||||||
|
};
|
||||||
|
|
||||||
|
template <typename TMessage>
|
||||||
|
struct BaseMessageHandler : MessageHandler {
|
||||||
|
virtual void Run(TMessage* message) = 0;
|
||||||
|
|
||||||
|
// MessageHandler:
|
||||||
|
IpcId GetId() const override { return TMessage::kIpcId; }
|
||||||
|
void Run(std::unique_ptr<BaseIpcMessage> message) override {
|
||||||
|
Run(message->As<TMessage>());
|
||||||
|
}
|
||||||
|
};
|
176
src/messages/initialize.cc
Normal file
176
src/messages/initialize.cc
Normal file
@ -0,0 +1,176 @@
|
|||||||
|
#include "entry_points.h"
|
||||||
|
#include "message_handler.h"
|
||||||
|
#include "platform.h"
|
||||||
|
#include "timer.h"
|
||||||
|
|
||||||
|
#include <loguru.hpp>
|
||||||
|
|
||||||
|
struct InitializeHandler : BaseMessageHandler<Ipc_InitializeRequest> {
|
||||||
|
void Run(Ipc_InitializeRequest* request) override {
|
||||||
|
// Log initialization parameters.
|
||||||
|
rapidjson::StringBuffer output;
|
||||||
|
Writer writer(output);
|
||||||
|
Reflect(writer, request->params.initializationOptions);
|
||||||
|
LOG_S(INFO) << "Init parameters: " << output.GetString();
|
||||||
|
|
||||||
|
if (request->params.rootUri) {
|
||||||
|
std::string project_path = request->params.rootUri->GetPath();
|
||||||
|
LOG_S(INFO) << "[querydb] Initialize in directory " << project_path
|
||||||
|
<< " with uri " << request->params.rootUri->raw_uri;
|
||||||
|
|
||||||
|
if (!request->params.initializationOptions) {
|
||||||
|
LOG_S(FATAL) << "Initialization parameters (particularily "
|
||||||
|
"cacheDirectory) are required";
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
*config = *request->params.initializationOptions;
|
||||||
|
|
||||||
|
// Check client version.
|
||||||
|
if (config->clientVersion.has_value() &&
|
||||||
|
*config->clientVersion != kExpectedClientVersion) {
|
||||||
|
Out_ShowLogMessage out;
|
||||||
|
out.display_type = Out_ShowLogMessage::DisplayType::Show;
|
||||||
|
out.params.type = lsMessageType::Error;
|
||||||
|
out.params.message =
|
||||||
|
"cquery client (v" + std::to_string(*config->clientVersion) +
|
||||||
|
") and server (v" + std::to_string(kExpectedClientVersion) +
|
||||||
|
") version mismatch. Please update ";
|
||||||
|
if (config->clientVersion > kExpectedClientVersion)
|
||||||
|
out.params.message += "the cquery binary.";
|
||||||
|
else
|
||||||
|
out.params.message +=
|
||||||
|
"your extension client (VSIX file). Make sure to uninstall "
|
||||||
|
"the cquery extension and restart vscode before "
|
||||||
|
"reinstalling.";
|
||||||
|
out.Write(std::cout);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Make sure cache directory is valid.
|
||||||
|
if (config->cacheDirectory.empty()) {
|
||||||
|
LOG_S(FATAL) << "Exiting; no cache directory";
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
config->cacheDirectory = NormalizePath(config->cacheDirectory);
|
||||||
|
EnsureEndsInSlash(config->cacheDirectory);
|
||||||
|
|
||||||
|
// Ensure there is a resource directory.
|
||||||
|
if (config->resourceDirectory.empty()) {
|
||||||
|
config->resourceDirectory = GetWorkingDirectory();
|
||||||
|
#if defined(_WIN32)
|
||||||
|
config->resourceDirectory += std::string("../../clang_resource_dir/");
|
||||||
|
#else
|
||||||
|
config->resourceDirectory += std::string("../clang_resource_dir/");
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
config->resourceDirectory = NormalizePath(config->resourceDirectory);
|
||||||
|
LOG_S(INFO) << "Using -resource-dir=" << config->resourceDirectory;
|
||||||
|
|
||||||
|
// Send initialization before starting indexers, so we don't send a
|
||||||
|
// status update too early.
|
||||||
|
// TODO: query request->params.capabilities.textDocument and support
|
||||||
|
// only things the client supports.
|
||||||
|
|
||||||
|
Out_InitializeResponse out;
|
||||||
|
out.id = request->id;
|
||||||
|
|
||||||
|
// out.result.capabilities.textDocumentSync =
|
||||||
|
// lsTextDocumentSyncOptions();
|
||||||
|
// out.result.capabilities.textDocumentSync->openClose = true;
|
||||||
|
// out.result.capabilities.textDocumentSync->change =
|
||||||
|
// lsTextDocumentSyncKind::Full;
|
||||||
|
// out.result.capabilities.textDocumentSync->willSave = true;
|
||||||
|
// out.result.capabilities.textDocumentSync->willSaveWaitUntil =
|
||||||
|
// true;
|
||||||
|
out.result.capabilities.textDocumentSync =
|
||||||
|
lsTextDocumentSyncKind::Incremental;
|
||||||
|
|
||||||
|
out.result.capabilities.renameProvider = true;
|
||||||
|
|
||||||
|
out.result.capabilities.completionProvider = lsCompletionOptions();
|
||||||
|
out.result.capabilities.completionProvider->resolveProvider = false;
|
||||||
|
// vscode doesn't support trigger character sequences, so we use ':'
|
||||||
|
// for
|
||||||
|
// '::' and '>' for '->'. See
|
||||||
|
// https://github.com/Microsoft/language-server-protocol/issues/138.
|
||||||
|
out.result.capabilities.completionProvider->triggerCharacters = {
|
||||||
|
".", ":", ">", "#"};
|
||||||
|
|
||||||
|
out.result.capabilities.signatureHelpProvider = lsSignatureHelpOptions();
|
||||||
|
// NOTE: If updating signature help tokens make sure to also update
|
||||||
|
// WorkingFile::FindClosestCallNameInBuffer.
|
||||||
|
out.result.capabilities.signatureHelpProvider->triggerCharacters = {"(",
|
||||||
|
","};
|
||||||
|
|
||||||
|
out.result.capabilities.codeLensProvider = lsCodeLensOptions();
|
||||||
|
out.result.capabilities.codeLensProvider->resolveProvider = false;
|
||||||
|
|
||||||
|
out.result.capabilities.definitionProvider = true;
|
||||||
|
out.result.capabilities.documentHighlightProvider = true;
|
||||||
|
out.result.capabilities.hoverProvider = true;
|
||||||
|
out.result.capabilities.referencesProvider = true;
|
||||||
|
|
||||||
|
out.result.capabilities.codeActionProvider = true;
|
||||||
|
|
||||||
|
out.result.capabilities.documentSymbolProvider = true;
|
||||||
|
out.result.capabilities.workspaceSymbolProvider = true;
|
||||||
|
|
||||||
|
out.result.capabilities.documentLinkProvider = lsDocumentLinkOptions();
|
||||||
|
out.result.capabilities.documentLinkProvider->resolveProvider = false;
|
||||||
|
|
||||||
|
IpcManager::WriteStdout(IpcId::Initialize, out);
|
||||||
|
|
||||||
|
// Set project root.
|
||||||
|
config->projectRoot = NormalizePath(request->params.rootUri->GetPath());
|
||||||
|
EnsureEndsInSlash(config->projectRoot);
|
||||||
|
MakeDirectoryRecursive(config->cacheDirectory +
|
||||||
|
EscapeFileName(config->projectRoot));
|
||||||
|
|
||||||
|
// Start indexer threads.
|
||||||
|
if (config->indexerCount == 0) {
|
||||||
|
// If the user has not specified how many indexers to run, try to
|
||||||
|
// guess an appropriate value. Default to 80% utilization.
|
||||||
|
const float kDefaultTargetUtilization = 0.8;
|
||||||
|
config->indexerCount =
|
||||||
|
std::thread::hardware_concurrency() * kDefaultTargetUtilization;
|
||||||
|
if (config->indexerCount <= 0)
|
||||||
|
config->indexerCount = 1;
|
||||||
|
}
|
||||||
|
LOG_S(INFO) << "Starting " << config->indexerCount << " indexers";
|
||||||
|
for (int i = 0; i < config->indexerCount; ++i) {
|
||||||
|
WorkThread::StartThread("indexer" + std::to_string(i), [=]() {
|
||||||
|
return IndexMain(config, file_consumer_shared, timestamp_manager,
|
||||||
|
import_manager, project, working_files, waiter,
|
||||||
|
queue);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
Timer time;
|
||||||
|
|
||||||
|
// Open up / load the project.
|
||||||
|
project->Load(config->extraClangArguments,
|
||||||
|
config->compilationDatabaseDirectory, project_path,
|
||||||
|
config->resourceDirectory);
|
||||||
|
time.ResetAndPrint("[perf] Loaded compilation entries (" +
|
||||||
|
std::to_string(project->entries.size()) + " files)");
|
||||||
|
|
||||||
|
// Start scanning include directories before dispatching project
|
||||||
|
// files, because that takes a long time.
|
||||||
|
include_complete->Rescan();
|
||||||
|
|
||||||
|
time.Reset();
|
||||||
|
project->ForAllFilteredFiles(
|
||||||
|
config, [&](int i, const Project::Entry& entry) {
|
||||||
|
bool is_interactive =
|
||||||
|
working_files->GetFileByFilename(entry.filename) != nullptr;
|
||||||
|
queue->index_request.Enqueue(Index_Request(
|
||||||
|
entry.filename, entry.args, is_interactive, nullopt));
|
||||||
|
});
|
||||||
|
|
||||||
|
// We need to support multiple concurrent index processes.
|
||||||
|
time.ResetAndPrint("[perf] Dispatched initial index requests");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
REGISTER_MESSAGE_HANDLER(InitializeHandler);
|
26
src/timestamp_manager.cc
Normal file
26
src/timestamp_manager.cc
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
#include "timestamp_manager.h"
|
||||||
|
|
||||||
|
#include "indexer.h"
|
||||||
|
|
||||||
|
optional<int64_t> TimestampManager::GetLastCachedModificationTime(
|
||||||
|
CacheLoader* cache_loader,
|
||||||
|
const std::string& path) {
|
||||||
|
{
|
||||||
|
std::lock_guard<std::mutex> guard(mutex_);
|
||||||
|
auto it = timestamps_.find(path);
|
||||||
|
if (it != timestamps_.end())
|
||||||
|
return it->second;
|
||||||
|
}
|
||||||
|
IndexFile* file = cache_loader->TryLoad(path);
|
||||||
|
if (!file)
|
||||||
|
return nullopt;
|
||||||
|
|
||||||
|
UpdateCachedModificationTime(path, file->last_modification_time);
|
||||||
|
return file->last_modification_time;
|
||||||
|
}
|
||||||
|
|
||||||
|
void TimestampManager::UpdateCachedModificationTime(const std::string& path,
|
||||||
|
int64_t timestamp) {
|
||||||
|
std::lock_guard<std::mutex> guard(mutex_);
|
||||||
|
timestamps_[path] = timestamp;
|
||||||
|
}
|
25
src/timestamp_manager.h
Normal file
25
src/timestamp_manager.h
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "cache_loader.h"
|
||||||
|
|
||||||
|
#include <optional.h>
|
||||||
|
|
||||||
|
#include <mutex>
|
||||||
|
#include <unordered_map>
|
||||||
|
|
||||||
|
using namespace std::experimental;
|
||||||
|
using std::experimental::nullopt;
|
||||||
|
|
||||||
|
// Caches timestamps of cc files so we can avoid a filesystem reads. This is
|
||||||
|
// important for import perf, as during dependency checking the same files are
|
||||||
|
// checked over and over again if they are common headers.
|
||||||
|
struct TimestampManager {
|
||||||
|
optional<int64_t> GetLastCachedModificationTime(CacheLoader* cache_loader,
|
||||||
|
const std::string& path);
|
||||||
|
|
||||||
|
void UpdateCachedModificationTime(const std::string& path, int64_t timestamp);
|
||||||
|
|
||||||
|
// TODO: use std::shared_mutex so we can have multiple readers.
|
||||||
|
std::mutex mutex_;
|
||||||
|
std::unordered_map<std::string, int64_t> timestamps_;
|
||||||
|
};
|
Loading…
Reference in New Issue
Block a user