Whalesong: Racket to JavaScript compiler
Go to file
2011-05-13 14:03:00 -04:00
dead simulator in separate directory 2011-05-13 14:03:00 -04:00
experiments/gauss getting the evaluator to do something. 2011-02-21 20:28:03 -05:00
simulator simulator in separate directory 2011-05-13 14:03:00 -04:00
tests baffled: I'm getting a different set of parse trees out of earley, and not enough of them. 2011-04-14 16:26:06 -04:00
assemble-expression.rkt working on labels 2011-05-03 17:36:50 -04:00
assemble-helpers.rkt more of module invoke 2011-05-13 00:57:45 -04:00
assemble-open-coded.rkt passing test cases again 2011-05-12 15:43:01 -04:00
assemble-perform-statement.rkt getting into bad compilation times again. Trying to isolate 2011-05-11 14:22:47 -04:00
assemble-structs.rkt in the middle of moving getControlStackLabel as an oparg, so we don't need to write into val register before jumping 2011-04-17 13:11:49 -04:00
assemble.rkt more of module invoke 2011-05-13 00:57:45 -04:00
bootstrapped-primitives.rkt trying to trace why test-compiler is failing on the derivative example. Something broke. 2011-05-12 02:33:31 -04:00
browser-evaluate.rkt moved the type check stuff into the compiler 2011-03-29 21:49:11 -04:00
collect-jump-targets.rkt more of module invoke 2011-05-13 00:57:45 -04:00
compiler-structs.rkt moving some of the definitions of il-structs into separate compiler-structs.rkt module 2011-05-04 18:38:14 -04:00
compiler.rkt more of module invoke 2011-05-13 00:57:45 -04:00
expression-structs.rkt fixing misquoting of label in compiler output for begin0 2011-05-12 16:42:08 -04:00
helpers.rkt traced bug with list intersection 2011-03-23 19:19:49 -04:00
il-structs.rkt more of module invoke 2011-05-13 00:57:45 -04:00
kernel-primitives.rkt trying to run earley using Racket's compiler 2011-05-12 18:48:15 -04:00
language-namespace.rkt trying to clean up; code is getting too large to manage 2011-05-13 13:57:17 -04:00
lexical-env.rkt extending prefixes to also take globalbuckets, although I don't think the simulator or assembler is treating them and modulevariables correctly yet. 2011-05-06 15:53:00 -04:00
lexical-structs.rkt using resolve-module-path-index 2011-05-11 00:46:08 -04:00
NOTES more of module invoke 2011-05-13 00:57:45 -04:00
optimize-il.rkt more of module invoke 2011-05-13 00:57:45 -04:00
package.rkt moving some of the definitions of il-structs into separate compiler-structs.rkt module 2011-05-04 18:38:14 -04:00
parameters.rkt trying to add positional names to lambdas 2011-05-08 19:22:13 -04:00
parse-bytecode-5.1.1.rkt trying to deal with case-lam's revised structure 2011-05-12 14:36:06 -04:00
parse.rkt passing test cases again 2011-05-12 15:43:01 -04:00
racket-expander.rkt kernel domain checks have caller 2011-04-13 14:33:02 -04:00
README test cases for parsing caselam 2011-05-03 16:06:10 -04:00
runtime.compressed.js closure compile can shrink down the size of the runtime to something more managable. 2011-04-03 17:40:01 -04:00
runtime.js trying to complete the bytecode parse 2011-05-11 18:03:46 -04:00
sets.rkt relooper start 2011-02-28 23:01:45 -05:00
test-all.rkt adding test-parse-bytecode-5.1.1 to the test suite. 2011-05-10 00:15:39 -04:00
test-assemble.rkt test case for assembling make-CompiledProcedureClosureReference 2011-05-04 18:27:42 -04:00
test-browser-evaluate.rkt added man-or-boy test 2011-04-22 13:01:34 -04:00
test-compiler-2.rkt trying to run earley using Racket's compiler 2011-05-12 18:48:15 -04:00
test-compiler.rkt simulator in separate directory 2011-05-13 14:03:00 -04:00
test-conform-browser.rkt initial namespacing done 2011-04-02 17:53:43 -04:00
test-conform.rkt moving some of the definitions of il-structs into separate compiler-structs.rkt module 2011-05-04 18:38:14 -04:00
test-earley-browser.rkt trying to trace why earley is failing 2011-04-14 15:20:07 -04:00
test-earley.rkt trying to run earley using Racket's compiler 2011-05-12 18:48:15 -04:00
test-helpers.rkt trying to clean up; code is getting too large to manage 2011-05-13 13:57:17 -04:00
test-package.rkt kludging the call/cc code in 2011-03-12 03:46:15 -05:00
test-parse-bytecode-5.1.1.rkt trying to deal with case-lam's revised structure 2011-05-12 14:36:06 -04:00
test-parse.rkt passing test cases again 2011-05-12 15:43:01 -04:00
test-simulator.rkt fixing test case since simulator structure changed a little 2011-05-11 18:11:56 -04:00
typed-module-path.rkt using resolve-module-path-index 2011-05-11 00:46:08 -04:00
typed-parse.rkt some name cleanup 2011-03-29 20:25:52 -04:00

Compiler from Racket to JavaScript.




Prerequisite: Racket 5.1.1.  The majority of the project is written
Typed Racket, and I highly recommend you use a version of Racket
that's at least 5.1.1; otherwise, compilation may take an unusual
amount of time.



======================================================================

Architecture:

The basic idea is to reuse most of the Racket compiler infrastructure.
The underlying Racket compiler will produce bytecode from Racket
source, and perform macro expansion and module-level optimizations for
us.  We will parse that bytecode using the compiler/zo-parse
collection to get an AST, compile that to an intermediate language,
and finally assemble JavaScript.


                 AST          IL                     JS
 bytecode.rkt ----------> compiler.rkt --------> assembler.rkt ------->
    (todo)


The IL is intended to be translated straightforwardly.  We currently
have an assembler to JavaScript, as well as a simulator
(simulator.rkt).  The simulator allows us to test the compiler in a
controlled environment.



======================================================================

bytecode.rkt

dyoo is currently working on bytecode.rkt.  Not done yet.  This is
intended to reuse the Racket compiler to produce the AST structures
defined in compiler/zo-parse.



For the moment there's a hacky parser in parse.rkt that produces the
AST expression structures that are consumed by the rest of the system.




======================================================================

compiler.rkt translates the AST to the intermediate language.  The
compiler is similar to that of the register compiler in Structure and
Interpretation of Computer Programs:

    http://mitpress.mit.edu/sicp/full-text/book/book-Z-H-35.html#%_sec_5.5

but with some significant modifications.  Since this is a stack
machine, we don't need any of the register-saving infrastructure in
the original compiler.  We also need to support slightly different
linkage structures, since we want to support multiple value contexts.
We're trying to generate code that works effectively on a machine like
the one described in:

    http://plt.eecs.northwestern.edu/racket-machine/




The intermediate language is defined in il-structs.rkt, and a
simulator for the IL in simulator.rkt.  See test-simulator.rkt to see
the simulator in action, and test-compiler.rkt to see how the output
of the compiler can be fed into the simulator.



The assumed machine is a stack machine with the following atomic
registers:

    val: value
    proc: procedure
    argcount: number of arguments

and two stack registers:

    env: environment stack
    control: control stack



======================================================================


The JavaScript assembler is playing a few tricks to make things like
tail calls work:


   * Each basic block is translated to a function taking a MACHINE
     argument.

   * Every GOTO becomes a function call.

   * The head of each basic-blocked function checks to see if we
     should trampoline
     (http://en.wikipedia.org/wiki/Trampoline_(computers))

   * We support a limited form of computed jump by assigning an
     attribute to the function corresponding to a return point.  See
     the code related to the LinkedLabel structure for details.


Otherwise, the assembler is fairly straightforward.  It depends on
library functions defined in runtime.js.  As soon as the compiler
stabilizes, we will be pulling in the runtime library in Moby Scheme
into this project.


The assembled output distinguishes between Primitives and Closures.
Primitives are only allowed to return single values back, and are not
allowed to do any higher-order procedure calls.  Closures, on the
other hand, have full access to the machine, but they are responsible
for calling the continuation and popping off their arguments when
they're finished.




======================================================================

Tests

The test suite in test-all.rkt runs the test suite.  You'll need to
run this on a system with a web browser, as the suite will evaluate
JavaScript and make sure it is producing values.  A bridge module
browser-evaluate.rkt brings up a temporary web server that allows us
to pass values between Racket and the JavaScript evaluator on the
browser.