Skip to content

Latest commit

 

History

History
137 lines (107 loc) · 3.94 KB

README.md

File metadata and controls

137 lines (107 loc) · 3.94 KB

progress-banner

Lox Interpreter in Rust

This is a Rust implementation of the Lox language interpreter for the "Build Your Own Interpreter" Challenge on CodeCrafters.

Features

This interpreter supports a rich set of language features:

  • Expressions: Arithmetic operations, comparisons, logical operations
  • Variables: Declaration and assignment
  • Control Flow: If statements, while loops, for loops
  • Functions: Declaration, calling, and returning values
  • Closures: Functions can capture variables from their enclosing scope
  • Classes: Object-oriented programming with class declarations
  • Inheritance: Classes can inherit from another class using the < operator
  • Methods: Classes can have methods including special init constructor
  • Native Functions: Built-in functions like clock()

Getting Started

  1. Ensure you have cargo (1.77) installed locally
  2. Run ./your_program.sh run <filename.lox> to interpret a Lox script

Usage Examples

The interpreter supports several modes:

# Run a full Lox program
./your_program.sh run test/constructor_calls/3.lox

# Just tokenize the file to see lexical analysis
./your_program.sh tokenize test/constructor_calls/3.lox

# Just evaluate an expression
./your_program.sh evaluate test/string_concat.lox

# Just parse the expression and output the AST
./your_program.sh parse test/string_concat.lox

Lox Language Examples

Variables and Control Flow

var a = 1;
var b = 2;
if (a < b) {
  print "a is less than b";
} else {
  print "a is greater than or equal to b";
}

Functions

fun fibonacci(n) {
  if (n <= 1) return n;
  return fibonacci(n - 2) + fibonacci(n - 1);
}

print fibonacci(10);

Classes and Inheritance

class Animal {
  init(name) {
    this.name = name;
  }
  
  speak() {
    print this.name + " makes a noise";
  }
}

class Dog < Animal {
  speak() {
    print this.name + " barks";
  }
}

var dog = Dog("Rex");
dog.speak();  // Prints: Rex barks

Higher-Order Functions

fun makeCounter() {
  var i = 0;
  fun count() {
    i = i + 1;
    return i;
  }
  return count;
}

var counter = makeCounter();
print counter(); // 1
print counter(); // 2

Architecture

The interpreter is structured in these key components:

  • Scanner (scanner.rs): Tokenizes the source code into lexical tokens
  • Parser (parser.rs): Builds an abstract syntax tree from tokens
  • Resolver (resolver.rs): Performs variable resolution and semantic analysis
  • Interpreter (interpreter.rs): Executes the resolved syntax tree
  • Environment (environment.rs): Manages variable scopes and lookups
  • Callable (callable.rs): Implements functions, classes, and method calls
  • Grammar (grammar.rs): Defines the AST node types and tokens

Implementation Details

The interpreter follows a classic compilation pipeline:

  1. Scanning: Converts source code into tokens
  2. Parsing: Transforms tokens into an Abstract Syntax Tree
  3. Resolution: Resolves variable references and performs static analysis
  4. Interpretation: Executes the program

Error Handling

The interpreter implements comprehensive error handling:

  • Lexical errors: Detected during scanning
  • Syntax errors: Reported during parsing
  • Semantic errors: Caught during resolution (e.g., undefined variables)
  • Runtime errors: Thrown during execution (e.g., type errors, division by zero)

Each error provides meaningful context with line numbers and error messages.

Performance Considerations

  • The interpreter uses Rust's ownership model for memory safety
  • Garbage collection is handled through Rust's reference counting (Rc) and interior mutability (RefCell)
  • Function and method lookups are optimized with hashmaps