From 1ec535d5237e3eb8725285b07e26d686ace28090 Mon Sep 17 00:00:00 2001 From: Leonardo de Moura Date: Wed, 23 Nov 2022 18:45:17 -0800 Subject: [PATCH] test: alternative encoding experiment for `decEq` and `noConfusion` --- tests/playground/noConfusionDecEqExp.lean | 97 +++++++++++++++++++++++ 1 file changed, 97 insertions(+) create mode 100644 tests/playground/noConfusionDecEqExp.lean diff --git a/tests/playground/noConfusionDecEqExp.lean b/tests/playground/noConfusionDecEqExp.lean new file mode 100644 index 0000000000..2e6c04646e --- /dev/null +++ b/tests/playground/noConfusionDecEqExp.lean @@ -0,0 +1,97 @@ +inductive Foo (α : Type u) where + | mk1 (val : α) + | mk2 (left : Foo α) (right : Foo α) + | mk3 (val : Nat) + | mk4 (val : String) + | mk5 (head : α) (tail : Foo α) + +def Foo.ctorIdx : Foo α → Nat + | .mk1 .. => 0 + | .mk2 .. => 1 + | .mk3 .. => 2 + | .mk4 .. => 3 + | .mk5 .. => 4 + +@[elab_as_elim] +def Foo.elimCtor1 {motive : Foo α → Sort v} (a : Foo α) (hIdx : a.ctorIdx == 0) (h : (val : α) → motive (Foo.mk1 val)) : motive a := + match a with + | .mk1 a => h a + | .mk2 .. => Bool.noConfusion hIdx + | .mk3 .. => Bool.noConfusion hIdx + | .mk4 .. => Bool.noConfusion hIdx + | .mk5 .. => Bool.noConfusion hIdx + +@[elab_as_elim] +def Foo.elimCtor2 {motive : Foo α → Sort v} (a : Foo α) (hIdx : a.ctorIdx == 1) (h : (left : Foo α) → (right : Foo α) → motive (Foo.mk2 left right)) : motive a := + match a with + | .mk1 .. => Bool.noConfusion hIdx + | .mk2 left right => h left right + | .mk3 .. => Bool.noConfusion hIdx + | .mk4 .. => Bool.noConfusion hIdx + | .mk5 .. => Bool.noConfusion hIdx + +@[elab_as_elim] +def Foo.elimCtor3 {motive : Foo α → Sort v} (a : Foo α) (hIdx : a.ctorIdx == 2) (h : (val : Nat) → motive (Foo.mk3 val)) : motive a := + match a with + | .mk1 .. => Bool.noConfusion hIdx + | .mk2 .. => Bool.noConfusion hIdx + | .mk3 val => h val + | .mk4 .. => Bool.noConfusion hIdx + | .mk5 .. => Bool.noConfusion hIdx + +@[elab_as_elim] +def Foo.elimCtor4 {motive : Foo α → Sort v} (a : Foo α) (hIdx : a.ctorIdx == 3) (h : (val : String) → motive (Foo.mk4 val)) : motive a := + match a with + | .mk1 .. => Bool.noConfusion hIdx + | .mk2 .. => Bool.noConfusion hIdx + | .mk3 .. => Bool.noConfusion hIdx + | .mk4 val => h val + | .mk5 .. => Bool.noConfusion hIdx + +@[elab_as_elim] +def Foo.elimCtor5 {motive : Foo α → Sort v} (a : Foo α) (hIdx : a.ctorIdx == 4) (h : (head : α) → (tail : Foo α) → motive (Foo.mk5 head tail)) : motive a := + match a with + | .mk1 .. => Bool.noConfusion hIdx + | .mk2 .. => Bool.noConfusion hIdx + | .mk3 .. => Bool.noConfusion hIdx + | .mk4 .. => Bool.noConfusion hIdx + | .mk5 head tail => h head tail + +@[reducible] def Foo.noConfusionType' {α : Type u} (P : Sort v) (a b : Foo α) : Sort v := + if h : b.ctorIdx == a.ctorIdx then + match a with + | .mk1 val1 => Foo.elimCtor1 (motive := fun _ => Sort v) b h (fun val2 => (val1 = val2 → P) → P) + | .mk2 left1 right1 => Foo.elimCtor2 (motive := fun _ => Sort v) b h (fun left2 right2 => (left1 = left2 → right1 = right2 → P) → P) + | .mk3 val1 => Foo.elimCtor3 (motive := fun _ => Sort v) b h (fun val2 => (val1 = val2 → P) → P) + | .mk4 val1 => Foo.elimCtor4 (motive := fun _ => Sort v) b h (fun val2 => (val1 = val2 → P) → P) + | .mk5 head1 tail1 => Foo.elimCtor5 (motive := fun _ => Sort v) b h (fun head2 tail2 => (head1 = head2 → tail1 = tail2 → P) → P) + else + P + +@[reducible] def Foo.noConfusion' {α : Type u} {P : Sort u_1} {v1 v2 : Foo α} (h : v1 = v2) : Foo.noConfusionType' P v1 v2 := + Eq.ndrec (motive := fun a => v1 = a → Foo.noConfusionType' P v1 a) + (fun (_ : v1 = v1) => + Foo.casesOn v1 (fun _ h => h rfl) (fun _ _ h => h rfl rfl) (fun _ h => h rfl) (fun _ h => h rfl) (fun _ _ h => h rfl rfl)) + h h + +def Foo.decEq {α : Type u} [DecidableEq α] (a b : Foo α) : Decidable (a = b) := + if hc : b.ctorIdx == a.ctorIdx then + match a with + | mk1 val1 => Foo.elimCtor1 b hc fun val2 => if he : val1 = val2 then isTrue (he ▸ rfl) else isFalse (fun h => Foo.noConfusion' h fun h => he h) + | mk2 left1 right1 => Foo.elimCtor2 b hc fun left2 right2 => + match decEq left1 left2 with + | isTrue he1 => match decEq right1 right2 with + | isTrue he2 => isTrue (he1 ▸ he2 ▸ rfl) + | isFalse he2 => isFalse (fun h => Foo.noConfusion' h fun _ h => he2 h) + | isFalse he1 => isFalse (fun h => Foo.noConfusion' h fun h _ => he1 h) + | mk3 val1 => Foo.elimCtor3 b hc fun val2 => if he : val1 = val2 then isTrue (he ▸ rfl) else isFalse (fun h => Foo.noConfusion' h fun h => he h) + | mk4 val1 => Foo.elimCtor4 b hc fun val2 => if he : val1 = val2 then isTrue (he ▸ rfl) else isFalse (fun h => Foo.noConfusion' h fun h => he h) + | mk5 head1 tail1 => Foo.elimCtor5 b hc fun head2 tail2 => + if he1 : head1 = head2 then + match decEq tail1 tail2 with + | isTrue he2 => isTrue (he1 ▸ he2 ▸ rfl) + | isFalse he2 => isFalse (fun h => Foo.noConfusion' h fun _ h => he2 h) + else + isFalse (fun h => Foo.noConfusion' h fun h _ => he1 h) + else + isFalse (fun h => have : ¬ b.ctorIdx == b.ctorIdx := h ▸ hc; this LawfulBEq.rfl)