Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: setup Std.Sat with definitions of SAT and CNF #4933

Merged
merged 18 commits into from
Aug 7, 2024
Merged
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 src/Std.lean
Original file line number Diff line number Diff line change
Expand Up @@ -5,3 +5,4 @@ Authors: Sebastian Ullrich
-/
prelude
import Std.Data
import Std.Sat
7 changes: 7 additions & 0 deletions src/Std/Sat.lean
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
/-
Copyright (c) 2024 Lean FRO, LLC. All rights reserved.
Released under Apache 2.0 license as described in the file LICENSE.
Authors: Henrik Böving
-/
prelude
import Std.Sat.CNF
10 changes: 10 additions & 0 deletions src/Std/Sat/CNF.lean
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
/-
Copyright (c) 2024 Lean FRO, LLC. All rights reserved.
Released under Apache 2.0 license as described in the file LICENSE.
Authors: Henrik Böving
-/
prelude
import Std.Sat.CNF.Basic
import Std.Sat.CNF.Literal
import Std.Sat.CNF.Relabel
import Std.Sat.CNF.RelabelFin
190 changes: 190 additions & 0 deletions src/Std/Sat/CNF/Basic.lean
Original file line number Diff line number Diff line change
@@ -0,0 +1,190 @@
/-
Copyright (c) 2024 Lean FRO, LLC. All rights reserved.
Released under Apache 2.0 license as described in the file LICENSE.
Authors: Kim Morrison
-/
prelude
import Init.Data.List.Lemmas
import Init.Data.List.Impl
import Std.Sat.CNF.Literal

namespace Std
namespace Sat

/--
A clause in a CNF.

The literal `(i, b)` is satisfied if the assignment to `i` agrees with `b`.
-/
abbrev CNF.Clause (α : Type u) : Type u := List (Literal α)

/--
A CNF formula.

Literals are identified by members of `α`.
-/
abbrev CNF (α : Type u) : Type u := List (CNF.Clause α)

namespace CNF

/--
Evaluating a `Clause` with respect to an assignment `a`.
-/
def Clause.eval (a : α → Bool) (c : Clause α) : Bool := c.any fun (i, n) => a i == n

@[simp] theorem Clause.eval_nil (a : α → Bool) : Clause.eval a [] = false := rfl
@[simp] theorem Clause.eval_cons (a : α → Bool) :
Clause.eval a (i :: c) = (a i.1 == i.2 || Clause.eval a c) := rfl

/--
Evaluating a `CNF` formula with respect to an assignment `a`.
-/
def eval (a : α → Bool) (f : CNF α) : Bool := f.all fun c => c.eval a

@[simp] theorem eval_nil (a : α → Bool) : eval a [] = true := rfl
hargoniX marked this conversation as resolved.
Show resolved Hide resolved
@[simp] theorem eval_cons (a : α → Bool) : eval a (c :: f) = (c.eval a && eval a f) := rfl

@[simp] theorem eval_append (a : α → Bool) (f1 f2 : CNF α) :
eval a (f1 ++ f2) = (eval a f1 && eval a f2) := List.all_append

def Sat (a : α → Bool) (f : CNF α) : Prop := eval a f = true
def Unsat (f : CNF α) : Prop := ∀ a, eval a f = false

theorem sat_def (a : α → Bool) (f : CNF α) : Sat a f ↔ (eval a f = true) := by rfl
theorem unsat_def (f : CNF α) : Unsat f ↔ (∀ a, eval a f = false) := by rfl


@[simp] theorem not_unsat_nil : ¬Unsat ([] : CNF α) :=
fun h => by simp [unsat_def] at h

@[simp] theorem sat_nil {assign : α → Bool} : Sat assign ([] : CNF α) := by
simp [sat_def]

@[simp] theorem unsat_nil_cons {g : CNF α} : Unsat ([] :: g) := by
simp [unsat_def]

namespace Clause

/--
Variable `v` occurs in `Clause` `c`.
-/
def Mem (v : α) (c : Clause α) : Prop := (v, false) ∈ c ∨ (v, true) ∈ c

instance {v : α} {c : Clause α} [DecidableEq α] : Decidable (Mem v c) :=
inferInstanceAs <| Decidable (_ ∨ _)

@[simp] theorem not_mem_nil {v : α} : ¬Mem v ([] : Clause α) := by simp [Mem]
@[simp] theorem mem_cons {v : α} : Mem v (l :: c : Clause α) ↔ (v = l.1 ∨ Mem v c) := by
rcases l with ⟨b, (_|_)⟩
· simp [Mem, or_assoc]
· simp [Mem]
rw [or_left_comm]

theorem mem_of (h : (v, p) ∈ c) : Mem v c := by
cases p
· left; exact h
· right; exact h

theorem eval_congr (a1 a2 : α → Bool) (c : Clause α) (hw : ∀ i, Mem i c → a1 i = a2 i) :
eval a1 c = eval a2 c := by
induction c
case nil => rfl
case cons i c ih =>
simp only [eval_cons]
rw [ih, hw]
· rcases i with ⟨b, (_|_)⟩ <;> simp [Mem]
· intro j h
apply hw
rcases h with h | h
· left
apply List.mem_cons_of_mem _ h
· right
apply List.mem_cons_of_mem _ h

end Clause

/--
Variable `v` occurs in `CNF` formula `f`.
-/
def Mem (v : α) (f : CNF α) : Prop := ∃ c, c ∈ f ∧ c.Mem v

instance {v : α} {f : CNF α} [DecidableEq α] : Decidable (Mem v f) :=
inferInstanceAs <| Decidable (∃ _, _)

theorem any_not_isEmpty_iff_exists_mem {f : CNF α} :
(List.any f fun c => !List.isEmpty c) = true ↔ ∃ v, Mem v f := by
simp only [List.any_eq_true, Bool.not_eq_true', List.isEmpty_false_iff_exists_mem, Mem,
Clause.Mem]
constructor
. intro h
rcases h with ⟨clause, ⟨hclause1, hclause2⟩⟩
rcases hclause2 with ⟨lit, hlit⟩
exists lit.fst, clause
constructor
. assumption
. rcases lit with ⟨_, ⟨_ | _⟩⟩ <;> simp_all
. intro h
rcases h with ⟨lit, clause, ⟨hclause1, hclause2⟩⟩
exists clause
constructor
. assumption
. cases hclause2 with
| inl hl => exact Exists.intro _ hl
| inr hr => exact Exists.intro _ hr

@[simp] theorem not_exists_mem : (¬ ∃ v, Mem v f) ↔ ∃ n, f = List.replicate n [] := by
simp only [← any_not_isEmpty_iff_exists_mem]
simp only [List.any_eq_true, Bool.not_eq_true', not_exists, not_and, Bool.not_eq_false]
induction f with
| nil =>
simp only [List.not_mem_nil, List.isEmpty_iff, false_implies, forall_const, true_iff]
exact ⟨0, rfl⟩
| cons c f ih =>
simp_all [ih, List.isEmpty_iff]
constructor
· rintro ⟨rfl, n, rfl⟩
exact ⟨n+1, rfl⟩
· rintro ⟨n, h⟩
cases n
· simp at h
· simp_all only [List.replicate, List.cons.injEq, true_and]
exact ⟨_, rfl⟩

instance {f : CNF α} [DecidableEq α] : Decidable (∃ v, Mem v f) :=
decidable_of_iff (f.any fun c => !c.isEmpty) any_not_isEmpty_iff_exists_mem

@[simp] theorem not_mem_nil {v : α} : ¬Mem v ([] : CNF α) := by simp [Mem]
@[simp] theorem mem_cons {v : α} {c} {f : CNF α} :
Mem v (c :: f : CNF α) ↔ (Clause.Mem v c ∨ Mem v f) := by simp [Mem]

theorem mem_of (h : c ∈ f) (w : Clause.Mem v c) : Mem v f := by
apply Exists.intro c
constructor <;> assumption

@[simp] theorem mem_append {v : α} {f1 f2 : CNF α} : Mem v (f1 ++ f2) ↔ Mem v f1 ∨ Mem v f2 := by
simp [Mem, List.mem_append]
constructor
· rintro ⟨c, (mf1 | mf2), mc⟩
· left
exact ⟨c, mf1, mc⟩
· right
exact ⟨c, mf2, mc⟩
· rintro (⟨c, mf1, mc⟩ | ⟨c, mf2, mc⟩)
· exact ⟨c, Or.inl mf1, mc⟩
· exact ⟨c, Or.inr mf2, mc⟩

theorem eval_congr (a1 a2 : α → Bool) (f : CNF α) (hw : ∀ v, Mem v f → a1 v = a2 v) :
eval a1 f = eval a2 f := by
induction f
case nil => rfl
case cons c x ih =>
simp only [eval_cons]
rw [ih, Clause.eval_congr] <;>
· intro i h
apply hw
simp [h]

end CNF

end Sat
end Std
38 changes: 38 additions & 0 deletions src/Std/Sat/CNF/Literal.lean
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
/-
Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
Released under Apache 2.0 license as described in the file LICENSE.
Authors: Josh Clune
-/
prelude
import Init.Data.Hashable
import Init.Data.ToString

namespace Std
namespace Sat

/--
CNF literals identified by some type `α`. The `Bool` is the polarity of the literal.
`true` means positive polarity.
-/
abbrev Literal (α : Type u) := α × Bool
hargoniX marked this conversation as resolved.
Show resolved Hide resolved

namespace Literal

/--
Flip the polarity of `l`.
-/
def negate (l : Literal α) : Literal α := (l.1, not l.2)

/--
Output `l` as a DIMACS literal identifier.
-/
def dimacs [ToString α] (l : Literal α) : String :=
if l.2 then
s!"{l.1}"
else
s!"-{l.1}"

end Literal

end Sat
end Std
123 changes: 123 additions & 0 deletions src/Std/Sat/CNF/Relabel.lean
Original file line number Diff line number Diff line change
@@ -0,0 +1,123 @@
/-
Copyright (c) 2024 Lean FRO, LLC. All rights reserved.
Released under Apache 2.0 license as described in the file LICENSE.
Authors: Kim Morrison
-/
prelude
import Std.Sat.CNF.Basic

namespace Std
namespace Sat

namespace CNF

namespace Clause

/--
Change the literal type in a `Clause` from `α` to `β` by using `r`.
-/
def relabel (r : α → β) (c : Clause α) : Clause β := c.map (fun (i, n) => (r i, n))

@[simp] theorem eval_relabel {r : α → β} {a : β → Bool} {c : Clause α} :
(relabel r c).eval a = c.eval (a ∘ r) := by
induction c <;> simp_all [relabel]

@[simp] theorem relabel_id' : relabel (id : α → α) = id := by funext; simp [relabel]

theorem relabel_congr {c : Clause α} {r1 r2 : α → β} (hw : ∀ v, Mem v c → r1 v = r2 v) :
relabel r1 c = relabel r2 c := by
simp only [relabel]
rw [List.map_congr_left]
intro ⟨v, p⟩ h
congr
apply hw _ (mem_of h)

-- We need the unapplied equality later.
@[simp] theorem relabel_relabel' : relabel r1 ∘ relabel r2 = relabel (r1 ∘ r2) := by
funext i
simp only [Function.comp_apply, relabel, List.map_map]
rfl

end Clause

/-! ### Relabelling

It is convenient to be able to construct a CNF using a more complicated literal type,
but eventually we need to embed in `Nat`.
-/

/--
Change the literal type in a `CNF` formula from `α` to `β` by using `r`.
-/
def relabel (r : α → β) (f : CNF α) : CNF β := f.map (Clause.relabel r)
hargoniX marked this conversation as resolved.
Show resolved Hide resolved

@[simp] theorem relabel_nil {r : α → β} : relabel r [] = [] := by simp [relabel]
@[simp] theorem relabel_cons {r : α → β} : relabel r (c :: f) = (c.relabel r) :: relabel r f := by
simp [relabel]

@[simp] theorem eval_relabel (r : α → β) (a : β → Bool) (f : CNF α) :
(relabel r f).eval a = f.eval (a ∘ r) := by
induction f <;> simp_all

@[simp] theorem relabel_append : relabel r (f1 ++ f2) = relabel r f1 ++ relabel r f2 :=
List.map_append _ _ _

@[simp] theorem relabel_relabel : relabel r1 (relabel r2 f) = relabel (r1 ∘ r2) f := by
simp only [relabel, List.map_map, Clause.relabel_relabel']

@[simp] theorem relabel_id : relabel id x = x := by simp [relabel]

theorem relabel_congr {f : CNF α} {r1 r2 : α → β} (hw : ∀ v, Mem v f → r1 v = r2 v) :
relabel r1 f = relabel r2 f := by
dsimp only [relabel]
rw [List.map_congr_left]
intro c h
apply Clause.relabel_congr
intro v m
exact hw _ (mem_of h m)

theorem sat_relabel {f : CNF α} (h : Sat (r1 ∘ r2) f) : Sat r1 (relabel r2 f) := by
simp_all [sat_def]

theorem unsat_relabel {f : CNF α} (r : α → β) (h : Unsat f) :
Unsat (relabel r f) := by
simp_all [unsat_def]

theorem nonempty_or_impossible (f : CNF α) : Nonempty α ∨ ∃ n, f = List.replicate n [] := by
induction f with
| nil => exact Or.inr ⟨0, rfl⟩
| cons c x ih => match c with
| [] => cases ih with
| inl h => left; exact h
| inr h =>
obtain ⟨n, rfl⟩ := h
right
exact ⟨n + 1, rfl⟩
| ⟨a, b⟩ :: c => exact Or.inl ⟨a⟩

theorem unsat_relabel_iff {f : CNF α} {r : α → β}
(hw : ∀ {v1 v2}, Mem v1 f → Mem v2 f → r v1 = r v2 → v1 = v2) :
Unsat (relabel r f) ↔ Unsat f := by
rcases nonempty_or_impossible f with (⟨⟨a₀⟩⟩ | ⟨n, rfl⟩)
· refine ⟨fun h => ?_, unsat_relabel r⟩
have em := Classical.propDecidable
let g : β → α := fun b =>
if h : ∃ a, Mem a f ∧ r a = b then h.choose else a₀
have h' := unsat_relabel g h
suffices w : relabel g (relabel r f) = f by
rwa [w] at h'
have : ∀ a, Mem a f → g (r a) = a := by
intro v h
dsimp [g]
rw [dif_pos ⟨v, h, rfl⟩]
apply hw _ h
· exact (Exists.choose_spec (⟨v, h, rfl⟩ : ∃ a', Mem a' f ∧ r a' = r v)).2
· exact (Exists.choose_spec (⟨v, h, rfl⟩ : ∃ a', Mem a' f ∧ r a' = r v)).1
rw [relabel_relabel, relabel_congr, relabel_id]
exact this
· cases n <;> simp [unsat_def, List.replicate_succ]

end CNF

end Sat
end Std
Loading
Loading