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

Side by Side Diff: dart/sdk/lib/_internal/compiler/implementation/js_emitter/container_builder.dart

Issue 27524003: Generate tear-off closures dynamically. (Closed) Base URL: https://dart.googlecode.com/svn/branches/bleeding_edge
Patch Set: Fixed unit tests. Created 7 years 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 | Annotate | Revision Log
OLDNEW
1 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file 1 // Copyright (c) 2013, 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 part of dart2js.js_emitter; 5 part of dart2js.js_emitter;
6 6
7 /// This class should morph into something that makes it easy to build 7 /// This class should morph into something that makes it easy to build
8 /// JavaScript representations of libraries, class-sides, and instance-sides. 8 /// JavaScript representations of libraries, class-sides, and instance-sides.
9 /// Initially, it is just a placeholder for code that is moved from 9 /// Initially, it is just a placeholder for code that is moved from
10 /// [CodeEmitterTask]. 10 /// [CodeEmitterTask].
11 class ContainerBuilder extends CodeEmitterHelper { 11 class ContainerBuilder extends CodeEmitterHelper {
12 final Map<Element, Element> staticGetters = new Map<Element, Element>(); 12 final Map<Element, Element> staticGetters = new Map<Element, Element>();
13 13
14 /// A cache of synthesized closures for top-level, static or 14 /// A cache of synthesized closures for top-level, static or
15 /// instance methods. 15 /// instance methods.
16 final Map<String, Element> methodClosures = <String, Element>{}; 16 final Map<String, Element> methodClosures = <String, Element>{};
17 17
18 /** 18 /**
19 * Generate stubs to handle invocation of methods with optional 19 * Generate stubs to handle invocation of methods with optional
20 * arguments. 20 * arguments.
21 * 21 *
22 * A method like [: foo([x]) :] may be invoked by the following 22 * A method like [: foo([x]) :] may be invoked by the following
23 * calls: [: foo(), foo(1), foo(x: 1) :]. See the sources of this 23 * calls: [: foo(), foo(1), foo(x: 1) :]. See the sources of this
24 * function for detailed examples. 24 * function for detailed examples.
25 */ 25 */
26 void addParameterStub(FunctionElement member, 26 void addParameterStub(FunctionElement member,
27 Selector selector, 27 Selector selector,
28 DefineStubFunction defineStub, 28 AddStubFunction addStub,
29 Set<String> alreadyGenerated) { 29 Set<String> alreadyGenerated) {
30 FunctionSignature parameters = member.computeSignature(compiler); 30 FunctionSignature parameters = member.computeSignature(compiler);
31 int positionalArgumentCount = selector.positionalArgumentCount; 31 int positionalArgumentCount = selector.positionalArgumentCount;
32 if (positionalArgumentCount == parameters.parameterCount) { 32 if (positionalArgumentCount == parameters.parameterCount) {
33 assert(selector.namedArgumentCount == 0); 33 assert(selector.namedArgumentCount == 0);
34 return; 34 return;
35 } 35 }
36 if (parameters.optionalParametersAreNamed 36 if (parameters.optionalParametersAreNamed
37 && selector.namedArgumentCount == parameters.optionalParameterCount) { 37 && selector.namedArgumentCount == parameters.optionalParameterCount) {
38 // If the selector has the same number of named arguments as the element, 38 // If the selector has the same number of named arguments as the element,
(...skipping 35 matching lines...) Expand 10 before | Expand all | Expand 10 after
74 74
75 int optionalParameterStart = positionalArgumentCount + extraArgumentCount; 75 int optionalParameterStart = positionalArgumentCount + extraArgumentCount;
76 // Includes extra receiver argument when using interceptor convention 76 // Includes extra receiver argument when using interceptor convention
77 int indexOfLastOptionalArgumentInParameters = optionalParameterStart - 1; 77 int indexOfLastOptionalArgumentInParameters = optionalParameterStart - 1;
78 78
79 TreeElements elements = 79 TreeElements elements =
80 compiler.enqueuer.resolution.getCachedElements(member); 80 compiler.enqueuer.resolution.getCachedElements(member);
81 81
82 int parameterIndex = 0; 82 int parameterIndex = 0;
83 parameters.orderedForEachParameter((Element element) { 83 parameters.orderedForEachParameter((Element element) {
84 // Use generic names for closures to facilitate code sharing. 84 String jsName = backend.namer.safeName(element.name);
85 String jsName = member is ClosureInvocationElement
86 ? 'p${parameterIndex++}'
87 : backend.namer.safeName(element.name);
88 assert(jsName != receiverArgumentName); 85 assert(jsName != receiverArgumentName);
89 if (count < optionalParameterStart) { 86 if (count < optionalParameterStart) {
90 parametersBuffer[count] = new jsAst.Parameter(jsName); 87 parametersBuffer[count] = new jsAst.Parameter(jsName);
91 argumentsBuffer[count] = js(jsName); 88 argumentsBuffer[count] = js(jsName);
92 } else { 89 } else {
93 int index = names.indexOf(element.name); 90 int index = names.indexOf(element.name);
94 if (index != -1) { 91 if (index != -1) {
95 indexOfLastOptionalArgumentInParameters = count; 92 indexOfLastOptionalArgumentInParameters = count;
96 // The order of the named arguments is not the same as the 93 // The order of the named arguments is not the same as the
97 // one in the real method (which is in Dart source order). 94 // one in the real method (which is in Dart source order).
(...skipping 16 matching lines...) Expand all
114 } 111 }
115 count++; 112 count++;
116 }); 113 });
117 114
118 List body; 115 List body;
119 if (member.hasFixedBackendName()) { 116 if (member.hasFixedBackendName()) {
120 body = task.nativeEmitter.generateParameterStubStatements( 117 body = task.nativeEmitter.generateParameterStubStatements(
121 member, isInterceptedMethod, invocationName, 118 member, isInterceptedMethod, invocationName,
122 parametersBuffer, argumentsBuffer, 119 parametersBuffer, argumentsBuffer,
123 indexOfLastOptionalArgumentInParameters); 120 indexOfLastOptionalArgumentInParameters);
124 } else { 121 } else if (member.isInstanceMember()) {
125 body = [js.return_( 122 body = [js.return_(
126 js('this')[namer.getNameOfInstanceMember(member)](argumentsBuffer))]; 123 js('this')[namer.getNameOfInstanceMember(member)](argumentsBuffer))];
124 } else {
125 body = [js.return_(namer.elementAccess(member)(argumentsBuffer))];
127 } 126 }
128 127
129 jsAst.Fun function = js.fun(parametersBuffer, body); 128 jsAst.Fun function = js.fun(parametersBuffer, body);
130 129
131 defineStub(invocationName, function); 130 addStub(selector, function);
132
133 String reflectionName = task.getReflectionName(selector, invocationName);
134 if (reflectionName != null) {
135 var reflectable =
136 js(backend.isAccessibleByReflection(member) ? '1' : '0');
137 defineStub('+$reflectionName', reflectable);
138 }
139 } 131 }
140 132
141 void addParameterStubs(FunctionElement member, 133 void addParameterStubs(FunctionElement member, AddStubFunction defineStub,
142 DefineStubFunction defineStub) { 134 [bool canTearOff = false]) {
135 if (member.enclosingElement.isClosure()) {
136 ClosureClassElement cls = member.enclosingElement;
137 if (cls.supertype.element == compiler.boundClosureClass) {
138 compiler.internalErrorOnElement(cls.methodElement, 'Bound closure1.');
Johnni Winther 2013/12/05 11:49:56 Better error message.
ahe 2013/12/06 15:57:54 I'm not sure what you suggest would be better. I o
139 }
140 if (cls.methodElement.isInstanceMember()) {
141 compiler.internalErrorOnElement(cls.methodElement, 'Bound closure2.');
Johnni Winther 2013/12/05 11:49:56 Ditto.
ahe 2013/12/06 15:57:54 Ditto ditto :-)
142 }
143 }
144
143 // We fill the lists depending on the selector. For example, 145 // We fill the lists depending on the selector. For example,
144 // take method foo: 146 // take method foo:
145 // foo(a, b, {c, d}); 147 // foo(a, b, {c, d});
146 // 148 //
147 // We may have multiple ways of calling foo: 149 // We may have multiple ways of calling foo:
148 // (1) foo(1, 2); 150 // (1) foo(1, 2);
149 // (2) foo(1, 2, c: 3); 151 // (2) foo(1, 2, c: 3);
150 // (3) foo(1, 2, d: 4); 152 // (3) foo(1, 2, d: 4);
151 // (4) foo(1, 2, c: 3, d: 4); 153 // (4) foo(1, 2, c: 3, d: 4);
152 // (5) foo(1, 2, d: 4, c: 3); 154 // (5) foo(1, 2, d: 4, c: 3);
153 // 155 //
154 // What we generate at the call sites are: 156 // What we generate at the call sites are:
155 // (1) foo$2(1, 2); 157 // (1) foo$2(1, 2);
156 // (2) foo$3$c(1, 2, 3); 158 // (2) foo$3$c(1, 2, 3);
157 // (3) foo$3$d(1, 2, 4); 159 // (3) foo$3$d(1, 2, 4);
158 // (4) foo$4$c$d(1, 2, 3, 4); 160 // (4) foo$4$c$d(1, 2, 3, 4);
159 // (5) foo$4$c$d(1, 2, 3, 4); 161 // (5) foo$4$c$d(1, 2, 3, 4);
160 // 162 //
161 // The stubs we generate are (expressed in Dart): 163 // The stubs we generate are (expressed in Dart):
162 // (1) foo$2(a, b) => foo$4$c$d(a, b, null, null) 164 // (1) foo$2(a, b) => foo$4$c$d(a, b, null, null)
163 // (2) foo$3$c(a, b, c) => foo$4$c$d(a, b, c, null); 165 // (2) foo$3$c(a, b, c) => foo$4$c$d(a, b, c, null);
164 // (3) foo$3$d(a, b, d) => foo$4$c$d(a, b, null, d); 166 // (3) foo$3$d(a, b, d) => foo$4$c$d(a, b, null, d);
165 // (4) No stub generated, call is direct. 167 // (4) No stub generated, call is direct.
166 // (5) No stub generated, call is direct. 168 // (5) No stub generated, call is direct.
167 169
168 // Keep a cache of which stubs have already been generated, to 170 // Keep a cache of which stubs have already been generated, to
169 // avoid duplicates. Note that even if selectors are 171 // avoid duplicates. Note that even if selectors are
170 // canonicalized, we would still need this cache: a typed selector 172 // canonicalized, we would still need this cache: a typed selector
171 // on A and a typed selector on B could yield the same stub. 173 // on A and a typed selector on B could yield the same stub.
172 Set<String> generatedStubNames = new Set<String>(); 174 Set<Selector> selectors = null;
173 bool isClosureInvocation = 175 if (member.isInstanceMember()) {
174 member.name == namer.closureInvocationSelectorName; 176 // No stubs needed for static methods.
175 if (backend.isNeededForReflection(member) || 177 selectors = compiler.codegenWorld.invokedNames[member.name];
176 (compiler.enabledFunctionApply && isClosureInvocation)) { 178 }
177 // If [Function.apply] is called, we pessimistically compile all 179
178 // possible stubs for this closure. 180 /// Returns all closure call selectors renamed to match this member.
179 FunctionSignature signature = member.computeSignature(compiler); 181 Set<Selector> callSelectorsAsNamed() {
180 Set<Selector> selectors = signature.optionalParametersAreNamed 182 if (!canTearOff) return null;
181 ? computeSeenNamedSelectors(member) 183 Set<Selector> callSelectors = compiler.codegenWorld.invokedNames[
182 : computeOptionalSelectors(signature, member); 184 namer.closureInvocationSelectorName];
185 if (callSelectors == null) return null;
186 return callSelectors.map((Selector callSelector) {
187 return new Selector.call(
188 member.name, member.getLibrary(),
189 callSelector.argumentCount, callSelector.namedArguments);
190 }).toSet();
191 }
192 if (selectors == null) {
193 selectors = callSelectorsAsNamed();
194 if (selectors == null) return;
195 } else {
196 Set<Selector> callSelectors = callSelectorsAsNamed();
197 if (callSelectors != null) {
198 selectors = selectors.union(callSelectors);
199 }
200 }
201 Set<Selector> untypedSelectors = new Set();
Johnni Winther 2013/12/05 11:49:56 `new Set()` -> `new Set<Selector>()`
ahe 2013/12/06 15:57:54 Done.
202 if (selectors != null) {
183 for (Selector selector in selectors) { 203 for (Selector selector in selectors) {
184 addParameterStub(member, selector, defineStub, generatedStubNames); 204 if (!selector.appliesUnnamed(member, compiler)) continue;
205 if (untypedSelectors.add(selector.asUntyped)) {
206 // TODO(ahe): Is the last argument to [addParameterStub] needed?
207 addParameterStub(member, selector, defineStub, new Set<String>());
208 }
185 } 209 }
186 if (signature.optionalParametersAreNamed && isClosureInvocation) { 210 }
187 addCatchAllParameterStub(member, signature, defineStub); 211 if (canTearOff) {
188 } 212 selectors = compiler.codegenWorld.invokedNames[
189 } else { 213 namer.closureInvocationSelectorName];
190 Set<Selector> selectors = compiler.codegenWorld.invokedNames[member.name]; 214 if (selectors != null) {
191 if (selectors == null) return; 215 for (Selector selector in selectors) {
192 for (Selector selector in selectors) { 216 selector = new Selector.call(
193 if (!selector.applies(member, compiler)) continue; 217 member.name, member.getLibrary(),
194 addParameterStub(member, selector, defineStub, generatedStubNames); 218 selector.argumentCount, selector.namedArguments);
219 if (!selector.appliesUnnamed(member, compiler)) continue;
220 if (untypedSelectors.add(selector)) {
221 // TODO(ahe): Is the last argument to [addParameterStub] needed?
222 addParameterStub(member, selector, defineStub, new Set<String>());
223 }
224 }
195 } 225 }
196 } 226 }
197 } 227 }
198 228
199 Set<Selector> computeSeenNamedSelectors(FunctionElement element) {
200 Set<Selector> selectors = compiler.codegenWorld.invokedNames[element.name];
201 Set<Selector> result = new Set<Selector>();
202 if (selectors == null) return result;
203 for (Selector selector in selectors) {
204 if (!selector.applies(element, compiler)) continue;
205 result.add(selector);
206 }
207 return result;
208 }
209
210 void addCatchAllParameterStub(FunctionElement member,
211 FunctionSignature signature,
212 DefineStubFunction defineStub) {
213 // See Primities.applyFunction in js_helper.dart for details.
214 List<jsAst.Property> properties = <jsAst.Property>[];
215 for (Element element in signature.orderedOptionalParameters) {
216 String jsName = backend.namer.safeName(element.name);
217 Constant value = compiler.constantHandler.initialVariableValues[element];
218 jsAst.Expression reference = null;
219 if (value == null) {
220 reference = new jsAst.LiteralNull();
221 } else {
222 reference = task.constantReference(value);
223 }
224 properties.add(new jsAst.Property(js.string(jsName), reference));
225 }
226 defineStub(
227 backend.namer.callCatchAllName,
228 js.fun([], js.return_(new jsAst.ObjectInitializer(properties))));
229 }
230
231 /** 229 /**
232 * Compute the set of possible selectors in the presence of optional
233 * non-named parameters.
234 */
235 Set<Selector> computeOptionalSelectors(FunctionSignature signature,
236 FunctionElement element) {
237 Set<Selector> selectors = new Set<Selector>();
238 // Add the selector that does not have any optional argument.
239 selectors.add(new Selector(SelectorKind.CALL,
240 element.name,
241 element.getLibrary(),
242 signature.requiredParameterCount,
243 <String>[]));
244
245 // For each optional parameter, we increment the number of passed
246 // argument.
247 for (int i = 1; i <= signature.optionalParameterCount; i++) {
248 selectors.add(new Selector(SelectorKind.CALL,
249 element.name,
250 element.getLibrary(),
251 signature.requiredParameterCount + i,
252 <String>[]));
253 }
254 return selectors;
255 }
256
257 void emitStaticFunctionGetters(CodeBuffer eagerBuffer) {
258 task.addComment('Static function getters', task.mainBuffer);
259 for (FunctionElement element in
260 Elements.sortedByPosition(staticGetters.keys)) {
261 Element closure = staticGetters[element];
262 CodeBuffer buffer =
263 task.isDeferred(element) ? task.deferredConstants : eagerBuffer;
264 String closureClass = namer.isolateAccess(closure);
265 String name = namer.getStaticClosureName(element);
266
267 String closureName = namer.getStaticClosureName(element);
268 jsAst.Node assignment = js(
269 'init.globalFunctions["$closureName"] ='
270 ' ${namer.globalObjectFor(element)}.$name ='
271 ' new $closureClass(#, "$closureName")',
272 namer.elementAccess(element));
273 buffer.write(jsAst.prettyPrint(assignment, compiler));
274 buffer.write('$N');
275 }
276 }
277
278 void emitStaticFunctionClosures() {
279 Set<FunctionElement> functionsNeedingGetter =
280 compiler.codegenWorld.staticFunctionsNeedingGetter;
281 for (FunctionElement element in
282 Elements.sortedByPosition(functionsNeedingGetter)) {
283 String superName = namer.getNameOfClass(compiler.closureClass);
284 int parameterCount = element.functionSignature.parameterCount;
285 String name = 'Closure\$$parameterCount';
286 assert(task.instantiatedClasses.contains(compiler.closureClass));
287
288 ClassElement closureClassElement = new ClosureClassElement(
289 null, name, compiler, element,
290 element.getCompilationUnit());
291 // Now add the methods on the closure class. The instance method does not
292 // have the correct name. Since [addParameterStubs] use the name to create
293 // its stubs we simply create a fake element with the correct name.
294 // Note: the callElement will not have any enclosingElement.
295 FunctionElement callElement =
296 new ClosureInvocationElement(namer.closureInvocationSelectorName,
297 element);
298
299 String invocationName = namer.instanceMethodName(callElement);
300 String mangledName = namer.getNameOfClass(closureClassElement);
301
302 // Define the constructor with a name so that Object.toString can
303 // find the class name of the closure class.
304 ClassBuilder closureBuilder = new ClassBuilder();
305 // If a static function is used as a closure we need to add its name
306 // in case it is used in spawnFunction.
307 String methodName = namer.STATIC_CLOSURE_NAME_NAME;
308 List<String> fieldNames = <String>[invocationName, methodName];
309 closureBuilder.addProperty('',
310 js.string("$superName;${fieldNames.join(',')}"));
311
312 addParameterStubs(callElement, closureBuilder.addProperty);
313
314 void emitFunctionTypeSignature(Element method, FunctionType methodType) {
315 RuntimeTypes rti = backend.rti;
316 // [:() => null:] is dummy encoding of [this] which is never needed for
317 // the encoding of the type of the static [method].
318 jsAst.Expression encoding =
319 rti.getSignatureEncoding(methodType, js('null'));
320 String operatorSignature = namer.operatorSignature();
321 // TODO(johnniwinther): Make MiniJsParser support function expressions.
322 closureBuilder.addProperty(operatorSignature, encoding);
323 }
324
325 FunctionType methodType = element.computeType(compiler);
326 Map<FunctionType, bool> functionTypeChecks =
327 task.typeTestEmitter.getFunctionTypeChecksOn(methodType);
328 task.typeTestEmitter.generateFunctionTypeTests(
329 element, methodType, functionTypeChecks,
330 emitFunctionTypeSignature);
331
332 closureClassElement =
333 addClosureIfNew(closureBuilder, closureClassElement, fieldNames);
334 staticGetters[element] = closureClassElement;
335
336 }
337 }
338
339 ClassElement addClosureIfNew(ClassBuilder builder,
340 ClassElement closure,
341 List<String> fieldNames) {
342 String key =
343 jsAst.prettyPrint(builder.toObjectInitializer(), compiler).getText();
344 return methodClosures.putIfAbsent(key, () {
345 String mangledName = namer.getNameOfClass(closure);
346 emitClosureInPrecompiledFunction(mangledName, fieldNames);
347 return closure;
348 });
349 }
350
351 void emitClosureInPrecompiledFunction(String mangledName,
352 List<String> fieldNames) {
353 List<String> fields = fieldNames;
354 String constructorName = mangledName;
355 task.precompiledFunction.add(new jsAst.FunctionDeclaration(
356 new jsAst.VariableDeclaration(constructorName),
357 js.fun(fields, fields.map(
358 (name) => js('this.$name = $name')).toList())));
359 task.precompiledFunction.addAll([
360 js('$constructorName.builtin\$cls = "$constructorName"'),
361 js('\$desc=\$collectedClasses.$constructorName'),
362 js.if_('\$desc instanceof Array', js('\$desc = \$desc[1]')),
363 js('$constructorName.prototype = \$desc'),
364 ]);
365
366 task.precompiledConstructorNames.add(js(constructorName));
367 }
368
369 /**
370 * Documentation wanted -- johnniwinther
371 *
372 * Invariant: [member] must be a declaration element.
373 */
374 void emitDynamicFunctionGetter(FunctionElement member,
375 DefineStubFunction defineStub) {
376 assert(invariant(member, member.isDeclaration));
377 assert(task.instantiatedClasses.contains(compiler.boundClosureClass));
378 // For every method that has the same name as a property-get we create a
379 // getter that returns a bound closure. Say we have a class 'A' with method
380 // 'foo' and somewhere in the code there is a dynamic property get of
381 // 'foo'. Then we generate the following code (in pseudo Dart/JavaScript):
382 //
383 // class A {
384 // foo(x, y, z) { ... } // Original function.
385 // get foo { return new BoundClosure499(this, "foo"); }
386 // }
387 // class BoundClosure499 extends BoundClosure {
388 // BoundClosure499(this.self, this.name);
389 // $call3(x, y, z) { return self[name](x, y, z); }
390 // }
391
392 bool hasOptionalParameters = member.optionalParameterCount(compiler) != 0;
393 int parameterCount = member.parameterCount(compiler);
394
395 // Intercepted methods take an extra parameter, which is the
396 // receiver of the call.
397 bool inInterceptor = backend.isInterceptedMethod(member);
398 List<String> fieldNames = <String>[];
399 compiler.boundClosureClass.forEachInstanceField((_, Element field) {
400 fieldNames.add(namer.instanceFieldPropertyName(field));
401 });
402
403 ClassElement classElement = member.getEnclosingClass();
404 String name = inInterceptor
405 ? 'BoundClosure\$i${parameterCount}'
406 : 'BoundClosure\$${parameterCount}';
407
408 ClassElement closureClassElement = new ClosureClassElement(
409 null, name, compiler, member,
410 member.getCompilationUnit());
411 String superName = namer.getNameOfClass(closureClassElement.superclass);
412
413 // Define the constructor with a name so that Object.toString can
414 // find the class name of the closure class.
415 ClassBuilder boundClosureBuilder = new ClassBuilder();
416 boundClosureBuilder.addProperty('',
417 js.string("$superName;${fieldNames.join(',')}"));
418 // Now add the methods on the closure class. The instance method does not
419 // have the correct name. Since [addParameterStubs] use the name to create
420 // its stubs we simply create a fake element with the correct name.
421 // Note: the callElement will not have any enclosingElement.
422 FunctionElement callElement = new ClosureInvocationElement(
423 namer.closureInvocationSelectorName, member);
424
425 String invocationName = namer.instanceMethodName(callElement);
426
427 List<String> parameters = <String>[];
428 List<jsAst.Expression> arguments =
429 <jsAst.Expression>[js('this')[fieldNames[0]]];
430 if (inInterceptor) {
431 arguments.add(js('this')[fieldNames[2]]);
432 }
433 for (int i = 0; i < parameterCount; i++) {
434 String name = 'p$i';
435 parameters.add(name);
436 arguments.add(js(name));
437 }
438
439 jsAst.Expression fun = js.fun(
440 parameters,
441 js.return_(
442 js('this')[fieldNames[1]]['call'](arguments)));
443 boundClosureBuilder.addProperty(invocationName, fun);
444
445 addParameterStubs(callElement, boundClosureBuilder.addProperty);
446
447 void emitFunctionTypeSignature(Element method, FunctionType methodType) {
448 jsAst.Expression encoding = backend.rti.getSignatureEncoding(
449 methodType, js('this')[fieldNames[0]]);
450 String operatorSignature = namer.operatorSignature();
451 boundClosureBuilder.addProperty(operatorSignature, encoding);
452 }
453
454 DartType memberType = member.computeType(compiler);
455 Map<FunctionType, bool> functionTypeChecks =
456 task.typeTestEmitter.getFunctionTypeChecksOn(memberType);
457
458 task.typeTestEmitter.generateFunctionTypeTests(
459 member, memberType, functionTypeChecks,
460 emitFunctionTypeSignature);
461
462 closureClassElement =
463 addClosureIfNew(boundClosureBuilder, closureClassElement, fieldNames);
464
465 String closureClass = namer.isolateAccess(closureClassElement);
466
467 // And finally the getter.
468 String getterName = namer.getterName(member);
469 String targetName = namer.instanceMethodName(member);
470
471 parameters = <String>[];
472 jsAst.PropertyAccess method =
473 backend.namer.elementAccess(classElement)['prototype'][targetName];
474 arguments = <jsAst.Expression>[js('this'), method];
475
476 if (inInterceptor) {
477 String receiverArg = fieldNames[2];
478 parameters.add(receiverArg);
479 arguments.add(js(receiverArg));
480 } else {
481 // Put null in the intercepted receiver field.
482 arguments.add(new jsAst.LiteralNull());
483 }
484
485 arguments.add(js.string(targetName));
486
487 jsAst.Expression getterFunction = js.fun(
488 parameters, js.return_(js(closureClass).newWith(arguments)));
489
490 defineStub(getterName, getterFunction);
491 }
492
493 /**
494 * Documentation wanted -- johnniwinther 230 * Documentation wanted -- johnniwinther
495 * 231 *
496 * Invariant: [member] must be a declaration element. 232 * Invariant: [member] must be a declaration element.
497 */ 233 */
498 void emitCallStubForGetter(Element member, 234 void emitCallStubForGetter(Element member,
499 Set<Selector> selectors, 235 Set<Selector> selectors,
500 DefineStubFunction defineStub) { 236 AddPropertyFunction addProperty) {
501 assert(invariant(member, member.isDeclaration)); 237 assert(invariant(member, member.isDeclaration));
502 LibraryElement memberLibrary = member.getLibrary(); 238 LibraryElement memberLibrary = member.getLibrary();
503 // If the method is intercepted, the stub gets the 239 // If the method is intercepted, the stub gets the
504 // receiver explicitely and we need to pass it to the getter call. 240 // receiver explicitely and we need to pass it to the getter call.
505 bool isInterceptedMethod = backend.isInterceptedMethod(member); 241 bool isInterceptedMethod = backend.isInterceptedMethod(member);
506 242
507 const String receiverArgumentName = r'$receiver'; 243 const String receiverArgumentName = r'$receiver';
508 244
509 jsAst.Expression buildGetter() { 245 jsAst.Expression buildGetter() {
510 if (member.isGetter()) { 246 if (member.isGetter()) {
(...skipping 31 matching lines...) Expand 10 before | Expand all | Expand 10 after
542 for (int i = 0; i < selector.argumentCount; i++) { 278 for (int i = 0; i < selector.argumentCount; i++) {
543 String name = 'arg$i'; 279 String name = 'arg$i';
544 parameters.add(new jsAst.Parameter(name)); 280 parameters.add(new jsAst.Parameter(name));
545 arguments.add(js(name)); 281 arguments.add(js(name));
546 } 282 }
547 283
548 jsAst.Fun function = js.fun( 284 jsAst.Fun function = js.fun(
549 parameters, 285 parameters,
550 js.return_(buildGetter()[closureCallName](arguments))); 286 js.return_(buildGetter()[closureCallName](arguments)));
551 287
552 defineStub(invocationName, function); 288 addProperty(invocationName, function);
553 } 289 }
554 } 290 }
555 } 291 }
556 292
557 /** 293 /**
558 * Documentation wanted -- johnniwinther 294 * Documentation wanted -- johnniwinther
559 * 295 *
560 * Invariant: [member] must be a declaration element. 296 * Invariant: [member] must be a declaration element.
561 */ 297 */
562 void emitExtraAccessors(Element member, ClassBuilder builder) { 298 void emitExtraAccessors(Element member, ClassBuilder builder) {
563 assert(invariant(member, member.isDeclaration)); 299 assert(invariant(member, member.isDeclaration));
564 if (member.isGetter() || member.isField()) { 300 if (member.isGetter() || member.isField()) {
565 Set<Selector> selectors = compiler.codegenWorld.invokedNames[member.name]; 301 Set<Selector> selectors = compiler.codegenWorld.invokedNames[member.name];
566 if (selectors != null && !selectors.isEmpty) { 302 if (selectors != null && !selectors.isEmpty) {
567 emitCallStubForGetter(member, selectors, builder.addProperty); 303 emitCallStubForGetter(member, selectors, builder.addProperty);
568 } 304 }
569 } else if (member.isFunction()) {
570 if (compiler.codegenWorld.hasInvokedGetter(member, compiler)) {
571 emitDynamicFunctionGetter(member, builder.addProperty);
572 }
573 } 305 }
574 } 306 }
575 307
576 void addMember(Element member, ClassBuilder builder) { 308 void addMember(Element member, ClassBuilder builder) {
577 assert(invariant(member, member.isDeclaration)); 309 assert(invariant(member, member.isDeclaration));
578 310
579 if (member.isField()) { 311 if (member.isField()) {
580 addMemberField(member, builder); 312 addMemberField(member, builder);
581 } else if (member.isFunction() || 313 } else if (member.isFunction() ||
582 member.isGenerativeConstructorBody() || 314 member.isGenerativeConstructorBody() ||
583 member.isGenerativeConstructor() || 315 member.isGenerativeConstructor() ||
584 member.isAccessor()) { 316 member.isAccessor()) {
585 addMemberMethod(member, builder); 317 addMemberMethod(member, builder);
586 } else { 318 } else {
587 compiler.internalErrorOnElement( 319 compiler.internalErrorOnElement(
588 member, 'unexpected kind: "${member.kind}"'); 320 member, 'unexpected kind: "${member.kind}"');
589 } 321 }
590 if (member.isInstanceMember()) emitExtraAccessors(member, builder); 322 if (member.isInstanceMember()) emitExtraAccessors(member, builder);
591 } 323 }
592 324
593 void addMemberMethod(FunctionElement member, ClassBuilder builder) { 325 void addMemberMethod(FunctionElement member, ClassBuilder builder) {
ahe 2013/12/04 09:55:53 FYI: this is a good place to start your review.
594 if (member.isAbstract) return; 326 if (member.isAbstract) return;
595 jsAst.Expression code = backend.generatedCode[member]; 327 jsAst.Expression code = backend.generatedCode[member];
596 if (code == null) return; 328 if (code == null) return;
597 String name = namer.getNameOfMember(member); 329 String name = namer.getNameOfMember(member);
598 if (backend.isInterceptedMethod(member)) { 330 task.interceptorEmitter.recordMangledNameOfMemberMethod(member, name);
599 task.interceptorEmitter.interceptorInvocationNames.add(name); 331 FunctionSignature parameters = member.computeSignature(compiler);
332 bool needsStubs = !parameters.optionalParameters.isEmpty;
333 bool canTearOff = false;
334 bool isClosure = false;
335 String tearOffName;
336 if (!member.isFunction() || member.isConstructor() || member.isAccessor()) {
337 canTearOff = false;
338 } else if (member.isInstanceMember()) {
339 if (member.getEnclosingClass().isClosure()) {
340 canTearOff = false;
341 isClosure = true;
342 } else {
343 // Careful with operators.
344 canTearOff = compiler.codegenWorld.hasInvokedGetter(member, compiler);
345 tearOffName = namer.getterName(member);
346 }
347 } else {
348 canTearOff =
349 compiler.codegenWorld.staticFunctionsNeedingGetter.contains(member);
350 tearOffName = namer.getStaticClosureName(member);
600 } 351 }
601 code = task.metadataEmitter.extendWithMetadata(member, code); 352
602 builder.addProperty(name, code); 353 bool canBeReflected = backend.isAccessibleByReflection(member);
603 String reflectionName = task.getReflectionName(member, name); 354 bool needStructuredInfo = canTearOff || canBeReflected || compiler.enabledFu nctionApply;
ngeoffray 2013/12/04 14:38:28 Line too long.
Johnni Winther 2013/12/05 11:49:56 Long line.
ahe 2013/12/06 15:57:54 Done.
ahe 2013/12/06 15:57:54 Done.
604 if (reflectionName != null) { 355 if (!needStructuredInfo) {
605 var reflectable = 356 builder.addProperty(name, code);
606 js(backend.isAccessibleByReflection(member) ? '1' : '0'); 357 if (needsStubs) {
607 builder.addProperty('+$reflectionName', reflectable); 358 addParameterStubs(
608 jsAst.Node defaultValues = 359 member,
609 task.metadataEmitter.reifyDefaultArguments(member); 360 (Selector selector, jsAst.Fun function) {
610 if (defaultValues != null) { 361 builder.addProperty(namer.invocationName(selector), function);
611 String unmangledName = member.name; 362 });
612 builder.addProperty('*$unmangledName', defaultValues);
613 } 363 }
364 return;
614 } 365 }
615 if (member.isInstanceMember()) { 366
616 // TODO(ahe): Where is this done for static/top-level methods? 367 if (canTearOff) {
617 FunctionSignature parameters = member.computeSignature(compiler); 368 assert(invariant(member, !member.isGenerativeConstructor()));
618 if (!parameters.optionalParameters.isEmpty) { 369 assert(invariant(member, !member.isGenerativeConstructorBody()));
619 addParameterStubs(member, builder.addProperty); 370 assert(invariant(member, !member.isConstructor()));
371 }
372
373 // This element is needed for reflection or needs additional stubs. So we
374 // need to retain additional information.
375
376 // The information is stored in an array with this format:
377 //
378 // 1. The JS function for this member.
379 // 2. First stub.
380 // 3. Name of first stub.
381 // ...
382 // M. Call name of this member.
383 // M+1. Call name of first stub.
384 // ...
385 // N. Getter name for tearOff.
386 // N+1. (Required parameter count << 1) + (member.isAccessor() ? 1 : 0).
387 // N+2. (Optional parameter count << 1) +
388 // (parameters.optionalParametersAreNamed ? 1 : 0).
389 // N+3. Index to function type in constant pool.
390 // N+4. First default argument.
391 // ...
392 // O. First parameter name (if needed for reflection or Function.apply).
393 // ...
394 // P. Unmangled name (if reflectable).
395 // P+1. First metadata (if reflectable).
396 // ...
397
398 List expressions = [];
399
400 Selector callSelector;
401 if (member.isFunction()) {
402 callSelector =
403 new Selector.fromElement(member, compiler).toCallSelector();
404 }
405
406 // On [requiredParameterCount], the lower bit is set if this method can be
407 // called reflectively.
408 int requiredParameterCount = parameters.requiredParameterCount << 1;
409 if (member.isAccessor()) requiredParameterCount++;
410
411 int optionalParameterCount = parameters.optionalParameterCount << 1;
412 if (parameters.optionalParametersAreNamed) optionalParameterCount++;
413
414 expressions.add(code);
415
416 String callSelectorString =
417 callSelector == null ? null : '"${namer.invocationName(callSelector)}"';
Johnni Winther 2013/12/05 11:49:56 Compute this when [callSelector] is computed (in l
ahe 2013/12/06 15:57:54 Done.
418 List tearOffInfo =
419 [new jsAst.LiteralString('$callSelectorString /* tearOffInfo */')];
Johnni Winther 2013/12/05 11:49:56 Remove /* tearOffInfo */ (at least from minified c
ahe 2013/12/06 15:57:54 Added TODO.
420
421 if (needsStubs || canTearOff) {
422 addParameterStubs(member, (Selector selector, jsAst.Fun function) {
423 expressions.add(function);
424 if (member.isInstanceMember()) {
425 Set invokedSelectors =
426 compiler.codegenWorld.invokedNames[member.name];
427 if (invokedSelectors != null && invokedSelectors.contains(selector)) {
428 expressions.add(js.string(namer.invocationName(selector)));
429 } else {
430 // Don't add a stub for calling this as a regular instance method,
431 // we only need the "call" stub for implicit closures of this
432 // method.
433 expressions.add("null");
434 }
435 } else {
436 // Static methods don't need "named" stubs as the default arguments
437 // are inlined at call sites. But static methods might need "call"
438 // stubs for implicit closures.
439 expressions.add("null");
440 // TOOD(ahe): Since we know when reading static data versus instance
441 // data, we can eliminate this element.
442 }
443 Set<Selector> callSelectors = compiler.codegenWorld.invokedNames[
444 namer.closureInvocationSelectorName];
445 Selector callSelector = selector.toCallSelector();
446 String callSelectorString = 'null';
447 if (canTearOff && callSelectors != null && callSelectors.contains(callSe lector)) {
Johnni Winther 2013/12/05 11:49:56 Long line.
ahe 2013/12/06 15:57:54 Done.
448 callSelectorString = '"${namer.invocationName(callSelector)}"';
449 }
450 tearOffInfo.add(new jsAst.LiteralString('$callSelectorString /* tearOffI nfo */'));
Johnni Winther 2013/12/05 11:49:56 Remove /* tearOffInfo */.
ahe 2013/12/06 15:57:54 TODO...
451 }, canTearOff);
452 }
453
454 jsAst.Expression memberTypeExpression;
455 if ((canTearOff || canBeReflected) && !member.isGenerativeConstructorBody()) {
Johnni Winther 2013/12/05 11:49:56 Long line.
ahe 2013/12/06 15:57:54 Done.
456 DartType memberType = member.computeType(compiler);
457 if (memberType.containsTypeVariables) {
458 jsAst.Expression thisAccess = js(r'this.$receiver');
459 memberTypeExpression =
460 backend.rti.getSignatureEncoding(memberType, thisAccess);
461 } else {
462 memberTypeExpression =
463 js.toExpression(task.metadataEmitter.reifyType(memberType));
620 } 464 }
465 } else {
466 memberTypeExpression = js('null');
621 } 467 }
468
469 expressions
470 ..addAll(tearOffInfo)
471 ..add((tearOffName == null || member.isAccessor())
472 ? js("null") : js.string(tearOffName))
473 ..add(requiredParameterCount)
474 ..add(optionalParameterCount)
475 ..add(memberTypeExpression)
476 ..addAll(task.metadataEmitter.reifyDefaultArguments(member));
477
478 if (canBeReflected || compiler.enabledFunctionApply) {
479 parameters.orderedForEachParameter((Element parameter) {
480 expressions.add(task.metadataEmitter.reifyName(parameter.name));
481 });
482 }
483 if (canBeReflected) {
484 jsAst.LiteralString reflectionName;
485 if (member.isConstructor()) {
486 String reflectionNameString = task.getReflectionName(member, name);
487 reflectionName =
488 new jsAst.LiteralString(
489 '"new ${Elements.reconstructConstructorName(member)}" /* $reflec tionNameString */');
Johnni Winther 2013/12/05 11:49:56 Long line. Remove /* $reflectionNameString */ (at
ahe 2013/12/06 15:57:54 Done.
490 } else {
491 reflectionName = js.string(member.name);
492 }
493 expressions
494 ..add(reflectionName)
495 ..addAll(task.metadataEmitter.computeMetadata(member));
496 } else if (isClosure && compiler.enabledFunctionApply) {
497 expressions.add(js.string(member.name));
498 }
499
500 builder.addProperty(name, js.toExpression(expressions));
622 } 501 }
623 502
624 void addMemberField(VariableElement member, ClassBuilder builder) { 503 void addMemberField(VariableElement member, ClassBuilder builder) {
625 // For now, do nothing. 504 // For now, do nothing.
626 } 505 }
627 } 506 }
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698