This PR adds `Invariant.withEarlyReturnNewDo`, `StringInvariant.withEarlyReturnNewDo`, and `StringSliceInvariant.withEarlyReturnNewDo` which use `Prod` instead of `MProd` for the state tuple, matching the new do elaborator's output. The existing `withEarlyReturn` definitions are reverted to `MProd` for backwards compatibility with the legacy do elaborator. Tests and invariant suggestions are updated to use the `NewDo` variants. 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-authored-by: Claude Opus 4.6 <noreply@anthropic.com>
832 lines
24 KiB
Text
832 lines
24 KiB
Text
/-
|
||
Copyright (c) 2025 Lean FRO LLC. All rights reserved.
|
||
Released under Apache 2.0 license as described in the file LICENSE.
|
||
Authors: Sebastian Graf
|
||
-/
|
||
|
||
import Std
|
||
import Lean.Elab.Tactic.Do.VCGen
|
||
import Lean
|
||
|
||
open Std.Do
|
||
|
||
set_option grind.warning false
|
||
set_option mvcgen.warning false
|
||
set_option warn.sorry false
|
||
set_option backward.do.legacy false
|
||
|
||
namespace Code
|
||
|
||
def fib_impl (n : Nat) : Id Nat := do
|
||
if n = 0 then return 0
|
||
let mut a := 0
|
||
let mut b := 1
|
||
for _ in [1:n] do
|
||
let a' := a
|
||
a := b
|
||
b := a' + b
|
||
return b
|
||
|
||
abbrev fib_spec : Nat → Nat
|
||
| 0 => 0
|
||
| 1 => 1
|
||
| n+2 => fib_spec n + fib_spec (n+1)
|
||
|
||
abbrev AppState := Nat × Nat
|
||
|
||
def mkFreshNat [Monad m] [MonadStateOf AppState m] : m Nat := do
|
||
let n ← Prod.fst <$> get
|
||
modify (fun s => (s.1 + 1, s.2))
|
||
pure n
|
||
|
||
def mkFreshPair : StateM AppState (Nat × Nat) := do
|
||
let a ← mkFreshNat
|
||
let b ← mkFreshNat
|
||
pure (a, b)
|
||
|
||
def sum_loop : Id Nat := do
|
||
let mut x := 0
|
||
for i in [1:5] do x := x + i
|
||
return x
|
||
|
||
def throwing_loop : ExceptT Nat (StateT Nat Id) Nat := do
|
||
let mut x := 0
|
||
let s ← get
|
||
for i in [1:s] do { x := x + i; if x > 4 then throw 42 }
|
||
(set 1 : ExceptT Nat (StateT Nat Id) PUnit)
|
||
return x
|
||
|
||
def breaking_loop : StateT Nat Id Nat := do
|
||
let mut x := 0
|
||
let s ← get
|
||
for i in [1:s] do { x := x + i; if x > 4 then break }
|
||
set 1
|
||
return x
|
||
|
||
def returning_loop : StateT Nat Id Nat := do
|
||
let mut x := 0
|
||
let s ← get
|
||
for i in [1:s] do { x := x + i; if x > 4 then return 42 }
|
||
set 1
|
||
return x
|
||
|
||
def unfold_to_expose_match : StateM Nat Nat :=
|
||
(some get).getD (pure 3)
|
||
|
||
end Code
|
||
|
||
namespace ByHand
|
||
|
||
open Code
|
||
|
||
open Lean.Parser.Tactic
|
||
|
||
theorem sum_loop_spec :
|
||
⦃⌜True⌝⦄
|
||
sum_loop
|
||
⦃⇓r => ⌜r < 30⌝⦄ := by
|
||
mintro -
|
||
unfold sum_loop
|
||
mspec
|
||
case inv => exact (⇓ (xs, r) => ⌜(∀ x ∈ xs.suffix, x ≤ 5) ∧ r + xs.suffix.length * 5 ≤ 25⌝)
|
||
all_goals simp_all +decide; try omega
|
||
intros
|
||
mintro _
|
||
mspec
|
||
simp_all +decide
|
||
omega
|
||
|
||
theorem mkFreshNat_spec [Monad m] [WPMonad m sh] :
|
||
⦃fun p => ⌜p.1 = n ∧ p.2 = o⌝⦄
|
||
(mkFreshNat : StateT (Nat × Nat) m Nat)
|
||
⦃⇓ r p => ⌜r = n ∧ p.1 = n + 1 ∧ p.2 = o⌝⦄ := by
|
||
mintro _
|
||
dsimp only [mkFreshNat, get, getThe, instMonadStateOfOfMonadLift, liftM, monadLift, modify, modifyGet]
|
||
mspec
|
||
mspec
|
||
mspec
|
||
mspec
|
||
simp [*]
|
||
|
||
attribute [local spec] mkFreshNat_spec
|
||
|
||
theorem mkFreshPair_spec :
|
||
⦃⌜True⌝⦄
|
||
mkFreshPair
|
||
⦃⇓ (a, b) => ⌜a ≠ b⌝⦄ := by
|
||
unfold mkFreshPair
|
||
mintro -
|
||
mspec mkFreshNat_spec
|
||
mintro ∀s
|
||
mrename_i h
|
||
mcases h with ⌜h₁⌝
|
||
mspec mkFreshNat_spec
|
||
mintro ∀s
|
||
mrename_i h
|
||
mcases h with ⌜h₂⌝
|
||
simp_all
|
||
|
||
theorem mkFreshPair_spec_no_eta :
|
||
⦃⌜True⌝⦄
|
||
mkFreshPair
|
||
⦃⇓ (a, b) => ⌜a ≠ b⌝⦄ := by
|
||
unfold mkFreshPair
|
||
mintro -
|
||
mspec mkFreshNat_spec
|
||
mspec mkFreshNat_spec
|
||
mspec
|
||
intro _; simp_all
|
||
|
||
example : PostCond (Nat × List.Cursor (List.range' 1 3 1)) (PostShape.except Nat (PostShape.arg Nat PostShape.pure)) :=
|
||
⟨fun (r, xs) s => ⌜r ≤ 4 ∧ s = 4 ∧ r + xs.suffix.sum > 4⌝, fun e s => ⌜e = 42 ∧ s = 4⌝, ()⟩
|
||
|
||
example : PostCond (Nat × List.Cursor (List.range' 1 3 1)) (PostShape.except Nat (PostShape.arg Nat PostShape.pure)) :=
|
||
post⟨fun (r, xs) s => ⌜r ≤ 4 ∧ s = 4 ∧ r + xs.suffix.sum > 4⌝, fun e s => ⌜e = 42 ∧ s = 4⌝⟩
|
||
|
||
theorem throwing_loop_spec :
|
||
⦃fun s => ⌜s = 4⌝⦄
|
||
throwing_loop
|
||
⦃post⟨fun _ _ => ⌜False⌝,
|
||
fun e s => ⌜e = 42 ∧ s = 4⌝⟩⦄ := by
|
||
mintro hs
|
||
dsimp +instances only [throwing_loop, get, getThe, instMonadStateOfOfMonadLift, liftM, monadLift]
|
||
mspec
|
||
mspec
|
||
mspec
|
||
case inv => exact post⟨fun (xs, r) s => ⌜r ≤ 4 ∧ s = 4 ∧ r + xs.suffix.sum > 4⌝, fun e s => ⌜e = 42 ∧ s = 4⌝⟩
|
||
case post.success =>
|
||
mspec
|
||
mspec
|
||
mspec
|
||
simp_all only [List.sum_nil, Nat.add_zero, gt_iff_lt, SPred.down_pure, SPred.entails_nil,
|
||
imp_false, not_true_eq_false]
|
||
omega
|
||
case post.except => simp
|
||
case pre => simp_all +decide
|
||
case step =>
|
||
simp_all
|
||
intros
|
||
mintro _
|
||
split
|
||
case isTrue => intro _; mintro _; mspec; mspec; intro _; simp_all
|
||
case isFalse => intro _; mintro _; mspec; intro _; simp_all +arith
|
||
|
||
theorem beaking_loop_spec :
|
||
⦃fun s => ⌜s = 42⌝⦄
|
||
breaking_loop
|
||
⦃⇓ r s => ⌜r > 4 ∧ s = 1⌝⦄ := by
|
||
mintro hs
|
||
dsimp only [breaking_loop, get, getThe, instMonadStateOfOfMonadLift, liftM, monadLift]
|
||
mspec
|
||
mspec
|
||
case inv => exact (⇓ (xs, r) s => ⌜(r ≤ 4 ∧ r = xs.prefix.sum ∨ r > 4) ∧ s = 42⌝)
|
||
all_goals simp_all
|
||
case post => grind
|
||
case step =>
|
||
intros
|
||
mintro _
|
||
split
|
||
case isTrue => intro _; mintro _; mspec; simp_all
|
||
case isFalse => intro _; mintro _; mspec; simp_all; omega
|
||
|
||
theorem returning_loop_spec :
|
||
⦃fun s => ⌜s = 4⌝⦄
|
||
returning_loop
|
||
⦃⇓ r s => ⌜r = 42 ∧ s = 4⌝⦄ := by
|
||
mintro hs
|
||
dsimp only [returning_loop, get, getThe, instMonadStateOfOfMonadLift, liftM, monadLift]
|
||
mspec
|
||
mspec
|
||
case inv => exact (⇓ (xs, r) s => ⌜(r.1 = none ∧ r.2 = xs.prefix.sum ∧ r.2 ≤ 4 ∨ r.1 = some 42 ∧ r.2 > 4) ∧ s = 4⌝)
|
||
all_goals simp_all [-SPred.entails_1]
|
||
case post =>
|
||
split
|
||
· mspec
|
||
intro _ h
|
||
simp at h
|
||
grind
|
||
· mspec
|
||
intro _ h
|
||
simp at h ⊢
|
||
grind
|
||
case step =>
|
||
intros
|
||
mintro _
|
||
split
|
||
case isTrue => intro _; mintro _; mspec; simp_all
|
||
case isFalse => intro _; mintro _; mspec; simp_all; omega
|
||
|
||
section fib
|
||
|
||
def fib_impl (n : Nat) : Id Nat := do
|
||
if n = 0 then return 0
|
||
let mut a := 0
|
||
let mut b := 1
|
||
for _ in [1:n] do
|
||
let a' := a
|
||
a := b
|
||
b := a' + b
|
||
return b
|
||
|
||
theorem fib_triple : ⦃⌜True⌝⦄ fib_impl n ⦃⇓ r => ⌜r = fib_spec n⌝⦄ := by
|
||
unfold fib_impl
|
||
dsimp
|
||
mintro _
|
||
if h : n = 0 then simp [h] else
|
||
simp only [h, reduceIte]
|
||
mspec Spec.forIn_range (⇓ ⟨xs, a, b⟩ => ⌜a = fib_spec xs.pos ∧ b = fib_spec (xs.pos + 1)⌝) ?step
|
||
case step => intros; mintro _; simp_all
|
||
simp_all [Nat.sub_one_add_one]
|
||
|
||
theorem fib_triple_cases : ⦃⌜True⌝⦄ fib_impl n ⦃⇓ r => ⌜r = fib_spec n⌝⦄ := by
|
||
apply fib_impl.fun_cases n _ ?case1 ?case2
|
||
case case1 => rintro rfl; mintro -; simp only [fib_impl, ↓reduceIte]; mspec
|
||
intro h
|
||
mintro -
|
||
simp only [fib_impl, h, reduceIte]
|
||
mspec Spec.forIn_range (⇓ ⟨xs, a, b⟩ => ⌜a = fib_spec xs.pos ∧ b = fib_spec (xs.pos + 1)⌝) ?step
|
||
case step => intros; mintro _; mspec; simp_all
|
||
simp_all [Nat.sub_one_add_one]
|
||
|
||
theorem fib_impl_vcs
|
||
(Q : Nat → PostCond Nat PostShape.pure)
|
||
(I : (n : Nat) → (_ : ¬n = 0) →
|
||
Invariant [1:n].toList (Prod Nat Nat) PostShape.pure)
|
||
(ret : ⊢ₛ (Q 0).1 0)
|
||
(loop_pre : ∀ n (hn : ¬n = 0), ⊢ₛ (I n hn).1 ⟨⟨[], [1:n].toList, rfl⟩, 0, 1⟩)
|
||
(loop_post : ∀ n (hn : ¬n = 0) r, (I n hn).1 ⟨⟨[1:n].toList, [], by simp⟩, r⟩ ⊢ₛ (Q n).1 r.2)
|
||
(loop_step : ∀ n (hn : ¬n = 0) r pref cur suff (h : [1:n].toList = pref ++ cur :: suff),
|
||
(I n hn).1 ⟨⟨pref, cur::suff, by simp[h]⟩, r⟩ ⊢ₛ (I n hn).1 ⟨⟨pref ++ [cur], suff, by simp[h]⟩, r.2, r.1+r.2⟩)
|
||
: ⊢ₛ wp⟦fib_impl n⟧ (Q n) := by
|
||
apply fib_impl.fun_cases n _ ?case1 ?case2
|
||
case case1 => intro h; simp only [fib_impl, h, ↓reduceIte]; mstart; mspec
|
||
intro hn
|
||
simp only [fib_impl, hn, ↓reduceIte]
|
||
mstart
|
||
mspec
|
||
case pre => exact loop_pre n hn
|
||
case post.success => mspec; mpure_intro; apply_rules [loop_post]
|
||
case step =>
|
||
intro _ _ _ _ h;
|
||
mintro _;
|
||
mspec
|
||
mpure_intro
|
||
apply_rules [loop_step]
|
||
|
||
theorem fib_triple_vcs : ⦃⌜True⌝⦄ fib_impl n ⦃⇓ r => ⌜r = fib_spec n⌝⦄ := by
|
||
apply fib_impl_vcs
|
||
case I => intro n hn; exact (⇓ ⟨xs, a, b⟩ => ⌜a = fib_spec xs.pos ∧ b = fib_spec (xs.pos + 1)⌝)
|
||
case ret => mpure_intro; rfl
|
||
case loop_pre => intros; mpure_intro; trivial
|
||
case loop_post => simp_all [Nat.sub_one_add_one]
|
||
case loop_step => simp_all
|
||
|
||
theorem fib_correct {n} : (fib_impl n).run = fib_spec n := by
|
||
generalize h : (fib_impl n).run = x
|
||
apply Id.of_wp_run_eq h
|
||
apply fib_triple
|
||
|
||
end fib
|
||
|
||
section regressions
|
||
|
||
def mySum (l : Array Nat) : Nat := Id.run do
|
||
let mut out := 0
|
||
for i in l do
|
||
out := out + i
|
||
return out
|
||
|
||
theorem mySum_correct (l : Array Nat) : mySum l = l.sum := by
|
||
generalize h : mySum l = x
|
||
apply Id.of_wp_run_eq h
|
||
-- This tests that `mspec` properly replaces the main goal.
|
||
-- Previously, we would get `No goals to be solved` here.
|
||
mspec
|
||
all_goals admit
|
||
|
||
end regressions
|
||
|
||
section WeNeedAProofMode
|
||
|
||
abbrev M := StateT Nat (StateT Char (StateT Bool (StateT String Id)))
|
||
axiom op : Nat → M Nat
|
||
noncomputable def prog (n : Nat) : M Nat := do
|
||
let a ← op n
|
||
let b ← op a
|
||
let c ← op b
|
||
return (a + b + c)
|
||
|
||
axiom isValid : Nat → Char → Bool → String → ULift Prop
|
||
|
||
axiom op.spec {n} : ⦃isValid⦄ op n ⦃⇓r => ⌜r > 42⌝ ∧ isValid⦄
|
||
|
||
theorem prog.spec : ⦃isValid⦄ prog n ⦃⇓r => ⌜r > 100⌝ ∧ isValid⦄ := by
|
||
unfold prog
|
||
mintro h
|
||
mspec op.spec
|
||
mrename_i h
|
||
mcases h with ⟨⌜hr₁⌝, □h⟩
|
||
/-
|
||
n r : Nat
|
||
hr₁ : r > 42
|
||
⊢
|
||
h : isValid
|
||
⊢ₛ
|
||
wp⟦do
|
||
let b ← op r
|
||
let c ← op b
|
||
pure (r + b + c)⟧
|
||
(⇓r => ⌜r > 100⌝ ∧ isValid)
|
||
-/
|
||
mspec op.spec
|
||
mrename_i h
|
||
mcases h with ⟨⌜hr₂⌝, □h⟩
|
||
mspec op.spec
|
||
mrename_i h
|
||
mcases h with ⟨⌜hr₃⌝, □h⟩
|
||
mspec
|
||
mrefine ⟨?_, h⟩
|
||
mpure_intro
|
||
omega
|
||
|
||
end WeNeedAProofMode
|
||
|
||
end ByHand
|
||
|
||
namespace Automated
|
||
|
||
open Code
|
||
|
||
theorem fib_triple : ⦃⌜True⌝⦄ fib_impl n ⦃⇓ r => ⌜r = fib_spec n⌝⦄ := by
|
||
unfold fib_impl
|
||
mvcgen
|
||
case inv1 => exact ⇓ (xs, ⟨a, b⟩) =>
|
||
⌜a = fib_spec xs.pos ∧ b = fib_spec (xs.pos + 1)⌝
|
||
all_goals simp_all +zetaDelta [Nat.sub_one_add_one]
|
||
|
||
theorem fib_triple_step : ⦃⌜True⌝⦄ fib_impl n ⦃⇓ r => ⌜r = fib_spec n⌝⦄ := by
|
||
unfold fib_impl
|
||
mvcgen (stepLimit := some 14) -- 13 still has a wp⟦·⟧
|
||
case inv1 => exact ⇓ ⟨xs, a, b⟩ =>
|
||
⌜a = fib_spec xs.pos ∧ b = fib_spec (xs.pos + 1)⌝
|
||
all_goals simp_all +zetaDelta [Nat.sub_one_add_one]
|
||
|
||
attribute [local spec] fib_triple in
|
||
theorem fib_triple_attr : ⦃⌜True⌝⦄ fib_impl n ⦃⇓ r => ⌜r = fib_spec n⌝⦄ := by
|
||
mvcgen
|
||
|
||
attribute [local spec] fib_triple in
|
||
theorem fib_triple_erase : ⦃⌜True⌝⦄ fib_impl n ⦃⇓ r => ⌜r = fib_spec n⌝⦄ := by
|
||
mvcgen [-fib_triple] -- should not make any progress
|
||
fail_if_success done
|
||
admit
|
||
|
||
theorem fib_impl_vcs
|
||
(Q : Nat → PostCond Nat PostShape.pure)
|
||
(I : (n : Nat) → (_ : ¬n = 0) →
|
||
Invariant [1:n].toList (Prod Nat Nat) PostShape.pure)
|
||
(ret : ⊢ₛ (Q 0).1 0)
|
||
(loop_pre : ∀ n (hn : ¬n = 0), ⊢ₛ (I n hn).1 ⟨⟨[], [1:n].toList, rfl⟩, 0, 1⟩)
|
||
(loop_post : ∀ n (hn : ¬n = 0) r, (I n hn).1 ⟨⟨[1:n].toList, [], by simp⟩, r⟩ ⊢ₛ (Q n).1 r.2)
|
||
(loop_step : ∀ n (hn : ¬n = 0) r pref cur suff (h : [1:n].toList = pref ++ cur :: suff),
|
||
(I n hn).1 ⟨⟨pref, cur::suff, by simp[h]⟩, r⟩ ⊢ₛ (I n hn).1 ⟨⟨pref ++ [cur], suff, by simp[h]⟩, r.2, r.1+r.2⟩)
|
||
: ⊢ₛ wp⟦fib_impl n⟧ (Q n) := by
|
||
unfold fib_impl
|
||
mvcgen
|
||
case inv1 h => exact I n h
|
||
case vc1 h => subst h; apply_rules [ret]
|
||
case vc2 h => apply_rules [loop_pre]
|
||
case vc3 => apply_rules [loop_step]
|
||
case vc4 => apply_rules [loop_post]
|
||
|
||
@[spec]
|
||
theorem mkFreshNat_spec [Monad m] [WPMonad m sh] :
|
||
⦃fun s => ⌜s.1 = n ∧ s.2 = o⌝⦄
|
||
(mkFreshNat : StateT AppState m Nat)
|
||
⦃⇓ r s => ⌜r = n ∧ s.1 = n + 1 ∧ s.2 = o⌝⦄ := by
|
||
mvcgen [mkFreshNat]
|
||
simp_all +zetaDelta
|
||
|
||
theorem erase_unfold [Monad m] [WPMonad m sh] :
|
||
⦃fun s => ⌜s.1 = n ∧ s.2 = o⌝⦄
|
||
(mkFreshNat : StateT AppState m Nat)
|
||
⦃⇓ r s => ⌜r = n ∧ s.1 = n + 1 ∧ s.2 = o⌝⦄ := by
|
||
unfold mkFreshNat
|
||
mvcgen [-modify]
|
||
simp_all [-WP.modify_MonadStateOf]
|
||
fail_if_success done
|
||
admit
|
||
|
||
theorem add_unfold [Monad m] [WPMonad m sh] :
|
||
⦃fun s => ⌜s.1 = n ∧ s.2 = o⌝⦄
|
||
(mkFreshNat : StateT AppState m Nat)
|
||
⦃⇓ r s => ⌜r = n ∧ s.1 = n + 1 ∧ s.2 = o⌝⦄ := by
|
||
mvcgen [mkFreshNat]
|
||
|
||
theorem mkFreshPair_triple : ⦃⌜True⌝⦄ mkFreshPair ⦃⇓ (a, b) => ⌜a ≠ b⌝⦄ := by
|
||
mvcgen -elimLets +trivial [mkFreshPair]
|
||
-- this tests whether `mSpec` immediately discharges by `rfl` and `And.intro` and in the process
|
||
-- eagerly instantiates some schematic variables.
|
||
simp_all
|
||
|
||
theorem sum_loop_spec :
|
||
⦃⌜True⌝⦄
|
||
sum_loop
|
||
⦃⇓r => ⌜r < 30⌝⦄ := by
|
||
-- cf. `ByHand.sum_loop_spec`
|
||
mintro -
|
||
mvcgen [sum_loop]
|
||
case inv1 => exact (⇓ (xs, r) => ⌜r + xs.suffix.length * 5 ≤ 25⌝)
|
||
all_goals simp_all; try grind
|
||
|
||
theorem throwing_loop_spec :
|
||
⦃fun s => ⌜s = 4⌝⦄
|
||
throwing_loop
|
||
⦃post⟨fun _ _ => ⌜False⌝,
|
||
fun e s => ⌜e = 42 ∧ s = 4⌝⟩⦄ := by
|
||
mvcgen [throwing_loop]
|
||
case inv1 => exact post⟨fun (xs, r) s => ⌜r ≤ 4 ∧ s = 4 ∧ r + xs.suffix.sum > 4⌝,
|
||
fun e s => ⌜e = 42 ∧ s = 4⌝⟩
|
||
all_goals mleave; try (subst_vars; grind)
|
||
|
||
theorem test_loop_break :
|
||
⦃fun s => ⌜s = 42⌝⦄
|
||
breaking_loop
|
||
⦃⇓ r s => ⌜r > 4 ∧ s = 1⌝⦄ := by
|
||
mvcgen [breaking_loop]
|
||
case inv1 => exact (⇓ (xs, r) s => ⌜(r ≤ 4 ∧ r = xs.prefix.sum ∨ r > 4) ∧ s = 42⌝)
|
||
all_goals mleave; try grind
|
||
|
||
theorem test_loop_early_return :
|
||
⦃fun s => ⌜s = 4⌝⦄
|
||
returning_loop
|
||
⦃⇓ r s => ⌜r = 42 ∧ s = 4⌝⦄ := by
|
||
mvcgen [returning_loop]
|
||
case inv1 => exact (⇓ (xs, r) s => ⌜(r.1 = none ∧ r.2 = xs.prefix.sum ∧ r.2 ≤ 4 ∨ r.1 = some 42 ∧ r.2 > 4) ∧ s = 4⌝)
|
||
all_goals simp_all; try grind
|
||
|
||
theorem unfold_to_expose_match_spec :
|
||
⦃fun s => ⌜s = 4⌝⦄
|
||
unfold_to_expose_match
|
||
⦃⇓ r => ⌜r = 4⌝⦄ := by
|
||
-- should unfold `Option.getD`, reduce the `match (some get) with | some e => e`
|
||
-- and then apply the spec for `get`.
|
||
mvcgen [unfold_to_expose_match, Option.getD]
|
||
|
||
theorem test_match_splitting {m : Option Nat} (h : m = some 4) :
|
||
⦃⌜True⌝⦄
|
||
(match m with
|
||
| some n => (set n : StateM Nat PUnit)
|
||
| none => set 0)
|
||
⦃⇓ _ s => ⌜s = 4⌝⦄ := by
|
||
mvcgen <;> simp_all
|
||
|
||
theorem test_sum :
|
||
⦃⌜True⌝⦄
|
||
(do
|
||
let mut x := 0
|
||
for i in [1:5] do
|
||
x := x + i
|
||
pure x : Id _)
|
||
⦃⇓r => ⌜r < 30⌝⦄ := by
|
||
mvcgen
|
||
case inv1 => exact (⇓ (xs, r) => ⌜r + xs.suffix.length * 5 ≤ 25⌝)
|
||
all_goals simp_all; try grind
|
||
|
||
/--
|
||
The main point about this test is that `mSpec` should return all unassigned MVars it creates.
|
||
This used to be untrue because `mkForallFVars` etc. would instantiate MVars and introduce new
|
||
unassigned MVars in turn. It is important for `mSpec` to return these new MVars, otherwise
|
||
we get `(kernel) declaration has metavariables 'MPL.Test.VC.mspec_forwards_mvars'`
|
||
-/
|
||
theorem mspec_forwards_mvars {n : Nat} :
|
||
⦃⌜True⌝⦄
|
||
(do
|
||
for i in [2:n] do
|
||
if n < i * i then
|
||
return 1
|
||
return 1 : Id Nat)
|
||
⦃⇓ r => ⌜True⌝⦄ := by
|
||
mvcgen
|
||
all_goals admit
|
||
|
||
def check_all (p : Nat → Prop) [DecidablePred p] (n : Nat) : Bool := Id.run do
|
||
for i in [0:n] do
|
||
if ¬ p i then
|
||
return false
|
||
return true
|
||
|
||
example (p : Nat → Prop) [DecidablePred p] (n : Nat) :
|
||
(∀ i, i < n → p i) ↔ check_all p n := by
|
||
generalize h : check_all p n = x
|
||
apply Id.of_wp_run_eq h
|
||
mvcgen
|
||
case inv1 =>
|
||
exact Invariant.withEarlyReturnNewDo
|
||
(onReturn := fun ret _ => ⌜ret = false ∧ ¬ ∀ i < n, p i⌝)
|
||
(onContinue := fun xs _ => ⌜∀ i, i ∈ xs.prefix → p i⌝)
|
||
all_goals simp_all [-Classical.not_forall]; try grind
|
||
|
||
end Automated
|
||
|
||
namespace VSTTE2010
|
||
|
||
namespace MaxAndSum
|
||
|
||
def max_and_sum (xs : Array Nat) : Id (Nat × Nat) := do
|
||
let mut max := 0
|
||
let mut sum := 0
|
||
for h : i in [0:xs.size] do
|
||
sum := sum + xs[i]
|
||
if xs[i] > max then
|
||
max := xs[i]
|
||
return (max, sum)
|
||
|
||
theorem max_and_sum_spec (xs : Array Nat) :
|
||
⦃⌜∀ i, (h : i < xs.size) → xs[i] ≥ 0⌝⦄ max_and_sum xs ⦃⇓ (m, s) => ⌜s ≤ m * xs.size⌝⦄ := by
|
||
mvcgen [max_and_sum]
|
||
case inv1 => exact (⇓ ⟨xs, m, s⟩ => ⌜s ≤ m * xs.pos⌝)
|
||
all_goals simp_all +zetaDelta
|
||
· rw [Nat.left_distrib]
|
||
simp +zetaDelta only [Nat.mul_one, Nat.add_le_add_iff_right]
|
||
rename_i h
|
||
apply Nat.le_trans h
|
||
apply Nat.mul_le_mul_right
|
||
grind
|
||
· rw [Nat.left_distrib]
|
||
grind
|
||
|
||
end MaxAndSum
|
||
|
||
end VSTTE2010
|
||
|
||
namespace RishsConstApproxBug
|
||
|
||
@[spec]
|
||
theorem Spec.get_StateT' [Monad m] [WPMonad m psm] :
|
||
⦃fun s => Q.1 s s⦄ (MonadState.get : StateT σ m σ) ⦃Q⦄ := Spec.get_StateT
|
||
|
||
@[inline] def test : StateM Unit Unit := do
|
||
let _ ← get
|
||
if True then
|
||
pure ()
|
||
|
||
theorem need_const_approx :
|
||
⦃fun x => ⌜x = ()⌝⦄
|
||
test
|
||
⦃⇓ _ => ⌜True⌝⦄ := by
|
||
unfold test
|
||
mintro _
|
||
mspec
|
||
split <;> mspec
|
||
|
||
theorem need_const_approx' :
|
||
⦃fun x => ⌜x = ()⌝⦄
|
||
test
|
||
⦃⇓ _ => ⌜True⌝⦄ := by
|
||
mvcgen [test]
|
||
|
||
end RishsConstApproxBug
|
||
|
||
namespace RishsTailContextBug
|
||
|
||
axiom Specs.get_StateT' [Monad m] [WPMonad m psm] :
|
||
⦃fun s => Q.1 s s⦄ (MonadState.get : StateT σ m σ) ⦃Q⦄
|
||
attribute [local spec] Specs.get_StateT'
|
||
|
||
axiom I : StateM Nat Unit
|
||
axiom F : StateM Nat Unit
|
||
axiom G : StateM Nat Unit
|
||
axiom P : Assertion (PostShape.arg Nat PostShape.pure)
|
||
axiom Q: PostCond Unit (PostShape.arg Nat PostShape.pure)
|
||
axiom hI : ⦃⌜True⌝⦄ I ⦃⇓ _ => P⦄
|
||
axiom hF : ⦃P⦄ F ⦃Q⦄
|
||
axiom hG : ⦃P⦄ G ⦃Q⦄
|
||
attribute [local spec] hI hF hG
|
||
|
||
@[inline] noncomputable def test_ite : StateM Nat Unit := do
|
||
I
|
||
let n ← get
|
||
if n < 1 then
|
||
F
|
||
else
|
||
G
|
||
|
||
theorem ex : ⦃⌜True⌝⦄ test_ite ⦃Q⦄ := by
|
||
mvcgen [test_ite]
|
||
-- We used to get
|
||
-- s✝ : Nat
|
||
-- h : P s✝
|
||
-- a✝ : s✝ < 1
|
||
-- ⊢
|
||
-- h✝ : fun s => ⌜True⌝
|
||
-- ⊢ₛ P
|
||
-- and this is unsatisfiable. We need to remember the tail context `· s✝`.
|
||
-- The simplest way to do so is to use `split` in `mvcgen`, which we do now.
|
||
|
||
-- Same with explicit `split` and `mspec`
|
||
theorem ex': ⦃⌜True⌝⦄ test_ite ⦃Q⦄ := by
|
||
unfold test_ite
|
||
mintro _
|
||
mspec
|
||
mspec
|
||
mintro ∀ s
|
||
split <;> mspec
|
||
|
||
end RishsTailContextBug
|
||
|
||
namespace KimsUnivPolyUseCase
|
||
|
||
open Std
|
||
|
||
variable {α : Type u} {β : Type v} {cmp : α → α → Ordering} [TransCmp cmp]
|
||
|
||
def mergeWithAll (m₁ m₂ : ExtTreeMap α β cmp) (f : α → Option β → Option β → Option β) : ExtTreeMap α β cmp :=
|
||
Id.run do
|
||
let mut r := ∅
|
||
for (a, b₁) in m₁ do
|
||
if let some b := f a (some b₁) m₂[a]? then
|
||
r := r.insert a b
|
||
for (a, b₂) in m₂ do
|
||
if a ∉ m₁ then
|
||
if let some b := f a none (some b₂) then
|
||
r := r.insert a b
|
||
return r
|
||
|
||
theorem mem_mergeWithAll [LawfulEqCmp cmp] {m₁ m₂ : ExtTreeMap α β cmp} {f : α → Option β → Option β → Option β} {a : α} :
|
||
a ∈ mergeWithAll m₁ m₂ f ↔ (a ∈ m₁ ∨ a ∈ m₂) ∧ (f a m₁[a]? m₂[a]?).isSome := by
|
||
generalize h : mergeWithAll m₁ m₂ f = x
|
||
apply Id.of_wp_run_eq h
|
||
mvcgen
|
||
-- this was only to demonstrate that `Id.of_wp_run_eq` and `mvcgen` applies here despite the universe polymorphism
|
||
admit
|
||
|
||
end KimsUnivPolyUseCase
|
||
|
||
namespace Slices
|
||
|
||
def subarraySum (xs : Subarray Nat) : Nat := Id.run do
|
||
let mut sum := 0
|
||
for x in xs do
|
||
sum := sum + x
|
||
return sum
|
||
|
||
theorem subarraySum_correct {xs : Subarray Nat} : subarraySum xs = xs.toList.sum := by
|
||
generalize h : subarraySum xs = r
|
||
apply Id.of_wp_run_eq h
|
||
mvcgen
|
||
case inv1 => exact ⇓⟨cursor, prefixSum⟩ => ⌜prefixSum = cursor.prefix.sum⌝
|
||
all_goals simp_all
|
||
|
||
end Slices
|
||
|
||
namespace PatricksFastExp
|
||
|
||
def naive_expo (x n : Nat) : Nat := Id.run do
|
||
let mut y := 1
|
||
for _ in [:n] do
|
||
y := y*x
|
||
return y
|
||
|
||
def fast_expo (x n : Nat) : Nat := Id.run do
|
||
let mut x := x
|
||
let mut y := 1
|
||
let mut e := n
|
||
for _ in [:n] do -- simulating a while loop running at most n times
|
||
if e = 0 then break
|
||
if e % 2 = 1 then
|
||
y := x * y
|
||
e := e - 1
|
||
else
|
||
x := x*x
|
||
e := e/2
|
||
|
||
return y
|
||
|
||
theorem naive_expo_correct (x n : Nat) : naive_expo x n = x^n := by
|
||
generalize h : naive_expo x n = r
|
||
apply Id.of_wp_run_eq h
|
||
mvcgen
|
||
case inv1 => exact ⇓⟨xs, r⟩ => ⌜r = x^xs.pos⌝
|
||
all_goals simp_all [Nat.pow_add_one]
|
||
|
||
theorem fast_expo_correct (x n : Nat) : fast_expo x n = x^n := by
|
||
generalize h : fast_expo x n = r
|
||
apply Id.of_wp_run_eq h
|
||
mvcgen
|
||
case inv1 => exact ⇓⟨xs, x', y, e⟩ => ⌜x' ^ e * y = x ^ n ∧ e ≤ n - xs.pos⌝
|
||
all_goals simp_all +zetaDelta
|
||
case vc2 b _ _ _ _ _ _ ih =>
|
||
obtain ⟨x', y, e⟩ := b
|
||
simp at *
|
||
constructor
|
||
· rw [← Nat.mul_assoc, ← Nat.pow_add_one, ← ih.1]
|
||
have : e - 1 + 1 = e := by grind
|
||
rw [this]
|
||
· grind
|
||
case vc3 b _ _ _ _ _ ih _ =>
|
||
obtain ⟨x', y, e⟩ := b
|
||
simp at *
|
||
constructor
|
||
· rw [← Nat.pow_two, ← Nat.pow_mul]
|
||
grind
|
||
· grind
|
||
case vc5 b ih =>
|
||
obtain ⟨x', y, e⟩ := b
|
||
simp at *
|
||
rw [← ih.1, ih.2, Nat.pow_zero, Nat.one_mul]
|
||
|
||
theorem same_func (x n : Nat) : fast_expo x n = naive_expo x n := by
|
||
rw [naive_expo_correct, fast_expo_correct]
|
||
|
||
end PatricksFastExp
|
||
|
||
-- WIP example to reproduce a bug with delayed assignments
|
||
|
||
syntax (name := specialTactic) "specialTactic" : tactic
|
||
|
||
open Lean Meta Elab Tactic in
|
||
@[tactic specialTactic]
|
||
meta def evalSpecialTactic : Tactic := fun _ => do
|
||
let mv ← getMainGoal
|
||
let .forallE _ hpTy (.forallE _ hinvTy _ _) _ := (← mv.getType) | failure
|
||
let prf ←
|
||
withLocalDecl `hp .default hpTy fun hp => do
|
||
withLocalDecl `hinv .default hinvTy fun hinv => do
|
||
let n ← mkFreshExprMVar (mkConst ``Nat) .natural `n
|
||
let inv ← mkFreshExprSyntheticOpaqueMVar (← mkArrow (mkConst ``Nat) (mkSort 0)) `inv
|
||
let prf₁ ← mkFreshExprSyntheticOpaqueMVar (mkApp inv (mkNatLit 13)) `prf1
|
||
let hq := mkApp2 hinv inv prf₁
|
||
let prf₂ ← mkFreshExprSyntheticOpaqueMVar (← mkEq n (mkNatLit 42)) `prf2
|
||
replaceMainGoal <| [n, inv, prf₁, prf₂].map (·.mvarId!)
|
||
mkLambdaFVars #[hp, hinv] (mkApp3 hp n prf₂ hq)
|
||
mv.assign prf
|
||
|
||
example : (hp : ∀m, m = 42 → q → p) → (hinv : ∀ (inv : Nat → Prop), inv 13 → q) → p := by
|
||
specialTactic
|
||
-- intro hp
|
||
-- let ?n : Nat
|
||
-- let ?inv : Nat → Prop
|
||
-- let ?prf1 : inv 13
|
||
-- let hinv : inv 13 := ?prf1
|
||
-- let ?prf2 : n = 42
|
||
-- exact hp ?n ?prf2
|
||
case prf2 => rfl
|
||
case inv => exact fun _ => True
|
||
case prf1 => grind
|
||
|
||
variable {m} [Monad m]
|
||
open Std Std.Iterators
|
||
|
||
theorem forIn_eq_sum (xs : Array Nat) {m ps} [Monad m] [WPMonad m ps] :
|
||
Triple (m := m) (do
|
||
let mut sum : Nat := 0
|
||
for n in xs.iter do
|
||
sum := sum + n
|
||
return sum) ⌜True⌝ (⇓r => ⌜r = xs.sum⌝) := by
|
||
mvcgen
|
||
case inv1 => exact ⇓⟨cur, n⟩ => ⌜n = cur.prefix.sum⌝
|
||
all_goals grind
|
||
|
||
theorem forIn_map_eq_sum_add_size (xs : Array Nat) {m ps} [Monad m] [LawfulMonad m]
|
||
[WPMonad m ps] :
|
||
Triple (m := m) (do
|
||
let mut sum : Nat := 0
|
||
for n in (xs.iterM Id).map (· + 1) do
|
||
sum := sum + n
|
||
return sum) ⌜True⌝ (⇓r => ⌜r = xs.sum + xs.size⌝) := by
|
||
mvcgen
|
||
case inv1 => exact ⇓⟨cur, n⟩ => ⌜n = cur.prefix.sum + cur.prefix.length⌝
|
||
all_goals grind
|
||
|
||
theorem forIn_mapM_eq_sum_add_size (xs : Array Nat) {m ps} [Monad m] [MonadAttach m]
|
||
[LawfulMonad m] [WeaklyLawfulMonadAttach m] [WPMonad m ps] :
|
||
Triple (m := m) (do
|
||
let mut sum : Nat := 0
|
||
for n in (xs.iterM Id).mapM (pure (f := m) <| · + 1) do
|
||
sum := sum + n
|
||
return sum) ⌜True⌝ (⇓r => ⌜r = xs.sum + xs.size⌝) := by
|
||
mvcgen
|
||
case inv1 => exact ⇓⟨cur, n⟩ => ⌜n = cur.prefix.sum + cur.prefix.length⌝
|
||
all_goals grind
|
||
|
||
theorem forIn_filterMapM_eq_sum_add_size (xs : Array Nat) {m ps}
|
||
[Monad m] [LawfulMonad m] [MonadAttach m] [WeaklyLawfulMonadAttach m] [WPMonad m ps] :
|
||
Triple (m := m) (do
|
||
let mut sum : Nat := 0
|
||
for n in (xs.iterM Id).filterMapM (pure (f := m) <| some <| · + 1) do
|
||
sum := sum + n
|
||
return sum) ⌜True⌝ (⇓r => ⌜r = xs.sum + xs.size⌝) := by
|
||
mvcgen
|
||
case inv1 => exact ⇓⟨cur, n⟩ => ⌜n = cur.prefix.sum + cur.prefix.length⌝
|
||
all_goals grind
|
||
|
||
theorem foldM_eq_sum (xs : Array Nat) {m ps} [Monad m] [LawfulMonad m]
|
||
[WPMonad m ps] :
|
||
Triple (m := m)
|
||
(xs.iter.foldM (m := m) (init := 0) (pure <| · + ·))
|
||
⌜True⌝
|
||
(⇓r => ⌜r = xs.sum⌝) := by
|
||
mvcgen
|
||
case inv1 => exact ⇓⟨cur, n⟩ => ⌜n = cur.prefix.sum⌝
|
||
all_goals grind
|