2013-10-01 00:58:46 +00:00
|
|
|
/*
|
|
|
|
Copyright (c) 2013 Microsoft Corporation. All rights reserved.
|
|
|
|
Released under Apache 2.0 license as described in the file LICENSE.
|
|
|
|
|
|
|
|
Author: Leonardo de Moura
|
|
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <algorithm>
|
|
|
|
#include "util/debug.h"
|
|
|
|
#include "util/rc.h"
|
|
|
|
#include "util/buffer.h"
|
|
|
|
#include "util/sexpr/format.h"
|
2013-10-01 17:25:18 +00:00
|
|
|
#include "kernel/expr.h"
|
|
|
|
#include "kernel/formatter.h"
|
2013-10-14 21:37:35 +00:00
|
|
|
#include "kernel/pos_info_provider.h"
|
2013-10-01 00:58:46 +00:00
|
|
|
|
|
|
|
namespace lean {
|
2014-02-21 01:19:03 +00:00
|
|
|
class substitution;
|
|
|
|
struct justification_cell;
|
2013-10-01 00:58:46 +00:00
|
|
|
|
2013-10-26 21:21:29 +00:00
|
|
|
/**
|
2014-02-21 01:19:03 +00:00
|
|
|
\brief When a set of constraints cannot be solved, we must print some "error" message to the user.
|
|
|
|
The pp_jst_fn is a generic funciton that produces these messages. We can associate these functions
|
|
|
|
to justification objects.
|
2013-10-26 21:21:29 +00:00
|
|
|
*/
|
2015-06-15 02:44:00 +00:00
|
|
|
typedef std::function<format(formatter const &, pos_info_provider const *, substitution const &, bool, bool)> pp_jst_fn;
|
2015-05-21 00:21:27 +00:00
|
|
|
|
|
|
|
class justification_set;
|
2013-10-26 21:21:29 +00:00
|
|
|
|
2013-10-01 00:58:46 +00:00
|
|
|
/**
|
2014-02-21 01:19:03 +00:00
|
|
|
\brief Objects used to justify unification (and level) constraints and metavariable assignments.
|
|
|
|
|
|
|
|
There are 4 basic kinds of justification:
|
|
|
|
- None: the null justification.
|
|
|
|
- Asserted: a justification produced by the type checker.
|
|
|
|
- Assumption: for case-splits inside the elaborator and other constraint solvers.
|
|
|
|
- Composite: the combination of two other justification objects.
|
|
|
|
|
|
|
|
A composite justification is used, for example, to justify a constraint produced by
|
|
|
|
replacing a metavariable ?M by its assignment in a constraint C.
|
2013-10-01 00:58:46 +00:00
|
|
|
*/
|
2013-10-23 20:42:14 +00:00
|
|
|
class justification {
|
|
|
|
justification_cell * m_ptr;
|
2014-02-21 01:19:03 +00:00
|
|
|
justification(justification_cell * ptr);
|
2015-05-21 00:21:27 +00:00
|
|
|
format pp_core(formatter const & fmt, pos_info_provider const * p, substitution const & s,
|
2015-06-15 02:44:00 +00:00
|
|
|
justification_set & visited, bool is_main, bool as_error) const;
|
2013-10-01 00:58:46 +00:00
|
|
|
public:
|
2014-02-21 01:19:03 +00:00
|
|
|
justification();
|
|
|
|
justification(justification const & s);
|
|
|
|
justification(justification && s);
|
|
|
|
~justification();
|
2013-10-01 00:58:46 +00:00
|
|
|
|
2014-02-21 01:19:03 +00:00
|
|
|
friend void swap(justification & a, justification & b) { std::swap(a.m_ptr, b.m_ptr); }
|
|
|
|
|
|
|
|
bool is_none() const;
|
|
|
|
bool is_asserted() const;
|
|
|
|
bool is_assumption() const;
|
|
|
|
bool is_composite() const;
|
2015-04-24 21:39:26 +00:00
|
|
|
bool is_wrapper() const;
|
2013-10-01 00:58:46 +00:00
|
|
|
|
2013-10-23 20:42:14 +00:00
|
|
|
justification_cell * raw() const { return m_ptr; }
|
2013-10-01 00:58:46 +00:00
|
|
|
|
2014-02-21 01:19:03 +00:00
|
|
|
justification & operator=(justification const & s);
|
|
|
|
justification & operator=(justification && s);
|
|
|
|
/**
|
|
|
|
\brief Convert this justification into a format object. This method is usually used to report
|
|
|
|
"error" messages to users.
|
|
|
|
*/
|
2015-06-15 02:44:00 +00:00
|
|
|
format pp(formatter const & fmt, pos_info_provider const * p, substitution const & s, bool as_error = true) const;
|
2014-02-21 01:19:03 +00:00
|
|
|
/**
|
|
|
|
\brief Return an expression associated with the justification object.
|
|
|
|
*/
|
|
|
|
optional<expr> get_main_expr() const;
|
2013-10-01 00:58:46 +00:00
|
|
|
|
2014-02-25 00:30:56 +00:00
|
|
|
friend justification mk_composite(justification const & j1, justification const & j2, optional<expr> const & s, pp_jst_fn const & fn);
|
2014-02-21 01:19:03 +00:00
|
|
|
friend justification mk_composite1(justification const & j1, justification const & j2);
|
2015-04-24 21:39:26 +00:00
|
|
|
friend justification mk_wrapper(justification const & j, optional<expr> const & s, pp_jst_fn const & fn);
|
2014-02-25 00:30:56 +00:00
|
|
|
friend justification mk_assumption_justification(unsigned idx, optional<expr> const & s, pp_jst_fn const & fn);
|
2014-02-21 01:19:03 +00:00
|
|
|
friend justification mk_assumption_justification(unsigned idx);
|
2014-02-25 00:30:56 +00:00
|
|
|
friend justification mk_justification(optional<expr> const & s, pp_jst_fn const & fn);
|
2014-05-02 19:15:29 +00:00
|
|
|
|
|
|
|
friend bool is_eqp(justification const & j1, justification const & j2) { return j1.raw() == j2.raw(); }
|
2013-10-01 00:58:46 +00:00
|
|
|
};
|
2013-10-14 18:50:51 +00:00
|
|
|
|
2014-02-21 01:19:03 +00:00
|
|
|
/**
|
2014-04-24 22:08:10 +00:00
|
|
|
\brief Simpler version of pp_jst_fn
|
2014-02-21 01:19:03 +00:00
|
|
|
*/
|
2015-06-15 02:44:00 +00:00
|
|
|
typedef std::function<format(formatter const &, substitution const &, bool)> pp_jst_sfn;
|
2014-02-21 01:19:03 +00:00
|
|
|
|
|
|
|
/** \brief Return a format object containing position information for the given expression (if available) */
|
|
|
|
format to_pos(optional<expr> const & e, pos_info_provider const * p);
|
2015-05-21 00:21:27 +00:00
|
|
|
format pp_previous_error_header(formatter const &, pos_info_provider const * pos_prov, optional<expr> const & ref, bool is_main);
|
2013-10-26 21:21:29 +00:00
|
|
|
|
2015-04-24 21:39:26 +00:00
|
|
|
/** \brief Provide a custom pretty printer for \c j */
|
|
|
|
justification mk_wrapper(justification const & j, optional<expr> const & s, pp_jst_fn const & fn);
|
2013-10-14 18:50:51 +00:00
|
|
|
/**
|
2014-02-21 01:19:03 +00:00
|
|
|
\brief Combine the two given justifications into a new justification object, and use
|
|
|
|
the given function to convert the justification into a format object.
|
|
|
|
*/
|
2014-02-25 00:30:56 +00:00
|
|
|
justification mk_composite(justification const & j1, justification const & j2, optional<expr> const & s, pp_jst_fn const & fn);
|
2014-02-21 01:19:03 +00:00
|
|
|
/**
|
|
|
|
\brief Similar to \c mk_composite, but uses \c j1.pp to convert the
|
|
|
|
resulting justification into a format object.
|
|
|
|
*/
|
|
|
|
justification mk_composite1(justification const & j1, justification const & j2);
|
|
|
|
/**
|
|
|
|
\brief Alias for \c mk_composite1
|
|
|
|
*/
|
|
|
|
justification mk_substitution_justification(justification const & j1, justification const & j2);
|
|
|
|
/**
|
|
|
|
\brief Create a "case-split" justification with the given \c idx.
|
|
|
|
*/
|
2014-02-25 00:30:56 +00:00
|
|
|
justification mk_assumption_justification(unsigned idx, optional<expr> const & s, pp_jst_fn const & fn);
|
2014-02-21 01:19:03 +00:00
|
|
|
/**
|
|
|
|
\brief Similar to the previous function, but fn just returns the format object "assumption idx", and s is the none.
|
|
|
|
*/
|
|
|
|
justification mk_assumption_justification(unsigned idx);
|
|
|
|
/**
|
|
|
|
\brief Create a justification for constraints produced by the type checker.
|
|
|
|
*/
|
2014-02-25 00:30:56 +00:00
|
|
|
justification mk_justification(optional<expr> const & s, pp_jst_fn const & fn);
|
2014-05-15 18:16:48 +00:00
|
|
|
justification mk_justification(char const * msg, optional<expr> const & s = none_expr());
|
2014-02-21 01:19:03 +00:00
|
|
|
/**
|
|
|
|
\brief Create a justification for constraints produced by the type checker.
|
|
|
|
It is similar to the previous function, but the position of \c s will be automatically included.
|
2013-10-14 18:50:51 +00:00
|
|
|
*/
|
2014-02-25 00:30:56 +00:00
|
|
|
justification mk_justification(optional<expr> const & s, pp_jst_sfn const & fn);
|
|
|
|
inline justification mk_justification(expr const & s, pp_jst_sfn const & fn) {
|
|
|
|
return mk_justification(some_expr(s), fn);
|
|
|
|
}
|
2013-10-27 18:02:29 +00:00
|
|
|
|
2015-04-24 21:39:26 +00:00
|
|
|
/**
|
|
|
|
\brief Return the child of a wrapper justification.
|
|
|
|
\pre j.is_composite()
|
|
|
|
*/
|
|
|
|
justification const & wrapper_child(justification const & j);
|
2014-02-21 01:19:03 +00:00
|
|
|
/**
|
|
|
|
\brief Return the first child of a composite justification.
|
|
|
|
\pre j.is_composite()
|
|
|
|
*/
|
|
|
|
justification const & composite_child1(justification const & j);
|
|
|
|
/**
|
|
|
|
\brief Return the second child of a composite justification.
|
|
|
|
\pre j.is_composite()
|
|
|
|
*/
|
|
|
|
justification const & composite_child2(justification const & j);
|
|
|
|
/**
|
|
|
|
\brief Return the index of an assumption justification.
|
|
|
|
\pre j.is_assumption()
|
|
|
|
*/
|
|
|
|
unsigned assumption_idx(justification const & j);
|
2013-10-27 18:02:29 +00:00
|
|
|
|
2014-02-21 01:19:03 +00:00
|
|
|
/** \brief Return true iff \c j depends on the assumption with index \c i. */
|
|
|
|
bool depends_on(justification const & j, unsigned i);
|
2014-02-23 00:12:06 +00:00
|
|
|
|
|
|
|
/** \brief Printer for debugging purposes */
|
|
|
|
std::ostream & operator<<(std::ostream & out, justification const & j);
|
2014-05-15 17:46:04 +00:00
|
|
|
|
|
|
|
/** \brief Object to simulate delayed justification creation. */
|
|
|
|
class delayed_justification {
|
2014-05-28 06:16:49 +00:00
|
|
|
public:
|
|
|
|
virtual ~delayed_justification() {}
|
|
|
|
virtual justification get() = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
class no_delayed_justification : public delayed_justification {
|
|
|
|
public:
|
|
|
|
virtual justification get() { return justification(); }
|
|
|
|
};
|
|
|
|
|
|
|
|
class simple_delayed_justification : public delayed_justification {
|
2014-05-15 17:46:04 +00:00
|
|
|
optional<justification> m_jst;
|
|
|
|
std::function<justification()> m_mk;
|
|
|
|
public:
|
2014-05-28 06:16:49 +00:00
|
|
|
template<typename Mk> simple_delayed_justification(Mk && mk):m_mk(mk) {}
|
|
|
|
virtual justification get() { if (!m_jst) { m_jst = m_mk(); } return *m_jst; }
|
2014-05-15 17:46:04 +00:00
|
|
|
};
|
2014-06-22 02:45:20 +00:00
|
|
|
|
|
|
|
class as_delayed_justification : public delayed_justification {
|
|
|
|
justification m_jst;
|
|
|
|
public:
|
|
|
|
as_delayed_justification(justification const & j):m_jst(j) {}
|
|
|
|
virtual justification get() { return m_jst; }
|
|
|
|
};
|
2013-10-01 00:58:46 +00:00
|
|
|
}
|