OLD | NEW |
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 #include "vm/code_generator.h" | 5 #include "vm/code_generator.h" |
6 | 6 |
7 #include "vm/assembler.h" | 7 #include "vm/assembler.h" |
8 #include "vm/ast.h" | 8 #include "vm/ast.h" |
9 #include "vm/code_patcher.h" | 9 #include "vm/code_patcher.h" |
10 #include "vm/compiler.h" | 10 #include "vm/compiler.h" |
(...skipping 719 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
730 | 730 |
731 // An instance call of the form o.f(...) could not be resolved. Check if | 731 // An instance call of the form o.f(...) could not be resolved. Check if |
732 // there is a getter with the same name. If so, invoke it. If the value is | 732 // there is a getter with the same name. If so, invoke it. If the value is |
733 // a closure, invoke it with the given arguments. If the value is a | 733 // a closure, invoke it with the given arguments. If the value is a |
734 // non-closure, attempt to invoke "call" on it. | 734 // non-closure, attempt to invoke "call" on it. |
735 static bool ResolveCallThroughGetter(const Instance& receiver, | 735 static bool ResolveCallThroughGetter(const Instance& receiver, |
736 const Class& receiver_class, | 736 const Class& receiver_class, |
737 const String& target_name, | 737 const String& target_name, |
738 const Array& arguments_descriptor, | 738 const Array& arguments_descriptor, |
739 Function* result) { | 739 Function* result) { |
740 ASSERT(FLAG_lazy_dispatchers); | |
741 // 1. Check if there is a getter with the same name. | 740 // 1. Check if there is a getter with the same name. |
742 const String& getter_name = String::Handle(Field::GetterName(target_name)); | 741 const String& getter_name = String::Handle(Field::GetterName(target_name)); |
743 const int kNumArguments = 1; | 742 const int kNumArguments = 1; |
744 ArgumentsDescriptor args_desc( | 743 ArgumentsDescriptor args_desc( |
745 Array::Handle(ArgumentsDescriptor::New(kNumArguments))); | 744 Array::Handle(ArgumentsDescriptor::New(kNumArguments))); |
746 const Function& getter = Function::Handle( | 745 const Function& getter = Function::Handle( |
747 Resolver::ResolveDynamicForReceiverClass(receiver_class, | 746 Resolver::ResolveDynamicForReceiverClass(receiver_class, |
748 getter_name, | 747 getter_name, |
749 args_desc)); | 748 args_desc)); |
750 if (getter.IsNull() || getter.IsMethodExtractor()) { | 749 if (getter.IsNull() || getter.IsMethodExtractor()) { |
751 return false; | 750 return false; |
752 } | 751 } |
753 const Class& cache_class = Class::Handle(receiver_class.IsSignatureClass() | 752 const Class& cache_class = Class::Handle(receiver_class.IsSignatureClass() |
754 ? receiver_class.SuperClass() | 753 ? receiver_class.SuperClass() |
755 : receiver_class.raw()); | 754 : receiver_class.raw()); |
756 ASSERT( | 755 ASSERT( |
757 !receiver_class.IsSignatureClass() || | 756 !receiver_class.IsSignatureClass() || |
758 (receiver_class.SuperClass() == Type::Handle( | 757 (receiver_class.SuperClass() == Type::Handle( |
759 Isolate::Current()->object_store()->function_impl_type()).type_class())); | 758 Isolate::Current()->object_store()->function_impl_type()).type_class())); |
760 const Function& target_function = | 759 const Function& target_function = |
761 Function::Handle(cache_class.GetInvocationDispatcher( | 760 Function::Handle(cache_class.GetInvocationDispatcher( |
762 target_name, | 761 target_name, |
763 arguments_descriptor, | 762 arguments_descriptor, |
764 RawFunction::kInvokeFieldDispatcher)); | 763 RawFunction::kInvokeFieldDispatcher, |
765 ASSERT(!target_function.IsNull()); | 764 FLAG_lazy_dispatchers)); |
| 765 ASSERT(!target_function.IsNull() || !FLAG_lazy_dispatchers); |
766 if (FLAG_trace_ic) { | 766 if (FLAG_trace_ic) { |
767 OS::PrintErr("InvokeField IC miss: adding <%s> id:%" Pd " -> <%s>\n", | 767 OS::PrintErr("InvokeField IC miss: adding <%s> id:%" Pd " -> <%s>\n", |
768 Class::Handle(receiver.clazz()).ToCString(), | 768 Class::Handle(receiver.clazz()).ToCString(), |
769 receiver.GetClassId(), | 769 receiver.GetClassId(), |
770 target_function.ToCString()); | 770 target_function.ToCString()); |
771 } | 771 } |
772 *result = target_function.raw(); | 772 *result = target_function.raw(); |
773 return true; | 773 return true; |
774 } | 774 } |
775 | 775 |
776 | 776 |
777 // Handle other invocations (implicit closures, noSuchMethod). | 777 // Handle other invocations (implicit closures, noSuchMethod). |
778 RawFunction* InlineCacheMissHelper( | 778 RawFunction* InlineCacheMissHelper( |
779 const Instance& receiver, | 779 const Instance& receiver, |
780 const ICData& ic_data) { | 780 const ICData& ic_data) { |
781 if (!FLAG_lazy_dispatchers) { | |
782 return Function::null(); // We'll handle it in the runtime. | |
783 } | |
784 | |
785 const Array& args_descriptor = Array::Handle(ic_data.arguments_descriptor()); | 781 const Array& args_descriptor = Array::Handle(ic_data.arguments_descriptor()); |
786 | 782 |
787 const Class& receiver_class = Class::Handle(receiver.clazz()); | 783 const Class& receiver_class = Class::Handle(receiver.clazz()); |
788 const String& target_name = String::Handle(ic_data.target_name()); | 784 const String& target_name = String::Handle(ic_data.target_name()); |
789 | 785 |
790 Function& result = Function::Handle(); | 786 Function& result = Function::Handle(); |
791 if (!ResolveCallThroughGetter(receiver, | 787 if (!ResolveCallThroughGetter(receiver, |
792 receiver_class, | 788 receiver_class, |
793 target_name, | 789 target_name, |
794 args_descriptor, | 790 args_descriptor, |
795 &result)) { | 791 &result)) { |
796 ArgumentsDescriptor desc(args_descriptor); | 792 ArgumentsDescriptor desc(args_descriptor); |
797 const Function& target_function = | 793 const Function& target_function = |
798 Function::Handle(receiver_class.GetInvocationDispatcher( | 794 Function::Handle(receiver_class.GetInvocationDispatcher( |
799 target_name, | 795 target_name, |
800 args_descriptor, | 796 args_descriptor, |
801 RawFunction::kNoSuchMethodDispatcher)); | 797 RawFunction::kNoSuchMethodDispatcher, |
| 798 FLAG_lazy_dispatchers)); |
802 if (FLAG_trace_ic) { | 799 if (FLAG_trace_ic) { |
803 OS::PrintErr("NoSuchMethod IC miss: adding <%s> id:%" Pd " -> <%s>\n", | 800 OS::PrintErr("NoSuchMethod IC miss: adding <%s> id:%" Pd " -> <%s>\n", |
804 Class::Handle(receiver.clazz()).ToCString(), | 801 Class::Handle(receiver.clazz()).ToCString(), |
805 receiver.GetClassId(), | 802 receiver.GetClassId(), |
806 target_function.ToCString()); | 803 target_function.ToCString()); |
807 } | 804 } |
808 result = target_function.raw(); | 805 result = target_function.raw(); |
809 } | 806 } |
| 807 // May be null if --no-lazy-dispatchers, in which case dispatch will be |
| 808 // handled by InvokeNoSuchMethodDispatcher. |
| 809 ASSERT(!result.IsNull() || !FLAG_lazy_dispatchers); |
810 return result.raw(); | 810 return result.raw(); |
811 } | 811 } |
812 | 812 |
813 static RawFunction* InlineCacheMissHandler( | 813 static RawFunction* InlineCacheMissHandler( |
814 const GrowableArray<const Instance*>& args, | 814 const GrowableArray<const Instance*>& args, |
815 const ICData& ic_data) { | 815 const ICData& ic_data) { |
816 const Instance& receiver = *args[0]; | 816 const Instance& receiver = *args[0]; |
817 ArgumentsDescriptor | 817 ArgumentsDescriptor |
818 arguments_descriptor(Array::Handle(ic_data.arguments_descriptor())); | 818 arguments_descriptor(Array::Handle(ic_data.arguments_descriptor())); |
819 String& function_name = String::Handle(ic_data.target_name()); | 819 String& function_name = String::Handle(ic_data.target_name()); |
(...skipping 1005 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1825 const intptr_t elm_size = old_data.ElementSizeInBytes(); | 1825 const intptr_t elm_size = old_data.ElementSizeInBytes(); |
1826 const TypedData& new_data = | 1826 const TypedData& new_data = |
1827 TypedData::Handle(TypedData::New(cid, new_size, Heap::kOld)); | 1827 TypedData::Handle(TypedData::New(cid, new_size, Heap::kOld)); |
1828 TypedData::Copy(new_data, 0, old_data, 0, old_size * elm_size); | 1828 TypedData::Copy(new_data, 0, old_data, 0, old_size * elm_size); |
1829 typed_data_cell.SetAt(0, new_data); | 1829 typed_data_cell.SetAt(0, new_data); |
1830 arguments.SetReturn(new_data); | 1830 arguments.SetReturn(new_data); |
1831 } | 1831 } |
1832 | 1832 |
1833 | 1833 |
1834 } // namespace dart | 1834 } // namespace dart |
OLD | NEW |