/- 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 Various multiplicative and additive structures. Partially modeled on Isabelle's library. -/ import logic.eq data.sigma data.prod import algebra.binary algebra.priority open binary variable {A : Type} /- semigroup -/ /- TODO(Leo): decide whether we keep this annotation or not -/ -- attribute inv [light 3] -- attribute neg [light 3] structure semigroup [class] (A : Type) extends has_mul A := (mul_assoc : ∀a b c : A, a * b * c = a * (b * c)) -- We add pattern hints to the following lemma because we want it to be used in both directions -- at inst_simp strategy. attribute [simp] theorem mul.assoc [semigroup A] (a b c : A) : a * b * c = a * (b * c) := semigroup.mul_assoc a b c set_option pp.all true structure comm_semigroup [class] (A : Type) extends semigroup A := (mul_comm : ∀a b : A, a * b = b * a) attribute [simp] theorem mul.comm [comm_semigroup A] (a b : A) : a * b = b * a := comm_semigroup.mul_comm a b attribute [simp] theorem mul.left_comm [comm_semigroup A] (a b c : A) : a * (b * c) = b * (a * c) := binary.left_comm (@mul.comm A _) (@mul.assoc A _) a b c theorem mul.right_comm [comm_semigroup A] (a b c : A) : (a * b) * c = (a * c) * b := sorry -- by simp structure left_cancel_semigroup [class] (A : Type) extends semigroup A := (mul_left_cancel : ∀a b c : A, a * b = a * c → b = c) theorem mul.left_cancel [left_cancel_semigroup A] {a b c : A} : a * b = a * c → b = c := left_cancel_semigroup.mul_left_cancel a b c abbreviation eq_of_mul_eq_mul_left' := @mul.left_cancel structure right_cancel_semigroup [class] (A : Type) extends semigroup A := (mul_right_cancel : ∀a b c : A, a * b = c * b → a = c) theorem mul.right_cancel [right_cancel_semigroup A] {a b c : A} : a * b = c * b → a = c := right_cancel_semigroup.mul_right_cancel a b c abbreviation eq_of_mul_eq_mul_right' := @mul.right_cancel /- additive semigroup -/ structure add_semigroup [class] (A : Type) extends has_add A := (add_assoc : ∀a b c : A, a + b + c = a + (b + c)) attribute [simp] theorem add.assoc [add_semigroup A] (a b c : A) : a + b + c = a + (b + c) := add_semigroup.add_assoc a b c structure add_comm_semigroup [class] (A : Type) extends add_semigroup A := (add_comm : ∀a b : A, a + b = b + a) attribute [simp] theorem add.comm [add_comm_semigroup A] (a b : A) : a + b = b + a := add_comm_semigroup.add_comm a b attribute [simp] theorem add.left_comm [add_comm_semigroup A] (a b c : A) : a + (b + c) = b + (a + c) := binary.left_comm (@add.comm A _) (@add.assoc A _) a b c theorem add.right_comm [add_comm_semigroup A] (a b c : A) : (a + b) + c = (a + c) + b := sorry -- by simp structure add_left_cancel_semigroup [class] (A : Type) extends add_semigroup A := (add_left_cancel : ∀a b c : A, a + b = a + c → b = c) theorem add.left_cancel [add_left_cancel_semigroup A] {a b c : A} : a + b = a + c → b = c := add_left_cancel_semigroup.add_left_cancel a b c abbreviation eq_of_add_eq_add_left := @add.left_cancel structure add_right_cancel_semigroup [class] (A : Type) extends add_semigroup A := (add_right_cancel : ∀a b c : A, a + b = c + b → a = c) theorem add.right_cancel [add_right_cancel_semigroup A] {a b c : A} : a + b = c + b → a = c := add_right_cancel_semigroup.add_right_cancel a b c abbreviation eq_of_add_eq_add_right := @add.right_cancel /- monoid -/ structure monoid [class] (A : Type) extends semigroup A, has_one A := (one_mul : ∀a : A, 1 * a = a) (mul_one : ∀a : A, a * 1 = a) attribute [simp] theorem one_mul [monoid A] (a : A) : 1 * a = a := monoid.one_mul a attribute [simp] theorem mul_one [monoid A] (a : A) : a * 1 = a := monoid.mul_one a structure comm_monoid [class] (A : Type) extends monoid A, comm_semigroup A /- additive monoid -/ structure add_monoid [class] (A : Type) extends add_semigroup A, has_zero A := (zero_add : ∀a : A, 0 + a = a) (add_zero : ∀a : A, a + 0 = a) attribute [simp] theorem zero_add [add_monoid A] (a : A) : 0 + a = a := add_monoid.zero_add a attribute [simp] theorem add_zero [add_monoid A] (a : A) : a + 0 = a := add_monoid.add_zero a structure add_comm_monoid [class] (A : Type) extends add_monoid A, add_comm_semigroup A definition add_monoid.to_monoid {A : Type} [add_monoid A] : monoid A := ⦃ monoid, mul := add_monoid.add, mul_assoc := add_monoid.add_assoc, one := add_monoid.zero A, mul_one := add_monoid.add_zero, one_mul := add_monoid.zero_add ⦄ definition add_comm_monoid.to_comm_monoid {A : Type} [add_comm_monoid A] : comm_monoid A := ⦃ comm_monoid, add_monoid.to_monoid, mul_comm := add_comm_monoid.add_comm ⦄ section add_comm_monoid variables [add_comm_monoid A] theorem add_comm_three (a b c : A) : a + b + c = c + b + a := sorry -- by simp theorem add.comm4 : ∀ (n m k l : A), n + m + (k + l) = n + k + (m + l) := sorry -- by simp end add_comm_monoid /- group -/ structure group [class] (A : Type) extends monoid A, has_inv A := (mul_left_inv : ∀a : A, a⁻¹ * a = 1) -- Note: with more work, we could derive the axiom one_mul section group variable [group A] attribute [simp] theorem mul.left_inv (a : A) : a⁻¹ * a = 1 := group.mul_left_inv a attribute [simp] theorem inv_mul_cancel_left (a b : A) : a⁻¹ * (a * b) = b := sorry -- by rewrite [-mul.assoc, mul.left_inv, one_mul] attribute [simp] theorem inv_mul_cancel_right (a b : A) : a * b⁻¹ * b = a := sorry -- by simp theorem inv_eq_of_mul_eq_one {a b : A} (H : a * b = 1) : a⁻¹ = b := sorry /- have a⁻¹ * 1 = b, by inst_simp, by inst_simp -/ attribute [simp] theorem one_inv : 1⁻¹ = (1 : A) := inv_eq_of_mul_eq_one (one_mul 1) attribute [simp] theorem inv_inv (a : A) : (a⁻¹)⁻¹ = a := inv_eq_of_mul_eq_one (mul.left_inv a) variable (A) theorem left_inverse_inv : function.left_inverse (λ a : A, a⁻¹) (λ a, a⁻¹) := take a, inv_inv a variable {A} theorem inv.inj {a b : A} (H : a⁻¹ = b⁻¹) : a = b := sorry /- have a = a⁻¹⁻¹, by simp_nohyps, by inst_simp -/ theorem inv_eq_inv_iff_eq (a b : A) : a⁻¹ = b⁻¹ ↔ a = b := sorry -- iff.intro (assume H, inv.inj H) (by simp) theorem inv_eq_one_iff_eq_one (a : A) : a⁻¹ = 1 ↔ a = 1 := sorry /- have a⁻¹ = 1⁻¹ ↔ a = 1, from inv_eq_inv_iff_eq a 1, by simp -/ theorem eq_one_of_inv_eq_one (a : A) : a⁻¹ = 1 → a = 1 := iff.mp (inv_eq_one_iff_eq_one a) theorem eq_inv_of_eq_inv {a b : A} (H : a = b⁻¹) : b = a⁻¹ := sorry -- by simp theorem eq_inv_iff_eq_inv (a b : A) : a = b⁻¹ ↔ b = a⁻¹ := iff.intro eq_inv_of_eq_inv eq_inv_of_eq_inv theorem eq_inv_of_mul_eq_one {a b : A} (H : a * b = 1) : a = b⁻¹ := sorry /- have a⁻¹ = b, from inv_eq_of_mul_eq_one H, by inst_simp -/ attribute [simp] theorem mul.right_inv (a : A) : a * a⁻¹ = 1 := sorry /- have a = a⁻¹⁻¹, by simp, by inst_simp -/ attribute [simp] theorem mul_inv_cancel_left (a b : A) : a * (a⁻¹ * b) = b := sorry -- by inst_simp attribute [simp] theorem mul_inv_cancel_right (a b : A) : a * b * b⁻¹ = a := sorry -- by inst_simp attribute [simp] theorem mul_inv (a b : A) : (a * b)⁻¹ = b⁻¹ * a⁻¹ := sorry -- inv_eq_of_mul_eq_one (by inst_simp) theorem eq_of_mul_inv_eq_one {a b : A} (H : a * b⁻¹ = 1) : a = b := sorry /- have a⁻¹ * 1 = a⁻¹, by inst_simp, by inst_simp -/ theorem eq_mul_inv_of_mul_eq {a b c : A} (H : a * c = b) : a = b * c⁻¹ := sorry -- by simp theorem eq_inv_mul_of_mul_eq {a b c : A} (H : b * a = c) : a = b⁻¹ * c := sorry -- by simp theorem inv_mul_eq_of_eq_mul {a b c : A} (H : b = a * c) : a⁻¹ * b = c := sorry -- by simp theorem mul_inv_eq_of_eq_mul {a b c : A} (H : a = c * b) : a * b⁻¹ = c := sorry -- by simp theorem eq_mul_of_mul_inv_eq {a b c : A} (H : a * c⁻¹ = b) : a = b * c := sorry -- by simp theorem eq_mul_of_inv_mul_eq {a b c : A} (H : b⁻¹ * a = c) : a = b * c := sorry -- by simp theorem mul_eq_of_eq_inv_mul {a b c : A} (H : b = a⁻¹ * c) : a * b = c := sorry -- by simp theorem mul_eq_of_eq_mul_inv {a b c : A} (H : a = c * b⁻¹) : a * b = c := sorry -- by simp theorem mul_eq_iff_eq_inv_mul (a b c : A) : a * b = c ↔ b = a⁻¹ * c := iff.intro eq_inv_mul_of_mul_eq mul_eq_of_eq_inv_mul theorem mul_eq_iff_eq_mul_inv (a b c : A) : a * b = c ↔ a = c * b⁻¹ := iff.intro eq_mul_inv_of_mul_eq mul_eq_of_eq_mul_inv theorem mul_left_cancel {a b c : A} (H : a * b = a * c) : b = c := sorry /- have a⁻¹ * (a * b) = b, by inst_simp, by inst_simp -/ theorem mul_right_cancel {a b c : A} (H : a * b = c * b) : a = c := sorry /- have a * b * b⁻¹ = a, by inst_simp, by inst_simp -/ theorem mul_eq_one_of_mul_eq_one {a b : A} (H : b * a = 1) : a * b = 1 := sorry -- by rewrite [-inv_eq_of_mul_eq_one H, mul.left_inv] theorem mul_eq_one_iff_mul_eq_one (a b : A) : a * b = 1 ↔ b * a = 1 := iff.intro mul_eq_one_of_mul_eq_one mul_eq_one_of_mul_eq_one definition conj_by (g a : A) := g * a * g⁻¹ definition is_conjugate (a b : A) := ∃ x, conj_by x b = a local infixl ` ~ ` := is_conjugate local infixr ` ∘c `:55 := conj_by local attribute conj_by [reducible] attribute [simp] lemma conj_compose (f g a : A) : f ∘c g ∘c a = f*g ∘c a := sorry -- by inst_simp attribute [simp] lemma conj_id (a : A) : 1 ∘c a = a := sorry -- by inst_simp attribute [simp] lemma conj_one (g : A) : g ∘c 1 = 1 := sorry -- by inst_simp attribute [simp] lemma conj_inv_cancel (g : A) : ∀ a, g⁻¹ ∘c g ∘c a = a := sorry -- by inst_simp attribute [simp] lemma conj_inv (g : A) : ∀ a, (g ∘c a)⁻¹ = g ∘c a⁻¹ := sorry -- by inst_simp lemma is_conj.refl (a : A) : a ~ a := exists.intro 1 (conj_id a) lemma is_conj.symm (a b : A) : a ~ b → b ~ a := sorry /- assume Pab, obtain x (Pconj : x ∘c b = a), from Pab, have Pxinv : x⁻¹ ∘c x ∘c b = x⁻¹ ∘c a, by simp, exists.intro x⁻¹ (by simp) -/ lemma is_conj.trans (a b c : A) : a ~ b → b ~ c → a ~ c := sorry /- assume Pab, assume Pbc, obtain x (Px : x ∘c b = a), from Pab, obtain y (Py : y ∘c c = b), from Pbc, exists.intro (x*y) (by inst_simp) -/ end group attribute [instance] definition group.to_left_cancel_semigroup [s : group A] : left_cancel_semigroup A := ⦃ left_cancel_semigroup, s, mul_left_cancel := @mul_left_cancel A s ⦄ attribute [instance] definition group.to_right_cancel_semigroup [s : group A] : right_cancel_semigroup A := ⦃ right_cancel_semigroup, s, mul_right_cancel := @mul_right_cancel A s ⦄ structure comm_group [class] (A : Type) extends group A, comm_monoid A /- additive group -/ structure add_group [class] (A : Type) extends add_monoid A, has_neg A := (add_left_inv : ∀a : A, -a + a = 0) definition add_group.to_group {A : Type} [add_group A] : group A := ⦃ group, add_monoid.to_monoid, mul_left_inv := add_group.add_left_inv ⦄ section add_group variables [s : add_group A] include s attribute [simp] theorem add.left_inv (a : A) : -a + a = 0 := add_group.add_left_inv a attribute [simp] theorem neg_add_cancel_left (a b : A) : -a + (a + b) = b := calc -a + (a + b) = (-a + a) + b : sorry -- by rewrite add.assoc ... = b : sorry -- by simp attribute [simp] theorem neg_add_cancel_right (a b : A) : a + -b + b = a := sorry -- by simp theorem neg_eq_of_add_eq_zero {a b : A} (H : a + b = 0) : -a = b := sorry /- have -a + 0 = b, by inst_simp, by inst_simp -/ attribute [simp] theorem neg_zero : -0 = (0 : A) := neg_eq_of_add_eq_zero (zero_add 0) attribute [simp] theorem neg_neg (a : A) : -(-a) = a := neg_eq_of_add_eq_zero (add.left_inv a) variable (A) theorem left_inverse_neg : function.left_inverse (λ a : A, - a) (λ a, - a) := take a, neg_neg a variable {A} theorem eq_neg_of_add_eq_zero {a b : A} (H : a + b = 0) : a = -b := have -a = b, from neg_eq_of_add_eq_zero H, sorry -- by inst_simp theorem neg.inj {a b : A} (H : -a = -b) : a = b := sorry /- have a = -(-a), by simp_nohyps, by inst_simp -/ theorem neg_eq_neg_iff_eq (a b : A) : -a = -b ↔ a = b := sorry -- iff.intro (assume H, neg.inj H) (by simp) theorem eq_of_neg_eq_neg {a b : A} : -a = -b → a = b := iff.mp (neg_eq_neg_iff_eq a b) theorem neg_eq_zero_iff_eq_zero (a : A) : -a = 0 ↔ a = 0 := have -a = -0 ↔ a = 0, from neg_eq_neg_iff_eq a 0, sorry -- by simp theorem eq_zero_of_neg_eq_zero {a : A} : -a = 0 → a = 0 := iff.mp (neg_eq_zero_iff_eq_zero a) theorem eq_neg_of_eq_neg {a b : A} (H : a = -b) : b = -a := sorry -- by simp theorem eq_neg_iff_eq_neg (a b : A) : a = -b ↔ b = -a := iff.intro eq_neg_of_eq_neg eq_neg_of_eq_neg attribute [simp] theorem add.right_inv (a : A) : a + -a = 0 := sorry /- have a = -(-a), by simp, by inst_simp -/ attribute [simp] theorem add_neg_cancel_left (a b : A) : a + (-a + b) = b := sorry -- by inst_simp attribute [simp] theorem add_neg_cancel_right (a b : A) : a + b + -b = a := sorry -- by simp attribute [simp] theorem neg_add_rev (a b : A) : -(a + b) = -b + -a := sorry -- neg_eq_of_add_eq_zero (by simp) -- TODO: delete these in favor of sub rules? theorem eq_add_neg_of_add_eq {a b c : A} (H : a + c = b) : a = b + -c := sorry -- by simp theorem eq_neg_add_of_add_eq {a b c : A} (H : b + a = c) : a = -b + c := sorry -- by simp theorem neg_add_eq_of_eq_add {a b c : A} (H : b = a + c) : -a + b = c := sorry -- by simp theorem add_neg_eq_of_eq_add {a b c : A} (H : a = c + b) : a + -b = c := sorry -- by simp theorem eq_add_of_add_neg_eq {a b c : A} (H : a + -c = b) : a = b + c := sorry -- by simp theorem eq_add_of_neg_add_eq {a b c : A} (H : -b + a = c) : a = b + c := sorry -- by simp theorem add_eq_of_eq_neg_add {a b c : A} (H : b = -a + c) : a + b = c := sorry -- by simp theorem add_eq_of_eq_add_neg {a b c : A} (H : a = c + -b) : a + b = c := sorry -- by simp theorem add_eq_iff_eq_neg_add (a b c : A) : a + b = c ↔ b = -a + c := iff.intro eq_neg_add_of_add_eq add_eq_of_eq_neg_add theorem add_eq_iff_eq_add_neg (a b c : A) : a + b = c ↔ a = c + -b := iff.intro eq_add_neg_of_add_eq add_eq_of_eq_add_neg theorem add_left_cancel {a b c : A} (H : a + b = a + c) : b = c := sorry /- have -a + (a + b) = b, by inst_simp, by inst_simp -/ theorem add_right_cancel {a b c : A} (H : a + b = c + b) : a = c := sorry /- have a + b + -b = a, by inst_simp, by inst_simp -/ attribute [instance] definition add_group.to_left_cancel_semigroup : add_left_cancel_semigroup A := ⦃ add_left_cancel_semigroup, s, add_left_cancel := @add_left_cancel A s ⦄ attribute [instance] definition add_group.to_add_right_cancel_semigroup : add_right_cancel_semigroup A := ⦃ add_right_cancel_semigroup, s, add_right_cancel := @add_right_cancel A s ⦄ theorem add_neg_eq_neg_add_rev {a b : A} : a + -b = -(b + -a) := sorry -- by simp theorem ne_add_of_ne_zero_right (a : A) {b : A} (H : b ≠ 0) : a ≠ b + a := sorry /- begin intro Heq, apply H, rewrite [-zero_add a at Heq{1}], let Heq' := eq_of_add_eq_add_right Heq, apply eq.symm Heq' end -/ theorem ne_add_of_ne_zero_left (a : A) {b : A} (H : b ≠ 0) : a ≠ a + b := sorry /- begin intro Heq, apply H, rewrite [-add_zero a at Heq{1}], let Heq' := eq_of_add_eq_add_left Heq, apply eq.symm Heq' end -/ /- sub -/ -- TODO: derive corresponding facts for div in a field attribute [reducible] protected definition algebra.sub (a b : A) : A := a + -b attribute [instance] definition add_group_has_sub : has_sub A := has_sub.mk algebra.sub attribute [simp] theorem sub_eq_add_neg (a b : A) : a - b = a + -b := rfl theorem sub_self (a : A) : a - a = 0 := add.right_inv a theorem sub_add_cancel (a b : A) : a - b + b = a := neg_add_cancel_right a b theorem add_sub_cancel (a b : A) : a + b - b = a := add_neg_cancel_right a b theorem add_sub_assoc (a b c : A) : a + b - c = a + (b - c) := sorry -- by rewrite [sub_eq_add_neg, add.assoc, -sub_eq_add_neg] theorem eq_of_sub_eq_zero {a b : A} (H : a - b = 0) : a = b := sorry /- have -a + 0 = -a, by inst_simp, by inst_simp -/ theorem eq_iff_sub_eq_zero (a b : A) : a = b ↔ a - b = 0 := iff.intro (assume H, eq.subst H (sub_self _)) (assume H, eq_of_sub_eq_zero H) theorem zero_sub (a : A) : 0 - a = -a := zero_add (-a) theorem sub_zero (a : A) : a - 0 = a := sorry -- by simp theorem sub_ne_zero_of_ne {a b : A} (H : a ≠ b) : a - b ≠ 0 := sorry /- begin intro Hab, apply H, apply eq_of_sub_eq_zero Hab end -/ theorem sub_neg_eq_add (a b : A) : a - (-b) = a + b := sorry -- by simp theorem neg_sub (a b : A) : -(a - b) = b - a := sorry -- neg_eq_of_add_eq_zero (by inst_simp) theorem add_sub (a b c : A) : a + (b - c) = a + b - c := sorry -- by simp theorem sub_add_eq_sub_sub_swap (a b c : A) : a - (b + c) = a - c - b := sorry -- by inst_simp theorem sub_eq_iff_eq_add (a b c : A) : a - b = c ↔ a = c + b := iff.intro (assume H, eq_add_of_add_neg_eq H) (assume H, add_neg_eq_of_eq_add H) theorem eq_sub_iff_add_eq (a b c : A) : a = b - c ↔ a + c = b := iff.intro (assume H, add_eq_of_eq_add_neg H) (assume H, eq_add_neg_of_add_eq H) theorem eq_iff_eq_of_sub_eq_sub {a b c d : A} (H : a - b = c - d) : a = b ↔ c = d := calc a = b ↔ a - b = 0 : eq_iff_sub_eq_zero a b ... = (c - d = 0) : sorry -- by rewrite H ... ↔ c = d : iff.symm (eq_iff_sub_eq_zero c d) theorem eq_sub_of_add_eq {a b c : A} (H : a + c = b) : a = b - c := sorry -- by simp theorem sub_eq_of_eq_add {a b c : A} (H : a = c + b) : a - b = c := sorry -- by simp theorem eq_add_of_sub_eq {a b c : A} (H : a - c = b) : a = b + c := sorry -- by simp theorem add_eq_of_eq_sub {a b c : A} (H : a = c - b) : a + b = c := sorry -- by simp theorem left_inverse_sub_add_left (c : A) : function.left_inverse (λ x, x - c) (λ x, x + c) := take x, add_sub_cancel x c theorem left_inverse_add_left_sub (c : A) : function.left_inverse (λ x, x + c) (λ x, x - c) := take x, sub_add_cancel x c theorem left_inverse_add_right_neg_add (c : A) : function.left_inverse (λ x, c + x) (λ x, - c + x) := take x, add_neg_cancel_left c x theorem left_inverse_neg_add_add_right (c : A) : function.left_inverse (λ x, - c + x) (λ x, c + x) := take x, neg_add_cancel_left c x end add_group structure add_comm_group [class] (A : Type) extends add_group A, add_comm_monoid A section add_comm_group variable [s : add_comm_group A] include s theorem sub_add_eq_sub_sub (a b c : A) : a - (b + c) = a - b - c := sorry -- by simp theorem neg_add_eq_sub (a b : A) : -a + b = b - a := sorry -- by simp theorem neg_add (a b : A) : -(a + b) = -a + -b := sorry -- by simp theorem sub_add_eq_add_sub (a b c : A) : a - b + c = a + c - b := sorry -- by simp theorem sub_sub (a b c : A) : a - b - c = a - (b + c) := sorry -- by simp theorem add_sub_add_left_eq_sub (a b c : A) : (c + a) - (c + b) = a - b := sorry -- by simp theorem eq_sub_of_add_eq' {a b c : A} (H : c + a = b) : a = b - c := sorry -- by simp theorem sub_eq_of_eq_add' {a b c : A} (H : a = b + c) : a - b = c := sorry -- by simp theorem eq_add_of_sub_eq' {a b c : A} (H : a - b = c) : a = b + c := sorry -- by simp theorem add_eq_of_eq_sub' {a b c : A} (H : b = c - a) : a + b = c := sorry -- by simp theorem sub_sub_self (a b : A) : a - (a - b) = b := sorry -- by simp theorem add_sub_comm (a b c d : A) : a + b - (c + d) = (a - c) + (b - d) := sorry -- by simp theorem sub_eq_sub_add_sub (a b c : A) : a - b = c - b + (a - c) := sorry -- by simp theorem neg_neg_sub_neg (a b : A) : - (-a - -b) = a - b := sorry -- by simp end add_comm_group definition group_of_add_group (A : Type) [G : add_group A] : group A := ⦃group, mul := has_add.add, mul_assoc := add.assoc, one := has_zero.zero A, one_mul := zero_add, mul_one := add_zero, inv := has_neg.neg, mul_left_inv := add.left_inv⦄ namespace norm_num reveal add.assoc definition add1 [has_add A] [has_one A] (a : A) : A := add a one local attribute add1 bit0 bit1 [reducible] theorem add_comm_four [add_comm_semigroup A] (a b : A) : a + a + (b + b) = (a + b) + (a + b) := sorry -- by simp theorem add_comm_middle [add_comm_semigroup A] (a b c : A) : a + b + c = a + c + b := sorry -- by simp theorem bit0_add_bit0 [add_comm_semigroup A] (a b : A) : bit0 a + bit0 b = bit0 (a + b) := sorry -- by simp theorem bit0_add_bit0_helper [add_comm_semigroup A] (a b t : A) (H : a + b = t) : bit0 a + bit0 b = bit0 t := sorry -- by rewrite -H; simp theorem bit1_add_bit0 [add_comm_semigroup A] [has_one A] (a b : A) : bit1 a + bit0 b = bit1 (a + b) := sorry -- by simp theorem bit1_add_bit0_helper [add_comm_semigroup A] [has_one A] (a b t : A) (H : a + b = t) : bit1 a + bit0 b = bit1 t := sorry -- by rewrite -H; simp theorem bit0_add_bit1 [add_comm_semigroup A] [has_one A] (a b : A) : bit0 a + bit1 b = bit1 (a + b) := sorry -- by simp theorem bit0_add_bit1_helper [add_comm_semigroup A] [has_one A] (a b t : A) (H : a + b = t) : bit0 a + bit1 b = bit1 t := sorry -- by rewrite -H; simp theorem bit1_add_bit1 [add_comm_semigroup A] [has_one A] (a b : A) : bit1 a + bit1 b = bit0 (add1 (a + b)) := sorry -- by simp theorem bit1_add_bit1_helper [add_comm_semigroup A] [has_one A] (a b t s: A) (H : (a + b) = t) (H2 : add1 t = s) : bit1 a + bit1 b = bit0 s := sorry -- by inst_simp theorem bin_add_zero [add_monoid A] (a : A) : a + zero = a := sorry -- by simp theorem bin_zero_add [add_monoid A] (a : A) : zero + a = a := sorry -- by simp theorem one_add_bit0 [add_comm_semigroup A] [has_one A] (a : A) : one + bit0 a = bit1 a := sorry -- by simp theorem bit0_add_one [has_add A] [has_one A] (a : A) : bit0 a + one = bit1 a := rfl theorem bit1_add_one [has_add A] [has_one A] (a : A) : bit1 a + one = add1 (bit1 a) := rfl theorem bit1_add_one_helper [has_add A] [has_one A] (a t : A) (H : add1 (bit1 a) = t) : bit1 a + one = t := sorry -- by inst_simp theorem one_add_bit1 [add_comm_semigroup A] [has_one A] (a : A) : one + bit1 a = add1 (bit1 a) := sorry -- by simp theorem one_add_bit1_helper [add_comm_semigroup A] [has_one A] (a t : A) (H : add1 (bit1 a) = t) : one + bit1 a = t := sorry -- by inst_simp theorem add1_bit0 [has_add A] [has_one A] (a : A) : add1 (bit0 a) = bit1 a := rfl theorem add1_bit1 [add_comm_semigroup A] [has_one A] (a : A) : add1 (bit1 a) = bit0 (add1 a) := sorry -- by simp theorem add1_bit1_helper [add_comm_semigroup A] [has_one A] (a t : A) (H : add1 a = t) : add1 (bit1 a) = bit0 t := sorry -- by inst_simp theorem add1_one [has_add A] [has_one A] : add1 (one : A) = bit0 one := rfl theorem add1_zero [add_monoid A] [has_one A] : add1 (zero : A) = one := sorry -- by simp theorem one_add_one [has_add A] [has_one A] : (one : A) + one = bit0 one := rfl theorem subst_into_sum [has_add A] (l r tl tr t : A) (prl : l = tl) (prr : r = tr) (prt : tl + tr = t) : l + r = t := sorry -- by simp theorem neg_zero_helper [add_group A] (a : A) (H : a = 0) : - a = 0 := sorry -- by simp end norm_num attribute [simp] zero_add add_zero one_mul mul_one attribute [simp] neg_neg sub_eq_add_neg attribute [simp] add.assoc add.comm add.left_comm mul.left_comm mul.comm mul.assoc