OLD | NEW |
1 // Copyright (c) 2015, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2015, 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 | 4 |
5 import 'package:analyzer/dart/ast/ast.dart'; | 5 import 'package:analyzer/dart/ast/ast.dart'; |
6 import 'package:analyzer/dart/ast/visitor.dart'; | 6 import 'package:analyzer/dart/ast/visitor.dart'; |
7 import 'package:analyzer/dart/element/element.dart'; | 7 import 'package:analyzer/dart/element/element.dart'; |
8 import 'package:analyzer/src/generated/constant.dart'; | 8 import 'package:analyzer/src/generated/constant.dart'; |
9 import 'package:analyzer/error/listener.dart' | 9 import 'package:analyzer/error/listener.dart' |
10 show AnalysisErrorListener, ErrorReporter; | 10 show AnalysisErrorListener, ErrorReporter; |
11 import 'package:analyzer/src/generated/resolver.dart' show TypeProvider; | 11 import 'package:analyzer/src/generated/engine.dart'; |
12 import 'package:analyzer/src/generated/source.dart' show Source; | 12 import 'package:analyzer/src/generated/source.dart' show Source; |
13 import 'package:analyzer/src/dart/ast/ast.dart'; | 13 import 'package:analyzer/src/dart/ast/ast.dart'; |
14 | 14 |
15 /// True is the expression can be evaluated multiple times without causing | 15 /// True is the expression can be evaluated multiple times without causing |
16 /// code execution. This is true for final fields. This can be true for local | 16 /// code execution. This is true for final fields. This can be true for local |
17 /// variables, if: | 17 /// variables, if: |
18 /// * they are not assigned within the [context]. | 18 /// * they are not assigned within the [context]. |
19 /// * they are not assigned in a function closure anywhere. | 19 /// * they are not assigned in a function closure anywhere. |
20 /// True is the expression can be evaluated multiple times without causing | 20 /// True is the expression can be evaluated multiple times without causing |
21 /// code execution. This is true for final fields. This can be true for local | 21 /// code execution. This is true for final fields. This can be true for local |
(...skipping 76 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
98 if (parent is ConstructorName) return; | 98 if (parent is ConstructorName) return; |
99 if (parent is Label) return; | 99 if (parent is Label) return; |
100 | 100 |
101 if (node.inSetterContext() && node.staticElement == _variable) { | 101 if (node.inSetterContext() && node.staticElement == _variable) { |
102 _potentiallyMutated = true; | 102 _potentiallyMutated = true; |
103 } | 103 } |
104 } | 104 } |
105 } | 105 } |
106 | 106 |
107 class ConstFieldVisitor { | 107 class ConstFieldVisitor { |
108 final ConstantVisitor _constantVisitor; | 108 final ConstantVisitor constantVisitor; |
109 | 109 |
110 ConstFieldVisitor(TypeProvider types, {Source dummySource}) | 110 ConstFieldVisitor(AnalysisContext context, {Source dummySource}) |
111 // TODO(jmesserly): support -D variables on the command line | 111 : constantVisitor = new ConstantVisitor( |
112 : _constantVisitor = new ConstantVisitor( | 112 new ConstantEvaluationEngine( |
113 new ConstantEvaluationEngine(types, new DeclaredVariables()), | 113 context.typeProvider, context.declaredVariables), |
114 new ErrorReporter( | 114 new ErrorReporter( |
115 AnalysisErrorListener.NULL_LISTENER, dummySource)); | 115 AnalysisErrorListener.NULL_LISTENER, dummySource)); |
116 | 116 |
117 // TODO(jmesserly): this is used to determine if the field initialization is | 117 // TODO(jmesserly): this is used to determine if the field initialization is |
118 // side effect free. We should make the check more general, as things like | 118 // side effect free. We should make the check more general, as things like |
119 // list/map literals/regexp are also side effect free and fairly common | 119 // list/map literals/regexp are also side effect free and fairly common |
120 // to use as field initializers. | 120 // to use as field initializers. |
121 bool isFieldInitConstant(VariableDeclaration field) => | 121 bool isFieldInitConstant(VariableDeclaration field) => |
122 field.initializer == null || computeConstant(field) != null; | 122 field.initializer == null || computeConstant(field) != null; |
123 | 123 |
124 DartObject computeConstant(VariableDeclaration field) { | 124 DartObject computeConstant(VariableDeclaration field) { |
125 // If the constant is already computed by ConstantEvaluator, just return it. | 125 // If the constant is already computed by ConstantEvaluator, just return it. |
126 VariableElement element = field.element; | 126 VariableElement element = field.element; |
127 var result = element.constantValue; | 127 var result = element.constantValue; |
128 if (result != null) return result; | 128 if (result != null) return result; |
129 | 129 |
130 // ConstantEvaluator will not compute constants for non-const fields, | 130 // ConstantEvaluator will not compute constants for non-const fields, |
131 // so run ConstantVisitor for those to figure out if the initializer is | 131 // so run ConstantVisitor for those to figure out if the initializer is |
132 // actually a constant (and therefore side effect free to evaluate). | 132 // actually a constant (and therefore side effect free to evaluate). |
133 assert(!field.isConst); | 133 assert(!field.isConst); |
134 | 134 |
135 var initializer = field.initializer; | 135 var initializer = field.initializer; |
136 if (initializer == null) return null; | 136 if (initializer == null) return null; |
137 return initializer.accept(_constantVisitor); | 137 return initializer.accept(constantVisitor); |
138 } | 138 } |
139 } | 139 } |
OLD | NEW |