项目作者: tdp2110

项目描述 :
Crafting Interpreters in Rust
高级语言: Rust
项目地址: git://github.com/tdp2110/crafting-interpreters-rs.git
创建时间: 2020-06-08T01:23:44Z
项目社区:https://github.com/tdp2110/crafting-interpreters-rs

开源协议:Boost Software License 1.0

下载


Crafting Interpreters in Rust

Giving https://craftinginterpreters.com/ a try, while learning Rust at the same time.

Just getting started with both :)

:crab: :crab: :crab: This now includes two fairly complete implementations of Bob Nystrom’s Lox language: one as a tree-walk interpreter, and the other as a bytecode interpreter. The treewalk interpreter does not include a garbage collector (the bytecode interpreter does). The bytecode interpreter is written completely in safe Rust (though an unsafe version would likely be much faster). :crab: :crab: :crab:

Examples

Consider fib.lox

  1. fun fib(n) {
  2. if (n < 2) return n;
  3. return fib(n - 1) + fib(n - 2);
  4. }
  5. var before = clock();
  6. print fib(25);
  7. var after = clock();
  8. print after - before;

We can run this in the treewalk interpreter using

  1. cargo run --release --quiet -- fib.lox --treewalk

On my laptop, this prints a timing of 1755 milliseconds. We can run the same thing in the bytecode interpreter using

  1. cargo run --release --quiet -- fib.lox

On the same laptop, this shows a timing of 401 milliseconds.
For comparison, on the same laptop, the tiger compiler
computes the same answer in 0.00s user 0.00s system 4% cpu 0.077 total (not counting compilation :)). A C compiler,
or tigerc using the llvm backend :), computes this in 0.00s user 0.00s system 65% cpu 0.004 total.

Now consider hello_world.lox

  1. print "hello world!";

We can tokenize this with

  1. cargo run --release --quiet -- hello_world.lox --show-tokens

Which gives output

  1. [
  2. Token { ty: Print, lexeme: "print", literal: None, line: 1, col: 4},
  3. Token { ty: String, lexeme: ""hello world!"", literal: Some(Str("hello world!")), line: 1, col: 19},
  4. Token { ty: Semicolon, lexeme: ";", literal: None, line: 1, col: 20},
  5. Token { ty: Eof, lexeme: "", literal: None, line: 1, col: 20},
  6. ]

We can show the AST with

  1. cargo run --release --quiet -- hello_world.lox --show-ast

Which gives

  1. [
  2. Print(
  3. Literal(
  4. String(
  5. "hello world!",
  6. ),
  7. ),
  8. ),
  9. ]

Finally, we can show compiled bytecode with

  1. cargo run --release --quiet -- hello_world.lox --disassemble

Giving

  1. ============ hello_world.lox ============
  2. ------------ constants -----------
  3. 0 "hello world!"
  4. ------------ code -----------------
  5. 0000 OP_CONSTANT "hello world!" (idx=0) line 1
  6. 0001 OP_PRINT line 1
  7. 0002 OP_NIL line 1
  8. 0003 OP_RETURN line 1

Debugger

This project includes a basic (in-progress, possibly never to progress further) debugger.

For example, consider f.lox

  1. fun a() { b(); }
  2. fun b() { c(); }
  3. fun c() {
  4. c("too", "many");
  5. }
  6. a();

We can explore this in the debugger with

  1. $ cargo run --release --quiet -- f.lox --debug
  2. (loxdb) b 4
  3. inserted breakpoint at line 4
  4. (loxdb) g
  5. reached breakpoint at line 4
  6. (loxdb) list
  7. 2 fun b() { c(); }
  8. 3 fun c() {
  9. ==> 4 c("too", "many");
  10. 5 }
  11. 6
  12. 7 a();
  13. ==> 0000 OP_GET_GLOBAL String("c") (idx=0) line 4
  14. 0001 OP_CONSTANT "too" (idx=1) line 4
  15. 0002 OP_CONSTANT "many" (idx=2) line 4
  16. 0003 OP_CALL 2 line 4
  17. (loxdb) bt
  18. [line 7] in script
  19. [line 1] in a()
  20. [line 2] in b()
  21. [line 4] in c()
  22. (loxdb) g
  23. Lox runtime error: Expected 0 arguments but found 2..
  24. Traceback:
  25. [line 7] in script
  26. [line 1] in a()
  27. [line 2] in b()
  28. [line 4] in c()

REPL

A REPL for interactive development is also available, which uses the slower treewalk interpreter. Launch with

  1. cargo run --release --quiet

Here’s an example session:

  1. $ cargo run --release --quiet
  2. ============================================
  3. Welcome to lox! using tree-walk interpreter.
  4. ============================================
  5. >>> var x = 42;
  6. >>> fun f(n) { return n + 1; }
  7. >>> f(x);
  8. 43

Extensions

Using the --Xlists command line switch (eg cargo run --release --quiet -- --Xlists), we can enable lists

  1. ===================================================
  2. Welcome to lox 0.1.0! Using tree-walk interpreter.
  3. Authors: Thomas Peters <thomas.d.peters@gmail.com>
  4. ===================================================
  5. >>> var xs = [1,2,3]
  6. >>> xs
  7. [1, 2, 3]

Lists don’t have much functionality yet, but they have lengths

  1. >>> len(xs)
  2. 3

can be concatenated

  1. >>> var ys = xs + xs
  2. >>> ys
  3. [1, 2, 3, 1, 2, 3]

can be mapped over

  1. >>> fun square(x) { return x * x; }
  2. >>> map(square, xs)
  3. [1, 4, 9]
  4. >>>

can be iterated

  1. >>> fun printFun(elt) { print elt; }
  2. >>> forEach(xs, printFun)
  3. 1
  4. 2
  5. 3

and also have expected indexing operators

  1. >>> xs[0] = -xs[0]
  2. -1
  3. >>> xs
  4. [-1, 2, 3]