Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
110 commits
Select commit Hold shift + click to select a range
0fb6da6
reduction system attribute
Jul 22, 2025
62b1776
evaluation results
Jul 22, 2025
b11c414
injectivity, rice's theorem
Jul 22, 2025
86c8407
rice's theorem
Sep 5, 2025
7df1910
documentation
Sep 5, 2025
957e3d9
Merge remote-tracking branch 'origin/rice-theorem'
Sep 8, 2025
e693e05
Add proofs for tensor-zero and parr-top equivalences (#15)
m-ow Jul 21, 2025
6993f88
Locally Nameless Beta Confluence (#11)
chenson2018 Jul 21, 2025
175ac3a
Testing CCS
fmontesi Jul 22, 2025
47c7db8
arrow shortcuts
fmontesi Jul 22, 2025
8ba26a3
better CCS example
fmontesi Jul 22, 2025
db126c8
Switch to the standard type for relations. (#16)
fmontesi Jul 23, 2025
8efce25
Make calc work with LTS
fmontesi Jul 23, 2025
2f588e0
Remove some simps and a commented out def
fmontesi Jul 23, 2025
acb0c94
Adopt \sup and Eq instead of custom defs
fmontesi Jul 25, 2025
8e34a71
Activate mathlib linters and fix some warnings. (#19)
fmontesi Jul 25, 2025
174a932
Trans instances for behavioural equivalences
fmontesi Jul 25, 2025
4585fe4
Remove Relation.inv in favour of the built-in flip
fmontesi Jul 26, 2025
84115c1
Bisimilarity life improvements (use Subrelation and implicit in large…
fmontesi Jul 26, 2025
60d00fc
Bisimilarity: better implicits
fmontesi Jul 26, 2025
b1e5e73
Bisimulations equipped with union form a join-semilattice and a bound…
fmontesi Jul 26, 2025
9698145
minor moving around
fmontesi Jul 26, 2025
f351e36
minor moving around
fmontesi Jul 26, 2025
cc122d3
Bisimilarity is included in trace equivalence
fmontesi Jul 26, 2025
4a4f267
Docs fixes (#21)
chenson2018 Jul 27, 2025
742627a
upgrade lean-toolchain
fmontesi Jul 27, 2025
4fab70b
Check the docs toolchain is up to date
fmontesi Jul 27, 2025
a5b81ca
rename LTS to Lts
fmontesi Jul 27, 2025
90d312d
checkout before checking toolchains
fmontesi Jul 27, 2025
02ab4e8
checkout in the same job
fmontesi Jul 27, 2025
d034d8c
remove useless parens
fmontesi Jul 27, 2025
9ab1bc0
Class for well-formedness. (#22)
fmontesi Jul 28, 2025
9e74096
dep bumps
fmontesi Jul 28, 2025
8d44d55
fixes for Rel
fmontesi Jul 28, 2025
3abfe96
docgen4 bump
fmontesi Jul 28, 2025
ca10096
Sorry doc(s), you should be ok now
fmontesi Jul 28, 2025
50e3365
bib fixes
fmontesi Jul 28, 2025
730a236
bib fixes
fmontesi Jul 28, 2025
ae0207a
fix links
fmontesi Jul 28, 2025
bcf7938
grind is awesome
fmontesi Jul 28, 2025
eb8ac4b
Use parent namespace in lts and reduction_sys attributes (#24)
chenson2018 Jul 30, 2025
56fdfeb
try fixing the linter action
fmontesi Jul 30, 2025
8369ceb
fix linting action
fmontesi Jul 30, 2025
d5fd4e2
begin parassoc
fmontesi Jul 28, 2025
49da602
some proof_wanted in CCS
fmontesi Jul 31, 2025
276144a
minor move
fmontesi Jul 31, 2025
2a3dbc1
use notation3 for Lts and ReductionSystem (#25)
chenson2018 Aug 4, 2025
01b9d00
fix linting again
fmontesi Aug 4, 2025
1253ae4
Locally nameless STLC (#17)
chenson2018 Aug 5, 2025
e1da3c1
All Lints Corrected! (#26)
chenson2018 Aug 5, 2025
23ddc19
add wfail option to CI build (#27)
chenson2018 Aug 5, 2025
00a438a
Toolchain v4.22.0 (#31)
chenson2018 Aug 14, 2025
13d5d5b
feat(HasFresh): characterize (#29)
tristan-f-r Aug 14, 2025
276f02e
move to leanprover
fmontesi Aug 15, 2025
55f88dc
Fix type name in CONTRIBUTING.md (#33)
fmontesi Aug 15, 2025
05e5310
feat(LinearLogic): logical equivalence is an equivalence relation (#34)
tristan-f-r Aug 16, 2025
2f7bfa9
term elaborator for selecting fresh variables (#32)
chenson2018 Aug 18, 2025
26fa6a0
chore(Data): golf entire `FinFun.congrFinFun` (#36)
euprunin Aug 19, 2025
bfcb34f
Migrate LambdaCalculus.LocallyNameless to `grind` (#35)
chenson2018 Aug 19, 2025
826cabb
Update README.md
fmontesi Aug 20, 2025
4fe6694
chore(Data): golf entire `FinFun.eq_char₁`. golf using grind. remove …
euprunin Aug 20, 2025
9c0aee9
chore(Semantics/Lts): golf `Lts.deterministic_image_char`, `Lts.strN.…
euprunin Aug 22, 2025
4ed0f87
chore(Computability/CombinatoryLogic): golf `RFindAbove_correct` usin…
euprunin Aug 22, 2025
df2bea9
Allow multiple maps in `free_union` (#46)
chenson2018 Aug 25, 2025
ba25dd7
add type paramater to HasSubstitution (#45)
chenson2018 Aug 25, 2025
08689a8
chore(Semantics/Lts): golf `Bisimilarity.refl` and `Bisimulation.simu…
euprunin Aug 25, 2025
2145d1a
chore(Semantics/Lts): golf `SimulationEquiv.refl`, `SimulationEquiv.s…
euprunin Aug 26, 2025
1831471
fix: `LambdaCalculus.Named.Term.subst` scoping problem (#48)
thelissimus Aug 26, 2025
89945fe
Fix citation
fmontesi Aug 26, 2025
08ebd48
Further Simulation golfing
fmontesi Aug 26, 2025
b7add91
CLL equivalences and substitution of equivalent formulas (#49)
jpyamamoto Aug 27, 2025
ea6fc77
Reorg of directories
fmontesi Aug 27, 2025
69ad85b
Reorg: fix imports
fmontesi Aug 27, 2025
11112da
update CODEOWNERS (#51)
chenson2018 Aug 27, 2025
4c329db
Notation for duality in CLL
fmontesi Aug 28, 2025
3bb54f3
use notation for ?
fmontesi Aug 28, 2025
8edf9ee
Some notation fixes and statement of cut admissibility and elimination
fmontesi Aug 28, 2025
3d7dc8c
add to lakefile.toml (#52)
chenson2018 Aug 28, 2025
2b32fea
scope some grinds
fmontesi Aug 29, 2025
c2ce28d
grinding grind on CCS
fmontesi Aug 29, 2025
86e9fab
Governance
fmontesi Sep 3, 2025
d0e4a5e
chore: use grind in CCS.BehaviouralTheory (#54)
kim-em Sep 3, 2025
b3ce876
Make lts look nice
fmontesi Sep 3, 2025
6644835
small update in affiliation (#56)
arademaker Sep 3, 2025
dd02d60
Added Boole directory
barrettcw Sep 5, 2025
8e94a0d
Fix CLL exchange, add HasSize, add cut'
fmontesi Sep 3, 2025
6e7ce4a
Bisimulation -> IsBisimulation and lots of grinding
fmontesi Sep 8, 2025
009def2
Better support for dot-notation in IsBisimulation
fmontesi Sep 8, 2025
7c9a1ba
reduction system attribute
Jul 22, 2025
75ad098
evaluation results
Jul 22, 2025
8b32be6
fix imports, build
Sep 8, 2025
8591e57
linting
Sep 8, 2025
6b1a05b
spaces after transitions
Sep 10, 2025
b03bb3a
documentation
Sep 24, 2025
752d2c5
dot notation
Sep 24, 2025
1aa92ab
more descriptive names
Sep 27, 2025
d7ba53a
lint
Sep 27, 2025
3473f87
fix imports
Oct 6, 2025
7f63756
Merge branch 'main' into rice-theorem
thomaskwaring Oct 6, 2025
26c3296
fix spaces in notation
Oct 6, 2025
97f9495
fix lake-manifest in docs
thomaskwaring Oct 15, 2025
26fdf29
fix other lake-manifest
thomaskwaring Oct 15, 2025
7a6d62e
fix again?
Oct 15, 2025
b7152df
once more with feeling
Oct 15, 2025
cb2365e
namespace Evaluation
thomaskwaring Oct 15, 2025
13c3c7f
take manifest from main
chenson2018 Oct 15, 2025
014a5f3
rm unicode id
Oct 15, 2025
0984610
Merge branch 'rice-theorem' of https://github.com/thomaskwaring/cslib…
Oct 15, 2025
de53b60
fix namespaces
Oct 15, 2025
30ba08f
remove commented-out code
Oct 15, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
22 changes: 13 additions & 9 deletions Cslib/Foundations/Semantics/ReductionSystem/Basic.lean
Original file line number Diff line number Diff line change
Expand Up @@ -46,11 +46,15 @@ open Relation Relation.ReflTransGen
instance (rs : ReductionSystem Term) : Trans rs.Red rs.MRed rs.MRed := by infer_instance
instance (rs : ReductionSystem Term) : Trans rs.MRed rs.Red rs.MRed := by infer_instance

instance (rs : ReductionSystem Term) : IsTrans Term rs.MRed := by infer_instance
instance (rs : ReductionSystem Term) : Transitive rs.MRed := transitive_of_trans rs.MRed
instance (rs : ReductionSystem Term) : Trans rs.MRed rs.MRed rs.MRed := instTransOfIsTrans

end MultiStep

open Lean Elab Meta Command Term

-- thank you to Kyle Miller for this:
-- thank you to Kyle Miller for this:
-- https://leanprover.zulipchat.com/#narrow/channel/239415-metaprogramming-.2F-tactics/topic/Working.20with.20variables.20in.20a.20command/near/529324084

/-- A command to create a `ReductionSystem` from a relation, robust to use of `variable `-/
Expand Down Expand Up @@ -81,10 +85,10 @@ elab "create_reduction_sys" rel:ident name:ident : command => do
}
addTermInfo' name (.const name.getId params) (isBinder := true)
addDeclarationRangesFromSyntax name.getId name
/--

/--
This command adds notations for a `ReductionSystem.Red` and `ReductionSystem.MRed`. This should
not usually be called directly, but from the `reduction_sys` attribute.
not usually be called directly, but from the `reduction_sys` attribute.

As an example `reduction_notation foo "β"` will add the notations "⭢β" and "↠β".

Expand All @@ -94,19 +98,19 @@ elab "create_reduction_sys" rel:ident name:ident : command => do
-/
syntax "reduction_notation" ident (str)? : command
macro_rules
| `(reduction_notation $rs $sym) =>
| `(reduction_notation $rs $sym) =>
`(
notation3 t:39 " ⭢" $sym:str t':39 => (ReductionSystem.Red $rs) t t'
notation3 t:39 " ↠" $sym:str t':39 => (ReductionSystem.MRed $rs) t t'
)
| `(reduction_notation $rs) =>
| `(reduction_notation $rs) =>
`(
notation3 t:39 " ⭢" t':39 => (ReductionSystem.Red $rs) t t'
notation3 t:39 " ↠" t':39 => (ReductionSystem.MRed $rs) t t'
notation3 t:39 " ⭢ " t':39 => (ReductionSystem.Red $rs) t t'
notation3 t:39 " ↠ " t':39 => (ReductionSystem.MRed $rs) t t'
)


/--
/--
This attribute calls the `reduction_notation` command for the annotated declaration, such as in:

```
Expand Down
80 changes: 42 additions & 38 deletions Cslib/Languages/CombinatoryLogic/Basic.lean
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,8 @@ For a presentation of the bracket abstraction algorithm see:
<https://web.archive.org/web/19970727171324/http://www.cs.oberlin.edu/classes/cs280/labs/lab4/lab43.html#@l13>
-/

namespace Cslib

namespace SKI

open Red MRed
Expand Down Expand Up @@ -61,7 +63,7 @@ def Polynomial.eval {n : Nat} (Γ : SKI.Polynomial n) (l : List SKI) (hl : List.
def Polynomial.varFreeToSKI (Γ : SKI.Polynomial 0) : SKI := Γ.eval [] (by trivial)

/-- Inductively define a polynomial `Γ'` so that (up to the fact that we haven't
defined reduction on polynomials) `Γ' ⬝ t ⇒* Γ[xₙ ← t]`. -/
defined reduction on polynomials) `Γ' ⬝ t Γ[xₙ ← t]`. -/
def Polynomial.elimVar {n : Nat} : SKI.Polynomial (n+1) → SKI.Polynomial n
/- The K-combinator leaves plain terms unchanged by substitution `K ⬝ x ⬝ t ⇒ x` -/
| SKI.Polynomial.term x => K ⬝' x
Expand All @@ -83,7 +85,7 @@ for the inner variables.
-/
theorem Polynomial.elimVar_correct {n : Nat} (Γ : SKI.Polynomial (n + 1)) {ys : List SKI}
(hys : ys.length = n) (z : SKI) :
Γ.elimVar.eval ys hys ⬝ z ⇒* Γ.eval (ys ++ [z])
(Γ.elimVar.eval ys hys ⬝ z) ↠ Γ.eval (ys ++ [z])
(by rw [List.length_append, hys, List.length_singleton])
:= by
match n, Γ with
Expand Down Expand Up @@ -125,13 +127,13 @@ def Polynomial.toSKI {n : Nat} (Γ : SKI.Polynomial n) : SKI :=

/-- Correctness for the toSKI (bracket abstraction) algorithm. -/
theorem Polynomial.toSKI_correct {n : Nat} (Γ : SKI.Polynomial n) (xs : List SKI)
(hxs : xs.length = n) : Γ.toSKI.applyList xs ⇒* Γ.eval xs hxs := by
(hxs : xs.length = n) : Γ.toSKI.applyList xs Γ.eval xs hxs := by
match n with
| 0 =>
unfold toSKI varFreeToSKI applyList
rw [List.length_eq_zero_iff] at hxs
simp_rw [hxs, List.foldl_nil]
apply MRed.refl
rfl
| n+1 =>
-- show that xs = ys + [z]
have : xs ≠ [] := List.ne_nil_of_length_eq_add_one hxs
Expand Down Expand Up @@ -165,63 +167,63 @@ choose a descriptive name.
def RPoly : SKI.Polynomial 2 := &1 ⬝' &0
/-- A SKI term representing R -/
def R : SKI := RPoly.toSKI
theorem R_def (x y : SKI) : R ⬝ x ⬝ y ⇒* y ⬝ x :=
theorem R_def (x y : SKI) : (R ⬝ x ⬝ y) ↠ y ⬝ x :=
RPoly.toSKI_correct [x, y] (by simp)


/-- Composition: B := λ f g x. f (g x) -/
def BPoly : SKI.Polynomial 3 := &0 ⬝' (&1 ⬝' &2)
/-- A SKI term representing B -/
def B : SKI := BPoly.toSKI
theorem B_def (f g x : SKI) : B ⬝ f ⬝ g ⬝ x ⇒* f ⬝ (g ⬝ x) :=
theorem B_def (f g x : SKI) : (B ⬝ f ⬝ g ⬝ x) ↠ f ⬝ (g ⬝ x) :=
BPoly.toSKI_correct [f, g, x] (by simp)


/-- C := λ f x y. f y x -/
def CPoly : SKI.Polynomial 3 := &0 ⬝' &2 ⬝' &1
/-- A SKI term representing C -/
def C : SKI := CPoly.toSKI
theorem C_def (f x y : SKI) : C ⬝ f ⬝ x ⬝ y ⇒* f ⬝ y ⬝ x :=
theorem C_def (f x y : SKI) : (C ⬝ f ⬝ x ⬝ y) ↠ f ⬝ y ⬝ x :=
CPoly.toSKI_correct [f, x, y] (by simp)


/-- Rotate right: RotR := λ x y z. z x y -/
def RotRPoly : SKI.Polynomial 3 := &2 ⬝' &0 ⬝' &1
/-- A SKI term representing RotR -/
def RotR : SKI := RotRPoly.toSKI
theorem rotR_def (x y z : SKI) : RotR ⬝ x ⬝ y ⬝ z ⇒* z ⬝ x ⬝ y :=
theorem rotR_def (x y z : SKI) : (RotR ⬝ x ⬝ y ⬝ z) ↠ z ⬝ x ⬝ y :=
RotRPoly.toSKI_correct [x, y, z] (by simp)


/-- Rotate left: RotR := λ x y z. y z x -/
def RotLPoly : SKI.Polynomial 3 := &1 ⬝' &2 ⬝' &0
/-- A SKI term representing RotL -/
def RotL : SKI := RotLPoly.toSKI
theorem rotL_def (x y z : SKI) : RotL ⬝ x ⬝ y ⬝ z ⇒* y ⬝ z ⬝ x :=
theorem rotL_def (x y z : SKI) : (RotL ⬝ x ⬝ y ⬝ z) ↠ y ⬝ z ⬝ x :=
RotLPoly.toSKI_correct [x, y, z] (by simp)


/-- Self application: δ := λ x. x x -/
def δPoly : SKI.Polynomial 1 := &0 ⬝' &0
def DelPoly : SKI.Polynomial 1 := &0 ⬝' &0
/-- A SKI term representing δ -/
def δ : SKI := δPoly.toSKI
theorem δ_def (x : SKI) : δ ⬝ x ⇒* x ⬝ x :=
δPoly.toSKI_correct [x] (by simp)
def Del : SKI := DelPoly.toSKI
theorem del_def (x : SKI) : (Del ⬝ x) ↠ x ⬝ x :=
DelPoly.toSKI_correct [x] (by simp)


/-- H := λ f x. f (x x) -/
def HPoly : SKI.Polynomial 2 := &0 ⬝' (&1 ⬝' &1)
/-- A SKI term representing H -/
def H : SKI := HPoly.toSKI
theorem H_def (f x : SKI) : H ⬝ f ⬝ x ⇒* f ⬝ (x ⬝ x) :=
theorem H_def (f x : SKI) : (H ⬝ f ⬝ x) ↠ f ⬝ (x ⬝ x) :=
HPoly.toSKI_correct [f, x] (by simp)


/-- Curry's fixed-point combinator: Y := λ f. H f (H f) -/
def YPoly : SKI.Polynomial 1 := H ⬝' &0 ⬝' (H ⬝' &0)
/-- A SKI term representing Y -/
def Y : SKI := YPoly.toSKI
theorem Y_def (f : SKI) : Y ⬝ f ⇒* H ⬝ f ⬝ (H ⬝ f) :=
theorem Y_def (f : SKI) : (Y ⬝ f) ↠ H ⬝ f ⬝ (H ⬝ f) :=
YPoly.toSKI_correct [f] (by simp)


Expand All @@ -239,29 +241,29 @@ rather than up to a common reduct. An alternative is to use Turing's fixed-point
(defined below).
-/
def fixedPoint (f : SKI) : SKI := H ⬝ f ⬝ (H ⬝ f)
theorem fixedPoint_correct (f : SKI) : f.fixedPoint ⇒* f ⬝ f.fixedPoint := H_def f (H ⬝ f)
theorem fixedPoint_correct (f : SKI) : f.fixedPoint f ⬝ f.fixedPoint := H_def f (H ⬝ f)

/-- Auxiliary definition for Turing's fixed-point combinator: ΘAux := λ x y. y (x x y) -/
def ΘAuxPoly : SKI.Polynomial 2 := &1 ⬝' (&0 ⬝' &0 ⬝' &1)
def ThAuxPoly : SKI.Polynomial 2 := &1 ⬝' (&0 ⬝' &0 ⬝' &1)
/-- A term representing ΘAux -/
def ΘAux : SKI := ΘAuxPoly.toSKI
theorem ΘAux_def (x y : SKI) : ΘAux ⬝ x ⬝ y ⇒* y ⬝ (x ⬝ x ⬝ y) :=
ΘAuxPoly.toSKI_correct [x, y] (by simp)
def ThAux : SKI := ThAuxPoly.toSKI
theorem ThAux_def (x y : SKI) : (ThAux ⬝ x ⬝ y) ↠ y ⬝ (x ⬝ x ⬝ y) :=
ThAuxPoly.toSKI_correct [x, y] (by simp)


/-- Turing's fixed-point combinator: Θ := (λ x y. y (x x y)) (λ x y. y (x x y)) -/
def Θ : SKI := ΘAuxΘAux
def Th : SKI := ThAuxThAux
/-- A SKI term representing Θ -/
theorem Θ_correct (f : SKI) : Θ ⬝ f ⇒* f ⬝ (Θ ⬝ f) := ΘAux_def ΘAux f
theorem Th_correct (f : SKI) : (Th ⬝ f) ↠ f ⬝ (Th ⬝ f) := ThAux_def ThAux f


/-! ### Church Booleans -/

/-- A term a represents the boolean value u if it is βη-equivalent to a standard Church boolean. -/
def IsBool (u : Bool) (a : SKI) : Prop :=
∀ x y : SKI, a ⬝ x ⬝ y ⇒* (if u then x else y)
∀ x y : SKI, (a ⬝ x ⬝ y) ↠ (if u then x else y)

theorem isBool_trans (u : Bool) (a a' : SKI) (h : a ⇒* a') (ha' : IsBool u a') :
theorem isBool_trans (u : Bool) (a a' : SKI) (h : a a') (ha' : IsBool u a') :
IsBool u a := by
intro x y
trans a' ⬝ x ⬝ y
Expand All @@ -278,13 +280,13 @@ theorem TT_correct : IsBool true TT := fun x y ↦ MRed.K x y
def FF : SKI := K ⬝ I
theorem FF_correct : IsBool false FF :=
fun x y ↦ calc
FF ⬝ x ⬝ y I ⬝ y := by apply red_head; exact red_K I x
_ y := red_I y
(FF ⬝ x ⬝ y) ↠ I ⬝ y := by apply Relation.ReflTransGen.single; apply red_head; exact red_K I x
_ y := red_I y

/-- Conditional: Cond x y b := if b then x else y -/
protected def Cond : SKI := RotR
theorem cond_correct (a x y : SKI) (u : Bool) (h : IsBool u a) :
SKI.Cond ⬝ x ⬝ y ⬝ a ⇒* if u then x else y := by
(SKI.Cond ⬝ x ⬝ y ⬝ a) ↠ if u then x else y := by
trans a ⬝ x ⬝ y
· exact rotR_def x y a
· exact h x y
Expand All @@ -302,7 +304,7 @@ theorem neg_correct (a : SKI) (ua : Bool) (h : IsBool ua a) : IsBool (¬ ua) (SK
def AndPoly : SKI.Polynomial 2 := SKI.Cond ⬝' (SKI.Cond ⬝ TT ⬝ FF ⬝' &1) ⬝' FF ⬝' &0
/-- A SKI term representing And -/
protected def And : SKI := AndPoly.toSKI
theorem and_def (a b : SKI) : SKI.And ⬝ a ⬝ b ⇒* SKI.Cond ⬝ (SKI.Cond ⬝ TT ⬝ FF ⬝ b) ⬝ FF ⬝ a :=
theorem and_def (a b : SKI) : (SKI.And ⬝ a ⬝ b) ↠ SKI.Cond ⬝ (SKI.Cond ⬝ TT ⬝ FF ⬝ b) ⬝ FF ⬝ a :=
AndPoly.toSKI_correct [a, b] (by simp)

theorem and_correct (a b : SKI) (ua ub : Bool) (ha : IsBool ua a) (hb : IsBool ub b) :
Expand All @@ -321,7 +323,7 @@ theorem and_correct (a b : SKI) (ua ub : Bool) (ha : IsBool ua a) (hb : IsBool u
def OrPoly : SKI.Polynomial 2 := SKI.Cond ⬝' TT ⬝' (SKI.Cond ⬝ TT ⬝ FF ⬝' &1) ⬝' &0
/-- A SKI term representing Or -/
protected def Or : SKI := OrPoly.toSKI
theorem or_def (a b : SKI) : SKI.Or ⬝ a ⬝ b ⇒* SKI.Cond ⬝ TT ⬝ (SKI.Cond ⬝ TT ⬝ FF ⬝ b) ⬝ a :=
theorem or_def (a b : SKI) : (SKI.Or ⬝ a ⬝ b) ↠ SKI.Cond ⬝ TT ⬝ (SKI.Cond ⬝ TT ⬝ FF ⬝ b) ⬝ a :=
OrPoly.toSKI_correct [a, b] (by simp)

theorem or_correct (a b : SKI) (ua ub : Bool) (ha : IsBool ua a) (hb : IsBool ub b) :
Expand Down Expand Up @@ -350,22 +352,22 @@ def Fst : SKI := R ⬝ TT
/-- Second projection -/
def Snd : SKI := R ⬝ FF

theorem fst_correct (a b : SKI) : Fst ⬝ (MkPair ⬝ a ⬝ b) ⇒* a := by calc
_ ⇒* SKI.Cond ⬝ a ⬝ b ⬝ TT := R_def _ _
_ ⇒* a := cond_correct TT a b true TT_correct
theorem fst_correct (a b : SKI) : (Fst ⬝ (MkPair ⬝ a ⬝ b)) ↠ a := by calc
_ SKI.Cond ⬝ a ⬝ b ⬝ TT := R_def _ _
_ a := cond_correct TT a b true TT_correct

theorem snd_correct (a b : SKI) : Snd ⬝ (MkPair ⬝ a ⬝ b) ⇒* b := by calc
_ ⇒* SKI.Cond ⬝ a ⬝ b ⬝ FF := R_def _ _
_ ⇒* b := cond_correct FF a b false FF_correct
theorem snd_correct (a b : SKI) : (Snd ⬝ (MkPair ⬝ a ⬝ b)) ↠ b := by calc
_ SKI.Cond ⬝ a ⬝ b ⬝ FF := R_def _ _
_ b := cond_correct FF a b false FF_correct

/-- Unpaired f ⟨x, y⟩ := f x y, cf `Nat.unparied`. -/
def UnpairedPoly : SKI.Polynomial 2 := &0 ⬝' (Fst ⬝' &1) ⬝' (Snd ⬝' &1)
/-- A term representing Unpaired -/
protected def Unpaired : SKI := UnpairedPoly.toSKI
theorem unpaired_def (f p : SKI) : SKI.Unpaired ⬝ f ⬝ p ⇒* f ⬝ (Fst ⬝ p) ⬝ (Snd ⬝ p) :=
theorem unpaired_def (f p : SKI) : (SKI.Unpaired ⬝ f ⬝ p) ↠ f ⬝ (Fst ⬝ p) ⬝ (Snd ⬝ p) :=
UnpairedPoly.toSKI_correct [f, p] (by simp)

theorem unpaired_correct (f x y : SKI) : SKI.Unpaired ⬝ f ⬝ (MkPair ⬝ x ⬝ y) ⇒* f ⬝ x ⬝ y := by
theorem unpaired_correct (f x y : SKI) : (SKI.Unpaired ⬝ f ⬝ (MkPair ⬝ x ⬝ y)) ↠ f ⬝ x ⬝ y := by
trans f ⬝ (Fst ⬝ (MkPair ⬝ x ⬝ y)) ⬝ (Snd ⬝ (MkPair ⬝ x ⬝ y))
· exact unpaired_def f _
· apply parallel_mRed
Expand All @@ -377,7 +379,9 @@ theorem unpaired_correct (f x y : SKI) : SKI.Unpaired ⬝ f ⬝ (MkPair ⬝ x
def PairPoly : SKI.Polynomial 3 := MkPair ⬝' (&0 ⬝' &2) ⬝' (&1 ⬝' &2)
/-- A SKI term representing Pair -/
protected def Pair : SKI := PairPoly.toSKI
theorem pair_def (f g x : SKI) : SKI.Pair ⬝ f ⬝ g ⬝ x ⇒* MkPair ⬝ (f ⬝ x) ⬝ (g ⬝ x) :=
theorem pair_def (f g x : SKI) : (SKI.Pair ⬝ f ⬝ g ⬝ x) ↠ MkPair ⬝ (f ⬝ x) ⬝ (g ⬝ x) :=
PairPoly.toSKI_correct [f, g, x] (by simp)

end SKI

end Cslib
32 changes: 18 additions & 14 deletions Cslib/Languages/CombinatoryLogic/Confluence.lean
Original file line number Diff line number Diff line change
Expand Up @@ -9,11 +9,11 @@ import Cslib.Foundations.Data.Relation
/-!
# SKI reduction is confluent

This file proves the **Church-Rosser** theorem for the SKI calculus, that is, if `a ⇒* b` and
`a ⇒* c`, `b ⇒* d` and `c ⇒* d` for some term `d`. More strongly (though equivalently), we show
This file proves the **Church-Rosser** theorem for the SKI calculus, that is, if `a b` and
`a c`, `b d` and `c d` for some term `d`. More strongly (though equivalently), we show
that the relation of having a common reduct is transitive — in the above situation, `a` and `b`,
and `a` and `c` have common reducts, so the result implies the same of `b` and `c`. Note that
`CommonReduct` is symmetric (trivially) and reflexive (since `⇒*` is), so we in fact show that
`CommonReduct` is symmetric (trivially) and reflexive (since `` is), so we in fact show that
`CommonReduct` is an equivalence.

Our proof
Expand All @@ -23,21 +23,23 @@ Chapter 4 of Peter Selinger's notes:

## Main definitions

- `ParallelReduction` : a relation `⇒ₚ` on terms such that `⇒ ⊆ ⇒ₚ ⊆ ⇒*`, allowing simultaneous
- `ParallelReduction` : a relation `⇒ₚ` on terms such that `⇒ ⊆ ⇒ₚ ⊆ `, allowing simultaneous
reduction on the head and tail of a term.

## Main results

- `parallelReduction_diamond` : parallel reduction satisfies the diamond property, that is, it is
confluent in a single step.
- `commonReduct_equivalence` : by a general result, the diamond property for `⇒ₚ` implies the same
for its reflexive-transitive closure. This closure is exactly `⇒*`, which implies the
for its reflexive-transitive closure. This closure is exactly ``, which implies the
**Church-Rosser** theorem as sketched above.
-/

namespace Cslib

namespace SKI

open Red MRed
open Red MRed ReductionSystem

/-- A reduction step allowing simultaneous reduction of disjoint redexes -/
inductive ParallelReduction : SKI → SKI → Prop
Expand All @@ -55,8 +57,8 @@ inductive ParallelReduction : SKI → SKI → Prop
/-- Notation for parallel reduction -/
scoped infix:90 " ⇒ₚ " => ParallelReduction

/-- The inclusion `⇒ₚ ⊆ ⇒*` -/
theorem mRed_of_parallelReduction {a a' : SKI} (h : a ⇒ₚ a') : a ⇒* a' := by
/-- The inclusion `⇒ₚ ⊆ ` -/
theorem mRed_of_parallelReduction {a a' : SKI} (h : a ⇒ₚ a') : a a' := by
cases h
case refl => exact Relation.ReflTransGen.refl
case par a a' b b' ha hb =>
Expand All @@ -68,7 +70,7 @@ theorem mRed_of_parallelReduction {a a' : SKI} (h : a ⇒ₚ a') : a ⇒* a' :=
case red_S a b c => exact Relation.ReflTransGen.single (red_S a b c)

/-- The inclusion `⇒ ⊆ ⇒ₚ` -/
theorem parallelReduction_of_red {a a' : SKI} (h : a a') : a ⇒ₚ a' := by
theorem parallelReduction_of_red {a a' : SKI} (h : a a') : a ⇒ₚ a' := by
cases h
case red_S => apply ParallelReduction.red_S
case red_K => apply ParallelReduction.red_K
Expand All @@ -86,12 +88,12 @@ theorem parallelReduction_of_red {a a' : SKI} (h : a ⇒ a') : a ⇒ₚ a' := by
`parallelReduction_of_red` imply that `⇒` and `⇒ₚ` have the same reflexive-transitive
closure. -/
theorem reflTransGen_parallelReduction_mRed :
Relation.ReflTransGen ParallelReduction = MRed := by
Relation.ReflTransGen ParallelReduction = RedSKI.MRed := by
ext a b
constructor
· apply Relation.reflTransGen_minimal
· exact MRed.reflexive
· exact MRed.transitive
· exact fun _ => by rfl
· exact instTransitiveMRed RedSKI
· exact @mRed_of_parallelReduction
· apply Relation.reflTransGen_minimal
· exact Relation.reflexive_reflTransGen
Expand All @@ -101,7 +103,7 @@ theorem reflTransGen_parallelReduction_mRed :
/-!
Irreducibility for the (partially applied) primitive combinators.

TODO: possibly these should be proven more generally (in another file) for `⇒*`.
TODO: possibly these should be proven more generally (in another file) for ``.
-/

lemma I_irreducible (a : SKI) (h : I ⇒ₚ a) : a = I := by
Expand Down Expand Up @@ -234,9 +236,11 @@ theorem commonReduct_equivalence : Equivalence CommonReduct := by
exact join_parallelReduction_equivalence

/-- The **Church-Rosser** theorem in the form it is usually stated. -/
theorem MRed.diamond (a b c : SKI) (hab : a ⇒* b) (hac : a ⇒* c) : CommonReduct b c := by
theorem MRed.diamond (a b c : SKI) (hab : a b) (hac : a c) : CommonReduct b c := by
apply commonReduct_equivalence.trans (y := a)
· exact commonReduct_equivalence.symm (commonReduct_of_single hab)
· exact commonReduct_of_single hac

end SKI

end Cslib
Loading