Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(578)

Side by Side Diff: pkg/compiler/lib/src/js_model/closure_visitors.dart

Issue 2981723002: Distinguish between actual closure scopes and non-closure scopes. (Closed)
Patch Set: . Created 3 years, 5 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
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 4
5 import 'package:kernel/ast.dart' as ir; 5 import 'package:kernel/ast.dart' as ir;
6 6
7 import '../closure.dart'; 7 import '../closure.dart';
8 import '../elements/entities.dart'; 8 import '../elements/entities.dart';
9 import '../kernel/element_map.dart'; 9 import '../kernel/element_map.dart';
10 import 'closure.dart'; 10 import 'closure.dart';
11 11
12 /// This builder walks the code to determine what variables are captured/free at 12 /// This builder walks the code to determine what variables are captured/free at
13 /// various points to build ClosureScope that can respond to queries 13 /// various points to build ClosureScope that can respond to queries
14 /// about how a particular variable is being used at any point in the code. 14 /// about how a particular variable is being used at any point in the code.
15 class ClosureScopeBuilder extends ir.Visitor { 15 class ClosureScopeBuilder extends ir.Visitor {
16 /// A map of each visited call node with the associated information about what 16 /// A map of each visited call node with the associated information about what
17 /// variables are captured/used. Each ir.Node key corresponds to a scope that 17 /// variables are captured/used. Each ir.Node key corresponds to a scope that
18 /// was encountered while visiting a closure (initially called through 18 /// was encountered while visiting a closure (initially called through
19 /// [translateLazyIntializer] or [translateConstructorOrProcedure]). 19 /// [translateLazyIntializer] or [translateConstructorOrProcedure]).
20 Map<ir.Node, ClosureScope> _closureInfoMap = <ir.Node, ClosureScope>{}; 20 Map<ir.Node, ClosureScope> _closureInfoMap = <ir.Node, ClosureScope>{};
21 21
22 /// Map entities to their corresponding scope information (such as what
23 /// variables are captured/used). The distinction between this map and
24 /// [_closureInfoMap] is that [_closureInfoMap] stores this data for closures
25 /// specifically, whereas [_scopeInfoMap] stores this information for entities
26 /// that are *not* closures (this information is used by the locals handler).
27 /// The union of these two maps represents all the scopes encountered.
28 Map<Entity, ScopeInfo> _scopeInfoMap = <Entity, ScopeInfo>{};
sra1 2017/07/12 23:57:41 final, don't initialize.
Emily Fortuna 2017/07/13 21:03:39 Done.
29
22 /// A map of the nodes that we have flagged as necessary to generate closure 30 /// A map of the nodes that we have flagged as necessary to generate closure
23 /// classes for in a later stage. We map that node to information ascertained 31 /// classes for in a later stage. We map that node to information ascertained
24 /// about variable usage in the surrounding scope. 32 /// about variable usage in the surrounding scope.
25 Map<ir.TreeNode /* ir.Field | ir.FunctionNode */, ScopeInfo> 33 Map<ir.TreeNode /* ir.Field | ir.FunctionNode */, ScopeInfo>
26 _closuresToGenerate = <ir.TreeNode, ScopeInfo>{}; 34 _closuresToGenerate = <ir.TreeNode, ScopeInfo>{};
27 35
28 /// The local variables that have been declared in the current scope. 36 /// The local variables that have been declared in the current scope.
29 List<ir.VariableDeclaration> _scopeVariables; 37 List<ir.VariableDeclaration> _scopeVariables;
30 38
31 /// Pointer to the context in which this closure is executed. 39 /// Pointer to the context in which this closure is executed.
(...skipping 22 matching lines...) Expand all
54 bool _inTry = false; 62 bool _inTry = false;
55 63
56 /// Lookup the local entity that corresponds to a kernel variable declaration. 64 /// Lookup the local entity that corresponds to a kernel variable declaration.
57 final KernelToLocalsMap _localsMap; 65 final KernelToLocalsMap _localsMap;
58 66
59 /// The current scope we are in. 67 /// The current scope we are in.
60 KernelScopeInfo _currentScopeInfo; 68 KernelScopeInfo _currentScopeInfo;
61 69
62 final KernelToElementMap _kernelToElementMap; 70 final KernelToElementMap _kernelToElementMap;
63 71
64 ClosureScopeBuilder(this._closureInfoMap, this._closuresToGenerate, 72 ClosureScopeBuilder(this._closureInfoMap, this._scopeInfoMap,
65 this._localsMap, this._kernelToElementMap); 73 this._closuresToGenerate, this._localsMap, this._kernelToElementMap);
66 74
67 /// Update the [ClosureScope] object corresponding to 75 /// Update the [ClosureScope] object corresponding to
68 /// this node if any variables are captured. 76 /// this node if any variables are captured.
69 void attachCapturedScopeVariables(ir.Node node) { 77 void attachCapturedScopeVariables(ir.Node node) {
70 Set<Local> capturedVariablesForScope = new Set<Local>(); 78 Set<Local> capturedVariablesForScope = new Set<Local>();
71 79
72 for (ir.VariableDeclaration variable in _scopeVariables) { 80 for (ir.VariableDeclaration variable in _scopeVariables) {
73 // No need to box non-assignable elements. 81 // No need to box non-assignable elements.
74 if (variable.isFinal || variable.isConst) continue; 82 if (variable.isFinal || variable.isConst) continue;
75 if (!_mutatedVariables.contains(variable)) continue; 83 if (!_mutatedVariables.contains(variable)) continue;
(...skipping 129 matching lines...) Expand 10 before | Expand all | Expand 10 after
205 213
206 void visitInvokable(ir.TreeNode node) { 214 void visitInvokable(ir.TreeNode node) {
207 bool oldIsInsideClosure = _isInsideClosure; 215 bool oldIsInsideClosure = _isInsideClosure;
208 ir.Node oldExecutableContext = _executableContext; 216 ir.Node oldExecutableContext = _executableContext;
209 KernelScopeInfo oldScopeInfo = _currentScopeInfo; 217 KernelScopeInfo oldScopeInfo = _currentScopeInfo;
210 218
211 // _outermostNode is only null the first time we enter the body of the 219 // _outermostNode is only null the first time we enter the body of the
212 // field, constructor, or method that is being analyzed. 220 // field, constructor, or method that is being analyzed.
213 _isInsideClosure = _outermostNode != null; 221 _isInsideClosure = _outermostNode != null;
214 _executableContext = node; 222 _executableContext = node;
215 if (!_isInsideClosure) { 223 _currentScopeInfo = new KernelScopeInfo(_nodeToThisLocal(node));
224 if (_isInsideClosure) {
225 _closuresToGenerate[node] = _currentScopeInfo;
226 } else {
216 _outermostNode = node; 227 _outermostNode = node;
228 Entity entity;
229 ir.TreeNode tempNode = node;
230 if (tempNode is ir.Member) {
231 entity = _kernelToElementMap.getMember(tempNode);
Johnni Winther 2017/07/13 08:56:14 Maybe the visitor should have the member entity as
Emily Fortuna 2017/07/13 21:03:39 done.
232 } else {
233 entity = _kernelToElementMap.getLocalFunction(tempNode);
Johnni Winther 2017/07/13 08:56:14 Can this ever happen? We always start visiting on
Emily Fortuna 2017/07/13 21:03:39 You're right it won't. Took it out.
234 }
235 _scopeInfoMap[entity] = _currentScopeInfo;
217 } 236 }
218 _currentScopeInfo = new KernelScopeInfo(_nodeToThisLocal(node));
219 _closuresToGenerate[node] = _currentScopeInfo;
220 237
221 enterNewScope(node, () { 238 enterNewScope(node, () {
222 node.visitChildren(this); 239 node.visitChildren(this);
223 }); 240 });
224 241
225 KernelScopeInfo savedScopeInfo = _currentScopeInfo; 242 KernelScopeInfo savedScopeInfo = _currentScopeInfo;
226 bool savedIsInsideClosure = _isInsideClosure; 243 bool savedIsInsideClosure = _isInsideClosure;
227 244
228 // Restore old values. 245 // Restore old values.
229 _isInsideClosure = oldIsInsideClosure; 246 _isInsideClosure = oldIsInsideClosure;
(...skipping 55 matching lines...) Expand 10 before | Expand all | Expand 10 after
285 nodeToConvert.isInstanceMember)) { 302 nodeToConvert.isInstanceMember)) {
286 return new ThisLocal(_kernelToElementMap.getConstructor(nodeToConvert)); 303 return new ThisLocal(_kernelToElementMap.getConstructor(nodeToConvert));
287 } else if (nodeToConvert is ir.Procedure && 304 } else if (nodeToConvert is ir.Procedure &&
288 nodeToConvert.isInstanceMember) { 305 nodeToConvert.isInstanceMember) {
289 return new ThisLocal(_kernelToElementMap.getMethod(nodeToConvert)); 306 return new ThisLocal(_kernelToElementMap.getMethod(nodeToConvert));
290 } 307 }
291 } 308 }
292 return null; 309 return null;
293 } 310 }
294 } 311 }
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698