2024-04-13 18:39:38 -07:00
|
|
|
|
open import Language hiding (_[_])
|
2024-03-22 17:50:29 -07:00
|
|
|
|
open import Lattice
|
|
|
|
|
|
|
|
|
|
module Analysis.Forward
|
2025-01-04 21:16:22 -08:00
|
|
|
|
(L : Set) {h}
|
2024-03-22 17:50:29 -07:00
|
|
|
|
{_≈ˡ_ : L → L → Set} {_⊔ˡ_ : L → L → L} {_⊓ˡ_ : L → L → L}
|
2025-01-04 21:16:22 -08:00
|
|
|
|
{{isFiniteHeightLatticeˡ : IsFiniteHeightLattice L h _≈ˡ_ _⊔ˡ_ _⊓ˡ_}}
|
|
|
|
|
{{≈ˡ-dec : IsDecidable _≈ˡ_}} where
|
2024-03-22 17:50:29 -07:00
|
|
|
|
|
2024-05-08 20:50:21 -07:00
|
|
|
|
open import Data.Empty using (⊥-elim)
|
2025-01-05 19:35:56 -08:00
|
|
|
|
open import Data.Unit using (⊤)
|
2024-12-31 17:31:01 -08:00
|
|
|
|
open import Data.String using (String)
|
|
|
|
|
open import Data.Product using (_,_)
|
|
|
|
|
open import Data.List using (_∷_; []; foldr; foldl)
|
2024-05-08 20:50:21 -07:00
|
|
|
|
open import Data.List.Relation.Unary.Any as Any using ()
|
2024-12-31 17:31:01 -08:00
|
|
|
|
open import Relation.Binary.PropositionalEquality using (_≡_; refl; cong; sym; subst)
|
|
|
|
|
open import Relation.Nullary using (yes; no)
|
2024-05-08 21:50:38 -07:00
|
|
|
|
open import Function using (_∘_; flip)
|
2024-03-22 17:50:29 -07:00
|
|
|
|
|
|
|
|
|
open IsFiniteHeightLattice isFiniteHeightLatticeˡ
|
2024-12-31 17:31:01 -08:00
|
|
|
|
using () renaming (isLattice to isLatticeˡ)
|
2024-03-22 17:50:29 -07:00
|
|
|
|
|
|
|
|
|
module WithProg (prog : Program) where
|
2025-01-04 21:16:22 -08:00
|
|
|
|
open import Analysis.Forward.Lattices L prog hiding (≈ᵛ-Decidable) -- to disambiguate instance resolution
|
|
|
|
|
open import Analysis.Forward.Evaluation L prog
|
2024-03-22 17:50:29 -07:00
|
|
|
|
open Program prog
|
|
|
|
|
|
2024-12-31 17:31:01 -08:00
|
|
|
|
private module WithStmtEvaluator {{evaluator : StmtEvaluator}} where
|
|
|
|
|
open StmtEvaluator evaluator
|
2024-04-13 14:08:40 -07:00
|
|
|
|
|
2024-03-22 17:50:29 -07:00
|
|
|
|
updateVariablesForState : State → StateVariables → VariableValues
|
2024-04-13 14:08:40 -07:00
|
|
|
|
updateVariablesForState s sv =
|
2024-12-31 17:31:01 -08:00
|
|
|
|
foldl (flip (eval s)) (variablesAt s sv) (code s)
|
2024-03-22 17:50:29 -07:00
|
|
|
|
|
|
|
|
|
updateVariablesForState-Monoʳ : ∀ (s : State) → Monotonic _≼ᵐ_ _≼ᵛ_ (updateVariablesForState s)
|
2024-04-13 14:08:40 -07:00
|
|
|
|
updateVariablesForState-Monoʳ s {sv₁} {sv₂} sv₁≼sv₂ =
|
|
|
|
|
let
|
|
|
|
|
bss = code s
|
|
|
|
|
(vs₁ , s,vs₁∈sv₁) = locateᵐ {s} {sv₁} (states-in-Map s sv₁)
|
|
|
|
|
(vs₂ , s,vs₂∈sv₂) = locateᵐ {s} {sv₂} (states-in-Map s sv₂)
|
|
|
|
|
vs₁≼vs₂ = m₁≼m₂⇒m₁[k]ᵐ≼m₂[k]ᵐ sv₁ sv₂ sv₁≼sv₂ s,vs₁∈sv₁ s,vs₂∈sv₂
|
|
|
|
|
in
|
2024-05-08 21:50:38 -07:00
|
|
|
|
foldl-Mono' (IsLattice.joinSemilattice isLatticeᵛ) bss
|
2024-12-31 17:31:01 -08:00
|
|
|
|
(flip (eval s)) (eval-Monoʳ s)
|
2024-04-13 14:08:40 -07:00
|
|
|
|
vs₁≼vs₂
|
2024-03-22 17:50:29 -07:00
|
|
|
|
|
2025-01-04 21:16:22 -08:00
|
|
|
|
open StateVariablesFiniteMap.GeneralizedUpdate {{isLatticeᵐ}} (λ x → x) (λ a₁≼a₂ → a₁≼a₂) updateVariablesForState updateVariablesForState-Monoʳ states
|
2024-12-31 00:21:10 -08:00
|
|
|
|
using ()
|
2024-03-22 17:50:29 -07:00
|
|
|
|
renaming
|
|
|
|
|
( f' to updateAll
|
|
|
|
|
; f'-Monotonic to updateAll-Mono
|
2024-05-08 22:29:36 -07:00
|
|
|
|
; f'-k∈ks-≡ to updateAll-k∈ks-≡
|
2024-03-22 17:50:29 -07:00
|
|
|
|
)
|
2024-12-31 00:21:10 -08:00
|
|
|
|
public
|
2024-03-22 17:50:29 -07:00
|
|
|
|
|
2024-03-23 12:09:14 -07:00
|
|
|
|
-- Finally, the whole analysis consists of getting the 'join'
|
2024-03-22 17:50:29 -07:00
|
|
|
|
-- of all incoming states, then applying the per-state evaluation
|
|
|
|
|
-- function. This is just a composition, and is trivially monotonic.
|
|
|
|
|
|
|
|
|
|
analyze : StateVariables → StateVariables
|
|
|
|
|
analyze = updateAll ∘ joinAll
|
|
|
|
|
|
|
|
|
|
analyze-Mono : Monotonic _≼ᵐ_ _≼ᵐ_ analyze
|
2024-03-23 12:09:14 -07:00
|
|
|
|
analyze-Mono {sv₁} {sv₂} sv₁≼sv₂ =
|
|
|
|
|
updateAll-Mono {joinAll sv₁} {joinAll sv₂}
|
|
|
|
|
(joinAll-Mono {sv₁} {sv₂} sv₁≼sv₂)
|
2024-03-22 17:50:29 -07:00
|
|
|
|
|
|
|
|
|
-- The fixed point of the 'analyze' function is our final goal.
|
2025-01-04 18:58:56 -08:00
|
|
|
|
open import Fixedpoint ≈ᵐ-Decidable isFiniteHeightLatticeᵐ analyze (λ {m₁} {m₂} m₁≼m₂ → analyze-Mono {m₁} {m₂} m₁≼m₂)
|
2024-03-22 17:50:29 -07:00
|
|
|
|
using ()
|
2024-05-08 22:53:21 -07:00
|
|
|
|
renaming (aᶠ to result; aᶠ≈faᶠ to result≈analyze-result)
|
2024-03-22 17:50:29 -07:00
|
|
|
|
public
|
2024-05-08 20:50:21 -07:00
|
|
|
|
|
2024-05-08 22:29:36 -07:00
|
|
|
|
variablesAt-updateAll : ∀ (s : State) (sv : StateVariables) →
|
|
|
|
|
variablesAt s (updateAll sv) ≡ updateVariablesForState s sv
|
|
|
|
|
variablesAt-updateAll s sv
|
|
|
|
|
with (vs , s,vs∈usv) ← locateᵐ {s} {updateAll sv} (states-in-Map s (updateAll sv)) =
|
|
|
|
|
updateAll-k∈ks-≡ {l = sv} (states-complete s) s,vs∈usv
|
|
|
|
|
|
2024-12-31 17:31:01 -08:00
|
|
|
|
module WithValidInterpretation {{latticeInterpretationˡ : LatticeInterpretation isLatticeˡ}}
|
2025-01-05 19:35:56 -08:00
|
|
|
|
{{validEvaluator : ValidStmtEvaluator evaluator latticeInterpretationˡ}} (dummy : ⊤) where
|
2024-12-31 17:31:01 -08:00
|
|
|
|
open ValidStmtEvaluator validEvaluator
|
|
|
|
|
|
|
|
|
|
eval-fold-valid : ∀ {s bss vs ρ₁ ρ₂} → ρ₁ , bss ⇒ᵇˢ ρ₂ → ⟦ vs ⟧ᵛ ρ₁ → ⟦ foldl (flip (eval s)) vs bss ⟧ᵛ ρ₂
|
|
|
|
|
eval-fold-valid {_} [] ⟦vs⟧ρ = ⟦vs⟧ρ
|
|
|
|
|
eval-fold-valid {s} {bs ∷ bss'} {vs} {ρ₁} {ρ₂} (ρ₁,bs⇒ρ ∷ ρ,bss'⇒ρ₂) ⟦vs⟧ρ₁ =
|
|
|
|
|
eval-fold-valid
|
|
|
|
|
{bss = bss'} {eval s bs vs} ρ,bss'⇒ρ₂
|
|
|
|
|
(valid ρ₁,bs⇒ρ ⟦vs⟧ρ₁)
|
|
|
|
|
|
|
|
|
|
updateVariablesForState-matches : ∀ {s sv ρ₁ ρ₂} → ρ₁ , (code s) ⇒ᵇˢ ρ₂ → ⟦ variablesAt s sv ⟧ᵛ ρ₁ → ⟦ updateVariablesForState s sv ⟧ᵛ ρ₂
|
|
|
|
|
updateVariablesForState-matches = eval-fold-valid
|
|
|
|
|
|
|
|
|
|
updateAll-matches : ∀ {s sv ρ₁ ρ₂} → ρ₁ , (code s) ⇒ᵇˢ ρ₂ → ⟦ variablesAt s sv ⟧ᵛ ρ₁ → ⟦ variablesAt s (updateAll sv) ⟧ᵛ ρ₂
|
|
|
|
|
updateAll-matches {s} {sv} ρ₁,bss⇒ρ₂ ⟦vs⟧ρ₁
|
|
|
|
|
rewrite variablesAt-updateAll s sv =
|
|
|
|
|
updateVariablesForState-matches {s} {sv} ρ₁,bss⇒ρ₂ ⟦vs⟧ρ₁
|
|
|
|
|
|
|
|
|
|
stepTrace : ∀ {s₁ ρ₁ ρ₂} → ⟦ joinForKey s₁ result ⟧ᵛ ρ₁ → ρ₁ , (code s₁) ⇒ᵇˢ ρ₂ → ⟦ variablesAt s₁ result ⟧ᵛ ρ₂
|
|
|
|
|
stepTrace {s₁} {ρ₁} {ρ₂} ⟦joinForKey-s₁⟧ρ₁ ρ₁,bss⇒ρ₂ =
|
|
|
|
|
let
|
|
|
|
|
-- I'd use rewrite, but Agda gets a memory overflow (?!).
|
|
|
|
|
⟦joinAll-result⟧ρ₁ =
|
|
|
|
|
subst (λ vs → ⟦ vs ⟧ᵛ ρ₁)
|
|
|
|
|
(sym (variablesAt-joinAll s₁ result))
|
|
|
|
|
⟦joinForKey-s₁⟧ρ₁
|
|
|
|
|
⟦analyze-result⟧ρ₂ =
|
|
|
|
|
updateAll-matches {sv = joinAll result}
|
|
|
|
|
ρ₁,bss⇒ρ₂ ⟦joinAll-result⟧ρ₁
|
|
|
|
|
analyze-result≈result =
|
|
|
|
|
≈ᵐ-sym {result} {updateAll (joinAll result)}
|
|
|
|
|
result≈analyze-result
|
|
|
|
|
analyze-s₁≈s₁ =
|
|
|
|
|
variablesAt-≈ s₁ (updateAll (joinAll result))
|
|
|
|
|
result (analyze-result≈result)
|
|
|
|
|
in
|
|
|
|
|
⟦⟧ᵛ-respects-≈ᵛ {variablesAt s₁ (updateAll (joinAll result))} {variablesAt s₁ result} (analyze-s₁≈s₁) ρ₂ ⟦analyze-result⟧ρ₂
|
|
|
|
|
|
|
|
|
|
walkTrace : ∀ {s₁ s₂ ρ₁ ρ₂} → ⟦ joinForKey s₁ result ⟧ᵛ ρ₁ → Trace {graph} s₁ s₂ ρ₁ ρ₂ → ⟦ variablesAt s₂ result ⟧ᵛ ρ₂
|
|
|
|
|
walkTrace {s₁} {s₁} {ρ₁} {ρ₂} ⟦joinForKey-s₁⟧ρ₁ (Trace-single ρ₁,bss⇒ρ₂) =
|
|
|
|
|
stepTrace {s₁} {ρ₁} {ρ₂} ⟦joinForKey-s₁⟧ρ₁ ρ₁,bss⇒ρ₂
|
|
|
|
|
walkTrace {s₁} {s₂} {ρ₁} {ρ₂} ⟦joinForKey-s₁⟧ρ₁ (Trace-edge {ρ₂ = ρ} {idx₂ = s} ρ₁,bss⇒ρ s₁→s₂ tr) =
|
2024-05-08 22:53:21 -07:00
|
|
|
|
let
|
2024-12-31 17:31:01 -08:00
|
|
|
|
⟦result-s₁⟧ρ =
|
|
|
|
|
stepTrace {s₁} {ρ₁} {ρ} ⟦joinForKey-s₁⟧ρ₁ ρ₁,bss⇒ρ
|
|
|
|
|
s₁∈incomingStates =
|
|
|
|
|
[]-∈ result (edge⇒incoming s₁→s₂)
|
|
|
|
|
(variablesAt-∈ s₁ result)
|
|
|
|
|
⟦joinForKey-s⟧ρ =
|
|
|
|
|
⟦⟧ᵛ-foldr ⟦result-s₁⟧ρ s₁∈incomingStates
|
2024-05-08 22:53:21 -07:00
|
|
|
|
in
|
2024-12-31 17:31:01 -08:00
|
|
|
|
walkTrace ⟦joinForKey-s⟧ρ tr
|
2024-12-31 00:21:10 -08:00
|
|
|
|
|
2024-12-31 17:31:01 -08:00
|
|
|
|
joinForKey-initialState-⊥ᵛ : joinForKey initialState result ≡ ⊥ᵛ
|
|
|
|
|
joinForKey-initialState-⊥ᵛ = cong (λ ins → foldr _⊔ᵛ_ ⊥ᵛ (result [ ins ])) initialState-pred-∅
|
2024-12-31 00:21:10 -08:00
|
|
|
|
|
2024-12-31 17:31:01 -08:00
|
|
|
|
⟦joinAll-initialState⟧ᵛ∅ : ⟦ joinForKey initialState result ⟧ᵛ []
|
|
|
|
|
⟦joinAll-initialState⟧ᵛ∅ = subst (λ vs → ⟦ vs ⟧ᵛ []) (sym joinForKey-initialState-⊥ᵛ) ⟦⊥ᵛ⟧ᵛ∅
|
2024-12-31 00:21:10 -08:00
|
|
|
|
|
2024-12-31 17:31:01 -08:00
|
|
|
|
analyze-correct : ∀ {ρ : Env} → [] , rootStmt ⇒ˢ ρ → ⟦ variablesAt finalState result ⟧ᵛ ρ
|
|
|
|
|
analyze-correct {ρ} ∅,s⇒ρ = walkTrace {initialState} {finalState} {[]} {ρ} ⟦joinAll-initialState⟧ᵛ∅ (trace ∅,s⇒ρ)
|
2024-12-31 00:29:39 -08:00
|
|
|
|
|
2024-12-31 17:31:01 -08:00
|
|
|
|
open WithStmtEvaluator using (result; analyze; result≈analyze-result) public
|
|
|
|
|
open WithStmtEvaluator.WithValidInterpretation using (analyze-correct) public
|