Signed-off-by: Leonardo de Moura <leonardo@microsoft.com>
This commit is contained in:
parent
c7e9e238ec
commit
19c14fcca7
4 changed files with 187 additions and 107 deletions
|
@ -211,22 +211,21 @@ struct info_manager::imp {
|
||||||
friend bool operator<(env_info const & i1, env_info const & i2) { return i1.m_line < i2.m_line; }
|
friend bool operator<(env_info const & i1, env_info const & i2) { return i1.m_line < i2.m_line; }
|
||||||
};
|
};
|
||||||
mutex m_mutex;
|
mutex m_mutex;
|
||||||
bool m_block_updates;
|
bool m_block_new_info;
|
||||||
std::vector<info_data_set> m_line_data;
|
std::vector<info_data_set> m_line_data;
|
||||||
std::vector<bool> m_line_valid;
|
std::vector<bool> m_line_valid;
|
||||||
std::set<env_info> m_env_info;
|
std::set<env_info> m_env_info;
|
||||||
unsigned m_iteration; // current interation
|
unsigned m_iteration; // current interation
|
||||||
unsigned m_processed_upto;
|
unsigned m_processed_upto;
|
||||||
|
|
||||||
imp():m_block_updates(false), m_iteration(0), m_processed_upto(0) {}
|
imp():m_block_new_info(false), m_iteration(0), m_processed_upto(0) {}
|
||||||
|
|
||||||
void block_updates(bool f) {
|
void block_new_info(bool f) {
|
||||||
lock_guard<mutex> lc(m_mutex);
|
lock_guard<mutex> lc(m_mutex);
|
||||||
m_block_updates = f;
|
m_block_new_info = f;
|
||||||
}
|
}
|
||||||
|
|
||||||
void synch_line(unsigned l) {
|
void synch_line(unsigned l) {
|
||||||
lean_assert(!m_block_updates);
|
|
||||||
lean_assert(l > 0);
|
lean_assert(l > 0);
|
||||||
if (l >= m_line_data.size()) {
|
if (l >= m_line_data.size()) {
|
||||||
m_line_data.resize(l+1);
|
m_line_data.resize(l+1);
|
||||||
|
@ -236,7 +235,7 @@ struct info_manager::imp {
|
||||||
|
|
||||||
void save_environment_options(unsigned l, environment const & env, options const & o) {
|
void save_environment_options(unsigned l, environment const & env, options const & o) {
|
||||||
lock_guard<mutex> lc(m_mutex);
|
lock_guard<mutex> lc(m_mutex);
|
||||||
if (m_block_updates)
|
if (m_block_new_info)
|
||||||
return;
|
return;
|
||||||
// erase all entries in m_env_info such that e.m_line <= l and e.m_iteration < m_iteration
|
// erase all entries in m_env_info such that e.m_line <= l and e.m_iteration < m_iteration
|
||||||
auto it = m_env_info.begin();
|
auto it = m_env_info.begin();
|
||||||
|
@ -254,7 +253,7 @@ struct info_manager::imp {
|
||||||
|
|
||||||
void add_type_info(unsigned l, unsigned c, expr const & e) {
|
void add_type_info(unsigned l, unsigned c, expr const & e) {
|
||||||
lock_guard<mutex> lc(m_mutex);
|
lock_guard<mutex> lc(m_mutex);
|
||||||
if (m_block_updates)
|
if (m_block_new_info)
|
||||||
return;
|
return;
|
||||||
synch_line(l);
|
synch_line(l);
|
||||||
m_line_data[l].insert(mk_type_info(c, e));
|
m_line_data[l].insert(mk_type_info(c, e));
|
||||||
|
@ -262,7 +261,7 @@ struct info_manager::imp {
|
||||||
|
|
||||||
void add_synth_info(unsigned l, unsigned c, expr const & e) {
|
void add_synth_info(unsigned l, unsigned c, expr const & e) {
|
||||||
lock_guard<mutex> lc(m_mutex);
|
lock_guard<mutex> lc(m_mutex);
|
||||||
if (m_block_updates)
|
if (m_block_new_info)
|
||||||
return;
|
return;
|
||||||
synch_line(l);
|
synch_line(l);
|
||||||
m_line_data[l].insert(mk_synth_info(c, e));
|
m_line_data[l].insert(mk_synth_info(c, e));
|
||||||
|
@ -270,7 +269,7 @@ struct info_manager::imp {
|
||||||
|
|
||||||
void add_overload_info(unsigned l, unsigned c, expr const & e) {
|
void add_overload_info(unsigned l, unsigned c, expr const & e) {
|
||||||
lock_guard<mutex> lc(m_mutex);
|
lock_guard<mutex> lc(m_mutex);
|
||||||
if (m_block_updates)
|
if (m_block_new_info)
|
||||||
return;
|
return;
|
||||||
synch_line(l);
|
synch_line(l);
|
||||||
m_line_data[l].insert(mk_overload_info(c, e));
|
m_line_data[l].insert(mk_overload_info(c, e));
|
||||||
|
@ -278,7 +277,7 @@ struct info_manager::imp {
|
||||||
|
|
||||||
void add_coercion_info(unsigned l, unsigned c, expr const & e) {
|
void add_coercion_info(unsigned l, unsigned c, expr const & e) {
|
||||||
lock_guard<mutex> lc(m_mutex);
|
lock_guard<mutex> lc(m_mutex);
|
||||||
if (m_block_updates)
|
if (m_block_new_info)
|
||||||
return;
|
return;
|
||||||
synch_line(l);
|
synch_line(l);
|
||||||
m_line_data[l].insert(mk_coercion_info(c, e));
|
m_line_data[l].insert(mk_coercion_info(c, e));
|
||||||
|
@ -286,7 +285,7 @@ struct info_manager::imp {
|
||||||
|
|
||||||
void add_symbol_info(unsigned l, unsigned c, name const & s) {
|
void add_symbol_info(unsigned l, unsigned c, name const & s) {
|
||||||
lock_guard<mutex> lc(m_mutex);
|
lock_guard<mutex> lc(m_mutex);
|
||||||
if (m_block_updates)
|
if (m_block_new_info)
|
||||||
return;
|
return;
|
||||||
synch_line(l);
|
synch_line(l);
|
||||||
m_line_data[l].insert(mk_symbol_info(c, s));
|
m_line_data[l].insert(mk_symbol_info(c, s));
|
||||||
|
@ -294,7 +293,7 @@ struct info_manager::imp {
|
||||||
|
|
||||||
void add_identifier_info(unsigned l, unsigned c, name const & full_id) {
|
void add_identifier_info(unsigned l, unsigned c, name const & full_id) {
|
||||||
lock_guard<mutex> lc(m_mutex);
|
lock_guard<mutex> lc(m_mutex);
|
||||||
if (m_block_updates)
|
if (m_block_new_info)
|
||||||
return;
|
return;
|
||||||
synch_line(l);
|
synch_line(l);
|
||||||
m_line_data[l].insert(mk_identifier_info(c, full_id));
|
m_line_data[l].insert(mk_identifier_info(c, full_id));
|
||||||
|
@ -310,7 +309,7 @@ struct info_manager::imp {
|
||||||
|
|
||||||
void instantiate(substitution const & s) {
|
void instantiate(substitution const & s) {
|
||||||
lock_guard<mutex> lc(m_mutex);
|
lock_guard<mutex> lc(m_mutex);
|
||||||
if (m_block_updates)
|
if (m_block_new_info)
|
||||||
return;
|
return;
|
||||||
substitution tmp_s = s;
|
substitution tmp_s = s;
|
||||||
unsigned sz = m_line_data.size();
|
unsigned sz = m_line_data.size();
|
||||||
|
@ -323,7 +322,7 @@ struct info_manager::imp {
|
||||||
if (m.m_line_data.empty())
|
if (m.m_line_data.empty())
|
||||||
return;
|
return;
|
||||||
lock_guard<mutex> lc(m_mutex);
|
lock_guard<mutex> lc(m_mutex);
|
||||||
if (m_block_updates)
|
if (m_block_new_info)
|
||||||
return;
|
return;
|
||||||
unsigned sz = m.m_line_data.size();
|
unsigned sz = m.m_line_data.size();
|
||||||
for (unsigned i = 1; i < sz; i++) {
|
for (unsigned i = 1; i < sz; i++) {
|
||||||
|
@ -338,8 +337,6 @@ struct info_manager::imp {
|
||||||
|
|
||||||
void insert_line(unsigned l) {
|
void insert_line(unsigned l) {
|
||||||
lock_guard<mutex> lc(m_mutex);
|
lock_guard<mutex> lc(m_mutex);
|
||||||
if (m_block_updates)
|
|
||||||
return;
|
|
||||||
synch_line(l);
|
synch_line(l);
|
||||||
if (m_processed_upto > l - 1)
|
if (m_processed_upto > l - 1)
|
||||||
m_processed_upto = l - 1;
|
m_processed_upto = l - 1;
|
||||||
|
@ -355,8 +352,6 @@ struct info_manager::imp {
|
||||||
|
|
||||||
void remove_line(unsigned l) {
|
void remove_line(unsigned l) {
|
||||||
lock_guard<mutex> lc(m_mutex);
|
lock_guard<mutex> lc(m_mutex);
|
||||||
if (m_block_updates)
|
|
||||||
return;
|
|
||||||
lean_assert(l > 0);
|
lean_assert(l > 0);
|
||||||
if (l >= m_line_data.size())
|
if (l >= m_line_data.size())
|
||||||
return;
|
return;
|
||||||
|
@ -372,8 +367,6 @@ struct info_manager::imp {
|
||||||
|
|
||||||
void invalidate_line(unsigned l) {
|
void invalidate_line(unsigned l) {
|
||||||
lock_guard<mutex> lc(m_mutex);
|
lock_guard<mutex> lc(m_mutex);
|
||||||
if (m_block_updates)
|
|
||||||
return;
|
|
||||||
synch_line(l);
|
synch_line(l);
|
||||||
if (m_processed_upto > l - 1)
|
if (m_processed_upto > l - 1)
|
||||||
m_processed_upto = l - 1;
|
m_processed_upto = l - 1;
|
||||||
|
@ -383,7 +376,7 @@ struct info_manager::imp {
|
||||||
|
|
||||||
void commit_upto(unsigned l, bool valid) {
|
void commit_upto(unsigned l, bool valid) {
|
||||||
lock_guard<mutex> lc(m_mutex);
|
lock_guard<mutex> lc(m_mutex);
|
||||||
if (m_block_updates)
|
if (m_block_new_info)
|
||||||
return;
|
return;
|
||||||
for (unsigned i = m_processed_upto; i < l && i < m_line_valid.size(); i++)
|
for (unsigned i = m_processed_upto; i < l && i < m_line_valid.size(); i++)
|
||||||
m_line_valid[i] = valid;
|
m_line_valid[i] = valid;
|
||||||
|
@ -447,7 +440,7 @@ struct info_manager::imp {
|
||||||
|
|
||||||
void clear() {
|
void clear() {
|
||||||
lock_guard<mutex> lc(m_mutex);
|
lock_guard<mutex> lc(m_mutex);
|
||||||
if (m_block_updates)
|
if (m_block_new_info)
|
||||||
return;
|
return;
|
||||||
m_line_data.clear();
|
m_line_data.clear();
|
||||||
m_line_valid.clear();
|
m_line_valid.clear();
|
||||||
|
@ -478,7 +471,10 @@ void info_manager::clear() { m_ptr->clear(); }
|
||||||
optional<pair<environment, options>> info_manager::get_final_env_opts() const {
|
optional<pair<environment, options>> info_manager::get_final_env_opts() const {
|
||||||
return m_ptr->get_final_env_opts();
|
return m_ptr->get_final_env_opts();
|
||||||
}
|
}
|
||||||
void info_manager::display(environment const & env, io_state const & ios, unsigned line) {
|
void info_manager::display(environment const & env, io_state const & ios, unsigned line) const {
|
||||||
m_ptr->display(env, ios, line);
|
m_ptr->display(env, ios, line);
|
||||||
}
|
}
|
||||||
|
void info_manager::block_new_info(bool f) {
|
||||||
|
m_ptr->block_new_info(f);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -34,6 +34,7 @@ public:
|
||||||
void save_environment_options(unsigned l, environment const & env, options const & o);
|
void save_environment_options(unsigned l, environment const & env, options const & o);
|
||||||
optional<pair<environment, options>> get_final_env_opts() const;
|
optional<pair<environment, options>> get_final_env_opts() const;
|
||||||
void clear();
|
void clear();
|
||||||
void display(environment const & env, io_state const & ios, unsigned line);
|
void display(environment const & env, io_state const & ios, unsigned line) const;
|
||||||
|
void block_new_info(bool f);
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,15 +11,25 @@ Author: Leonardo de Moura
|
||||||
#include "frontends/lean/parser.h"
|
#include "frontends/lean/parser.h"
|
||||||
|
|
||||||
namespace lean {
|
namespace lean {
|
||||||
server::file::file(std::string const & fname):m_fname(fname) {}
|
server::file::file(std::istream & in, std::string const & fname):m_fname(fname) {
|
||||||
|
for (std::string line; std::getline(in, line);) {
|
||||||
|
m_lines.push_back(line);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void server::file::replace_line(unsigned linenum, std::string const & new_line) {
|
void server::file::replace_line(unsigned linenum, std::string const & new_line) {
|
||||||
|
lock_guard<mutex> lk(m_lines_mutex);
|
||||||
|
m_info.block_new_info(true);
|
||||||
|
m_info.invalidate_line(linenum+1);
|
||||||
while (linenum >= m_lines.size())
|
while (linenum >= m_lines.size())
|
||||||
m_lines.push_back("");
|
m_lines.push_back("");
|
||||||
m_lines[linenum] = new_line;
|
m_lines[linenum] = new_line;
|
||||||
}
|
}
|
||||||
|
|
||||||
void server::file::insert_line(unsigned linenum, std::string const & new_line) {
|
void server::file::insert_line(unsigned linenum, std::string const & new_line) {
|
||||||
|
lock_guard<mutex> lk(m_lines_mutex);
|
||||||
|
m_info.block_new_info(true);
|
||||||
|
m_info.insert_line(linenum+1);
|
||||||
while (linenum >= m_lines.size())
|
while (linenum >= m_lines.size())
|
||||||
m_lines.push_back("");
|
m_lines.push_back("");
|
||||||
m_lines.push_back("");
|
m_lines.push_back("");
|
||||||
|
@ -33,6 +43,9 @@ void server::file::insert_line(unsigned linenum, std::string const & new_line) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void server::file::remove_line(unsigned linenum) {
|
void server::file::remove_line(unsigned linenum) {
|
||||||
|
lock_guard<mutex> lk(m_lines_mutex);
|
||||||
|
m_info.block_new_info(true);
|
||||||
|
m_info.remove_line(linenum+1);
|
||||||
if (linenum >= m_lines.size())
|
if (linenum >= m_lines.size())
|
||||||
return;
|
return;
|
||||||
lean_assert(!m_lines.empty());
|
lean_assert(!m_lines.empty());
|
||||||
|
@ -65,26 +78,125 @@ unsigned server::file::find(unsigned linenum) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** \brief Copy lines [starting_from, m_lines.size()) to block and return the total number of lines */
|
||||||
|
unsigned server::file::copy_to(std::string & block, unsigned starting_from) {
|
||||||
|
unsigned num_lines = m_lines.size();
|
||||||
|
for (unsigned j = starting_from; j < num_lines; j++) {
|
||||||
|
block += m_lines[j];
|
||||||
|
block += '\n';
|
||||||
|
}
|
||||||
|
return num_lines;
|
||||||
|
}
|
||||||
|
|
||||||
|
server::worker::worker(environment const & env, io_state const & ios, definition_cache & cache):
|
||||||
|
m_empty_snapshot(env, ios.get_options()),
|
||||||
|
m_cache(cache),
|
||||||
|
m_todo_linenum(0),
|
||||||
|
m_todo_options(ios.get_options()),
|
||||||
|
m_terminate(false),
|
||||||
|
m_thread([=]() {
|
||||||
|
io_state _ios(ios);
|
||||||
|
while (!m_terminate) {
|
||||||
|
file_ptr todo_file;
|
||||||
|
unsigned todo_linenum;
|
||||||
|
options todo_options;
|
||||||
|
// wait for next task
|
||||||
|
while (!m_terminate) {
|
||||||
|
unique_lock<mutex> lk(m_todo_mutex);
|
||||||
|
if (m_todo_file) {
|
||||||
|
todo_file = m_todo_file;
|
||||||
|
todo_linenum = m_todo_linenum;
|
||||||
|
todo_options = m_todo_options;
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
m_todo_cv.wait(lk);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// extract block of code and snapshot from todo_file
|
||||||
|
reset_interrupt();
|
||||||
|
if (m_terminate)
|
||||||
|
break;
|
||||||
|
std::string block;
|
||||||
|
unsigned num_lines;
|
||||||
|
snapshot s;
|
||||||
|
{
|
||||||
|
lean_assert(todo_file);
|
||||||
|
lock_guard<mutex> lk(todo_file->m_lines_mutex);
|
||||||
|
todo_file->m_info.block_new_info(false);
|
||||||
|
unsigned i = todo_file->find(todo_linenum);
|
||||||
|
todo_file->m_snapshots.resize(i);
|
||||||
|
s = i == 0 ? m_empty_snapshot : todo_file->m_snapshots[i-1];
|
||||||
|
lean_assert(s.m_line > 0);
|
||||||
|
num_lines = todo_file->copy_to(block, s.m_line - 1);
|
||||||
|
}
|
||||||
|
if (m_terminate)
|
||||||
|
break;
|
||||||
|
// parse block of code with respect to snapshot
|
||||||
|
try {
|
||||||
|
std::istringstream strm(block);
|
||||||
|
std::shared_ptr<output_channel> out1(new string_output_channel());
|
||||||
|
std::shared_ptr<output_channel> out2(new string_output_channel());
|
||||||
|
io_state tmp_ios(_ios, out1, out2);
|
||||||
|
tmp_ios.set_options(join(s.m_options, _ios.get_options()));
|
||||||
|
bool use_exceptions = false;
|
||||||
|
unsigned num_threads = 1;
|
||||||
|
parser p(s.m_env, tmp_ios, strm, todo_file->m_fname.c_str(), use_exceptions, num_threads,
|
||||||
|
s.m_lds, s.m_eds, s.m_line, &todo_file->m_snapshots, &todo_file->m_info);
|
||||||
|
p.set_cache(&m_cache);
|
||||||
|
p();
|
||||||
|
} catch (exception&) {}
|
||||||
|
if (!m_terminate) {
|
||||||
|
unique_lock<mutex> lk(m_todo_mutex);
|
||||||
|
if (m_todo_file == todo_file && m_last_file == todo_file && m_todo_linenum == todo_linenum) {
|
||||||
|
m_todo_linenum = num_lines + 1;
|
||||||
|
m_todo_file = nullptr;
|
||||||
|
m_todo_cv.notify_all();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}) {}
|
||||||
|
|
||||||
|
server::worker::~worker() {
|
||||||
|
m_terminate = true;
|
||||||
|
request_interrupt();
|
||||||
|
m_thread.join();
|
||||||
|
}
|
||||||
|
|
||||||
|
void server::worker::request_interrupt() {
|
||||||
|
m_todo_cv.notify_all();
|
||||||
|
m_thread.request_interrupt();
|
||||||
|
}
|
||||||
|
|
||||||
|
void server::worker::wait() {
|
||||||
|
while (true) {
|
||||||
|
unique_lock<mutex> lk(m_todo_mutex);
|
||||||
|
if (!m_todo_file)
|
||||||
|
break;
|
||||||
|
m_todo_cv.wait(lk);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void server::worker::set_todo(file_ptr const & f, unsigned linenum, options const & o) {
|
||||||
|
lock_guard<mutex> lk(m_todo_mutex);
|
||||||
|
if (m_last_file != f || linenum < m_todo_linenum)
|
||||||
|
m_todo_linenum = linenum;
|
||||||
|
m_todo_file = f;
|
||||||
|
m_last_file = f;
|
||||||
|
m_todo_options = o;
|
||||||
|
m_todo_cv.notify_all();
|
||||||
|
}
|
||||||
|
|
||||||
server::server(environment const & env, io_state const & ios, unsigned num_threads):
|
server::server(environment const & env, io_state const & ios, unsigned num_threads):
|
||||||
m_env(env), m_ios(ios), m_out(ios.get_regular_channel().get_stream()),
|
m_env(env), m_ios(ios), m_out(ios.get_regular_channel().get_stream()),
|
||||||
m_num_threads(num_threads), m_empty_snapshot(m_env, m_ios.get_options()) {
|
m_num_threads(num_threads), m_empty_snapshot(m_env, m_ios.get_options()),
|
||||||
|
m_worker(env, ios, m_cache) {
|
||||||
}
|
}
|
||||||
|
|
||||||
server::~server() {
|
server::~server() {
|
||||||
reset_thread();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void server::interrupt_thread() {
|
void server::interrupt_worker() {
|
||||||
if (m_thread_ptr)
|
m_worker.request_interrupt();
|
||||||
m_thread_ptr->request_interrupt();
|
|
||||||
}
|
|
||||||
|
|
||||||
void server::reset_thread() {
|
|
||||||
if (m_thread_ptr) {
|
|
||||||
m_thread_ptr->request_interrupt();
|
|
||||||
m_thread_ptr->join();
|
|
||||||
m_thread_ptr.reset(nullptr);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static std::string g_load("LOAD");
|
static std::string g_load("LOAD");
|
||||||
|
@ -92,7 +204,6 @@ static std::string g_visit("VISIT");
|
||||||
static std::string g_replace("REPLACE");
|
static std::string g_replace("REPLACE");
|
||||||
static std::string g_insert("INSERT");
|
static std::string g_insert("INSERT");
|
||||||
static std::string g_remove("REMOVE");
|
static std::string g_remove("REMOVE");
|
||||||
static std::string g_check("CHECK");
|
|
||||||
static std::string g_info("INFO");
|
static std::string g_info("INFO");
|
||||||
static std::string g_set("SET");
|
static std::string g_set("SET");
|
||||||
static std::string g_eval("EVAL");
|
static std::string g_eval("EVAL");
|
||||||
|
@ -118,55 +229,27 @@ static std::string & trim(std::string & s) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void server::process_from(unsigned linenum) {
|
void server::process_from(unsigned linenum) {
|
||||||
reset_thread();
|
m_worker.set_todo(m_file, linenum, m_ios.get_options());
|
||||||
unsigned i = m_file->find(linenum);
|
|
||||||
m_file->m_snapshots.resize(i);
|
|
||||||
snapshot & s = i == 0 ? m_empty_snapshot : m_file->m_snapshots[i-1];
|
|
||||||
std::string block;
|
|
||||||
lean_assert(s.m_line > 0);
|
|
||||||
for (unsigned j = s.m_line-1; j < m_file->m_lines.size(); j++) {
|
|
||||||
block += m_file->m_lines[j];
|
|
||||||
block += '\n';
|
|
||||||
}
|
|
||||||
m_thread_ptr.reset(new interruptible_thread([=]() {
|
|
||||||
try {
|
|
||||||
snapshot & s = i == 0 ? m_empty_snapshot : m_file->m_snapshots[i-1];
|
|
||||||
std::istringstream strm(block);
|
|
||||||
std::shared_ptr<output_channel> out1(new string_output_channel());
|
|
||||||
std::shared_ptr<output_channel> out2(new string_output_channel());
|
|
||||||
io_state ios(m_ios, out1, out2);
|
|
||||||
ios.set_options(s.m_options);
|
|
||||||
parser p(s.m_env, ios, strm, m_file->m_fname.c_str(), false, 1, s.m_lds, s.m_eds, s.m_line,
|
|
||||||
&m_file->m_snapshots, &m_file->m_info);
|
|
||||||
p.set_cache(&m_cache);
|
|
||||||
p();
|
|
||||||
} catch (exception& ex) {}
|
|
||||||
}));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void server::load_file(std::string const & fname) {
|
void server::load_file(std::string const & fname) {
|
||||||
interrupt_thread();
|
interrupt_worker();
|
||||||
std::ifstream in(fname);
|
std::ifstream in(fname);
|
||||||
if (in.bad() || in.fail()) {
|
if (in.bad() || in.fail()) {
|
||||||
m_out << "-- ERROR failed to open file '" << fname << "'" << std::endl;
|
m_out << "-- ERROR failed to open file '" << fname << "'" << std::endl;
|
||||||
} else {
|
} else {
|
||||||
reset_thread();
|
m_file.reset(new file(in, fname));
|
||||||
m_file.reset(new file(fname));
|
|
||||||
m_file_map.insert(mk_pair(fname, m_file));
|
m_file_map.insert(mk_pair(fname, m_file));
|
||||||
for (std::string line; std::getline(in, line);) {
|
|
||||||
m_file->m_lines.push_back(line);
|
|
||||||
}
|
|
||||||
process_from(0);
|
process_from(0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void server::visit_file(std::string const & fname) {
|
void server::visit_file(std::string const & fname) {
|
||||||
interrupt_thread();
|
interrupt_worker();
|
||||||
auto it = m_file_map.find(fname);
|
auto it = m_file_map.find(fname);
|
||||||
if (it == m_file_map.end()) {
|
if (it == m_file_map.end()) {
|
||||||
load_file(fname);
|
load_file(fname);
|
||||||
} else {
|
} else {
|
||||||
reset_thread();
|
|
||||||
m_file = it->second;
|
m_file = it->second;
|
||||||
process_from(0);
|
process_from(0);
|
||||||
}
|
}
|
||||||
|
@ -193,43 +276,26 @@ void server::check_file() {
|
||||||
}
|
}
|
||||||
|
|
||||||
void server::replace_line(unsigned linenum, std::string const & new_line) {
|
void server::replace_line(unsigned linenum, std::string const & new_line) {
|
||||||
interrupt_thread();
|
interrupt_worker();
|
||||||
check_file();
|
check_file();
|
||||||
m_file->replace_line(linenum, new_line);
|
m_file->replace_line(linenum, new_line);
|
||||||
reset_thread();
|
|
||||||
m_file->m_info.invalidate_line(linenum+1);
|
|
||||||
process_from(linenum);
|
process_from(linenum);
|
||||||
}
|
}
|
||||||
|
|
||||||
void server::insert_line(unsigned linenum, std::string const & new_line) {
|
void server::insert_line(unsigned linenum, std::string const & new_line) {
|
||||||
interrupt_thread();
|
interrupt_worker();
|
||||||
check_file();
|
check_file();
|
||||||
m_file->insert_line(linenum, new_line);
|
m_file->insert_line(linenum, new_line);
|
||||||
reset_thread();
|
|
||||||
m_file->m_info.insert_line(linenum+1);
|
|
||||||
process_from(linenum);
|
process_from(linenum);
|
||||||
}
|
}
|
||||||
|
|
||||||
void server::remove_line(unsigned linenum) {
|
void server::remove_line(unsigned linenum) {
|
||||||
interrupt_thread();
|
interrupt_worker();
|
||||||
check_file();
|
check_file();
|
||||||
m_file->remove_line(linenum);
|
m_file->remove_line(linenum);
|
||||||
reset_thread();
|
|
||||||
m_file->m_info.remove_line(linenum+1);
|
|
||||||
process_from(linenum);
|
process_from(linenum);
|
||||||
}
|
}
|
||||||
|
|
||||||
void server::check_line(unsigned linenum, std::string const & line) {
|
|
||||||
check_file();
|
|
||||||
if (linenum >= m_file->m_lines.size()) {
|
|
||||||
m_out << "-- MISMATCH line out of range" << std::endl;
|
|
||||||
} else if (m_file->m_lines[linenum] != line) {
|
|
||||||
m_out << "-- MISMATCH expected " << m_file->m_lines[linenum] << std::endl;
|
|
||||||
} else {
|
|
||||||
m_out << "-- OK" << std::endl;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void server::set_option(std::string const & line) {
|
void server::set_option(std::string const & line) {
|
||||||
std::string cmd = "set_option ";
|
std::string cmd = "set_option ";
|
||||||
cmd += line;
|
cmd += line;
|
||||||
|
@ -248,7 +314,7 @@ void server::set_option(std::string const & line) {
|
||||||
void server::show_info(unsigned linenum) {
|
void server::show_info(unsigned linenum) {
|
||||||
check_file();
|
check_file();
|
||||||
m_out << "-- BEGININFO" << std::endl;
|
m_out << "-- BEGININFO" << std::endl;
|
||||||
m_file->m_info.display(m_env, m_ios, linenum);
|
m_file->infom().display(m_env, m_ios, linenum);
|
||||||
m_out << "-- ENDINFO" << std::endl;
|
m_out << "-- ENDINFO" << std::endl;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -268,7 +334,7 @@ void server::eval_core(environment const & env, options const & o, std::string c
|
||||||
void server::eval(std::string const & line) {
|
void server::eval(std::string const & line) {
|
||||||
if (!m_file) {
|
if (!m_file) {
|
||||||
eval_core(m_env, m_ios.get_options(), line);
|
eval_core(m_env, m_ios.get_options(), line);
|
||||||
} else if (auto p = m_file->m_info.get_final_env_opts()) {
|
} else if (auto p = m_file->infom().get_final_env_opts()) {
|
||||||
eval_core(p->first, join(p->second, m_ios.get_options()), line);
|
eval_core(p->first, join(p->second, m_ios.get_options()), line);
|
||||||
} else {
|
} else {
|
||||||
eval_core(m_env, m_ios.get_options(), line);
|
eval_core(m_env, m_ios.get_options(), line);
|
||||||
|
@ -286,10 +352,6 @@ bool server::operator()(std::istream & in) {
|
||||||
std::string fname = line.substr(g_visit.size());
|
std::string fname = line.substr(g_visit.size());
|
||||||
trim(fname);
|
trim(fname);
|
||||||
visit_file(fname);
|
visit_file(fname);
|
||||||
} else if (is_command(g_check, line)) {
|
|
||||||
unsigned linenum = get_linenum(line, g_check);
|
|
||||||
read_line(in, line);
|
|
||||||
check_line(linenum-1, line);
|
|
||||||
} else if (is_command(g_replace, line)) {
|
} else if (is_command(g_replace, line)) {
|
||||||
unsigned linenum = get_linenum(line, g_replace);
|
unsigned linenum = get_linenum(line, g_replace);
|
||||||
read_line(in, line);
|
read_line(in, line);
|
||||||
|
@ -313,10 +375,7 @@ bool server::operator()(std::istream & in) {
|
||||||
} else if (is_command(g_clear_cache, line)) {
|
} else if (is_command(g_clear_cache, line)) {
|
||||||
m_cache.clear();
|
m_cache.clear();
|
||||||
} else if (is_command(g_wait, line)) {
|
} else if (is_command(g_wait, line)) {
|
||||||
if (m_thread_ptr) {
|
m_worker.wait();
|
||||||
m_thread_ptr->join();
|
|
||||||
m_thread_ptr.reset(nullptr);
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
throw exception(sstream() << "unexpected command line: " << line);
|
throw exception(sstream() << "unexpected command line: " << line);
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,20 +19,46 @@ namespace lean {
|
||||||
external processes.
|
external processes.
|
||||||
*/
|
*/
|
||||||
class server {
|
class server {
|
||||||
struct file {
|
class worker;
|
||||||
|
class file {
|
||||||
|
friend class server::worker;
|
||||||
std::string m_fname;
|
std::string m_fname;
|
||||||
|
mutable mutex m_lines_mutex;
|
||||||
std::vector<std::string> m_lines;
|
std::vector<std::string> m_lines;
|
||||||
snapshot_vector m_snapshots;
|
snapshot_vector m_snapshots;
|
||||||
info_manager m_info;
|
info_manager m_info;
|
||||||
|
|
||||||
file(std::string const & fname);
|
|
||||||
unsigned find(unsigned linenum);
|
unsigned find(unsigned linenum);
|
||||||
|
unsigned copy_to(std::string & block, unsigned starting_from);
|
||||||
|
public:
|
||||||
|
file(std::istream & in, std::string const & fname);
|
||||||
void replace_line(unsigned linenum, std::string const & new_line);
|
void replace_line(unsigned linenum, std::string const & new_line);
|
||||||
void insert_line(unsigned linenum, std::string const & new_line);
|
void insert_line(unsigned linenum, std::string const & new_line);
|
||||||
void remove_line(unsigned linenum);
|
void remove_line(unsigned linenum);
|
||||||
|
info_manager const & infom() const { return m_info; }
|
||||||
};
|
};
|
||||||
typedef std::shared_ptr<file> file_ptr;
|
typedef std::shared_ptr<file> file_ptr;
|
||||||
typedef std::unordered_map<std::string, file_ptr> file_map;
|
typedef std::unordered_map<std::string, file_ptr> file_map;
|
||||||
|
class worker {
|
||||||
|
snapshot m_empty_snapshot;
|
||||||
|
definition_cache & m_cache;
|
||||||
|
atomic_bool m_busy;
|
||||||
|
file_ptr m_todo_file;
|
||||||
|
unsigned m_todo_linenum;
|
||||||
|
options m_todo_options;
|
||||||
|
mutex m_todo_mutex;
|
||||||
|
condition_variable m_todo_cv;
|
||||||
|
file_ptr m_last_file;
|
||||||
|
atomic_bool m_terminate;
|
||||||
|
interruptible_thread m_thread;
|
||||||
|
public:
|
||||||
|
worker(environment const & env, io_state const & ios, definition_cache & cache);
|
||||||
|
~worker();
|
||||||
|
void set_todo(file_ptr const & f, unsigned linenum, options const & o);
|
||||||
|
void request_interrupt();
|
||||||
|
void wait();
|
||||||
|
};
|
||||||
|
|
||||||
file_map m_file_map;
|
file_map m_file_map;
|
||||||
file_ptr m_file;
|
file_ptr m_file;
|
||||||
environment m_env;
|
environment m_env;
|
||||||
|
@ -41,7 +67,7 @@ class server {
|
||||||
unsigned m_num_threads;
|
unsigned m_num_threads;
|
||||||
snapshot m_empty_snapshot;
|
snapshot m_empty_snapshot;
|
||||||
definition_cache m_cache;
|
definition_cache m_cache;
|
||||||
std::unique_ptr<interruptible_thread> m_thread_ptr;
|
worker m_worker;
|
||||||
|
|
||||||
void load_file(std::string const & fname);
|
void load_file(std::string const & fname);
|
||||||
void visit_file(std::string const & fname);
|
void visit_file(std::string const & fname);
|
||||||
|
@ -49,7 +75,6 @@ class server {
|
||||||
void replace_line(unsigned linenum, std::string const & new_line);
|
void replace_line(unsigned linenum, std::string const & new_line);
|
||||||
void insert_line(unsigned linenum, std::string const & new_line);
|
void insert_line(unsigned linenum, std::string const & new_line);
|
||||||
void remove_line(unsigned linenum);
|
void remove_line(unsigned linenum);
|
||||||
void check_line(unsigned linenum, std::string const & line);
|
|
||||||
void show_info(unsigned linenum);
|
void show_info(unsigned linenum);
|
||||||
void process_from(unsigned linenum);
|
void process_from(unsigned linenum);
|
||||||
void set_option(std::string const & line);
|
void set_option(std::string const & line);
|
||||||
|
@ -57,8 +82,7 @@ class server {
|
||||||
void eval(std::string const & line);
|
void eval(std::string const & line);
|
||||||
unsigned find(unsigned linenum);
|
unsigned find(unsigned linenum);
|
||||||
void read_line(std::istream & in, std::string & line);
|
void read_line(std::istream & in, std::string & line);
|
||||||
void reset_thread();
|
void interrupt_worker();
|
||||||
void interrupt_thread();
|
|
||||||
unsigned get_linenum(std::string const & line, std::string const & cmd);
|
unsigned get_linenum(std::string const & line, std::string const & cmd);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
Loading…
Reference in a new issue