2017-04-17 01:22:59 +00:00
|
|
|
#include "atomic_object.h"
|
2017-03-26 21:40:34 +00:00
|
|
|
#include "language_server_api.h"
|
|
|
|
#include "libclangmm/Index.h"
|
|
|
|
#include "libclangmm/TranslationUnit.h"
|
|
|
|
#include "project.h"
|
2017-05-26 06:40:38 +00:00
|
|
|
#include "threaded_queue.h"
|
2017-03-26 21:40:34 +00:00
|
|
|
#include "working_files.h"
|
|
|
|
|
|
|
|
#include <clang-c/Index.h>
|
|
|
|
|
2017-04-17 01:22:59 +00:00
|
|
|
#include <functional>
|
2017-05-26 06:40:38 +00:00
|
|
|
#include <memory>
|
2017-05-10 04:52:15 +00:00
|
|
|
#include <mutex>
|
2017-05-26 06:40:38 +00:00
|
|
|
#include <string>
|
2017-04-17 01:22:59 +00:00
|
|
|
|
2017-09-22 01:14:57 +00:00
|
|
|
struct CompletionSession
|
|
|
|
: public std::enable_shared_from_this<CompletionSession> {
|
2017-04-20 05:01:36 +00:00
|
|
|
Project::Entry file;
|
2017-05-10 04:52:15 +00:00
|
|
|
WorkingFiles* working_files;
|
2017-05-26 06:40:38 +00:00
|
|
|
clang::Index index;
|
2017-05-10 04:52:15 +00:00
|
|
|
|
2017-05-26 06:40:38 +00:00
|
|
|
// When |tu| was last parsed.
|
2017-09-22 01:14:57 +00:00
|
|
|
optional<std::chrono::time_point<std::chrono::high_resolution_clock>>
|
|
|
|
tu_last_parsed_at;
|
2017-03-26 21:40:34 +00:00
|
|
|
|
2017-05-26 06:40:38 +00:00
|
|
|
// Acquired when |tu| is being used.
|
|
|
|
std::mutex tu_lock;
|
2017-03-26 21:40:34 +00:00
|
|
|
|
2017-05-26 06:40:38 +00:00
|
|
|
// The active translation unit.
|
|
|
|
std::unique_ptr<clang::TranslationUnit> tu;
|
2017-03-26 21:40:34 +00:00
|
|
|
|
2017-04-26 04:03:22 +00:00
|
|
|
CompletionSession(const Project::Entry& file, WorkingFiles* working_files);
|
2017-03-26 21:40:34 +00:00
|
|
|
~CompletionSession();
|
|
|
|
};
|
|
|
|
|
2017-05-26 06:40:38 +00:00
|
|
|
struct LruSessionCache {
|
2017-06-10 04:13:16 +00:00
|
|
|
std::vector<std::shared_ptr<CompletionSession>> entries_;
|
2017-05-26 06:40:38 +00:00
|
|
|
int max_entries_;
|
|
|
|
|
|
|
|
LruSessionCache(int max_entries);
|
|
|
|
|
|
|
|
// Fetches the entry for |filename| and updates it's usage so it is less
|
|
|
|
// likely to be evicted.
|
2017-06-10 04:15:33 +00:00
|
|
|
std::shared_ptr<CompletionSession> TryGetEntry(const std::string& filename);
|
2017-05-27 04:21:00 +00:00
|
|
|
// TryGetEntry, except the return value captures ownership.
|
2017-06-10 04:13:16 +00:00
|
|
|
std::shared_ptr<CompletionSession> TryTakeEntry(const std::string& fiilename);
|
2017-05-26 06:40:38 +00:00
|
|
|
// Inserts an entry. Evicts the oldest unused entry if there is no space.
|
2017-06-10 04:13:16 +00:00
|
|
|
void InsertEntry(std::shared_ptr<CompletionSession> session);
|
2017-05-26 06:40:38 +00:00
|
|
|
};
|
2017-03-26 21:40:34 +00:00
|
|
|
|
2017-05-27 04:21:00 +00:00
|
|
|
struct ClangCompleteManager {
|
2017-09-22 01:14:57 +00:00
|
|
|
using OnDiagnostic =
|
|
|
|
std::function<void(std::string path,
|
|
|
|
NonElidedVector<lsDiagnostic> diagnostics)>;
|
2017-09-27 06:03:43 +00:00
|
|
|
using OnIndex = std::function<void(clang::TranslationUnit* tu,
|
|
|
|
const std::vector<CXUnsavedFile>& unsaved,
|
|
|
|
const std::string& path,
|
|
|
|
const std::vector<std::string>& args)>;
|
2017-09-22 01:14:57 +00:00
|
|
|
using OnComplete =
|
|
|
|
std::function<void(const NonElidedVector<lsCompletionItem>& results,
|
|
|
|
bool is_cached_result)>;
|
2017-06-29 02:50:30 +00:00
|
|
|
|
2017-05-26 06:40:38 +00:00
|
|
|
struct ParseRequest {
|
|
|
|
ParseRequest(const std::string& path);
|
|
|
|
|
|
|
|
std::chrono::time_point<std::chrono::high_resolution_clock> request_time;
|
|
|
|
std::string path;
|
|
|
|
};
|
2017-04-17 01:22:59 +00:00
|
|
|
struct CompletionRequest {
|
2017-09-22 02:25:33 +00:00
|
|
|
lsTextDocumentIdentifier document;
|
|
|
|
optional<lsPosition> position;
|
|
|
|
OnComplete on_complete; // May be null/empty.
|
|
|
|
bool emit_diagnostics = false;
|
2017-04-17 01:22:59 +00:00
|
|
|
};
|
2017-05-10 04:52:15 +00:00
|
|
|
|
2017-09-22 01:14:57 +00:00
|
|
|
ClangCompleteManager(Config* config,
|
|
|
|
Project* project,
|
|
|
|
WorkingFiles* working_files,
|
2017-09-27 06:03:43 +00:00
|
|
|
OnDiagnostic on_diagnostic,
|
|
|
|
OnIndex on_index);
|
2017-06-29 02:50:30 +00:00
|
|
|
~ClangCompleteManager();
|
2017-03-26 21:40:34 +00:00
|
|
|
|
2017-04-17 01:22:59 +00:00
|
|
|
// Start a code completion at the given location. |on_complete| will run when
|
|
|
|
// completion results are available. |on_complete| may run on any thread.
|
2017-05-26 06:40:38 +00:00
|
|
|
void CodeComplete(const lsTextDocumentPositionParams& completion_location,
|
|
|
|
const OnComplete& on_complete);
|
2017-09-22 02:25:33 +00:00
|
|
|
// Request a diagnostics update.
|
|
|
|
void DiagnosticsUpdate(const lsTextDocumentIdentifier& document);
|
2017-05-26 06:40:38 +00:00
|
|
|
|
|
|
|
// Notify the completion manager that |filename| has been viewed and we
|
|
|
|
// should begin preloading completion data.
|
|
|
|
void NotifyView(const std::string& filename);
|
|
|
|
// Notify the completion manager that |filename| has been edited.
|
|
|
|
void NotifyEdit(const std::string& filename);
|
|
|
|
// Notify the completion manager that |filename| has been saved. This
|
|
|
|
// triggers a reparse.
|
|
|
|
void NotifySave(const std::string& filename);
|
2017-10-17 18:43:33 +00:00
|
|
|
// Notify the completion manager that |filename| has been closed. Any existing
|
|
|
|
// completion session will be dropped.
|
|
|
|
void NotifyClose(const std::string& filename);
|
2017-05-26 06:40:38 +00:00
|
|
|
|
2017-09-22 01:14:57 +00:00
|
|
|
std::shared_ptr<CompletionSession> TryGetSession(const std::string& filename,
|
|
|
|
bool create_if_needed);
|
2017-05-26 06:40:38 +00:00
|
|
|
|
|
|
|
// TODO: make these configurable.
|
2017-10-12 15:38:44 +00:00
|
|
|
const int kMaxViewSessions = 10;
|
|
|
|
const int kMaxEditSessions = 5;
|
2017-05-26 06:40:38 +00:00
|
|
|
|
|
|
|
// Global state.
|
|
|
|
Config* config_;
|
|
|
|
Project* project_;
|
|
|
|
WorkingFiles* working_files_;
|
2017-06-10 04:13:16 +00:00
|
|
|
OnDiagnostic on_diagnostic_;
|
2017-09-27 06:03:43 +00:00
|
|
|
OnIndex on_index_;
|
2017-05-26 06:40:38 +00:00
|
|
|
|
|
|
|
// Sessions which have never had a real text-edit applied, but are preloaded
|
|
|
|
// to give a fast initial experience.
|
|
|
|
LruSessionCache view_sessions_;
|
|
|
|
// Completion sessions which have been edited.
|
|
|
|
LruSessionCache edit_sessions_;
|
|
|
|
// Mutex which protects |view_sessions_| and |edit_sessions_|.
|
|
|
|
std::mutex sessions_lock_;
|
|
|
|
|
|
|
|
// Request a code completion at the given location.
|
|
|
|
AtomicObject<CompletionRequest> completion_request_;
|
|
|
|
// Parse requests. The path may already be parsed, in which case it should be
|
|
|
|
// reparsed.
|
|
|
|
ThreadedQueue<ParseRequest> parse_requests_;
|
2017-06-29 02:50:30 +00:00
|
|
|
};
|