Skip to content
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
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
1 change: 1 addition & 0 deletions Cslib.lean
Original file line number Diff line number Diff line change
Expand Up @@ -69,6 +69,7 @@ public import Cslib.Foundations.Data.RelatesInSteps
public import Cslib.Foundations.Data.Set.Saturation
public import Cslib.Foundations.Data.StackTape
public import Cslib.Foundations.Lint.Basic
public import Cslib.Foundations.Logic.Connectives
public import Cslib.Foundations.Logic.InferenceSystem
public import Cslib.Foundations.Logic.LogicalEquivalence
public import Cslib.Foundations.Relation.Attr
Expand Down
96 changes: 96 additions & 0 deletions Cslib/Foundations/Logic/Connectives.lean
Original file line number Diff line number Diff line change
@@ -0,0 +1,96 @@
/-
Copyright (c) 2026 Benjamin Brast-McKie. All rights reserved.
Released under Apache 2.0 license as described in the file LICENSE.
Authors: Benjamin Brast-McKie
-/

module

import Cslib.Init

/-! # Connective Typeclasses for Propositional Logic

This module defines a typeclass hierarchy for propositional logical connectives. Each formula
type registers itself as an instance of the appropriate connective class, enabling polymorphic
axiom definitions and notation that work uniformly across different formula types.

## Design

The hierarchy adopts a hybrid five-primitive propositional signature `{atom, bot, imp, and, or}`,
following the operator-typeclass direction of fmontesi's PR #607 (one class per operator):
- **Atomic classes**: `HasBot`, `HasImp`, `HasAnd`, `HasOr`
- **Bundled class**: `PropositionalConnectives` (extends `HasBot` and `HasImp`)

Conjunction (`HasAnd`) and disjunction (`HasOr`) are treated as independent primitives rather
than Łukasiewicz-derived connectives. The classical encodings `φ ∧ ψ := ¬(φ → ¬ψ)` and
`φ ∨ ψ := ¬φ → ψ` are only propositionally equivalent to `∧` and `∨` in classical logic
([Wajsberg1938], [McKinsey1939]); they fail in intuitionistic and minimal logic. Making `and`
and `or` primitive via `HasAnd`/`HasOr` supports all three logic strengths with a single
typeclass hierarchy.

Negation and verum stay derived: each concrete formula type defines `neg φ := φ → ⊥` and
`top := ⊥ → ⊥` as `abbrev`s, which are valid in minimal, intuitionistic, and classical logic
alike, so no typeclass machinery is needed for them.

## Why Not Mathlib's `Bot` / `HImp`?

Mathlib's `Bot` class is for algebraic bottom elements (lattice theory), and `HImp` is for
Heyting implication in a lattice. Neither is appropriate here:
- Formula types are not lattices; they are freely generated algebras.
- We need `imp` to be a binary constructor (taking two formula arguments), not a binary
operation on a lattice.
- The typeclass hierarchy here enables polymorphic axiom definitions that quantify over any
formula type satisfying the appropriate connective class, independently of any algebraic
structure.

## References

* [I. Johansson, *Der Minimalkalkül, ein reduzierter intuitionistischer Formalismus*][Johansson1937]
* [M. Wajsberg, *Untersuchungen über den Aussagenkalkül von A. Heyting*][Wajsberg1938]
* [J. C. C. McKinsey,
*Proof of the Independence of the Primitive Symbols of Heyting's Calculus*][McKinsey1939]
* [D. Prawitz, *Natural Deduction: A Proof-Theoretical Study*][Prawitz1965]
* [A. S. Troelstra, D. van Dalen,
*Constructivism in Mathematics: An Introduction*][TroelstraVanDalen1988]
* [A. Church, *Introduction to Mathematical Logic*][Church1956]
* [G. Gentzen, *Untersuchungen über das logische Schließen*][Gentzen1935]
-/

@[expose] public section

namespace Cslib.Logic

/-- A type has a falsum (bottom) connective. -/
class HasBot (F : Type*) where
/-- The falsum/bottom connective. -/
bot : F

/-- A type has an implication connective. -/
class HasImp (F : Type*) where
/-- The implication connective. -/
imp : F → F → F

/-- A type has a conjunction connective. -/
class HasAnd (F : Type*) where
/-- The conjunction connective. -/
and : F → F → F

/-- A type has a disjunction connective. -/
class HasOr (F : Type*) where
/-- The disjunction connective. -/
or : F → F → F

/-- Propositional connectives: falsum, implication, conjunction, and disjunction.

`HasAnd` and `HasOr` are included because all four connectives are needed for a
five-primitive propositional signature `{atom, bot, imp, and, or}`. This bundled class
allows axiom schemas to be stated polymorphically over any formula type providing
all four connectives.

Note: this does not extend `HasAnd` and `HasOr` directly to avoid forcing all
formula types to bundle conjunction and disjunction together with the minimal
`HasBot`/`HasImp` interface. When all four are needed, use
`[PropositionalConnectives F] [HasAnd F] [HasOr F]`. -/
class PropositionalConnectives (F : Type*) extends HasBot F, HasImp F

end Cslib.Logic
143 changes: 107 additions & 36 deletions Cslib/Logics/Propositional/Defs.lean
Original file line number Diff line number Diff line change
@@ -1,37 +1,88 @@
/-
Copyright (c) 2025 Thomas Waring. All rights reserved.
Copyright (c) 2025 Thomas Waring, 2026 Benjamin Brast-McKie. All rights reserved.
Released under Apache 2.0 license as described in the file LICENSE.
Authors: Thomas Waring
Authors: Thomas Waring, Benjamin Brast-McKie
-/

module

public import Cslib.Init
public import Cslib.Foundations.Logic.Connectives
public import Mathlib.Data.FunLike.Basic
public import Mathlib.Data.Set.Basic
public import Mathlib.Data.Set.Image
public import Mathlib.Order.TypeTags

/-! # Propositions and theories

## Main definitions

- `Proposition` : the type of propositions over a given type of atom. This type has a `Bot`
instance whenever `Atom` does, and a `Top` whenever `Atom` is inhabited.
- `Proposition` : the type of propositions over a given type of atom. Primitives are `atom`,
`bot` (falsum), `imp` (implication), `and` (conjunction), and `or` (disjunction), following
the standard five-primitive signature of Gentzen/Prawitz/Troelstra-van Dalen. Negation (`neg`),
verum (`top`), and biconditional (`iff`) are derived connectives (`abbrev`s) rather than
constructors.
- `Theory` : set of `Proposition`.
- `IsIntuitionistic` : a theory is intuitionistic if it contains the principle of explosion.
- `IsClassical` : an intuitionistic theory is classical if it further contains double negation
elimination.
elimination.
- `Proposition.subst` : replace `atom x` in a `A : Proposition Atom` with `f x`, for a function
`f : Atom → Proposition Atom'`. This induces a monad structure on `Proposition`, with
`pure := Proposition.atom`. `Theory` is a functor, by mapping each proposition `A ∈ T` to
`f <$> A`.
- `Theory.intuitionisticCompletion` : the freely generated intuitionistic theory extending a given
theory.
theory.

## Design: Five Primitives

Upstream used four constructors `{atom, and, or, impl}` with a `Bot` typeclass constraint
to simulate falsum. This PR changes to five primitives `{atom, bot, imp, and, or}` with derived
negation and verum, for the following reasons:

1. **Minimal logic requires primitive `bot`**: Johansson's minimal logic (1937, [Johansson1937])
is the standard basis for proof-relevant propositional logic. Minimal logic requires a
distinguishable bottom formula; without it, one obtains only the *positive fragment* (no
negation at all). The `[Bot Atom]` constraint approach simulates bottom via an atomic formula,
which conflates the logical bottom with atomic data.

2. **Uniform negation**: With primitive `bot`, negation `¬A := A → ⊥` is uniformly available
without typeclass constraints. The old `[Bot Atom]` approach required `Inhabited Atom` for
`top` and `Bot Atom` for `neg`, producing a constraint proliferation.

3. **Standard notation**: The constructor name `impl` is non-standard; Gentzen (1935,
[Gentzen1935]) and Prawitz (1965, [Prawitz1965]) both write the connective as `⊃` or `→`.
We rename to `imp` for clarity and alignment with the broader logic literature.

4. **Functional completeness**: Adding `bot` as a primitive makes `{bot, imp, and, or}` the
standard five-primitive basis studied in [TroelstraVanDalen1988] Chapter 2 and
[Church1956] §24. This resolves the objection (CSLib PR #635) that the four-primitive
`{and, or, impl}` set is not functionally complete for classical logic (it lacks the
ability to express unsatisfiable formulas).

## Notation

We introduce notation for the logical connectives: `⊥ ⊤ ∧ ∨ → ¬` for, respectively, falsum, verum,
conjunction, disjunction, implication and negation.
We introduce notation for the logical connectives: `⊥ ⊤ ∧ ∨ → ↔ ¬` for, respectively, falsum,
verum, conjunction, disjunction, implication, biconditional, and negation.

## Architecture

Two proof systems are defined for this propositional language:

- **Layer 1 — Natural Deduction** (`NaturalDeduction/Basic.lean`): a `Theory.Derivation` inductive
with 10 primitive constructors. Logic strength is controlled by the theory parameter: `MPL`
(Johansson's minimal logic, [Johansson1937]), `IPL` (intuitionistic), and `CPL` (classical).

- **Layer 2 — Hilbert System** (`ProofSystem/`): an axiom predicate hierarchy with sequent
derivability and a Hilbert-style proof-theoretic treatment (future PR).

## References

* [I. Johansson, *Der Minimalkalkül, ein reduzierter intuitionistischer Formalismus*][Johansson1937]
* [G. Gentzen, *Untersuchungen über das logische Schließen*][Gentzen1935]
* [D. Prawitz, *Natural Deduction: A Proof-Theoretical Study*][Prawitz1965]
* [A. S. Troelstra, D. van Dalen,
*Constructivism in Mathematics: An Introduction*][TroelstraVanDalen1988]
* [A. Church, *Introduction to Mathematical Logic*][Church1956]
* [A. Chagrov, M. Zakharyaschev, *Modal Logic*][ChagrovZakharyaschev1997], Chapter 1
-/

@[expose] public section
Expand All @@ -42,44 +93,64 @@ variable {Atom : Type u} [DecidableEq Atom]

namespace Cslib.Logic.PL

/-- Propositions. -/
/-- Propositions. Primitives are atoms, falsum, implication, conjunction, and disjunction. -/
inductive Proposition (Atom : Type u) : Type u where
/-- Propositional atoms -/
| atom (x : Atom)
/-- Falsum / bottom — a primitive, not an atomic formula encoding of bottom. Making `bot`
primitive is required by Johansson's minimal logic ([Johansson1937]); without it, the
type only captures the positive fragment. -/
| bot
/-- Implication -/
| imp (a b : Proposition Atom)
/-- Conjunction -/
| and (a b : Proposition Atom)
/-- Disjunction -/
| or (a b : Proposition Atom)
/-- Implication -/
| impl (a b : Proposition Atom)
deriving DecidableEq, BEq

instance instBotProposition [Bot Atom] : Bot (Proposition Atom) := ⟨.atom ⊥⟩
instance instInhabitedOfBot [Bot Atom] : Inhabited Atom := ⟨⊥⟩
/-- Negation as a derived connective: `¬A := A → ⊥`. Valid in minimal, intuitionistic, and
classical logic alike. -/
abbrev Proposition.neg : Proposition Atom → Proposition Atom := (Proposition.imp · .bot)

/-- We view negation as a defined connective ~A := A → ⊥ -/
abbrev Proposition.neg [Bot Atom] : Proposition Atom → Proposition Atom := (Proposition.impl · ⊥)
/-- Verum as a derived connective: `⊤ := → ⊥`. Derivable in minimal logic. -/
abbrev Proposition.top : Proposition Atom := .imp .bot .bot

/-- A fixed choice of a derivable proposition (of course any two are equivalent). -/
abbrev Proposition.top [Inhabited Atom] : Proposition Atom := impl (.atom default) (.atom default)
/-- Biconditional as a derived connective: `A ↔ B := (A → B) ∧ (B → A)`. -/
abbrev Proposition.iff (A B : Proposition Atom) : Proposition Atom :=
(A.imp B).and (B.imp A)

instance instTopProposition [Inhabited Atom] : Top (Proposition Atom) := ⟨.top⟩

example [Bot Atom] : (⊤ : Proposition Atom) = Proposition.impl ⊥ ⊥ := rfl
instance : Bot (Proposition Atom) := ⟨.bot⟩
instance : Top (Proposition Atom) := ⟨.top⟩

@[inherit_doc] scoped infix:36 " ∧ " => Proposition.and
@[inherit_doc] scoped infix:35 " ∨ " => Proposition.or
@[inherit_doc] scoped infix:30 " → " => Proposition.impl
@[inherit_doc] scoped infix:30 " → " => Proposition.imp
@[inherit_doc] scoped infix:20 " ↔ " => Proposition.iff
@[inherit_doc] scoped prefix:40 " ¬ " => Proposition.neg

/-- Register `Proposition` as an instance of `PropositionalConnectives`. -/
instance : PropositionalConnectives (Proposition Atom) where
bot := .bot
imp := .imp

/-- Register `HasAnd` instance for `Proposition`. -/
instance : HasAnd (Proposition Atom) where
and := .and

/-- Register `HasOr` instance for `Proposition`. -/
instance : HasOr (Proposition Atom) where
or := .or

/-- Substitute each atom in a proposition for a proposition, possibly changing the atomic
language. -/
def Proposition.subst {Atom Atom' : Type u} (f : Atom → Proposition Atom') :
Proposition Atom → Proposition Atom'
| atom x => f x
| and A B => (A.subst f) ∧ (B.subst f)
| or A B => (A.subst f) ∨ (B.subst f)
| impl A B => (A.subst f) → (B.subst f)
| bot => .bot
| imp A B => .imp (A.subst f) (B.subst f)
| and A B => .and (A.subst f) (B.subst f)
| or A B => .or (A.subst f) (B.subst f)

-- This is probably a lawful monad, but that doesn't seem to be important.
instance : Monad Proposition where
Expand All @@ -102,45 +173,45 @@ instance : Functor Theory where
abbrev MPL : Theory (Atom) := ∅

/-- Intuitionistic propositional logic adds the principle of explosion (ex falso quodlibet). -/
abbrev IPL [Bot Atom] : Theory Atom :=
Set.range (⊥ → ·)
abbrev IPL : Theory Atom :=
Set.range (Proposition.imp ⊥ ·)

/-- Classical logic further adds double negation elimination. -/
abbrev CPL [Bot Atom] : Theory Atom :=
abbrev CPL : Theory Atom :=
Set.range (fun (A : Proposition Atom) ↦ ¬¬A → A)

/-- A theory is intuitionistic if it validates ex falso quodlibet. -/
@[scoped grind]
class IsIntuitionistic [Bot Atom] (T : Theory Atom) where
class IsIntuitionistic (T : Theory Atom) where
efq (A : Proposition Atom) : (⊥ → A) ∈ T

omit [DecidableEq Atom] in
@[scoped grind =]
theorem isIntuitionisticIff [Bot Atom] (T : Theory Atom) : IsIntuitionistic T ↔ IPL ⊆ T := by grind
theorem isIntuitionisticIff (T : Theory Atom) : IsIntuitionistic T ↔ IPL ⊆ T := by grind

/-- A theory is classical if it validates double-negation elimination. -/
@[scoped grind]
class IsClassical [Bot Atom] (T : Theory Atom) where
class IsClassical (T : Theory Atom) where
dne (A : Proposition Atom) : (¬¬A → A) ∈ T

omit [DecidableEq Atom] in
@[scoped grind =]
theorem isClassicalIff [Bot Atom] (T : Theory Atom) : IsClassical T ↔ CPL ⊆ T := by grind
theorem isClassicalIff (T : Theory Atom) : IsClassical T ↔ CPL ⊆ T := by grind

instance instIsIntuitionisticIPL [Bot Atom] : IsIntuitionistic (Atom := Atom) IPL where
instance instIsIntuitionisticIPL : IsIntuitionistic (Atom := Atom) IPL where
efq A := Set.mem_range.mpr ⟨A, rfl⟩

instance instIsClassicalCPL [Bot Atom] : IsClassical (Atom := Atom) CPL where
instance instIsClassicalCPL : IsClassical (Atom := Atom) CPL where
dne A := Set.mem_range.mpr ⟨A, rfl⟩

omit [DecidableEq Atom] in
@[scoped grind →]
theorem instIsIntuitionisticExtention [Bot Atom] {T T' : Theory Atom} [IsIntuitionistic T]
theorem instIsIntuitionisticExtention {T T' : Theory Atom} [IsIntuitionistic T]
(h : T ⊆ T') : IsIntuitionistic T' := by grind

omit [DecidableEq Atom] in
@[scoped grind →]
theorem instIsClassicalExtention [Bot Atom] {T T' : Theory Atom} [IsClassical T] (h : T ⊆ T') :
theorem instIsClassicalExtention {T T' : Theory Atom} [IsClassical T] (h : T ⊆ T') :
IsClassical T' := by grind

/-- Attach a bottom element to a theory `T`, and the principle of explosion for that bottom. -/
Expand Down
Loading
Loading