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 #include "vm/dart_entry.h" | 5 #include "vm/dart_entry.h" |
6 | 6 |
7 #include "vm/class_finalizer.h" | 7 #include "vm/class_finalizer.h" |
8 #include "vm/code_generator.h" | 8 #include "vm/code_generator.h" |
9 #include "vm/compiler.h" | 9 #include "vm/compiler.h" |
10 #include "vm/debugger.h" | 10 #include "vm/debugger.h" |
(...skipping 133 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
144 // Only invoke the function if its arguments are compatible. | 144 // Only invoke the function if its arguments are compatible. |
145 const ArgumentsDescriptor args_desc(arguments_descriptor); | 145 const ArgumentsDescriptor args_desc(arguments_descriptor); |
146 if (function.AreValidArgumentCounts(args_desc.Count(), | 146 if (function.AreValidArgumentCounts(args_desc.Count(), |
147 args_desc.NamedCount(), | 147 args_desc.NamedCount(), |
148 NULL)) { | 148 NULL)) { |
149 // The closure or non-closure object (receiver) is passed as implicit | 149 // The closure or non-closure object (receiver) is passed as implicit |
150 // first argument. It is already included in the arguments array. | 150 // first argument. It is already included in the arguments array. |
151 return InvokeFunction(function, arguments, arguments_descriptor); | 151 return InvokeFunction(function, arguments, arguments_descriptor); |
152 } | 152 } |
153 } | 153 } |
154 // There is no compatible 'call' method, so invoke noSuchMethod. | 154 |
155 // There is no compatible 'call' method, see if there's a getter. | |
156 if (instance.IsClosure()) { | |
157 // Special case: closures are implemented with a call getter instead of a | |
158 // call method. If the arguments didn't match, go to noSuchMethod instead | |
159 // of infinitely recursing on the getter. | |
160 } else { | |
Florian Schneider
2015/06/29 10:37:53
When is InvokeClosure used with a non-closure obje
rmacnak
2015/06/29 21:54:19
From the runtime stub, e.g.
class FakeFunction {
| |
161 const String& getter_name = String::Handle(Symbols::New("get:call")); | |
162 Class& cls = Class::Handle(instance.clazz()); | |
163 while (!cls.IsNull()) { | |
164 function ^= cls.LookupDynamicFunction(getter_name); | |
165 if (!function.IsNull()) { | |
166 // Getters don't have a stack overflow check, so do one in C++. | |
167 | |
168 Isolate* isolate = Isolate::Current(); | |
169 #if defined(USING_SIMULATOR) | |
170 uword stack_pos = Simulator::Current()->get_register(SPREG); | |
171 #else | |
172 uword stack_pos = Isolate::GetCurrentStackPointer(); | |
173 #endif | |
174 if (stack_pos < isolate->saved_stack_limit()) { | |
175 const Instance& exception = | |
176 Instance::Handle(isolate->object_store()->stack_overflow()); | |
177 return UnhandledException::New(exception, Stacktrace::Handle()); | |
178 } | |
179 | |
180 const Array& getter_arguments = Array::Handle(Array::New(1)); | |
181 getter_arguments.SetAt(0, instance); | |
182 const Object& getter_result = | |
183 Object::Handle(DartEntry::InvokeFunction(function, | |
184 getter_arguments)); | |
185 if (getter_result.IsError()) { | |
186 return getter_result.raw(); | |
187 } | |
188 ASSERT(getter_result.IsNull() || getter_result.IsInstance()); | |
189 | |
190 arguments.SetAt(0, getter_result); // Safe to reuse this? | |
Florian Schneider
2015/06/30 10:28:35
Why shouldn't it be safe?
rmacnak
2015/06/30 22:13:49
I didn't initially notice a new one was allocated
| |
191 return InvokeClosure(arguments, arguments_descriptor); | |
192 } | |
193 cls = cls.SuperClass(); | |
194 } | |
195 } | |
196 | |
197 // No compataible method or getter so invoke noSuchMethod. | |
Florian Schneider
2015/06/30 10:28:35
s/compataible/compatible/
rmacnak
2015/06/30 22:13:49
Done.
| |
155 return InvokeNoSuchMethod(instance, | 198 return InvokeNoSuchMethod(instance, |
156 Symbols::Call(), | 199 Symbols::Call(), |
157 arguments, | 200 arguments, |
158 arguments_descriptor); | 201 arguments_descriptor); |
159 } | 202 } |
160 | 203 |
161 | 204 |
162 RawObject* DartEntry::InvokeNoSuchMethod(const Instance& receiver, | 205 RawObject* DartEntry::InvokeNoSuchMethod(const Instance& receiver, |
163 const String& target_name, | 206 const String& target_name, |
164 const Array& arguments, | 207 const Array& arguments, |
(...skipping 364 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
529 const Array& args = Array::Handle(Array::New(kNumArguments)); | 572 const Array& args = Array::Handle(Array::New(kNumArguments)); |
530 args.SetAt(0, map); | 573 args.SetAt(0, map); |
531 args.SetAt(1, key); | 574 args.SetAt(1, key); |
532 args.SetAt(2, value); | 575 args.SetAt(2, value); |
533 const Object& result = Object::Handle(DartEntry::InvokeFunction(function, | 576 const Object& result = Object::Handle(DartEntry::InvokeFunction(function, |
534 args)); | 577 args)); |
535 return result.raw(); | 578 return result.raw(); |
536 } | 579 } |
537 | 580 |
538 } // namespace dart | 581 } // namespace dart |
OLD | NEW |