Skip to content

Auxiliary Functions

The \(\lambda\)-notation in the Agda definitions of auxiliary functions for Scm corresponds closely to that in its published denotational semantics (Mosses2025CSE).

{-# OPTIONS --rewriting --confluence-check --lossy-unification #-}

module Examples.Scm.Auxiliary-Functions where

open import Examples.Scm.Abstract-Syntax
open import Examples.Scm.Domain-Equations
import Notation
open Notation.Domains using (⟪_⟫; )
open Notation.Functions using (_→ᶜ_; fix)
open Notation.Flat using ()
open Notation.Flat.Booleans using (_⟶_,_; Eq; _==⊥_; true; false)
open Notation.Sums using (_in⊥_; _∈⊥_; _|⊥_)
open Notation.Products using (_,_; _↓₁; _↓₂)
open Notation.Products.Sequences using (⟨_⟩; #; _↓_; _†_)
open Notation.Updates using (_[_/_]⊥)

postulate instance
  eqL   : Eq Loc
  eqM   : Eq Misc
  eqR   : Eq Int

postulate
  _<ᴿ_  :  𝐑 →ᶜ 𝐑 →ᶜ 𝐓 
  _+ᴿ_  :  𝐑 →ᶜ 𝐑 →ᶜ 𝐑 
  _∧ᵀ_  :  𝐓 →ᶜ 𝐓 →ᶜ 𝐓 

postulate instance
  eqIde : Eq Ide

postulate unknown : Loc

postulate initial-env :  𝐔 

assign :  𝐋 →ᶜ 𝐄 →ᶜ 𝐂 →ᶜ 𝐂       -- assign α ϵ stores ϵ at location α
assign α ϵ θ σ = θ (σ [ ϵ / α ]⊥)

hold :  𝐋 →ᶜ (𝐄 →ᶜ 𝐂) →ᶜ 𝐂       -- hold α gives the value stored at α
hold α κ σ = κ (σ α) σ

In the continuation-passing style used to define auxilary functions for Scm, giving explicit continuity proofs would be particulary tedious. For example, the function hold is simply a combination of \(\lambda\)-abstraction and application, which is wellknown to ensure continuity.

postulate new :  (𝐋 →ᶜ 𝐂) →ᶜ 𝐂   -- new gives an unallocated location

alloc :  𝐄 →ᶜ (𝐋 →ᶜ 𝐂) →ᶜ 𝐂      -- alloc ϵ allocates a location for ϵ
alloc ϵ κ = new  α  assign α ϵ (κ α))

postulate initial-store :  𝐒     -- may have initialised locations

postulate finished :  𝐒 →ᶜ 𝐀     -- obtain answer from the final store

Conventional denotational definitions usually leave the injection function from sets into flat domains implicit, in contrast to the embedding of the definition of truish:

truish :  𝐄 →ᶜ 𝐓                 -- truish ε is true for all ε except false
truish ϵ =  (ϵ ∈⊥ 𝐓)  (((ϵ |⊥ 𝐓) ==⊥  false)   false ,  true) ,
             true

The remaining auxiliary function definitions shown here involve the operations for (finite) sequences ϵ⋆ declared in the module Notation.Products.Sequences.

cons :  𝐅                        -- cons ⟨ ϵ₁ , ϵ₂ ⟩ allocates and initialises a pair
cons ϵ⋆ κ =  (# ϵ⋆ ==⊥  2) 
                alloc (ϵ⋆  1)  α₁ 
                  alloc (ϵ⋆  2)  α₂  κ ((α₁ , α₂) in⊥ 𝐄))) ,
              

In (Mosses2025CSE) the auxiliary function \(\textit{list}\) is defined by recursion on ϵ⋆. Agda accepts recursive definitions only when it can mechanically prove that the recursion terminates, which is not the case for arguments in postulated types. The following definition uses the postulated operation fix to avoid recursion.

list :  𝐅                        -- list ϵ⋆ allocates and initialises a list
list =   fix λ (list′ :  𝐅 )  λ ϵ⋆ κ 
           (# ϵ⋆ ==⊥  0)  κ ( null in⊥ 𝐄) ,
           list′ (ϵ⋆  1)  ϵ  cons  (ϵ⋆  1) , ϵ  κ)

car :  𝐅                         -- car ⟨ ϵ ⟩ gives the head of the list ϵ
car ϵ⋆ κ = (# ϵ⋆ ==⊥  1)  hold (((ϵ⋆  1) |⊥ 𝐏) ↓₁) κ , 

cdr :  𝐅                         -- cdr ⟨ ϵ ⟩ gives the tail of the list ϵ
cdr ϵ⋆ κ = (# ϵ⋆ ==⊥  1)  hold (((ϵ⋆  1) |⊥ 𝐏) ↓₂) κ , 

setcar :  𝐅                      -- setcar ⟨ ϵ₁ , ϵ₂ ⟩ stores ϵ₂ in the head of list ϵ₁
setcar ϵ⋆ κ =
  (# ϵ⋆ ==⊥  2) 
    assign (((ϵ⋆  1) |⊥ 𝐏) ↓₁) (ϵ⋆  2) (κ ( unspecified in⊥ 𝐄)) ,
  

setcdr :  𝐅                      -- setcdr ⟨ ϵ₁ , ϵ₂ ⟩ stores ϵ₂ in the tail of list ϵ₁
setcdr ϵ⋆ κ =
  (# ϵ⋆ ==⊥  2) 
    assign (((ϵ⋆  1) |⊥ 𝐏) ↓₂) (ϵ⋆  2) (κ ( unspecified in⊥ 𝐄)) ,