2015-02-02 22:51:42 +00:00
|
|
|
/*
|
|
|
|
Copyright (c) 2015 Microsoft Corporation. All rights reserved.
|
|
|
|
Released under Apache 2.0 license as described in the file LICENSE.
|
|
|
|
|
|
|
|
Author: Leonardo de Moura
|
|
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include "library/kernel_serializer.h"
|
|
|
|
|
|
|
|
namespace lean {
|
|
|
|
/*
|
|
|
|
We can specify the scope of application of some clauses using
|
|
|
|
"locations". Locations specify goal and/or hypotheses and
|
|
|
|
"occurrences" inside a given location. The occurrences are
|
|
|
|
just numeric values and the specify which occurrences of a
|
|
|
|
term instance are considered by the tactic.
|
|
|
|
|
|
|
|
Examples:
|
|
|
|
|
|
|
|
- rewrite H1 in H at (1, 3)
|
|
|
|
|
|
|
|
says the scope of (rewrite H1) is the hypothesis H, and
|
|
|
|
only the first and third occurrences of the H1 left-hand-side
|
|
|
|
will be considered.
|
|
|
|
|
|
|
|
- rewrite H1 in H at -1
|
|
|
|
|
|
|
|
says the scope of (rewrite H1) is the hypothesis H and all
|
|
|
|
but the first occurrence of the H1 left-hand-side
|
|
|
|
will be considered.
|
|
|
|
*/
|
|
|
|
|
|
|
|
class occurrence {
|
|
|
|
enum kind { All, Pos, Neg };
|
|
|
|
kind m_kind;
|
|
|
|
list<unsigned> m_occs;
|
|
|
|
occurrence(kind k, list<unsigned> const & occs):m_kind(k), m_occs(occs) {}
|
|
|
|
public:
|
|
|
|
occurrence():m_kind(All) {}
|
|
|
|
static occurrence mk_occurrences(buffer<unsigned> const & occs) { return occurrence(Pos, to_list(occs)); }
|
|
|
|
static occurrence mk_occurrences(list<unsigned> const & occs) { return occurrence(Pos, occs); }
|
|
|
|
static occurrence mk_except_occurrences(buffer<unsigned> const & occs) { return occurrence(Neg, to_list(occs)); }
|
|
|
|
static occurrence mk_except_occurrences(list<unsigned> const & occs) { return occurrence(Neg, occs); }
|
|
|
|
bool is_all() const { return m_kind == All; }
|
|
|
|
bool is_except() const { return m_kind == Neg; }
|
|
|
|
/** \brief Return true iff this occurrence object contains the given occurrence index. */
|
|
|
|
unsigned contains(unsigned occ_idx) const;
|
|
|
|
friend serializer & operator<<(serializer & s, occurrence const & o);
|
|
|
|
friend deserializer & operator>>(deserializer & d, occurrence & o);
|
2015-03-14 02:24:09 +00:00
|
|
|
|
|
|
|
bool operator==(occurrence const & o) const { return m_kind == o.m_kind && m_occs == o.m_occs; }
|
|
|
|
bool operator!=(occurrence const & o) const { return !operator==(o); }
|
2015-02-02 22:51:42 +00:00
|
|
|
};
|
|
|
|
|
2015-02-04 23:17:58 +00:00
|
|
|
/** \brief Replace occurrences of \c t in \c e with the free variable #vidx.
|
|
|
|
The j-th occurrence is replaced iff occ.contains(j)
|
2015-06-28 19:05:00 +00:00
|
|
|
|
|
|
|
Return none when no occurrence was replaced.
|
2015-02-04 23:17:58 +00:00
|
|
|
*/
|
2015-06-28 19:05:00 +00:00
|
|
|
optional<expr> replace_occurrences(expr const & e, expr const & t, occurrence const & occ, unsigned vidx);
|
2015-02-04 23:17:58 +00:00
|
|
|
|
2015-02-02 22:51:42 +00:00
|
|
|
class location {
|
|
|
|
public:
|
|
|
|
enum kind { Everywhere, GoalOnly, AllHypotheses, Hypotheses, GoalHypotheses };
|
|
|
|
private:
|
|
|
|
kind m_kind;
|
|
|
|
occurrence m_goal;
|
|
|
|
list<pair<name, occurrence>> m_hyps;
|
2015-02-05 02:48:23 +00:00
|
|
|
location(kind k, occurrence const & g, list<pair<name, occurrence>> const & hs):
|
2015-02-02 22:51:42 +00:00
|
|
|
m_kind(k), m_goal(g), m_hyps(hs) {}
|
2015-02-05 02:48:23 +00:00
|
|
|
location(kind k, occurrence const & g):m_kind(k), m_goal(g) {}
|
|
|
|
location(kind k):location(k, occurrence()) {}
|
2015-02-02 22:51:42 +00:00
|
|
|
public:
|
|
|
|
location():m_kind(GoalOnly) {}
|
|
|
|
|
|
|
|
static location mk_goal_only() { return location(); }
|
|
|
|
static location mk_everywhere() { return location(Everywhere); }
|
|
|
|
static location mk_all_hypotheses() { return location(AllHypotheses); }
|
|
|
|
static location mk_hypotheses(buffer<name> const & hs);
|
|
|
|
static location mk_goal_hypotheses(buffer<name> const & hs);
|
|
|
|
static location mk_goal_at(occurrence const & o) { return location(GoalOnly, o); }
|
|
|
|
static location mk_hypotheses_at(buffer<name> const & hs, buffer<occurrence> const & occs);
|
|
|
|
static location mk_at(occurrence const & g_occs, buffer<name> const & hs, buffer<occurrence> const & hs_occs);
|
|
|
|
|
2015-02-04 17:59:34 +00:00
|
|
|
bool is_goal_only() const { return m_kind == GoalOnly; }
|
|
|
|
|
2015-02-02 22:51:42 +00:00
|
|
|
optional<occurrence> includes_goal() const;
|
|
|
|
optional<occurrence> includes_hypothesis(name const & h) const;
|
|
|
|
void get_explicit_hypotheses_names(buffer<name> & r) const;
|
|
|
|
|
2015-03-14 02:24:09 +00:00
|
|
|
bool operator==(location const & l) const { return m_kind == l.m_kind && m_goal == l.m_goal && m_hyps == l.m_hyps; }
|
|
|
|
bool operator!=(location const & l) const { return !operator==(l); }
|
|
|
|
|
2015-02-02 22:51:42 +00:00
|
|
|
friend serializer & operator<<(serializer & s, location const & loc);
|
|
|
|
friend deserializer & operator>>(deserializer & d, location & loc);
|
|
|
|
};
|
2015-07-13 21:56:42 +00:00
|
|
|
|
|
|
|
expr mk_location_expr(location const & loc);
|
|
|
|
bool is_location_expr(expr const & e);
|
|
|
|
location const & get_location_expr_location(expr const & e);
|
|
|
|
|
|
|
|
void initialize_location();
|
|
|
|
void finalize_location();
|
2015-02-02 22:51:42 +00:00
|
|
|
}
|