2014-12-06 05:46:17 +00:00
|
|
|
|
/-
|
2015-04-24 23:58:50 +00:00
|
|
|
|
Copyright (c) 2014-2015 Microsoft Corporation. All rights reserved.
|
2014-12-06 05:46:17 +00:00
|
|
|
|
Released under Apache 2.0 license as described in the file LICENSE.
|
2015-04-24 23:58:50 +00:00
|
|
|
|
Authors: Leonardo de Moura, Jeremy Avigad, Floris van Doorn, Jakob von Raumer
|
2014-12-06 05:46:17 +00:00
|
|
|
|
-/
|
2015-02-26 18:19:54 +00:00
|
|
|
|
|
2014-12-06 05:46:17 +00:00
|
|
|
|
prelude
|
2015-10-12 23:49:10 +00:00
|
|
|
|
import init.num init.wf
|
2015-11-21 04:58:06 +00:00
|
|
|
|
open iff
|
2015-04-24 23:58:50 +00:00
|
|
|
|
|
|
|
|
|
-- Empty type
|
|
|
|
|
-- ----------
|
|
|
|
|
|
|
|
|
|
namespace empty
|
|
|
|
|
|
2015-05-05 21:25:35 +00:00
|
|
|
|
protected theorem elim {A : Type} (H : empty) : A :=
|
2015-04-24 23:58:50 +00:00
|
|
|
|
empty.rec (λe, A) H
|
|
|
|
|
|
|
|
|
|
end empty
|
|
|
|
|
|
|
|
|
|
protected definition empty.has_decidable_eq [instance] : decidable_eq empty :=
|
|
|
|
|
take (a b : empty), decidable.inl (!empty.elim a)
|
|
|
|
|
|
|
|
|
|
-- Unit type
|
|
|
|
|
-- ---------
|
|
|
|
|
|
|
|
|
|
namespace unit
|
|
|
|
|
|
|
|
|
|
notation `⋆` := star
|
|
|
|
|
|
|
|
|
|
end unit
|
|
|
|
|
|
|
|
|
|
-- Sigma type
|
|
|
|
|
-- ----------
|
|
|
|
|
|
2015-09-30 23:52:34 +00:00
|
|
|
|
notation `Σ` binders `, ` r:(scoped P, sigma P) := r
|
2015-10-26 22:29:19 +00:00
|
|
|
|
abbreviation dpair [constructor] := @sigma.mk
|
2015-04-24 23:58:50 +00:00
|
|
|
|
namespace sigma
|
2015-09-30 23:52:34 +00:00
|
|
|
|
notation `⟨`:max t:(foldr `, ` (e r, mk e r)) `⟩`:0 := t --input ⟨ ⟩ as \< \>
|
2015-04-24 23:58:50 +00:00
|
|
|
|
|
|
|
|
|
namespace ops
|
|
|
|
|
postfix `.1`:(max+1) := pr1
|
|
|
|
|
postfix `.2`:(max+1) := pr2
|
|
|
|
|
abbreviation pr₁ := @pr1
|
|
|
|
|
abbreviation pr₂ := @pr2
|
|
|
|
|
end ops
|
|
|
|
|
end sigma
|
|
|
|
|
|
|
|
|
|
-- Sum type
|
|
|
|
|
-- --------
|
2015-12-08 17:57:55 +00:00
|
|
|
|
infixr ⊎ := sum
|
|
|
|
|
infixr + := sum
|
2015-04-24 23:58:50 +00:00
|
|
|
|
|
|
|
|
|
namespace sum
|
2015-10-28 03:32:12 +00:00
|
|
|
|
infixr [parsing_only] `+t`:25 := sum -- notation which is never overloaded
|
2015-04-24 23:58:50 +00:00
|
|
|
|
namespace low_precedence_plus
|
2015-09-30 23:52:34 +00:00
|
|
|
|
reserve infixr ` + `:25 -- conflicts with notation for addition
|
|
|
|
|
infixr ` + ` := sum
|
2015-04-24 23:58:50 +00:00
|
|
|
|
end low_precedence_plus
|
2015-05-05 21:25:35 +00:00
|
|
|
|
|
|
|
|
|
variables {a b c d : Type}
|
2015-12-08 17:57:55 +00:00
|
|
|
|
|
|
|
|
|
protected definition elim (H : a ⊎ b) (f : a → c) (g : b → c) := sum.rec_on H f g
|
|
|
|
|
|
2015-05-05 21:25:35 +00:00
|
|
|
|
definition sum_of_sum_of_imp_of_imp (H₁ : a ⊎ b) (H₂ : a → c) (H₃ : b → d) : c ⊎ d :=
|
|
|
|
|
sum.rec_on H₁
|
|
|
|
|
(assume Ha : a, sum.inl (H₂ Ha))
|
|
|
|
|
(assume Hb : b, sum.inr (H₃ Hb))
|
|
|
|
|
|
|
|
|
|
definition sum_of_sum_of_imp_left (H₁ : a ⊎ c) (H : a → b) : b ⊎ c :=
|
|
|
|
|
sum.rec_on H₁
|
|
|
|
|
(assume H₂ : a, sum.inl (H H₂))
|
|
|
|
|
(assume H₂ : c, sum.inr H₂)
|
|
|
|
|
|
|
|
|
|
definition sum_of_sum_of_imp_right (H₁ : c ⊎ a) (H : a → b) : c ⊎ b :=
|
|
|
|
|
sum.rec_on H₁
|
|
|
|
|
(assume H₂ : c, sum.inl H₂)
|
|
|
|
|
(assume H₂ : a, sum.inr (H H₂))
|
2015-04-24 23:58:50 +00:00
|
|
|
|
end sum
|
|
|
|
|
|
|
|
|
|
-- Product type
|
|
|
|
|
-- ------------
|
2014-12-06 05:46:17 +00:00
|
|
|
|
|
2015-09-22 16:01:55 +00:00
|
|
|
|
abbreviation pair [constructor] := @prod.mk
|
2015-12-08 17:57:55 +00:00
|
|
|
|
infixr × := prod
|
2014-12-06 05:46:17 +00:00
|
|
|
|
|
|
|
|
|
namespace prod
|
2014-12-16 20:10:12 +00:00
|
|
|
|
|
2015-05-02 21:17:50 +00:00
|
|
|
|
-- notation for n-ary tuples
|
2015-09-30 23:52:34 +00:00
|
|
|
|
notation `(` h `, ` t:(foldl `,` (e r, prod.mk r e) h) `)` := t
|
2015-10-28 03:32:12 +00:00
|
|
|
|
infixr [parsing_only] `×t`:30 := prod -- notation which is never overloaded
|
2014-12-16 20:10:12 +00:00
|
|
|
|
|
2015-02-21 00:30:32 +00:00
|
|
|
|
namespace ops
|
2015-10-09 20:21:03 +00:00
|
|
|
|
infixr [parsing_only] * := prod
|
2015-02-21 00:30:32 +00:00
|
|
|
|
postfix `.1`:(max+1) := pr1
|
|
|
|
|
postfix `.2`:(max+1) := pr2
|
|
|
|
|
abbreviation pr₁ := @pr1
|
|
|
|
|
abbreviation pr₂ := @pr2
|
|
|
|
|
end ops
|
|
|
|
|
|
2014-12-06 05:46:17 +00:00
|
|
|
|
namespace low_precedence_times
|
2014-12-16 20:10:12 +00:00
|
|
|
|
|
2015-09-30 23:52:34 +00:00
|
|
|
|
reserve infixr ` * `:30 -- conflicts with notation for multiplication
|
|
|
|
|
infixr ` * ` := prod
|
2014-12-16 20:10:12 +00:00
|
|
|
|
|
2014-12-06 05:46:17 +00:00
|
|
|
|
end low_precedence_times
|
|
|
|
|
|
2015-05-02 21:17:50 +00:00
|
|
|
|
open prod.ops
|
2014-12-12 18:17:50 +00:00
|
|
|
|
|
2015-09-22 16:01:55 +00:00
|
|
|
|
definition flip [unfold 3] {A B : Type} (a : A × B) : B × A := pair (pr2 a) (pr1 a)
|
2014-12-06 05:46:17 +00:00
|
|
|
|
|
|
|
|
|
open well_founded
|
|
|
|
|
|
|
|
|
|
section
|
|
|
|
|
variables {A B : Type}
|
|
|
|
|
variable (Ra : A → A → Type)
|
|
|
|
|
variable (Rb : B → B → Type)
|
|
|
|
|
|
|
|
|
|
-- Lexicographical order based on Ra and Rb
|
|
|
|
|
inductive lex : A × B → A × B → Type :=
|
2015-02-26 01:00:10 +00:00
|
|
|
|
| left : ∀{a₁ b₁} a₂ b₂, Ra a₁ a₂ → lex (a₁, b₁) (a₂, b₂)
|
|
|
|
|
| right : ∀a {b₁ b₂}, Rb b₁ b₂ → lex (a, b₁) (a, b₂)
|
2014-12-06 05:46:17 +00:00
|
|
|
|
|
|
|
|
|
-- Relational product based on Ra and Rb
|
|
|
|
|
inductive rprod : A × B → A × B → Type :=
|
|
|
|
|
intro : ∀{a₁ b₁ a₂ b₂}, Ra a₁ a₂ → Rb b₁ b₂ → rprod (a₁, b₁) (a₂, b₂)
|
|
|
|
|
end
|
|
|
|
|
|
2015-04-22 02:17:59 +00:00
|
|
|
|
section
|
2014-12-06 05:46:17 +00:00
|
|
|
|
parameters {A B : Type}
|
|
|
|
|
parameters {Ra : A → A → Type} {Rb : B → B → Type}
|
2015-04-22 02:17:59 +00:00
|
|
|
|
local infix `≺`:50 := lex Ra Rb
|
2014-12-06 05:46:17 +00:00
|
|
|
|
|
|
|
|
|
definition lex.accessible {a} (aca : acc Ra a) (acb : ∀b, acc Rb b): ∀b, acc (lex Ra Rb) (a, b) :=
|
|
|
|
|
acc.rec_on aca
|
|
|
|
|
(λxa aca (iHa : ∀y, Ra y xa → ∀b, acc (lex Ra Rb) (y, b)),
|
|
|
|
|
λb, acc.rec_on (acb b)
|
|
|
|
|
(λxb acb
|
|
|
|
|
(iHb : ∀y, Rb y xb → acc (lex Ra Rb) (xa, y)),
|
|
|
|
|
acc.intro (xa, xb) (λp (lt : p ≺ (xa, xb)),
|
|
|
|
|
have aux : xa = xa → xb = xb → acc (lex Ra Rb) p, from
|
2015-02-11 20:49:27 +00:00
|
|
|
|
@prod.lex.rec_on A B Ra Rb (λp₁ p₂ h, pr₁ p₂ = xa → pr₂ p₂ = xb → acc (lex Ra Rb) p₁)
|
|
|
|
|
p (xa, xb) lt
|
2014-12-06 05:46:17 +00:00
|
|
|
|
(λa₁ b₁ a₂ b₂ (H : Ra a₁ a₂) (eq₂ : a₂ = xa) (eq₃ : b₂ = xb),
|
|
|
|
|
show acc (lex Ra Rb) (a₁, b₁), from
|
|
|
|
|
have Ra₁ : Ra a₁ xa, from eq.rec_on eq₂ H,
|
|
|
|
|
iHa a₁ Ra₁ b₁)
|
|
|
|
|
(λa b₁ b₂ (H : Rb b₁ b₂) (eq₂ : a = xa) (eq₃ : b₂ = xb),
|
|
|
|
|
show acc (lex Ra Rb) (a, b₁), from
|
|
|
|
|
have Rb₁ : Rb b₁ xb, from eq.rec_on eq₃ H,
|
|
|
|
|
have eq₂' : xa = a, from eq.rec_on eq₂ rfl,
|
|
|
|
|
eq.rec_on eq₂' (iHb b₁ Rb₁)),
|
|
|
|
|
aux rfl rfl)))
|
|
|
|
|
|
|
|
|
|
-- The lexicographical order of well founded relations is well-founded
|
|
|
|
|
definition lex.wf (Ha : well_founded Ra) (Hb : well_founded Rb) : well_founded (lex Ra Rb) :=
|
|
|
|
|
well_founded.intro (λp, destruct p (λa b, lex.accessible (Ha a) (well_founded.apply Hb) b))
|
|
|
|
|
|
|
|
|
|
-- Relational product is a subrelation of the lex
|
|
|
|
|
definition rprod.sub_lex : ∀ a b, rprod Ra Rb a b → lex Ra Rb a b :=
|
2015-02-11 20:49:27 +00:00
|
|
|
|
λa b H, prod.rprod.rec_on H (λ a₁ b₁ a₂ b₂ H₁ H₂, lex.left Rb a₂ b₂ H₁)
|
2014-12-06 05:46:17 +00:00
|
|
|
|
|
|
|
|
|
-- The relational product of well founded relations is well-founded
|
|
|
|
|
definition rprod.wf (Ha : well_founded Ra) (Hb : well_founded Rb) : well_founded (rprod Ra Rb) :=
|
|
|
|
|
subrelation.wf (rprod.sub_lex) (lex.wf Ha Hb)
|
|
|
|
|
|
|
|
|
|
end
|
2014-12-12 18:17:50 +00:00
|
|
|
|
|
2014-12-06 05:46:17 +00:00
|
|
|
|
end prod
|
2015-05-01 21:26:31 +00:00
|
|
|
|
|
|
|
|
|
/- logic using prod and sum -/
|
|
|
|
|
|
|
|
|
|
variables {a b c d : Type}
|
|
|
|
|
open prod sum unit
|
|
|
|
|
|
|
|
|
|
/- prod -/
|
|
|
|
|
|
|
|
|
|
definition not_prod_of_not_left (b : Type) (Hna : ¬a) : ¬(a × b) :=
|
|
|
|
|
assume H : a × b, absurd (pr1 H) Hna
|
|
|
|
|
|
|
|
|
|
definition not_prod_of_not_right (a : Type) {b : Type} (Hnb : ¬b) : ¬(a × b) :=
|
|
|
|
|
assume H : a × b, absurd (pr2 H) Hnb
|
|
|
|
|
|
|
|
|
|
definition prod.swap (H : a × b) : b × a :=
|
|
|
|
|
pair (pr2 H) (pr1 H)
|
|
|
|
|
|
|
|
|
|
definition prod_of_prod_of_imp_of_imp (H₁ : a × b) (H₂ : a → c) (H₃ : b → d) : c × d :=
|
|
|
|
|
by cases H₁ with aa bb; exact (H₂ aa, H₃ bb)
|
|
|
|
|
|
|
|
|
|
definition prod_of_prod_of_imp_left (H₁ : a × c) (H : a → b) : b × c :=
|
|
|
|
|
by cases H₁ with aa cc; exact (H aa, cc)
|
|
|
|
|
|
|
|
|
|
definition prod_of_prod_of_imp_right (H₁ : c × a) (H : a → b) : c × b :=
|
|
|
|
|
by cases H₁ with cc aa; exact (cc, H aa)
|
|
|
|
|
|
|
|
|
|
definition prod.comm : a × b ↔ b × a :=
|
|
|
|
|
iff.intro (λH, prod.swap H) (λH, prod.swap H)
|
|
|
|
|
|
|
|
|
|
definition prod.assoc : (a × b) × c ↔ a × (b × c) :=
|
|
|
|
|
iff.intro
|
|
|
|
|
(assume H, pair
|
|
|
|
|
(pr1 (pr1 H))
|
|
|
|
|
(pair (pr2 (pr1 H)) (pr2 H)))
|
|
|
|
|
(assume H, pair
|
|
|
|
|
(pair (pr1 H) (pr1 (pr2 H)))
|
|
|
|
|
(pr2 (pr2 H)))
|
|
|
|
|
|
|
|
|
|
definition prod_unit (a : Type) : a × unit ↔ a :=
|
|
|
|
|
iff.intro (assume H, pr1 H) (assume H, pair H star)
|
|
|
|
|
|
|
|
|
|
definition unit_prod (a : Type) : unit × a ↔ a :=
|
|
|
|
|
iff.intro (assume H, pr2 H) (assume H, pair star H)
|
|
|
|
|
|
2015-05-05 22:05:07 +00:00
|
|
|
|
definition prod_empty (a : Type) : a × empty ↔ empty :=
|
2015-05-01 21:26:31 +00:00
|
|
|
|
iff.intro (assume H, pr2 H) (assume H, !empty.elim H)
|
|
|
|
|
|
|
|
|
|
definition empty_prod (a : Type) : empty × a ↔ empty :=
|
|
|
|
|
iff.intro (assume H, pr1 H) (assume H, !empty.elim H)
|
|
|
|
|
|
|
|
|
|
definition prod_self (a : Type) : a × a ↔ a :=
|
|
|
|
|
iff.intro (assume H, pr1 H) (assume H, pair H H)
|
|
|
|
|
|
|
|
|
|
/- sum -/
|
|
|
|
|
|
|
|
|
|
definition not_sum (Hna : ¬a) (Hnb : ¬b) : ¬(a ⊎ b) :=
|
|
|
|
|
assume H : a ⊎ b, sum.rec_on H
|
|
|
|
|
(assume Ha, absurd Ha Hna)
|
|
|
|
|
(assume Hb, absurd Hb Hnb)
|
|
|
|
|
|
|
|
|
|
definition sum_of_sum_of_imp_of_imp (H₁ : a ⊎ b) (H₂ : a → c) (H₃ : b → d) : c ⊎ d :=
|
|
|
|
|
sum.rec_on H₁
|
|
|
|
|
(assume Ha : a, sum.inl (H₂ Ha))
|
|
|
|
|
(assume Hb : b, sum.inr (H₃ Hb))
|
|
|
|
|
|
|
|
|
|
definition sum_of_sum_of_imp_left (H₁ : a ⊎ c) (H : a → b) : b ⊎ c :=
|
|
|
|
|
sum.rec_on H₁
|
|
|
|
|
(assume H₂ : a, sum.inl (H H₂))
|
|
|
|
|
(assume H₂ : c, sum.inr H₂)
|
|
|
|
|
|
|
|
|
|
definition sum_of_sum_of_imp_right (H₁ : c ⊎ a) (H : a → b) : c ⊎ b :=
|
|
|
|
|
sum.rec_on H₁
|
|
|
|
|
(assume H₂ : c, sum.inl H₂)
|
|
|
|
|
(assume H₂ : a, sum.inr (H H₂))
|
|
|
|
|
|
|
|
|
|
definition sum.elim3 (H : a ⊎ b ⊎ c) (Ha : a → d) (Hb : b → d) (Hc : c → d) : d :=
|
|
|
|
|
sum.rec_on H Ha (assume H₂, sum.rec_on H₂ Hb Hc)
|
|
|
|
|
|
|
|
|
|
definition sum_resolve_right (H₁ : a ⊎ b) (H₂ : ¬a) : b :=
|
|
|
|
|
sum.rec_on H₁ (assume Ha, absurd Ha H₂) (assume Hb, Hb)
|
|
|
|
|
|
|
|
|
|
definition sum_resolve_left (H₁ : a ⊎ b) (H₂ : ¬b) : a :=
|
|
|
|
|
sum.rec_on H₁ (assume Ha, Ha) (assume Hb, absurd Hb H₂)
|
|
|
|
|
|
|
|
|
|
definition sum.swap (H : a ⊎ b) : b ⊎ a :=
|
|
|
|
|
sum.rec_on H (assume Ha, sum.inr Ha) (assume Hb, sum.inl Hb)
|
|
|
|
|
|
|
|
|
|
definition sum.comm : a ⊎ b ↔ b ⊎ a :=
|
|
|
|
|
iff.intro (λH, sum.swap H) (λH, sum.swap H)
|
|
|
|
|
|
|
|
|
|
definition sum.assoc : (a ⊎ b) ⊎ c ↔ a ⊎ (b ⊎ c) :=
|
|
|
|
|
iff.intro
|
|
|
|
|
(assume H, sum.rec_on H
|
|
|
|
|
(assume H₁, sum.rec_on H₁
|
|
|
|
|
(assume Ha, sum.inl Ha)
|
|
|
|
|
(assume Hb, sum.inr (sum.inl Hb)))
|
|
|
|
|
(assume Hc, sum.inr (sum.inr Hc)))
|
|
|
|
|
(assume H, sum.rec_on H
|
|
|
|
|
(assume Ha, (sum.inl (sum.inl Ha)))
|
|
|
|
|
(assume H₁, sum.rec_on H₁
|
|
|
|
|
(assume Hb, sum.inl (sum.inr Hb))
|
|
|
|
|
(assume Hc, sum.inr Hc)))
|
|
|
|
|
|
|
|
|
|
definition sum_unit (a : Type) : a ⊎ unit ↔ unit :=
|
|
|
|
|
iff.intro (assume H, star) (assume H, sum.inr H)
|
|
|
|
|
|
|
|
|
|
definition unit_sum (a : Type) : unit ⊎ a ↔ unit :=
|
|
|
|
|
iff.intro (assume H, star) (assume H, sum.inl H)
|
|
|
|
|
|
|
|
|
|
definition sum_empty (a : Type) : a ⊎ empty ↔ a :=
|
|
|
|
|
iff.intro
|
|
|
|
|
(assume H, sum.rec_on H (assume H1 : a, H1) (assume H1 : empty, !empty.elim H1))
|
|
|
|
|
(assume H, sum.inl H)
|
|
|
|
|
|
|
|
|
|
definition empty_sum (a : Type) : empty ⊎ a ↔ a :=
|
|
|
|
|
iff.intro
|
|
|
|
|
(assume H, sum.rec_on H (assume H1 : empty, !empty.elim H1) (assume H1 : a, H1))
|
|
|
|
|
(assume H, sum.inr H)
|
|
|
|
|
|
|
|
|
|
definition sum_self (a : Type) : a ⊎ a ↔ a :=
|
|
|
|
|
iff.intro
|
|
|
|
|
(assume H, sum.rec_on H (assume H1, H1) (assume H1, H1))
|
|
|
|
|
(assume H, sum.inl H)
|
2015-11-21 04:58:06 +00:00
|
|
|
|
|
|
|
|
|
/- TODO
|
|
|
|
|
theorem sum.right_comm (a b c : Type) : (a + b) + c ↔ (a + c) + b :=
|
|
|
|
|
calc
|
|
|
|
|
(a + b) + c ↔ a + (b + c) : sum.assoc
|
|
|
|
|
... ↔ a + (c + b) : {sum.comm}
|
|
|
|
|
... ↔ (a + c) + b : iff.symm sum.assoc
|
|
|
|
|
|
|
|
|
|
theorem sum.left_comm (a b c : Type) : a + (b + c) ↔ b + (a + c) :=
|
|
|
|
|
calc
|
|
|
|
|
a + (b + c) ↔ (a + b) + c : iff.symm sum.assoc
|
|
|
|
|
... ↔ (b + a) + c : {sum.comm}
|
|
|
|
|
... ↔ b + (a + c) : sum.assoc
|
|
|
|
|
|
|
|
|
|
theorem prod.right_comm (a b c : Type) : (a × b) × c ↔ (a × c) × b :=
|
|
|
|
|
calc
|
|
|
|
|
(a × b) × c ↔ a × (b × c) : prod.assoc
|
|
|
|
|
... ↔ a × (c × b) : _
|
|
|
|
|
... ↔ (a × c) × b : iff.symm prod.assoc
|
|
|
|
|
|
|
|
|
|
theorem prod_not_self_iff {a : Type} : a × ¬ a ↔ false :=
|
|
|
|
|
iff.intro (assume H, (prod.right H) (prod.left H)) (assume H, false.elim H)
|
|
|
|
|
|
|
|
|
|
theorem not_prod_self_iff {a : Type} : ¬ a × a ↔ false :=
|
|
|
|
|
!prod.comm ▸ !prod_not_self_iff
|
|
|
|
|
|
|
|
|
|
theorem prod.left_comm [simp] (a b c : Type) : a × (b × c) ↔ b × (a × c) :=
|
|
|
|
|
calc
|
|
|
|
|
a × (b × c) ↔ (a × b) × c : iff.symm prod.assoc
|
|
|
|
|
... ↔ (b × a) × c : {prod.comm}
|
|
|
|
|
... ↔ b × (a × c) : prod.assoc
|
|
|
|
|
-/
|
|
|
|
|
|
|
|
|
|
theorem imp.syl (H : a → b) (H₂ : c → a) (Hc : c) : b :=
|
|
|
|
|
H (H₂ Hc)
|
|
|
|
|
|
|
|
|
|
theorem sum.imp_distrib : ((a + b) → c) ↔ ((a → c) × (b → c)) :=
|
|
|
|
|
iff.intro
|
|
|
|
|
(λH, prod.mk (imp.syl H sum.inl) (imp.syl H sum.inr))
|
|
|
|
|
(prod.rec sum.rec)
|
|
|
|
|
|
|
|
|
|
theorem not_sum_iff_not_prod_not {a b : Type} : ¬(a + b) ↔ ¬a × ¬b :=
|
|
|
|
|
sum.imp_distrib
|