TL;DR: What does call/cc do, (semi-)formally speaking?
The long version: I'm vaguely familiar with continuations and call/cc, but I don't have a strong formal understanding. I would like one.
In the SICP video lectures we are presented with the substitution model and a metacircular interpreter. In Shriram Krishnamurti's programming language course we are shown an environment-and-store-passing style. In the compiler course I took in uni I evaluated Java expressions by manipulating a stack.
What's the simplest evaluation model in which call/cc can be expressed, and how do you express call/cc in it?
I found this excellent presentation (in German), which answered my question: https://www.youtube.com/watch?v=iuaM9-PX1ls
To evaluate the lambda calculus with call/CC, you pass around an evaluation context consisting of an environment (as usual) and a call stack. A call to
call/cc
creates a special function-like continuation object which stores the evaluation context. The result of applying the special object to an expressionexpr
is the result of interpretingexpr
in the evaluation context captured in the continuation object.TL;DR: you can implement
call/cc
with an environment-and-call-stack-passing interpreter.If you want to also thread around a mutable store the continuation objects should not capture it. Rather, when invoking a continuation you pass the store as an argument to the interpretation in the restored evaluation context. (The store can be of a linear type.)
Here is one such implementation in Haskell. Here's a sample expression you may want to evaluate:
interpret 0 (Application (Lambda (1, (CallCC (Lambda (2, (Application (Variable 2) (Lambda (3, (Variable 4))))))))) (Lambda (4, (Variable 5))))
.(The numbers are just plain old names, not (e.g.) De Bruijn indices. If you wish to use characters or strings, change
type Name = Integer
totype Name = Char
.)Note especially
interp
applied toCallCC
andInvokeContinuation
as well ascontinue
applied toContinuationArgument
.