2013-07-22 11:03:46 +00:00
|
|
|
/*
|
2014-04-25 22:26:03 +00:00
|
|
|
Copyright (c) 2013-2014 Microsoft Corporation. All rights reserved.
|
2013-07-22 11:03:46 +00:00
|
|
|
Released under Apache 2.0 license as described in the file LICENSE.
|
|
|
|
|
|
|
|
Author: Leonardo de Moura
|
|
|
|
*/
|
|
|
|
#pragma once
|
2013-09-13 23:14:24 +00:00
|
|
|
#include <algorithm>
|
2013-07-22 11:03:46 +00:00
|
|
|
#include <iostream>
|
|
|
|
#include <limits>
|
2013-09-13 10:35:29 +00:00
|
|
|
#include <utility>
|
2013-10-26 18:07:06 +00:00
|
|
|
#include <tuple>
|
2013-12-28 09:16:50 +00:00
|
|
|
#include <string>
|
2013-12-09 22:56:48 +00:00
|
|
|
#include "util/thread.h"
|
2013-11-27 03:15:49 +00:00
|
|
|
#include "util/lua.h"
|
2013-09-13 03:04:10 +00:00
|
|
|
#include "util/rc.h"
|
|
|
|
#include "util/name.h"
|
|
|
|
#include "util/hash.h"
|
|
|
|
#include "util/buffer.h"
|
2013-09-13 01:25:38 +00:00
|
|
|
#include "util/list_fn.h"
|
2013-12-08 07:21:07 +00:00
|
|
|
#include "util/optional.h"
|
2013-12-28 09:16:50 +00:00
|
|
|
#include "util/serializer.h"
|
2013-09-13 03:04:10 +00:00
|
|
|
#include "util/sexpr/format.h"
|
2014-06-30 16:14:55 +00:00
|
|
|
#include "util/name_generator.h"
|
2013-09-13 03:04:10 +00:00
|
|
|
#include "kernel/level.h"
|
2014-07-10 17:32:00 +00:00
|
|
|
#include "kernel/formatter.h"
|
2014-04-25 22:26:03 +00:00
|
|
|
#include "kernel/extension_context.h"
|
2014-10-17 23:39:39 +00:00
|
|
|
#include "kernel/expr_eq_fn.h"
|
2013-07-22 11:03:46 +00:00
|
|
|
|
|
|
|
namespace lean {
|
2014-05-12 23:33:22 +00:00
|
|
|
// Tags are used by frontends to mark expressions. They are automatically propagated by
|
|
|
|
// procedures such as update_app, update_binder, etc.
|
|
|
|
typedef unsigned tag;
|
|
|
|
constexpr tag nulltag = std::numeric_limits<unsigned>::max();
|
2014-02-15 03:55:02 +00:00
|
|
|
class expr;
|
2013-07-22 11:03:46 +00:00
|
|
|
/* =======================================
|
|
|
|
Expressions
|
2013-08-03 23:09:21 +00:00
|
|
|
expr ::= Var idx
|
2014-02-15 03:55:02 +00:00
|
|
|
| Sort level
|
|
|
|
| Constant name [levels]
|
|
|
|
| Meta name expr
|
|
|
|
| Local name expr
|
|
|
|
| App expr expr
|
2013-08-03 23:09:21 +00:00
|
|
|
| Lambda name expr expr
|
|
|
|
| Pi name expr expr
|
2013-07-22 11:03:46 +00:00
|
|
|
|
2014-02-15 03:55:02 +00:00
|
|
|
| Macro macro
|
2013-07-22 11:03:46 +00:00
|
|
|
*/
|
2014-06-24 23:27:23 +00:00
|
|
|
enum class expr_kind { Var, Sort, Constant, Meta, Local, App, Lambda, Pi, Macro };
|
2013-07-22 11:03:46 +00:00
|
|
|
class expr_cell {
|
|
|
|
protected:
|
2013-12-21 21:59:45 +00:00
|
|
|
// The bits of the following field mean:
|
2014-04-22 22:38:18 +00:00
|
|
|
// 0-1 - term is an arrow (0 - not initialized, 1 - is arrow, 2 - is not arrow)
|
2014-02-20 18:07:38 +00:00
|
|
|
// Remark: we use atomic_uchar because these flags are computed lazily (i.e., after the expression is created)
|
|
|
|
atomic_uchar m_flags;
|
2014-05-12 23:33:22 +00:00
|
|
|
unsigned m_kind:8;
|
2014-07-05 20:11:30 +00:00
|
|
|
unsigned m_has_expr_mv:1; // term contains expression metavariables
|
|
|
|
unsigned m_has_univ_mv:1; // term contains universe metavariables
|
2014-03-01 01:16:34 +00:00
|
|
|
unsigned m_has_local:1; // term contains local constants
|
|
|
|
unsigned m_has_param_univ:1; // term constains parametric universe levels
|
2014-05-12 23:33:22 +00:00
|
|
|
unsigned m_hash; // hash based on the structure of the expression (this is a good hash for structural equality)
|
|
|
|
unsigned m_hash_alloc; // hash based on 'time' of allocation (this is a good hash for pointer-based equality)
|
|
|
|
atomic_uint m_tag;
|
2013-07-22 11:03:46 +00:00
|
|
|
MK_LEAN_RC(); // Declare m_rc counter
|
|
|
|
void dealloc();
|
2013-07-23 18:31:31 +00:00
|
|
|
|
2013-12-21 21:59:45 +00:00
|
|
|
optional<bool> is_arrow() const;
|
|
|
|
void set_is_arrow(bool flag);
|
|
|
|
friend bool is_arrow(expr const & e);
|
|
|
|
|
2014-04-17 18:31:48 +00:00
|
|
|
static void dec_ref(expr & c, buffer<expr_cell*> & todelete);
|
2013-07-22 11:03:46 +00:00
|
|
|
public:
|
2014-10-15 20:12:09 +00:00
|
|
|
expr_cell(expr_kind k, unsigned h, bool has_expr_mv, bool has_univ_mv, bool has_local, bool has_param_univ, tag g);
|
2013-07-23 15:59:39 +00:00
|
|
|
expr_kind kind() const { return static_cast<expr_kind>(m_kind); }
|
2013-07-22 11:03:46 +00:00
|
|
|
unsigned hash() const { return m_hash; }
|
2013-11-14 10:31:54 +00:00
|
|
|
unsigned hash_alloc() const { return m_hash_alloc; }
|
2014-07-05 20:11:30 +00:00
|
|
|
bool has_expr_metavar() const { return m_has_expr_mv; }
|
|
|
|
bool has_univ_metavar() const { return m_has_univ_mv; }
|
2014-02-20 18:07:38 +00:00
|
|
|
bool has_local() const { return m_has_local; }
|
2014-03-01 01:16:34 +00:00
|
|
|
bool has_param_univ() const { return m_has_param_univ; }
|
2014-05-12 23:33:22 +00:00
|
|
|
void set_tag(tag t);
|
|
|
|
tag get_tag() const { return m_tag; }
|
2013-07-22 11:03:46 +00:00
|
|
|
};
|
2014-02-15 03:55:02 +00:00
|
|
|
|
2014-10-16 19:48:49 +00:00
|
|
|
typedef expr_cell * expr_ptr;
|
|
|
|
|
2014-04-25 22:02:52 +00:00
|
|
|
class macro_definition;
|
2014-05-16 18:13:50 +00:00
|
|
|
class binder_info;
|
2014-02-15 03:55:02 +00:00
|
|
|
|
2013-07-22 11:03:46 +00:00
|
|
|
/**
|
|
|
|
\brief Exprs for encoding formulas/expressions, types and proofs.
|
|
|
|
*/
|
|
|
|
class expr {
|
|
|
|
private:
|
|
|
|
expr_cell * m_ptr;
|
2013-12-08 18:17:29 +00:00
|
|
|
explicit expr(expr_cell * ptr):m_ptr(ptr) { if (m_ptr) m_ptr->inc_ref(); }
|
2013-11-19 02:41:08 +00:00
|
|
|
friend class expr_cell;
|
|
|
|
expr_cell * steal_ptr() { expr_cell * r = m_ptr; m_ptr = nullptr; return r; }
|
2013-12-08 18:17:29 +00:00
|
|
|
friend class optional<expr>;
|
2013-07-22 11:03:46 +00:00
|
|
|
public:
|
2013-12-08 22:45:18 +00:00
|
|
|
/**
|
|
|
|
\brief The default constructor creates a reference to a "dummy"
|
|
|
|
expression. The actual "dummy" expression is not relevant, and
|
|
|
|
no procedure should rely on the kind of expression used.
|
|
|
|
|
|
|
|
We have a default constructor because some collections only work
|
|
|
|
with types that have a default constructor.
|
|
|
|
*/
|
2013-12-08 07:21:07 +00:00
|
|
|
expr();
|
2013-07-22 11:03:46 +00:00
|
|
|
expr(expr const & s):m_ptr(s.m_ptr) { if (m_ptr) m_ptr->inc_ref(); }
|
2013-08-01 20:57:43 +00:00
|
|
|
expr(expr && s):m_ptr(s.m_ptr) { s.m_ptr = nullptr; }
|
2013-07-22 11:03:46 +00:00
|
|
|
~expr() { if (m_ptr) m_ptr->dec_ref(); }
|
|
|
|
|
|
|
|
friend void swap(expr & a, expr & b) { std::swap(a.m_ptr, b.m_ptr); }
|
|
|
|
|
2013-12-13 23:00:30 +00:00
|
|
|
expr & operator=(expr const & s) { LEAN_COPY_REF(s); }
|
|
|
|
expr & operator=(expr && s) { LEAN_MOVE_REF(s); }
|
2013-07-22 11:03:46 +00:00
|
|
|
|
|
|
|
expr_kind kind() const { return m_ptr->kind(); }
|
2013-07-30 08:39:29 +00:00
|
|
|
unsigned hash() const { return m_ptr ? m_ptr->hash() : 23; }
|
2013-11-14 10:31:54 +00:00
|
|
|
unsigned hash_alloc() const { return m_ptr ? m_ptr->hash_alloc() : 23; }
|
2014-07-05 20:11:30 +00:00
|
|
|
bool has_expr_metavar() const { return m_ptr->has_expr_metavar(); }
|
|
|
|
bool has_univ_metavar() const { return m_ptr->has_univ_metavar(); }
|
|
|
|
bool has_metavar() const { return has_expr_metavar() || has_univ_metavar(); }
|
2014-02-20 18:07:38 +00:00
|
|
|
bool has_local() const { return m_ptr->has_local(); }
|
2014-03-01 01:16:34 +00:00
|
|
|
bool has_param_univ() const { return m_ptr->has_param_univ(); }
|
2013-07-22 11:03:46 +00:00
|
|
|
|
2014-09-10 17:24:35 +00:00
|
|
|
expr set_tag(tag t) { m_ptr->set_tag(t); return *this; }
|
2014-05-12 23:33:22 +00:00
|
|
|
tag get_tag() const { return m_ptr->get_tag(); }
|
|
|
|
|
2014-10-16 19:48:49 +00:00
|
|
|
operator expr_ptr() const { return m_ptr; }
|
2013-07-22 11:03:46 +00:00
|
|
|
expr_cell * raw() const { return m_ptr; }
|
|
|
|
|
2014-10-15 20:12:09 +00:00
|
|
|
friend expr mk_var(unsigned idx, tag g);
|
|
|
|
friend expr mk_sort(level const & l, tag g);
|
|
|
|
friend expr mk_constant(name const & n, levels const & ls, tag g);
|
|
|
|
friend expr mk_metavar(name const & n, expr const & t, tag g);
|
|
|
|
friend expr mk_local(name const & n, name const & pp_n, expr const & t, binder_info const & bi,
|
|
|
|
tag g);
|
|
|
|
friend expr mk_app(expr const & f, expr const & a, tag g);
|
|
|
|
friend expr mk_binding(expr_kind k, name const & n, expr const & t, expr const & e, binder_info const & i,
|
|
|
|
tag g);
|
|
|
|
friend expr mk_macro(macro_definition const & m, unsigned num, expr const * args, tag g);
|
2013-07-22 11:03:46 +00:00
|
|
|
|
2013-08-04 16:41:49 +00:00
|
|
|
friend bool is_eqp(expr const & a, expr const & b) { return a.m_ptr == b.m_ptr; }
|
2013-07-22 11:03:46 +00:00
|
|
|
};
|
|
|
|
|
2014-06-16 20:00:13 +00:00
|
|
|
expr copy_tag(expr const & e, expr && new_e);
|
|
|
|
|
2013-12-08 18:17:29 +00:00
|
|
|
SPECIALIZE_OPTIONAL_FOR_SMART_PTR(expr)
|
|
|
|
|
2013-12-08 18:34:38 +00:00
|
|
|
inline optional<expr> none_expr() { return optional<expr>(); }
|
|
|
|
inline optional<expr> some_expr(expr const & e) { return optional<expr>(e); }
|
|
|
|
inline optional<expr> some_expr(expr && e) { return optional<expr>(std::forward<expr>(e)); }
|
|
|
|
|
2013-12-08 18:17:29 +00:00
|
|
|
inline bool is_eqp(optional<expr> const & a, optional<expr> const & b) {
|
|
|
|
return static_cast<bool>(a) == static_cast<bool>(b) && (!a || is_eqp(*a, *b));
|
|
|
|
}
|
|
|
|
|
2014-02-15 03:55:02 +00:00
|
|
|
/** \brief Bounded variables. They are encoded using de Bruijn's indices. */
|
2013-07-22 11:03:46 +00:00
|
|
|
class expr_var : public expr_cell {
|
|
|
|
unsigned m_vidx; // de Bruijn index
|
2014-07-22 01:46:47 +00:00
|
|
|
friend expr_cell;
|
|
|
|
void dealloc();
|
2013-07-22 11:03:46 +00:00
|
|
|
public:
|
2014-10-15 20:12:09 +00:00
|
|
|
expr_var(unsigned idx, tag g);
|
2013-07-22 11:03:46 +00:00
|
|
|
unsigned get_vidx() const { return m_vidx; }
|
|
|
|
};
|
2014-02-15 03:55:02 +00:00
|
|
|
|
|
|
|
/** \brief (parametric) Constants. */
|
2013-07-22 11:03:46 +00:00
|
|
|
class expr_const : public expr_cell {
|
2014-02-16 17:53:55 +00:00
|
|
|
name m_name;
|
|
|
|
levels m_levels;
|
2014-07-22 01:46:47 +00:00
|
|
|
friend expr_cell;
|
|
|
|
void dealloc();
|
2013-07-22 11:03:46 +00:00
|
|
|
public:
|
2014-10-15 20:12:09 +00:00
|
|
|
expr_const(name const & n, levels const & ls, tag g);
|
2013-07-22 11:03:46 +00:00
|
|
|
name const & get_name() const { return m_name; }
|
2014-05-16 18:13:50 +00:00
|
|
|
levels const & get_levels() const { return m_levels; }
|
2014-02-15 03:55:02 +00:00
|
|
|
};
|
|
|
|
|
2014-07-17 07:51:46 +00:00
|
|
|
/** \brief Composite expressions */
|
|
|
|
class expr_composite : public expr_cell {
|
|
|
|
protected:
|
2014-08-04 22:48:24 +00:00
|
|
|
unsigned m_weight;
|
2014-07-17 07:51:46 +00:00
|
|
|
unsigned m_free_var_range;
|
2014-08-04 22:48:24 +00:00
|
|
|
friend unsigned get_weight(expr const & e);
|
2014-07-17 07:51:46 +00:00
|
|
|
friend unsigned get_free_var_range(expr const & e);
|
|
|
|
public:
|
|
|
|
expr_composite(expr_kind k, unsigned h, bool has_expr_mv, bool has_univ_mv, bool has_local,
|
2014-10-15 20:12:09 +00:00
|
|
|
bool has_param_univ, unsigned w, unsigned fv_range, tag g);
|
2014-07-17 07:51:46 +00:00
|
|
|
};
|
|
|
|
|
2014-02-15 03:55:02 +00:00
|
|
|
/** \brief Metavariables and local constants */
|
2014-07-17 07:51:46 +00:00
|
|
|
class expr_mlocal : public expr_composite {
|
2014-05-16 22:47:30 +00:00
|
|
|
protected:
|
2014-02-15 03:55:02 +00:00
|
|
|
name m_name;
|
|
|
|
expr m_type;
|
2014-02-16 17:53:55 +00:00
|
|
|
friend expr_cell;
|
|
|
|
void dealloc(buffer<expr_cell*> & todelete);
|
2014-02-15 03:55:02 +00:00
|
|
|
public:
|
2014-10-15 20:12:09 +00:00
|
|
|
expr_mlocal(bool is_meta, name const & n, expr const & t, tag g);
|
2014-02-15 03:55:02 +00:00
|
|
|
name const & get_name() const { return m_name; }
|
|
|
|
expr const & get_type() const { return m_type; }
|
2013-07-22 11:03:46 +00:00
|
|
|
};
|
2014-02-15 03:55:02 +00:00
|
|
|
|
2014-06-30 16:14:55 +00:00
|
|
|
/**
|
|
|
|
\brief Auxiliary annotation for binders (Lambda and Pi). This information
|
|
|
|
is only used for elaboration.
|
|
|
|
*/
|
|
|
|
class binder_info {
|
|
|
|
unsigned m_implicit:1; //! if true, binder argument is an implicit argument
|
2014-10-12 20:06:00 +00:00
|
|
|
/** if m_contextual is true, binder argument is assumed to be part of the context,
|
|
|
|
and may be argument for metavariables. */
|
|
|
|
unsigned m_contextual:1;
|
2014-06-30 16:14:55 +00:00
|
|
|
unsigned m_strict_implicit:1; //! if true, binder argument is assumed to be a strict implicit argument
|
2014-10-12 20:06:00 +00:00
|
|
|
/** \brief if m_inst_implicit is true, binder argument is an implicit argument, and should be
|
|
|
|
inferred by class-instance resolution. */
|
|
|
|
unsigned m_inst_implicit:1;
|
2014-06-30 16:14:55 +00:00
|
|
|
public:
|
2015-06-12 01:11:22 +00:00
|
|
|
binder_info(bool implicit = false, bool contextual = true, bool strict_implicit = false, bool inst_implicit = false):
|
|
|
|
m_implicit(implicit), m_contextual(contextual), m_strict_implicit(strict_implicit),
|
2014-10-12 20:06:00 +00:00
|
|
|
m_inst_implicit(inst_implicit) {}
|
2014-06-30 16:14:55 +00:00
|
|
|
bool is_implicit() const { return m_implicit; }
|
|
|
|
bool is_contextual() const { return m_contextual; }
|
|
|
|
bool is_strict_implicit() const { return m_strict_implicit; }
|
2014-10-12 20:06:00 +00:00
|
|
|
bool is_inst_implicit() const { return m_inst_implicit; }
|
2014-09-29 23:19:01 +00:00
|
|
|
unsigned hash() const;
|
2015-06-12 01:11:22 +00:00
|
|
|
binder_info update_contextual(bool f) const { return binder_info(m_implicit, f, m_strict_implicit, m_inst_implicit); }
|
2014-06-30 16:14:55 +00:00
|
|
|
};
|
|
|
|
|
2015-06-12 01:11:22 +00:00
|
|
|
inline binder_info mk_implicit_binder_info() { return binder_info(true); }
|
|
|
|
inline binder_info mk_strict_implicit_binder_info() { return binder_info(false, true, true); }
|
|
|
|
inline binder_info mk_inst_implicit_binder_info() { return binder_info(false, true, false, true); }
|
|
|
|
inline binder_info mk_contextual_info(bool f) { return binder_info(false, f); }
|
2014-06-30 16:14:55 +00:00
|
|
|
|
2014-10-18 22:11:26 +00:00
|
|
|
inline bool is_explicit(binder_info const & bi) {
|
|
|
|
return !bi.is_implicit() && !bi.is_strict_implicit() && !bi.is_inst_implicit();
|
|
|
|
}
|
|
|
|
|
2014-06-30 16:14:55 +00:00
|
|
|
bool operator==(binder_info const & i1, binder_info const & i2);
|
|
|
|
inline bool operator!=(binder_info const & i1, binder_info const & i2) { return !(i1 == i2); }
|
|
|
|
|
2014-09-29 23:19:01 +00:00
|
|
|
/** \brief Compute a hash code that takes binder_info into account.
|
|
|
|
\remark This information is not cached like hash(). */
|
|
|
|
unsigned hash_bi(expr const & e);
|
|
|
|
|
2014-05-16 20:08:09 +00:00
|
|
|
/** \brief expr_mlocal subclass for local constants. */
|
|
|
|
class expr_local : public expr_mlocal {
|
|
|
|
// The name used in the binder that generate this local,
|
|
|
|
// it is only used for pretty printing. This field is ignored
|
|
|
|
// when comparing expressions.
|
2014-06-30 16:14:55 +00:00
|
|
|
name m_pp_name;
|
|
|
|
binder_info m_bi;
|
2014-05-16 22:47:30 +00:00
|
|
|
friend expr_cell;
|
|
|
|
void dealloc(buffer<expr_cell*> & todelete);
|
2014-05-16 20:08:09 +00:00
|
|
|
public:
|
2014-10-15 20:12:09 +00:00
|
|
|
expr_local(name const & n, name const & pp_name, expr const & t, binder_info const & bi, tag g);
|
2014-05-16 20:08:09 +00:00
|
|
|
name const & get_pp_name() const { return m_pp_name; }
|
2014-06-30 16:14:55 +00:00
|
|
|
binder_info const & get_info() const { return m_bi; }
|
2014-05-16 20:08:09 +00:00
|
|
|
};
|
|
|
|
|
2014-02-15 03:55:02 +00:00
|
|
|
/** \brief Applications */
|
|
|
|
class expr_app : public expr_composite {
|
|
|
|
expr m_fn;
|
|
|
|
expr m_arg;
|
2013-11-19 02:41:08 +00:00
|
|
|
friend expr_cell;
|
|
|
|
void dealloc(buffer<expr_cell*> & todelete);
|
2013-07-22 11:03:46 +00:00
|
|
|
public:
|
2014-10-15 20:12:09 +00:00
|
|
|
expr_app(expr const & fn, expr const & arg, tag g);
|
2014-02-15 03:55:02 +00:00
|
|
|
expr const & get_fn() const { return m_fn; }
|
|
|
|
expr const & get_arg() const { return m_arg; }
|
2013-07-22 11:03:46 +00:00
|
|
|
};
|
2014-02-15 03:55:02 +00:00
|
|
|
|
2014-05-16 17:40:53 +00:00
|
|
|
class binder {
|
2014-05-16 18:13:50 +00:00
|
|
|
friend class expr_binding;
|
2014-05-16 17:40:53 +00:00
|
|
|
name m_name;
|
|
|
|
expr m_type;
|
2014-05-16 18:52:43 +00:00
|
|
|
binder_info m_info;
|
2014-05-16 17:40:53 +00:00
|
|
|
public:
|
2014-10-15 20:12:09 +00:00
|
|
|
binder(name const & n, expr const & t, binder_info const & bi):
|
2014-05-16 17:40:53 +00:00
|
|
|
m_name(n), m_type(t), m_info(bi) {}
|
|
|
|
name const & get_name() const { return m_name; }
|
|
|
|
expr const & get_type() const { return m_type; }
|
2014-05-16 18:13:50 +00:00
|
|
|
binder_info const & get_info() const { return m_info; }
|
2014-05-16 18:52:43 +00:00
|
|
|
binder update_type(expr const & t) const { return binder(m_name, t, m_info); }
|
2014-05-16 17:40:53 +00:00
|
|
|
};
|
|
|
|
|
2014-05-16 18:13:50 +00:00
|
|
|
/** \brief Lambda and Pi expressions */
|
|
|
|
class expr_binding : public expr_composite {
|
2014-05-16 17:40:53 +00:00
|
|
|
binder m_binder;
|
2014-05-01 17:35:02 +00:00
|
|
|
expr m_body;
|
2013-11-19 02:41:08 +00:00
|
|
|
friend class expr_cell;
|
|
|
|
void dealloc(buffer<expr_cell*> & todelete);
|
2013-07-22 11:03:46 +00:00
|
|
|
public:
|
2014-10-15 20:12:09 +00:00
|
|
|
expr_binding(expr_kind k, name const & n, expr const & t, expr const & e,
|
|
|
|
binder_info const & i, tag g);
|
2014-05-16 17:40:53 +00:00
|
|
|
name const & get_name() const { return m_binder.get_name(); }
|
|
|
|
expr const & get_domain() const { return m_binder.get_type(); }
|
2013-08-03 18:31:42 +00:00
|
|
|
expr const & get_body() const { return m_body; }
|
2014-05-16 18:13:50 +00:00
|
|
|
binder_info const & get_info() const { return m_binder.get_info(); }
|
|
|
|
binder const & get_binder() const { return m_binder; }
|
2013-07-22 11:03:46 +00:00
|
|
|
};
|
2014-02-15 03:55:02 +00:00
|
|
|
|
|
|
|
/** \brief Sort */
|
|
|
|
class expr_sort : public expr_cell {
|
2013-07-30 02:44:26 +00:00
|
|
|
level m_level;
|
2014-07-22 01:46:47 +00:00
|
|
|
friend expr_cell;
|
|
|
|
void dealloc();
|
2013-07-22 11:03:46 +00:00
|
|
|
public:
|
2014-10-15 20:12:09 +00:00
|
|
|
expr_sort(level const & l, tag g);
|
2014-02-15 03:55:02 +00:00
|
|
|
~expr_sort();
|
2013-07-30 02:44:26 +00:00
|
|
|
level const & get_level() const { return m_level; }
|
2013-07-22 11:03:46 +00:00
|
|
|
};
|
2014-02-15 03:55:02 +00:00
|
|
|
|
2014-05-02 01:05:46 +00:00
|
|
|
/** \brief Abstract class for macro_definitions */
|
|
|
|
class macro_definition_cell {
|
|
|
|
protected:
|
2013-08-03 23:09:21 +00:00
|
|
|
void dealloc() { delete this; }
|
|
|
|
MK_LEAN_RC();
|
2013-09-24 19:16:32 +00:00
|
|
|
/**
|
2014-05-02 01:05:46 +00:00
|
|
|
\brief Auxiliary method used for implementing a total order on macro
|
|
|
|
attachments. It is invoked by operator<, and it is only invoked when
|
|
|
|
<tt>get_name() == other.get_name()</tt>
|
2013-09-24 19:16:32 +00:00
|
|
|
*/
|
2014-05-02 01:05:46 +00:00
|
|
|
virtual bool lt(macro_definition_cell const &) const;
|
2013-07-22 11:03:46 +00:00
|
|
|
public:
|
2014-05-02 01:05:46 +00:00
|
|
|
macro_definition_cell():m_rc(0) {}
|
|
|
|
virtual ~macro_definition_cell() {}
|
2013-09-04 06:15:37 +00:00
|
|
|
virtual name get_name() const = 0;
|
2015-05-08 19:31:34 +00:00
|
|
|
virtual pair<expr, constraint_seq> check_type(expr const & m, extension_context & ctx, bool infer_only) const = 0;
|
2014-05-15 17:29:35 +00:00
|
|
|
virtual optional<expr> expand(expr const & m, extension_context & ctx) const = 0;
|
|
|
|
virtual optional<expr> expand1(expr const & m, extension_context & ctx) const { return expand(m, ctx); }
|
2014-05-02 01:05:46 +00:00
|
|
|
virtual unsigned trust_level() const;
|
|
|
|
virtual bool operator==(macro_definition_cell const & other) const;
|
2014-02-15 03:55:02 +00:00
|
|
|
virtual void display(std::ostream & out) const;
|
2014-07-10 17:32:00 +00:00
|
|
|
virtual format pp(formatter const & fmt) const;
|
2014-02-15 03:55:02 +00:00
|
|
|
virtual bool is_atomic_pp(bool unicode, bool coercion) const;
|
2013-09-04 06:15:37 +00:00
|
|
|
virtual unsigned hash() const;
|
2013-12-28 09:16:50 +00:00
|
|
|
virtual void write(serializer & s) const = 0;
|
2014-04-25 22:02:52 +00:00
|
|
|
typedef std::function<expr(deserializer&, unsigned, expr const *)> reader;
|
2013-12-28 09:16:50 +00:00
|
|
|
};
|
|
|
|
|
2014-05-02 01:05:46 +00:00
|
|
|
/** \brief Smart pointer for macro definitions */
|
|
|
|
class macro_definition {
|
|
|
|
public:
|
|
|
|
macro_definition_cell * m_ptr;
|
|
|
|
public:
|
|
|
|
explicit macro_definition(macro_definition_cell * ptr);
|
|
|
|
macro_definition(macro_definition const & s);
|
|
|
|
macro_definition(macro_definition && s);
|
|
|
|
~macro_definition();
|
|
|
|
|
|
|
|
macro_definition & operator=(macro_definition const & s);
|
|
|
|
macro_definition & operator=(macro_definition && s);
|
|
|
|
|
|
|
|
name get_name() const { return m_ptr->get_name(); }
|
2015-05-08 19:31:34 +00:00
|
|
|
pair<expr, constraint_seq> check_type(expr const & m, extension_context & ctx, bool infer_only) const {
|
|
|
|
return m_ptr->check_type(m, ctx, infer_only);
|
2014-08-20 05:31:26 +00:00
|
|
|
}
|
2014-05-15 17:29:35 +00:00
|
|
|
optional<expr> expand(expr const & m, extension_context & ctx) const { return m_ptr->expand(m, ctx); }
|
|
|
|
optional<expr> expand1(expr const & m, extension_context & ctx) const { return m_ptr->expand1(m, ctx); }
|
2014-05-02 01:05:46 +00:00
|
|
|
unsigned trust_level() const { return m_ptr->trust_level(); }
|
|
|
|
bool operator==(macro_definition const & other) const { return m_ptr->operator==(*other.m_ptr); }
|
|
|
|
bool operator!=(macro_definition const & other) const { return !operator==(other); }
|
|
|
|
bool operator<(macro_definition const & other) const;
|
|
|
|
void display(std::ostream & out) const { return m_ptr->display(out); }
|
2014-07-10 17:32:00 +00:00
|
|
|
format pp(formatter const & fmt) const { return m_ptr->pp(fmt); }
|
2014-05-02 01:05:46 +00:00
|
|
|
bool is_atomic_pp(bool unicode, bool coercion) const { return m_ptr->is_atomic_pp(unicode, coercion); }
|
|
|
|
unsigned hash() const { return m_ptr->hash(); }
|
|
|
|
void write(serializer & s) const { return m_ptr->write(s); }
|
2014-07-02 17:00:55 +00:00
|
|
|
macro_definition_cell const * raw() const { return m_ptr; }
|
2014-05-02 01:05:46 +00:00
|
|
|
};
|
|
|
|
|
2014-02-15 03:55:02 +00:00
|
|
|
/** \brief Macro attachments */
|
2014-04-25 22:02:52 +00:00
|
|
|
class expr_macro : public expr_composite {
|
2014-05-02 01:05:46 +00:00
|
|
|
macro_definition m_definition;
|
|
|
|
unsigned m_num_args;
|
|
|
|
expr * m_args;
|
2014-04-25 22:02:52 +00:00
|
|
|
friend class expr_cell;
|
2013-08-03 23:09:21 +00:00
|
|
|
friend expr copy(expr const & a);
|
2014-04-25 22:02:52 +00:00
|
|
|
friend expr update_macro(expr const & e, unsigned num, expr const * args);
|
|
|
|
void dealloc(buffer<expr_cell*> & todelete);
|
2013-08-03 23:09:21 +00:00
|
|
|
public:
|
2014-10-15 20:12:09 +00:00
|
|
|
expr_macro(macro_definition const & v, unsigned num, expr const * args, tag g);
|
2014-02-15 03:55:02 +00:00
|
|
|
~expr_macro();
|
2013-09-13 01:25:38 +00:00
|
|
|
|
2014-05-02 01:05:46 +00:00
|
|
|
macro_definition const & get_def() const { return m_definition; }
|
2014-04-25 22:02:52 +00:00
|
|
|
expr const * get_args() const { return m_args; }
|
|
|
|
expr const & get_arg(unsigned idx) const { lean_assert(idx < m_num_args); return m_args[idx]; }
|
|
|
|
unsigned get_num_args() const { return m_num_args; }
|
2013-09-13 01:25:38 +00:00
|
|
|
};
|
2013-07-22 11:03:46 +00:00
|
|
|
|
|
|
|
// =======================================
|
|
|
|
// Testers
|
2014-10-16 19:48:49 +00:00
|
|
|
inline bool is_var(expr_ptr e) { return e->kind() == expr_kind::Var; }
|
|
|
|
inline bool is_constant(expr_ptr e) { return e->kind() == expr_kind::Constant; }
|
|
|
|
inline bool is_local(expr_ptr e) { return e->kind() == expr_kind::Local; }
|
|
|
|
inline bool is_metavar(expr_ptr e) { return e->kind() == expr_kind::Meta; }
|
|
|
|
inline bool is_macro(expr_ptr e) { return e->kind() == expr_kind::Macro; }
|
|
|
|
inline bool is_app(expr_ptr e) { return e->kind() == expr_kind::App; }
|
|
|
|
inline bool is_lambda(expr_ptr e) { return e->kind() == expr_kind::Lambda; }
|
|
|
|
inline bool is_pi(expr_ptr e) { return e->kind() == expr_kind::Pi; }
|
|
|
|
inline bool is_sort(expr_ptr e) { return e->kind() == expr_kind::Sort; }
|
|
|
|
inline bool is_binding(expr_ptr e) { return is_lambda(e) || is_pi(e); }
|
|
|
|
inline bool is_mlocal(expr_ptr e) { return is_metavar(e) || is_local(e); }
|
2014-02-18 00:10:11 +00:00
|
|
|
|
|
|
|
bool is_atomic(expr const & e);
|
|
|
|
bool is_arrow(expr const & t);
|
2014-04-24 03:16:14 +00:00
|
|
|
/** \brief Return true iff \c e is a metavariable or an application of a metavariable */
|
|
|
|
bool is_meta(expr const & e);
|
2013-07-22 11:03:46 +00:00
|
|
|
// =======================================
|
|
|
|
|
|
|
|
// =======================================
|
|
|
|
// Constructors
|
2014-10-15 20:12:09 +00:00
|
|
|
expr mk_var(unsigned idx, tag g = nulltag);
|
2013-08-06 18:27:14 +00:00
|
|
|
inline expr Var(unsigned idx) { return mk_var(idx); }
|
2014-10-15 20:12:09 +00:00
|
|
|
expr mk_constant(name const & n, levels const & ls, tag g = nulltag);
|
2014-02-16 17:53:55 +00:00
|
|
|
inline expr mk_constant(name const & n) { return mk_constant(n, levels()); }
|
2013-08-06 18:27:14 +00:00
|
|
|
inline expr Const(name const & n) { return mk_constant(n); }
|
2014-10-15 20:12:09 +00:00
|
|
|
expr mk_macro(macro_definition const & m, unsigned num = 0, expr const * args = nullptr, tag g = nulltag);
|
|
|
|
expr mk_metavar(name const & n, expr const & t, tag g = nulltag);
|
|
|
|
expr mk_local(name const & n, name const & pp_n, expr const & t, binder_info const & bi, tag g = nulltag);
|
|
|
|
inline expr mk_local(name const & n, expr const & t, tag g = nulltag) { return mk_local(n, n, t, binder_info(), g); }
|
|
|
|
inline expr mk_local(name const & n, expr const & t, binder_info const & bi, tag g = nulltag) {
|
|
|
|
return mk_local(n, n, t, bi, g);
|
2014-05-01 17:35:02 +00:00
|
|
|
}
|
2014-10-15 20:12:09 +00:00
|
|
|
inline expr Local(name const & n, expr const & t, binder_info const & bi = binder_info(), tag g = nulltag) {
|
|
|
|
return mk_local(n, t, bi, g);
|
2014-05-01 17:35:02 +00:00
|
|
|
}
|
2014-10-15 20:12:09 +00:00
|
|
|
expr mk_app(expr const & f, expr const & a, tag g = nulltag);
|
|
|
|
expr mk_app(expr const & f, unsigned num_args, expr const * args, tag g = nulltag);
|
|
|
|
expr mk_app(unsigned num_args, expr const * args, tag g = nulltag);
|
|
|
|
inline expr mk_app(std::initializer_list<expr> const & l, tag g = nulltag) {
|
|
|
|
return mk_app(l.size(), l.begin(), g);
|
|
|
|
}
|
2014-10-16 20:37:55 +00:00
|
|
|
inline expr mk_app(buffer<expr> const & args, tag g = nulltag) { return mk_app(args.size(), args.data(), g); }
|
|
|
|
inline expr mk_app(expr const & f, buffer<expr> const & args, tag g = nulltag) {
|
2014-10-15 20:12:09 +00:00
|
|
|
return mk_app(f, args.size(), args.data(), g);
|
|
|
|
}
|
|
|
|
expr mk_rev_app(expr const & f, unsigned num_args, expr const * args, tag g = nulltag);
|
|
|
|
expr mk_rev_app(unsigned num_args, expr const * args, tag g = nulltag);
|
2014-10-16 20:37:55 +00:00
|
|
|
inline expr mk_rev_app(buffer<expr> const & args, tag g = nulltag) { return mk_rev_app(args.size(), args.data(), g); }
|
|
|
|
inline expr mk_rev_app(expr const & f, buffer<expr> const & args, tag g = nulltag) {
|
2014-10-15 20:12:09 +00:00
|
|
|
return mk_rev_app(f, args.size(), args.data(), g);
|
|
|
|
}
|
|
|
|
expr mk_binding(expr_kind k, name const & n, expr const & t, expr const & e,
|
|
|
|
binder_info const & i = binder_info(), tag g = nulltag);
|
|
|
|
inline expr mk_lambda(name const & n, expr const & t, expr const & e,
|
|
|
|
binder_info const & i = binder_info(), tag g = nulltag) {
|
|
|
|
return mk_binding(expr_kind::Lambda, n, t, e, i, g);
|
|
|
|
}
|
|
|
|
inline expr mk_pi(name const & n, expr const & t, expr const & e, binder_info const & i = binder_info(), tag g = nulltag) {
|
|
|
|
return mk_binding(expr_kind::Pi, n, t, e, i, g);
|
|
|
|
}
|
|
|
|
expr mk_sort(level const & l, tag g = nulltag);
|
2014-02-16 17:53:55 +00:00
|
|
|
|
2014-07-22 16:43:18 +00:00
|
|
|
expr mk_Prop();
|
2014-02-15 03:55:02 +00:00
|
|
|
expr mk_Type();
|
2013-08-06 18:27:14 +00:00
|
|
|
|
2014-02-16 17:53:55 +00:00
|
|
|
bool is_default_var_name(name const & n);
|
2014-10-15 20:12:09 +00:00
|
|
|
expr mk_arrow(expr const & t, expr const & e, tag g = nulltag);
|
2014-02-16 17:53:55 +00:00
|
|
|
inline expr operator>>(expr const & t, expr const & e) { return mk_arrow(t, e); }
|
|
|
|
|
2014-02-15 03:55:02 +00:00
|
|
|
// Auxiliary
|
2014-10-15 20:12:09 +00:00
|
|
|
inline expr mk_app(expr const & e1, expr const & e2, expr const & e3, tag g = nulltag) {
|
|
|
|
return mk_app({e1, e2, e3}, g);
|
|
|
|
}
|
|
|
|
inline expr mk_app(expr const & e1, expr const & e2, expr const & e3, expr const & e4, tag g = nulltag) {
|
|
|
|
return mk_app({e1, e2, e3, e4}, g);
|
|
|
|
}
|
|
|
|
inline expr mk_app(expr const & e1, expr const & e2, expr const & e3, expr const & e4, expr const & e5,
|
|
|
|
tag g = nulltag) {
|
|
|
|
return mk_app({e1, e2, e3, e4, e5}, g);
|
|
|
|
}
|
2014-04-26 01:17:32 +00:00
|
|
|
|
|
|
|
/** \brief Return application (...((f x_{n-1}) x_{n-2}) ... x_0) */
|
2014-10-15 20:12:09 +00:00
|
|
|
expr mk_app_vars(expr const & f, unsigned n, tag g = nulltag);
|
2014-06-03 20:32:02 +00:00
|
|
|
|
2015-09-30 20:31:42 +00:00
|
|
|
/** \brief Enable hash-consing (caching) for expressions (and universe levels) */
|
2014-06-03 20:32:02 +00:00
|
|
|
bool enable_expr_caching(bool f);
|
|
|
|
/** \brief Helper class for temporarily enabling/disabling expression caching */
|
|
|
|
struct scoped_expr_caching {
|
|
|
|
bool m_old;
|
|
|
|
scoped_expr_caching(bool f) { m_old = enable_expr_caching(f); }
|
|
|
|
~scoped_expr_caching() { enable_expr_caching(m_old); }
|
|
|
|
};
|
2015-09-30 19:28:42 +00:00
|
|
|
/** \brief Return true iff \c e is in the cache */
|
|
|
|
bool is_cached(expr const & e);
|
2013-07-22 11:03:46 +00:00
|
|
|
// =======================================
|
|
|
|
|
|
|
|
// =======================================
|
2013-07-23 15:59:39 +00:00
|
|
|
// Casting (these functions are only needed for low-level code)
|
2014-10-16 19:48:49 +00:00
|
|
|
inline expr_var * to_var(expr_ptr e) { lean_assert(is_var(e)); return static_cast<expr_var*>(e); }
|
|
|
|
inline expr_const * to_constant(expr_ptr e) { lean_assert(is_constant(e)); return static_cast<expr_const*>(e); }
|
|
|
|
inline expr_app * to_app(expr_ptr e) { lean_assert(is_app(e)); return static_cast<expr_app*>(e); }
|
|
|
|
inline expr_binding * to_binding(expr_ptr e) { lean_assert(is_binding(e)); return static_cast<expr_binding*>(e); }
|
|
|
|
inline expr_sort * to_sort(expr_ptr e) { lean_assert(is_sort(e)); return static_cast<expr_sort*>(e); }
|
|
|
|
inline expr_mlocal * to_mlocal(expr_ptr e) { lean_assert(is_mlocal(e)); return static_cast<expr_mlocal*>(e); }
|
|
|
|
inline expr_local * to_local(expr_ptr e) { lean_assert(is_local(e)); return static_cast<expr_local*>(e); }
|
|
|
|
inline expr_mlocal * to_metavar(expr_ptr e) { lean_assert(is_metavar(e)); return static_cast<expr_mlocal*>(e); }
|
|
|
|
inline expr_macro * to_macro(expr_ptr e) { lean_assert(is_macro(e)); return static_cast<expr_macro*>(e); }
|
2013-07-22 11:03:46 +00:00
|
|
|
// =======================================
|
|
|
|
|
2014-02-15 03:55:02 +00:00
|
|
|
|
2013-07-22 11:03:46 +00:00
|
|
|
// =======================================
|
|
|
|
// Accessors
|
2014-10-16 19:48:49 +00:00
|
|
|
inline unsigned get_rc(expr_ptr e) { return e->get_rc(); }
|
|
|
|
inline bool is_shared(expr_ptr e) { return get_rc(e) > 1; }
|
|
|
|
inline unsigned var_idx(expr_ptr e) { return to_var(e)->get_vidx(); }
|
|
|
|
inline bool is_var(expr_ptr e, unsigned i) { return is_var(e) && var_idx(e) == i; }
|
|
|
|
inline name const & const_name(expr_ptr e) { return to_constant(e)->get_name(); }
|
|
|
|
inline levels const & const_levels(expr_ptr e) { return to_constant(e)->get_levels(); }
|
|
|
|
inline macro_definition const & macro_def(expr_ptr e) { return to_macro(e)->get_def(); }
|
|
|
|
inline expr const * macro_args(expr_ptr e) { return to_macro(e)->get_args(); }
|
|
|
|
inline expr const & macro_arg(expr_ptr e, unsigned i) { return to_macro(e)->get_arg(i); }
|
|
|
|
inline unsigned macro_num_args(expr_ptr e) { return to_macro(e)->get_num_args(); }
|
|
|
|
inline expr const & app_fn(expr_ptr e) { return to_app(e)->get_fn(); }
|
|
|
|
inline expr const & app_arg(expr_ptr e) { return to_app(e)->get_arg(); }
|
|
|
|
inline name const & binding_name(expr_ptr e) { return to_binding(e)->get_name(); }
|
|
|
|
inline expr const & binding_domain(expr_ptr e) { return to_binding(e)->get_domain(); }
|
|
|
|
inline expr const & binding_body(expr_ptr e) { return to_binding(e)->get_body(); }
|
|
|
|
inline binder_info const & binding_info(expr_ptr e) { return to_binding(e)->get_info(); }
|
|
|
|
inline binder const & binding_binder(expr_ptr e) { return to_binding(e)->get_binder(); }
|
|
|
|
inline level const & sort_level(expr_ptr e) { return to_sort(e)->get_level(); }
|
|
|
|
inline name const & mlocal_name(expr_ptr e) { return to_mlocal(e)->get_name(); }
|
|
|
|
inline expr const & mlocal_type(expr_ptr e) { return to_mlocal(e)->get_type(); }
|
|
|
|
inline name const & local_pp_name(expr_ptr e) { return to_local(e)->get_pp_name(); }
|
|
|
|
inline binder_info const & local_info(expr_ptr e) { return to_local(e)->get_info(); }
|
2014-02-16 17:53:55 +00:00
|
|
|
|
|
|
|
inline bool is_constant(expr const & e, name const & n) { return is_constant(e) && const_name(e) == n; }
|
2013-09-13 01:25:38 +00:00
|
|
|
inline bool has_metavar(expr const & e) { return e.has_metavar(); }
|
2014-07-05 20:11:30 +00:00
|
|
|
inline bool has_expr_metavar(expr const & e) { return e.has_expr_metavar(); }
|
|
|
|
inline bool has_univ_metavar(expr const & e) { return e.has_univ_metavar(); }
|
2014-10-27 20:16:50 +00:00
|
|
|
/** \brief Similar to \c has_expr_metavar, but ignores metavariables occurring in local constant types.
|
|
|
|
It also returns the meta-variable application found in \c e.
|
|
|
|
*/
|
|
|
|
optional<expr> has_expr_metavar_strict(expr const & e);
|
2014-02-20 18:07:38 +00:00
|
|
|
inline bool has_local(expr const & e) { return e.has_local(); }
|
2014-03-01 01:16:34 +00:00
|
|
|
inline bool has_param_univ(expr const & e) { return e.has_param_univ(); }
|
2014-08-04 22:48:24 +00:00
|
|
|
unsigned get_weight(expr const & e);
|
2014-04-17 18:31:48 +00:00
|
|
|
/**
|
|
|
|
\brief Return \c R s.t. the de Bruijn index of all free variables
|
|
|
|
occurring in \c e is in the interval <tt>[0, R)</tt>.
|
|
|
|
*/
|
2014-07-17 07:51:46 +00:00
|
|
|
inline unsigned get_free_var_range(expr const & e) {
|
|
|
|
switch (e.kind()) {
|
|
|
|
case expr_kind::Var: return var_idx(e) + 1;
|
|
|
|
case expr_kind::Constant: case expr_kind::Sort: return 0;
|
|
|
|
default: return static_cast<expr_composite*>(e.raw())->m_free_var_range;
|
|
|
|
}
|
|
|
|
}
|
2014-04-17 18:31:48 +00:00
|
|
|
/** \brief Return true iff the given expression has free variables. */
|
|
|
|
inline bool has_free_vars(expr const & e) { return get_free_var_range(e) > 0; }
|
|
|
|
/** \brief Return true iff the given expression does not have free variables. */
|
|
|
|
inline bool closed(expr const & e) { return !has_free_vars(e); }
|
2014-04-26 01:17:32 +00:00
|
|
|
/**
|
|
|
|
\brief Given \c e of the form <tt>(...(f a1) ... an)</tt>, store a1 ... an in args.
|
|
|
|
If \c e is not an application, then nothing is stored in args.
|
2014-05-18 02:19:10 +00:00
|
|
|
|
|
|
|
It returns the f.
|
2014-04-26 01:17:32 +00:00
|
|
|
*/
|
2014-05-19 18:11:19 +00:00
|
|
|
expr const & get_app_args(expr const & e, buffer<expr> & args);
|
|
|
|
/**
|
|
|
|
\brief Similar to \c get_app_args, but arguments are stored in reverse order in \c args.
|
|
|
|
If e is of the form <tt>(...(f a1) ... an)</tt>, then the procedure stores [an, ..., a1] in \c args.
|
|
|
|
*/
|
|
|
|
expr const & get_app_rev_args(expr const & e, buffer<expr> & args);
|
2014-05-24 17:45:00 +00:00
|
|
|
/** \brief Given \c e of the form <tt>(...(f a_1) ... a_n)</tt>, return \c f. If \c e is not an application, then return \c e. */
|
2014-05-19 18:11:19 +00:00
|
|
|
expr const & get_app_fn(expr const & e);
|
2014-05-24 17:45:00 +00:00
|
|
|
/** \brief Given \c e of the form <tt>(...(f a_1) ... a_n)</tt>, return \c n. If \c e is not an application, then return 0. */
|
|
|
|
unsigned get_app_num_args(expr const & e);
|
2013-07-22 11:03:46 +00:00
|
|
|
// =======================================
|
|
|
|
|
2013-07-23 17:09:04 +00:00
|
|
|
// =======================================
|
2013-07-24 07:45:38 +00:00
|
|
|
// Auxiliary functionals
|
2013-07-26 18:43:53 +00:00
|
|
|
/** \brief Functional object for hashing kernel expressions. */
|
2013-07-23 17:09:04 +00:00
|
|
|
struct expr_hash { unsigned operator()(expr const & e) const { return e.hash(); } };
|
2013-11-14 10:31:54 +00:00
|
|
|
/**
|
|
|
|
\brief Functional object for hashing (based on allocation time) kernel expressions.
|
|
|
|
|
|
|
|
This hash is compatible with pointer equality.
|
|
|
|
\warning This hash is incompatible with structural equality (i.e., std::equal_to<expr>)
|
|
|
|
*/
|
|
|
|
struct expr_hash_alloc { unsigned operator()(expr const & e) const { return e.hash_alloc(); } };
|
2013-07-26 18:43:53 +00:00
|
|
|
/** \brief Functional object for testing pointer equality between kernel expressions. */
|
2013-08-04 16:41:49 +00:00
|
|
|
struct expr_eqp { bool operator()(expr const & e1, expr const & e2) const { return is_eqp(e1, e2); } };
|
2013-07-26 18:43:53 +00:00
|
|
|
/** \brief Functional object for hashing kernel expression cells. */
|
2014-10-16 19:48:49 +00:00
|
|
|
struct expr_cell_hash { unsigned operator()(expr_ptr e) const { return e->hash_alloc(); } };
|
2013-07-26 18:43:53 +00:00
|
|
|
/** \brief Functional object for testing pointer equality between kernel cell expressions. */
|
2014-10-16 19:48:49 +00:00
|
|
|
struct expr_cell_eqp { bool operator()(expr_ptr e1, expr_ptr e2) const { return e1 == e2; } };
|
2013-07-23 17:09:04 +00:00
|
|
|
// =======================================
|
2013-07-22 11:03:46 +00:00
|
|
|
|
2013-07-30 04:28:22 +00:00
|
|
|
// =======================================
|
|
|
|
// Update
|
2014-02-16 17:53:55 +00:00
|
|
|
expr update_app(expr const & e, expr const & new_fn, expr const & new_arg);
|
2014-05-17 18:37:27 +00:00
|
|
|
expr update_binding(expr const & e, expr const & new_domain, expr const & new_body);
|
2014-06-28 20:57:36 +00:00
|
|
|
expr update_binding(expr const & e, expr const & new_domain, expr const & new_body, binder_info const & bi);
|
2014-02-16 17:53:55 +00:00
|
|
|
expr update_mlocal(expr const & e, expr const & new_type);
|
2014-06-30 16:14:55 +00:00
|
|
|
expr update_local(expr const & e, expr const & new_type, binder_info const & bi);
|
2014-09-07 19:28:58 +00:00
|
|
|
expr update_local(expr const & e, binder_info const & bi);
|
2014-02-25 02:31:38 +00:00
|
|
|
expr update_sort(expr const & e, level const & new_level);
|
|
|
|
expr update_constant(expr const & e, levels const & new_levels);
|
2014-04-25 22:02:52 +00:00
|
|
|
expr update_macro(expr const & e, unsigned num, expr const * args);
|
2013-07-30 04:28:22 +00:00
|
|
|
// =======================================
|
2013-09-26 00:45:13 +00:00
|
|
|
|
2014-06-28 20:57:36 +00:00
|
|
|
// =======================================
|
|
|
|
// Implicit argument inference
|
|
|
|
/**
|
|
|
|
\brief Given \c t of the form <tt>Pi (x_1 : A_1) ... (x_k : A_k), B</tt>,
|
|
|
|
mark the first \c num_params as implicit if they are not already marked, and
|
|
|
|
they occur in the remaining arguments. If \c strict is false, then we
|
|
|
|
also mark it implicit if it occurs in \c B.
|
|
|
|
*/
|
|
|
|
expr infer_implicit(expr const & t, unsigned num_params, bool strict);
|
|
|
|
expr infer_implicit(expr const & t, bool strict);
|
|
|
|
// =======================================
|
|
|
|
|
2013-09-26 00:45:13 +00:00
|
|
|
std::ostream & operator<<(std::ostream & out, expr const & e);
|
2014-09-23 19:09:13 +00:00
|
|
|
|
|
|
|
void initialize_expr();
|
|
|
|
void finalize_expr();
|
2013-07-22 11:03:46 +00:00
|
|
|
}
|