feat(library/meng_paulson): ignore class and class-instances when computing set of relevant theorems

This commit is contained in:
Leonardo de Moura 2015-10-14 20:46:03 -07:00
parent 023ec1ba76
commit ee347772d7
5 changed files with 101 additions and 18 deletions

View file

@ -1132,16 +1132,9 @@ static environment relevant_thms_cmd(parser & p) {
} }
name_set TS = get_relevant_thms(env, p.get_options(), R); name_set TS = get_relevant_thms(env, p.get_options(), R);
io_state_stream out = p.regular_stream(); io_state_stream out = p.regular_stream();
out << "{";
bool first = true;
TS.for_each([&](name const & T) { TS.for_each([&](name const & T) {
if (first) out << T << "\n";
first = false;
else
out << ", ";
out << T;
}); });
out << "}";
return env; return env;
} }

View file

@ -15,6 +15,7 @@ Author: Leonardo de Moura
#include "library/tc_multigraph.h" #include "library/tc_multigraph.h"
#include "library/protected.h" #include "library/protected.h"
#include "library/class.h" #include "library/class.h"
#include "library/decl_stats.h"
#ifndef LEAN_INSTANCE_DEFAULT_PRIORITY #ifndef LEAN_INSTANCE_DEFAULT_PRIORITY
#define LEAN_INSTANCE_DEFAULT_PRIORITY 1000 #define LEAN_INSTANCE_DEFAULT_PRIORITY 1000
@ -236,7 +237,8 @@ name get_class_name(environment const & env, expr const & e) {
environment add_class(environment const & env, name const & n, bool persistent) { environment add_class(environment const & env, name const & n, bool persistent) {
check_class(env, n); check_class(env, n);
return class_ext::add_entry(env, get_dummy_ios(), class_entry(n), persistent); environment new_env = class_ext::add_entry(env, get_dummy_ios(), class_entry(n), persistent);
return mark_class_instance_somewhere(new_env, n);
} }
void get_classes(environment const & env, buffer<name> & classes) { void get_classes(environment const & env, buffer<name> & classes) {
@ -273,7 +275,8 @@ environment add_instance(environment const & env, name const & n, unsigned prior
} }
name c = get_class_name(env, get_app_fn(type)); name c = get_class_name(env, get_app_fn(type));
check_is_class(env, c); check_is_class(env, c);
return class_ext::add_entry(env, get_dummy_ios(), class_entry(class_entry_kind::Instance, c, n, priority), persistent); environment new_env = class_ext::add_entry(env, get_dummy_ios(), class_entry(class_entry_kind::Instance, c, n, priority), persistent);
return mark_class_instance_somewhere(new_env, n);
} }
environment add_instance(environment const & env, name const & n, bool persistent) { environment add_instance(environment const & env, name const & n, bool persistent) {
@ -313,11 +316,13 @@ environment add_trans_instance(environment const & env, name const & n, unsigned
environment new_env = new_env_insts.first; environment new_env = new_env_insts.first;
new_env = class_ext::add_entry(new_env, get_dummy_ios(), new_env = class_ext::add_entry(new_env, get_dummy_ios(),
class_entry::mk_trans_inst(src_tgt.first, src_tgt.second, n, priority), persistent); class_entry::mk_trans_inst(src_tgt.first, src_tgt.second, n, priority), persistent);
new_env = mark_class_instance_somewhere(new_env, n);
for (tc_edge const & edge : new_env_insts.second) { for (tc_edge const & edge : new_env_insts.second) {
new_env = class_ext::add_entry(new_env, get_dummy_ios(), new_env = class_ext::add_entry(new_env, get_dummy_ios(),
class_entry::mk_derived_trans_inst(edge.m_from, edge.m_to, edge.m_cnst), persistent); class_entry::mk_derived_trans_inst(edge.m_from, edge.m_to, edge.m_cnst), persistent);
new_env = set_reducible(new_env, edge.m_cnst, reducible_status::Reducible, persistent); new_env = set_reducible(new_env, edge.m_cnst, reducible_status::Reducible, persistent);
new_env = add_protected(new_env, edge.m_cnst); new_env = add_protected(new_env, edge.m_cnst);
new_env = mark_class_instance_somewhere(new_env, edge.m_cnst);
} }
return new_env; return new_env;
} }

View file

@ -4,12 +4,15 @@ Released under Apache 2.0 license as described in the file LICENSE.
Author: Leonardo de Moura Author: Leonardo de Moura
*/ */
#include <string>
#include "kernel/environment.h" #include "kernel/environment.h"
#include "kernel/for_each_fn.h" #include "kernel/for_each_fn.h"
#include "library/reducible.h" #include "library/reducible.h"
#include "library/module.h"
namespace lean { namespace lean {
struct decl_stats_ext : public environment_extension { struct decl_stats_ext : public environment_extension {
name_set m_class_instance; // set of all classes and class instances declared in some namespace
name_map<unsigned> m_num_occs; name_map<unsigned> m_num_occs;
name_map<name_set> m_use_set; name_map<name_set> m_use_set;
name_map<name_set> m_used_by_set; name_map<name_set> m_used_by_set;
@ -89,6 +92,33 @@ public:
} }
}; };
environment mark_class_instance_somewhere_core(environment const & env, name const & n) {
decl_stats_ext ext = get_extension(env);
ext.m_class_instance.insert(n);
return update(env, ext);
}
static std::string * g_key = nullptr;
environment mark_class_instance_somewhere(environment const & env, name const & n) {
environment new_env = mark_class_instance_somewhere_core(env, n);
return module::add(new_env, *g_key, [=](environment const &, serializer & s) { s << n; });
}
bool is_class_instance_somewhere(environment const & env, name const & n) {
return get_extension(env).m_class_instance.contains(n);
}
static void class_instance_mark_reader(deserializer & d, shared_environment & senv,
std::function<void(asynch_update_fn const &)> &,
std::function<void(delayed_update_fn const &)> &) {
name n;
d >> n;
senv.update([=](environment const & env) -> environment {
return mark_class_instance_somewhere_core(env, n);
});
}
environment update_decl_stats(environment const & env, declaration const & d) { environment update_decl_stats(environment const & env, declaration const & d) {
return update_decl_stats_fn(env)(d); return update_decl_stats_fn(env)(d);
} }
@ -115,10 +145,13 @@ name_set get_used_by_set(environment const & env, name const & n) {
} }
void initialize_decl_stats() { void initialize_decl_stats() {
g_ext = new decl_stats_ext_reg(); g_ext = new decl_stats_ext_reg();
g_key = new std::string("CIGLB");
register_module_object_reader(*g_key, class_instance_mark_reader);
} }
void finalize_decl_stats() { void finalize_decl_stats() {
delete g_ext; delete g_ext;
delete g_key;
} }
} }

View file

@ -20,6 +20,12 @@ unsigned get_num_occs(environment const & env, name const & n);
name_set get_use_set(environment const & env, name const & n); name_set get_use_set(environment const & env, name const & n);
/** \brief Return the set of constants s.t. \c n occur in their type. */ /** \brief Return the set of constants s.t. \c n occur in their type. */
name_set get_used_by_set(environment const & env, name const & n); name_set get_used_by_set(environment const & env, name const & n);
/** \brief Mark that \c n is a class or class-instance in some namespace. */
environment mark_class_instance_somewhere(environment const & env, name const & n);
/** \brief Return true iff \c n is a class or class-instance in some namespace. */
bool is_class_instance_somewhere(environment const & env, name const & n);
void initialize_decl_stats(); void initialize_decl_stats();
void finalize_decl_stats(); void finalize_decl_stats();
} }

View file

@ -14,8 +14,11 @@ the approach described at
#include <math.h> #include <math.h>
#include "util/sexpr/option_declarations.h" #include "util/sexpr/option_declarations.h"
#include "kernel/environment.h" #include "kernel/environment.h"
#include "kernel/inductive/inductive.h"
#include "library/decl_stats.h" #include "library/decl_stats.h"
#include "library/private.h" #include "library/private.h"
#include "library/class.h"
#include "library/constants.h"
#ifndef LEAN_DEFAULT_MENG_PAULSON_P #ifndef LEAN_DEFAULT_MENG_PAULSON_P
#define LEAN_DEFAULT_MENG_PAULSON_P 0.6 #define LEAN_DEFAULT_MENG_PAULSON_P 0.6
@ -65,17 +68,55 @@ class relevant_thms_fn {
return 1.0 + 2.0 / log(r + 1.0); return 1.0 + 2.0 / log(r + 1.0);
} }
bool is_connective(name const & n) const {
return
n == get_or_name() ||
n == get_and_name() ||
n == get_not_name() ||
n == get_iff_name() ||
n == get_not_name() ||
n == get_ite_name() ||
n == get_true_name() ||
n == get_false_name();
}
// constants symbols in theorem types that should be ignored
bool ignore_F(name const & F) const {
if (is_private(m_env, F))
return true; // we ignore private decls
if (is_class_instance_somewhere(m_env, F))
return true; // ignore if F is a class or class-instance in some namespace
if (is_connective(F))
return true;
return false;
}
// Theorems/Axioms that should be ignored
bool ignore_T(name const & T) const {
if (is_private(m_env, T))
return true;
if (inductive::is_elim_rule(m_env, T))
return true;
if (inductive::is_intro_rule(m_env, T))
return true;
return false;
}
double get_thm_score(name const & n) const { double get_thm_score(name const & n) const {
name_set s = get_use_set(m_env, n); name_set s = get_use_set(m_env, n);
unsigned IR = 0; unsigned IR = 0;
double M = 0.0; double M = 0.0;
s.for_each([&](name const & F) { s.for_each([&](name const & F) {
if (ignore_F(F))
return;
if (m_relevant.contains(F)) { if (m_relevant.contains(F)) {
M += get_weight(F); M += get_weight(F);
} else { } else {
// std::cout << "IR: " << F << "\n";
IR++; IR++;
} }
}); });
// std::cout << n << " M: " << M << " IR: " << IR << "\n";
if (M > 0.0) if (M > 0.0)
return M / (M + IR); return M / (M + IR);
else else
@ -89,7 +130,9 @@ public:
name_set operator()() { name_set operator()() {
name_set A; name_set A;
// unsigned i = 1;
while (true) { while (true) {
// std::cout << "#" << i << ", p: " << m_p << "\n";
name_set Rel; name_set Rel;
m_relevant.for_each([&](name const & c) { m_relevant.for_each([&](name const & c) {
name_set used_by = get_used_by_set(m_env, c); name_set used_by = get_used_by_set(m_env, c);
@ -99,9 +142,10 @@ public:
return; // T is already in the result set return; // T is already in the result set
if (!T_decl.is_theorem() && !T_decl.is_axiom()) if (!T_decl.is_theorem() && !T_decl.is_axiom())
return; // we only care about axioms and theorems return; // we only care about axioms and theorems
if (is_private(m_env, T)) if (ignore_T(T))
return; // we ignore private decls return; // we ignore private decls
double M = get_thm_score(T); double M = get_thm_score(T);
// std::cout << T << " : " << M << "\n";
if (M < m_p) if (M < m_p)
return; // score is to low return; // score is to low
Rel.insert(T); Rel.insert(T);
@ -114,13 +158,15 @@ public:
// include symbols of new theorems in m_relevant // include symbols of new theorems in m_relevant
Rel.for_each([&](name const & T) { Rel.for_each([&](name const & T) {
name_set uses = get_use_set(m_env, T); name_set uses = get_use_set(m_env, T);
uses.for_each([&](name const & c) { uses.for_each([&](name const & F) {
declaration const & c_decl = m_env.get(c); declaration const & F_decl = m_env.get(F);
if (c_decl.is_theorem() || c_decl.is_axiom()) if (F_decl.is_theorem() || F_decl.is_axiom())
return; // we ignore theorems occurring in types return; // we ignore theorems occurring in types
if (is_private(m_env, c)) if (ignore_F(F))
return; // we ignore private decls return;
m_relevant.insert(c); // if (!m_relevant.contains(F))
// std::cout << "new relevant: " << F << "\n";
m_relevant.insert(F);
}); });
}); });
m_p = m_p + (1.0 - m_p) / m_c; m_p = m_p + (1.0 - m_p) / m_c;