aboutsummaryrefslogtreecommitdiffstats
path: root/src/com/craftinginterpreters/lox
diff options
context:
space:
mode:
authorGravatar Tom Willemse2021-03-17 22:52:35 -0700
committerGravatar Tom Willemse2021-03-17 22:52:35 -0700
commit580a5acb580a858c8ffa705f5f64f095cb0b4b37 (patch)
treee0ac4b327ad2993b21304b1e5ba84819cad64034 /src/com/craftinginterpreters/lox
parentb04b6bcf2f31526f6cfa593c342165a65cb9e7ac (diff)
downloadcrafting-interpreters-580a5acb580a858c8ffa705f5f64f095cb0b4b37.tar.gz
crafting-interpreters-580a5acb580a858c8ffa705f5f64f095cb0b4b37.zip
Chapter 12, Sections 1, 2, and 3
Diffstat (limited to 'src/com/craftinginterpreters/lox')
-rw-r--r--src/com/craftinginterpreters/lox/CMakeLists.txt2
-rw-r--r--src/com/craftinginterpreters/lox/Interpreter.java31
-rw-r--r--src/com/craftinginterpreters/lox/LoxClass.java27
-rw-r--r--src/com/craftinginterpreters/lox/LoxInstance.java30
-rw-r--r--src/com/craftinginterpreters/lox/Parser.java22
-rw-r--r--src/com/craftinginterpreters/lox/Resolver.java20
6 files changed, 132 insertions, 0 deletions
diff --git a/src/com/craftinginterpreters/lox/CMakeLists.txt b/src/com/craftinginterpreters/lox/CMakeLists.txt
index 391331a..aeab240 100644
--- a/src/com/craftinginterpreters/lox/CMakeLists.txt
+++ b/src/com/craftinginterpreters/lox/CMakeLists.txt
@@ -23,4 +23,6 @@ add_jar(Lox
LoxFunction.java
Return.java
Resolver.java
+ LoxClass.java
+ LoxInstance.java
ENTRY_POINT com/craftinginterpreters/lox/Lox)
diff --git a/src/com/craftinginterpreters/lox/Interpreter.java b/src/com/craftinginterpreters/lox/Interpreter.java
index 4092191..5c1ea50 100644
--- a/src/com/craftinginterpreters/lox/Interpreter.java
+++ b/src/com/craftinginterpreters/lox/Interpreter.java
@@ -50,6 +50,19 @@ class Interpreter implements Expr.Visitor<Object>, Stmt.Visitor<Void> {
}
@Override
+ public Object visitSetExpr(Expr.Set expr) {
+ Object object = evaluate(expr.object);
+
+ if (!(object instanceof LoxInstance)) {
+ throw new RuntimeError(expr.name, "Only instances have fields.");
+ }
+
+ Object value = evaluate(expr.value);
+ ((LoxInstance) object).set(expr.name, value);
+ return value;
+ }
+
+ @Override
public Object visitUnaryExpr(Expr.Unary expr) {
Object right = evaluate(expr.right);
@@ -162,6 +175,14 @@ class Interpreter implements Expr.Visitor<Object>, Stmt.Visitor<Void> {
}
@Override
+ public Void visitClassStmt(Stmt.Class stmt) {
+ environment.define(stmt.name.lexeme, null);
+ LoxClass klass = new LoxClass(stmt.name.lexeme);
+ environment.assign(stmt.name, klass);
+ return null;
+ }
+
+ @Override
public Void visitExpressionStmt(Stmt.Expression stmt) {
evaluate(stmt.expression);
return null;
@@ -304,6 +325,16 @@ class Interpreter implements Expr.Visitor<Object>, Stmt.Visitor<Void> {
return function.call(this, arguments);
}
+ @Override
+ public Object visitGetExpr(Expr.Get expr) {
+ Object object = evaluate(expr.object);
+ if (object instanceof LoxInstance) {
+ return ((LoxInstance) object).get(expr.name);
+ }
+
+ throw new RuntimeError(expr.name, "Only instances have properties.");
+ }
+
public void interpret(List<Stmt> statements) {
try {
for (Stmt statement : statements) {
diff --git a/src/com/craftinginterpreters/lox/LoxClass.java b/src/com/craftinginterpreters/lox/LoxClass.java
new file mode 100644
index 0000000..4f7ef4f
--- /dev/null
+++ b/src/com/craftinginterpreters/lox/LoxClass.java
@@ -0,0 +1,27 @@
+package com.craftinginterpreters.lox;
+
+import java.util.List;
+
+class LoxClass implements LoxCallable {
+ final String name;
+
+ LoxClass(String name) {
+ this.name = name;
+ }
+
+ @Override
+ public String toString() {
+ return name;
+ }
+
+ @Override
+ public Object call(Interpreter interpreter, List<Object> arguments) {
+ LoxInstance instance = new LoxInstance(this);
+ return instance;
+ }
+
+ @Override
+ public int arity() {
+ return 0;
+ }
+}
diff --git a/src/com/craftinginterpreters/lox/LoxInstance.java b/src/com/craftinginterpreters/lox/LoxInstance.java
new file mode 100644
index 0000000..4e5eb9e
--- /dev/null
+++ b/src/com/craftinginterpreters/lox/LoxInstance.java
@@ -0,0 +1,30 @@
+package com.craftinginterpreters.lox;
+
+import java.util.HashMap;
+import java.util.Map;
+
+class LoxInstance {
+ private LoxClass klass;
+ private final Map<String, Object> fields = new HashMap<>();
+
+ LoxInstance(LoxClass klass) {
+ this.klass = klass;
+ }
+
+ Object get(Token name) {
+ if (fields.containsKey(name.lexeme)) {
+ return fields.get(name.lexeme);
+ }
+
+ throw new RuntimeError(name, "Undefined proprety '" + name.lexeme + "'.");
+ }
+
+ void set(Token name, Object value) {
+ fields.put(name.lexeme, value);
+ }
+
+ @Override
+ public String toString() {
+ return klass.name + " instance";
+ }
+}
diff --git a/src/com/craftinginterpreters/lox/Parser.java b/src/com/craftinginterpreters/lox/Parser.java
index 528326c..fe51e45 100644
--- a/src/com/craftinginterpreters/lox/Parser.java
+++ b/src/com/craftinginterpreters/lox/Parser.java
@@ -33,6 +33,8 @@ class Parser {
private Stmt declaration() {
try {
+ if (match(CLASS))
+ return classDeclaration();
if (match(FUN))
return function("function");
if (match(VAR))
@@ -45,6 +47,20 @@ class Parser {
}
}
+ private Stmt classDeclaration() {
+ Token name = consume(IDENTIFIER, "Expect class name.");
+ consume(LEFT_BRACE, "Expect '{' before class body.");
+
+ List<Stmt.Function> methods = new ArrayList<>();
+ while (!check(RIGHT_BRACE) && !isAtEnd()) {
+ methods.add(function("method"));
+ }
+
+ consume(RIGHT_BRACE, "Expect '}' after class body.");
+
+ return new Stmt.Class(name, methods);
+ }
+
private Stmt statement() {
if (match(FOR))
return forStatement();
@@ -201,6 +217,9 @@ class Parser {
if (expr instanceof Expr.Variable) {
Token name = ((Expr.Variable) expr).name;
return new Expr.Assign(name, value);
+ } else if (expr instanceof Expr.Get) {
+ Expr.Get get = (Expr.Get)expr;
+ return new Expr.Set(get.object, get.name, value);
}
error(equals, "Invalid assignment target.");
@@ -313,6 +332,9 @@ class Parser {
while (true) {
if (match(LEFT_PAREN)) {
expr = finishCall(expr);
+ } else if (match(DOT)) {
+ Token name = consume(IDENTIFIER, "Expect property name after '.'");
+ expr = new Expr.Get(expr, name);
} else {
break;
}
diff --git a/src/com/craftinginterpreters/lox/Resolver.java b/src/com/craftinginterpreters/lox/Resolver.java
index 2a870ab..a7c9911 100644
--- a/src/com/craftinginterpreters/lox/Resolver.java
+++ b/src/com/craftinginterpreters/lox/Resolver.java
@@ -33,6 +33,13 @@ class Resolver implements Expr.Visitor<Void>, Stmt.Visitor<Void> {
}
@Override
+ public Void visitClassStmt(Stmt.Class stmt) {
+ declare(stmt.name);
+ define(stmt.name);
+ return null;
+ }
+
+ @Override
public Void visitExpressionStmt(Stmt.Expression stmt) {
resolve(stmt.expression);
return null;
@@ -118,6 +125,12 @@ class Resolver implements Expr.Visitor<Void>, Stmt.Visitor<Void> {
}
@Override
+ public Void visitGetExpr(Expr.Get expr) {
+ resolve(expr.object);
+ return null;
+ }
+
+ @Override
public Void visitGroupingExpr(Expr.Grouping expr) {
resolve(expr.expression);
return null;
@@ -129,6 +142,13 @@ class Resolver implements Expr.Visitor<Void>, Stmt.Visitor<Void> {
}
@Override
+ public Void visitSetExpr(Expr.Set expr) {
+ resolve(expr.value);
+ resolve(expr.object);
+ return null;
+ }
+
+ @Override
public Void visitLogicalExpr(Expr.Logical expr) {
resolve(expr.left);
resolve(expr.right);