OLD | NEW |
1 // Copyright (c) 2011, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2011, 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 /** A dynamic member stub. */ | 5 /** A dynamic member stub. */ |
6 class VarMember { | 6 class VarMember { |
7 final String name; | 7 final String name; |
8 | 8 |
9 VarMember(this.name); | 9 VarMember(this.name); |
10 | 10 |
(...skipping 106 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
117 VarMethodStub(String name, this.member, this.args, this.body): super(name); | 117 VarMethodStub(String name, this.member, this.args, this.body): super(name); |
118 | 118 |
119 Type get returnType() => | 119 Type get returnType() => |
120 member != null ? member.returnType : world.varType; | 120 member != null ? member.returnType : world.varType; |
121 | 121 |
122 String get typeName() => | 122 String get typeName() => |
123 member != null ? member.declaringType.jsname : 'Object'; | 123 member != null ? member.declaringType.jsname : 'Object'; |
124 | 124 |
125 void generate(CodeWriter code) { | 125 void generate(CodeWriter code) { |
126 code.write('$typeName.prototype.$name = '); | 126 code.write('$typeName.prototype.$name = '); |
127 generateBody(code); | 127 generateBody(code, ';'); |
128 code.writeln(';'); | |
129 } | 128 } |
130 | 129 |
131 void generateBody(CodeWriter code) { | 130 void generateBody(CodeWriter code, String end) { |
132 if (_useDirectCall(member, args)) { | 131 if (_useDirectCall(member, args)) { |
133 code.write('$typeName.prototype.${member.jsname}'); | 132 code.writeln('$typeName.prototype.${member.jsname}$end'); |
134 } else { | 133 } else { |
135 code.enterBlock('function(${args.getCode()}) {'); | 134 code.enterBlock('function(${args.getCode()}) {'); |
136 code.writeln('return ${body.code};'); | 135 code.writeln('return ${body.code};'); |
137 code.exitBlock('}'); | 136 code.exitBlock('}$end'); |
138 } | 137 } |
139 } | 138 } |
140 | 139 |
141 bool _useDirectCall(Member member, Arguments args) { | 140 bool _useDirectCall(Member member, Arguments args) { |
142 // TODO(jmesserly): for now disallow direct references to DOM types until we | 141 // TODO(jmesserly): for now disallow direct references to DOM types until we |
143 // figure out which types can be patched reliably. | 142 // figure out which types can be patched reliably. |
144 // I don't think our other native libs have this issue. | 143 // I don't think our other native libs have this issue. |
145 if (member is MethodMember && member.declaringType.library != world.dom) { | 144 if (member is MethodMember && member.declaringType.library != world.dom) { |
146 MethodMember method = member; | 145 MethodMember method = member; |
147 if (method.needsArgumentConversion(args)) { | 146 if (method.needsArgumentConversion(args)) { |
(...skipping 37 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
185 | 184 |
186 Value invoke(MethodGenerator context, Node node, Value target, Arguments args)
{ | 185 Value invoke(MethodGenerator context, Node node, Value target, Arguments args)
{ |
187 _invokeMembers(context, node); | 186 _invokeMembers(context, node); |
188 return super.invoke(context, node, target, args); | 187 return super.invoke(context, node, target, args); |
189 } | 188 } |
190 | 189 |
191 /** Invokes members to ensure they're generated. */ | 190 /** Invokes members to ensure they're generated. */ |
192 _invokeMembers(MethodGenerator context, Node node) { | 191 _invokeMembers(MethodGenerator context, Node node) { |
193 if (_fallbackStubs != null) return; | 192 if (_fallbackStubs != null) return; |
194 | 193 |
| 194 var objectStub = null; |
195 _fallbackStubs = []; | 195 _fallbackStubs = []; |
196 for (var member in members) { | 196 for (var member in members) { |
197 // Invoke the member with the stub args (this gives us the method body), | 197 // Invoke the member with the stub args (this gives us the method body), |
198 // then create the stub method. | 198 // then create the stub method. |
199 final target = new Value(member.declaringType, 'this', node.span); | 199 final target = new Value(member.declaringType, 'this', node.span); |
200 var result = member.invoke(context, node, target, args); | 200 var result = member.invoke(context, node, target, args); |
201 var stub = new VarMethodStub(name, member, args, result); | 201 var stub = new VarMethodStub(name, member, args, result); |
202 | 202 |
203 // Put the stub on the type directly if possible. Otherwise | 203 // Put the stub on the type directly if possible. Otherwise |
204 // put the stub on Object.prototype. | 204 // put the stub on Object.prototype. |
205 var type = member.declaringType; | 205 var type = member.declaringType; |
206 if (type.library != world.dom && !type.isObject) { | 206 if (type.isObject) { |
| 207 objectStub = stub; |
| 208 } else if (type.library != world.dom) { |
207 _addVarStub(type, stub); | 209 _addVarStub(type, stub); |
208 } else { | 210 } else { |
209 _fallbackStubs.add(stub); | 211 _fallbackStubs.add(stub); |
210 } | 212 } |
211 } | 213 } |
212 | 214 |
213 // Finally, invoke noSuchMethod | 215 // Create a noSuchMethod fallback on Object if needed. |
214 final target = new Value(world.objectType, 'this', node.span); | 216 // Some methods, like toString and == already have a fallback on Object. |
215 var result = target.invokeNoSuchMethod(context, baseName, node, args); | 217 if (objectStub == null) { |
216 var stub = new VarMethodStub(name, null, args, result); | 218 final target = new Value(world.objectType, 'this', node.span); |
| 219 var result = target.invokeNoSuchMethod(context, baseName, node, args); |
| 220 objectStub = new VarMethodStub(name, null, args, result); |
| 221 } |
217 if (_fallbackStubs.length == 0) { | 222 if (_fallbackStubs.length == 0) { |
218 _addVarStub(world.objectType, stub); | 223 _addVarStub(world.objectType, objectStub); |
219 } else { | 224 } else { |
220 _fallbackStubs.add(stub); | 225 _fallbackStubs.add(objectStub); |
221 world.gen.corejs.useVarMethod = true; | 226 world.gen.corejs.useVarMethod = true; |
222 } | 227 } |
223 } | 228 } |
224 | 229 |
225 static _addVarStub(Type type, VarMember stub) { | 230 static _addVarStub(Type type, VarMember stub) { |
226 if (type.varStubs == null) type.varStubs = {}; | 231 if (type.varStubs == null) type.varStubs = {}; |
227 type.varStubs[stub.name] = stub; | 232 type.varStubs[stub.name] = stub; |
228 } | 233 } |
229 | 234 |
230 /** | 235 /** |
231 * Generate var call fallbacks, like this: | 236 * Generate var call fallbacks, like this: |
232 * | 237 * |
233 * $varMethod('addEventListener$1$capture', { | 238 * $varMethod('addEventListener$1$capture', { |
234 * 'HTMLElement': function($0, capture) { | 239 * 'HTMLElement': function($0, capture) { |
235 * return this.addEventListener($0, capture); | 240 * return this.addEventListener($0, capture); |
236 * }, | 241 * }, |
237 * 'SomeOtherDOMType': function($0, capture) { | 242 * 'SomeOtherDOMType': function($0, capture) { |
238 * return this.addEventListener($0, false, true, capture); | 243 * return this.addEventListener($0, false, true, capture); |
239 * }, | 244 * }, |
240 * 'Object': function($0, capture) { | 245 * 'Object': function($0, capture) { |
241 * return this.noSuchMethod('addEventListener', [$0], | 246 * return this.noSuchMethod('addEventListener', [$0], |
242 * {'capture': capture}); | 247 * {'capture': capture}); |
243 * } | 248 * } |
244 * }); | 249 * }); |
245 */ | 250 */ |
246 void generate(CodeWriter code) { | 251 void generate(CodeWriter code) { |
247 if (_fallbackStubs.length == 0) return; | 252 if (_fallbackStubs.length == 0) return; |
248 | 253 |
249 code.enterBlock('\$varMethod("$name", {'); | 254 code.enterBlock('\$varMethod("$name", {'); |
250 var lastOne = _fallbackStubs[_fallbackStubs.length - 1]; | 255 var lastOne = _fallbackStubs.last(); |
251 for (var stub in _fallbackStubs) { | 256 for (var stub in _fallbackStubs) { |
252 code.write('"${stub.typeName}": '); | 257 code.write('"${stub.typeName}": '); |
253 stub.generateBody(code); | 258 stub.generateBody(code, stub == lastOne ? '' : ','); |
254 code.writeln(stub == lastOne ? '' : ','); | |
255 } | 259 } |
256 code.exitBlock('});'); | 260 code.exitBlock('});'); |
257 } | 261 } |
258 } | 262 } |
259 | 263 |
260 String _getCallStubName(String name, Arguments args) { | 264 String _getCallStubName(String name, Arguments args) { |
261 final nameBuilder = new StringBuffer('${name}\$${args.bareCount}'); | 265 final nameBuilder = new StringBuffer('${name}\$${args.bareCount}'); |
262 for (int i = args.bareCount; i < args.length; i++) { | 266 for (int i = args.bareCount; i < args.length; i++) { |
263 nameBuilder.add('\$').add(args.getName(i)); | 267 nameBuilder.add('\$').add(args.getName(i)); |
264 } | 268 } |
265 return nameBuilder.toString(); | 269 return nameBuilder.toString(); |
266 } | 270 } |
OLD | NEW |