Skip to content

Commit

Permalink
feat: Switch bvOmegaBench to use MetaM
Browse files Browse the repository at this point in the history
This allows us to gradually convert all our tactics to `MetaM`,
which provides the correct level to discipline to make sure
that bugs like #154 don't happen anymore.
Also, as a mild bonus, one hopes that eliminating `evalTactic` makes stuff a bit faster,
but I would not hold my breath on this.
  • Loading branch information
bollu committed Oct 11, 2024
1 parent 8baad8c commit 475d1c4
Show file tree
Hide file tree
Showing 2 changed files with 90 additions and 19 deletions.
1 change: 1 addition & 0 deletions Arm/Memory/Common.lean
Original file line number Diff line number Diff line change
Expand Up @@ -290,6 +290,7 @@ def omega (bvToNatSimpCtx : Simp.Context) (bvToNatSimprocs : Array Simp.Simprocs
TacticM.withTraceNode' m!"goal post `bv_toNat` reductions (Note: can be large)" do
trace[simp_mem.info] "{goal}"
-- @bollu: TODO: understand what precisely we are recovering from.
-- | TODO: replace with bvOmegaBench, and make THAT work in TacticM.
withoutRecover do
evalTactic (← `(tactic| bv_omega_bench))

Expand Down
108 changes: 89 additions & 19 deletions Tactics/BvOmegaBench.lean
Original file line number Diff line number Diff line change
Expand Up @@ -10,42 +10,112 @@ and allows us to send bug reports to the Lean developers.
-/
import Tactics.Attr
import Lean
open Lean Elab Meta Tactic
import Lean.Elab.Tactic.Omega.Frontend
import Lean.Elab.Tactic.Omega
import Tactics.Simp


open Lean Elab Meta Tactic Omega

namespace BvOmegaBench


-- Adapted mkSimpContext:
-- from https://github.com/leanprover/lean4/blob/master/src/Lean/Elab/Tactic/Simp.lean#L287
def bvOmegaSimpCtx : MetaM (Simp.Context × Array Simp.Simprocs) := do
let mut simprocs := #[]
let mut simpTheorems := #[]

let some ext ← (getSimpExtension? `bv_toNat)
| throwError m!"[omega] Error: unable to find `bv_toNat"
simpTheorems := simpTheorems.push (← ext.getTheorems)

if let some ext ← (Simp.getSimprocExtension? `bv_toNat) then
let s ← ext.getSimprocs
simprocs := simprocs.push s

let congrTheorems ← Meta.getSimpCongrTheorems
let config : Simp.Config := { failIfUnchanged := false }
let ctx : Simp.Context := { config, simpTheorems, congrTheorems }
return (ctx, simprocs)


/--
Run bv_omega, gather the results, and then store them at the value that is given by the option.
By default, it's stored at `pwd`, with filename `omega-bench`. The file is appended to,
with the goal state that is being run, and the time elapsed to solve the goal is written.
Code adapted from:
- https://github.com/leanprover/lean4/blob/master/src/Lean/Elab/Tactic/Simp.lean#L406
- https://github.com/leanprover/lean4/blob/master/src/Lean/Elab/Tactic/Omega/Frontend.lean#L685
-/
def run : TacticM Unit := do
def _run : TacticM Unit := do
let minMs ← getBvOmegaBenchMinMs
let goal ← getMainGoal
let goalStr ← ppGoal goal
let startTime ← IO.monoMsNow
try
withMainContext do
withoutRecover do
evalTactic (← `(tactic| bv_omega))
if !(← getBvOmegaBenchIsEnabled) then
return ()
let endTime ← IO.monoMsNow
let delta := endTime - startTime
let filePath ← getBvOmegaBenchFilePath
IO.FS.withFile filePath IO.FS.Mode.append fun h => do
if delta >= minMs then
h.putStrLn "\n---\n"
h.putStrLn s!"time"
h.putStrLn s!"{delta}"
h.putStrLn s!"endtime"
h.putStrLn s!"goal"
h.putStrLn goalStr.pretty
h.putStrLn s!"endgoal"
liftMetaFinishingTactic fun g => do
g.withContext do
let (bvToNatSimpCtx, bvToNatSimprocs) ← bvOmegaSimpCtx
let nonDepHyps ← g.getNondepPropHyps
let mut g := g

try
let (result?, _stats) ←
simpGoal g bvToNatSimpCtx bvToNatSimprocs
(simplifyTarget := true) (discharge? := .none) nonDepHyps
let .some (_, g') := result? | return ()
g := g'
trace[omega] "ran simpGoal, goal is now {indentD g}"
catch e =>
trace[omega] "ran simpGoal, got error {indentD e.toMessageData}"
pure ()

g.withContext do
let some g ← g.falseOrByContra | return ()
g.withContext do
let hyps := (← getLocalHyps).toList
trace[omega] "analyzing {hyps.length} hypotheses:\n{← hyps.mapM inferType}"
omega hyps g {}
-- evalTactic (← `(tactic| bv_omega))
if !(← getBvOmegaBenchIsEnabled) then return ()
let endTime ← IO.monoMsNow
let delta := endTime - startTime
let filePath ← getBvOmegaBenchFilePath
IO.FS.withFile filePath IO.FS.Mode.append fun h => do
if delta >= minMs then
h.putStrLn "\n---\n"
h.putStrLn s!"time"
h.putStrLn s!"{delta}"
h.putStrLn s!"endtime"
h.putStrLn s!"goal"
h.putStrLn goalStr.pretty
h.putStrLn s!"endgoal"
catch e =>
throw e
return ()

def run_ : TacticM Unit := do
focus do
let g ← getGoals
withMainContext do
-- dbg_trace (← m!"bv_omega_bench: goal before `simp only`: {← getMainGoal}".toString)
try
evalTactic (← `(tactic| simp only [bv_toNat] at *))
catch _ =>
setGoals g

evalTactic (← `(tactic| all_goals omega))

def run__ : TacticM Unit := do
focus do
evalTactic (← `(tactic| bv_omega))

def run : TacticM Unit := do
evalTactic (← `(tactic| bv_omega))


end BvOmegaBench

syntax (name := bvOmegaBenchTac) "bv_omega_bench" : tactic
Expand Down

0 comments on commit 475d1c4

Please sign in to comment.