2026-06-23 15:12:37 -05:00
|
|
|
|
import Spa.Analysis.Forward
|
2026-06-28 09:46:54 -05:00
|
|
|
|
import Spa.Lattice.Finset
|
2026-06-27 16:29:16 -05:00
|
|
|
|
import Spa.Language.Tagged.Graphs
|
2026-06-23 15:12:37 -05:00
|
|
|
|
import Spa.Showable
|
|
|
|
|
|
|
|
|
|
|
|
namespace Spa
|
|
|
|
|
|
|
|
|
|
|
|
open Forward
|
|
|
|
|
|
|
2026-06-28 09:46:54 -05:00
|
|
|
|
instance {n : ℕ} : Showable (Finset (Fin n)) :=
|
|
|
|
|
|
⟨fun s =>
|
2026-06-27 16:29:16 -05:00
|
|
|
|
"{" ++ (List.finRange n).foldr
|
2026-06-28 09:46:54 -05:00
|
|
|
|
(fun i rest => if i ∈ s then show' i ++ ", " ++ rest else rest) ""
|
2026-06-27 16:29:16 -05:00
|
|
|
|
++ "}"⟩
|
|
|
|
|
|
|
2026-06-28 09:46:54 -05:00
|
|
|
|
abbrev DefSet (prog : Program) : Type := Finset prog.NodeId
|
2026-06-23 15:12:37 -05:00
|
|
|
|
|
|
|
|
|
|
namespace ReachingAnalysis
|
|
|
|
|
|
|
|
|
|
|
|
variable (prog : Program)
|
|
|
|
|
|
|
2026-06-30 23:21:00 -05:00
|
|
|
|
def genSet (s : prog.State) : DefSet prog := (prog.nodeIdOf s).elim {} (fun x => {x})
|
|
|
|
|
|
|
|
|
|
|
|
def eval (s : prog.State) (vs : VariableValues (DefSet prog) prog) : VariableValues (DefSet prog) prog :=
|
|
|
|
|
|
match prog.code s with
|
|
|
|
|
|
| none => vs
|
|
|
|
|
|
| some bs =>
|
|
|
|
|
|
match bs with
|
|
|
|
|
|
| .assign k _ => FiniteMap.generalizedUpdate id (fun _ _ => genSet prog s) [k] vs
|
|
|
|
|
|
| .noop => vs
|
|
|
|
|
|
|
|
|
|
|
|
lemma eval_mono (s : prog.State) :
|
|
|
|
|
|
Monotone (eval prog s) := by
|
|
|
|
|
|
intros vs₁ vs₂ hle
|
|
|
|
|
|
unfold eval; split <;> try simpa
|
|
|
|
|
|
split <;> try simpa
|
|
|
|
|
|
apply FiniteMap.generalizedUpdate_monotone monotone_id (fun _ => monotone_const)
|
|
|
|
|
|
assumption
|
2026-06-23 15:12:37 -05:00
|
|
|
|
|
|
|
|
|
|
instance stmtEvaluator : StmtEvaluator (DefSet prog) prog :=
|
|
|
|
|
|
⟨eval prog, eval_mono prog⟩
|
|
|
|
|
|
|
|
|
|
|
|
def output : String :=
|
|
|
|
|
|
show' (result (DefSet prog) prog)
|
|
|
|
|
|
|
2026-07-02 09:01:09 -05:00
|
|
|
|
/-- The statements a trace executed, paired with the state each executed at,
|
|
|
|
|
|
most recent first (matching `LastAssign`, which scans for the most recent
|
|
|
|
|
|
assignment). This is `Trace.steps` (chronological) reversed, so facts about
|
|
|
|
|
|
concatenating traces reduce to mathlib's `List.append`/`List.reverse` lemmas. -/
|
|
|
|
|
|
abbrev Run (prog : Program) : Type := List (prog.State × BasicStmt)
|
2026-06-27 16:29:16 -05:00
|
|
|
|
|
2026-06-27 19:30:01 -05:00
|
|
|
|
@[aesop unsafe cases]
|
2026-06-27 16:29:16 -05:00
|
|
|
|
inductive LastAssign (prog : Program) (x : String) : Run prog → prog.NodeId → Prop
|
|
|
|
|
|
| here (s : prog.State) (e : Expr) (hc : prog.code s = some (.assign x e))
|
|
|
|
|
|
(rest : Run prog) :
|
2026-07-02 09:01:09 -05:00
|
|
|
|
LastAssign prog x ((s, .assign x e) :: rest) (prog.nodeIdOfNonempty s hc)
|
2026-06-27 16:29:16 -05:00
|
|
|
|
| there (s : prog.State) (bs : BasicStmt) (hc : prog.code s = some bs)
|
|
|
|
|
|
(rest : Run prog) {n : prog.NodeId} :
|
|
|
|
|
|
(∀ e, bs ≠ .assign x e) → LastAssign prog x rest n →
|
2026-07-02 09:01:09 -05:00
|
|
|
|
LastAssign prog x ((s, bs) :: rest) n
|
|
|
|
|
|
|
|
|
|
|
|
def runOfTraceₗ {s₁ s₂ : prog.State} {ρ₁ ρ₂ : Env}
|
|
|
|
|
|
(tr : Traceₗ prog.cfg s₁ s₂ ρ₁ ρ₂) : Run prog :=
|
|
|
|
|
|
tr.steps.reverse
|
|
|
|
|
|
|
|
|
|
|
|
def runOfTrace {s₁ s₂ : prog.State} {ρ₁ ρ₂ : Env}
|
|
|
|
|
|
(tr : Trace prog.cfg s₁ s₂ ρ₁ ρ₂) : Run prog :=
|
|
|
|
|
|
tr.steps.reverse
|
2026-06-27 16:29:16 -05:00
|
|
|
|
|
2026-06-29 10:00:01 -05:00
|
|
|
|
instance stateInterp : StateInterpretation (DefSet prog) prog where
|
2026-07-02 09:01:09 -05:00
|
|
|
|
Proj := Run prog
|
|
|
|
|
|
Pre := @runOfTraceₗ prog
|
|
|
|
|
|
Post := @runOfTrace prog
|
|
|
|
|
|
|
|
|
|
|
|
interp vs run := ∀ (x : String) (assigners : DefSet prog), (x, assigners) ∈ vs →
|
2026-06-28 09:46:54 -05:00
|
|
|
|
∀ (n : prog.NodeId), LastAssign prog x run n → n ∈ assigners
|
2026-06-27 16:29:16 -05:00
|
|
|
|
interp_sup := by
|
2026-07-02 09:01:09 -05:00
|
|
|
|
intro vs₁ vs₂ run h x assigners hmem n hla
|
2026-06-27 16:29:16 -05:00
|
|
|
|
obtain ⟨a₁, a₂, rfl, h₁, h₂⟩ := FiniteMap.mem_sup hmem
|
2026-06-28 09:46:54 -05:00
|
|
|
|
aesop (add simp Finset.mem_union)
|
2026-06-27 16:29:16 -05:00
|
|
|
|
interp_inf := by
|
2026-07-02 09:01:09 -05:00
|
|
|
|
intro vs₁ vs₂ run h x assigners hmem n hla
|
2026-06-27 16:29:16 -05:00
|
|
|
|
obtain ⟨a₁, a₂, rfl, h₁, h₂⟩ := FiniteMap.mem_inf hmem
|
2026-06-28 09:46:54 -05:00
|
|
|
|
aesop (add simp Finset.mem_inter)
|
2026-06-27 16:29:16 -05:00
|
|
|
|
|
2026-07-02 09:01:09 -05:00
|
|
|
|
post_pre := by
|
|
|
|
|
|
intro vs s₁ s₂ s₃ ρ₁ ρ₂ tr hedge hvs
|
|
|
|
|
|
simpa [runOfTrace, runOfTraceₗ] using hvs
|
|
|
|
|
|
|
|
|
|
|
|
private lemma valid_step (s : prog.State) {ρ₁ ρ₂ : Env}
|
|
|
|
|
|
{obs : Option BasicStmt} (hcode : prog.code s = obs)
|
|
|
|
|
|
(hbs : EvalBasicStmtOpt ρ₁ obs ρ₂)
|
|
|
|
|
|
{vs : VariableValues (DefSet prog) prog} {run : Run prog}
|
|
|
|
|
|
(hvs : ⟦vs⟧ run) :
|
|
|
|
|
|
⟦eval prog s vs⟧ ((hbs.steps s).reverse ++ run) := by
|
|
|
|
|
|
cases hbs with
|
|
|
|
|
|
| none => simpa [eval, hcode, EvalBasicStmtOpt.steps] using hvs
|
|
|
|
|
|
| some hbs =>
|
|
|
|
|
|
cases hbs with
|
|
|
|
|
|
| noop =>
|
|
|
|
|
|
simp [eval, hcode, EvalBasicStmtOpt.steps]
|
|
|
|
|
|
intro x assigners hmem n hla; aesop
|
|
|
|
|
|
| assign x e v hev =>
|
|
|
|
|
|
simp [eval, hcode, EvalBasicStmtOpt.steps]; intro k assigners hmem n hla
|
|
|
|
|
|
by_cases hx : k = x
|
|
|
|
|
|
· subst hx
|
|
|
|
|
|
have hd := FiniteMap.generalizedUpdate_mem_eq (List.mem_singleton.mpr rfl) hmem
|
|
|
|
|
|
rcases hla
|
|
|
|
|
|
<;> simp [Program.nodeIdOfNonempty, hd, genSet, Option.get] <;> aesop
|
|
|
|
|
|
· have hmem' := FiniteMap.generalizedUpdate_not_mem_backward
|
|
|
|
|
|
(fun hc => hx (List.mem_singleton.mp hc)) hmem
|
|
|
|
|
|
aesop
|
2026-06-30 23:21:00 -05:00
|
|
|
|
|
2026-06-27 16:29:16 -05:00
|
|
|
|
instance validStateEvaluator : ValidStateEvaluator (DefSet prog) prog where
|
|
|
|
|
|
valid := by
|
2026-07-02 09:01:09 -05:00
|
|
|
|
intro s₁ s₂ ρ₁ ρ₂ ρ₃ vs tr hbs hvs
|
|
|
|
|
|
show ⟦eval prog s₂ vs⟧ (runOfTrace prog (tr ++ hbs))
|
|
|
|
|
|
simpa [runOfTrace, runOfTraceₗ] using valid_step prog s₂ rfl hbs hvs
|
2026-06-27 16:29:16 -05:00
|
|
|
|
botV_init := by intro x assigners _ n hla; cases hla
|
|
|
|
|
|
|
|
|
|
|
|
theorem analyze_correct {ρ : Env} (hrun : EvalStmt [] prog.rootStmt ρ) :
|
2026-07-02 09:01:09 -05:00
|
|
|
|
⟦ variablesAt prog.finalState (result (DefSet prog) prog) ⟧
|
|
|
|
|
|
(runOfTrace prog (prog.trace hrun)) :=
|
|
|
|
|
|
Forward.analyze_correct' (DefSet prog) prog hrun
|
2026-06-27 16:29:16 -05:00
|
|
|
|
|
2026-06-28 14:24:46 -05:00
|
|
|
|
theorem analyze_correct_at {ρf : Env} (hrun : EvalStmt [] prog.rootStmt ρf)
|
2026-07-02 09:01:09 -05:00
|
|
|
|
{s : prog.State} {ρin ρout : Env}
|
|
|
|
|
|
(hr : Reaches (prog.trace hrun) s ρin ρout) :
|
|
|
|
|
|
⟦ joinForKey s (result (DefSet prog) prog) ⟧ (runOfTraceₗ prog hr.pre)
|
|
|
|
|
|
∧ ⟦ variablesAt s (result (DefSet prog) prog) ⟧ (runOfTrace prog hr.post) :=
|
2026-06-28 14:24:46 -05:00
|
|
|
|
Forward.analyze_correct_at (DefSet prog) prog hrun hr
|
|
|
|
|
|
|
2026-06-23 15:12:37 -05:00
|
|
|
|
end ReachingAnalysis
|
|
|
|
|
|
|
|
|
|
|
|
end Spa
|