Skip to content

Commit

Permalink
refactor: reimplement initNextStep without evalTactic
Browse files Browse the repository at this point in the history
  • Loading branch information
alexkeizer committed Sep 27, 2024
1 parent 4e90fe1 commit 5bf77e7
Show file tree
Hide file tree
Showing 4 changed files with 50 additions and 22 deletions.
7 changes: 7 additions & 0 deletions Arm/Exec.lean
Original file line number Diff line number Diff line change
Expand Up @@ -162,6 +162,13 @@ theorem run_onestep {s s': ArmState} {n : Nat} :
(s' = run (n + 1) s) → ∃ s'', stepi s = s'' ∧ s' = run n s'' := by
simp only [run, exists_eq_left', imp_self]

/-- helper lemma for automation -/
theorem run_of_run_succ_of_stepi_eq {s0 s1 sf : ArmState} {n : Nat}
(h_run : sf = run (n + 1) s0)
(h_stepi : stepi s0 = s1) :
sf = run n s1 := by
simp_all only [run]

/-- helper lemma for automation -/
theorem stepi_eq_of_fetch_inst_of_decode_raw_inst
(s : ArmState) (addr : BitVec 64) (rawInst : BitVec 32) (inst : ArmInst)
Expand Down
57 changes: 36 additions & 21 deletions Tactics/Sym.lean
Original file line number Diff line number Diff line change
Expand Up @@ -160,37 +160,47 @@ def unfoldRun (whileTac : Unit → TacticM Unit) : SymM Expr := do
setGoals (rwRes.mvarIds ++ originalGoals ++ newGoal)
instantiateMVars runStepsPred

/--
/-- TODO: better docstring
`initNextStep` returns expressions
- `sn : ArmState`, and
- `stepiEq : stepi <currentState> = sn`
- `nextState : ArmState`, and
- `stepiEq : stepi <currentState> = nextState`
In that order, it also modifies `hRun` to be of type:
`<finalState> = hRun _ sn`
-/
def initNextStep (whileTac : TSyntax `tactic) : SymM (Expr × Expr) :=
withMainContext' do
let stepi_eq := Lean.mkIdent (.mkSimple s!"stepi_{← getCurrentStateName}")
let goal ← getMainGoal

-- Add next state to local context
let currentState ← getCurrentState
let nextStateVal := -- `stepi <currentState>`
mkApp (mkConst ``stepi) currentState
let (nextStateId, goal) ← do
let name ← getNextStateName
goal.note name nextStateVal mkArmState
let nextState := Expr.fvar nextStateId

let stepiEq : Expr := -- `stepiEq : stepi <currentState> = nextState`
let ty := mkEqArmState nextStateVal nextState
mkApp2 (.const ``id [0]) ty <| mkEqReflArmState nextState
let (_, goal) ← do
let name := Name.mkSimple s!"stepi_{← getCurrentStateName}"
goal.note name stepiEq
replaceMainGoal [goal]

-- Ensure we have one more step to simulate
let runStepsPred ← unfoldRun (fun _ => evalTacticAndTrace whileTac)
-- Add new state to local context
let hRunId := mkIdent <|← getHRunName
let nextStateId := mkIdent <|← getNextStateName
withMainContext' <| evalTacticAndTrace <|← `(tactic|
init_next_step $hRunId:ident $stepi_eq:ident $nextStateId:ident
)
withMainContext' <| do
-- Update `hRun`
let hRun := hRunId.getId
let some hRun := (← getLCtx).findFromUserName? hRun
| throwError "internal error: couldn't find {hRun}"
modifyThe SymContext ({ · with
hRun := hRun.toExpr
})

let sn ← SymContext.findFromUserName nextStateId.getId
let stepiEq ← SymContext.findFromUserName stepi_eq.getId
-- Change `hRun`
let hRun ← getHRun
let hRun := -- `run_of_run_succ_of_stepi_eq <hRun> <stepiEq>`
mkAppN (mkConst ``run_of_run_succ_of_stepi_eq) #[
currentState, nextState, ← getFinalState, runStepsPred,
hRun, stepiEq
]
modifyThe SymContext ({ · with hRun })

return (sn.toExpr, stepiEq.toExpr)
return (nextState, stepiEq)

/-- Break an equality `h_step : s{i+1} = w ... (... (w ... s{i})...)` into an
`AxEffects` that characterizes the effects in terms of reads from `s{i+1}`,
Expand Down Expand Up @@ -461,6 +471,11 @@ elab "sym_n" whileTac?:(sym_while)? n:num s:(sym_at)? : tactic => do
trace[Tactic.sym] "performed subsitutition in:\n{goal}"

replaceMainGoal [goal]
else -- Replace `h_run` in the local context
let goal ← getMainGoal
let res ← goal.replace c.hRunId c.hRun
replaceMainGoal [res.mvarId]


-- Rudimentary aggregation: we feed all the axiomatic effect hypotheses
-- added while symbolically evaluating to `simp`
Expand Down
2 changes: 1 addition & 1 deletion Tactics/Sym/AxEffects.lean
Original file line number Diff line number Diff line change
Expand Up @@ -105,7 +105,7 @@ def getCurrentStateName : m Name := do
@id (MetaM _) <| do
let state ← instantiateMVars state
let Expr.fvar id := state.consumeMData
| throwError "error: expected a free variable, found:\n {state} WHHOPS"
| throwError "error: expected a free variable, found:\n {state}"
let lctx ← getLCtx
let some decl := lctx.find? id
| throwError "error: unknown fvar: {state}"
Expand Down
6 changes: 6 additions & 0 deletions Tactics/Sym/Context.lean
Original file line number Diff line number Diff line change
Expand Up @@ -52,6 +52,9 @@ structure SymContext where
See also `SymContext.runSteps?` -/
hRun : Expr
/-- The id of the variable with which `hRun` was initialized -/
hRunId : FVarId

/-- `programInfo` is the relevant cached `ProgramInfo` -/
programInfo : ProgramInfo

Expand Down Expand Up @@ -167,6 +170,8 @@ variable {m} [Monad m] [MonadReaderOf SymContext m]

def getCurrentStateNumber : m Nat := do return (← read).currentStateNumber

def getFinalState : m Expr := do return (← read).finalState

/-- Return an expression of type
`<finalState> = run <runSteps> <initialState>` -/
def getHRun : m Expr := do return (← read).hRun
Expand Down Expand Up @@ -367,6 +372,7 @@ def fromLocalContext (state? : Option Name) : TacticM SymContext := do
let c : SymContext := {
finalState, runSteps?, pc,
hRun := h_run.toExpr,
hRunId := h_run.fvarId,
h_sp? := (·.userName) <$> h_sp?,
programInfo,
effects,
Expand Down

0 comments on commit 5bf77e7

Please sign in to comment.