/* Copyright (c) 2015 Microsoft Corporation. All rights reserved. Released under Apache 2.0 license as described in the file LICENSE. Author: Robert Y. Lewis */ #include "library/norm_num.h" #include "library/constants.h" namespace lean { static bool is_numeral_aux(expr const & e, bool is_first) { buffer args; expr const & f = get_app_args(e, args); if (!is_constant(f)) { return false; } if (const_name(f) == get_one_name()) { return args.size() == 2; } else if (const_name(f) == get_zero_name()) { return is_first && args.size() == 2; } else if (const_name(f) == get_bit0_name()) { return args.size() == 3 && is_numeral_aux(args[2], false); } else if (const_name(f) == get_bit1_name()) { return args.size() == 4 && is_numeral_aux(args[3], false); } return false; } bool norm_num_context::is_numeral(expr const & e) const { return is_numeral_aux(e, true); } bool norm_num_context::is_neg_app(expr const & e) const { return is_const_app(e, get_neg_name(), 3); } bool norm_num_context::is_div(expr const & e) const { return is_const_app(e, get_div_name(), 4); } /* Takes A : Type, and tries to synthesize has_add A. */ expr norm_num_context::mk_has_add(expr const & e) { auto l_name = get_has_add_name(); if (instances.find(l_name) != instances.end()) { return instances[l_name]; } expr t = mk_app(mk_constant(l_name, m_lvls), e); optional inst = m_type_ctx.mk_class_instance(t); if (inst) { instances[l_name] = *inst; return *inst; } else { throw exception("failed to synthesize has_add instance"); } } expr norm_num_context::mk_has_mul(expr const & e) { auto l_name = get_has_mul_name(); if (instances.find(l_name) != instances.end()) { return instances[l_name]; } expr t = mk_app(mk_constant(l_name, m_lvls), e); optional inst = m_type_ctx.mk_class_instance(t); if (inst) { instances[l_name] = *inst; return *inst; } else { throw exception("failed to synthesize has_mul instance"); } } expr norm_num_context::mk_has_one(expr const & e) { auto l_name = get_has_one_name(); if (instances.find(l_name) != instances.end()) { return instances[l_name]; } expr t = mk_app(mk_constant(l_name, m_lvls), e); optional inst = m_type_ctx.mk_class_instance(t); if (inst) { instances[l_name] = *inst; return *inst; } else { throw exception("failed to synthesize has_one instance"); } } expr norm_num_context::mk_has_zero(expr const & e) { auto l_name = get_has_zero_name(); if (instances.find(l_name) != instances.end()) { return instances[l_name]; } expr t = mk_app(mk_constant(l_name, m_lvls), e); optional inst = m_type_ctx.mk_class_instance(t); if (inst) { instances[l_name] = *inst; return *inst; } else { throw exception("failed to synthesize has_one instance"); } } expr norm_num_context::mk_add_monoid(expr const & e) { auto l_name = get_add_monoid_name(); if (instances.find(l_name) != instances.end()) { return instances[l_name]; } expr t = mk_app(mk_constant(l_name, m_lvls), e); optional inst = m_type_ctx.mk_class_instance(t); if (inst) { instances[l_name] = *inst; return *inst; } else { throw exception("failed to synthesize add_monoid instance"); } } expr norm_num_context::mk_monoid(expr const & e) { auto l_name = get_monoid_name(); if (instances.find(l_name) != instances.end()) { return instances[l_name]; } expr t = mk_app(mk_constant(l_name, m_lvls), e); optional inst = m_type_ctx.mk_class_instance(t); if (inst) { instances[l_name] = *inst; return *inst; } else { throw exception("failed to synthesize monoid instance"); } } expr norm_num_context::mk_field(expr const & e) { expr t = mk_app(mk_constant(get_field_name(), m_lvls), e); optional inst = m_type_ctx.mk_class_instance(t); if (inst) { return *inst; } else { throw exception("failed to synthesize field instance"); } } expr norm_num_context::mk_add_comm(expr const & e) { auto l_name = get_add_comm_semigroup_name(); if (instances.find(l_name) != instances.end()) { return instances[l_name]; } expr t = mk_app(mk_constant(l_name, m_lvls), e); optional inst = m_type_ctx.mk_class_instance(t); if (inst) { instances[l_name] = *inst; return *inst; } else { throw exception("failed to synthesize add_comm_semigroup instance"); } } expr norm_num_context::mk_add_group(expr const & e) { auto l_name = get_add_group_name(); if (instances.find(l_name) != instances.end()) { return instances[l_name]; } expr t = mk_app(mk_constant(l_name, m_lvls), e); optional inst = m_type_ctx.mk_class_instance(t); if (inst) { instances[l_name] = *inst; return *inst; } else { throw exception("failed to synthesize add_comm_semigroup instance"); } } expr norm_num_context::mk_has_distrib(expr const & e) { auto l_name = get_distrib_name(); if (instances.find(l_name) != instances.end()) { return instances[l_name]; } expr t = mk_app(mk_constant(l_name, m_lvls), e); optional inst = m_type_ctx.mk_class_instance(t); if (inst) { instances[l_name] = *inst; return *inst; } else { throw exception("failed to synthesize has_distrib instance"); } } expr norm_num_context::mk_mul_zero_class(expr const & e) { auto l_name = get_mul_zero_class_name(); if (instances.find(l_name) != instances.end()) { return instances[l_name]; } expr t = mk_app(mk_constant(l_name, m_lvls), e); optional inst = m_type_ctx.mk_class_instance(t); if (inst) { instances[l_name] = *inst; return *inst; } else { throw exception("failed to synthesize mul_zero instance"); } } expr norm_num_context::mk_semiring(expr const & e) { auto l_name = get_semiring_name(); if (instances.find(l_name) != instances.end()) { return instances[l_name]; } expr t = mk_app(mk_constant(l_name, m_lvls), e); optional inst = m_type_ctx.mk_class_instance(t); if (inst) { instances[l_name] = *inst; return *inst; } else { throw exception("failed to synthesize semiring instance"); } } expr norm_num_context::mk_has_neg(expr const & e) { auto l_name = get_has_neg_name(); if (instances.find(l_name) != instances.end()) { return instances[l_name]; } expr t = mk_app(mk_constant(l_name, m_lvls), e); optional inst = m_type_ctx.mk_class_instance(t); if (inst) { instances[l_name] = *inst; return *inst; } else { throw exception("failed to synthesize has_neg instance"); } } expr norm_num_context::mk_has_sub(expr const & e) { auto l_name = get_has_sub_name(); if (instances.find(l_name) != instances.end()) { return instances[l_name]; } expr t = mk_app(mk_constant(l_name, m_lvls), e); optional inst = m_type_ctx.mk_class_instance(t); if (inst) { instances[l_name] = *inst; return *inst; } else { throw exception("failed to synthesize has_sub instance"); } } expr norm_num_context::mk_has_div(expr const & e) { auto l_name = get_has_div_name(); if (instances.find(l_name) != instances.end()) { return instances[l_name]; } expr t = mk_app(mk_constant(l_name, m_lvls), e); optional inst = m_type_ctx.mk_class_instance(t); if (inst) { instances[l_name] = *inst; return *inst; } else { throw exception("failed to synthesize has_div instance"); } } expr norm_num_context::mk_add_comm_group(expr const & e) { auto l_name = get_add_comm_group_name(); if (instances.find(l_name) != instances.end()) { return instances[l_name]; } expr t = mk_app(mk_constant(l_name, m_lvls), e); optional inst = m_type_ctx.mk_class_instance(t); if (inst) { instances[l_name] = *inst; return *inst; } else { throw exception("failed to synthesize add_comm_group instance"); } } expr norm_num_context::mk_ring(expr const & e) { auto l_name = get_ring_name(); if (instances.find(l_name) != instances.end()) { return instances[l_name]; } expr t = mk_app(mk_constant(l_name, m_lvls), e); optional inst = m_type_ctx.mk_class_instance(t); if (inst) { instances[l_name] = *inst; return *inst; } else { throw exception("failed to synthesize ring instance"); } } expr norm_num_context::mk_lin_ord_ring(expr const & e) { auto l_name = get_linear_ordered_ring_name(); if (instances.find(l_name) != instances.end()) { return instances[l_name]; } expr t = mk_app(mk_constant(l_name, m_lvls), e); optional inst = m_type_ctx.mk_class_instance(t); if (inst) { instances[l_name] = *inst; return *inst; } else { throw exception("failed to synthesize lin_ord_ring instance"); } } expr norm_num_context::mk_lin_ord_semiring(expr const & e) { auto l_name = get_linear_ordered_semiring_name(); if (instances.find(l_name) != instances.end()) { return instances[l_name]; } expr t = mk_app(mk_constant(l_name, m_lvls), e); optional inst = m_type_ctx.mk_class_instance(t); if (inst) { instances[l_name] = *inst; return *inst; } else { throw exception("failed to synthesize lin_ord_semiring instance"); } } expr norm_num_context::mk_wk_order(expr const & e) { auto l_name = get_weak_order_name(); if (instances.find(l_name) != instances.end()) { return instances[l_name]; } expr t = mk_app(mk_constant(l_name, m_lvls), e); optional inst = m_type_ctx.mk_class_instance(t); if (inst) { instances[l_name] = *inst; return *inst; } else { throw exception("failed to synthesize weak_order instance"); } } expr norm_num_context::mk_const(name const & n) { return mk_constant(n, m_lvls); } expr norm_num_context::mk_cong(expr const & op, expr const & type, expr const & a, expr const & b, expr const & eq) { return mk_app({mk_const(get_norm_num_mk_cong_name()), type, op, a, b, eq}); } // returns such that p is a proof that lhs + rhs = t. pair norm_num_context::mk_norm_add(expr const & lhs, expr const & rhs) { buffer args_lhs; buffer args_rhs; expr lhs_head = get_app_args (lhs, args_lhs); expr rhs_head = get_app_args (rhs, args_rhs); if (!is_constant(lhs_head) || !is_constant(rhs_head)) { throw exception("cannot take norm_add of nonconstant"); } auto type = args_lhs[0]; auto typec = args_lhs[1]; expr rv; expr prf; if (is_bit0(lhs) && is_bit0(rhs)) { // typec is has_add auto p = mk_norm_add(args_lhs[2], args_rhs[2]); rv = mk_app(lhs_head, type, typec, p.first); prf = mk_app({mk_const(get_norm_num_bit0_add_bit0_helper_name()), type, mk_add_comm(type), args_lhs[2], args_rhs[2], p.first, p.second}); } else if (is_bit0(lhs) && is_bit1(rhs)) { auto p = mk_norm_add(args_lhs[2], args_rhs[3]); rv = mk_app({rhs_head, type, args_rhs[1], args_rhs[2], p.first}); prf = mk_app({mk_const(get_norm_num_bit0_add_bit1_helper_name()), type, mk_add_comm(type), args_rhs[1], args_lhs[2], args_rhs[3], p.first, p.second}); } else if (is_bit0(lhs) && is_one(rhs)) { rv = mk_app({mk_const(get_bit1_name()), type, args_rhs[1], args_lhs[1], args_lhs[2]}); prf = mk_app({mk_const(get_norm_num_bit0_add_one_name()), type, typec, args_rhs[1], args_lhs[2]}); } else if (is_bit1(lhs) && is_bit0(rhs)) { // typec is has_one auto p = mk_norm_add(args_lhs[3], args_rhs[2]); rv = mk_app(lhs_head, type, typec, args_lhs[2], p.first); prf = mk_app({mk_const(get_norm_num_bit1_add_bit0_helper_name()), type, mk_add_comm(type), typec, args_lhs[3], args_rhs[2], p.first, p.second}); } else if (is_bit1(lhs) && is_bit1(rhs)) { // typec is has_one auto add_ts = mk_norm_add(args_lhs[3], args_rhs[3]); expr add1 = mk_app({mk_const(get_norm_num_add1_name()), type, args_lhs[2], typec, add_ts.first}); auto p = mk_norm_add1(add1); rv = mk_app({mk_const(get_bit0_name()), type, args_lhs[2], p.first}); prf = mk_app({mk_const(get_norm_num_bit1_add_bit1_helper_name()), type, mk_add_comm(type), typec, args_lhs[3], args_rhs[3], add_ts.first, p.first, add_ts.second, p.second}); } else if (is_bit1(lhs) && is_one(rhs)) { // typec is has_one expr add1 = mk_app({mk_const(get_norm_num_add1_name()), type, args_lhs[2], typec, lhs}); auto p = mk_norm_add1(add1); rv = p.first; prf = mk_app({mk_const(get_norm_num_bit1_add_one_helper_name()), type, args_lhs[2], typec, args_lhs[3], p.first, p.second}); } else if (is_one(lhs) && is_bit0(rhs)) { // typec is has_one rv = mk_app({mk_const(get_bit1_name()), type, typec, args_rhs[1], args_rhs[2]}); prf = mk_app({mk_const(get_norm_num_one_add_bit0_name()), type, mk_add_comm(type), typec, args_rhs[2]}); } else if (is_one(lhs) && is_bit1(rhs)) { // typec is has_one expr add1 = mk_app({mk_const(get_norm_num_add1_name()), type, args_rhs[2], args_rhs[1], rhs}); auto p = mk_norm_add1(add1); rv = p.first; prf = mk_app({mk_const(get_norm_num_one_add_bit1_helper_name()), type, mk_add_comm(type), typec, args_rhs[3], p.first, p.second}); } else if (is_one(lhs) && is_one(rhs)) { rv = mk_app({mk_const(get_bit0_name()), type, mk_has_add(type), lhs}); prf = mk_app({mk_const(get_norm_num_one_add_one_name()), type, mk_has_add(type), typec}); } else if (is_zero(lhs)) { rv = rhs; prf = mk_app({mk_const(get_norm_num_bin_zero_add_name()), type, mk_add_monoid(type), rhs}); } else if (is_zero(rhs)) { rv = lhs; prf = mk_app({mk_const(get_norm_num_bin_add_zero_name()), type, mk_add_monoid(type), lhs}); } else { throw exception("mk_norm_add got malformed args"); } return pair(rv, prf); } pair norm_num_context::mk_norm_add1(expr const & e) { buffer args; expr f = get_app_args(e, args); expr p = args[3]; buffer ne_args; expr ne = get_app_args(p, ne_args); expr rv; expr prf; // args[1] = has_add, args[2] = has_one if (is_bit0(p)) { auto has_one = args[2]; rv = mk_app({mk_const(get_bit1_name()), args[0], args[2], args[1], ne_args[2]}); prf = mk_app({mk_const(get_norm_num_add1_bit0_name()), args[0], args[1], args[2], ne_args[2]}); } else if (is_bit1(p)) { // ne_args : has_one, has_add auto np = mk_norm_add1(mk_app({mk_const(get_norm_num_add1_name()), args[0], args[1], args[2], ne_args[3]})); rv = mk_app({mk_const(get_bit0_name()), args[0], args[1], np.first}); prf = mk_app({mk_const(get_norm_num_add1_bit1_helper_name()), args[0], mk_add_comm(args[0]), args[2], ne_args[3], np.first, np.second}); } else if (is_zero(p)) { rv = mk_app({mk_const(get_one_name()), args[0], args[2]}); prf = mk_app({mk_const(get_norm_num_add1_zero_name()), args[0], mk_add_monoid(args[0]), args[2]}); } else if (is_one(p)) { rv = mk_app({mk_const(get_bit0_name()), args[0], args[1], mk_app({mk_const(get_one_name()), args[0], args[2]})}); prf = mk_app({mk_const(get_norm_num_add1_one_name()), args[0], args[1], args[2]}); } else { throw exception("malformed add1"); } return pair(rv, prf); } pair norm_num_context::mk_norm_mul(expr const & lhs, expr const & rhs) { buffer args_lhs; buffer args_rhs; expr lhs_head = get_app_args (lhs, args_lhs); expr rhs_head = get_app_args (rhs, args_rhs); if (!is_constant(lhs_head) || !is_constant(rhs_head)) { throw exception("cannot take norm_add of nonconstant"); } auto type = args_rhs[0]; auto typec = args_rhs[1]; expr rv; expr prf; if (is_zero(rhs)) { rv = rhs; prf = mk_app({mk_const(get_mul_zero_name()), type, mk_mul_zero_class(type), lhs}); } else if (is_zero(lhs)) { rv = lhs; prf = mk_app({mk_const(get_zero_mul_name()), type, mk_mul_zero_class(type), rhs}); } else if (is_one(rhs)) { rv = lhs; prf = mk_app({mk_const(get_mul_one_name()), type, mk_monoid(type), lhs}); } else if (is_bit0(rhs)) { auto mtp = mk_norm_mul(lhs, args_rhs[2]); rv = mk_app({rhs_head, type, typec, mtp.first}); prf = mk_app({mk_const(get_norm_num_mul_bit0_helper_name()), type, mk_has_distrib(type), lhs, args_rhs[2], mtp.first, mtp.second}); } else if (is_bit1(rhs)) { auto mtp = mk_norm_mul(lhs, args_rhs[3]); auto atp = mk_norm_add(mk_app({mk_const(get_bit0_name()), type, args_rhs[2], mtp.first}), lhs); rv = atp.first; prf = mk_app({mk_const(get_norm_num_mul_bit1_helper_name()), type, mk_semiring(type), lhs, args_rhs[3], mtp.first, atp.first, mtp.second, atp.second}); } else { throw exception("mk_norm_mul got malformed args"); } return pair(rv, prf); } optional norm_num_context::to_mpq(expr const & e) { auto v = to_num(e); if (v) { return optional(mpq(*v)); } else { return optional(); } } mpq norm_num_context:: mpq_of_expr(expr const & e){ buffer args; expr f = get_app_args(e, args); if (!is_constant(f)) { throw exception("cannot find num of nonconstant"); } else if (const_name(f) == get_add_name() && args.size() == 4) { return mpq_of_expr(args[2]) + mpq_of_expr(args[3]); } else if (const_name(f) == get_mul_name() && args.size() == 4) { return mpq_of_expr(args[2]) * mpq_of_expr(args[3]); } else if (const_name(f) == get_sub_name() && args.size() == 4) { return mpq_of_expr(args[2]) - mpq_of_expr(args[3]); } else if (const_name(f) == get_div_name() && args.size() == 4) { mpq num = mpq_of_expr(args[2]), den = mpq_of_expr(args[3]); if (den != 0) return mpq_of_expr(args[2]) / mpq_of_expr(args[3]); else throw exception("divide by 0"); } else if (const_name(f) == get_neg_name() && args.size() == 3) { return neg(mpq_of_expr(args[2])); } else { auto v = to_mpq(e); if (v) { return *v; } else { throw exception("expression in mpq_of_expr is malfomed"); } } } mpz norm_num_context::num_of_expr(expr const & e) { buffer args; expr f = get_app_args(e, args); if (!is_constant(f)) { throw exception("cannot find num of nonconstant"); } auto v = to_num(e); if (v) { return *v; } if (const_name(f) == get_add_name() && args.size() == 4) { return num_of_expr(args[2]) + num_of_expr(args[3]); } else if (const_name(f) == get_mul_name() && args.size() == 4) { return num_of_expr(args[2]) * num_of_expr(args[3]); } else if (const_name(f) == get_sub_name() && args.size() == 4) { return num_of_expr(args[2]) - num_of_expr(args[3]); } else if (const_name(f) == get_neg_name() && args.size() == 3) { return neg(num_of_expr(args[2])); } else { throw exception("expression in num_of_expr is malfomed"); } } pair norm_num_context::get_type_and_arg_of_neg(expr & e) { lean_assert(is_neg_app(e)); buffer args; expr f = get_app_args(e, args); return pair(args[0], args[2]); } // returns a proof that s_lhs + s_rhs = rhs, where all are negated numerals expr norm_num_context::mk_norm_eq_neg_add_neg(expr & s_lhs, expr & s_rhs, expr & rhs) { lean_assert(is_neg_app(s_lhs)); lean_assert(is_neg_app(s_rhs)); lean_assert(is_neg_app(rhs)); auto s_lhs_v = get_type_and_arg_of_neg(s_lhs).second; auto s_rhs_v = get_type_and_arg_of_neg(s_rhs).second; auto rhs_v = get_type_and_arg_of_neg(rhs); expr type = rhs_v.first; auto sum_pr = mk_norm(mk_add(type, s_lhs_v, s_rhs_v)).second; return mk_app({mk_const(get_norm_num_neg_add_neg_helper_name()), type, mk_add_comm_group(type), s_lhs_v, s_rhs_v, rhs_v.second, sum_pr}); } expr norm_num_context::mk_norm_eq_neg_add_pos(expr & s_lhs, expr & s_rhs, expr & rhs) { lean_assert(is_neg_app(s_lhs)); lean_assert(!is_neg_app(s_rhs)); auto s_lhs_v = get_type_and_arg_of_neg(s_lhs); expr type = s_lhs_v.first; if (is_neg_app(rhs)) { auto rhs_v = get_type_and_arg_of_neg(rhs).second; auto sum_pr = mk_norm(mk_add(type, s_rhs, rhs_v)).second; return mk_app({mk_const(get_norm_num_neg_add_pos_helper1_name()), type, mk_add_comm_group(type), s_lhs_v.second, s_rhs, rhs_v, sum_pr}); } else { auto sum_pr = mk_norm(mk_add(type, s_lhs_v.second, rhs)).second; return mk_app({mk_const(get_norm_num_neg_add_pos_helper2_name()), type, mk_add_comm_group(type), s_lhs_v.second, s_rhs, rhs, sum_pr}); } } expr norm_num_context::mk_norm_eq_pos_add_neg(expr & s_lhs, expr & s_rhs, expr & rhs) { lean_assert(is_neg_app(s_rhs)); lean_assert(!is_neg_app(s_lhs)); expr prf = mk_norm_eq_neg_add_pos(s_rhs, s_lhs, rhs); expr type = get_type_and_arg_of_neg(s_rhs).first; return mk_app({mk_const(get_norm_num_pos_add_neg_helper_name()), type, mk_add_comm_group(type), s_lhs, s_rhs, rhs, prf}); } // returns a proof that s_lhs + s_rhs = rhs, where all are nonneg normalized numerals expr norm_num_context::mk_norm_eq_pos_add_pos(expr & s_lhs, expr & s_rhs, expr & DEBUG_CODE(rhs)) { lean_assert(!is_neg_app(s_lhs)); lean_assert(!is_neg_app(s_rhs)); lean_assert(!is_neg_app(rhs)); auto p = mk_norm_add(s_lhs, s_rhs); lean_assert(to_num(rhs) == to_num(p.first)); return p.second; } expr norm_num_context::mk_norm_eq_neg_mul_neg(expr & s_lhs, expr & s_rhs, expr & rhs) { lean_assert(is_neg_app(s_lhs)); lean_assert(is_neg_app(s_rhs)); lean_assert(!is_neg_app(rhs)); auto s_lhs_v = get_type_and_arg_of_neg(s_lhs).second; expr s_rhs_v, type; std::tie(type, s_rhs_v) = get_type_and_arg_of_neg(s_rhs); auto prod_pr = mk_norm(mk_mul(type, s_lhs_v, s_rhs_v)); lean_assert(to_num(rhs) == to_num(prod_pr.first)); return mk_app({mk_const(get_norm_num_neg_mul_neg_helper_name()), type, mk_ring(type), s_lhs_v, s_rhs_v, rhs, prod_pr.second}); } expr norm_num_context::mk_norm_eq_neg_mul_pos(expr & s_lhs, expr & s_rhs, expr & rhs) { lean_assert(is_neg_app(s_lhs)); lean_assert(!is_neg_app(s_rhs)); lean_assert(is_neg_app(rhs)); expr s_lhs_v, type; std::tie(type, s_lhs_v) = get_type_and_arg_of_neg(s_lhs); auto rhs_v = get_type_and_arg_of_neg(rhs).second; auto prod_pr = mk_norm(mk_mul(type, s_lhs_v, s_rhs)); return mk_app({mk_const(get_norm_num_neg_mul_pos_helper_name()), type, mk_ring(type), s_lhs_v, s_rhs, rhs_v, prod_pr.second}); } expr norm_num_context::mk_norm_eq_pos_mul_neg(expr & s_lhs, expr & s_rhs, expr & rhs) { lean_assert(!is_neg_app(s_lhs)); lean_assert(is_neg_app(s_rhs)); lean_assert(is_neg_app(rhs)); expr s_rhs_v, type; std::tie(type, s_rhs_v) = get_type_and_arg_of_neg(s_rhs); auto rhs_v = get_type_and_arg_of_neg(rhs).second; auto prod_pr = mk_norm(mk_mul(type, s_lhs, s_rhs_v)); return mk_app({mk_const(get_norm_num_pos_mul_neg_helper_name()), type, mk_ring(type), s_lhs, s_rhs_v, rhs_v, prod_pr.second}); } // returns a proof that s_lhs + s_rhs = rhs, where all are nonneg normalized numerals expr norm_num_context::mk_norm_eq_pos_mul_pos(expr & s_lhs, expr & s_rhs, expr & DEBUG_CODE(rhs)) { lean_assert(!is_neg_app(s_lhs)); lean_assert(!is_neg_app(s_rhs)); lean_assert(!is_neg_app(rhs)); auto p = mk_norm_mul(s_lhs, s_rhs); lean_assert(to_num(rhs) == to_num(p.first)); return p.second; } expr norm_num_context::from_pos_num(mpz const & n, expr const & type) { lean_assert(n > 0); if (n == 1) return mk_app({mk_const(get_one_name()), type, mk_has_one(type)}); if (n % mpz(2) == 1) return mk_app({mk_const(get_bit1_name()), type, mk_has_one(type), mk_has_add(type), from_pos_num(n/2, type)}); else return mk_app({mk_const(get_bit0_name()), type, mk_has_add(type), from_pos_num(n/2, type)}); } expr norm_num_context::from_num(mpz const & n, expr const & type) { expr r; lean_assert(n >= 0); if (n == 0) r = mk_app(mk_const(get_zero_name()), type, mk_has_zero(type)); else r = from_pos_num(n, type); lean_assert(*to_num(r) == n); return r; } // assumes q >= 0 expr norm_num_context::from_mpq(mpq const & q, expr const & type) { mpz numer = q.get_numerator(); mpz denom = q.get_denominator(); lean_assert(numer >= 0 && denom >= 0); if (denom == 1) { return from_num(numer, type); } else { return mk_div(type, from_num(numer, type), from_num(denom, type)); } } expr norm_num_context::mk_div(expr const & type, expr const & e1, expr const & e2) { auto has_div = mk_has_div(type); return mk_app({mk_const(get_div_name()), type, has_div, e1, e2}); } expr norm_num_context::mk_neg(expr const & type, expr const & e) { auto has_neg = mk_has_neg(type); return mk_app({mk_const(get_neg_name()), type, has_neg, e}); } expr norm_num_context::mk_add(expr const & type, expr const & e1, expr const & e2) { auto has_add = mk_has_add(type); return mk_app({mk_const(get_add_name()), type, has_add, e1, e2}); } expr norm_num_context::mk_mul(expr const & type, expr const & e1, expr const & e2) { auto has_mul = mk_has_mul(type); return mk_app({mk_const(get_mul_name()), type, has_mul, e1, e2}); } // s_lhs is div. returns proof that s_lhs + s_rhs = rhs expr norm_num_context::mk_norm_div_add(expr & s_lhs, expr & s_rhs, expr & rhs) { buffer s_lhs_args; get_app_args(s_lhs, s_lhs_args); expr type = s_lhs_args[0]; expr num = s_lhs_args[2], den = s_lhs_args[3]; expr new_lhs = mk_add(type, num, mk_mul(type, s_rhs, den)); auto npr_l = mk_norm(new_lhs); auto npr_r = mk_norm(mk_mul(type, rhs, den)); lean_assert(to_mpq(npr_l.first) == to_mpq(npr_r.first)); expr den_neq_zero = mk_nonzero_prf(den); return mk_app({mk_const(get_norm_num_div_add_helper_name()), type, mk_field(type), num, den, s_rhs, rhs, npr_l.first, den_neq_zero, npr_l.second, npr_r.second}); } // s_rhs is div. returns proof that s_lhs + s_rhs = rhs expr norm_num_context::mk_norm_add_div(expr & s_lhs, expr & s_rhs, expr & rhs) { buffer s_rhs_args; get_app_args(s_rhs, s_rhs_args); expr type = s_rhs_args[0]; expr num = s_rhs_args[2], den = s_rhs_args[3]; expr new_lhs = mk_add(type, mk_mul(type, den, s_lhs), num); auto npr_l = mk_norm(new_lhs); auto npr_r = mk_norm(mk_mul(type, den, rhs)); lean_assert(to_mpq(npr_l.first) == to_mpq(npr_r.first)); expr den_neq_zero = mk_nonzero_prf(den); return mk_app({mk_const(get_norm_num_add_div_helper_name()), type, mk_field(type), num, den, s_lhs, rhs, npr_l.first, den_neq_zero, npr_l.second, npr_r.second}); } // if e is a numeral or a negation of a numeral or division, returns proof that e != 0 expr norm_num_context::mk_nonzero_prf(expr const & e) { buffer args; expr f = get_app_args(e, args); if (const_name(f) == get_neg_name()) { return mk_app({mk_const(get_norm_num_nonzero_of_neg_helper_name()), args[0], mk_lin_ord_ring(args[0]), args[2], mk_nonzero_prf(args[2])}); } else if (const_name(f) == get_div_name()) { expr num_pr = mk_nonzero_prf(args[2]), den_pr = mk_nonzero_prf(args[3]); return mk_app({mk_const(get_norm_num_nonzero_of_div_helper_name()), args[0], mk_field(args[0]), args[2], args[3], num_pr, den_pr}); } else { return mk_app({mk_const(get_norm_num_nonzero_of_pos_helper_name()), args[0], mk_lin_ord_semiring(args[0]), e, mk_pos_prf(e)}); } } // if e is a numeral, makes a proof that e > 0 expr norm_num_context::mk_pos_prf(expr const & e) { buffer args; get_app_args(e, args); expr type = args[0]; expr prf; if (is_bit0(e)) { prf = mk_pos_prf(args[2]); return mk_app({mk_const(get_norm_num_pos_bit0_helper_name()), type, mk_lin_ord_semiring(type), args[2], prf}); } else if (is_bit1(e)) { prf = mk_nonneg_prf(args[3]); return mk_app({mk_const(get_norm_num_pos_bit1_helper_name()), type, mk_lin_ord_semiring(type), args[3], prf}); } else if (is_one(e)) { return mk_app({mk_const(get_zero_lt_one_name()), type, mk_lin_ord_semiring(type)}); } else { throw exception("mk_pos_proof called on zero or non_numeral"); } } expr norm_num_context::mk_nonneg_prf(expr const & e) { buffer args; get_app_args(e, args); expr type = args[0]; expr prf; if (is_bit0(e)) { prf = mk_nonneg_prf(args[2]); return mk_app({mk_const(get_norm_num_nonneg_bit0_helper_name()), type, mk_lin_ord_semiring(type), args[2], prf}); } else if (is_bit1(e)) { prf = mk_nonneg_prf(args[3]); return mk_app({mk_const(get_norm_num_nonneg_bit1_helper_name()), type, mk_lin_ord_semiring(type), args[3], prf}); } else if (is_one(e)) { return mk_app({mk_const(get_zero_le_one_name()), type, mk_lin_ord_ring(type)}); } else if (is_zero(e)) { return mk_app({mk_const(get_le_refl_name()), type, mk_wk_order(type), mk_app({mk_const(get_zero_name()), type, mk_has_zero(type)})}); } else { throw exception("mk_nonneg_proof called on zero or non_numeral"); } } // s_lhs is div. returns proof that s_lhs * s_rhs = rhs expr norm_num_context::mk_norm_div_mul(expr & s_lhs, expr & s_rhs, expr & rhs) { buffer args; get_app_args(s_lhs, args); expr type = args[0]; expr new_num = mk_mul(type, args[2], s_rhs); auto prf = mk_norm(mk_div(type, new_num, args[3])); lean_assert(to_mpq(prf.first) == to_mpq(rhs)); expr den_ne_zero = mk_nonzero_prf(args[3]); return mk_app({mk_const(get_norm_num_div_mul_helper_name()), type, mk_field(type), args[2], args[3], s_rhs, rhs, den_ne_zero, prf.second}); } expr norm_num_context::mk_norm_mul_div(expr & s_lhs, expr & s_rhs, expr & rhs) { buffer args; get_app_args(s_rhs, args); expr type = args[0]; expr new_num = mk_mul(type, s_lhs, args[2]); auto prf = mk_norm(mk_div(type, new_num, args[3])); lean_assert(to_mpq(prf.first) == to_mpq(rhs)); expr den_ne_zero = mk_nonzero_prf(args[3]); return mk_app({mk_const(get_norm_num_mul_div_helper_name()), type, mk_field(type), s_lhs, args[2], args[3], rhs, den_ne_zero, prf.second}); } pair norm_num_context::mk_norm(expr const & e) { buffer args; expr f = get_app_args(e, args); if (!is_constant(f) || args.size() == 0) { throw exception("malformed argument to mk_norm_expr"); } m_lvls = const_levels(f); expr type = args[0]; if (is_numeral(e)) { expr prf = mk_app({mk_const(get_eq_refl_name()), type, e}); return pair(e, prf); } mpq val = mpq_of_expr(e); expr nval; // e = nval if (val >= 0) { nval = from_mpq(val, type); } else { nval = mk_neg(type, from_mpq(neg(val), type)); } if (const_name(f) == get_add_name() && args.size() == 4) { expr prf; auto lhs_p = mk_norm(args[2]); auto rhs_p = mk_norm(args[3]); if (is_neg_app(lhs_p.first)) { if (is_neg_app(rhs_p.first)) { prf = mk_norm_eq_neg_add_neg(lhs_p.first, rhs_p.first, nval); } else { prf = mk_norm_eq_neg_add_pos(lhs_p.first, rhs_p.first, nval); } } else { if (is_neg_app(rhs_p.first)) { prf = mk_norm_eq_pos_add_neg(lhs_p.first, rhs_p.first, nval); } else { if (is_div(lhs_p.first)) { prf = mk_norm_div_add(lhs_p.first, rhs_p.first, nval); } else if (is_div(rhs_p.first)) { prf = mk_norm_add_div(lhs_p.first, rhs_p.first, nval); } else { prf = mk_norm_eq_pos_add_pos(lhs_p.first, rhs_p.first, nval); } } } expr rprf = mk_app({mk_const(get_norm_num_subst_into_sum_name()), type, mk_has_add(type), args[2], args[3], lhs_p.first, rhs_p.first, nval, lhs_p.second, rhs_p.second, prf}); return pair(nval, rprf); } else if (const_name(f) == get_sub_name() && args.size() == 4) { expr sum = mk_add(args[0], args[2], mk_neg(args[0], args[3])); auto anprf = mk_norm(sum); expr rprf = mk_app({mk_const(get_norm_num_subst_into_subtr_name()), type, mk_add_group(type), args[2], args[3], anprf.first, anprf.second}); return pair(nval, rprf); } else if (const_name(f) == get_neg_name() && args.size() == 3) { auto prf = mk_norm(args[2]); lean_assert(mpq_of_expr(prf.first) == neg(val)); if (is_zero(prf.first)) { expr rprf = mk_app({mk_const(get_norm_num_neg_zero_helper_name()), type, mk_add_group(type), args[2], prf.second}); return pair(prf.first, rprf); } if (is_neg_app(nval)) { buffer nval_args; get_app_args(nval, nval_args); expr rprf = mk_cong(mk_app(f, args[0], args[1]), type, args[2], nval_args[2], prf.second); return pair(nval, rprf); } else { expr rprf = mk_app({mk_const(get_norm_num_neg_neg_helper_name()), type, mk_add_group(type), args[2], nval, prf.second}); return pair(nval, rprf); } } else if (const_name(f) == get_mul_name() && args.size() == 4) { auto lhs_p = mk_norm(args[2]); auto rhs_p = mk_norm(args[3]); expr prf; if (is_div(lhs_p.first)) { prf = mk_norm_div_mul(lhs_p.first, rhs_p.first, nval); } else if (is_div(rhs_p.first)) { prf = mk_norm_mul_div(lhs_p.first, rhs_p.first, nval); } else if (is_zero(lhs_p.first) || is_zero(rhs_p.first)) { prf = mk_norm_mul(lhs_p.first, rhs_p.first).second; } else if (is_neg_app(lhs_p.first)) { if (is_neg_app(rhs_p.first)) { prf = mk_norm_eq_neg_mul_neg(lhs_p.first, rhs_p.first, nval); } else { // bad args passing here prf = mk_norm_eq_neg_mul_pos(lhs_p.first, rhs_p.first, nval); } } else { if (is_neg_app(rhs_p.first)) { prf = mk_norm_eq_pos_mul_neg(lhs_p.first, rhs_p.first, nval); } else { prf = mk_norm_eq_pos_mul_pos(lhs_p.first, rhs_p.first, nval); } } expr rprf = mk_app({mk_const(get_norm_num_subst_into_prod_name()), type, mk_has_mul(args[0]), args[2], args[3], lhs_p.first, rhs_p.first, nval, lhs_p.second, rhs_p.second, prf}); return pair(nval, rprf); } else if (const_name(f) == get_div_name() && args.size() == 4) { auto lhs_p = mk_norm(args[2]); auto rhs_p = mk_norm(args[3]); expr prf; if (is_div(nval)) { buffer nval_args; get_app_args(nval, nval_args); expr nval_num = nval_args[2], nval_den = nval_args[3]; auto lhs_mul = mk_norm(mk_mul(type, lhs_p.first, nval_den)); auto rhs_mul = mk_norm(mk_mul(type, nval_num, rhs_p.first)); expr den_nonzero = mk_nonzero_prf(rhs_p.first); expr nval_den_nonzero = mk_nonzero_prf(nval_den); prf = mk_app({mk_const(get_norm_num_div_eq_div_helper_name()), type, mk_field(type), lhs_p.first, rhs_p.first, nval_num, nval_den, lhs_mul.first, lhs_mul.second, rhs_mul.second, den_nonzero, nval_den_nonzero}); } else { auto prod = mk_norm(mk_mul(type, nval, rhs_p.first)); auto val1 = to_mpq(prod.first), val2 = to_mpq(lhs_p.first); if (val1 && val2) { lean_assert(*val1 == *val2); } expr den_nonzero = mk_nonzero_prf(rhs_p.first); prf = mk_app({mk_const(get_norm_num_div_helper_name()), type, mk_field(type), lhs_p.first, rhs_p.first, nval, den_nonzero, prod.second}); } expr rprf = mk_app({mk_const(get_norm_num_subst_into_div_name()), type, mk_has_div(type), lhs_p.first, rhs_p.first, args[2], args[3], nval, prf, lhs_p.second, rhs_p.second}); return pair(nval, rprf); } else if (const_name(f) == get_bit0_name() && args.size() == 3) { lean_assert(is_bit0(nval)); buffer nval_args; get_app_args(nval, nval_args); auto prf = mk_norm(args[2]); auto rprf = mk_cong(mk_app(f, args[0], args[1]), type, args[2], nval_args[2], prf.second); return pair(nval, rprf); } else if (const_name(f) == get_bit1_name() && args.size() == 4) { lean_assert(is_bit1(nval)); buffer nval_args; get_app_args(nval, nval_args); auto prf = mk_norm(args[3]); auto rprf = mk_cong(mk_app(f, args[0], args[1], args[2]), type, args[3], nval_args[3], prf.second); return pair(nval, rprf); } else if ((const_name(f) == get_zero_name() || const_name(f) == get_one_name()) && args.size() == 2) { return pair(e, mk_app({mk_const(get_eq_refl_name()), args[0], e})); } else { throw exception("mk_norm found unrecognized combo "); } } }