We can implement a delimited continuation monad in Agda rather easily.
There is, however, no need to, as the Agda "standard library" has an implementation of a delimited continuation monad. What confuses me about this implementation, though, is the addition of an extra parameter to the DCont
type.
DCont : ∀ {i f} {I : Set i} → (I → Set f) → IFun I f
DCont K = DContT K Identity
My question is: why is the extra parameter K
there? And how would I use the DContIMonadDCont
instance? Can I open
it in such a way that I'll get something akin to the below reference implementation in (global) scope?
All my attempts to use it are leading to unsolvable metas.
Reference implementation of delimited continuations not using the Agda "standard library".
DCont : Set → Set → Set → Set
DCont r i a = (a → i) → r
return : ∀ {r a} → a → DCont r r a
return x = λ k → k x
_>>=_ : ∀ {r i j a b} → DCont r i a → (a → DCont i j b) → DCont r j b
c >>= f = λ k → c (λ x → f x k)
join : ∀ {r i j a} → DCont r i (DCont i j a) → DCont r j a
join c = c >>= id
shift : ∀ {r o i j a} → ((a → DCont i i o) → DCont r j j) → DCont r o a
shift f = λ k → f (λ x → λ k′ → k′ (k x)) id
reset : ∀ {r i a} → DCont a i i → DCont r r a
reset a = λ k → k (a id)
Let me answer your second and third questions first. Looking at how
DContT
is defined:We can recover the requested definition by specifying
M = id
andK = id
(M
also has to be a monad, but we have theIdentity
monad).DCont
already fixesM
to beid
, so we are left withK
.Now, we can open the
RawIMonadDCont
module provided we have an instance of the corresponding record. And luckily, we do:Category.Monad.Continuation
has one such record under the nameDContIMonadDCont
.And that's it. Let's make sure the required operations are really there:
And indeed, this typechecks. You can also check if the implementation matches. For example, using
C-c C-n
(normalize) onshift
, we get:Modulo renaming and some implicit parameters, this is exactly implementation of the
shift
in your question.Now the first question. The extra parameter is there to allow additional dependency on the indices. I haven't used delimited continuations in this way, so let me reach for an example somewhere else. Consider this indexed writer:
If we have some sort of indexed monoid, we can write a monad instance for
IWriter
:Now, how is this useful? Imagine you wanted to use the writer to produce a message log or something of the same ilk. With usual boring lists, this is not a problem; but if you wanted to use vectors, you are stuck. How to express that type of the log can change? With the indexed version, you could do something like this:
Well, that was a lot of (ad-hoc) code to make a point. I haven't given it much thought, so I'm fairly sure there's nicer/more principled approach, but it illustrates that such dependency allows your code to be more expressive.
Now, you could apply the same thing to
DCont
, for example:If we apply the definitions, the type reduces to
(ℕ → Vec ℕ 3) → Vec ℕ 2
. Not very convincing example, I know. But perhaps you can some up with something more useful now that you know what this parameter does.