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.reversed) { |
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.reversed) { | |
Dmitry Stefantsov
2017/07/14 00:43:08
Is it important that [bindings] is reversed?
zhivkag
2017/07/14 07:02:31
Done.
| |
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()); | |
Dmitry Stefantsov
2017/07/14 00:43:07
If we pass this `assert`, then at some iteration o
zhivkag
2017/07/14 07:02:31
The intent of this assert is similar to "contains"
Dmitry Stefantsov
2017/07/18 10:45:15
Yep. I misinterpreted [containsThis] above. Than
| |
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 |