Add sexpr test
Signed-off-by: Leonardo de Moura <leonardo@microsoft.com>
This commit is contained in:
parent
e6a694ebcc
commit
aed8a07c1b
5 changed files with 56 additions and 22 deletions
|
@ -1,2 +1,2 @@
|
||||||
add_library(kernel expr.cpp)
|
add_library(kernel expr.cpp expr_max_shared)
|
||||||
target_link_libraries(kernel ${EXTRA_LIBS})
|
target_link_libraries(kernel ${EXTRA_LIBS})
|
||||||
|
|
|
@ -123,7 +123,7 @@ bool eq(expr const & a, expr const & b) {
|
||||||
if (a.kind() != b.kind()) return false;
|
if (a.kind() != b.kind()) return false;
|
||||||
if (is_var(a)) return get_var_idx(a) == get_var_idx(b);
|
if (is_var(a)) return get_var_idx(a) == get_var_idx(b);
|
||||||
if (is_prop(a)) return true;
|
if (is_prop(a)) return true;
|
||||||
if (get_rc(a) > 1 && get_rc(b) > 1) {
|
if (is_shared(a) && is_shared(b)) {
|
||||||
auto p = std::make_pair(a.raw(), b.raw());
|
auto p = std::make_pair(a.raw(), b.raw());
|
||||||
if (g_eq_visited.find(p) != g_eq_visited.end())
|
if (g_eq_visited.find(p) != g_eq_visited.end())
|
||||||
return true;
|
return true;
|
||||||
|
|
|
@ -260,6 +260,7 @@ inline expr_numeral * to_numeral(expr const & e) { return to_numeral(e.r
|
||||||
// =======================================
|
// =======================================
|
||||||
// Accessors
|
// Accessors
|
||||||
inline unsigned get_rc(expr_cell * e) { return e->get_rc(); }
|
inline unsigned get_rc(expr_cell * e) { return e->get_rc(); }
|
||||||
|
inline bool is_shared(expr_cell * e) { return get_rc(e) > 1; }
|
||||||
inline unsigned get_var_idx(expr_cell * e) { return to_var(e)->get_vidx(); }
|
inline unsigned get_var_idx(expr_cell * e) { return to_var(e)->get_vidx(); }
|
||||||
inline name const & get_const_name(expr_cell * e) { return to_constant(e)->get_name(); }
|
inline name const & get_const_name(expr_cell * e) { return to_constant(e)->get_name(); }
|
||||||
inline unsigned get_const_pos(expr_cell * e) { return to_constant(e)->get_pos(); }
|
inline unsigned get_const_pos(expr_cell * e) { return to_constant(e)->get_pos(); }
|
||||||
|
@ -273,6 +274,7 @@ inline uvar const & get_ty_var(expr_cell * e, unsigned idx) { return to_type(e)-
|
||||||
inline mpz const & get_numeral(expr_cell * e) { return to_numeral(e)->get_num(); }
|
inline mpz const & get_numeral(expr_cell * e) { return to_numeral(e)->get_num(); }
|
||||||
|
|
||||||
inline unsigned get_rc(expr const & e) { return e.raw()->get_rc(); }
|
inline unsigned get_rc(expr const & e) { return e.raw()->get_rc(); }
|
||||||
|
inline bool is_shared(expr const & e) { return get_rc(e) > 1; }
|
||||||
inline unsigned get_var_idx(expr const & e) { return to_var(e)->get_vidx(); }
|
inline unsigned get_var_idx(expr const & e) { return to_var(e)->get_vidx(); }
|
||||||
inline name const & get_const_name(expr const & e) { return to_constant(e)->get_name(); }
|
inline name const & get_const_name(expr const & e) { return to_constant(e)->get_name(); }
|
||||||
inline unsigned get_const_pos(expr const & e) { return to_constant(e)->get_pos(); }
|
inline unsigned get_const_pos(expr const & e) { return to_constant(e)->get_pos(); }
|
||||||
|
|
|
@ -7,6 +7,7 @@ Author: Leonardo de Moura
|
||||||
#pragma once
|
#pragma once
|
||||||
#include <unordered_set>
|
#include <unordered_set>
|
||||||
#include "expr.h"
|
#include "expr.h"
|
||||||
|
#include "expr_functors.h"
|
||||||
#include "hash.h"
|
#include "hash.h"
|
||||||
|
|
||||||
namespace lean {
|
namespace lean {
|
||||||
|
@ -14,12 +15,6 @@ namespace lean {
|
||||||
// =======================================
|
// =======================================
|
||||||
// Expression Set
|
// Expression Set
|
||||||
// Remark: to expressions are assumed to be equal if they are "pointer-equal"
|
// Remark: to expressions are assumed to be equal if they are "pointer-equal"
|
||||||
struct expr_hash {
|
|
||||||
unsigned operator()(expr const & e) const { return e.hash(); }
|
|
||||||
};
|
|
||||||
struct expr_eqp {
|
|
||||||
bool operator()(expr const & e1, expr const & e2) const { return eqp(e1, e2); }
|
|
||||||
};
|
|
||||||
typedef std::unordered_set<expr, expr_hash, expr_eqp> expr_set;
|
typedef std::unordered_set<expr, expr_hash, expr_eqp> expr_set;
|
||||||
// =======================================
|
// =======================================
|
||||||
|
|
||||||
|
@ -30,12 +25,6 @@ typedef std::unordered_set<expr, expr_hash, expr_eqp> expr_set;
|
||||||
// WARNING: use with care, this kind of set
|
// WARNING: use with care, this kind of set
|
||||||
// does not prevent an expression from being
|
// does not prevent an expression from being
|
||||||
// garbage collected.
|
// garbage collected.
|
||||||
struct expr_cell_hash {
|
|
||||||
unsigned operator()(expr_cell * e) const { return e->hash(); }
|
|
||||||
};
|
|
||||||
struct expr_cell_eqp {
|
|
||||||
bool operator()(expr_cell * e1, expr_cell * e2) const { return e1 == e2; }
|
|
||||||
};
|
|
||||||
typedef std::unordered_set<expr_cell*, expr_cell_hash, expr_cell_eqp> expr_cell_set;
|
typedef std::unordered_set<expr_cell*, expr_cell_hash, expr_cell_eqp> expr_cell_set;
|
||||||
// =======================================
|
// =======================================
|
||||||
|
|
||||||
|
|
|
@ -5,11 +5,13 @@ Released under Apache 2.0 license as described in the file LICENSE.
|
||||||
Author: Leonardo de Moura
|
Author: Leonardo de Moura
|
||||||
*/
|
*/
|
||||||
#include "expr.h"
|
#include "expr.h"
|
||||||
|
#include "expr_max_shared.h"
|
||||||
|
#include "expr_set.h"
|
||||||
#include "test.h"
|
#include "test.h"
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
using namespace lean;
|
using namespace lean;
|
||||||
|
|
||||||
static void tst1() {
|
void tst1() {
|
||||||
expr a;
|
expr a;
|
||||||
a = numeral(mpz(10));
|
a = numeral(mpz(10));
|
||||||
expr f;
|
expr f;
|
||||||
|
@ -96,7 +98,7 @@ unsigned depth3(expr const & e) {
|
||||||
return m;
|
return m;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void tst2() {
|
void tst2() {
|
||||||
expr r1 = mk_dag(20);
|
expr r1 = mk_dag(20);
|
||||||
expr r2 = mk_dag(20);
|
expr r2 = mk_dag(20);
|
||||||
lean_verify(r1 == r2);
|
lean_verify(r1 == r2);
|
||||||
|
@ -111,14 +113,14 @@ expr mk_big(expr f, unsigned depth, unsigned val) {
|
||||||
return app({f, mk_big(f, depth - 1, val << 1), mk_big(f, depth - 1, (val << 1) + 1)});
|
return app({f, mk_big(f, depth - 1, val << 1), mk_big(f, depth - 1, (val << 1) + 1)});
|
||||||
}
|
}
|
||||||
|
|
||||||
static void tst3() {
|
void tst3() {
|
||||||
expr f = constant(name("f"));
|
expr f = constant(name("f"));
|
||||||
expr r1 = mk_big(f, 18, 0);
|
expr r1 = mk_big(f, 18, 0);
|
||||||
expr r2 = mk_big(f, 18, 0);
|
expr r2 = mk_big(f, 18, 0);
|
||||||
lean_verify(r1 == r2);
|
lean_verify(r1 == r2);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void tst4() {
|
void tst4() {
|
||||||
expr f = constant(name("f"));
|
expr f = constant(name("f"));
|
||||||
expr a = var(0);
|
expr a = var(0);
|
||||||
for (unsigned i = 0; i < 10000; i++) {
|
for (unsigned i = 0; i < 10000; i++) {
|
||||||
|
@ -126,14 +128,55 @@ static void tst4() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
expr mk_redundant_dag(expr f, unsigned depth) {
|
||||||
|
if (depth == 0)
|
||||||
|
return var(0);
|
||||||
|
else
|
||||||
|
return app({f, mk_redundant_dag(f, depth - 1), mk_redundant_dag(f, depth - 1)});
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
unsigned count_core(expr const & a, expr_set & s) {
|
||||||
|
if (s.find(a) != s.end())
|
||||||
|
return 0;
|
||||||
|
s.insert(a);
|
||||||
|
switch (a.kind()) {
|
||||||
|
case expr_kind::Var: case expr_kind::Constant: case expr_kind::Prop: case expr_kind::Type: case expr_kind::Numeral:
|
||||||
|
return 1;
|
||||||
|
case expr_kind::App:
|
||||||
|
return std::accumulate(begin_args(a), end_args(a), 1,
|
||||||
|
[&](unsigned sum, expr const & arg){ return sum + count_core(arg, s); });
|
||||||
|
case expr_kind::Lambda: case expr_kind::Pi:
|
||||||
|
return count_core(get_abs_type(a), s) + count_core(get_abs_expr(a), s) + 1;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned count(expr const & a) {
|
||||||
|
expr_set s;
|
||||||
|
return count_core(a, s);
|
||||||
|
}
|
||||||
|
|
||||||
|
void tst5() {
|
||||||
|
expr f = constant(name("f"));
|
||||||
|
expr r1 = mk_redundant_dag(f, 1);
|
||||||
|
expr r2 = max_shared(r1);
|
||||||
|
std::cout << "r1: " << r1 << "\n";
|
||||||
|
std::cout << "r2: " << r1 << "\n";
|
||||||
|
std::cout << "count(r1): " << count(r1) << "\n";
|
||||||
|
std::cout << "count(r2): " << count(r2) << "\n";
|
||||||
|
lean_assert(r1 == r2);
|
||||||
|
}
|
||||||
|
|
||||||
int main() {
|
int main() {
|
||||||
// continue_on_violation(true);
|
// continue_on_violation(true);
|
||||||
std::cout << "sizeof(expr): " << sizeof(expr) << "\n";
|
std::cout << "sizeof(expr): " << sizeof(expr) << "\n";
|
||||||
std::cout << "sizeof(expr_app): " << sizeof(expr_app) << "\n";
|
std::cout << "sizeof(expr_app): " << sizeof(expr_app) << "\n";
|
||||||
tst1();
|
// tst1();
|
||||||
tst2();
|
// tst2();
|
||||||
tst3();
|
// tst3();
|
||||||
tst4();
|
// tst4();
|
||||||
|
tst5();
|
||||||
std::cout << "done" << "\n";
|
std::cout << "done" << "\n";
|
||||||
return has_violations() ? 1 : 0;
|
return has_violations() ? 1 : 0;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue