| OLD | NEW |
| 1 // Copyright (c) 2017, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2017, the Dart project authors. Please see the AUTHORS file |
| 2 // for details. All rights reserved. Use of this source code is governed by a | 2 // for details. All rights reserved. Use of this source code is governed by a |
| 3 // BSD-style license that can be found in the LICENSE file. | 3 // BSD-style license that can be found in the LICENSE file. |
| 4 library kernel.interpreter; | 4 library kernel.interpreter; |
| 5 | 5 |
| 6 import '../ast.dart'; | 6 import '../ast.dart'; |
| 7 import '../ast.dart' as ast show Class; | 7 import '../ast.dart' as ast show Class; |
| 8 | 8 |
| 9 import '../log.dart'; | 9 import '../log.dart'; |
| 10 export '../log.dart'; | 10 export '../log.dart'; |
| (...skipping 35 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 46 final VariableDeclaration variable; | 46 final VariableDeclaration variable; |
| 47 final Location location; | 47 final Location location; |
| 48 | 48 |
| 49 Binding(this.variable, this.location); | 49 Binding(this.variable, this.location); |
| 50 } | 50 } |
| 51 | 51 |
| 52 class Environment { | 52 class Environment { |
| 53 final List<Binding> bindings = <Binding>[]; | 53 final List<Binding> bindings = <Binding>[]; |
| 54 final Environment parent; | 54 final Environment parent; |
| 55 | 55 |
| 56 Value get thisInstance { |
| 57 return containsThis() |
| 58 ? lookupThis().value |
| 59 : throw "Invalid reference to 'this' expression"; |
| 60 } |
| 61 |
| 56 Environment.empty() : parent = null; | 62 Environment.empty() : parent = null; |
| 57 Environment(this.parent); | 63 Environment(this.parent); |
| 58 | 64 |
| 59 bool contains(VariableDeclaration variable) { | 65 bool contains(VariableDeclaration variable) { |
| 60 for (Binding b in bindings.reversed) { | 66 for (Binding b in bindings) { |
| 61 if (identical(b.variable, variable)) return true; | 67 if (identical(b.variable, variable)) return true; |
| 62 } | 68 } |
| 63 return parent?.contains(variable) ?? false; | 69 return parent?.contains(variable) ?? false; |
| 64 } | 70 } |
| 65 | 71 |
| 72 bool containsThis() { |
| 73 for (Binding b in bindings) { |
| 74 if (identical(b.variable.name, 'this')) return true; |
| 75 } |
| 76 return parent?.containsThis() ?? false; |
| 77 } |
| 78 |
| 66 Binding lookupBinding(VariableDeclaration variable) { | 79 Binding lookupBinding(VariableDeclaration variable) { |
| 67 assert(contains(variable)); | 80 assert(contains(variable)); |
| 68 for (Binding b in bindings) { | 81 for (Binding b in bindings) { |
| 69 if (identical(b.variable, variable)) return b; | 82 if (identical(b.variable, variable)) return b; |
| 70 } | 83 } |
| 71 return parent.lookupBinding(variable); | 84 return parent.lookupBinding(variable); |
| 72 } | 85 } |
| 73 | 86 |
| 87 Location lookupThis() { |
| 88 assert(containsThis()); |
| 89 for (Binding b in bindings) { |
| 90 if (identical(b.variable.name, 'this')) return b.location; |
| 91 } |
| 92 return parent.lookupThis(); |
| 93 } |
| 94 |
| 74 Value lookup(VariableDeclaration variable) { | 95 Value lookup(VariableDeclaration variable) { |
| 75 return lookupBinding(variable).location.value; | 96 return lookupBinding(variable).location.value; |
| 76 } | 97 } |
| 77 | 98 |
| 78 void assign(VariableDeclaration variable, Value value) { | 99 void assign(VariableDeclaration variable, Value value) { |
| 79 assert(contains(variable)); | 100 assert(contains(variable)); |
| 80 lookupBinding(variable).location.value = value; | 101 lookupBinding(variable).location.value = value; |
| 81 } | 102 } |
| 82 | 103 |
| 83 Environment extend(VariableDeclaration variable, Value value) { | 104 Environment extend(VariableDeclaration variable, Value value) { |
| 84 assert(!contains(variable)); | 105 assert(!contains(variable)); |
| 85 return new Environment(this) | 106 return new Environment(this) |
| 86 ..bindings.add(new Binding(variable, new Location(value))); | 107 ..bindings.add(new Binding(variable, new Location(value))); |
| 87 } | 108 } |
| 109 |
| 110 Environment extendWithThis(ObjectValue v) { |
| 111 assert(!containsThis()); |
| 112 return extend(new VariableDeclaration('this'), v); |
| 113 } |
| 88 } | 114 } |
| 89 | 115 |
| 90 /// Evaluate expressions. | 116 /// Evaluate expressions. |
| 91 class Evaluator extends ExpressionVisitor1<Configuration, EvalConfiguration> { | 117 class Evaluator extends ExpressionVisitor1<Configuration, EvalConfiguration> { |
| 92 Configuration eval(Expression expr, EvalConfiguration config) => | 118 Configuration eval(Expression expr, EvalConfiguration config) => |
| 93 expr.accept1(this, config); | 119 expr.accept1(this, config); |
| 94 | 120 |
| 95 Configuration evalList(List<InterpreterExpression> list, Environment env, | 121 Configuration evalList(List<InterpreterExpression> list, Environment env, |
| 96 ApplicationContinuation cont) { | 122 ApplicationContinuation cont) { |
| 97 if (list.isNotEmpty) { | 123 if (list.isNotEmpty) { |
| (...skipping 1184 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1282 void _initializeNullFields(Class class_, ObjectValue newObject) { | 1308 void _initializeNullFields(Class class_, ObjectValue newObject) { |
| 1283 int superClassSize = class_.superclass?.instanceSize ?? 0; | 1309 int superClassSize = class_.superclass?.instanceSize ?? 0; |
| 1284 for (int i = superClassSize; i < class_.instanceSize; i++) { | 1310 for (int i = superClassSize; i < class_.instanceSize; i++) { |
| 1285 Field field = class_.instanceFields[i]; | 1311 Field field = class_.instanceFields[i]; |
| 1286 if (class_.getProperty(newObject, field) == null) { | 1312 if (class_.getProperty(newObject, field) == null) { |
| 1287 assert(field.initializer == null); | 1313 assert(field.initializer == null); |
| 1288 class_.setProperty(newObject, field, Value.nullInstance); | 1314 class_.setProperty(newObject, field, Value.nullInstance); |
| 1289 } | 1315 } |
| 1290 } | 1316 } |
| 1291 } | 1317 } |
| OLD | NEW |