2014-11-14 19:22:51 +00:00
|
|
|
|
/-
|
|
|
|
|
Copyright (c) 2014 Jeremy Avigad. All rights reserved.
|
|
|
|
|
Released under Apache 2.0 license as described in the file LICENSE.
|
|
|
|
|
Authors: Jeremy Avigad, Leonardo de Moura
|
|
|
|
|
|
|
|
|
|
Structures with multiplicative and additive components, including semirings, rings, and fields.
|
|
|
|
|
The development is modeled after Isabelle's library.
|
|
|
|
|
-/
|
|
|
|
|
|
2014-12-12 21:20:27 +00:00
|
|
|
|
import logic.eq logic.connectives data.unit data.sigma data.prod
|
2015-07-06 14:29:56 +00:00
|
|
|
|
import algebra.binary algebra.group
|
2014-11-14 19:22:51 +00:00
|
|
|
|
open eq eq.ops
|
|
|
|
|
|
|
|
|
|
variable {A : Type}
|
|
|
|
|
|
|
|
|
|
/- auxiliary classes -/
|
|
|
|
|
|
|
|
|
|
structure distrib [class] (A : Type) extends has_mul A, has_add A :=
|
2014-11-30 13:31:34 +00:00
|
|
|
|
(left_distrib : ∀a b c, mul a (add b c) = add (mul a b) (mul a c))
|
|
|
|
|
(right_distrib : ∀a b c, mul (add a b) c = add (mul a c) (mul b c))
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
2015-12-13 19:15:10 +00:00
|
|
|
|
theorem left_distrib [distrib A] (a b c : A) : a * (b + c) = a * b + a * c :=
|
2014-11-30 13:31:34 +00:00
|
|
|
|
!distrib.left_distrib
|
|
|
|
|
|
2015-12-13 19:15:10 +00:00
|
|
|
|
theorem right_distrib [distrib A] (a b c : A) : (a + b) * c = a * c + b * c :=
|
2014-11-30 13:31:34 +00:00
|
|
|
|
!distrib.right_distrib
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
2014-12-23 22:34:16 +00:00
|
|
|
|
structure mul_zero_class [class] (A : Type) extends has_mul A, has_zero A :=
|
|
|
|
|
(zero_mul : ∀a, mul zero a = zero)
|
|
|
|
|
(mul_zero : ∀a, mul a zero = zero)
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
2015-12-29 19:16:18 +00:00
|
|
|
|
theorem zero_mul [simp] [mul_zero_class A] (a : A) : 0 * a = 0 := !mul_zero_class.zero_mul
|
|
|
|
|
theorem mul_zero [simp] [mul_zero_class A] (a : A) : a * 0 = 0 := !mul_zero_class.mul_zero
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
|
|
|
|
structure zero_ne_one_class [class] (A : Type) extends has_zero A, has_one A :=
|
|
|
|
|
(zero_ne_one : zero ≠ one)
|
|
|
|
|
|
2015-05-18 22:45:23 +00:00
|
|
|
|
theorem zero_ne_one [s: zero_ne_one_class A] : 0 ≠ (1:A) := @zero_ne_one_class.zero_ne_one A s
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
|
|
|
|
/- semiring -/
|
2014-12-12 23:22:19 +00:00
|
|
|
|
|
2014-12-23 22:34:16 +00:00
|
|
|
|
structure semiring [class] (A : Type) extends add_comm_monoid A, monoid A, distrib A,
|
2015-03-25 21:08:36 +00:00
|
|
|
|
mul_zero_class A
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
2014-12-22 19:21:22 +00:00
|
|
|
|
section semiring
|
|
|
|
|
variables [s : semiring A] (a b c : A)
|
|
|
|
|
include s
|
|
|
|
|
|
2015-10-12 03:29:31 +00:00
|
|
|
|
theorem one_add_one_eq_two : 1 + 1 = (2:A) :=
|
|
|
|
|
by unfold bit0
|
|
|
|
|
|
2014-12-22 19:21:22 +00:00
|
|
|
|
theorem ne_zero_of_mul_ne_zero_right {a b : A} (H : a * b ≠ 0) : a ≠ 0 :=
|
2015-07-21 16:10:56 +00:00
|
|
|
|
suppose a = 0,
|
2016-02-04 04:07:12 +00:00
|
|
|
|
have a * b = 0, by rewrite [this, zero_mul],
|
2015-07-21 16:10:56 +00:00
|
|
|
|
H this
|
2014-12-22 19:21:22 +00:00
|
|
|
|
|
|
|
|
|
theorem ne_zero_of_mul_ne_zero_left {a b : A} (H : a * b ≠ 0) : b ≠ 0 :=
|
2015-07-21 16:10:56 +00:00
|
|
|
|
suppose b = 0,
|
2016-02-04 04:07:12 +00:00
|
|
|
|
have a * b = 0, by rewrite [this, mul_zero],
|
2015-07-21 16:10:56 +00:00
|
|
|
|
H this
|
2015-06-16 06:54:23 +00:00
|
|
|
|
|
2015-12-29 19:16:18 +00:00
|
|
|
|
local attribute right_distrib [simp]
|
|
|
|
|
|
2015-06-16 06:54:23 +00:00
|
|
|
|
theorem distrib_three_right (a b c d : A) : (a + b + c) * d = a * d + b * d + c * d :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
by simp
|
2014-12-22 19:21:22 +00:00
|
|
|
|
end semiring
|
|
|
|
|
|
|
|
|
|
/- comm semiring -/
|
|
|
|
|
|
2015-05-16 12:26:59 +00:00
|
|
|
|
structure comm_semiring [class] (A : Type) extends semiring A, comm_monoid A
|
2014-12-26 21:25:05 +00:00
|
|
|
|
-- TODO: we could also define a cancelative comm_semiring, i.e. satisfying
|
|
|
|
|
-- c ≠ 0 → c * a = c * b → a = b.
|
2014-12-12 23:22:19 +00:00
|
|
|
|
|
2014-12-22 17:21:53 +00:00
|
|
|
|
section comm_semiring
|
|
|
|
|
variables [s : comm_semiring A] (a b c : A)
|
|
|
|
|
include s
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
2015-12-06 07:27:46 +00:00
|
|
|
|
protected definition algebra.dvd (a b : A) : Prop := ∃c, b = a * c
|
2015-10-07 23:44:47 +00:00
|
|
|
|
|
2016-02-25 20:26:20 +00:00
|
|
|
|
definition comm_semiring_has_dvd [instance] [priority algebra.prio] : has_dvd A :=
|
2015-10-07 23:44:47 +00:00
|
|
|
|
has_dvd.mk algebra.dvd
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
2015-04-06 16:24:09 +00:00
|
|
|
|
theorem dvd.intro {a b c : A} (H : a * c = b) : a ∣ b :=
|
2015-02-01 02:53:47 +00:00
|
|
|
|
exists.intro _ H⁻¹
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
2015-08-10 01:23:44 +00:00
|
|
|
|
theorem dvd_of_mul_right_eq {a b c : A} (H : a * c = b) : a ∣ b := dvd.intro H
|
|
|
|
|
|
2015-04-06 16:24:09 +00:00
|
|
|
|
theorem dvd.intro_left {a b c : A} (H : c * a = b) : a ∣ b :=
|
2016-02-04 04:07:12 +00:00
|
|
|
|
dvd.intro (by rewrite mul.comm at H; exact H)
|
2014-12-26 21:25:05 +00:00
|
|
|
|
|
2015-08-10 01:23:44 +00:00
|
|
|
|
theorem dvd_of_mul_left_eq {a b c : A} (H : c * a = b) : a ∣ b := dvd.intro_left H
|
|
|
|
|
|
2015-04-06 16:24:09 +00:00
|
|
|
|
theorem exists_eq_mul_right_of_dvd {a b : A} (H : a ∣ b) : ∃c, b = a * c := H
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
2015-04-06 16:24:09 +00:00
|
|
|
|
theorem dvd.elim {P : Prop} {a b : A} (H₁ : a ∣ b) (H₂ : ∀c, b = a * c → P) : P :=
|
2014-12-22 17:21:53 +00:00
|
|
|
|
exists.elim H₁ H₂
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
2015-04-06 16:24:09 +00:00
|
|
|
|
theorem exists_eq_mul_left_of_dvd {a b : A} (H : a ∣ b) : ∃c, b = c * a :=
|
2015-02-01 02:53:47 +00:00
|
|
|
|
dvd.elim H (take c, assume H1 : b = a * c, exists.intro c (H1 ⬝ !mul.comm))
|
|
|
|
|
|
2015-04-06 16:24:09 +00:00
|
|
|
|
theorem dvd.elim_left {P : Prop} {a b : A} (H₁ : a ∣ b) (H₂ : ∀c, b = c * a → P) : P :=
|
2015-02-01 02:53:47 +00:00
|
|
|
|
exists.elim (exists_eq_mul_left_of_dvd H₁) (take c, assume H₃ : b = c * a, H₂ c H₃)
|
|
|
|
|
|
2015-12-29 19:16:18 +00:00
|
|
|
|
theorem dvd.refl [simp] : a ∣ a :=
|
|
|
|
|
dvd.intro !mul_one
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
2015-04-06 16:24:09 +00:00
|
|
|
|
theorem dvd.trans {a b c : A} (H₁ : a ∣ b) (H₂ : b ∣ c) : a ∣ c :=
|
2014-11-30 13:31:34 +00:00
|
|
|
|
dvd.elim H₁
|
2015-02-01 02:53:47 +00:00
|
|
|
|
(take d, assume H₃ : b = a * d,
|
2014-11-30 13:31:34 +00:00
|
|
|
|
dvd.elim H₂
|
2015-02-01 02:53:47 +00:00
|
|
|
|
(take e, assume H₄ : c = b * e,
|
|
|
|
|
dvd.intro
|
2015-03-28 00:26:06 +00:00
|
|
|
|
(show a * (d * e) = c, by rewrite [-mul.assoc, -H₃, H₄])))
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
2015-04-06 16:24:09 +00:00
|
|
|
|
theorem eq_zero_of_zero_dvd {a : A} (H : 0 ∣ a) : a = 0 :=
|
2015-02-01 02:53:47 +00:00
|
|
|
|
dvd.elim H (take c, assume H' : a = 0 * c, H' ⬝ !zero_mul)
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
2015-12-29 19:16:18 +00:00
|
|
|
|
theorem dvd_zero [simp] : a ∣ 0 := dvd.intro !mul_zero
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
2015-12-29 19:16:18 +00:00
|
|
|
|
theorem one_dvd [simp] : 1 ∣ a := dvd.intro !one_mul
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
2015-12-29 19:16:18 +00:00
|
|
|
|
theorem dvd_mul_right [simp] : a ∣ a * b := dvd.intro rfl
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
2015-12-29 19:16:18 +00:00
|
|
|
|
theorem dvd_mul_left [simp] : a ∣ b * a :=
|
|
|
|
|
by simp
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
2015-04-06 16:24:09 +00:00
|
|
|
|
theorem dvd_mul_of_dvd_left {a b : A} (H : a ∣ b) (c : A) : a ∣ b * c :=
|
2014-11-30 13:31:34 +00:00
|
|
|
|
dvd.elim H
|
2014-11-14 19:22:51 +00:00
|
|
|
|
(take d,
|
2015-07-21 16:10:56 +00:00
|
|
|
|
suppose b = a * d,
|
2014-11-30 13:31:34 +00:00
|
|
|
|
dvd.intro
|
2015-12-29 19:16:18 +00:00
|
|
|
|
(show a * (d * c) = b * c, by simp))
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
2015-04-06 16:24:09 +00:00
|
|
|
|
theorem dvd_mul_of_dvd_right {a b : A} (H : a ∣ b) (c : A) : a ∣ c * b :=
|
2014-11-30 13:31:34 +00:00
|
|
|
|
!mul.comm ▸ (dvd_mul_of_dvd_left H _)
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
2015-05-30 12:09:34 +00:00
|
|
|
|
theorem mul_dvd_mul {a b c d : A} (dvd_ab : a ∣ b) (dvd_cd : c ∣ d) : a * c ∣ b * d :=
|
2014-11-30 13:31:34 +00:00
|
|
|
|
dvd.elim dvd_ab
|
2015-07-21 16:10:56 +00:00
|
|
|
|
(take e, suppose b = a * e,
|
2014-11-30 13:31:34 +00:00
|
|
|
|
dvd.elim dvd_cd
|
2015-07-21 16:10:56 +00:00
|
|
|
|
(take f, suppose d = c * f,
|
2014-11-30 13:31:34 +00:00
|
|
|
|
dvd.intro
|
2015-02-06 02:48:02 +00:00
|
|
|
|
(show a * c * (e * f) = b * d,
|
2015-12-29 19:16:18 +00:00
|
|
|
|
by simp)))
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
2015-04-06 16:24:09 +00:00
|
|
|
|
theorem dvd_of_mul_right_dvd {a b c : A} (H : a * b ∣ c) : a ∣ c :=
|
2015-02-01 02:53:47 +00:00
|
|
|
|
dvd.elim H (take d, assume Habdc : c = a * b * d, dvd.intro (!mul.assoc⁻¹ ⬝ Habdc⁻¹))
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
2015-04-06 16:24:09 +00:00
|
|
|
|
theorem dvd_of_mul_left_dvd {a b c : A} (H : a * b ∣ c) : b ∣ c :=
|
2014-11-30 13:31:34 +00:00
|
|
|
|
dvd_of_mul_right_dvd (mul.comm a b ▸ H)
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
2015-04-06 16:24:09 +00:00
|
|
|
|
theorem dvd_add {a b c : A} (Hab : a ∣ b) (Hac : a ∣ c) : a ∣ b + c :=
|
2014-11-30 13:31:34 +00:00
|
|
|
|
dvd.elim Hab
|
2015-07-21 16:10:56 +00:00
|
|
|
|
(take d, suppose b = a * d,
|
2014-11-30 13:31:34 +00:00
|
|
|
|
dvd.elim Hac
|
2015-07-21 16:10:56 +00:00
|
|
|
|
(take e, suppose c = a * e,
|
2015-02-06 02:48:02 +00:00
|
|
|
|
dvd.intro (show a * (d + e) = b + c,
|
2015-07-21 16:10:56 +00:00
|
|
|
|
by rewrite [left_distrib]; substvars)))
|
2014-12-22 17:21:53 +00:00
|
|
|
|
end comm_semiring
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
|
|
|
|
/- ring -/
|
|
|
|
|
|
2015-03-25 21:08:36 +00:00
|
|
|
|
structure ring [class] (A : Type) extends add_comm_group A, monoid A, distrib A
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
2015-12-29 19:16:18 +00:00
|
|
|
|
theorem ring.mul_zero [simp] [ring A] (a : A) : a * 0 = 0 :=
|
2015-07-21 16:10:56 +00:00
|
|
|
|
have a * 0 + 0 = a * 0 + a * 0, from calc
|
2015-12-29 19:16:18 +00:00
|
|
|
|
a * 0 + 0 = a * (0 + 0) : by simp
|
|
|
|
|
... = a * 0 + a * 0 : by rewrite left_distrib,
|
2015-07-21 16:10:56 +00:00
|
|
|
|
show a * 0 = 0, from (add.left_cancel this)⁻¹
|
2015-01-19 21:00:24 +00:00
|
|
|
|
|
2015-12-29 19:16:18 +00:00
|
|
|
|
theorem ring.zero_mul [simp] [ring A] (a : A) : 0 * a = 0 :=
|
2015-07-21 16:10:56 +00:00
|
|
|
|
have 0 * a + 0 = 0 * a + 0 * a, from calc
|
2015-12-29 19:16:18 +00:00
|
|
|
|
0 * a + 0 = (0 + 0) * a : by simp
|
|
|
|
|
... = 0 * a + 0 * a : by rewrite right_distrib,
|
2015-07-21 16:10:56 +00:00
|
|
|
|
show 0 * a = 0, from (add.left_cancel this)⁻¹
|
2015-01-19 21:00:24 +00:00
|
|
|
|
|
2016-02-25 22:30:00 +00:00
|
|
|
|
definition ring.to_semiring [trans_instance] [s : ring A] : semiring A :=
|
2015-01-17 17:38:10 +00:00
|
|
|
|
⦃ semiring, s,
|
2015-01-19 21:00:24 +00:00
|
|
|
|
mul_zero := ring.mul_zero,
|
|
|
|
|
zero_mul := ring.zero_mul ⦄
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
|
|
|
|
section
|
2014-11-17 20:19:46 +00:00
|
|
|
|
variables [s : ring A] (a b c d e : A)
|
2014-11-14 19:22:51 +00:00
|
|
|
|
include s
|
|
|
|
|
|
2014-11-30 13:31:34 +00:00
|
|
|
|
theorem neg_mul_eq_neg_mul : -(a * b) = -a * b :=
|
|
|
|
|
neg_eq_of_add_eq_zero
|
2015-02-09 01:35:01 +00:00
|
|
|
|
begin
|
|
|
|
|
rewrite [-right_distrib, add.right_inv, zero_mul]
|
|
|
|
|
end
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
2014-11-30 13:31:34 +00:00
|
|
|
|
theorem neg_mul_eq_mul_neg : -(a * b) = a * -b :=
|
2015-01-21 20:50:42 +00:00
|
|
|
|
neg_eq_of_add_eq_zero
|
2015-02-09 01:35:01 +00:00
|
|
|
|
begin
|
|
|
|
|
rewrite [-left_distrib, add.right_inv, mul_zero]
|
|
|
|
|
end
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
2015-12-29 19:16:18 +00:00
|
|
|
|
theorem neg_mul_eq_neg_mul_symm [simp] : - a * b = - (a * b) := eq.symm !neg_mul_eq_neg_mul
|
|
|
|
|
theorem mul_neg_eq_neg_mul_symm [simp] : a * - b = - (a * b) := eq.symm !neg_mul_eq_mul_neg
|
2015-12-05 01:57:03 +00:00
|
|
|
|
|
2015-01-21 20:50:42 +00:00
|
|
|
|
theorem neg_mul_neg : -a * -b = a * b :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
by simp
|
2014-11-14 19:22:51 +00:00
|
|
|
|
|
2015-12-29 19:16:18 +00:00
|
|
|
|
theorem neg_mul_comm : -a * b = a * -b :=
|
|
|
|
|
by simp
|
2014-11-17 20:19:46 +00:00
|
|
|
|
|
2015-01-21 20:50:42 +00:00
|
|
|
|
theorem neg_eq_neg_one_mul : -a = -1 * a :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
by simp
|
2015-01-21 20:50:42 +00:00
|
|
|
|
|
2014-11-30 13:31:34 +00:00
|
|
|
|
theorem mul_sub_left_distrib : a * (b - c) = a * b - a * c :=
|
2014-11-17 20:19:46 +00:00
|
|
|
|
calc
|
2014-11-30 13:31:34 +00:00
|
|
|
|
a * (b - c) = a * b + a * -c : left_distrib
|
2015-12-29 19:16:18 +00:00
|
|
|
|
... = a * b - a * c : by simp
|
2014-11-17 20:19:46 +00:00
|
|
|
|
|
2014-11-30 13:31:34 +00:00
|
|
|
|
theorem mul_sub_right_distrib : (a - b) * c = a * c - b * c :=
|
2014-11-17 20:19:46 +00:00
|
|
|
|
calc
|
2014-11-30 13:31:34 +00:00
|
|
|
|
(a - b) * c = a * c + -b * c : right_distrib
|
2015-12-29 19:16:18 +00:00
|
|
|
|
... = a * c - b * c : by simp
|
2014-11-17 20:19:46 +00:00
|
|
|
|
|
|
|
|
|
-- TODO: can calc mode be improved to make this easier?
|
|
|
|
|
-- TODO: there is also the other direction. It will be easier when we
|
|
|
|
|
-- have the simplifier.
|
2014-11-30 13:31:34 +00:00
|
|
|
|
|
|
|
|
|
theorem mul_add_eq_mul_add_iff_sub_mul_add_eq : a * e + c = b * e + d ↔ (a - b) * e + c = d :=
|
2014-11-17 20:19:46 +00:00
|
|
|
|
calc
|
2015-02-09 01:35:01 +00:00
|
|
|
|
a * e + c = b * e + d ↔ a * e + c = d + b * e : by rewrite {b*e+_}add.comm
|
2014-11-30 13:31:34 +00:00
|
|
|
|
... ↔ a * e + c - b * e = d : iff.symm !sub_eq_iff_eq_add
|
2015-02-09 01:35:01 +00:00
|
|
|
|
... ↔ a * e - b * e + c = d : by rewrite sub_add_eq_add_sub
|
|
|
|
|
... ↔ (a - b) * e + c = d : by rewrite mul_sub_right_distrib
|
2015-03-27 17:11:23 +00:00
|
|
|
|
|
2015-08-05 21:03:46 +00:00
|
|
|
|
theorem mul_add_eq_mul_add_of_sub_mul_add_eq : (a - b) * e + c = d → a * e + c = b * e + d :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
iff.mpr !mul_add_eq_mul_add_iff_sub_mul_add_eq
|
2015-08-05 21:03:46 +00:00
|
|
|
|
|
|
|
|
|
theorem sub_mul_add_eq_of_mul_add_eq_mul_add : a * e + c = b * e + d → (a - b) * e + c = d :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
iff.mp !mul_add_eq_mul_add_iff_sub_mul_add_eq
|
2015-08-05 21:03:46 +00:00
|
|
|
|
|
2015-03-27 17:11:23 +00:00
|
|
|
|
theorem mul_neg_one_eq_neg : a * (-1) = -a :=
|
2015-07-21 16:10:56 +00:00
|
|
|
|
have a + a * -1 = 0, from calc
|
2015-12-29 19:16:18 +00:00
|
|
|
|
a + a * -1 = a * 1 + a * -1 : by simp
|
2015-03-27 17:11:23 +00:00
|
|
|
|
... = a * (1 + -1) : left_distrib
|
2015-12-29 19:16:18 +00:00
|
|
|
|
... = 0 : by simp,
|
2015-07-21 16:10:56 +00:00
|
|
|
|
symm (neg_eq_of_add_eq_zero this)
|
2015-03-27 17:11:23 +00:00
|
|
|
|
|
2015-05-23 04:05:06 +00:00
|
|
|
|
theorem ne_zero_and_ne_zero_of_mul_ne_zero {a b : A} (H : a * b ≠ 0) : a ≠ 0 ∧ b ≠ 0 :=
|
2015-07-21 16:10:56 +00:00
|
|
|
|
have a ≠ 0, from
|
|
|
|
|
(suppose a = 0,
|
|
|
|
|
have a * b = 0, by rewrite [this, zero_mul],
|
|
|
|
|
absurd this H),
|
|
|
|
|
have b ≠ 0, from
|
|
|
|
|
(suppose b = 0,
|
|
|
|
|
have a * b = 0, by rewrite [this, mul_zero],
|
|
|
|
|
absurd this H),
|
|
|
|
|
and.intro `a ≠ 0` `b ≠ 0`
|
2014-11-14 19:22:51 +00:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
structure comm_ring [class] (A : Type) extends ring A, comm_semigroup A
|
|
|
|
|
|
2016-02-25 22:30:00 +00:00
|
|
|
|
definition comm_ring.to_comm_semiring [trans_instance] [s : comm_ring A] : comm_semiring A :=
|
2015-01-17 17:38:10 +00:00
|
|
|
|
⦃ comm_semiring, s,
|
2015-01-17 01:42:30 +00:00
|
|
|
|
mul_zero := mul_zero,
|
2015-01-17 17:38:10 +00:00
|
|
|
|
zero_mul := zero_mul ⦄
|
2014-11-17 20:19:46 +00:00
|
|
|
|
|
|
|
|
|
section
|
|
|
|
|
variables [s : comm_ring A] (a b c d e : A)
|
|
|
|
|
include s
|
|
|
|
|
|
2015-12-29 19:16:18 +00:00
|
|
|
|
local attribute left_distrib right_distrib [simp]
|
|
|
|
|
|
2015-03-30 18:55:16 +00:00
|
|
|
|
theorem mul_self_sub_mul_self_eq : a * a - b * b = (a + b) * (a - b) :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
by simp
|
2014-11-17 20:19:46 +00:00
|
|
|
|
|
2014-11-30 13:31:34 +00:00
|
|
|
|
theorem mul_self_sub_one_eq : a * a - 1 = (a + 1) * (a - 1) :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
by simp
|
2014-11-17 20:19:46 +00:00
|
|
|
|
|
2016-01-31 00:34:59 +00:00
|
|
|
|
theorem add_mul_self_eq : (a + b) * (a + b) = a*a + 2*a*b + b*b :=
|
|
|
|
|
calc (a + b)*(a + b) = a*a + (1+1)*a*b + b*b : by simp
|
|
|
|
|
... = a*a + 2*a*b + b*b : by rewrite one_add_one_eq_two
|
|
|
|
|
|
2015-04-06 16:24:09 +00:00
|
|
|
|
theorem dvd_neg_iff_dvd : (a ∣ -b) ↔ (a ∣ b) :=
|
2014-11-17 20:19:46 +00:00
|
|
|
|
iff.intro
|
2015-07-21 16:10:56 +00:00
|
|
|
|
(suppose a ∣ -b,
|
|
|
|
|
dvd.elim this
|
|
|
|
|
(take c, suppose -b = a * c,
|
2014-11-30 13:31:34 +00:00
|
|
|
|
dvd.intro
|
2015-02-09 01:35:01 +00:00
|
|
|
|
(show a * -c = b,
|
2015-07-21 16:10:56 +00:00
|
|
|
|
by rewrite [-neg_mul_eq_mul_neg, -this, neg_neg])))
|
|
|
|
|
(suppose a ∣ b,
|
|
|
|
|
dvd.elim this
|
|
|
|
|
(take c, suppose b = a * c,
|
2014-11-30 13:31:34 +00:00
|
|
|
|
dvd.intro
|
2015-02-09 01:35:01 +00:00
|
|
|
|
(show a * -c = -b,
|
2015-07-21 16:10:56 +00:00
|
|
|
|
by rewrite [-neg_mul_eq_mul_neg, -this])))
|
2014-11-17 20:19:46 +00:00
|
|
|
|
|
2015-08-05 21:03:46 +00:00
|
|
|
|
theorem dvd_neg_of_dvd : (a ∣ b) → (a ∣ -b) :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
iff.mpr !dvd_neg_iff_dvd
|
2015-08-05 21:03:46 +00:00
|
|
|
|
|
|
|
|
|
theorem dvd_of_dvd_neg : (a ∣ -b) → (a ∣ b) :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
iff.mp !dvd_neg_iff_dvd
|
2015-08-05 21:03:46 +00:00
|
|
|
|
|
2015-04-06 16:24:09 +00:00
|
|
|
|
theorem neg_dvd_iff_dvd : (-a ∣ b) ↔ (a ∣ b) :=
|
2014-11-17 20:19:46 +00:00
|
|
|
|
iff.intro
|
2015-07-21 16:10:56 +00:00
|
|
|
|
(suppose -a ∣ b,
|
|
|
|
|
dvd.elim this
|
|
|
|
|
(take c, suppose b = -a * c,
|
2014-11-30 13:31:34 +00:00
|
|
|
|
dvd.intro
|
2015-07-21 16:10:56 +00:00
|
|
|
|
(show a * -c = b, by rewrite [-neg_mul_comm, this])))
|
|
|
|
|
(suppose a ∣ b,
|
|
|
|
|
dvd.elim this
|
|
|
|
|
(take c, suppose b = a * c,
|
2014-11-30 13:31:34 +00:00
|
|
|
|
dvd.intro
|
2015-07-21 16:10:56 +00:00
|
|
|
|
(show -a * -c = b, by rewrite [neg_mul_neg, this])))
|
2014-11-17 20:19:46 +00:00
|
|
|
|
|
2015-08-05 21:03:46 +00:00
|
|
|
|
theorem neg_dvd_of_dvd : (a ∣ b) → (-a ∣ b) :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
iff.mpr !neg_dvd_iff_dvd
|
2015-08-05 21:03:46 +00:00
|
|
|
|
|
|
|
|
|
theorem dvd_of_neg_dvd : (-a ∣ b) → (a ∣ b) :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
iff.mp !neg_dvd_iff_dvd
|
2015-08-05 21:03:46 +00:00
|
|
|
|
|
2015-04-06 16:24:09 +00:00
|
|
|
|
theorem dvd_sub (H₁ : (a ∣ b)) (H₂ : (a ∣ c)) : (a ∣ b - c) :=
|
2015-08-05 21:03:46 +00:00
|
|
|
|
dvd_add H₁ (!dvd_neg_of_dvd H₂)
|
2014-11-17 20:19:46 +00:00
|
|
|
|
end
|
|
|
|
|
|
2014-11-26 23:08:23 +00:00
|
|
|
|
/- integral domains -/
|
|
|
|
|
|
|
|
|
|
structure no_zero_divisors [class] (A : Type) extends has_mul A, has_zero A :=
|
|
|
|
|
(eq_zero_or_eq_zero_of_mul_eq_zero : ∀a b, mul a b = zero → a = zero ∨ b = zero)
|
|
|
|
|
|
2015-12-13 19:15:10 +00:00
|
|
|
|
theorem eq_zero_or_eq_zero_of_mul_eq_zero {A : Type} [no_zero_divisors A] {a b : A}
|
2014-11-30 13:31:34 +00:00
|
|
|
|
(H : a * b = 0) :
|
2016-02-20 13:27:59 +00:00
|
|
|
|
a = 0 ∨ b = 0 :=
|
|
|
|
|
!no_zero_divisors.eq_zero_or_eq_zero_of_mul_eq_zero H
|
|
|
|
|
|
|
|
|
|
theorem eq_zero_of_mul_self_eq_zero {A : Type} [no_zero_divisors A] {a : A} (H : a * a = 0) :
|
|
|
|
|
a = 0 :=
|
|
|
|
|
or.elim (eq_zero_or_eq_zero_of_mul_eq_zero H) (assume H', H') (assume H', H')
|
2014-11-26 23:08:23 +00:00
|
|
|
|
|
2015-08-17 03:23:03 +00:00
|
|
|
|
structure integral_domain [class] (A : Type) extends comm_ring A, no_zero_divisors A,
|
|
|
|
|
zero_ne_one_class A
|
2014-11-26 23:08:23 +00:00
|
|
|
|
|
|
|
|
|
section
|
|
|
|
|
variables [s : integral_domain A] (a b c d e : A)
|
|
|
|
|
include s
|
|
|
|
|
|
2014-12-22 19:21:22 +00:00
|
|
|
|
theorem mul_ne_zero {a b : A} (H1 : a ≠ 0) (H2 : b ≠ 0) : a * b ≠ 0 :=
|
2015-07-21 16:10:56 +00:00
|
|
|
|
suppose a * b = 0,
|
2015-12-29 19:16:18 +00:00
|
|
|
|
or.elim (eq_zero_or_eq_zero_of_mul_eq_zero this) (assume H3, H1 H3) (assume H4, H2 H4)
|
2014-11-26 23:08:23 +00:00
|
|
|
|
|
2015-06-01 01:39:59 +00:00
|
|
|
|
theorem eq_of_mul_eq_mul_right {a b c : A} (Ha : a ≠ 0) (H : b * a = c * a) : b = c :=
|
2015-07-21 16:10:56 +00:00
|
|
|
|
have b * a - c * a = 0, from iff.mp !eq_iff_sub_eq_zero H,
|
2016-02-29 19:28:20 +00:00
|
|
|
|
have (b - c) * a = 0, by rewrite [mul_sub_right_distrib, this],
|
2015-07-21 16:10:56 +00:00
|
|
|
|
have b - c = 0, from or_resolve_left (eq_zero_or_eq_zero_of_mul_eq_zero this) Ha,
|
|
|
|
|
iff.elim_right !eq_iff_sub_eq_zero this
|
2014-11-26 23:08:23 +00:00
|
|
|
|
|
2015-06-01 01:39:59 +00:00
|
|
|
|
theorem eq_of_mul_eq_mul_left {a b c : A} (Ha : a ≠ 0) (H : a * b = a * c) : b = c :=
|
2015-07-21 16:10:56 +00:00
|
|
|
|
have a * b - a * c = 0, from iff.mp !eq_iff_sub_eq_zero H,
|
2016-02-29 19:28:20 +00:00
|
|
|
|
have a * (b - c) = 0, by rewrite [mul_sub_left_distrib, this],
|
2015-07-21 16:10:56 +00:00
|
|
|
|
have b - c = 0, from or_resolve_right (eq_zero_or_eq_zero_of_mul_eq_zero this) Ha,
|
|
|
|
|
iff.elim_right !eq_iff_sub_eq_zero this
|
2014-11-26 23:08:23 +00:00
|
|
|
|
|
|
|
|
|
-- TODO: do we want the iff versions?
|
|
|
|
|
|
2015-09-21 00:47:30 +00:00
|
|
|
|
theorem eq_zero_of_mul_eq_self_right {a b : A} (H₁ : b ≠ 1) (H₂ : a * b = a) : a = 0 :=
|
|
|
|
|
have b - 1 ≠ 0, from
|
|
|
|
|
suppose b - 1 = 0, H₁ (!zero_add ▸ eq_add_of_sub_eq this),
|
2015-12-29 19:16:18 +00:00
|
|
|
|
have a * b - a = 0, by simp,
|
2015-09-21 00:47:30 +00:00
|
|
|
|
have a * (b - 1) = 0, by+ rewrite [mul_sub_left_distrib, mul_one]; apply this,
|
|
|
|
|
show a = 0, from or_resolve_left (eq_zero_or_eq_zero_of_mul_eq_zero this) `b - 1 ≠ 0`
|
|
|
|
|
|
|
|
|
|
theorem eq_zero_of_mul_eq_self_left {a b : A} (H₁ : b ≠ 1) (H₂ : b * a = a) : a = 0 :=
|
|
|
|
|
eq_zero_of_mul_eq_self_right H₁ (!mul.comm ▸ H₂)
|
|
|
|
|
|
2015-03-30 18:55:16 +00:00
|
|
|
|
theorem mul_self_eq_mul_self_iff (a b : A) : a * a = b * b ↔ a = b ∨ a = -b :=
|
|
|
|
|
iff.intro
|
2015-07-21 16:10:56 +00:00
|
|
|
|
(suppose a * a = b * b,
|
|
|
|
|
have (a - b) * (a + b) = 0,
|
|
|
|
|
by rewrite [mul.comm, -mul_self_sub_mul_self_eq, this, sub_self],
|
2016-02-29 19:47:33 +00:00
|
|
|
|
have a - b = 0 ∨ a + b = 0, from !eq_zero_or_eq_zero_of_mul_eq_zero this,
|
2015-07-21 16:10:56 +00:00
|
|
|
|
or.elim this
|
|
|
|
|
(suppose a - b = 0, or.inl (eq_of_sub_eq_zero this))
|
|
|
|
|
(suppose a + b = 0, or.inr (eq_neg_of_add_eq_zero this)))
|
|
|
|
|
(suppose a = b ∨ a = -b, or.elim this
|
|
|
|
|
(suppose a = b, by rewrite this)
|
|
|
|
|
(suppose a = -b, by rewrite [this, neg_mul_neg]))
|
2015-03-30 18:55:16 +00:00
|
|
|
|
|
|
|
|
|
theorem mul_self_eq_one_iff (a : A) : a * a = 1 ↔ a = 1 ∨ a = -1 :=
|
2016-02-29 19:47:33 +00:00
|
|
|
|
have a * a = 1 * 1 ↔ a = 1 ∨ a = -1, from mul_self_eq_mul_self_iff a 1,
|
2015-07-21 16:10:56 +00:00
|
|
|
|
by rewrite mul_one at this; exact this
|
2014-11-26 23:08:23 +00:00
|
|
|
|
|
|
|
|
|
-- TODO: c - b * c → c = 0 ∨ b = 1 and variants
|
|
|
|
|
|
2015-04-06 16:24:09 +00:00
|
|
|
|
theorem dvd_of_mul_dvd_mul_left {a b c : A} (Ha : a ≠ 0) (Hdvd : (a * b ∣ a * c)) : (b ∣ c) :=
|
2014-11-30 13:31:34 +00:00
|
|
|
|
dvd.elim Hdvd
|
2014-11-26 23:08:23 +00:00
|
|
|
|
(take d,
|
2015-07-21 16:10:56 +00:00
|
|
|
|
suppose a * c = a * b * d,
|
|
|
|
|
have b * d = c, from eq_of_mul_eq_mul_left Ha (mul.assoc a b d ▸ this⁻¹),
|
|
|
|
|
dvd.intro this)
|
2014-11-26 23:08:23 +00:00
|
|
|
|
|
2015-04-06 16:24:09 +00:00
|
|
|
|
theorem dvd_of_mul_dvd_mul_right {a b c : A} (Ha : a ≠ 0) (Hdvd : (b * a ∣ c * a)) : (b ∣ c) :=
|
2014-11-30 13:31:34 +00:00
|
|
|
|
dvd.elim Hdvd
|
2014-11-26 23:08:23 +00:00
|
|
|
|
(take d,
|
2015-07-21 16:10:56 +00:00
|
|
|
|
suppose c * a = b * a * d,
|
|
|
|
|
have b * d * a = c * a, from by rewrite [mul.right_comm, -this],
|
|
|
|
|
have b * d = c, from eq_of_mul_eq_mul_right Ha this,
|
|
|
|
|
dvd.intro this)
|
2014-11-26 23:08:23 +00:00
|
|
|
|
end
|
2014-11-17 20:19:46 +00:00
|
|
|
|
|
2015-12-05 03:13:49 +00:00
|
|
|
|
namespace norm_num
|
|
|
|
|
|
2015-12-29 19:16:18 +00:00
|
|
|
|
local attribute bit0 bit1 add1 [reducible]
|
|
|
|
|
local attribute right_distrib left_distrib [simp]
|
|
|
|
|
|
2015-12-13 19:15:10 +00:00
|
|
|
|
theorem mul_zero [mul_zero_class A] (a : A) : a * zero = zero :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
by simp
|
2015-12-05 03:13:49 +00:00
|
|
|
|
|
2015-12-13 19:15:10 +00:00
|
|
|
|
theorem zero_mul [mul_zero_class A] (a : A) : zero * a = zero :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
by simp
|
2015-12-05 03:13:49 +00:00
|
|
|
|
|
2015-12-13 19:15:10 +00:00
|
|
|
|
theorem mul_one [monoid A] (a : A) : a * one = a :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
by simp
|
2015-12-05 03:13:49 +00:00
|
|
|
|
|
2015-12-13 19:15:10 +00:00
|
|
|
|
theorem mul_bit0 [distrib A] (a b : A) : a * (bit0 b) = bit0 (a * b) :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
by simp
|
2015-12-05 03:13:49 +00:00
|
|
|
|
|
2015-12-13 19:15:10 +00:00
|
|
|
|
theorem mul_bit0_helper [distrib A] (a b t : A) (H : a * b = t) : a * (bit0 b) = bit0 t :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
by rewrite -H; simp
|
2015-12-05 03:13:49 +00:00
|
|
|
|
|
2015-12-13 19:15:10 +00:00
|
|
|
|
theorem mul_bit1 [semiring A] (a b : A) : a * (bit1 b) = bit0 (a * b) + a :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
by simp
|
2015-12-05 03:13:49 +00:00
|
|
|
|
|
2015-12-13 19:15:10 +00:00
|
|
|
|
theorem mul_bit1_helper [semiring A] (a b s t : A) (Hs : a * b = s) (Ht : bit0 s + a = t) :
|
2015-12-05 03:13:49 +00:00
|
|
|
|
a * (bit1 b) = t :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
by simp
|
2015-12-05 03:13:49 +00:00
|
|
|
|
|
2015-12-13 19:15:10 +00:00
|
|
|
|
theorem subst_into_prod [has_mul A] (l r tl tr t : A) (prl : l = tl) (prr : r = tr)
|
2015-12-05 03:13:49 +00:00
|
|
|
|
(prt : tl * tr = t) :
|
|
|
|
|
l * r = t :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
by simp
|
2015-12-05 03:13:49 +00:00
|
|
|
|
|
|
|
|
|
theorem mk_cong (op : A → A) (a b : A) (H : a = b) : op a = op b :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
by simp
|
2015-12-05 03:13:49 +00:00
|
|
|
|
|
2015-12-13 19:15:10 +00:00
|
|
|
|
theorem neg_add_neg_eq_of_add_add_eq_zero [add_comm_group A] (a b c : A) (H : c + a + b = 0) :
|
2015-12-05 03:13:49 +00:00
|
|
|
|
-a + -b = c :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
begin
|
|
|
|
|
apply add_neg_eq_of_eq_add,
|
|
|
|
|
apply neg_eq_of_add_eq_zero,
|
|
|
|
|
simp
|
|
|
|
|
end
|
2015-12-05 03:13:49 +00:00
|
|
|
|
|
2015-12-13 19:15:10 +00:00
|
|
|
|
theorem neg_add_neg_helper [add_comm_group A] (a b c : A) (H : a + b = c) : -a + -b = -c :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
begin apply iff.mp !neg_eq_neg_iff_eq, simp end
|
2015-12-05 03:13:49 +00:00
|
|
|
|
|
2015-12-13 19:15:10 +00:00
|
|
|
|
theorem neg_add_pos_eq_of_eq_add [add_comm_group A] (a b c : A) (H : b = c + a) : -a + b = c :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
begin apply neg_add_eq_of_eq_add, simp end
|
2015-12-05 03:13:49 +00:00
|
|
|
|
|
2015-12-13 19:15:10 +00:00
|
|
|
|
theorem neg_add_pos_helper1 [add_comm_group A] (a b c : A) (H : b + c = a) : -a + b = -c :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
begin apply neg_add_eq_of_eq_add, apply eq_add_neg_of_add_eq H end
|
2015-12-05 03:13:49 +00:00
|
|
|
|
|
2015-12-13 19:15:10 +00:00
|
|
|
|
theorem neg_add_pos_helper2 [add_comm_group A] (a b c : A) (H : a + c = b) : -a + b = c :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
begin apply neg_add_eq_of_eq_add, rewrite H end
|
2015-12-05 03:13:49 +00:00
|
|
|
|
|
2015-12-13 19:15:10 +00:00
|
|
|
|
theorem pos_add_neg_helper [add_comm_group A] (a b c : A) (H : b + a = c) : a + b = c :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
by simp
|
2015-12-05 03:13:49 +00:00
|
|
|
|
|
2015-12-13 19:15:10 +00:00
|
|
|
|
theorem sub_eq_add_neg_helper [add_comm_group A] (t₁ t₂ e w₁ w₂: A) (H₁ : t₁ = w₁)
|
2015-12-05 03:13:49 +00:00
|
|
|
|
(H₂ : t₂ = w₂) (H : w₁ + -w₂ = e) : t₁ - t₂ = e :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
by simp
|
2015-12-05 03:13:49 +00:00
|
|
|
|
|
2015-12-13 19:15:10 +00:00
|
|
|
|
theorem pos_add_pos_helper [add_comm_group A] (a b c h₁ h₂ : A) (H₁ : a = h₁) (H₂ : b = h₂)
|
2015-12-05 03:13:49 +00:00
|
|
|
|
(H : h₁ + h₂ = c) : a + b = c :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
by simp
|
2015-12-05 03:13:49 +00:00
|
|
|
|
|
2015-12-13 19:15:10 +00:00
|
|
|
|
theorem subst_into_subtr [add_group A] (l r t : A) (prt : l + -r = t) : l - r = t :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
by simp
|
2015-12-05 03:13:49 +00:00
|
|
|
|
|
2015-12-13 19:15:10 +00:00
|
|
|
|
theorem neg_neg_helper [add_group A] (a b : A) (H : a = -b) : -a = b :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
by simp
|
2015-12-05 03:13:49 +00:00
|
|
|
|
|
2015-12-13 19:15:10 +00:00
|
|
|
|
theorem neg_mul_neg_helper [ring A] (a b c : A) (H : a * b = c) : (-a) * (-b) = c :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
by simp
|
2015-12-05 03:13:49 +00:00
|
|
|
|
|
2015-12-13 19:15:10 +00:00
|
|
|
|
theorem neg_mul_pos_helper [ring A] (a b c : A) (H : a * b = c) : (-a) * b = -c :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
by simp
|
2015-12-05 03:13:49 +00:00
|
|
|
|
|
2015-12-13 19:15:10 +00:00
|
|
|
|
theorem pos_mul_neg_helper [ring A] (a b c : A) (H : a * b = c) : a * (-b) = -c :=
|
2015-12-29 19:16:18 +00:00
|
|
|
|
by simp
|
2015-12-05 03:13:49 +00:00
|
|
|
|
|
|
|
|
|
end norm_num
|
|
|
|
|
|
2015-12-05 19:50:08 +00:00
|
|
|
|
attribute [simp]
|
2015-12-06 07:27:46 +00:00
|
|
|
|
zero_mul mul_zero
|
2015-12-05 19:50:08 +00:00
|
|
|
|
at simplifier.unit
|
2015-12-05 01:57:03 +00:00
|
|
|
|
|
2015-12-05 19:50:08 +00:00
|
|
|
|
attribute [simp]
|
2015-12-06 07:27:46 +00:00
|
|
|
|
neg_mul_eq_neg_mul_symm mul_neg_eq_neg_mul_symm
|
2015-12-05 19:50:08 +00:00
|
|
|
|
at simplifier.neg
|
2015-12-05 01:57:03 +00:00
|
|
|
|
|
2015-12-05 19:50:08 +00:00
|
|
|
|
attribute [simp]
|
2015-12-06 07:27:46 +00:00
|
|
|
|
left_distrib right_distrib
|
2015-12-05 19:50:08 +00:00
|
|
|
|
at simplifier.distrib
|