OLD | NEW |
1 // Copyright (c) 2015, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2015, 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/precompiler.h" | 5 #include "vm/precompiler.h" |
6 | 6 |
| 7 #include "vm/code_patcher.h" |
7 #include "vm/compiler.h" | 8 #include "vm/compiler.h" |
8 #include "vm/isolate.h" | 9 #include "vm/isolate.h" |
9 #include "vm/log.h" | 10 #include "vm/log.h" |
10 #include "vm/longjump.h" | 11 #include "vm/longjump.h" |
11 #include "vm/object.h" | 12 #include "vm/object.h" |
12 #include "vm/object_store.h" | 13 #include "vm/object_store.h" |
13 #include "vm/resolver.h" | 14 #include "vm/resolver.h" |
14 #include "vm/symbols.h" | 15 #include "vm/symbols.h" |
15 | 16 |
16 namespace dart { | 17 namespace dart { |
(...skipping 304 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
321 } | 322 } |
322 } | 323 } |
323 | 324 |
324 | 325 |
325 void Precompiler::CleanUp() { | 326 void Precompiler::CleanUp() { |
326 I->set_collected_closures(GrowableObjectArray::Handle(Z)); | 327 I->set_collected_closures(GrowableObjectArray::Handle(Z)); |
327 | 328 |
328 DropUncompiledFunctions(); | 329 DropUncompiledFunctions(); |
329 | 330 |
330 // TODO(rmacnak): DropEmptyClasses(); | 331 // TODO(rmacnak): DropEmptyClasses(); |
| 332 |
| 333 BindStaticCalls(); |
331 } | 334 } |
332 | 335 |
333 | 336 |
334 void Precompiler::ProcessFunction(const Function& function) { | 337 void Precompiler::ProcessFunction(const Function& function) { |
335 if (!function.HasCode()) { | 338 if (!function.HasCode()) { |
336 function_count_++; | 339 function_count_++; |
337 | 340 |
338 if (FLAG_trace_precompiler) { | 341 if (FLAG_trace_precompiler) { |
339 THR_Print("Precompiling %" Pd " %s (%" Pd ", %s)\n", | 342 THR_Print("Precompiling %" Pd " %s (%" Pd ", %s)\n", |
340 function_count_, | 343 function_count_, |
(...skipping 343 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
684 if (!closures.IsNull()) { | 687 if (!closures.IsNull()) { |
685 for (intptr_t j = 0; j < closures.Length(); j++) { | 688 for (intptr_t j = 0; j < closures.Length(); j++) { |
686 function ^= closures.At(j); | 689 function ^= closures.At(j); |
687 ASSERT(function.HasCode()); | 690 ASSERT(function.HasCode()); |
688 } | 691 } |
689 } | 692 } |
690 } | 693 } |
691 } | 694 } |
692 } | 695 } |
693 | 696 |
| 697 |
| 698 void Precompiler::BindStaticCalls() { |
| 699 Library& lib = Library::Handle(Z); |
| 700 Class& cls = Class::Handle(Z); |
| 701 Array& functions = Array::Handle(Z); |
| 702 Function& function = Function::Handle(Z); |
| 703 GrowableObjectArray& closures = GrowableObjectArray::Handle(Z); |
| 704 |
| 705 for (intptr_t i = 0; i < libraries_.Length(); i++) { |
| 706 lib ^= libraries_.At(i); |
| 707 ClassDictionaryIterator it(lib, ClassDictionaryIterator::kIteratePrivate); |
| 708 while (it.HasNext()) { |
| 709 cls = it.GetNextClass(); |
| 710 if (cls.IsDynamicClass()) { |
| 711 continue; // class 'dynamic' is in the read-only VM isolate. |
| 712 } |
| 713 |
| 714 functions = cls.functions(); |
| 715 for (intptr_t j = 0; j < functions.Length(); j++) { |
| 716 function ^= functions.At(j); |
| 717 BindStaticCalls(function); |
| 718 } |
| 719 |
| 720 closures = cls.closures(); |
| 721 if (!closures.IsNull()) { |
| 722 for (intptr_t j = 0; j < closures.Length(); j++) { |
| 723 function ^= closures.At(j); |
| 724 BindStaticCalls(function); |
| 725 } |
| 726 } |
| 727 } |
| 728 } |
| 729 } |
| 730 |
| 731 |
| 732 void Precompiler::BindStaticCalls(const Function& function) { |
| 733 ASSERT(function.HasCode()); |
| 734 |
| 735 const Code& code = Code::Handle(Z, function.CurrentCode()); |
| 736 |
| 737 const Array& table = Array::Handle(Z, code.static_calls_target_table()); |
| 738 Smi& pc_offset = Smi::Handle(Z); |
| 739 Function& target = Function::Handle(Z); |
| 740 Code& target_code = Code::Handle(Z); |
| 741 |
| 742 for (intptr_t i = 0; i < table.Length(); i += Code::kSCallTableEntryLength) { |
| 743 pc_offset ^= table.At(i + Code::kSCallTableOffsetEntry); |
| 744 target ^= table.At(i + Code::kSCallTableFunctionEntry); |
| 745 if (target.IsNull()) { |
| 746 target_code ^= table.At(i + Code::kSCallTableCodeEntry); |
| 747 ASSERT(!target_code.IsNull()); |
| 748 ASSERT(!target_code.IsFunctionCode()); |
| 749 // Allocation stub or AllocateContext or AllocateArray or ... |
| 750 } else { |
| 751 // Cf. runtime entry PatchStaticCall called from CallStaticFunction stub. |
| 752 ASSERT(target.HasCode()); |
| 753 target_code ^= target.CurrentCode(); |
| 754 CodePatcher::PatchStaticCallAt(pc_offset.Value() + code.EntryPoint(), |
| 755 code, target_code); |
| 756 } |
| 757 } |
| 758 code.set_static_calls_target_table(Object::empty_array()); |
| 759 } |
| 760 |
694 } // namespace dart | 761 } // namespace dart |
OLD | NEW |