------------------------------------------------------------------------
-- The delay monad, defined coinductively, with a sized type parameter
------------------------------------------------------------------------

{-# OPTIONS --sized-types #-}

module Delay-monad.Sized where

open import Equality.Propositional
open import Prelude
open import Prelude.Size

open import Bijection equality-with-J using (_↔_)

-- The delay monad.

mutual

  data Delay {a} (A : Size  Type a) (i : Size) : Type a where
    now   : A i         Delay A i
    later : Delay′ A i  Delay A i

  record Delay′ {a} (A : Size  Type a) (i : Size) : Type a where
    coinductive
    field
      force : {j : Size< i}  Delay A j

open Delay′ public

module _ {a} {A : Size  Type a} where

  mutual

    -- A non-terminating computation.

    never :  {i}  Delay A i
    never = later never′

    never′ :  {i}  Delay′ A i
    force never′ = never

  -- Removes a later constructor, if possible.

  drop-later : Delay A   Delay A 
  drop-later (now   x) = now x
  drop-later (later x) = force x

  -- An unfolding lemma for Delay.

  Delay↔ :  {i}  Delay A i  A i  Delay′ A i
  Delay↔ = record
    { surjection = record
      { logical-equivalence = record
        { to   = λ { (now x)  inj₁ x; (later x)  inj₂ x }
        ; from = [ now , later ]
        }
      ; right-inverse-of = [  _  refl) ,  _  refl) ]
      }
    ; left-inverse-of = λ { (now _)  refl; (later _)  refl }
    }