feat(*): simplify interrupt propagation
Instead of having m_interrupted flags in several components. We use a thread_local global variable. The new approach is much simpler to get right since there is no risk of "forgetting" to propagate the set_interrupt method to sub-components. The plan is to support set_interrupt methods and m_interrupted flags only in tactic objects. We need to support them in tactics and tacticals because we want to implement combinators/tacticals such as (try_for T M) that fails if tactic T does not finish in M ms. For example, consider the tactic: try-for (T1 ORELSE T2) 5 It tries the tactic (T1 ORELSE T2) for 5ms. Thus, if T1 does not finish after 5ms an interrupt request is sent, and T1 is interrupted. Now, if you do not have a m_interrupted flag marking each tactic, the ORELSE combinator will try T2. The set_interrupt method for ORELSE tactical should turn on the m_interrupted flag. Signed-off-by: Leonardo de Moura <leonardo@microsoft.com>
This commit is contained in:
parent
b31233e8c2
commit
c4c548dc5d
26 changed files with 35 additions and 181 deletions
|
@ -189,13 +189,5 @@ public:
|
||||||
void set_regular_channel(std::shared_ptr<output_channel> const & out) { m_state.set_regular_channel(out); }
|
void set_regular_channel(std::shared_ptr<output_channel> const & out) { m_state.set_regular_channel(out); }
|
||||||
void set_diagnostic_channel(std::shared_ptr<output_channel> const & out) { m_state.set_diagnostic_channel(out); }
|
void set_diagnostic_channel(std::shared_ptr<output_channel> const & out) { m_state.set_diagnostic_channel(out); }
|
||||||
/*@}*/
|
/*@}*/
|
||||||
|
|
||||||
/**
|
|
||||||
@name Interrupts.
|
|
||||||
*/
|
|
||||||
void set_interrupt(bool flag) { m_env.set_interrupt(flag); m_state.set_interrupt(flag); }
|
|
||||||
void interrupt() { set_interrupt(true); }
|
|
||||||
void reset_interrupt() { set_interrupt(false); }
|
|
||||||
/*@}*/
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,7 +8,7 @@ Author: Leonardo de Moura
|
||||||
#include <utility>
|
#include <utility>
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include <limits>
|
#include <limits>
|
||||||
#include "util/interruptable_ptr.h"
|
#include "util/interrupt.h"
|
||||||
#include "kernel/type_checker.h"
|
#include "kernel/type_checker.h"
|
||||||
#include "kernel/type_checker_justification.h"
|
#include "kernel/type_checker_justification.h"
|
||||||
#include "kernel/normalizer.h"
|
#include "kernel/normalizer.h"
|
||||||
|
@ -112,8 +112,6 @@ class frontend_elaborator::imp {
|
||||||
// We need that because a frontend may associate line number information
|
// We need that because a frontend may associate line number information
|
||||||
// with the original non-elaborated expressions.
|
// with the original non-elaborated expressions.
|
||||||
expr_map<expr> m_trace;
|
expr_map<expr> m_trace;
|
||||||
interruptable_ptr<elaborator> m_elaborator;
|
|
||||||
volatile bool m_interrupted;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\brief Replace placeholders and choices with metavariables.
|
\brief Replace placeholders and choices with metavariables.
|
||||||
|
@ -378,7 +376,7 @@ class frontend_elaborator::imp {
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual expr visit(expr const & e, context const & ctx) {
|
virtual expr visit(expr const & e, context const & ctx) {
|
||||||
check_interrupted(m_ref.m_interrupted);
|
check_interrupted();
|
||||||
expr r = replace_visitor::visit(e, ctx);
|
expr r = replace_visitor::visit(e, ctx);
|
||||||
if (!is_eqp(r, e))
|
if (!is_eqp(r, e))
|
||||||
m_ref.m_trace[r] = e;
|
m_ref.m_trace[r] = e;
|
||||||
|
@ -392,7 +390,6 @@ class frontend_elaborator::imp {
|
||||||
// return !is_choice(c1) && is_choice(c2);
|
// return !is_choice(c1) && is_choice(c2);
|
||||||
// });
|
// });
|
||||||
elaborator elb(m_env, m_menv, m_ucs.size(), m_ucs.data());
|
elaborator elb(m_env, m_menv, m_ucs.size(), m_ucs.data());
|
||||||
scoped_set_interruptable_ptr<elaborator> set(m_elaborator, &elb);
|
|
||||||
return elb.next();
|
return elb.next();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -403,7 +400,6 @@ public:
|
||||||
m_type_checker(m_env),
|
m_type_checker(m_env),
|
||||||
m_type_inferer(m_env),
|
m_type_inferer(m_env),
|
||||||
m_normalizer(m_env) {
|
m_normalizer(m_env) {
|
||||||
m_interrupted = false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
expr elaborate(expr const & e) {
|
expr elaborate(expr const & e) {
|
||||||
|
@ -458,14 +454,6 @@ public:
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void set_interrupt(bool f) {
|
|
||||||
m_interrupted = f;
|
|
||||||
m_type_checker.set_interrupt(f);
|
|
||||||
m_type_inferer.set_interrupt(f);
|
|
||||||
m_normalizer.set_interrupt(f);
|
|
||||||
m_elaborator.set_interrupt(f);
|
|
||||||
}
|
|
||||||
|
|
||||||
void clear() {
|
void clear() {
|
||||||
m_menv = metavar_env();
|
m_menv = metavar_env();
|
||||||
m_ucs.clear();
|
m_ucs.clear();
|
||||||
|
@ -485,7 +473,6 @@ frontend_elaborator::~frontend_elaborator() {}
|
||||||
expr frontend_elaborator::operator()(expr const & e) { return m_ptr->elaborate(e); }
|
expr frontend_elaborator::operator()(expr const & e) { return m_ptr->elaborate(e); }
|
||||||
std::pair<expr, expr> frontend_elaborator::operator()(name const & n, expr const & t, expr const & e) { return m_ptr->elaborate(n, t, e); }
|
std::pair<expr, expr> frontend_elaborator::operator()(name const & n, expr const & t, expr const & e) { return m_ptr->elaborate(n, t, e); }
|
||||||
expr const & frontend_elaborator::get_original(expr const & e) const { return m_ptr->get_original(e); }
|
expr const & frontend_elaborator::get_original(expr const & e) const { return m_ptr->get_original(e); }
|
||||||
void frontend_elaborator::set_interrupt(bool f) { m_ptr->set_interrupt(f); }
|
|
||||||
void frontend_elaborator::clear() { m_ptr->clear(); }
|
void frontend_elaborator::clear() { m_ptr->clear(); }
|
||||||
environment const & frontend_elaborator::get_environment() const { return m_ptr->get_environment(); }
|
environment const & frontend_elaborator::get_environment() const { return m_ptr->get_environment(); }
|
||||||
}
|
}
|
||||||
|
|
|
@ -44,10 +44,6 @@ public:
|
||||||
*/
|
*/
|
||||||
expr const & get_original(expr const & e) const;
|
expr const & get_original(expr const & e) const;
|
||||||
|
|
||||||
void set_interrupt(bool flag);
|
|
||||||
void interrupt() { set_interrupt(true); }
|
|
||||||
void reset_interrupt() { set_interrupt(false); }
|
|
||||||
|
|
||||||
void clear();
|
void clear();
|
||||||
|
|
||||||
environment const & get_environment() const;
|
environment const & get_environment() const;
|
||||||
|
|
|
@ -20,6 +20,7 @@ Author: Leonardo de Moura
|
||||||
#include "util/exception.h"
|
#include "util/exception.h"
|
||||||
#include "util/sstream.h"
|
#include "util/sstream.h"
|
||||||
#include "util/sexpr/option_declarations.h"
|
#include "util/sexpr/option_declarations.h"
|
||||||
|
#include "util/interrupt.h"
|
||||||
#include "kernel/normalizer.h"
|
#include "kernel/normalizer.h"
|
||||||
#include "kernel/type_checker.h"
|
#include "kernel/type_checker.h"
|
||||||
#include "kernel/free_vars.h"
|
#include "kernel/free_vars.h"
|
||||||
|
@ -129,10 +130,6 @@ class parser::imp {
|
||||||
expr_pos_info m_expr_pos_info;
|
expr_pos_info m_expr_pos_info;
|
||||||
pos_info m_last_cmd_pos;
|
pos_info m_last_cmd_pos;
|
||||||
pos_info m_last_script_pos;
|
pos_info m_last_script_pos;
|
||||||
// Reference to temporary parser used to process import command.
|
|
||||||
// We need this reference to be able to interrupt it.
|
|
||||||
interruptable_ptr<parser> m_import_parser;
|
|
||||||
interruptable_ptr<normalizer> m_normalizer;
|
|
||||||
|
|
||||||
leanlua_state * m_leanlua_state;
|
leanlua_state * m_leanlua_state;
|
||||||
|
|
||||||
|
@ -1170,7 +1167,6 @@ class parser::imp {
|
||||||
next();
|
next();
|
||||||
expr v = m_elaborator(parse_expr());
|
expr v = m_elaborator(parse_expr());
|
||||||
normalizer norm(m_frontend);
|
normalizer norm(m_frontend);
|
||||||
scoped_set_interruptable_ptr<normalizer> set(m_normalizer, &norm);
|
|
||||||
expr r = norm(v);
|
expr r = norm(v);
|
||||||
regular(m_frontend) << r << endl;
|
regular(m_frontend) << r << endl;
|
||||||
}
|
}
|
||||||
|
@ -1403,7 +1399,6 @@ class parser::imp {
|
||||||
if (m_verbose)
|
if (m_verbose)
|
||||||
regular(m_frontend) << "Importing file '" << fname << "'" << endl;
|
regular(m_frontend) << "Importing file '" << fname << "'" << endl;
|
||||||
parser import_parser(m_frontend, in, m_leanlua_state, true /* use exceptions */, false /* not interactive */);
|
parser import_parser(m_frontend, in, m_leanlua_state, true /* use exceptions */, false /* not interactive */);
|
||||||
scoped_set_interruptable_ptr<parser> set(m_import_parser, &import_parser);
|
|
||||||
import_parser();
|
import_parser();
|
||||||
} catch (interrupted &) {
|
} catch (interrupted &) {
|
||||||
throw;
|
throw;
|
||||||
|
@ -1686,17 +1681,6 @@ public:
|
||||||
throw parser_exception(ex.what(), ex.m_pos.first, ex.m_pos.second);
|
throw parser_exception(ex.what(), ex.m_pos.first, ex.m_pos.second);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void set_interrupt(bool flag) {
|
|
||||||
m_frontend.set_interrupt(flag);
|
|
||||||
m_elaborator.set_interrupt(flag);
|
|
||||||
m_import_parser.set_interrupt(flag);
|
|
||||||
m_normalizer.set_interrupt(flag);
|
|
||||||
}
|
|
||||||
|
|
||||||
void reset_interrupt() {
|
|
||||||
set_interrupt(false);
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
parser::parser(frontend & fe, std::istream & in, leanlua_state * S, bool use_exceptions, bool interactive) {
|
parser::parser(frontend & fe, std::istream & in, leanlua_state * S, bool use_exceptions, bool interactive) {
|
||||||
|
@ -1711,10 +1695,6 @@ bool parser::operator()() {
|
||||||
return m_ptr->parse_commands();
|
return m_ptr->parse_commands();
|
||||||
}
|
}
|
||||||
|
|
||||||
void parser::set_interrupt(bool flag) {
|
|
||||||
m_ptr->set_interrupt(flag);
|
|
||||||
}
|
|
||||||
|
|
||||||
expr parser::parse_expr() {
|
expr parser::parse_expr() {
|
||||||
return m_ptr->parse_expr_main();
|
return m_ptr->parse_expr_main();
|
||||||
}
|
}
|
||||||
|
@ -1736,7 +1716,6 @@ bool shell::operator()() {
|
||||||
std::istringstream strm(input);
|
std::istringstream strm(input);
|
||||||
{
|
{
|
||||||
parser p(m_frontend, strm, m_leanlua_state, false, false);
|
parser p(m_frontend, strm, m_leanlua_state, false, false);
|
||||||
scoped_set_interruptable_ptr<parser> set(m_parser, &p);
|
|
||||||
if (!p())
|
if (!p())
|
||||||
errors = true;
|
errors = true;
|
||||||
}
|
}
|
||||||
|
@ -1744,12 +1723,7 @@ bool shell::operator()() {
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
parser p(m_frontend, std::cin, m_leanlua_state, false, true);
|
parser p(m_frontend, std::cin, m_leanlua_state, false, true);
|
||||||
scoped_set_interruptable_ptr<parser> set(m_parser, &p);
|
|
||||||
return p();
|
return p();
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
void shell::set_interrupt(bool flag) {
|
|
||||||
m_parser.set_interrupt(flag);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,7 +6,6 @@ Author: Leonardo de Moura
|
||||||
*/
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include "util/interruptable_ptr.h"
|
|
||||||
#include "frontends/lean/frontend.h"
|
#include "frontends/lean/frontend.h"
|
||||||
|
|
||||||
namespace lean {
|
namespace lean {
|
||||||
|
@ -24,26 +23,17 @@ public:
|
||||||
|
|
||||||
/** \brief Parse a single expression */
|
/** \brief Parse a single expression */
|
||||||
expr parse_expr();
|
expr parse_expr();
|
||||||
|
|
||||||
void set_interrupt(bool flag);
|
|
||||||
void interrupt() { set_interrupt(true); }
|
|
||||||
void reset_interrupt() { set_interrupt(false); }
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/** \brief Implements the Read Eval Print loop */
|
/** \brief Implements the Read Eval Print loop */
|
||||||
class shell {
|
class shell {
|
||||||
frontend & m_frontend;
|
frontend & m_frontend;
|
||||||
leanlua_state * m_leanlua_state;
|
leanlua_state * m_leanlua_state;
|
||||||
interruptable_ptr<parser> m_parser;
|
|
||||||
public:
|
public:
|
||||||
shell(frontend & fe, leanlua_state * S);
|
shell(frontend & fe, leanlua_state * S);
|
||||||
~shell();
|
~shell();
|
||||||
|
|
||||||
bool operator()();
|
bool operator()();
|
||||||
|
|
||||||
void set_interrupt(bool flag);
|
|
||||||
void interrupt() { set_interrupt(true); }
|
|
||||||
void reset_interrupt() { set_interrupt(false); }
|
|
||||||
};
|
};
|
||||||
|
|
||||||
inline bool parse_commands(frontend & fe, std::istream & in, leanlua_state * S = nullptr, bool use_exceptions = true, bool interactive = false) {
|
inline bool parse_commands(frontend & fe, std::istream & in, leanlua_state * S = nullptr, bool use_exceptions = true, bool interactive = false) {
|
||||||
|
|
|
@ -13,7 +13,7 @@ Author: Leonardo de Moura
|
||||||
#include "util/exception.h"
|
#include "util/exception.h"
|
||||||
#include "util/scoped_set.h"
|
#include "util/scoped_set.h"
|
||||||
#include "util/sexpr/options.h"
|
#include "util/sexpr/options.h"
|
||||||
#include "util/interruptable_ptr.h"
|
#include "util/interrupt.h"
|
||||||
#include "kernel/context.h"
|
#include "kernel/context.h"
|
||||||
#include "kernel/for_each.h"
|
#include "kernel/for_each.h"
|
||||||
#include "kernel/occurs.h"
|
#include "kernel/occurs.h"
|
||||||
|
@ -169,8 +169,6 @@ class pp_fn {
|
||||||
bool m_notation; //!< if true use notation
|
bool m_notation; //!< if true use notation
|
||||||
bool m_extra_lets; //!< introduce extra let-expression to cope with sharing.
|
bool m_extra_lets; //!< introduce extra let-expression to cope with sharing.
|
||||||
unsigned m_alias_min_weight; //!< minimal weight for creating an alias
|
unsigned m_alias_min_weight; //!< minimal weight for creating an alias
|
||||||
volatile bool m_interrupted;
|
|
||||||
|
|
||||||
|
|
||||||
// Create a scope for local definitions
|
// Create a scope for local definitions
|
||||||
struct mk_scope {
|
struct mk_scope {
|
||||||
|
@ -1072,7 +1070,7 @@ class pp_fn {
|
||||||
}
|
}
|
||||||
|
|
||||||
result pp(expr const & e, unsigned depth, bool main = false) {
|
result pp(expr const & e, unsigned depth, bool main = false) {
|
||||||
check_interrupted(m_interrupted);
|
check_interrupted();
|
||||||
if (!is_atomic(e) && (m_num_steps > m_max_steps || depth > m_max_depth)) {
|
if (!is_atomic(e) && (m_num_steps > m_max_steps || depth > m_max_depth)) {
|
||||||
return pp_ellipsis();
|
return pp_ellipsis();
|
||||||
} else {
|
} else {
|
||||||
|
@ -1170,7 +1168,6 @@ public:
|
||||||
pp_fn(frontend const & fe, options const & opts):
|
pp_fn(frontend const & fe, options const & opts):
|
||||||
m_frontend(fe) {
|
m_frontend(fe) {
|
||||||
set_options(opts);
|
set_options(opts);
|
||||||
m_interrupted = false;
|
|
||||||
m_num_steps = 0;
|
m_num_steps = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1190,10 +1187,6 @@ public:
|
||||||
return pp_abstraction_core(e, 0, expr(), &implicit_args).first;
|
return pp_abstraction_core(e, 0, expr(), &implicit_args).first;
|
||||||
}
|
}
|
||||||
|
|
||||||
void set_interrupt(bool flag) {
|
|
||||||
m_interrupted = flag;
|
|
||||||
}
|
|
||||||
|
|
||||||
void register_local(name const & n) {
|
void register_local(name const & n) {
|
||||||
m_local_names.insert(n);
|
m_local_names.insert(n);
|
||||||
}
|
}
|
||||||
|
@ -1201,24 +1194,20 @@ public:
|
||||||
|
|
||||||
class pp_formatter_cell : public formatter_cell {
|
class pp_formatter_cell : public formatter_cell {
|
||||||
frontend const & m_frontend;
|
frontend const & m_frontend;
|
||||||
interruptable_ptr<pp_fn> m_pp_fn;
|
|
||||||
volatile bool m_interrupted;
|
|
||||||
|
|
||||||
format pp(expr const & e, options const & opts) {
|
format pp(expr const & e, options const & opts) {
|
||||||
pp_fn fn(m_frontend, opts);
|
pp_fn fn(m_frontend, opts);
|
||||||
scoped_set_interruptable_ptr<pp_fn> set(m_pp_fn, &fn);
|
|
||||||
return fn(e);
|
return fn(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
format pp(context const & c, expr const & e, bool include_e, options const & opts) {
|
format pp(context const & c, expr const & e, bool include_e, options const & opts) {
|
||||||
pp_fn fn(m_frontend, opts);
|
pp_fn fn(m_frontend, opts);
|
||||||
scoped_set_interruptable_ptr<pp_fn> set(m_pp_fn, &fn);
|
|
||||||
unsigned indent = get_pp_indent(opts);
|
unsigned indent = get_pp_indent(opts);
|
||||||
format r;
|
format r;
|
||||||
bool first = true;
|
bool first = true;
|
||||||
expr c2 = context_to_lambda(c, e);
|
expr c2 = context_to_lambda(c, e);
|
||||||
while (is_fake_context(c2)) {
|
while (is_fake_context(c2)) {
|
||||||
check_interrupted(m_interrupted);
|
check_interrupted();
|
||||||
name n1 = get_unused_name(c2);
|
name n1 = get_unused_name(c2);
|
||||||
fn.register_local(n1);
|
fn.register_local(n1);
|
||||||
format entry = format{format(n1), space(), colon(), space(), fn(fake_context_domain(c2))};
|
format entry = format{format(n1), space(), colon(), space(), fn(fake_context_domain(c2))};
|
||||||
|
@ -1255,7 +1244,7 @@ class pp_formatter_cell : public formatter_cell {
|
||||||
expr it1 = t;
|
expr it1 = t;
|
||||||
expr it2 = v;
|
expr it2 = v;
|
||||||
while (is_pi(it1) && is_lambda(it2)) {
|
while (is_pi(it1) && is_lambda(it2)) {
|
||||||
check_interrupted(m_interrupted);
|
check_interrupted();
|
||||||
if (abst_domain(it1) != abst_domain(it2))
|
if (abst_domain(it1) != abst_domain(it2))
|
||||||
return pp_definition(kwd, n, t, v, opts);
|
return pp_definition(kwd, n, t, v, opts);
|
||||||
it1 = abst_body(it1);
|
it1 = abst_body(it1);
|
||||||
|
@ -1269,7 +1258,6 @@ class pp_formatter_cell : public formatter_cell {
|
||||||
if (m_frontend.has_implicit_arguments(n))
|
if (m_frontend.has_implicit_arguments(n))
|
||||||
implicit_args = &(m_frontend.get_implicit_arguments(n));
|
implicit_args = &(m_frontend.get_implicit_arguments(n));
|
||||||
pp_fn fn(m_frontend, opts);
|
pp_fn fn(m_frontend, opts);
|
||||||
scoped_set_interruptable_ptr<pp_fn> set(m_pp_fn, &fn);
|
|
||||||
format def = fn.pp_definition(v, t, implicit_args);
|
format def = fn.pp_definition(v, t, implicit_args);
|
||||||
return format{highlight_command(format(kwd)), space(), format(n), def};
|
return format{highlight_command(format(kwd)), space(), format(n), def};
|
||||||
}
|
}
|
||||||
|
@ -1286,7 +1274,6 @@ class pp_formatter_cell : public formatter_cell {
|
||||||
format r = format{highlight_command(format(kwd)), space(), format(n)};
|
format r = format{highlight_command(format(kwd)), space(), format(n)};
|
||||||
if (m_frontend.has_implicit_arguments(n)) {
|
if (m_frontend.has_implicit_arguments(n)) {
|
||||||
pp_fn fn(m_frontend, opts);
|
pp_fn fn(m_frontend, opts);
|
||||||
scoped_set_interruptable_ptr<pp_fn> set(m_pp_fn, &fn);
|
|
||||||
r += fn.pp_pi_with_implicit_args(obj.get_type(), m_frontend.get_implicit_arguments(n));
|
r += fn.pp_pi_with_implicit_args(obj.get_type(), m_frontend.get_implicit_arguments(n));
|
||||||
} else {
|
} else {
|
||||||
r += format{space(), colon(), space(), pp(obj.get_type(), opts)};
|
r += format{space(), colon(), space(), pp(obj.get_type(), opts)};
|
||||||
|
@ -1329,7 +1316,6 @@ class pp_formatter_cell : public formatter_cell {
|
||||||
public:
|
public:
|
||||||
pp_formatter_cell(frontend const & fe):
|
pp_formatter_cell(frontend const & fe):
|
||||||
m_frontend(fe) {
|
m_frontend(fe) {
|
||||||
m_interrupted = false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual ~pp_formatter_cell() {
|
virtual ~pp_formatter_cell() {
|
||||||
|
@ -1348,10 +1334,9 @@ public:
|
||||||
return pp(c, e, true, opts);
|
return pp(c, e, true, opts);
|
||||||
} else {
|
} else {
|
||||||
pp_fn fn(m_frontend, opts);
|
pp_fn fn(m_frontend, opts);
|
||||||
scoped_set_interruptable_ptr<pp_fn> set(m_pp_fn, &fn);
|
|
||||||
expr c2 = context_to_lambda(c, e);
|
expr c2 = context_to_lambda(c, e);
|
||||||
while (is_fake_context(c2)) {
|
while (is_fake_context(c2)) {
|
||||||
check_interrupted(m_interrupted);
|
check_interrupted();
|
||||||
name n1 = get_unused_name(c2);
|
name n1 = get_unused_name(c2);
|
||||||
fn.register_local(n1);
|
fn.register_local(n1);
|
||||||
expr const & rest = fake_context_rest(c2);
|
expr const & rest = fake_context_rest(c2);
|
||||||
|
@ -1389,17 +1374,12 @@ public:
|
||||||
std::for_each(env.begin_objects(),
|
std::for_each(env.begin_objects(),
|
||||||
env.end_objects(),
|
env.end_objects(),
|
||||||
[&](object const & obj) {
|
[&](object const & obj) {
|
||||||
check_interrupted(m_interrupted);
|
check_interrupted();
|
||||||
if (first) first = false; else r += line();
|
if (first) first = false; else r += line();
|
||||||
r += operator()(obj, opts);
|
r += operator()(obj, opts);
|
||||||
});
|
});
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual void set_interrupt(bool flag) {
|
|
||||||
m_pp_fn.set_interrupt(flag);
|
|
||||||
m_interrupted = flag;
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
formatter mk_pp_formatter(frontend const & fe) {
|
formatter mk_pp_formatter(frontend const & fe) {
|
||||||
|
|
|
@ -402,10 +402,6 @@ struct environment::imp {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void set_interrupt(bool flag) {
|
|
||||||
m_type_checker->set_interrupt(flag);
|
|
||||||
}
|
|
||||||
|
|
||||||
imp():
|
imp():
|
||||||
m_num_children(0) {
|
m_num_children(0) {
|
||||||
init_uvars();
|
init_uvars();
|
||||||
|
@ -531,10 +527,6 @@ void environment::display(std::ostream & out) const {
|
||||||
m_ptr->display(out, *this);
|
m_ptr->display(out, *this);
|
||||||
}
|
}
|
||||||
|
|
||||||
void environment::set_interrupt(bool flag) {
|
|
||||||
m_ptr->set_interrupt(flag);
|
|
||||||
}
|
|
||||||
|
|
||||||
environment::extension const & environment::get_extension_core(unsigned extid) const {
|
environment::extension const & environment::get_extension_core(unsigned extid) const {
|
||||||
return m_ptr->get_extension_core(extid);
|
return m_ptr->get_extension_core(extid);
|
||||||
}
|
}
|
||||||
|
|
|
@ -209,10 +209,6 @@ public:
|
||||||
/** \brief Display universal variable constraints and objects stored in this environment and its parents. */
|
/** \brief Display universal variable constraints and objects stored in this environment and its parents. */
|
||||||
void display(std::ostream & out) const;
|
void display(std::ostream & out) const;
|
||||||
|
|
||||||
void set_interrupt(bool flag);
|
|
||||||
void interrupt() { set_interrupt(true); }
|
|
||||||
void reset_interrupt() { set_interrupt(false); }
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\brief Frontend can store data in environment extensions.
|
\brief Frontend can store data in environment extensions.
|
||||||
Each extension is associated with a unique token/id.
|
Each extension is associated with a unique token/id.
|
||||||
|
|
|
@ -34,8 +34,6 @@ public:
|
||||||
virtual format operator()(object const & obj, options const & opts) = 0;
|
virtual format operator()(object const & obj, options const & opts) = 0;
|
||||||
/** \brief Format the given environment */
|
/** \brief Format the given environment */
|
||||||
virtual format operator()(environment const & env, options const & opts) = 0;
|
virtual format operator()(environment const & env, options const & opts) = 0;
|
||||||
/** \brief Request interruption */
|
|
||||||
virtual void set_interrupt(bool ) {}
|
|
||||||
};
|
};
|
||||||
/**
|
/**
|
||||||
\brief Smart-pointer for the actual formatter object (aka \c formatter_cell).
|
\brief Smart-pointer for the actual formatter object (aka \c formatter_cell).
|
||||||
|
@ -50,7 +48,6 @@ public:
|
||||||
format operator()(context const & c, expr const & e, bool format_ctx = false, options const & opts = options()) const { return (*m_cell)(c, e, format_ctx, opts); }
|
format operator()(context const & c, expr const & e, bool format_ctx = false, options const & opts = options()) const { return (*m_cell)(c, e, format_ctx, opts); }
|
||||||
format operator()(object const & obj, options const & opts = options()) const { return (*m_cell)(obj, opts); }
|
format operator()(object const & obj, options const & opts = options()) const { return (*m_cell)(obj, opts); }
|
||||||
format operator()(environment const & env, options const & opts = options()) const { return (*m_cell)(env, opts); }
|
format operator()(environment const & env, options const & opts = options()) const { return (*m_cell)(env, opts); }
|
||||||
void set_interrupt(bool flag) { m_cell->set_interrupt(flag); }
|
|
||||||
};
|
};
|
||||||
/**
|
/**
|
||||||
\brief Create a simple formatter object based on \c print function.
|
\brief Create a simple formatter object based on \c print function.
|
||||||
|
|
|
@ -10,6 +10,7 @@ Author: Leonardo de Moura
|
||||||
#include "util/list.h"
|
#include "util/list.h"
|
||||||
#include "util/flet.h"
|
#include "util/flet.h"
|
||||||
#include "util/buffer.h"
|
#include "util/buffer.h"
|
||||||
|
#include "util/interrupt.h"
|
||||||
#include "util/sexpr/options.h"
|
#include "util/sexpr/options.h"
|
||||||
#include "kernel/normalizer.h"
|
#include "kernel/normalizer.h"
|
||||||
#include "kernel/expr.h"
|
#include "kernel/expr.h"
|
||||||
|
@ -74,7 +75,6 @@ class normalizer::imp {
|
||||||
cache m_cache;
|
cache m_cache;
|
||||||
unsigned m_max_depth;
|
unsigned m_max_depth;
|
||||||
unsigned m_depth;
|
unsigned m_depth;
|
||||||
volatile bool m_interrupted;
|
|
||||||
|
|
||||||
environment env() const { return environment(m_env); }
|
environment env() const { return environment(m_env); }
|
||||||
|
|
||||||
|
@ -179,7 +179,7 @@ class normalizer::imp {
|
||||||
/** \brief Normalize the expression \c a in a context composed of stack \c s and \c k binders. */
|
/** \brief Normalize the expression \c a in a context composed of stack \c s and \c k binders. */
|
||||||
svalue normalize(expr const & a, value_stack const & s, unsigned k) {
|
svalue normalize(expr const & a, value_stack const & s, unsigned k) {
|
||||||
flet<unsigned> l(m_depth, m_depth+1);
|
flet<unsigned> l(m_depth, m_depth+1);
|
||||||
check_interrupted(m_interrupted);
|
check_interrupted();
|
||||||
if (m_depth > m_max_depth)
|
if (m_depth > m_max_depth)
|
||||||
throw kernel_exception(env(), "normalizer maximum recursion depth exceeded");
|
throw kernel_exception(env(), "normalizer maximum recursion depth exceeded");
|
||||||
bool shared = false;
|
bool shared = false;
|
||||||
|
@ -294,7 +294,6 @@ class normalizer::imp {
|
||||||
public:
|
public:
|
||||||
imp(environment const & env, unsigned max_depth):
|
imp(environment const & env, unsigned max_depth):
|
||||||
m_env(env.to_weak_ref()) {
|
m_env(env.to_weak_ref()) {
|
||||||
m_interrupted = false;
|
|
||||||
m_max_depth = max_depth;
|
m_max_depth = max_depth;
|
||||||
m_depth = 0;
|
m_depth = 0;
|
||||||
}
|
}
|
||||||
|
@ -306,7 +305,6 @@ public:
|
||||||
}
|
}
|
||||||
|
|
||||||
void clear() { m_ctx = context(); m_cache.clear(); }
|
void clear() { m_ctx = context(); m_cache.clear(); }
|
||||||
void set_interrupt(bool flag) { m_interrupted = flag; }
|
|
||||||
};
|
};
|
||||||
|
|
||||||
normalizer::normalizer(environment const & env, unsigned max_depth):m_ptr(new imp(env, max_depth)) {}
|
normalizer::normalizer(environment const & env, unsigned max_depth):m_ptr(new imp(env, max_depth)) {}
|
||||||
|
@ -315,7 +313,6 @@ normalizer::normalizer(environment const & env, options const & opts):normalizer
|
||||||
normalizer::~normalizer() {}
|
normalizer::~normalizer() {}
|
||||||
expr normalizer::operator()(expr const & e, context const & ctx) { return (*m_ptr)(e, ctx); }
|
expr normalizer::operator()(expr const & e, context const & ctx) { return (*m_ptr)(e, ctx); }
|
||||||
void normalizer::clear() { m_ptr->clear(); }
|
void normalizer::clear() { m_ptr->clear(); }
|
||||||
void normalizer::set_interrupt(bool flag) { m_ptr->set_interrupt(flag); }
|
|
||||||
|
|
||||||
expr normalize(expr const & e, environment const & env, context const & ctx) {
|
expr normalize(expr const & e, environment const & env, context const & ctx) {
|
||||||
return normalizer(env)(e, ctx);
|
return normalizer(env)(e, ctx);
|
||||||
|
|
|
@ -26,10 +26,6 @@ public:
|
||||||
expr operator()(expr const & e, context const & ctx = context());
|
expr operator()(expr const & e, context const & ctx = context());
|
||||||
|
|
||||||
void clear();
|
void clear();
|
||||||
|
|
||||||
void set_interrupt(bool flag);
|
|
||||||
void interrupt() { set_interrupt(true); }
|
|
||||||
void reset_interrupt() { set_interrupt(false); }
|
|
||||||
};
|
};
|
||||||
/** \brief Normalize \c e using the environment \c env and context \c ctx */
|
/** \brief Normalize \c e using the environment \c env and context \c ctx */
|
||||||
expr normalize(expr const & e, environment const & env, context const & ctx = context());
|
expr normalize(expr const & e, environment const & env, context const & ctx = context());
|
||||||
|
|
|
@ -6,6 +6,7 @@ Author: Leonardo de Moura
|
||||||
*/
|
*/
|
||||||
#include "util/scoped_map.h"
|
#include "util/scoped_map.h"
|
||||||
#include "util/flet.h"
|
#include "util/flet.h"
|
||||||
|
#include "util/interrupt.h"
|
||||||
#include "kernel/type_checker.h"
|
#include "kernel/type_checker.h"
|
||||||
#include "kernel/environment.h"
|
#include "kernel/environment.h"
|
||||||
#include "kernel/kernel_exception.h"
|
#include "kernel/kernel_exception.h"
|
||||||
|
@ -29,7 +30,6 @@ class type_checker::imp {
|
||||||
metavar_env * m_menv;
|
metavar_env * m_menv;
|
||||||
unsigned m_menv_timestamp;
|
unsigned m_menv_timestamp;
|
||||||
unification_constraints * m_uc;
|
unification_constraints * m_uc;
|
||||||
volatile bool m_interrupted;
|
|
||||||
|
|
||||||
environment env() const {
|
environment env() const {
|
||||||
return environment(m_env);
|
return environment(m_env);
|
||||||
|
@ -85,7 +85,7 @@ class type_checker::imp {
|
||||||
}
|
}
|
||||||
|
|
||||||
expr infer_type_core(expr const & e, context const & ctx) {
|
expr infer_type_core(expr const & e, context const & ctx) {
|
||||||
check_interrupted(m_interrupted);
|
check_interrupted();
|
||||||
bool shared = false;
|
bool shared = false;
|
||||||
if (is_shared(e)) {
|
if (is_shared(e)) {
|
||||||
shared = true;
|
shared = true;
|
||||||
|
@ -289,7 +289,6 @@ public:
|
||||||
m_menv = nullptr;
|
m_menv = nullptr;
|
||||||
m_menv_timestamp = 0;
|
m_menv_timestamp = 0;
|
||||||
m_uc = nullptr;
|
m_uc = nullptr;
|
||||||
m_interrupted = false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
expr infer_type(expr const & e, context const & ctx, metavar_env * menv, buffer<unification_constraint> & uc) {
|
expr infer_type(expr const & e, context const & ctx, metavar_env * menv, buffer<unification_constraint> & uc) {
|
||||||
|
@ -315,11 +314,6 @@ public:
|
||||||
check_type(t, e, ctx);
|
check_type(t, e, ctx);
|
||||||
}
|
}
|
||||||
|
|
||||||
void set_interrupt(bool flag) {
|
|
||||||
m_interrupted = flag;
|
|
||||||
m_normalizer.set_interrupt(flag);
|
|
||||||
}
|
|
||||||
|
|
||||||
void clear() {
|
void clear() {
|
||||||
m_cache.clear();
|
m_cache.clear();
|
||||||
m_normalizer.clear();
|
m_normalizer.clear();
|
||||||
|
@ -349,7 +343,6 @@ void type_checker::check_type(expr const & e, context const & ctx) {
|
||||||
m_ptr->check_type(e, ctx);
|
m_ptr->check_type(e, ctx);
|
||||||
}
|
}
|
||||||
void type_checker::clear() { m_ptr->clear(); }
|
void type_checker::clear() { m_ptr->clear(); }
|
||||||
void type_checker::set_interrupt(bool flag) { m_ptr->set_interrupt(flag); }
|
|
||||||
normalizer & type_checker::get_normalizer() { return m_ptr->get_normalizer(); }
|
normalizer & type_checker::get_normalizer() { return m_ptr->get_normalizer(); }
|
||||||
|
|
||||||
expr infer_type(expr const & e, environment const & env, context const & ctx) {
|
expr infer_type(expr const & e, environment const & env, context const & ctx) {
|
||||||
|
|
|
@ -56,11 +56,6 @@ public:
|
||||||
/** \brief Reset internal caches */
|
/** \brief Reset internal caches */
|
||||||
void clear();
|
void clear();
|
||||||
|
|
||||||
/** \brief Interrupt type checker */
|
|
||||||
void set_interrupt(bool flag);
|
|
||||||
void interrupt() { set_interrupt(true); }
|
|
||||||
void reset_interrupt() { set_interrupt(false); }
|
|
||||||
|
|
||||||
/** \brief Return reference to the normalizer used by this type checker. */
|
/** \brief Return reference to the normalizer used by this type checker. */
|
||||||
normalizer & get_normalizer();
|
normalizer & get_normalizer();
|
||||||
};
|
};
|
||||||
|
|
|
@ -8,6 +8,7 @@ Author: Leonardo de Moura
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include <utility>
|
#include <utility>
|
||||||
#include "util/pdeque.h"
|
#include "util/pdeque.h"
|
||||||
|
#include "util/interrupt.h"
|
||||||
#include "kernel/formatter.h"
|
#include "kernel/formatter.h"
|
||||||
#include "kernel/free_vars.h"
|
#include "kernel/free_vars.h"
|
||||||
#include "kernel/normalizer.h"
|
#include "kernel/normalizer.h"
|
||||||
|
@ -143,8 +144,6 @@ class elaborator::imp {
|
||||||
int m_quota;
|
int m_quota;
|
||||||
justification m_conflict;
|
justification m_conflict;
|
||||||
bool m_first;
|
bool m_first;
|
||||||
bool m_interrupted;
|
|
||||||
|
|
||||||
|
|
||||||
// options
|
// options
|
||||||
bool m_use_justifications;
|
bool m_use_justifications;
|
||||||
|
@ -596,7 +595,7 @@ class elaborator::imp {
|
||||||
context const & ctx = get_context(c);
|
context const & ctx = get_context(c);
|
||||||
bool modified = false;
|
bool modified = false;
|
||||||
while (true) {
|
while (true) {
|
||||||
check_interrupted(m_interrupted);
|
check_interrupted();
|
||||||
expr new_a = normalize_step(ctx, a);
|
expr new_a = normalize_step(ctx, a);
|
||||||
expr new_b = normalize_step(ctx, b);
|
expr new_b = normalize_step(ctx, b);
|
||||||
if (new_a == a && new_b == b) {
|
if (new_a == a && new_b == b) {
|
||||||
|
@ -1380,13 +1379,12 @@ public:
|
||||||
set_options(opts);
|
set_options(opts);
|
||||||
m_next_id = 0;
|
m_next_id = 0;
|
||||||
m_quota = 0;
|
m_quota = 0;
|
||||||
m_interrupted = false;
|
|
||||||
m_first = true;
|
m_first = true;
|
||||||
// display(std::cout);
|
// display(std::cout);
|
||||||
}
|
}
|
||||||
|
|
||||||
metavar_env next() {
|
metavar_env next() {
|
||||||
check_interrupted(m_interrupted);
|
check_interrupted();
|
||||||
if (m_conflict)
|
if (m_conflict)
|
||||||
throw elaborator_exception(m_conflict);
|
throw elaborator_exception(m_conflict);
|
||||||
if (!m_case_splits.empty()) {
|
if (!m_case_splits.empty()) {
|
||||||
|
@ -1404,7 +1402,7 @@ public:
|
||||||
}
|
}
|
||||||
reset_quota();
|
reset_quota();
|
||||||
while (true) {
|
while (true) {
|
||||||
check_interrupted(m_interrupted);
|
check_interrupted();
|
||||||
cnstr_queue & q = m_state.m_queue;
|
cnstr_queue & q = m_state.m_queue;
|
||||||
if (q.empty() || m_quota < - static_cast<int>(q.size()) - 10) {
|
if (q.empty() || m_quota < - static_cast<int>(q.size()) - 10) {
|
||||||
// TODO(Leo): implement interface with synthesizer
|
// TODO(Leo): implement interface with synthesizer
|
||||||
|
@ -1420,12 +1418,6 @@ public:
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void interrupt() {
|
|
||||||
m_interrupted = true;
|
|
||||||
m_type_inferer.set_interrupt(true);
|
|
||||||
m_normalizer.set_interrupt(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
void display(std::ostream & out, unification_constraint const & c) const {
|
void display(std::ostream & out, unification_constraint const & c) const {
|
||||||
formatter fmt = mk_simple_formatter();
|
formatter fmt = mk_simple_formatter();
|
||||||
out << c.pp(fmt, options(), nullptr, false) << "\n";
|
out << c.pp(fmt, options(), nullptr, false) << "\n";
|
||||||
|
@ -1462,8 +1454,4 @@ elaborator::~elaborator() {
|
||||||
metavar_env elaborator::next() {
|
metavar_env elaborator::next() {
|
||||||
return m_ptr->next();
|
return m_ptr->next();
|
||||||
}
|
}
|
||||||
|
|
||||||
void elaborator::interrupt() {
|
|
||||||
m_ptr->interrupt();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -61,7 +61,5 @@ public:
|
||||||
~elaborator();
|
~elaborator();
|
||||||
|
|
||||||
metavar_env next();
|
metavar_env next();
|
||||||
void interrupt();
|
|
||||||
void set_interrupt(bool ) { interrupt(); }
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
|
@ -29,7 +29,6 @@ public:
|
||||||
*/
|
*/
|
||||||
virtual std::pair<metavar_env, list<unification_constraint>> next(justification const & assumption) = 0;
|
virtual std::pair<metavar_env, list<unification_constraint>> next(justification const & assumption) = 0;
|
||||||
/** \brief Interrupt the computation for the next solution. */
|
/** \brief Interrupt the computation for the next solution. */
|
||||||
virtual void interrupt() = 0;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -25,7 +25,6 @@ public:
|
||||||
/** \brief Return the next possible solution. An elaborator_exception is throw in case of failure. */
|
/** \brief Return the next possible solution. An elaborator_exception is throw in case of failure. */
|
||||||
virtual expr next() = 0;
|
virtual expr next() = 0;
|
||||||
/** \brief Interrupt the computation for the next solution. */
|
/** \brief Interrupt the computation for the next solution. */
|
||||||
virtual void interrupt() = 0;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -39,7 +39,6 @@ public:
|
||||||
template<typename T> void set_option(name const & n, T const & v) {
|
template<typename T> void set_option(name const & n, T const & v) {
|
||||||
set_options(get_options().update(n, v));
|
set_options(get_options().update(n, v));
|
||||||
}
|
}
|
||||||
void set_interrupt(bool flag) { m_formatter.set_interrupt(flag); }
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -6,6 +6,7 @@ Author: Leonardo de Moura
|
||||||
*/
|
*/
|
||||||
#include "util/flet.h"
|
#include "util/flet.h"
|
||||||
#include "util/scoped_map.h"
|
#include "util/scoped_map.h"
|
||||||
|
#include "util/interrupt.h"
|
||||||
#include "kernel/environment.h"
|
#include "kernel/environment.h"
|
||||||
#include "kernel/normalizer.h"
|
#include "kernel/normalizer.h"
|
||||||
#include "kernel/builtin.h"
|
#include "kernel/builtin.h"
|
||||||
|
@ -29,7 +30,6 @@ class type_inferer::imp {
|
||||||
unification_constraints * m_uc;
|
unification_constraints * m_uc;
|
||||||
normalizer m_normalizer;
|
normalizer m_normalizer;
|
||||||
cache m_cache;
|
cache m_cache;
|
||||||
volatile bool m_interrupted;
|
|
||||||
|
|
||||||
expr normalize(expr const & e, context const & ctx) {
|
expr normalize(expr const & e, context const & ctx) {
|
||||||
return m_normalizer(e, ctx);
|
return m_normalizer(e, ctx);
|
||||||
|
@ -139,7 +139,7 @@ class type_inferer::imp {
|
||||||
break; // expensive cases
|
break; // expensive cases
|
||||||
}
|
}
|
||||||
|
|
||||||
check_interrupted(m_interrupted);
|
check_interrupted();
|
||||||
bool shared = false;
|
bool shared = false;
|
||||||
if (is_shared(e)) {
|
if (is_shared(e)) {
|
||||||
shared = true;
|
shared = true;
|
||||||
|
@ -214,7 +214,6 @@ public:
|
||||||
imp(environment const & env):
|
imp(environment const & env):
|
||||||
m_env(env),
|
m_env(env),
|
||||||
m_normalizer(env) {
|
m_normalizer(env) {
|
||||||
m_interrupted = false;
|
|
||||||
m_menv = nullptr;
|
m_menv = nullptr;
|
||||||
m_menv_timestamp = 0;
|
m_menv_timestamp = 0;
|
||||||
m_uc = nullptr;
|
m_uc = nullptr;
|
||||||
|
@ -227,11 +226,6 @@ public:
|
||||||
return infer_type(e, ctx);
|
return infer_type(e, ctx);
|
||||||
}
|
}
|
||||||
|
|
||||||
void set_interrupt(bool flag) {
|
|
||||||
m_interrupted = flag;
|
|
||||||
m_normalizer.set_interrupt(flag);
|
|
||||||
}
|
|
||||||
|
|
||||||
void clear() {
|
void clear() {
|
||||||
m_cache.clear();
|
m_cache.clear();
|
||||||
m_normalizer.clear();
|
m_normalizer.clear();
|
||||||
|
@ -250,5 +244,4 @@ expr type_inferer::operator()(expr const & e, context const & ctx) {
|
||||||
return operator()(e, ctx, nullptr, uc);
|
return operator()(e, ctx, nullptr, uc);
|
||||||
}
|
}
|
||||||
void type_inferer::clear() { m_ptr->clear(); }
|
void type_inferer::clear() { m_ptr->clear(); }
|
||||||
void type_inferer::set_interrupt(bool flag) { m_ptr->set_interrupt(flag); }
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -35,9 +35,5 @@ public:
|
||||||
expr operator()(expr const & e, context const & ctx = context());
|
expr operator()(expr const & e, context const & ctx = context());
|
||||||
|
|
||||||
void clear();
|
void clear();
|
||||||
|
|
||||||
void set_interrupt(bool flag);
|
|
||||||
void interrupt() { set_interrupt(true); }
|
|
||||||
void reset_interrupt() { set_interrupt(false); }
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,23 +7,19 @@ Author: Leonardo de Moura
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <fstream>
|
#include <fstream>
|
||||||
#include <signal.h>
|
#include <signal.h>
|
||||||
#include "util/interruptable_ptr.h"
|
#include "util/interrupt.h"
|
||||||
#include "kernel/printer.h"
|
#include "kernel/printer.h"
|
||||||
#include "frontends/lean/parser.h"
|
#include "frontends/lean/parser.h"
|
||||||
#include "bindings/lua/leanlua_state.h"
|
#include "bindings/lua/leanlua_state.h"
|
||||||
#include "version.h"
|
#include "version.h"
|
||||||
|
|
||||||
using lean::interruptable_ptr;
|
|
||||||
using lean::shell;
|
using lean::shell;
|
||||||
using lean::frontend;
|
using lean::frontend;
|
||||||
using lean::scoped_set_interruptable_ptr;
|
|
||||||
using lean::parser;
|
using lean::parser;
|
||||||
using lean::leanlua_state;
|
using lean::leanlua_state;
|
||||||
|
|
||||||
static interruptable_ptr<shell> g_lean_shell;
|
|
||||||
|
|
||||||
static void on_ctrl_c(int ) {
|
static void on_ctrl_c(int ) {
|
||||||
g_lean_shell.set_interrupt(true);
|
lean::request_interrupt();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool lean_shell() {
|
bool lean_shell() {
|
||||||
|
@ -32,7 +28,6 @@ bool lean_shell() {
|
||||||
frontend f;
|
frontend f;
|
||||||
leanlua_state S;
|
leanlua_state S;
|
||||||
shell sh(f, &S);
|
shell sh(f, &S);
|
||||||
scoped_set_interruptable_ptr<shell> set(g_lean_shell, &sh);
|
|
||||||
signal(SIGINT, on_ctrl_c);
|
signal(SIGINT, on_ctrl_c);
|
||||||
return sh();
|
return sh();
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,6 +10,7 @@ Author: Leonardo de Moura
|
||||||
#include "util/test.h"
|
#include "util/test.h"
|
||||||
#include "util/trace.h"
|
#include "util/trace.h"
|
||||||
#include "util/exception.h"
|
#include "util/exception.h"
|
||||||
|
#include "util/interrupt.h"
|
||||||
#include "kernel/normalizer.h"
|
#include "kernel/normalizer.h"
|
||||||
#include "kernel/builtin.h"
|
#include "kernel/builtin.h"
|
||||||
#include "kernel/expr_sets.h"
|
#include "kernel/expr_sets.h"
|
||||||
|
@ -219,7 +220,7 @@ static void tst5() {
|
||||||
env.add_var("a", Bool);
|
env.add_var("a", Bool);
|
||||||
normalizer proc(env);
|
normalizer proc(env);
|
||||||
std::chrono::milliseconds dura(50);
|
std::chrono::milliseconds dura(50);
|
||||||
std::thread thread([&]() {
|
interruptible_thread thread([&]() {
|
||||||
try {
|
try {
|
||||||
proc(t);
|
proc(t);
|
||||||
// Remark: if the following code is reached, we
|
// Remark: if the following code is reached, we
|
||||||
|
@ -230,7 +231,8 @@ static void tst5() {
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
std::this_thread::sleep_for(dura);
|
std::this_thread::sleep_for(dura);
|
||||||
proc.interrupt();
|
while (!thread.request_interrupt()) {
|
||||||
|
}
|
||||||
thread.join();
|
thread.join();
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,6 +11,7 @@ Author: Leonardo de Moura
|
||||||
#include "util/test.h"
|
#include "util/test.h"
|
||||||
#include "util/trace.h"
|
#include "util/trace.h"
|
||||||
#include "util/exception.h"
|
#include "util/exception.h"
|
||||||
|
#include "util/interrupt.h"
|
||||||
#include "kernel/type_checker.h"
|
#include "kernel/type_checker.h"
|
||||||
#include "kernel/environment.h"
|
#include "kernel/environment.h"
|
||||||
#include "kernel/abstract.h"
|
#include "kernel/abstract.h"
|
||||||
|
@ -227,7 +228,7 @@ static void tst12() {
|
||||||
env.add_var("a", Int);
|
env.add_var("a", Int);
|
||||||
type_checker checker(env);
|
type_checker checker(env);
|
||||||
std::chrono::milliseconds dura(100);
|
std::chrono::milliseconds dura(100);
|
||||||
std::thread thread([&]() {
|
interruptible_thread thread([&]() {
|
||||||
try {
|
try {
|
||||||
std::cout << checker.infer_type(t) << "\n";
|
std::cout << checker.infer_type(t) << "\n";
|
||||||
// Remark: if the following code is reached, we
|
// Remark: if the following code is reached, we
|
||||||
|
@ -238,7 +239,8 @@ static void tst12() {
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
std::this_thread::sleep_for(dura);
|
std::this_thread::sleep_for(dura);
|
||||||
checker.interrupt();
|
while (!thread.request_interrupt()) {
|
||||||
|
}
|
||||||
thread.join();
|
thread.join();
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
|
@ -39,8 +39,6 @@ static void tst1() {
|
||||||
check(fmt(ctx), "[x : f a; y : f x (N -> N); z : N := y == x]");
|
check(fmt(ctx), "[x : f a; y : f x (N -> N); z : N := y == x]");
|
||||||
check(fmt(ctx, f(Var(0), Var(2))), "f z x");
|
check(fmt(ctx, f(Var(0), Var(2))), "f z x");
|
||||||
check(fmt(ctx, f(Var(0), Var(2)), true), "[x : f a; y : f x (N -> N); z : N := y == x] |- f z x");
|
check(fmt(ctx, f(Var(0), Var(2)), true), "[x : f a; y : f x (N -> N); z : N := y == x] |- f z x");
|
||||||
fmt.set_interrupt(true);
|
|
||||||
fmt.set_interrupt(false);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int main() {
|
int main() {
|
||||||
|
|
|
@ -18,12 +18,12 @@ void reset_interrupt() {
|
||||||
g_interrupt.store(false);
|
g_interrupt.store(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool interrupted() {
|
bool interrupt_requested() {
|
||||||
return g_interrupt.load();
|
return g_interrupt.load();
|
||||||
}
|
}
|
||||||
|
|
||||||
void check_interrupt() {
|
void check_interrupted() {
|
||||||
if (interrupted())
|
if (interrupt_requested())
|
||||||
throw interrupted();
|
throw interrupted();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -22,12 +22,12 @@ void reset_interrupt();
|
||||||
/**
|
/**
|
||||||
\brief Return true iff the current thread was marked for interruption.
|
\brief Return true iff the current thread was marked for interruption.
|
||||||
*/
|
*/
|
||||||
bool interrupted();
|
bool interrupt_requested();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\brief Throw an interrupt exception if the (interrupt) flag is set.
|
\brief Throw an interrupted exception if the (interrupt) flag is set.
|
||||||
*/
|
*/
|
||||||
void check_interrupt();
|
void check_interrupted();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\brief Thread that provides a method for setting its interrupt flag.
|
\brief Thread that provides a method for setting its interrupt flag.
|
||||||
|
|
Loading…
Reference in a new issue