677
|
1 |
% !TEX program = xelatex
|
539
|
2 |
\documentclass{article}
|
|
3 |
\usepackage{../style}
|
|
4 |
\usepackage{../langs}
|
|
5 |
\usepackage{../graphics}
|
|
6 |
\usepackage{../grammar}
|
677
|
7 |
%%\usepackage{multicol}
|
539
|
8 |
|
677
|
9 |
%%\newcommand{\dn}{\stackrel{\mbox{\scriptsize def}}{=}}
|
539
|
10 |
|
|
11 |
\begin{document}
|
677
|
12 |
\fnote{\copyright{} Christian Urban, King's College London, 2019}
|
539
|
13 |
|
|
14 |
|
677
|
15 |
\section*{Handout 9 (LLVM, SSA and CPS)}
|
539
|
16 |
|
700
|
17 |
Reflecting on our two tiny compilers targetting the JVM, the code
|
|
18 |
generation part was actually not so hard, no? Pretty much just some
|
|
19 |
post-traversal of the abstract syntax tree, yes? One of the reasons for
|
|
20 |
this ease is that the JVM is a stack-based virtual machine and it is
|
|
21 |
therefore not hard to translate deeply-nested arithmetic expressions
|
|
22 |
into a sequence of instructions manipulating the stack. The problem is
|
|
23 |
that ``real'' CPUs, although supporting stack operations, are not really
|
|
24 |
designed to be \emph{stack machines}. The design of CPUs is more like,
|
|
25 |
here is a chunk of memory---compiler, or better compiler writers, do
|
|
26 |
something with it. Consequently, modern compilers need to go the extra
|
|
27 |
mile in order to generate code that is much easier and faster to process
|
|
28 |
by CPUs. To make this all tractable for this module, we target the LLVM
|
680
|
29 |
Intermediate Language. In this way we can take advantage of the tools
|
|
30 |
coming with LLVM. For example we do not have to worry about things like
|
|
31 |
register allocations.\bigskip
|
539
|
32 |
|
678
|
33 |
\noindent LLVM\footnote{\url{http://llvm.org}} is a beautiful example
|
700
|
34 |
that projects from Academia can make a difference in the World. LLVM
|
678
|
35 |
started in 2000 as a project by two researchers at the University of
|
|
36 |
Illinois at Urbana-Champaign. At the time the behemoth of compilers was
|
680
|
37 |
gcc with its myriad of front-ends for other languages (C++, Fortran,
|
678
|
38 |
Ada, Go, Objective-C, Pascal etc). The problem was that gcc morphed over
|
|
39 |
time into a monolithic gigantic piece of m\ldots ehm software, which you
|
|
40 |
could not mess about in an afternoon. In contrast, LLVM is designed to
|
700
|
41 |
be a modular suite of tools with which you can play around easily and
|
678
|
42 |
try out something new. LLVM became a big player once Apple hired one of
|
|
43 |
the original developers (I cannot remember the reason why Apple did not
|
|
44 |
want to use gcc, but maybe they were also just disgusted by its big
|
|
45 |
monolithic codebase). Anyway, LLVM is now the big player and gcc is more
|
|
46 |
or less legacy. This does not mean that programming languages like C and
|
|
47 |
C++ are dying out any time soon---they are nicely supported by LLVM.
|
539
|
48 |
|
700
|
49 |
We will target the LLVM Intermediate Language, or LLVM Intermediate
|
|
50 |
Representation (short LLVM-IR). The LLVM-IR looks very similar to the
|
|
51 |
assembly language of Jasmin and Krakatau. It will also allow us to
|
|
52 |
benefit from the modular structure of the LLVM compiler and let for
|
|
53 |
example the compiler generate code for different CPUs, like X86 or ARM.
|
|
54 |
That means we can be agnostic about where our code actually runs. We can
|
|
55 |
also be ignorant about optimising code and allocating memory
|
|
56 |
efficiently.
|
539
|
57 |
|
700
|
58 |
However, what we have to do for LLVM is to generate code in \emph{Static
|
|
59 |
Single-Assignment} format (short SSA), because that is what the LLVM-IR
|
|
60 |
expects from us. A reason why LLVM uses the SSA format, rather than
|
|
61 |
JVM-like stack instructions, is that stack instructions are difficult to
|
|
62 |
optimise---you cannot just re-arrange instructions without messing about
|
|
63 |
with what is calculated on the stack. Also it is hard to find out if all
|
|
64 |
the calculations on the stack are actually necessary and not by chance
|
|
65 |
dead code. The JVM has for all these obstacles sophisticated machinery
|
|
66 |
to make such ``high-level'' code still run fast, but let's say that for
|
|
67 |
the sake of argument we do not want to rely on it. We want to generate
|
|
68 |
fast code ourselves. This means we have to work around the intricacies
|
|
69 |
of what instructions CPUs can actually process fast. This is what the
|
|
70 |
SSA format is designed for.
|
|
71 |
|
|
72 |
|
|
73 |
The main idea behind the SSA format is to use very simple variable
|
678
|
74 |
assignments where every variable is assigned only once. The assignments
|
|
75 |
also need to be primitive in the sense that they can be just simple
|
|
76 |
operations like addition, multiplication, jumps, comparisons and so on.
|
680
|
77 |
Say, we have an expression $((1 + a) + (3 + (b * 5)))$, then the
|
700
|
78 |
corresponding SSA format is
|
680
|
79 |
|
|
80 |
\begin{lstlisting}[language=LLVMIR,numbers=left]
|
|
81 |
let tmp0 = add 1 a in
|
|
82 |
let tmp1 = mul b 5 in
|
|
83 |
let tmp2 = add 3 tmp1 in
|
700
|
84 |
let tmp3 = add tmp0 tmp2 in tmp3
|
677
|
85 |
\end{lstlisting}
|
539
|
86 |
|
678
|
87 |
\noindent where every variable is used only once (we could not write
|
680
|
88 |
\texttt{tmp1 = add 3 tmp1} in Line 3 for example). There are
|
678
|
89 |
sophisticated algorithms for imperative languages, like C, that
|
|
90 |
efficiently transform a high-level program into SSA format. But we can
|
|
91 |
ignore them here. We want to compile a functional language and there
|
|
92 |
things get much more interesting than just sophisticated. We will need
|
|
93 |
to have a look at CPS translations, where the CPS stands for
|
|
94 |
Continuation-Passing-Style---basically black programming art or
|
|
95 |
abracadabra programming. So sit tight.
|
539
|
96 |
|
678
|
97 |
\subsection*{LLVM-IR}
|
539
|
98 |
|
700
|
99 |
Before we start, let's first have a look at the \emph{LLVM Intermediate
|
|
100 |
Representation} in more detail. The LLVM-IR is in between the frontends
|
|
101 |
and backends of the LLVM framework. It allows compilation of multiple
|
|
102 |
source languages to multiple targets. It is also the place where most of
|
|
103 |
the target independent optimisations are performed.
|
|
104 |
|
|
105 |
What is good about our toy Fun language is that it basically only
|
|
106 |
contains expressions (be they arithmetic expressions, boolean
|
|
107 |
expressions or if-expressions). The exception are function definitions.
|
|
108 |
Luckily, for them we can use the mechanism of defining functions in the
|
|
109 |
LLVM-IR (this is similar to using JVM methods for functions in our
|
|
110 |
earlier compiler). For example the simple Fun program
|
539
|
111 |
|
|
112 |
|
677
|
113 |
\begin{lstlisting}[language=Scala,numbers=none]
|
678
|
114 |
def sqr(x) = x * x
|
677
|
115 |
\end{lstlisting}
|
539
|
116 |
|
677
|
117 |
\noindent
|
700
|
118 |
can be compiled to the following LLVM-IR function:
|
539
|
119 |
|
677
|
120 |
\begin{lstlisting}[language=LLVM]
|
678
|
121 |
define i32 @sqr(i32 %x) {
|
|
122 |
%tmp = mul i32 %x, %x
|
677
|
123 |
ret i32 %tmp
|
|
124 |
}
|
|
125 |
\end{lstlisting}
|
539
|
126 |
|
700
|
127 |
\noindent First notice that all variable names, in this case \texttt{x}
|
|
128 |
and \texttt{tmp}, are prefixed with \texttt{\%} in the LLVM-IR.
|
|
129 |
Temporary variables can be named with an identifier, such as
|
|
130 |
\texttt{tmp}, or numbers. Function names, since they are ``global'',
|
|
131 |
need to be prefixed with @-symbol. Also, the LLVM-IR is a fully typed
|
|
132 |
language. The \texttt{i32} type stands for 32-bit integers. There are
|
|
133 |
also types for 64-bit integers (\texttt{i64}), chars (\texttt{i8}),
|
|
134 |
floats, arrays and even pointer types. In the code above, \texttt{sqr}
|
|
135 |
takes an argument of type \texttt{i32} and produces a result of type
|
|
136 |
\texttt{i32} (the result type is in front of the function name, like in
|
|
137 |
C). Each arithmetic operation, for example addition and multiplication,
|
|
138 |
are also prefixed with the type they operate on. Obviously these types
|
|
139 |
need to match up\ldots{} but since we have in our programs only
|
|
140 |
integers, \texttt{i32} everywhere will do. We do not have to generate
|
701
|
141 |
any other types, but obviously this is a limitation in our Fun language.
|
539
|
142 |
|
700
|
143 |
There are a few interesting instructions in the LLVM-IR which are quite
|
701
|
144 |
different than what we have seen in the JVM. Can you remember the
|
|
145 |
kerfuffle we had to go through with boolean expressions and negating the
|
|
146 |
condition? In the LLVM-IR, branching if-conditions is implemented
|
|
147 |
differently: there is a separate \texttt{br}-instruction as follows:
|
700
|
148 |
|
|
149 |
\begin{lstlisting}[language=LLVM]
|
|
150 |
br i1 %var, label %if_br, label %else_br
|
|
151 |
\end{lstlisting}
|
|
152 |
|
|
153 |
\noindent
|
|
154 |
The type \texttt{i1} stands for booleans. If the variable is true, then
|
|
155 |
this instruction jumps to the if-branch, which needs an explicit label;
|
|
156 |
otherwise to the else-branch, again with its own label. This allows us
|
701
|
157 |
to keep the meaning of the boolean expression as is when compiling if's.
|
|
158 |
A value of type boolean is generated in the LLVM-IR by the
|
|
159 |
\texttt{icmp}-instruction. This instruction is for integers (hence the
|
|
160 |
\texttt{i}) and takes the comparison operation as argument. For example
|
700
|
161 |
|
|
162 |
\begin{lstlisting}[language=LLVM]
|
|
163 |
icmp eq i32 %x, %y ; for equal
|
|
164 |
icmp sle i32 %x, %y ; signed less or equal
|
|
165 |
icmp slt i32 %x, %y ; signed less than
|
|
166 |
icmp ult i32 %x, %y ; unsigned less than
|
|
167 |
\end{lstlisting}
|
|
168 |
|
|
169 |
\noindent
|
|
170 |
In some operations, the LLVM-IR distinguishes between signed and
|
|
171 |
unsigned representations of integers.
|
|
172 |
|
701
|
173 |
It is also easy to call another function in LLVM-IR: as can be
|
|
174 |
seen from Figure~\ref{lli} we can just call a function with the
|
|
175 |
instruction \texttt{call} and can also assign the result to
|
|
176 |
a variable. The syntax is as follows
|
|
177 |
|
|
178 |
\begin{lstlisting}[language=LLVM]
|
|
179 |
%var = call i32 @foo(...args...)
|
|
180 |
\end{lstlisting}
|
|
181 |
|
|
182 |
\noindent
|
|
183 |
where the arguments can only be simple variables, not compound
|
|
184 |
expressions.
|
|
185 |
|
679
|
186 |
Conveniently, you can use the program \texttt{lli}, which comes with
|
|
187 |
LLVM, to interpret programs written in the LLVM-IR. So you can easily
|
|
188 |
check whether the code you produced actually works. To get a running
|
|
189 |
program that does something interesting you need to add some boilerplate
|
701
|
190 |
about printing out numbers and a main-function that is the entry point
|
700
|
191 |
for the program (see Figure~\ref{lli} for a complete listing). Again
|
|
192 |
this is very similar to the boilerplate we needed to add in our JVM
|
|
193 |
compiler.
|
|
194 |
|
|
195 |
You can generate a binary for the program in Figure~\ref{lli} by using
|
|
196 |
the \texttt{llc}-compiler and then \texttt{gcc}, whereby \texttt{llc} generates
|
|
197 |
an object file and \texttt{gcc} (that is clang) generates the
|
|
198 |
executable binary:
|
678
|
199 |
|
679
|
200 |
\begin{lstlisting}[language=bash,numbers=none]
|
|
201 |
llc -filetype=obj sqr.ll
|
|
202 |
gcc sqr.o -o a.out
|
|
203 |
./a.out
|
680
|
204 |
> 25
|
679
|
205 |
\end{lstlisting}
|
|
206 |
|
|
207 |
\begin{figure}[t]\small
|
|
208 |
\lstinputlisting[language=LLVM,numbers=left]{../progs/sqr.ll}
|
700
|
209 |
\caption{An LLVM-IR program for calculating the square function. It
|
|
210 |
calls this function in \texttt{@main} with the argument \texttt{5}. The
|
|
211 |
code for the \texttt{sqr} function is in Lines 13 -- 16. The main
|
|
212 |
function calls \texttt{sqr} and then prints out the result. The other
|
679
|
213 |
code is boilerplate for printing out integers.\label{lli}}
|
678
|
214 |
\end{figure}
|
|
215 |
|
679
|
216 |
|
|
217 |
|
|
218 |
\subsection*{Our Own Intermediate Language}
|
|
219 |
|
|
220 |
Remember compilers have to solve the problem of bridging the gap between
|
680
|
221 |
``high-level'' programs and ``low-level'' hardware. If the gap is too
|
|
222 |
wide for one step, then a good strategy is to lay a stepping stone
|
|
223 |
somewhere in between. The LLVM-IR itself is such a stepping stone to
|
|
224 |
make the task of generating and optimising code easier. Like a real
|
|
225 |
compiler we will use our own stepping stone which I call the
|
700
|
226 |
\emph{K-language}. For what follows recall the various kinds of
|
|
227 |
expressions in the Fun language. For convenience the Scala code of the
|
|
228 |
corresponding abstract syntax trees is shown on top of
|
|
229 |
Figure~\ref{absfun}. Below is the code for the abstract syntax trees in
|
701
|
230 |
the K-language. In K, here are two kinds of syntactic entities, namely
|
700
|
231 |
\emph{K-values} and \emph{K-expressions}. The central constructor of the
|
701
|
232 |
K-language is \texttt{KLet}. For this recall in SSA that arithmetic
|
|
233 |
expressions such as $((1 + a) + (3 + (b * 5)))$ need to be broken up
|
|
234 |
into smaller ``atomic'' steps, like so
|
680
|
235 |
|
|
236 |
\begin{lstlisting}[language=LLVMIR,numbers=none]
|
|
237 |
let tmp0 = add 1 a in
|
|
238 |
let tmp1 = mul b 5 in
|
|
239 |
let tmp2 = add 3 tmp1 in
|
|
240 |
let tmp3 = add tmp0 tmp2 in
|
|
241 |
tmp3
|
|
242 |
\end{lstlisting}
|
|
243 |
|
|
244 |
\noindent
|
700
|
245 |
Here \texttt{tmp3} will contain the result of what the whole expression
|
|
246 |
stands for. In each individual step we can only perform an ``atomic''
|
|
247 |
operation, like addition or multiplication of a number and a variable.
|
|
248 |
We are not allowed to have for example an if-condition on the right-hand
|
|
249 |
side of an equals. Such constraints are enforced upon us because of how
|
|
250 |
the SSA format works in the LLVM-IR. By having in \texttt{KLet} taking
|
|
251 |
first a string (standing for an intermediate result) and second a value,
|
|
252 |
we can fulfil this constraint ``by construction''---there is no way we
|
|
253 |
could write anything else than a value.
|
|
254 |
|
|
255 |
To sum up, K-values are the atomic operations that can be on the
|
|
256 |
right-hand side of equal-signs. The K-language is restricted such that
|
|
257 |
it is easy to generate the SSA format for the LLVM-IR.
|
680
|
258 |
|
679
|
259 |
|
|
260 |
|
|
261 |
\begin{figure}[p]\small
|
678
|
262 |
\begin{lstlisting}[language=Scala,numbers=none]
|
701
|
263 |
// Fun language (expressions)
|
679
|
264 |
abstract class Exp
|
|
265 |
abstract class BExp
|
678
|
266 |
|
|
267 |
case class Call(name: String, args: List[Exp]) extends Exp
|
|
268 |
case class If(a: BExp, e1: Exp, e2: Exp) extends Exp
|
|
269 |
case class Write(e: Exp) extends Exp
|
|
270 |
case class Var(s: String) extends Exp
|
|
271 |
case class Num(i: Int) extends Exp
|
|
272 |
case class Aop(o: String, a1: Exp, a2: Exp) extends Exp
|
|
273 |
case class Sequence(e1: Exp, e2: Exp) extends Exp
|
679
|
274 |
case class Bop(o: String, a1: Exp, a2: Exp) extends BExp
|
|
275 |
|
|
276 |
|
|
277 |
|
|
278 |
// K-language (K-expressions, K-values)
|
|
279 |
abstract class KExp
|
|
280 |
abstract class KVal
|
|
281 |
|
|
282 |
case class KVar(s: String) extends KVal
|
|
283 |
case class KNum(i: Int) extends KVal
|
|
284 |
case class Kop(o: String, v1: KVal, v2: KVal) extends KVal
|
|
285 |
case class KCall(o: String, vrs: List[KVal]) extends KVal
|
|
286 |
case class KWrite(v: KVal) extends KVal
|
|
287 |
|
|
288 |
case class KIf(x1: String, e1: KExp, e2: KExp) extends KExp
|
680
|
289 |
case class KLet(x: String, v: KVal, e: KExp) extends KExp
|
679
|
290 |
case class KReturn(v: KVal) extends KExp
|
678
|
291 |
\end{lstlisting}
|
|
292 |
\caption{Abstract syntax trees for the Fun language.\label{absfun}}
|
|
293 |
\end{figure}
|
679
|
294 |
|
678
|
295 |
|
|
296 |
|
|
297 |
\subsection*{CPS-Translations}
|
|
298 |
|
704
|
299 |
CPS stands for Continuation-Passing-Style. It is a kind of programming
|
|
300 |
technique often used in advanced functional programming. Before we delve
|
|
301 |
into the CPS-translation for our Fun language, let us look at
|
|
302 |
CPS-versions of some well-known functions. Consider
|
|
303 |
|
|
304 |
\begin{lstlisting}[language=Scala, numbers=none]
|
|
305 |
def fact(n: Int) : Int =
|
|
306 |
if (n == 0) 1 else n * fact(n - 1)
|
|
307 |
\end{lstlisting}
|
|
308 |
|
|
309 |
\noindent
|
|
310 |
This is clearly the usual factorial function. But now consider the
|
|
311 |
following version of the factorial function:
|
|
312 |
|
|
313 |
\begin{lstlisting}[language=Scala, numbers=none]
|
|
314 |
def factC(n: Int, ret: Int => Int) : Int =
|
|
315 |
if (n == 0) ret(1)
|
|
316 |
else factC(n - 1, x => ret(n * x))
|
|
317 |
|
|
318 |
factC(3, identity)
|
|
319 |
\end{lstlisting}
|
|
320 |
|
|
321 |
\noindent
|
|
322 |
This function is called with the number, in this case 3, and the
|
|
323 |
identity-function (which returns just its input). The recursive
|
|
324 |
calls are:
|
|
325 |
|
|
326 |
\begin{lstlisting}[language=Scala, numbers=none]
|
|
327 |
factC(2, x => identity(3 * x))
|
|
328 |
factC(1, x => identity(3 * (2 * x)))
|
|
329 |
factC(0, x => identity(3 * (2 * (1 * x))))
|
|
330 |
\end{lstlisting}
|
|
331 |
|
|
332 |
\noindent
|
|
333 |
Having reached 0, we get out of the recursion and apply 1 to the
|
|
334 |
continuation (see if-branch above). This gives
|
|
335 |
|
|
336 |
\begin{lstlisting}[language=Scala, numbers=none]
|
|
337 |
identity(3 * (2 * (1 * 1)))
|
|
338 |
= 3 * (2 * (1 * 1))
|
|
339 |
= 6
|
|
340 |
\end{lstlisting}
|
|
341 |
|
|
342 |
\noindent
|
|
343 |
which is the expected result. If this looks somewhat familiar, then this
|
|
344 |
is not a 1000 miles off, because functions with continuations can be
|
|
345 |
seen as a kind of generalisation of tail-recursive functions. Anyway
|
|
346 |
notice how the continuations is ``stacked up'' during the recursion and
|
|
347 |
then ``unrolled'' when we apply 1 to the continuation. Interestingly, we
|
|
348 |
can do something similar to the Fibonacci function where in the traditional
|
|
349 |
version we have two recursive calls. Consider the following function
|
|
350 |
|
|
351 |
\begin{lstlisting}[language=Scala, numbers=none]
|
|
352 |
def fibC(n: Int, ret: Int => Int) : Int =
|
|
353 |
if (n == 0 || n == 1) ret(1)
|
|
354 |
else fibC(n - 1,
|
|
355 |
r1 => fibC(n - 2,
|
|
356 |
r2 => ret(r1 + r2)))
|
|
357 |
\end{lstlisting}
|
|
358 |
|
|
359 |
\noindent
|
|
360 |
Here the continuation is a nested function essentially wrapping up
|
|
361 |
the second recursive call. Let us check how the recursion unfolds
|
|
362 |
when called with 3 and the identity function:
|
|
363 |
|
|
364 |
\begin{lstlisting}[language=Scala, numbers=none]
|
|
365 |
fibC(3, id)
|
|
366 |
fibC(2, r1 => fibC(1, r2 => id(r1 + r2)))
|
|
367 |
fibC(1, r1 =>
|
|
368 |
fibC(0, r2 => fibC(1, r2a => id((r1 + r2) + r2a))))
|
|
369 |
fibC(0, r2 => fibC(1, r2a => id((1 + r2) + r2a)))
|
|
370 |
fibC(1, r2a => id((1 + 1) + r2a))
|
|
371 |
id((1 + 1) + 1)
|
|
372 |
(1 + 1) + 1
|
|
373 |
3
|
|
374 |
\end{lstlisting}
|
|
375 |
|
|
376 |
Let us now come back to the CPS-translations for the Fun language. The
|
|
377 |
main difficulty of generating instructions in SSA format is that large
|
|
378 |
compound expressions need to be broken up into smaller pieces and
|
700
|
379 |
intermediate results need to be chained into later instructions. To do
|
|
380 |
this conveniently, CPS-translations have been developed. They use
|
|
381 |
functions (``continuations'') to represent what is coming next in a
|
701
|
382 |
sequence of instructions. Continuations are functions of type
|
704
|
383 |
\code{KVal} to \code{KExp}. They can be seen as a sequence of
|
|
384 |
\code{KLet}s where there is a ``hole'' that needs to be filled. Consider
|
|
385 |
for example
|
678
|
386 |
|
701
|
387 |
\begin{lstlisting}[language=LLVMIR,numbers=left,escapeinside={(*@}{@*)}]
|
|
388 |
let tmp0 = add 1 a in
|
|
389 |
let tmp1 = mul (*@$\Box$@*) 5 in
|
|
390 |
let tmp2 = add 3 tmp1 in
|
|
391 |
let tmp3 = add tmp0 tmp2 in
|
|
392 |
tmp3
|
|
393 |
\end{lstlisting}
|
|
394 |
|
|
395 |
\noindent
|
|
396 |
where in the second line is a $\Box$ which still expects a \code{KVal}
|
|
397 |
to be filled in before it becomes a ``proper'' \code{KExp}. When we
|
|
398 |
apply and argument to the continuation (remember they are functions)
|
|
399 |
we essentially fill something into the corresponding hole. The code
|
|
400 |
of the CPS-translation is
|
679
|
401 |
|
701
|
402 |
\begin{lstlisting}[language=Scala,numbers=none]
|
|
403 |
def CPS(e: Exp)(k: KVal => KExp) : KExp =
|
|
404 |
e match { ... }
|
|
405 |
\end{lstlisting}
|
679
|
406 |
|
701
|
407 |
\noindent
|
|
408 |
where \code{k} is the continuation and \code{e} is the expression
|
|
409 |
to be compiled. In case we have numbers or variables, we can just
|
|
410 |
apply the continuation like
|
679
|
411 |
|
701
|
412 |
\begin{center}
|
|
413 |
\code{k(KNum(n))} \qquad \code{k(KVar(x))}
|
|
414 |
\end{center}
|
679
|
415 |
|
701
|
416 |
\noindent This would just fill in the $\Box$ in a \code{KLet}-expression.
|
|
417 |
More interesting is the case for an arithmetic expression.
|
|
418 |
|
|
419 |
\begin{lstlisting}[language=Scala,numbers=none]
|
|
420 |
case Aop(o, e1, e2) => {
|
|
421 |
val z = Fresh("tmp")
|
|
422 |
CPS(e1)(y1 =>
|
|
423 |
CPS(e2)(y2 => KLet(z, Kop(o, y1, y2), k(KVar(z)))))
|
|
424 |
}
|
|
425 |
\end{lstlisting}
|
|
426 |
|
|
427 |
\noindent
|
539
|
428 |
\end{document}
|
|
429 |
|
|
430 |
|
|
431 |
%%% Local Variables:
|
|
432 |
%%% mode: latex
|
|
433 |
%%% TeX-master: t
|
|
434 |
%%% End:
|