2014-07-30 10:43:47 -07:00
|
|
|
|
----------------------------------------------------------------------------------------------------
|
|
|
|
|
--- Copyright (c) 2014 Microsoft Corporation. All rights reserved.
|
|
|
|
|
--- Released under Apache 2.0 license as described in the file LICENSE.
|
|
|
|
|
--- Author: Jeremy Avigad
|
|
|
|
|
----------------------------------------------------------------------------------------------------
|
|
|
|
|
|
2015-07-06 15:05:01 -07:00
|
|
|
|
import logic
|
2014-09-04 18:41:06 -07:00
|
|
|
|
open eq
|
2014-09-03 16:00:38 -07:00
|
|
|
|
open function
|
2014-07-30 10:43:47 -07:00
|
|
|
|
|
|
|
|
|
namespace congruence
|
|
|
|
|
|
|
|
|
|
-- TODO: move this somewhere else
|
2014-09-29 08:18:10 -07:00
|
|
|
|
definition reflexive {T : Type} (R : T → T → Prop) : Prop := ∀x, R x x
|
2014-07-30 10:43:47 -07:00
|
|
|
|
|
|
|
|
|
-- Congruence classes for unary and binary functions
|
|
|
|
|
-- -------------------------------------------------
|
|
|
|
|
|
2014-10-07 18:02:15 -07:00
|
|
|
|
inductive congruence [class] {T1 : Type} {T2 : Type} (R1 : T1 → T1 → Prop) (R2 : T2 → T2 → Prop)
|
2014-07-30 10:43:47 -07:00
|
|
|
|
(f : T1 → T2) : Prop :=
|
2014-08-22 15:46:10 -07:00
|
|
|
|
mk : (∀x y : T1, R1 x y → R2 (f x) (f y)) → congruence R1 R2 f
|
2014-07-30 10:43:47 -07:00
|
|
|
|
|
|
|
|
|
-- to trigger class inference
|
|
|
|
|
theorem congr_app {T1 : Type} {T2 : Type} (R1 : T1 → T1 → Prop) (R2 : T2 → T2 → Prop)
|
|
|
|
|
(f : T1 → T2) {C : congruence R1 R2 f} {x y : T1} : R1 x y → R2 (f x) (f y) :=
|
2014-09-04 15:03:59 -07:00
|
|
|
|
congruence.rec id C x y
|
2014-07-30 10:43:47 -07:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
-- General tools to build instances
|
|
|
|
|
-- --------------------------------
|
|
|
|
|
|
|
|
|
|
theorem congr_trivial [instance] {T : Type} (R : T → T → Prop) : congruence R R id :=
|
2014-09-04 16:36:06 -07:00
|
|
|
|
congruence.mk (take x y H, H)
|
2014-07-30 10:43:47 -07:00
|
|
|
|
|
|
|
|
|
theorem congr_const {T2 : Type} (R2 : T2 → T2 → Prop) (H : reflexive R2) :
|
|
|
|
|
∀(T1 : Type) (R1 : T1 → T1 → Prop) (c : T2), congruence R1 R2 (const T1 c) :=
|
2014-09-04 16:36:06 -07:00
|
|
|
|
take T1 R1 c, congruence.mk (take x y H1, H c)
|
2014-07-30 10:43:47 -07:00
|
|
|
|
|
|
|
|
|
-- congruences for logic
|
|
|
|
|
|
|
|
|
|
theorem congr_const_iff [instance] (T1 : Type) (R1 : T1 → T1 → Prop) (c : Prop) :
|
2014-09-04 21:25:21 -07:00
|
|
|
|
congruence R1 iff (const T1 c) := congr_const iff iff.refl T1 R1 c
|
2014-07-30 10:43:47 -07:00
|
|
|
|
|
|
|
|
|
theorem congr_or [instance] (T : Type) (R : T → T → Prop) (f1 f2 : T → Prop)
|
2015-02-24 16:10:16 -08:00
|
|
|
|
[H1 : congruence R iff f1] [H2 : congruence R iff f2] :
|
2014-07-30 10:43:47 -07:00
|
|
|
|
congruence R iff (λx, f1 x ∨ f2 x) := sorry
|
|
|
|
|
|
|
|
|
|
theorem congr_implies [instance] (T : Type) (R : T → T → Prop) (f1 f2 : T → Prop)
|
2015-02-24 16:10:16 -08:00
|
|
|
|
[H1 : congruence R iff f1] [H2 : congruence R iff f2] :
|
2014-07-30 10:43:47 -07:00
|
|
|
|
congruence R iff (λx, f1 x → f2 x) := sorry
|
|
|
|
|
|
|
|
|
|
theorem congr_iff [instance] (T : Type) (R : T → T → Prop) (f1 f2 : T → Prop)
|
2015-02-24 16:10:16 -08:00
|
|
|
|
[H1 : congruence R iff f1] [H2 : congruence R iff f2] :
|
2014-07-30 10:43:47 -07:00
|
|
|
|
congruence R iff (λx, f1 x ↔ f2 x) := sorry
|
|
|
|
|
|
|
|
|
|
theorem congr_not [instance] (T : Type) (R : T → T → Prop) (f : T → Prop)
|
2015-02-24 16:10:16 -08:00
|
|
|
|
[H : congruence R iff f] :
|
2014-07-30 10:43:47 -07:00
|
|
|
|
congruence R iff (λx, ¬ f x) := sorry
|
|
|
|
|
|
2014-10-12 13:06:00 -07:00
|
|
|
|
theorem subst_iff {T : Type} {R : T → T → Prop} {P : T → Prop} [C : congruence R iff P]
|
2014-07-30 10:43:47 -07:00
|
|
|
|
{a b : T} (H : R a b) (H1 : P a) : P b :=
|
2014-09-04 15:03:59 -07:00
|
|
|
|
-- iff_mp_left (congruence.rec id C a b H) H1
|
2014-09-04 21:25:21 -07:00
|
|
|
|
iff.elim_left (@congr_app _ _ R iff P C a b H) H1
|
2014-07-30 10:43:47 -07:00
|
|
|
|
|
2014-09-03 16:00:38 -07:00
|
|
|
|
end congruence
|