Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright 2013 the V8 project authors. All rights reserved. | 1 // Copyright 2013 the V8 project authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "src/v8.h" | 5 #include "src/v8.h" |
| 6 | 6 |
| 7 #include "src/accessors.h" | 7 #include "src/accessors.h" |
| 8 #include "src/codegen.h" | 8 #include "src/codegen.h" |
| 9 #include "src/cpu-profiler.h" | 9 #include "src/cpu-profiler.h" |
| 10 #include "src/deoptimizer.h" | 10 #include "src/deoptimizer.h" |
| (...skipping 176 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 187 frame_index >= construct_offset && | 187 frame_index >= construct_offset && |
| 188 deoptimizer->output_[frame_index - construct_offset]->GetFrameType() == | 188 deoptimizer->output_[frame_index - construct_offset]->GetFrameType() == |
| 189 StackFrame::CONSTRUCT; | 189 StackFrame::CONSTRUCT; |
| 190 | 190 |
| 191 DeoptimizedFrameInfo* info = new DeoptimizedFrameInfo(deoptimizer, | 191 DeoptimizedFrameInfo* info = new DeoptimizedFrameInfo(deoptimizer, |
| 192 frame_index, | 192 frame_index, |
| 193 has_arguments_adaptor, | 193 has_arguments_adaptor, |
| 194 has_construct_stub); | 194 has_construct_stub); |
| 195 isolate->deoptimizer_data()->deoptimized_frame_info_ = info; | 195 isolate->deoptimizer_data()->deoptimized_frame_info_ = info; |
| 196 | 196 |
| 197 // Get the "simulated" top and size for the requested frame. | |
| 198 FrameDescription* parameters_frame = | |
| 199 deoptimizer->output_[ | |
| 200 has_arguments_adaptor ? (frame_index - 1) : frame_index]; | |
| 201 | |
| 202 uint32_t parameters_size = (info->parameters_count() + 1) * kPointerSize; | |
| 203 Address parameters_top = reinterpret_cast<Address>( | |
| 204 parameters_frame->GetTop() + (parameters_frame->GetFrameSize() - | |
| 205 parameters_size)); | |
| 206 | |
| 207 uint32_t expressions_size = info->expression_count() * kPointerSize; | |
| 208 Address expressions_top = reinterpret_cast<Address>( | |
| 209 deoptimizer->output_[frame_index]->GetTop()); | |
| 210 | |
| 211 // Done with the GC-unsafe frame descriptions. This re-enables allocation. | 197 // Done with the GC-unsafe frame descriptions. This re-enables allocation. |
| 212 deoptimizer->DeleteFrameDescriptions(); | 198 deoptimizer->DeleteFrameDescriptions(); |
| 213 | 199 |
| 214 // Allocate a heap number for the doubles belonging to this frame. | 200 // Allocate a heap number for the doubles belonging to this frame. |
| 215 deoptimizer->MaterializeHeapNumbersForDebuggerInspectableFrame( | 201 deoptimizer->MaterializeHeapNumbersForDebuggerInspectableFrame( |
| 216 parameters_top, parameters_size, expressions_top, expressions_size, info); | 202 frame_index, info->parameters_count(), info->expression_count(), info); |
| 217 | 203 |
| 218 // Finished using the deoptimizer instance. | 204 // Finished using the deoptimizer instance. |
| 219 delete deoptimizer; | 205 delete deoptimizer; |
| 220 | 206 |
| 221 return info; | 207 return info; |
| 222 } | 208 } |
| 223 | 209 |
| 224 | 210 |
| 225 void Deoptimizer::DeleteDebuggerInspectableFrame(DeoptimizedFrameInfo* info, | 211 void Deoptimizer::DeleteDebuggerInspectableFrame(DeoptimizedFrameInfo* info, |
| 226 Isolate* isolate) { | 212 Isolate* isolate) { |
| (...skipping 312 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 539 case EAGER: return "eager"; | 525 case EAGER: return "eager"; |
| 540 case SOFT: return "soft"; | 526 case SOFT: return "soft"; |
| 541 case LAZY: return "lazy"; | 527 case LAZY: return "lazy"; |
| 542 case DEBUGGER: return "debugger"; | 528 case DEBUGGER: return "debugger"; |
| 543 } | 529 } |
| 544 FATAL("Unsupported deopt type"); | 530 FATAL("Unsupported deopt type"); |
| 545 return NULL; | 531 return NULL; |
| 546 } | 532 } |
| 547 | 533 |
| 548 | 534 |
| 549 Deoptimizer::Deoptimizer(Isolate* isolate, | 535 Deoptimizer::Deoptimizer(Isolate* isolate, JSFunction* function, |
| 550 JSFunction* function, | 536 BailoutType type, unsigned bailout_id, Address from, |
| 551 BailoutType type, | 537 int fp_to_sp_delta, Code* optimized_code) |
| 552 unsigned bailout_id, | |
| 553 Address from, | |
| 554 int fp_to_sp_delta, | |
| 555 Code* optimized_code) | |
| 556 : isolate_(isolate), | 538 : isolate_(isolate), |
| 557 function_(function), | 539 function_(function), |
| 558 bailout_id_(bailout_id), | 540 bailout_id_(bailout_id), |
| 559 bailout_type_(type), | 541 bailout_type_(type), |
| 560 from_(from), | 542 from_(from), |
| 561 fp_to_sp_delta_(fp_to_sp_delta), | 543 fp_to_sp_delta_(fp_to_sp_delta), |
| 562 has_alignment_padding_(0), | 544 has_alignment_padding_(0), |
| 563 input_(NULL), | 545 input_(nullptr), |
| 564 output_count_(0), | 546 output_count_(0), |
| 565 jsframe_count_(0), | 547 jsframe_count_(0), |
| 566 output_(NULL), | 548 output_(nullptr), |
| 567 deferred_objects_tagged_values_(0), | |
| 568 deferred_objects_double_values_(0), | |
| 569 deferred_objects_(0), | |
| 570 deferred_heap_numbers_(0), | |
| 571 jsframe_functions_(0), | 549 jsframe_functions_(0), |
| 572 jsframe_has_adapted_arguments_(0), | 550 jsframe_has_adapted_arguments_(0), |
| 573 materialized_values_(NULL), | 551 trace_scope_(nullptr) { |
| 574 materialized_objects_(NULL), | |
| 575 materialization_value_index_(0), | |
| 576 materialization_object_index_(0), | |
| 577 trace_scope_(NULL) { | |
| 578 // For COMPILED_STUBs called from builtins, the function pointer is a SMI | 552 // For COMPILED_STUBs called from builtins, the function pointer is a SMI |
| 579 // indicating an internal frame. | 553 // indicating an internal frame. |
| 580 if (function->IsSmi()) { | 554 if (function->IsSmi()) { |
| 581 function = NULL; | 555 function = nullptr; |
| 582 } | 556 } |
| 583 DCHECK(from != NULL); | 557 DCHECK(from != nullptr); |
| 584 if (function != NULL && function->IsOptimized()) { | 558 if (function != nullptr && function->IsOptimized()) { |
| 585 function->shared()->increment_deopt_count(); | 559 function->shared()->increment_deopt_count(); |
| 586 if (bailout_type_ == Deoptimizer::SOFT) { | 560 if (bailout_type_ == Deoptimizer::SOFT) { |
| 587 isolate->counters()->soft_deopts_executed()->Increment(); | 561 isolate->counters()->soft_deopts_executed()->Increment(); |
| 588 // Soft deopts shouldn't count against the overall re-optimization count | 562 // Soft deopts shouldn't count against the overall re-optimization count |
| 589 // that can eventually lead to disabling optimization for a function. | 563 // that can eventually lead to disabling optimization for a function. |
| 590 int opt_count = function->shared()->opt_count(); | 564 int opt_count = function->shared()->opt_count(); |
| 591 if (opt_count > 0) opt_count--; | 565 if (opt_count > 0) opt_count--; |
| 592 function->shared()->set_opt_count(opt_count); | 566 function->shared()->set_opt_count(opt_count); |
| 593 } | 567 } |
| 594 } | 568 } |
| (...skipping 178 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 773 (compiled_code_->is_hydrogen_stub())) { | 747 (compiled_code_->is_hydrogen_stub())) { |
| 774 compiled_code_->PrintDeoptLocation(trace_scope_->file(), from_); | 748 compiled_code_->PrintDeoptLocation(trace_scope_->file(), from_); |
| 775 } | 749 } |
| 776 } | 750 } |
| 777 | 751 |
| 778 BailoutId node_id = input_data->AstId(bailout_id_); | 752 BailoutId node_id = input_data->AstId(bailout_id_); |
| 779 ByteArray* translations = input_data->TranslationByteArray(); | 753 ByteArray* translations = input_data->TranslationByteArray(); |
| 780 unsigned translation_index = | 754 unsigned translation_index = |
| 781 input_data->TranslationIndex(bailout_id_)->value(); | 755 input_data->TranslationIndex(bailout_id_)->value(); |
| 782 | 756 |
| 757 TranslationIterator state_iterator(translations, translation_index); | |
| 758 translated_state_.Init( | |
| 759 input_->GetFramePointerAddress(), function_, &state_iterator, | |
| 760 input_data->LiteralArray(), input_->GetRegisterValues(), | |
| 761 trace_scope_ == nullptr ? nullptr : trace_scope_->file()); | |
| 762 | |
| 783 // Do the input frame to output frame(s) translation. | 763 // Do the input frame to output frame(s) translation. |
| 784 TranslationIterator iterator(translations, translation_index); | 764 size_t count = translated_state_.frames().size(); |
| 785 Translation::Opcode opcode = | |
| 786 static_cast<Translation::Opcode>(iterator.Next()); | |
| 787 DCHECK(Translation::BEGIN == opcode); | |
| 788 USE(opcode); | |
| 789 // Read the number of output frames and allocate an array for their | |
| 790 // descriptions. | |
| 791 int count = iterator.Next(); | |
| 792 iterator.Next(); // Drop JS frames count. | |
| 793 DCHECK(output_ == NULL); | 765 DCHECK(output_ == NULL); |
| 794 output_ = new FrameDescription*[count]; | 766 output_ = new FrameDescription*[count]; |
| 795 for (int i = 0; i < count; ++i) { | 767 for (size_t i = 0; i < count; ++i) { |
| 796 output_[i] = NULL; | 768 output_[i] = NULL; |
| 797 } | 769 } |
| 798 output_count_ = count; | 770 output_count_ = static_cast<int>(count); |
| 799 | 771 |
| 800 Register fp_reg = JavaScriptFrame::fp_register(); | 772 Register fp_reg = JavaScriptFrame::fp_register(); |
| 801 stack_fp_ = reinterpret_cast<Address>( | 773 stack_fp_ = reinterpret_cast<Address>( |
| 802 input_->GetRegister(fp_reg.code()) + | 774 input_->GetRegister(fp_reg.code()) + |
| 803 has_alignment_padding_ * kPointerSize); | 775 has_alignment_padding_ * kPointerSize); |
| 804 | 776 |
| 805 // Translate each output frame. | 777 // Translate each output frame. |
| 806 for (int i = 0; i < count; ++i) { | 778 for (size_t i = 0; i < count; ++i) { |
| 807 // Read the ast node id, function, and frame height for this output frame. | 779 // Read the ast node id, function, and frame height for this output frame. |
| 808 Translation::Opcode opcode = | 780 int frame_index = static_cast<int>(i); |
|
Benedikt Meurer
2015/06/03 04:02:17
Remove this static_cast and change the functions b
Jarin
2015/06/03 09:28:07
Yeah, I tried that, but in those functions we comp
| |
| 809 static_cast<Translation::Opcode>(iterator.Next()); | 781 switch (translated_state_.frames()[i].kind()) { |
| 810 switch (opcode) { | 782 case TranslatedFrame::Function: |
| 811 case Translation::JS_FRAME: | 783 DoComputeJSFrame(nullptr, frame_index); |
| 812 DoComputeJSFrame(&iterator, i); | |
| 813 jsframe_count_++; | 784 jsframe_count_++; |
| 814 break; | 785 break; |
| 815 case Translation::ARGUMENTS_ADAPTOR_FRAME: | 786 case TranslatedFrame::ArgumentsAdaptor: |
| 816 DoComputeArgumentsAdaptorFrame(&iterator, i); | 787 DoComputeArgumentsAdaptorFrame(nullptr, frame_index); |
| 817 break; | 788 break; |
| 818 case Translation::CONSTRUCT_STUB_FRAME: | 789 case TranslatedFrame::ConstructStub: |
| 819 DoComputeConstructStubFrame(&iterator, i); | 790 DoComputeConstructStubFrame(nullptr, frame_index); |
| 820 break; | 791 break; |
| 821 case Translation::GETTER_STUB_FRAME: | 792 case TranslatedFrame::Getter: |
| 822 DoComputeAccessorStubFrame(&iterator, i, false); | 793 DoComputeAccessorStubFrame(nullptr, frame_index, false); |
| 823 break; | 794 break; |
| 824 case Translation::SETTER_STUB_FRAME: | 795 case TranslatedFrame::Setter: |
| 825 DoComputeAccessorStubFrame(&iterator, i, true); | 796 DoComputeAccessorStubFrame(nullptr, frame_index, true); |
| 826 break; | 797 break; |
| 827 case Translation::COMPILED_STUB_FRAME: | 798 case TranslatedFrame::CompiledStub: |
| 828 DoComputeCompiledStubFrame(&iterator, i); | 799 DoComputeCompiledStubFrame(nullptr, frame_index); |
| 829 break; | 800 break; |
| 830 case Translation::BEGIN: | 801 case TranslatedFrame::Invalid: |
| 831 case Translation::REGISTER: | 802 FATAL("invalid frame"); |
| 832 case Translation::INT32_REGISTER: | |
| 833 case Translation::UINT32_REGISTER: | |
| 834 case Translation::BOOL_REGISTER: | |
| 835 case Translation::DOUBLE_REGISTER: | |
| 836 case Translation::STACK_SLOT: | |
| 837 case Translation::INT32_STACK_SLOT: | |
| 838 case Translation::UINT32_STACK_SLOT: | |
| 839 case Translation::BOOL_STACK_SLOT: | |
| 840 case Translation::DOUBLE_STACK_SLOT: | |
| 841 case Translation::LITERAL: | |
| 842 case Translation::ARGUMENTS_OBJECT: | |
| 843 case Translation::DUPLICATED_OBJECT: | |
| 844 case Translation::CAPTURED_OBJECT: | |
| 845 FATAL("Unsupported translation"); | |
| 846 break; | 803 break; |
| 847 } | 804 } |
| 848 } | 805 } |
| 849 | 806 |
| 850 // Print some helpful diagnostic information. | 807 // Print some helpful diagnostic information. |
| 851 if (trace_scope_ != NULL) { | 808 if (trace_scope_ != NULL) { |
| 852 double ms = timer.Elapsed().InMillisecondsF(); | 809 double ms = timer.Elapsed().InMillisecondsF(); |
| 853 int index = output_count_ - 1; // Index of the topmost frame. | 810 int index = output_count_ - 1; // Index of the topmost frame. |
| 854 PrintF(trace_scope_->file(), "[deoptimizing (%s): end ", | 811 PrintF(trace_scope_->file(), "[deoptimizing (%s): end ", |
| 855 MessageFor(bailout_type_)); | 812 MessageFor(bailout_type_)); |
| 856 PrintFunctionName(); | 813 PrintFunctionName(); |
| 857 PrintF(trace_scope_->file(), | 814 PrintF(trace_scope_->file(), |
| 858 " @%d => node=%d, pc=0x%08" V8PRIxPTR ", state=%s, alignment=%s," | 815 " @%d => node=%d, pc=0x%08" V8PRIxPTR ", state=%s, alignment=%s," |
| 859 " took %0.3f ms]\n", | 816 " took %0.3f ms]\n", |
| 860 bailout_id_, | 817 bailout_id_, |
| 861 node_id.ToInt(), | 818 node_id.ToInt(), |
| 862 output_[index]->GetPc(), | 819 output_[index]->GetPc(), |
| 863 FullCodeGenerator::State2String( | 820 FullCodeGenerator::State2String( |
| 864 static_cast<FullCodeGenerator::State>( | 821 static_cast<FullCodeGenerator::State>( |
| 865 output_[index]->GetState()->value())), | 822 output_[index]->GetState()->value())), |
| 866 has_alignment_padding_ ? "with padding" : "no padding", | 823 has_alignment_padding_ ? "with padding" : "no padding", |
| 867 ms); | 824 ms); |
| 868 } | 825 } |
| 869 } | 826 } |
| 870 | 827 |
| 871 | 828 |
| 872 void Deoptimizer::DoComputeJSFrame(TranslationIterator* iterator, | 829 void Deoptimizer::DoComputeJSFrame(TranslationIterator* iterator, |
| 873 int frame_index) { | 830 int frame_index) { |
| 874 BailoutId node_id = BailoutId(iterator->Next()); | 831 TranslatedFrame* translated_frame = |
| 875 JSFunction* function; | 832 &(translated_state_.frames()[frame_index]); |
| 876 if (frame_index != 0) { | 833 TranslatedFrame::ValueIterator value_iterator = translated_frame->begin(); |
| 877 function = JSFunction::cast(ComputeLiteral(iterator->Next())); | 834 int input_index = 0; |
| 878 } else { | 835 |
| 879 int closure_id = iterator->Next(); | 836 BailoutId node_id = translated_frame->node_id(); |
| 880 USE(closure_id); | 837 JSFunction* function = translated_frame->raw_function(); |
| 881 CHECK_EQ(Translation::kSelfLiteralId, closure_id); | 838 unsigned height = |
| 882 function = function_; | 839 translated_frame->height() - 1; // Do not count the context. |
| 883 } | |
| 884 unsigned height = iterator->Next() - 1; // Do not count the context. | |
| 885 unsigned height_in_bytes = height * kPointerSize; | 840 unsigned height_in_bytes = height * kPointerSize; |
| 886 if (trace_scope_ != NULL) { | 841 if (trace_scope_ != NULL) { |
| 887 PrintF(trace_scope_->file(), " translating "); | 842 PrintF(trace_scope_->file(), " translating frame "); |
| 888 function->PrintName(trace_scope_->file()); | 843 function->PrintName(trace_scope_->file()); |
| 889 PrintF(trace_scope_->file(), | 844 PrintF(trace_scope_->file(), |
| 890 " => node=%d, height=%d\n", node_id.ToInt(), height_in_bytes); | 845 " => node=%d, height=%d\n", node_id.ToInt(), height_in_bytes); |
| 891 } | 846 } |
| 892 | 847 |
| 848 CHECK(translated_frame->kind() == TranslatedFrame::Function); | |
| 849 CHECK(function == translated_frame->raw_function()); | |
| 850 | |
| 893 // The 'fixed' part of the frame consists of the incoming parameters and | 851 // The 'fixed' part of the frame consists of the incoming parameters and |
| 894 // the part described by JavaScriptFrameConstants. | 852 // the part described by JavaScriptFrameConstants. |
| 895 unsigned fixed_frame_size = ComputeFixedSize(function); | 853 unsigned fixed_frame_size = ComputeFixedSize(function); |
| 896 unsigned input_frame_size = input_->GetFrameSize(); | 854 unsigned input_frame_size = input_->GetFrameSize(); |
| 897 unsigned output_frame_size = height_in_bytes + fixed_frame_size; | 855 unsigned output_frame_size = height_in_bytes + fixed_frame_size; |
| 898 | 856 |
| 899 // Allocate and store the output frame description. | 857 // Allocate and store the output frame description. |
| 900 FrameDescription* output_frame = | 858 FrameDescription* output_frame = |
| 901 new(output_frame_size) FrameDescription(output_frame_size, function); | 859 new(output_frame_size) FrameDescription(output_frame_size, function); |
| 902 output_frame->SetFrameType(StackFrame::JAVA_SCRIPT); | 860 output_frame->SetFrameType(StackFrame::JAVA_SCRIPT); |
| (...skipping 28 matching lines...) Expand all Loading... | |
| 931 } | 889 } |
| 932 output_frame->SetTop(top_address); | 890 output_frame->SetTop(top_address); |
| 933 | 891 |
| 934 // Compute the incoming parameter translation. | 892 // Compute the incoming parameter translation. |
| 935 int parameter_count = | 893 int parameter_count = |
| 936 function->shared()->internal_formal_parameter_count() + 1; | 894 function->shared()->internal_formal_parameter_count() + 1; |
| 937 unsigned output_offset = output_frame_size; | 895 unsigned output_offset = output_frame_size; |
| 938 unsigned input_offset = input_frame_size; | 896 unsigned input_offset = input_frame_size; |
| 939 for (int i = 0; i < parameter_count; ++i) { | 897 for (int i = 0; i < parameter_count; ++i) { |
| 940 output_offset -= kPointerSize; | 898 output_offset -= kPointerSize; |
| 941 DoTranslateCommand(iterator, frame_index, output_offset); | 899 WriteValueToOutput(&value_iterator, &input_index, frame_index, |
| 900 output_offset); | |
| 942 } | 901 } |
| 943 input_offset -= (parameter_count * kPointerSize); | 902 input_offset -= (parameter_count * kPointerSize); |
| 944 | 903 |
| 945 // There are no translation commands for the caller's pc and fp, the | 904 // There are no translation commands for the caller's pc and fp, the |
| 946 // context, and the function. Synthesize their values and set them up | 905 // context, and the function. Synthesize their values and set them up |
| 947 // explicitly. | 906 // explicitly. |
| 948 // | 907 // |
| 949 // The caller's pc for the bottommost output frame is the same as in the | 908 // The caller's pc for the bottommost output frame is the same as in the |
| 950 // input frame. For all subsequent output frames, it can be read from the | 909 // input frame. For all subsequent output frames, it can be read from the |
| 951 // previous one. This frame's pc can be computed from the non-optimized | 910 // previous one. This frame's pc can be computed from the non-optimized |
| (...skipping 59 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 1011 } | 970 } |
| 1012 } | 971 } |
| 1013 | 972 |
| 1014 // For the bottommost output frame the context can be gotten from the input | 973 // For the bottommost output frame the context can be gotten from the input |
| 1015 // frame. For all subsequent output frames it can be gotten from the function | 974 // frame. For all subsequent output frames it can be gotten from the function |
| 1016 // so long as we don't inline functions that need local contexts. | 975 // so long as we don't inline functions that need local contexts. |
| 1017 Register context_reg = JavaScriptFrame::context_register(); | 976 Register context_reg = JavaScriptFrame::context_register(); |
| 1018 output_offset -= kPointerSize; | 977 output_offset -= kPointerSize; |
| 1019 input_offset -= kPointerSize; | 978 input_offset -= kPointerSize; |
| 1020 // Read the context from the translations. | 979 // Read the context from the translations. |
| 1021 DoTranslateCommand(iterator, frame_index, output_offset); | 980 WriteValueToOutput(&value_iterator, &input_index, frame_index, output_offset); |
| 1022 value = output_frame->GetFrameSlot(output_offset); | 981 value = output_frame->GetFrameSlot(output_offset); |
| 1023 // The context should not be a placeholder for a materialized object. | 982 // The context should not be a placeholder for a materialized object. |
| 1024 CHECK(value != | 983 CHECK(value != |
| 1025 reinterpret_cast<intptr_t>(isolate_->heap()->arguments_marker())); | 984 reinterpret_cast<intptr_t>(isolate_->heap()->arguments_marker())); |
| 1026 if (value == | 985 if (value == |
| 1027 reinterpret_cast<intptr_t>(isolate_->heap()->undefined_value())) { | 986 reinterpret_cast<intptr_t>(isolate_->heap()->undefined_value())) { |
| 1028 // If the context was optimized away, just use the context from | 987 // If the context was optimized away, just use the context from |
| 1029 // the activation. This should only apply to Crankshaft code. | 988 // the activation. This should only apply to Crankshaft code. |
| 1030 CHECK(!compiled_code_->is_turbofanned()); | 989 CHECK(!compiled_code_->is_turbofanned()); |
| 1031 if (is_bottommost) { | 990 if (is_bottommost) { |
| (...skipping 23 matching lines...) Expand all Loading... | |
| 1055 if (trace_scope_ != NULL) { | 1014 if (trace_scope_ != NULL) { |
| 1056 PrintF(trace_scope_->file(), | 1015 PrintF(trace_scope_->file(), |
| 1057 " 0x%08" V8PRIxPTR ": [top + %d] <- 0x%08" | 1016 " 0x%08" V8PRIxPTR ": [top + %d] <- 0x%08" |
| 1058 V8PRIxPTR "; function\n", | 1017 V8PRIxPTR "; function\n", |
| 1059 top_address + output_offset, output_offset, value); | 1018 top_address + output_offset, output_offset, value); |
| 1060 } | 1019 } |
| 1061 | 1020 |
| 1062 // Translate the rest of the frame. | 1021 // Translate the rest of the frame. |
| 1063 for (unsigned i = 0; i < height; ++i) { | 1022 for (unsigned i = 0; i < height; ++i) { |
| 1064 output_offset -= kPointerSize; | 1023 output_offset -= kPointerSize; |
| 1065 DoTranslateCommand(iterator, frame_index, output_offset); | 1024 WriteValueToOutput(&value_iterator, &input_index, frame_index, |
| 1025 output_offset); | |
| 1066 } | 1026 } |
| 1067 CHECK_EQ(0u, output_offset); | 1027 CHECK_EQ(0u, output_offset); |
| 1068 | 1028 |
| 1069 // Compute this frame's PC, state, and continuation. | 1029 // Compute this frame's PC, state, and continuation. |
| 1070 Code* non_optimized_code = function->shared()->code(); | 1030 Code* non_optimized_code = function->shared()->code(); |
| 1071 FixedArray* raw_data = non_optimized_code->deoptimization_data(); | 1031 FixedArray* raw_data = non_optimized_code->deoptimization_data(); |
| 1072 DeoptimizationOutputData* data = DeoptimizationOutputData::cast(raw_data); | 1032 DeoptimizationOutputData* data = DeoptimizationOutputData::cast(raw_data); |
| 1073 Address start = non_optimized_code->instruction_start(); | 1033 Address start = non_optimized_code->instruction_start(); |
| 1074 unsigned pc_and_state = GetOutputInfo(data, node_id, function->shared()); | 1034 unsigned pc_and_state = GetOutputInfo(data, node_id, function->shared()); |
| 1075 unsigned pc_offset = FullCodeGenerator::PcField::decode(pc_and_state); | 1035 unsigned pc_offset = FullCodeGenerator::PcField::decode(pc_and_state); |
| (...skipping 28 matching lines...) Expand all Loading... | |
| 1104 CHECK_EQ(bailout_type_, EAGER); | 1064 CHECK_EQ(bailout_type_, EAGER); |
| 1105 } | 1065 } |
| 1106 output_frame->SetContinuation( | 1066 output_frame->SetContinuation( |
| 1107 reinterpret_cast<intptr_t>(continuation->entry())); | 1067 reinterpret_cast<intptr_t>(continuation->entry())); |
| 1108 } | 1068 } |
| 1109 } | 1069 } |
| 1110 | 1070 |
| 1111 | 1071 |
| 1112 void Deoptimizer::DoComputeArgumentsAdaptorFrame(TranslationIterator* iterator, | 1072 void Deoptimizer::DoComputeArgumentsAdaptorFrame(TranslationIterator* iterator, |
| 1113 int frame_index) { | 1073 int frame_index) { |
| 1114 JSFunction* function = JSFunction::cast(ComputeLiteral(iterator->Next())); | 1074 TranslatedFrame* translated_frame = |
| 1115 unsigned height = iterator->Next(); | 1075 &(translated_state_.frames()[frame_index]); |
| 1076 TranslatedFrame::ValueIterator value_iterator = translated_frame->begin(); | |
| 1077 int input_index = 0; | |
| 1078 | |
| 1079 JSFunction* function = translated_frame->raw_function(); | |
| 1080 unsigned height = translated_frame->height(); | |
| 1116 unsigned height_in_bytes = height * kPointerSize; | 1081 unsigned height_in_bytes = height * kPointerSize; |
| 1117 if (trace_scope_ != NULL) { | 1082 if (trace_scope_ != NULL) { |
| 1118 PrintF(trace_scope_->file(), | 1083 PrintF(trace_scope_->file(), |
| 1119 " translating arguments adaptor => height=%d\n", height_in_bytes); | 1084 " translating arguments adaptor => height=%d\n", height_in_bytes); |
| 1120 } | 1085 } |
| 1121 | 1086 |
| 1122 unsigned fixed_frame_size = ArgumentsAdaptorFrameConstants::kFrameSize; | 1087 unsigned fixed_frame_size = ArgumentsAdaptorFrameConstants::kFrameSize; |
| 1123 unsigned output_frame_size = height_in_bytes + fixed_frame_size; | 1088 unsigned output_frame_size = height_in_bytes + fixed_frame_size; |
| 1124 | 1089 |
| 1125 // Allocate and store the output frame description. | 1090 // Allocate and store the output frame description. |
| (...skipping 10 matching lines...) Expand all Loading... | |
| 1136 // frame's top and this frame's size. | 1101 // frame's top and this frame's size. |
| 1137 intptr_t top_address; | 1102 intptr_t top_address; |
| 1138 top_address = output_[frame_index - 1]->GetTop() - output_frame_size; | 1103 top_address = output_[frame_index - 1]->GetTop() - output_frame_size; |
| 1139 output_frame->SetTop(top_address); | 1104 output_frame->SetTop(top_address); |
| 1140 | 1105 |
| 1141 // Compute the incoming parameter translation. | 1106 // Compute the incoming parameter translation. |
| 1142 int parameter_count = height; | 1107 int parameter_count = height; |
| 1143 unsigned output_offset = output_frame_size; | 1108 unsigned output_offset = output_frame_size; |
| 1144 for (int i = 0; i < parameter_count; ++i) { | 1109 for (int i = 0; i < parameter_count; ++i) { |
| 1145 output_offset -= kPointerSize; | 1110 output_offset -= kPointerSize; |
| 1146 DoTranslateCommand(iterator, frame_index, output_offset); | 1111 WriteValueToOutput(&value_iterator, &input_index, frame_index, |
| 1112 output_offset); | |
| 1147 } | 1113 } |
| 1148 | 1114 |
| 1149 // Read caller's PC from the previous frame. | 1115 // Read caller's PC from the previous frame. |
| 1150 output_offset -= kPCOnStackSize; | 1116 output_offset -= kPCOnStackSize; |
| 1151 intptr_t callers_pc = output_[frame_index - 1]->GetPc(); | 1117 intptr_t callers_pc = output_[frame_index - 1]->GetPc(); |
| 1152 output_frame->SetCallerPc(output_offset, callers_pc); | 1118 output_frame->SetCallerPc(output_offset, callers_pc); |
| 1153 if (trace_scope_ != NULL) { | 1119 if (trace_scope_ != NULL) { |
| 1154 PrintF(trace_scope_->file(), | 1120 PrintF(trace_scope_->file(), |
| 1155 " 0x%08" V8PRIxPTR ": [top + %d] <- 0x%08" | 1121 " 0x%08" V8PRIxPTR ": [top + %d] <- 0x%08" |
| 1156 V8PRIxPTR " ; caller's pc\n", | 1122 V8PRIxPTR " ; caller's pc\n", |
| (...skipping 71 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 1228 if (FLAG_enable_ool_constant_pool) { | 1194 if (FLAG_enable_ool_constant_pool) { |
| 1229 intptr_t constant_pool_value = | 1195 intptr_t constant_pool_value = |
| 1230 reinterpret_cast<intptr_t>(adaptor_trampoline->constant_pool()); | 1196 reinterpret_cast<intptr_t>(adaptor_trampoline->constant_pool()); |
| 1231 output_frame->SetConstantPool(constant_pool_value); | 1197 output_frame->SetConstantPool(constant_pool_value); |
| 1232 } | 1198 } |
| 1233 } | 1199 } |
| 1234 | 1200 |
| 1235 | 1201 |
| 1236 void Deoptimizer::DoComputeConstructStubFrame(TranslationIterator* iterator, | 1202 void Deoptimizer::DoComputeConstructStubFrame(TranslationIterator* iterator, |
| 1237 int frame_index) { | 1203 int frame_index) { |
| 1204 TranslatedFrame* translated_frame = | |
| 1205 &(translated_state_.frames()[frame_index]); | |
| 1206 TranslatedFrame::ValueIterator value_iterator = translated_frame->begin(); | |
| 1207 int input_index = 0; | |
| 1208 | |
| 1238 Builtins* builtins = isolate_->builtins(); | 1209 Builtins* builtins = isolate_->builtins(); |
| 1239 Code* construct_stub = builtins->builtin(Builtins::kJSConstructStubGeneric); | 1210 Code* construct_stub = builtins->builtin(Builtins::kJSConstructStubGeneric); |
| 1240 JSFunction* function = JSFunction::cast(ComputeLiteral(iterator->Next())); | 1211 JSFunction* function = translated_frame->raw_function(); |
| 1241 unsigned height = iterator->Next(); | 1212 unsigned height = translated_frame->height(); |
| 1242 unsigned height_in_bytes = height * kPointerSize; | 1213 unsigned height_in_bytes = height * kPointerSize; |
| 1243 if (trace_scope_ != NULL) { | 1214 if (trace_scope_ != NULL) { |
| 1244 PrintF(trace_scope_->file(), | 1215 PrintF(trace_scope_->file(), |
| 1245 " translating construct stub => height=%d\n", height_in_bytes); | 1216 " translating construct stub => height=%d\n", height_in_bytes); |
| 1246 } | 1217 } |
| 1247 | 1218 |
| 1248 unsigned fixed_frame_size = ConstructFrameConstants::kFrameSize; | 1219 unsigned fixed_frame_size = ConstructFrameConstants::kFrameSize; |
| 1249 unsigned output_frame_size = height_in_bytes + fixed_frame_size; | 1220 unsigned output_frame_size = height_in_bytes + fixed_frame_size; |
| 1250 | 1221 |
| 1251 // Allocate and store the output frame description. | 1222 // Allocate and store the output frame description. |
| (...skipping 10 matching lines...) Expand all Loading... | |
| 1262 // frame's top and this frame's size. | 1233 // frame's top and this frame's size. |
| 1263 intptr_t top_address; | 1234 intptr_t top_address; |
| 1264 top_address = output_[frame_index - 1]->GetTop() - output_frame_size; | 1235 top_address = output_[frame_index - 1]->GetTop() - output_frame_size; |
| 1265 output_frame->SetTop(top_address); | 1236 output_frame->SetTop(top_address); |
| 1266 | 1237 |
| 1267 // Compute the incoming parameter translation. | 1238 // Compute the incoming parameter translation. |
| 1268 int parameter_count = height; | 1239 int parameter_count = height; |
| 1269 unsigned output_offset = output_frame_size; | 1240 unsigned output_offset = output_frame_size; |
| 1270 for (int i = 0; i < parameter_count; ++i) { | 1241 for (int i = 0; i < parameter_count; ++i) { |
| 1271 output_offset -= kPointerSize; | 1242 output_offset -= kPointerSize; |
| 1272 int deferred_object_index = deferred_objects_.length(); | |
| 1273 DoTranslateCommand(iterator, frame_index, output_offset); | |
| 1274 // The allocated receiver of a construct stub frame is passed as the | 1243 // The allocated receiver of a construct stub frame is passed as the |
| 1275 // receiver parameter through the translation. It might be encoding | 1244 // receiver parameter through the translation. It might be encoding |
| 1276 // a captured object, patch the slot address for a captured object. | 1245 // a captured object, override the slot address for a captured object. |
| 1277 if (i == 0 && deferred_objects_.length() > deferred_object_index) { | 1246 WriteValueToOutput( |
| 1278 CHECK(!deferred_objects_[deferred_object_index].is_arguments()); | 1247 &value_iterator, &input_index, frame_index, output_offset, |
| 1279 deferred_objects_[deferred_object_index].patch_slot_address(top_address); | 1248 (i == 0) ? reinterpret_cast<Address>(top_address) : nullptr); |
| 1280 } | |
| 1281 } | 1249 } |
| 1282 | 1250 |
| 1283 // Read caller's PC from the previous frame. | 1251 // Read caller's PC from the previous frame. |
| 1284 output_offset -= kPCOnStackSize; | 1252 output_offset -= kPCOnStackSize; |
| 1285 intptr_t callers_pc = output_[frame_index - 1]->GetPc(); | 1253 intptr_t callers_pc = output_[frame_index - 1]->GetPc(); |
| 1286 output_frame->SetCallerPc(output_offset, callers_pc); | 1254 output_frame->SetCallerPc(output_offset, callers_pc); |
| 1287 if (trace_scope_ != NULL) { | 1255 if (trace_scope_ != NULL) { |
| 1288 PrintF(trace_scope_->file(), | 1256 PrintF(trace_scope_->file(), |
| 1289 " 0x%08" V8PRIxPTR ": [top + %d] <- 0x%08" | 1257 " 0x%08" V8PRIxPTR ": [top + %d] <- 0x%08" |
| 1290 V8PRIxPTR " ; caller's pc\n", | 1258 V8PRIxPTR " ; caller's pc\n", |
| (...skipping 105 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 1396 intptr_t constant_pool_value = | 1364 intptr_t constant_pool_value = |
| 1397 reinterpret_cast<intptr_t>(construct_stub->constant_pool()); | 1365 reinterpret_cast<intptr_t>(construct_stub->constant_pool()); |
| 1398 output_frame->SetConstantPool(constant_pool_value); | 1366 output_frame->SetConstantPool(constant_pool_value); |
| 1399 } | 1367 } |
| 1400 } | 1368 } |
| 1401 | 1369 |
| 1402 | 1370 |
| 1403 void Deoptimizer::DoComputeAccessorStubFrame(TranslationIterator* iterator, | 1371 void Deoptimizer::DoComputeAccessorStubFrame(TranslationIterator* iterator, |
| 1404 int frame_index, | 1372 int frame_index, |
| 1405 bool is_setter_stub_frame) { | 1373 bool is_setter_stub_frame) { |
| 1406 JSFunction* accessor = JSFunction::cast(ComputeLiteral(iterator->Next())); | 1374 TranslatedFrame* translated_frame = |
| 1375 &(translated_state_.frames()[frame_index]); | |
| 1376 TranslatedFrame::ValueIterator value_iterator = translated_frame->begin(); | |
| 1377 int input_index = 0; | |
| 1378 | |
| 1379 JSFunction* accessor = translated_frame->raw_function(); | |
| 1407 // The receiver (and the implicit return value, if any) are expected in | 1380 // The receiver (and the implicit return value, if any) are expected in |
| 1408 // registers by the LoadIC/StoreIC, so they don't belong to the output stack | 1381 // registers by the LoadIC/StoreIC, so they don't belong to the output stack |
| 1409 // frame. This means that we have to use a height of 0. | 1382 // frame. This means that we have to use a height of 0. |
| 1410 unsigned height = 0; | 1383 unsigned height = 0; |
| 1411 unsigned height_in_bytes = height * kPointerSize; | 1384 unsigned height_in_bytes = height * kPointerSize; |
| 1412 const char* kind = is_setter_stub_frame ? "setter" : "getter"; | 1385 const char* kind = is_setter_stub_frame ? "setter" : "getter"; |
| 1413 if (trace_scope_ != NULL) { | 1386 if (trace_scope_ != NULL) { |
| 1414 PrintF(trace_scope_->file(), | 1387 PrintF(trace_scope_->file(), |
| 1415 " translating %s stub => height=%u\n", kind, height_in_bytes); | 1388 " translating %s stub => height=%u\n", kind, height_in_bytes); |
| 1416 } | 1389 } |
| (...skipping 93 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 1510 value = reinterpret_cast<intptr_t>(accessor_stub); | 1483 value = reinterpret_cast<intptr_t>(accessor_stub); |
| 1511 output_frame->SetFrameSlot(output_offset, value); | 1484 output_frame->SetFrameSlot(output_offset, value); |
| 1512 if (trace_scope_ != NULL) { | 1485 if (trace_scope_ != NULL) { |
| 1513 PrintF(trace_scope_->file(), | 1486 PrintF(trace_scope_->file(), |
| 1514 " 0x%08" V8PRIxPTR ": [top + %u] <- 0x%08" V8PRIxPTR | 1487 " 0x%08" V8PRIxPTR ": [top + %u] <- 0x%08" V8PRIxPTR |
| 1515 " ; code object\n", | 1488 " ; code object\n", |
| 1516 top_address + output_offset, output_offset, value); | 1489 top_address + output_offset, output_offset, value); |
| 1517 } | 1490 } |
| 1518 | 1491 |
| 1519 // Skip receiver. | 1492 // Skip receiver. |
| 1520 DoTranslateObjectAndSkip(iterator); | 1493 value_iterator++; |
| 1521 | 1494 |
| 1522 if (is_setter_stub_frame) { | 1495 if (is_setter_stub_frame) { |
| 1523 // The implicit return value was part of the artificial setter stub | 1496 // The implicit return value was part of the artificial setter stub |
| 1524 // environment. | 1497 // environment. |
| 1525 output_offset -= kPointerSize; | 1498 output_offset -= kPointerSize; |
| 1526 DoTranslateCommand(iterator, frame_index, output_offset); | 1499 WriteValueToOutput(&value_iterator, &input_index, frame_index, |
| 1500 output_offset); | |
| 1527 } | 1501 } |
| 1528 | 1502 |
| 1529 CHECK_EQ(0u, output_offset); | 1503 CHECK_EQ(0u, output_offset); |
| 1530 | 1504 |
| 1531 Smi* offset = is_setter_stub_frame ? | 1505 Smi* offset = is_setter_stub_frame ? |
| 1532 isolate_->heap()->setter_stub_deopt_pc_offset() : | 1506 isolate_->heap()->setter_stub_deopt_pc_offset() : |
| 1533 isolate_->heap()->getter_stub_deopt_pc_offset(); | 1507 isolate_->heap()->getter_stub_deopt_pc_offset(); |
| 1534 intptr_t pc = reinterpret_cast<intptr_t>( | 1508 intptr_t pc = reinterpret_cast<intptr_t>( |
| 1535 accessor_stub->instruction_start() + offset->value()); | 1509 accessor_stub->instruction_start() + offset->value()); |
| 1536 output_frame->SetPc(pc); | 1510 output_frame->SetPc(pc); |
| (...skipping 32 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 1569 // and spilled to stack | .... | | 1543 // and spilled to stack | .... | |
| 1570 // +-------------------------+ | 1544 // +-------------------------+ |
| 1571 // | caller stack param n | | 1545 // | caller stack param n | |
| 1572 // +-------------------------+<-spreg | 1546 // +-------------------------+<-spreg |
| 1573 // reg = number of parameters | 1547 // reg = number of parameters |
| 1574 // reg = failure handler address | 1548 // reg = failure handler address |
| 1575 // reg = saved frame | 1549 // reg = saved frame |
| 1576 // reg = JSFunction context | 1550 // reg = JSFunction context |
| 1577 // | 1551 // |
| 1578 | 1552 |
| 1553 TranslatedFrame* translated_frame = | |
| 1554 &(translated_state_.frames()[frame_index]); | |
| 1555 TranslatedFrame::ValueIterator value_iterator = translated_frame->begin(); | |
| 1556 int input_index = 0; | |
| 1557 | |
| 1579 CHECK(compiled_code_->is_hydrogen_stub()); | 1558 CHECK(compiled_code_->is_hydrogen_stub()); |
| 1580 int major_key = CodeStub::GetMajorKey(compiled_code_); | 1559 int major_key = CodeStub::GetMajorKey(compiled_code_); |
| 1581 CodeStubDescriptor descriptor(isolate_, compiled_code_->stub_key()); | 1560 CodeStubDescriptor descriptor(isolate_, compiled_code_->stub_key()); |
| 1582 | 1561 |
| 1583 // The output frame must have room for all pushed register parameters | 1562 // The output frame must have room for all pushed register parameters |
| 1584 // and the standard stack frame slots. Include space for an argument | 1563 // and the standard stack frame slots. Include space for an argument |
| 1585 // object to the callee and optionally the space to pass the argument | 1564 // object to the callee and optionally the space to pass the argument |
| 1586 // object to the stub failure handler. | 1565 // object to the stub failure handler. |
| 1587 int param_count = descriptor.GetEnvironmentParameterCount(); | 1566 int param_count = descriptor.GetEnvironmentParameterCount(); |
| 1567 CHECK_EQ(translated_frame->height(), param_count); | |
| 1588 CHECK_GE(param_count, 0); | 1568 CHECK_GE(param_count, 0); |
| 1589 | 1569 |
| 1590 int height_in_bytes = kPointerSize * param_count + sizeof(Arguments) + | 1570 int height_in_bytes = kPointerSize * param_count + sizeof(Arguments) + |
| 1591 kPointerSize; | 1571 kPointerSize; |
| 1592 int fixed_frame_size = StandardFrameConstants::kFixedFrameSize; | 1572 int fixed_frame_size = StandardFrameConstants::kFixedFrameSize; |
| 1593 int input_frame_size = input_->GetFrameSize(); | 1573 int input_frame_size = input_->GetFrameSize(); |
| 1594 int output_frame_size = height_in_bytes + fixed_frame_size; | 1574 int output_frame_size = height_in_bytes + fixed_frame_size; |
| 1595 if (trace_scope_ != NULL) { | 1575 if (trace_scope_ != NULL) { |
| 1596 PrintF(trace_scope_->file(), | 1576 PrintF(trace_scope_->file(), |
| 1597 " translating %s => StubFailureTrampolineStub, height=%d\n", | 1577 " translating %s => StubFailureTrampolineStub, height=%d\n", |
| (...skipping 127 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 1725 PrintF(trace_scope_->file(), | 1705 PrintF(trace_scope_->file(), |
| 1726 " 0x%08" V8PRIxPTR ": [top + %d] <- 0x%08" | 1706 " 0x%08" V8PRIxPTR ": [top + %d] <- 0x%08" |
| 1727 V8PRIxPTR " ; args*\n", | 1707 V8PRIxPTR " ; args*\n", |
| 1728 top_address + output_frame_offset, output_frame_offset, value); | 1708 top_address + output_frame_offset, output_frame_offset, value); |
| 1729 } | 1709 } |
| 1730 | 1710 |
| 1731 // Copy the register parameters to the failure frame. | 1711 // Copy the register parameters to the failure frame. |
| 1732 int arguments_length_offset = -1; | 1712 int arguments_length_offset = -1; |
| 1733 for (int i = 0; i < param_count; ++i) { | 1713 for (int i = 0; i < param_count; ++i) { |
| 1734 output_frame_offset -= kPointerSize; | 1714 output_frame_offset -= kPointerSize; |
| 1735 DoTranslateCommand(iterator, 0, output_frame_offset); | 1715 WriteValueToOutput(&value_iterator, &input_index, 0, output_frame_offset); |
| 1736 | 1716 |
| 1737 if (!arg_count_known && descriptor.IsEnvironmentParameterCountRegister(i)) { | 1717 if (!arg_count_known && descriptor.IsEnvironmentParameterCountRegister(i)) { |
| 1738 arguments_length_offset = output_frame_offset; | 1718 arguments_length_offset = output_frame_offset; |
| 1739 } | 1719 } |
| 1740 } | 1720 } |
| 1741 | 1721 |
| 1742 CHECK_EQ(0u, output_frame_offset); | 1722 CHECK_EQ(0u, output_frame_offset); |
| 1743 | 1723 |
| 1744 if (!arg_count_known) { | 1724 if (!arg_count_known) { |
| 1745 CHECK_GE(arguments_length_offset, 0); | 1725 CHECK_GE(arguments_length_offset, 0); |
| (...skipping 46 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 1792 output_frame->SetRegister(constant_pool_reg.code(), constant_pool_value); | 1772 output_frame->SetRegister(constant_pool_reg.code(), constant_pool_value); |
| 1793 } | 1773 } |
| 1794 output_frame->SetState(Smi::FromInt(FullCodeGenerator::NO_REGISTERS)); | 1774 output_frame->SetState(Smi::FromInt(FullCodeGenerator::NO_REGISTERS)); |
| 1795 Code* notify_failure = | 1775 Code* notify_failure = |
| 1796 isolate_->builtins()->builtin(Builtins::kNotifyStubFailureSaveDoubles); | 1776 isolate_->builtins()->builtin(Builtins::kNotifyStubFailureSaveDoubles); |
| 1797 output_frame->SetContinuation( | 1777 output_frame->SetContinuation( |
| 1798 reinterpret_cast<intptr_t>(notify_failure->entry())); | 1778 reinterpret_cast<intptr_t>(notify_failure->entry())); |
| 1799 } | 1779 } |
| 1800 | 1780 |
| 1801 | 1781 |
| 1802 Handle<Object> Deoptimizer::MaterializeNextHeapObject() { | |
| 1803 int object_index = materialization_object_index_++; | |
| 1804 ObjectMaterializationDescriptor desc = deferred_objects_[object_index]; | |
| 1805 const int length = desc.object_length(); | |
| 1806 | |
| 1807 if (desc.duplicate_object() >= 0) { | |
| 1808 // Found a previously materialized object by de-duplication. | |
| 1809 object_index = desc.duplicate_object(); | |
| 1810 materialized_objects_->Add(Handle<Object>()); | |
| 1811 } else if (desc.is_arguments() && ArgumentsObjectIsAdapted(object_index)) { | |
| 1812 // Use the arguments adapter frame we just built to materialize the | |
| 1813 // arguments object. FunctionGetArguments can't throw an exception. | |
| 1814 Handle<JSFunction> function = ArgumentsObjectFunction(object_index); | |
| 1815 Handle<JSObject> arguments = Handle<JSObject>::cast( | |
| 1816 Accessors::FunctionGetArguments(function)); | |
| 1817 materialized_objects_->Add(arguments); | |
| 1818 // To keep consistent object counters, we still materialize the | |
| 1819 // nested values (but we throw them away). | |
| 1820 for (int i = 0; i < length; ++i) { | |
| 1821 MaterializeNextValue(); | |
| 1822 } | |
| 1823 } else if (desc.is_arguments()) { | |
| 1824 // Construct an arguments object and copy the parameters to a newly | |
| 1825 // allocated arguments object backing store. | |
| 1826 Handle<JSFunction> function = ArgumentsObjectFunction(object_index); | |
| 1827 Handle<JSObject> arguments = | |
| 1828 isolate_->factory()->NewArgumentsObject(function, length); | |
| 1829 Handle<FixedArray> array = isolate_->factory()->NewFixedArray(length); | |
| 1830 DCHECK_EQ(array->length(), length); | |
| 1831 arguments->set_elements(*array); | |
| 1832 materialized_objects_->Add(arguments); | |
| 1833 for (int i = 0; i < length; ++i) { | |
| 1834 Handle<Object> value = MaterializeNextValue(); | |
| 1835 array->set(i, *value); | |
| 1836 } | |
| 1837 } else { | |
| 1838 // Dispatch on the instance type of the object to be materialized. | |
| 1839 // We also need to make sure that the representation of all fields | |
| 1840 // in the given object are general enough to hold a tagged value. | |
| 1841 Handle<Map> map = Map::GeneralizeAllFieldRepresentations( | |
| 1842 Handle<Map>::cast(MaterializeNextValue())); | |
| 1843 switch (map->instance_type()) { | |
| 1844 case MUTABLE_HEAP_NUMBER_TYPE: | |
| 1845 case HEAP_NUMBER_TYPE: { | |
| 1846 // Reuse the HeapNumber value directly as it is already properly | |
| 1847 // tagged and skip materializing the HeapNumber explicitly. Turn mutable | |
| 1848 // heap numbers immutable. | |
| 1849 Handle<Object> object = MaterializeNextValue(); | |
| 1850 if (object_index < prev_materialized_count_) { | |
| 1851 materialized_objects_->Add(Handle<Object>( | |
| 1852 previously_materialized_objects_->get(object_index), isolate_)); | |
| 1853 } else { | |
| 1854 materialized_objects_->Add(object); | |
| 1855 } | |
| 1856 materialization_value_index_ += kDoubleSize / kPointerSize - 1; | |
| 1857 break; | |
| 1858 } | |
| 1859 case JS_OBJECT_TYPE: { | |
| 1860 Handle<JSObject> object = | |
| 1861 isolate_->factory()->NewJSObjectFromMap(map, NOT_TENURED, false); | |
| 1862 if (object_index < prev_materialized_count_) { | |
| 1863 materialized_objects_->Add(Handle<Object>( | |
| 1864 previously_materialized_objects_->get(object_index), isolate_)); | |
| 1865 } else { | |
| 1866 materialized_objects_->Add(object); | |
| 1867 } | |
| 1868 Handle<Object> properties = MaterializeNextValue(); | |
| 1869 Handle<Object> elements = MaterializeNextValue(); | |
| 1870 object->set_properties(FixedArray::cast(*properties)); | |
| 1871 object->set_elements(FixedArrayBase::cast(*elements)); | |
| 1872 for (int i = 0; i < length - 3; ++i) { | |
| 1873 Handle<Object> value = MaterializeNextValue(); | |
| 1874 FieldIndex index = FieldIndex::ForPropertyIndex(object->map(), i); | |
| 1875 object->FastPropertyAtPut(index, *value); | |
| 1876 } | |
| 1877 break; | |
| 1878 } | |
| 1879 case JS_ARRAY_TYPE: { | |
| 1880 Handle<JSArray> object = | |
| 1881 isolate_->factory()->NewJSArray(0, map->elements_kind()); | |
| 1882 if (object_index < prev_materialized_count_) { | |
| 1883 materialized_objects_->Add(Handle<Object>( | |
| 1884 previously_materialized_objects_->get(object_index), isolate_)); | |
| 1885 } else { | |
| 1886 materialized_objects_->Add(object); | |
| 1887 } | |
| 1888 Handle<Object> properties = MaterializeNextValue(); | |
| 1889 Handle<Object> elements = MaterializeNextValue(); | |
| 1890 Handle<Object> length = MaterializeNextValue(); | |
| 1891 object->set_properties(FixedArray::cast(*properties)); | |
| 1892 object->set_elements(FixedArrayBase::cast(*elements)); | |
| 1893 object->set_length(*length); | |
| 1894 break; | |
| 1895 } | |
| 1896 default: | |
| 1897 PrintF(stderr, | |
| 1898 "[couldn't handle instance type %d]\n", map->instance_type()); | |
| 1899 FATAL("Unsupported instance type"); | |
| 1900 } | |
| 1901 } | |
| 1902 | |
| 1903 return materialized_objects_->at(object_index); | |
| 1904 } | |
| 1905 | |
| 1906 | |
| 1907 Handle<Object> Deoptimizer::MaterializeNextValue() { | |
| 1908 int value_index = materialization_value_index_++; | |
| 1909 Handle<Object> value = materialized_values_->at(value_index); | |
| 1910 if (value->IsMutableHeapNumber()) { | |
| 1911 HeapNumber::cast(*value)->set_map(isolate_->heap()->heap_number_map()); | |
| 1912 } | |
| 1913 if (*value == isolate_->heap()->arguments_marker()) { | |
| 1914 value = MaterializeNextHeapObject(); | |
| 1915 } | |
| 1916 return value; | |
| 1917 } | |
| 1918 | |
| 1919 | |
| 1920 void Deoptimizer::MaterializeHeapObjects(JavaScriptFrameIterator* it) { | 1782 void Deoptimizer::MaterializeHeapObjects(JavaScriptFrameIterator* it) { |
| 1921 DCHECK_NE(DEBUGGER, bailout_type_); | 1783 DCHECK_NE(DEBUGGER, bailout_type_); |
| 1922 | 1784 |
| 1923 MaterializedObjectStore* materialized_store = | 1785 MaterializedObjectStore* materialized_store = |
| 1924 isolate_->materialized_object_store(); | 1786 isolate_->materialized_object_store(); |
| 1925 previously_materialized_objects_ = materialized_store->Get(stack_fp_); | 1787 previously_materialized_objects_ = materialized_store->Get(stack_fp_); |
| 1926 prev_materialized_count_ = previously_materialized_objects_.is_null() ? | 1788 prev_materialized_count_ = previously_materialized_objects_.is_null() ? |
| 1927 0 : previously_materialized_objects_->length(); | 1789 0 : previously_materialized_objects_->length(); |
| 1928 | 1790 |
| 1929 // Walk all JavaScript output frames with the given frame iterator. | 1791 // Walk all JavaScript output frames with the given frame iterator. |
| 1930 for (int frame_index = 0; frame_index < jsframe_count(); ++frame_index) { | 1792 for (int frame_index = 0; frame_index < jsframe_count(); ++frame_index) { |
| 1931 if (frame_index != 0) it->Advance(); | 1793 if (frame_index != 0) it->Advance(); |
| 1932 JavaScriptFrame* frame = it->frame(); | 1794 JavaScriptFrame* frame = it->frame(); |
| 1933 jsframe_functions_.Add(handle(frame->function(), isolate_)); | 1795 jsframe_functions_.Add(handle(frame->function(), isolate_)); |
| 1934 jsframe_has_adapted_arguments_.Add(frame->has_adapted_arguments()); | 1796 jsframe_has_adapted_arguments_.Add(frame->has_adapted_arguments()); |
| 1935 } | 1797 } |
| 1936 | 1798 |
| 1937 // Handlify all tagged object values before triggering any allocation. | 1799 translated_state_.Prepare(it->frame()->has_adapted_arguments(), stack_fp_); |
| 1938 List<Handle<Object> > values(deferred_objects_tagged_values_.length()); | 1800 |
| 1939 for (int i = 0; i < deferred_objects_tagged_values_.length(); ++i) { | 1801 for (auto& materialization : values_to_materialize_) { |
| 1940 values.Add(Handle<Object>(deferred_objects_tagged_values_[i], isolate_)); | 1802 Handle<Object> value = materialization.value_->GetValue(); |
| 1803 | |
| 1804 if (trace_scope_ != nullptr) { | |
| 1805 PrintF("Materialization [0x%08" V8PRIxPTR "] <- 0x%08" V8PRIxPTR " ; ", | |
| 1806 reinterpret_cast<intptr_t>(materialization.output_slot_address_), | |
| 1807 reinterpret_cast<intptr_t>(*value)); | |
| 1808 value->ShortPrint(trace_scope_->file()); | |
| 1809 PrintF(trace_scope_->file(), "\n"); | |
| 1810 } | |
| 1811 | |
| 1812 *(reinterpret_cast<intptr_t*>(materialization.output_slot_address_)) = | |
| 1813 reinterpret_cast<intptr_t>(*value); | |
| 1941 } | 1814 } |
| 1942 | 1815 |
| 1943 // Play it safe and clear all unhandlified values before we continue. | 1816 isolate_->materialized_object_store()->Remove(stack_fp_); |
| 1944 deferred_objects_tagged_values_.Clear(); | |
| 1945 | |
| 1946 // Materialize all heap numbers before looking at arguments because when the | |
| 1947 // output frames are used to materialize arguments objects later on they need | |
| 1948 // to already contain valid heap numbers. | |
| 1949 for (int i = 0; i < deferred_heap_numbers_.length(); i++) { | |
| 1950 HeapNumberMaterializationDescriptor<Address> d = deferred_heap_numbers_[i]; | |
| 1951 Handle<Object> num = isolate_->factory()->NewNumber(d.value()); | |
| 1952 if (trace_scope_ != NULL) { | |
| 1953 PrintF(trace_scope_->file(), | |
| 1954 "Materialized a new heap number %p [%e] in slot %p\n", | |
| 1955 reinterpret_cast<void*>(*num), | |
| 1956 d.value(), | |
| 1957 d.destination()); | |
| 1958 } | |
| 1959 Memory::Object_at(d.destination()) = *num; | |
| 1960 } | |
| 1961 | |
| 1962 // Materialize all heap numbers required for arguments/captured objects. | |
| 1963 for (int i = 0; i < deferred_objects_double_values_.length(); i++) { | |
| 1964 HeapNumberMaterializationDescriptor<int> d = | |
| 1965 deferred_objects_double_values_[i]; | |
| 1966 Handle<Object> num = isolate_->factory()->NewNumber(d.value()); | |
| 1967 if (trace_scope_ != NULL) { | |
| 1968 PrintF(trace_scope_->file(), | |
| 1969 "Materialized a new heap number %p [%e] for object at %d\n", | |
| 1970 reinterpret_cast<void*>(*num), | |
| 1971 d.value(), | |
| 1972 d.destination()); | |
| 1973 } | |
| 1974 DCHECK(values.at(d.destination())->IsTheHole()); | |
| 1975 values.Set(d.destination(), num); | |
| 1976 } | |
| 1977 | |
| 1978 // Play it safe and clear all object double values before we continue. | |
| 1979 deferred_objects_double_values_.Clear(); | |
| 1980 | |
| 1981 // Materialize arguments/captured objects. | |
| 1982 if (!deferred_objects_.is_empty()) { | |
| 1983 List<Handle<Object> > materialized_objects(deferred_objects_.length()); | |
| 1984 materialized_objects_ = &materialized_objects; | |
| 1985 materialized_values_ = &values; | |
| 1986 | |
| 1987 while (materialization_object_index_ < deferred_objects_.length()) { | |
| 1988 int object_index = materialization_object_index_; | |
| 1989 ObjectMaterializationDescriptor descriptor = | |
| 1990 deferred_objects_.at(object_index); | |
| 1991 | |
| 1992 // Find a previously materialized object by de-duplication or | |
| 1993 // materialize a new instance of the object if necessary. Store | |
| 1994 // the materialized object into the frame slot. | |
| 1995 Handle<Object> object = MaterializeNextHeapObject(); | |
| 1996 if (descriptor.slot_address() != NULL) { | |
| 1997 Memory::Object_at(descriptor.slot_address()) = *object; | |
| 1998 } | |
| 1999 if (trace_scope_ != NULL) { | |
| 2000 if (descriptor.is_arguments()) { | |
| 2001 PrintF(trace_scope_->file(), | |
| 2002 "Materialized %sarguments object of length %d for %p: ", | |
| 2003 ArgumentsObjectIsAdapted(object_index) ? "(adapted) " : "", | |
| 2004 Handle<JSObject>::cast(object)->elements()->length(), | |
| 2005 reinterpret_cast<void*>(descriptor.slot_address())); | |
| 2006 } else { | |
| 2007 PrintF(trace_scope_->file(), | |
| 2008 "Materialized captured object of size %d for %p: ", | |
| 2009 Handle<HeapObject>::cast(object)->Size(), | |
| 2010 reinterpret_cast<void*>(descriptor.slot_address())); | |
| 2011 } | |
| 2012 object->ShortPrint(trace_scope_->file()); | |
| 2013 PrintF(trace_scope_->file(), "\n"); | |
| 2014 } | |
| 2015 } | |
| 2016 | |
| 2017 CHECK_EQ(materialization_object_index_, materialized_objects_->length()); | |
| 2018 CHECK_EQ(materialization_value_index_, materialized_values_->length()); | |
| 2019 } | |
| 2020 | |
| 2021 if (prev_materialized_count_ > 0) { | |
| 2022 bool removed = materialized_store->Remove(stack_fp_); | |
| 2023 CHECK(removed); | |
| 2024 } | |
| 2025 } | 1817 } |
| 2026 | 1818 |
| 2027 | 1819 |
| 2028 void Deoptimizer::MaterializeHeapNumbersForDebuggerInspectableFrame( | 1820 void Deoptimizer::MaterializeHeapNumbersForDebuggerInspectableFrame( |
| 2029 Address parameters_top, | 1821 int frame_index, int parameter_count, int expression_count, |
| 2030 uint32_t parameters_size, | |
| 2031 Address expressions_top, | |
| 2032 uint32_t expressions_size, | |
| 2033 DeoptimizedFrameInfo* info) { | 1822 DeoptimizedFrameInfo* info) { |
| 2034 CHECK_EQ(DEBUGGER, bailout_type_); | 1823 CHECK_EQ(DEBUGGER, bailout_type_); |
| 2035 Address parameters_bottom = parameters_top + parameters_size; | |
| 2036 Address expressions_bottom = expressions_top + expressions_size; | |
| 2037 for (int i = 0; i < deferred_heap_numbers_.length(); i++) { | |
| 2038 HeapNumberMaterializationDescriptor<Address> d = deferred_heap_numbers_[i]; | |
| 2039 | 1824 |
| 2040 // Check of the heap number to materialize actually belong to the frame | 1825 translated_state_.Prepare(false, nullptr); |
| 2041 // being extracted. | |
| 2042 Address slot = d.destination(); | |
| 2043 if (parameters_top <= slot && slot < parameters_bottom) { | |
| 2044 Handle<Object> num = isolate_->factory()->NewNumber(d.value()); | |
| 2045 | 1826 |
| 2046 int index = (info->parameters_count() - 1) - | 1827 TranslatedFrame* frame = &(translated_state_.frames()[frame_index]); |
| 2047 static_cast<int>(slot - parameters_top) / kPointerSize; | 1828 CHECK(frame->kind() == TranslatedFrame::Function); |
| 1829 int frame_arg_count = | |
| 1830 frame->function()->shared()->internal_formal_parameter_count(); | |
| 2048 | 1831 |
| 2049 if (trace_scope_ != NULL) { | 1832 // The height is #expressions + 1 for context. |
| 2050 PrintF(trace_scope_->file(), | 1833 CHECK_EQ(expression_count + 1, frame->height()); |
| 2051 "Materializing a new heap number %p [%e] in slot %p" | 1834 TranslatedFrame* argument_frame = frame; |
| 2052 "for parameter slot #%d\n", | 1835 if (frame_index > 0) { |
| 2053 reinterpret_cast<void*>(*num), | 1836 TranslatedFrame* previous_frame = |
| 2054 d.value(), | 1837 &(translated_state_.frames()[frame_index - 1]); |
| 2055 d.destination(), | 1838 if (previous_frame->kind() == TranslatedFrame::ArgumentsAdaptor) { |
| 2056 index); | 1839 argument_frame = previous_frame; |
| 2057 } | 1840 CHECK_EQ(parameter_count, argument_frame->height() - 1); |
| 1841 } else { | |
| 1842 CHECK_EQ(frame_arg_count, parameter_count); | |
| 1843 } | |
| 1844 } else { | |
| 1845 CHECK_EQ(frame_arg_count, parameter_count); | |
| 1846 } | |
| 2058 | 1847 |
| 2059 info->SetParameter(index, *num); | 1848 TranslatedFrame::ValueIterator arg_iter = argument_frame->begin(); |
| 2060 } else if (expressions_top <= slot && slot < expressions_bottom) { | 1849 arg_iter++; // Skip the receiver. |
| 2061 Handle<Object> num = isolate_->factory()->NewNumber(d.value()); | 1850 for (int i = 0; i < parameter_count; i++, arg_iter++) { |
| 1851 if (!arg_iter->IsMaterializedObject()) { | |
| 1852 info->SetParameter(i, *(arg_iter->GetValue())); | |
| 1853 } | |
| 1854 } | |
| 2062 | 1855 |
| 2063 int index = info->expression_count() - 1 - | 1856 TranslatedFrame::ValueIterator iter = frame->begin(); |
| 2064 static_cast<int>(slot - expressions_top) / kPointerSize; | 1857 // Skip the arguments, receiver and context. |
| 1858 for (int i = 0; i < frame_arg_count + 2; i++, iter++) { | |
| 1859 } | |
| 2065 | 1860 |
| 2066 if (trace_scope_ != NULL) { | 1861 for (int i = 0; i < expression_count; i++, iter++) { |
| 2067 PrintF(trace_scope_->file(), | 1862 if (!iter->IsMaterializedObject()) { |
| 2068 "Materializing a new heap number %p [%e] in slot %p" | 1863 info->SetExpression(i, *(iter->GetValue())); |
| 2069 "for expression slot #%d\n", | |
| 2070 reinterpret_cast<void*>(*num), | |
| 2071 d.value(), | |
| 2072 d.destination(), | |
| 2073 index); | |
| 2074 } | |
| 2075 | |
| 2076 info->SetExpression(index, *num); | |
| 2077 } | 1864 } |
| 2078 } | 1865 } |
| 2079 } | 1866 } |
| 2080 | 1867 |
| 2081 | 1868 |
| 2082 static const char* TraceValueType(bool is_smi) { | 1869 void Deoptimizer::WriteValueToOutput( |
| 2083 if (is_smi) { | 1870 TranslatedFrame::ValueIterator* iterator, int* input_index, int frame_index, |
| 2084 return "smi"; | 1871 unsigned output_offset, Address output_address_for_materialization) { |
| 1872 Object* value = (*iterator)->GetRawValue(); | |
| 1873 output_[frame_index]->SetFrameSlot(output_offset, | |
| 1874 reinterpret_cast<intptr_t>(value)); | |
| 1875 | |
| 1876 Address output_address = | |
| 1877 reinterpret_cast<Address>(output_[frame_index]->GetTop()) + output_offset; | |
| 1878 if (trace_scope_ != nullptr) { | |
| 1879 PrintF(trace_scope_->file(), | |
| 1880 " 0x%08" V8PRIxPTR ": [top + %d] <- 0x%08" V8PRIxPTR " ; ", | |
| 1881 reinterpret_cast<intptr_t>(output_address), output_offset, | |
| 1882 reinterpret_cast<intptr_t>(value)); | |
| 1883 value->ShortPrint(trace_scope_->file()); | |
| 1884 PrintF(trace_scope_->file(), " (input #%d)\n", *input_index); | |
| 2085 } | 1885 } |
| 2086 | 1886 |
| 2087 return "heap number"; | 1887 if (value == isolate_->heap()->arguments_marker()) { |
| 2088 } | 1888 if (output_address_for_materialization == nullptr) { |
| 2089 | 1889 output_address_for_materialization = output_address; |
| 2090 | |
| 2091 void Deoptimizer::DoTranslateObjectAndSkip(TranslationIterator* iterator) { | |
| 2092 Translation::Opcode opcode = | |
| 2093 static_cast<Translation::Opcode>(iterator->Next()); | |
| 2094 | |
| 2095 switch (opcode) { | |
| 2096 case Translation::BEGIN: | |
| 2097 case Translation::JS_FRAME: | |
| 2098 case Translation::ARGUMENTS_ADAPTOR_FRAME: | |
| 2099 case Translation::CONSTRUCT_STUB_FRAME: | |
| 2100 case Translation::GETTER_STUB_FRAME: | |
| 2101 case Translation::SETTER_STUB_FRAME: | |
| 2102 case Translation::COMPILED_STUB_FRAME: { | |
| 2103 FATAL("Unexpected frame start translation opcode"); | |
| 2104 return; | |
| 2105 } | 1890 } |
| 2106 | 1891 values_to_materialize_.push_back( |
| 2107 case Translation::REGISTER: | 1892 {output_address_for_materialization, *iterator}); |
| 2108 case Translation::INT32_REGISTER: | |
| 2109 case Translation::UINT32_REGISTER: | |
| 2110 case Translation::BOOL_REGISTER: | |
| 2111 case Translation::DOUBLE_REGISTER: | |
| 2112 case Translation::STACK_SLOT: | |
| 2113 case Translation::INT32_STACK_SLOT: | |
| 2114 case Translation::UINT32_STACK_SLOT: | |
| 2115 case Translation::BOOL_STACK_SLOT: | |
| 2116 case Translation::DOUBLE_STACK_SLOT: | |
| 2117 case Translation::LITERAL: { | |
| 2118 // The value is not part of any materialized object, so we can ignore it. | |
| 2119 iterator->Skip(Translation::NumberOfOperandsFor(opcode)); | |
| 2120 return; | |
| 2121 } | |
| 2122 | |
| 2123 case Translation::DUPLICATED_OBJECT: { | |
| 2124 int object_index = iterator->Next(); | |
| 2125 if (trace_scope_ != NULL) { | |
| 2126 PrintF(trace_scope_->file(), " skipping object "); | |
| 2127 PrintF(trace_scope_->file(), | |
| 2128 " ; duplicate of object #%d\n", object_index); | |
| 2129 } | |
| 2130 AddObjectDuplication(0, object_index); | |
| 2131 return; | |
| 2132 } | |
| 2133 | |
| 2134 case Translation::ARGUMENTS_OBJECT: | |
| 2135 case Translation::CAPTURED_OBJECT: { | |
| 2136 int length = iterator->Next(); | |
| 2137 bool is_args = opcode == Translation::ARGUMENTS_OBJECT; | |
| 2138 if (trace_scope_ != NULL) { | |
| 2139 PrintF(trace_scope_->file(), " skipping object "); | |
| 2140 PrintF(trace_scope_->file(), | |
| 2141 " ; object (length = %d, is_args = %d)\n", length, is_args); | |
| 2142 } | |
| 2143 | |
| 2144 AddObjectStart(0, length, is_args); | |
| 2145 | |
| 2146 // We save the object values on the side and materialize the actual | |
| 2147 // object after the deoptimized frame is built. | |
| 2148 int object_index = deferred_objects_.length() - 1; | |
| 2149 for (int i = 0; i < length; i++) { | |
| 2150 DoTranslateObject(iterator, object_index, i); | |
| 2151 } | |
| 2152 return; | |
| 2153 } | |
| 2154 } | 1893 } |
| 2155 | 1894 |
| 2156 FATAL("Unexpected translation opcode"); | 1895 (*iterator)++; |
| 2157 } | 1896 (*input_index)++; |
| 2158 | |
| 2159 | |
| 2160 void Deoptimizer::DoTranslateObject(TranslationIterator* iterator, | |
| 2161 int object_index, | |
| 2162 int field_index) { | |
| 2163 disasm::NameConverter converter; | |
| 2164 Address object_slot = deferred_objects_[object_index].slot_address(); | |
| 2165 | |
| 2166 Translation::Opcode opcode = | |
| 2167 static_cast<Translation::Opcode>(iterator->Next()); | |
| 2168 | |
| 2169 switch (opcode) { | |
| 2170 case Translation::BEGIN: | |
| 2171 case Translation::JS_FRAME: | |
| 2172 case Translation::ARGUMENTS_ADAPTOR_FRAME: | |
| 2173 case Translation::CONSTRUCT_STUB_FRAME: | |
| 2174 case Translation::GETTER_STUB_FRAME: | |
| 2175 case Translation::SETTER_STUB_FRAME: | |
| 2176 case Translation::COMPILED_STUB_FRAME: | |
| 2177 FATAL("Unexpected frame start translation opcode"); | |
| 2178 return; | |
| 2179 | |
| 2180 case Translation::REGISTER: { | |
| 2181 int input_reg = iterator->Next(); | |
| 2182 intptr_t input_value = input_->GetRegister(input_reg); | |
| 2183 if (trace_scope_ != NULL) { | |
| 2184 PrintF(trace_scope_->file(), | |
| 2185 " object @0x%08" V8PRIxPTR ": [field #%d] <- ", | |
| 2186 reinterpret_cast<intptr_t>(object_slot), | |
| 2187 field_index); | |
| 2188 PrintF(trace_scope_->file(), | |
| 2189 "0x%08" V8PRIxPTR " ; %s ", input_value, | |
| 2190 converter.NameOfCPURegister(input_reg)); | |
| 2191 reinterpret_cast<Object*>(input_value)->ShortPrint( | |
| 2192 trace_scope_->file()); | |
| 2193 PrintF(trace_scope_->file(), | |
| 2194 "\n"); | |
| 2195 } | |
| 2196 AddObjectTaggedValue(input_value); | |
| 2197 return; | |
| 2198 } | |
| 2199 | |
| 2200 case Translation::INT32_REGISTER: { | |
| 2201 int input_reg = iterator->Next(); | |
| 2202 intptr_t value = input_->GetRegister(input_reg); | |
| 2203 bool is_smi = Smi::IsValid(value); | |
| 2204 if (trace_scope_ != NULL) { | |
| 2205 PrintF(trace_scope_->file(), | |
| 2206 " object @0x%08" V8PRIxPTR ": [field #%d] <- ", | |
| 2207 reinterpret_cast<intptr_t>(object_slot), | |
| 2208 field_index); | |
| 2209 PrintF(trace_scope_->file(), | |
| 2210 "%" V8PRIdPTR " ; %s (%s)\n", value, | |
| 2211 converter.NameOfCPURegister(input_reg), | |
| 2212 TraceValueType(is_smi)); | |
| 2213 } | |
| 2214 if (is_smi) { | |
| 2215 intptr_t tagged_value = | |
| 2216 reinterpret_cast<intptr_t>(Smi::FromInt(static_cast<int>(value))); | |
| 2217 AddObjectTaggedValue(tagged_value); | |
| 2218 } else { | |
| 2219 double double_value = static_cast<double>(static_cast<int32_t>(value)); | |
| 2220 AddObjectDoubleValue(double_value); | |
| 2221 } | |
| 2222 return; | |
| 2223 } | |
| 2224 | |
| 2225 case Translation::UINT32_REGISTER: { | |
| 2226 int input_reg = iterator->Next(); | |
| 2227 uintptr_t value = static_cast<uintptr_t>(input_->GetRegister(input_reg)); | |
| 2228 bool is_smi = (value <= static_cast<uintptr_t>(Smi::kMaxValue)); | |
| 2229 if (trace_scope_ != NULL) { | |
| 2230 PrintF(trace_scope_->file(), | |
| 2231 " object @0x%08" V8PRIxPTR ": [field #%d] <- ", | |
| 2232 reinterpret_cast<intptr_t>(object_slot), | |
| 2233 field_index); | |
| 2234 PrintF(trace_scope_->file(), "%" V8PRIuPTR " ; uint %s (%s)\n", value, | |
| 2235 converter.NameOfCPURegister(input_reg), TraceValueType(is_smi)); | |
| 2236 } | |
| 2237 if (is_smi) { | |
| 2238 intptr_t tagged_value = | |
| 2239 reinterpret_cast<intptr_t>(Smi::FromInt(static_cast<int>(value))); | |
| 2240 AddObjectTaggedValue(tagged_value); | |
| 2241 } else { | |
| 2242 double double_value = static_cast<double>(static_cast<uint32_t>(value)); | |
| 2243 AddObjectDoubleValue(double_value); | |
| 2244 } | |
| 2245 return; | |
| 2246 } | |
| 2247 | |
| 2248 case Translation::BOOL_REGISTER: { | |
| 2249 int input_reg = iterator->Next(); | |
| 2250 uintptr_t value = static_cast<uintptr_t>(input_->GetRegister(input_reg)); | |
| 2251 bool is_smi = (value <= static_cast<uintptr_t>(Smi::kMaxValue)); | |
| 2252 if (trace_scope_ != NULL) { | |
| 2253 PrintF(trace_scope_->file(), | |
| 2254 " object @0x%08" V8PRIxPTR ": [field #%d] <- ", | |
| 2255 reinterpret_cast<intptr_t>(object_slot), field_index); | |
| 2256 PrintF(trace_scope_->file(), "%" V8PRIuPTR " ; bool %s (%s)\n", value, | |
| 2257 converter.NameOfCPURegister(input_reg), TraceValueType(is_smi)); | |
| 2258 } | |
| 2259 if (value == 0) { | |
| 2260 AddObjectTaggedValue( | |
| 2261 reinterpret_cast<intptr_t>(isolate_->heap()->false_value())); | |
| 2262 } else { | |
| 2263 DCHECK_EQ(1U, value); | |
| 2264 AddObjectTaggedValue( | |
| 2265 reinterpret_cast<intptr_t>(isolate_->heap()->true_value())); | |
| 2266 } | |
| 2267 return; | |
| 2268 } | |
| 2269 | |
| 2270 case Translation::DOUBLE_REGISTER: { | |
| 2271 int input_reg = iterator->Next(); | |
| 2272 double value = input_->GetDoubleRegister(input_reg); | |
| 2273 int int_value = FastD2IChecked(value); | |
| 2274 bool is_smi = | |
| 2275 !IsMinusZero(value) && value == int_value && Smi::IsValid(int_value); | |
| 2276 if (trace_scope_ != NULL) { | |
| 2277 PrintF(trace_scope_->file(), | |
| 2278 " object @0x%08" V8PRIxPTR ": [field #%d] <- ", | |
| 2279 reinterpret_cast<intptr_t>(object_slot), | |
| 2280 field_index); | |
| 2281 PrintF(trace_scope_->file(), | |
| 2282 "%e ; %s\n", value, | |
| 2283 DoubleRegister::AllocationIndexToString(input_reg)); | |
| 2284 } | |
| 2285 if (is_smi) { | |
| 2286 intptr_t tagged_value = | |
| 2287 reinterpret_cast<intptr_t>(Smi::FromInt(int_value)); | |
| 2288 AddObjectTaggedValue(tagged_value); | |
| 2289 } else { | |
| 2290 AddObjectDoubleValue(value); | |
| 2291 } | |
| 2292 return; | |
| 2293 } | |
| 2294 | |
| 2295 case Translation::STACK_SLOT: { | |
| 2296 int input_slot_index = iterator->Next(); | |
| 2297 unsigned input_offset = input_->GetOffsetFromSlotIndex(input_slot_index); | |
| 2298 intptr_t input_value = input_->GetFrameSlot(input_offset); | |
| 2299 if (trace_scope_ != NULL) { | |
| 2300 PrintF(trace_scope_->file(), | |
| 2301 " object @0x%08" V8PRIxPTR ": [field #%d] <- ", | |
| 2302 reinterpret_cast<intptr_t>(object_slot), | |
| 2303 field_index); | |
| 2304 PrintF(trace_scope_->file(), | |
| 2305 "0x%08" V8PRIxPTR " ; [sp + %d] ", input_value, input_offset); | |
| 2306 reinterpret_cast<Object*>(input_value)->ShortPrint( | |
| 2307 trace_scope_->file()); | |
| 2308 PrintF(trace_scope_->file(), | |
| 2309 "\n"); | |
| 2310 } | |
| 2311 AddObjectTaggedValue(input_value); | |
| 2312 return; | |
| 2313 } | |
| 2314 | |
| 2315 case Translation::INT32_STACK_SLOT: { | |
| 2316 int input_slot_index = iterator->Next(); | |
| 2317 unsigned input_offset = input_->GetOffsetFromSlotIndex(input_slot_index); | |
| 2318 intptr_t value = input_->GetFrameSlot(input_offset); | |
| 2319 bool is_smi = Smi::IsValid(value); | |
| 2320 if (trace_scope_ != NULL) { | |
| 2321 PrintF(trace_scope_->file(), | |
| 2322 " object @0x%08" V8PRIxPTR ": [field #%d] <- ", | |
| 2323 reinterpret_cast<intptr_t>(object_slot), | |
| 2324 field_index); | |
| 2325 PrintF(trace_scope_->file(), | |
| 2326 "%" V8PRIdPTR " ; [sp + %d] (%s)\n", | |
| 2327 value, input_offset, TraceValueType(is_smi)); | |
| 2328 } | |
| 2329 if (is_smi) { | |
| 2330 intptr_t tagged_value = | |
| 2331 reinterpret_cast<intptr_t>(Smi::FromInt(static_cast<int>(value))); | |
| 2332 AddObjectTaggedValue(tagged_value); | |
| 2333 } else { | |
| 2334 double double_value = static_cast<double>(static_cast<int32_t>(value)); | |
| 2335 AddObjectDoubleValue(double_value); | |
| 2336 } | |
| 2337 return; | |
| 2338 } | |
| 2339 | |
| 2340 case Translation::UINT32_STACK_SLOT: { | |
| 2341 int input_slot_index = iterator->Next(); | |
| 2342 unsigned input_offset = input_->GetOffsetFromSlotIndex(input_slot_index); | |
| 2343 uintptr_t value = | |
| 2344 static_cast<uintptr_t>(input_->GetFrameSlot(input_offset)); | |
| 2345 bool is_smi = (value <= static_cast<uintptr_t>(Smi::kMaxValue)); | |
| 2346 if (trace_scope_ != NULL) { | |
| 2347 PrintF(trace_scope_->file(), | |
| 2348 " object @0x%08" V8PRIxPTR ": [field #%d] <- ", | |
| 2349 reinterpret_cast<intptr_t>(object_slot), | |
| 2350 field_index); | |
| 2351 PrintF(trace_scope_->file(), "%" V8PRIuPTR " ; [sp + %d] (uint %s)\n", | |
| 2352 value, input_offset, TraceValueType(is_smi)); | |
| 2353 } | |
| 2354 if (is_smi) { | |
| 2355 intptr_t tagged_value = | |
| 2356 reinterpret_cast<intptr_t>(Smi::FromInt(static_cast<int>(value))); | |
| 2357 AddObjectTaggedValue(tagged_value); | |
| 2358 } else { | |
| 2359 double double_value = static_cast<double>(static_cast<uint32_t>(value)); | |
| 2360 AddObjectDoubleValue(double_value); | |
| 2361 } | |
| 2362 return; | |
| 2363 } | |
| 2364 | |
| 2365 case Translation::BOOL_STACK_SLOT: { | |
| 2366 int input_slot_index = iterator->Next(); | |
| 2367 unsigned input_offset = input_->GetOffsetFromSlotIndex(input_slot_index); | |
| 2368 uintptr_t value = | |
| 2369 static_cast<uintptr_t>(input_->GetFrameSlot(input_offset)); | |
| 2370 bool is_smi = (value <= static_cast<uintptr_t>(Smi::kMaxValue)); | |
| 2371 if (trace_scope_ != NULL) { | |
| 2372 PrintF(trace_scope_->file(), | |
| 2373 " object @0x%08" V8PRIxPTR ": [field #%d] <- ", | |
| 2374 reinterpret_cast<intptr_t>(object_slot), field_index); | |
| 2375 PrintF(trace_scope_->file(), "%" V8PRIuPTR " ; [sp + %d] (bool %s)\n", | |
| 2376 value, input_offset, TraceValueType(is_smi)); | |
| 2377 } | |
| 2378 if (value == 0) { | |
| 2379 AddObjectTaggedValue( | |
| 2380 reinterpret_cast<intptr_t>(isolate_->heap()->false_value())); | |
| 2381 } else { | |
| 2382 DCHECK_EQ(1U, value); | |
| 2383 AddObjectTaggedValue( | |
| 2384 reinterpret_cast<intptr_t>(isolate_->heap()->true_value())); | |
| 2385 } | |
| 2386 return; | |
| 2387 } | |
| 2388 | |
| 2389 case Translation::DOUBLE_STACK_SLOT: { | |
| 2390 int input_slot_index = iterator->Next(); | |
| 2391 unsigned input_offset = input_->GetOffsetFromSlotIndex(input_slot_index); | |
| 2392 double value = input_->GetDoubleFrameSlot(input_offset); | |
| 2393 int int_value = FastD2IChecked(value); | |
| 2394 bool is_smi = | |
| 2395 !IsMinusZero(value) && value == int_value && Smi::IsValid(int_value); | |
| 2396 if (trace_scope_ != NULL) { | |
| 2397 PrintF(trace_scope_->file(), | |
| 2398 " object @0x%08" V8PRIxPTR ": [field #%d] <- ", | |
| 2399 reinterpret_cast<intptr_t>(object_slot), | |
| 2400 field_index); | |
| 2401 PrintF(trace_scope_->file(), | |
| 2402 "%e ; [sp + %d]\n", value, input_offset); | |
| 2403 } | |
| 2404 if (is_smi) { | |
| 2405 intptr_t tagged_value = | |
| 2406 reinterpret_cast<intptr_t>(Smi::FromInt(int_value)); | |
| 2407 AddObjectTaggedValue(tagged_value); | |
| 2408 } else { | |
| 2409 AddObjectDoubleValue(value); | |
| 2410 } | |
| 2411 return; | |
| 2412 } | |
| 2413 | |
| 2414 case Translation::LITERAL: { | |
| 2415 Object* literal = ComputeLiteral(iterator->Next()); | |
| 2416 if (trace_scope_ != NULL) { | |
| 2417 PrintF(trace_scope_->file(), | |
| 2418 " object @0x%08" V8PRIxPTR ": [field #%d] <- ", | |
| 2419 reinterpret_cast<intptr_t>(object_slot), | |
| 2420 field_index); | |
| 2421 literal->ShortPrint(trace_scope_->file()); | |
| 2422 PrintF(trace_scope_->file(), | |
| 2423 " ; literal\n"); | |
| 2424 } | |
| 2425 intptr_t value = reinterpret_cast<intptr_t>(literal); | |
| 2426 AddObjectTaggedValue(value); | |
| 2427 return; | |
| 2428 } | |
| 2429 | |
| 2430 case Translation::DUPLICATED_OBJECT: { | |
| 2431 int object_index = iterator->Next(); | |
| 2432 if (trace_scope_ != NULL) { | |
| 2433 PrintF(trace_scope_->file(), | |
| 2434 " nested @0x%08" V8PRIxPTR ": [field #%d] <- ", | |
| 2435 reinterpret_cast<intptr_t>(object_slot), | |
| 2436 field_index); | |
| 2437 isolate_->heap()->arguments_marker()->ShortPrint(trace_scope_->file()); | |
| 2438 PrintF(trace_scope_->file(), | |
| 2439 " ; duplicate of object #%d\n", object_index); | |
| 2440 } | |
| 2441 // Use the materialization marker value as a sentinel and fill in | |
| 2442 // the object after the deoptimized frame is built. | |
| 2443 intptr_t value = reinterpret_cast<intptr_t>( | |
| 2444 isolate_->heap()->arguments_marker()); | |
| 2445 AddObjectDuplication(0, object_index); | |
| 2446 AddObjectTaggedValue(value); | |
| 2447 return; | |
| 2448 } | |
| 2449 | |
| 2450 case Translation::ARGUMENTS_OBJECT: | |
| 2451 case Translation::CAPTURED_OBJECT: { | |
| 2452 int length = iterator->Next(); | |
| 2453 bool is_args = opcode == Translation::ARGUMENTS_OBJECT; | |
| 2454 if (trace_scope_ != NULL) { | |
| 2455 PrintF(trace_scope_->file(), | |
| 2456 " nested @0x%08" V8PRIxPTR ": [field #%d] <- ", | |
| 2457 reinterpret_cast<intptr_t>(object_slot), | |
| 2458 field_index); | |
| 2459 isolate_->heap()->arguments_marker()->ShortPrint(trace_scope_->file()); | |
| 2460 PrintF(trace_scope_->file(), | |
| 2461 " ; object (length = %d, is_args = %d)\n", length, is_args); | |
| 2462 } | |
| 2463 // Use the materialization marker value as a sentinel and fill in | |
| 2464 // the object after the deoptimized frame is built. | |
| 2465 intptr_t value = reinterpret_cast<intptr_t>( | |
| 2466 isolate_->heap()->arguments_marker()); | |
| 2467 AddObjectStart(0, length, is_args); | |
| 2468 AddObjectTaggedValue(value); | |
| 2469 // We save the object values on the side and materialize the actual | |
| 2470 // object after the deoptimized frame is built. | |
| 2471 int object_index = deferred_objects_.length() - 1; | |
| 2472 for (int i = 0; i < length; i++) { | |
| 2473 DoTranslateObject(iterator, object_index, i); | |
| 2474 } | |
| 2475 return; | |
| 2476 } | |
| 2477 } | |
| 2478 | |
| 2479 FATAL("Unexpected translation opcode"); | |
| 2480 } | |
| 2481 | |
| 2482 | |
| 2483 void Deoptimizer::DoTranslateCommand(TranslationIterator* iterator, | |
| 2484 int frame_index, | |
| 2485 unsigned output_offset) { | |
| 2486 disasm::NameConverter converter; | |
| 2487 // A GC-safe temporary placeholder that we can put in the output frame. | |
| 2488 const intptr_t kPlaceholder = reinterpret_cast<intptr_t>(Smi::FromInt(0)); | |
| 2489 | |
| 2490 Translation::Opcode opcode = | |
| 2491 static_cast<Translation::Opcode>(iterator->Next()); | |
| 2492 | |
| 2493 switch (opcode) { | |
| 2494 case Translation::BEGIN: | |
| 2495 case Translation::JS_FRAME: | |
| 2496 case Translation::ARGUMENTS_ADAPTOR_FRAME: | |
| 2497 case Translation::CONSTRUCT_STUB_FRAME: | |
| 2498 case Translation::GETTER_STUB_FRAME: | |
| 2499 case Translation::SETTER_STUB_FRAME: | |
| 2500 case Translation::COMPILED_STUB_FRAME: | |
| 2501 FATAL("Unexpected translation opcode"); | |
| 2502 return; | |
| 2503 | |
| 2504 case Translation::REGISTER: { | |
| 2505 int input_reg = iterator->Next(); | |
| 2506 intptr_t input_value = input_->GetRegister(input_reg); | |
| 2507 if (trace_scope_ != NULL) { | |
| 2508 PrintF( | |
| 2509 trace_scope_->file(), | |
| 2510 " 0x%08" V8PRIxPTR ": [top + %d] <- 0x%08" V8PRIxPTR " ; %s ", | |
| 2511 output_[frame_index]->GetTop() + output_offset, | |
| 2512 output_offset, | |
| 2513 input_value, | |
| 2514 converter.NameOfCPURegister(input_reg)); | |
| 2515 reinterpret_cast<Object*>(input_value)->ShortPrint( | |
| 2516 trace_scope_->file()); | |
| 2517 PrintF(trace_scope_->file(), "\n"); | |
| 2518 } | |
| 2519 output_[frame_index]->SetFrameSlot(output_offset, input_value); | |
| 2520 return; | |
| 2521 } | |
| 2522 | |
| 2523 case Translation::INT32_REGISTER: { | |
| 2524 int input_reg = iterator->Next(); | |
| 2525 intptr_t value = input_->GetRegister(input_reg); | |
| 2526 bool is_smi = Smi::IsValid(value); | |
| 2527 if (trace_scope_ != NULL) { | |
| 2528 PrintF( | |
| 2529 trace_scope_->file(), | |
| 2530 " 0x%08" V8PRIxPTR ": [top + %d] <- %" V8PRIdPTR " ; %s (%s)\n", | |
| 2531 output_[frame_index]->GetTop() + output_offset, | |
| 2532 output_offset, | |
| 2533 value, | |
| 2534 converter.NameOfCPURegister(input_reg), | |
| 2535 TraceValueType(is_smi)); | |
| 2536 } | |
| 2537 if (is_smi) { | |
| 2538 intptr_t tagged_value = | |
| 2539 reinterpret_cast<intptr_t>(Smi::FromInt(static_cast<int>(value))); | |
| 2540 output_[frame_index]->SetFrameSlot(output_offset, tagged_value); | |
| 2541 } else { | |
| 2542 // We save the untagged value on the side and store a GC-safe | |
| 2543 // temporary placeholder in the frame. | |
| 2544 AddDoubleValue(output_[frame_index]->GetTop() + output_offset, | |
| 2545 static_cast<double>(static_cast<int32_t>(value))); | |
| 2546 output_[frame_index]->SetFrameSlot(output_offset, kPlaceholder); | |
| 2547 } | |
| 2548 return; | |
| 2549 } | |
| 2550 | |
| 2551 case Translation::UINT32_REGISTER: { | |
| 2552 int input_reg = iterator->Next(); | |
| 2553 uintptr_t value = static_cast<uintptr_t>(input_->GetRegister(input_reg)); | |
| 2554 bool is_smi = value <= static_cast<uintptr_t>(Smi::kMaxValue); | |
| 2555 if (trace_scope_ != NULL) { | |
| 2556 PrintF( | |
| 2557 trace_scope_->file(), | |
| 2558 " 0x%08" V8PRIxPTR ": [top + %d] <- %" V8PRIuPTR | |
| 2559 " ; uint %s (%s)\n", | |
| 2560 output_[frame_index]->GetTop() + output_offset, | |
| 2561 output_offset, | |
| 2562 value, | |
| 2563 converter.NameOfCPURegister(input_reg), | |
| 2564 TraceValueType(is_smi)); | |
| 2565 } | |
| 2566 if (is_smi) { | |
| 2567 intptr_t tagged_value = | |
| 2568 reinterpret_cast<intptr_t>(Smi::FromInt(static_cast<int>(value))); | |
| 2569 output_[frame_index]->SetFrameSlot(output_offset, tagged_value); | |
| 2570 } else { | |
| 2571 // We save the untagged value on the side and store a GC-safe | |
| 2572 // temporary placeholder in the frame. | |
| 2573 AddDoubleValue(output_[frame_index]->GetTop() + output_offset, | |
| 2574 static_cast<double>(static_cast<uint32_t>(value))); | |
| 2575 output_[frame_index]->SetFrameSlot(output_offset, kPlaceholder); | |
| 2576 } | |
| 2577 return; | |
| 2578 } | |
| 2579 | |
| 2580 case Translation::BOOL_REGISTER: { | |
| 2581 int input_reg = iterator->Next(); | |
| 2582 uintptr_t value = static_cast<uintptr_t>(input_->GetRegister(input_reg)); | |
| 2583 bool is_smi = value <= static_cast<uintptr_t>(Smi::kMaxValue); | |
| 2584 if (trace_scope_ != NULL) { | |
| 2585 PrintF(trace_scope_->file(), | |
| 2586 " 0x%08" V8PRIxPTR ": [top + %d] <- %" V8PRIuPTR | |
| 2587 " ; bool %s (%s)\n", | |
| 2588 output_[frame_index]->GetTop() + output_offset, output_offset, | |
| 2589 value, converter.NameOfCPURegister(input_reg), | |
| 2590 TraceValueType(is_smi)); | |
| 2591 } | |
| 2592 if (value == 0) { | |
| 2593 output_[frame_index]->SetFrameSlot( | |
| 2594 output_offset, | |
| 2595 reinterpret_cast<intptr_t>(isolate_->heap()->false_value())); | |
| 2596 } else { | |
| 2597 DCHECK_EQ(1U, value); | |
| 2598 output_[frame_index]->SetFrameSlot( | |
| 2599 output_offset, | |
| 2600 reinterpret_cast<intptr_t>(isolate_->heap()->true_value())); | |
| 2601 } | |
| 2602 return; | |
| 2603 } | |
| 2604 | |
| 2605 case Translation::DOUBLE_REGISTER: { | |
| 2606 int input_reg = iterator->Next(); | |
| 2607 double value = input_->GetDoubleRegister(input_reg); | |
| 2608 int int_value = FastD2IChecked(value); | |
| 2609 bool is_smi = | |
| 2610 !IsMinusZero(value) && value == int_value && Smi::IsValid(int_value); | |
| 2611 if (trace_scope_ != NULL) { | |
| 2612 PrintF(trace_scope_->file(), | |
| 2613 " 0x%08" V8PRIxPTR ": [top + %d] <- %e ; %s\n", | |
| 2614 output_[frame_index]->GetTop() + output_offset, output_offset, | |
| 2615 value, DoubleRegister::AllocationIndexToString(input_reg)); | |
| 2616 } | |
| 2617 if (is_smi) { | |
| 2618 intptr_t tagged_value = | |
| 2619 reinterpret_cast<intptr_t>(Smi::FromInt(int_value)); | |
| 2620 output_[frame_index]->SetFrameSlot(output_offset, tagged_value); | |
| 2621 } else { | |
| 2622 // We save the untagged value on the side and store a GC-safe | |
| 2623 // temporary placeholder in the frame. | |
| 2624 AddDoubleValue(output_[frame_index]->GetTop() + output_offset, value); | |
| 2625 output_[frame_index]->SetFrameSlot(output_offset, kPlaceholder); | |
| 2626 } | |
| 2627 return; | |
| 2628 } | |
| 2629 | |
| 2630 case Translation::STACK_SLOT: { | |
| 2631 int input_slot_index = iterator->Next(); | |
| 2632 unsigned input_offset = input_->GetOffsetFromSlotIndex(input_slot_index); | |
| 2633 intptr_t input_value = input_->GetFrameSlot(input_offset); | |
| 2634 if (trace_scope_ != NULL) { | |
| 2635 PrintF(trace_scope_->file(), | |
| 2636 " 0x%08" V8PRIxPTR ": ", | |
| 2637 output_[frame_index]->GetTop() + output_offset); | |
| 2638 PrintF(trace_scope_->file(), | |
| 2639 "[top + %d] <- 0x%08" V8PRIxPTR " ; [sp + %d] ", | |
| 2640 output_offset, | |
| 2641 input_value, | |
| 2642 input_offset); | |
| 2643 reinterpret_cast<Object*>(input_value)->ShortPrint( | |
| 2644 trace_scope_->file()); | |
| 2645 PrintF(trace_scope_->file(), "\n"); | |
| 2646 } | |
| 2647 output_[frame_index]->SetFrameSlot(output_offset, input_value); | |
| 2648 return; | |
| 2649 } | |
| 2650 | |
| 2651 case Translation::INT32_STACK_SLOT: { | |
| 2652 int input_slot_index = iterator->Next(); | |
| 2653 unsigned input_offset = input_->GetOffsetFromSlotIndex(input_slot_index); | |
| 2654 intptr_t value = input_->GetFrameSlot(input_offset); | |
| 2655 bool is_smi = Smi::IsValid(value); | |
| 2656 if (trace_scope_ != NULL) { | |
| 2657 PrintF(trace_scope_->file(), | |
| 2658 " 0x%08" V8PRIxPTR ": ", | |
| 2659 output_[frame_index]->GetTop() + output_offset); | |
| 2660 PrintF(trace_scope_->file(), | |
| 2661 "[top + %d] <- %" V8PRIdPTR " ; [sp + %d] (%s)\n", | |
| 2662 output_offset, | |
| 2663 value, | |
| 2664 input_offset, | |
| 2665 TraceValueType(is_smi)); | |
| 2666 } | |
| 2667 if (is_smi) { | |
| 2668 intptr_t tagged_value = | |
| 2669 reinterpret_cast<intptr_t>(Smi::FromInt(static_cast<int>(value))); | |
| 2670 output_[frame_index]->SetFrameSlot(output_offset, tagged_value); | |
| 2671 } else { | |
| 2672 // We save the untagged value on the side and store a GC-safe | |
| 2673 // temporary placeholder in the frame. | |
| 2674 AddDoubleValue(output_[frame_index]->GetTop() + output_offset, | |
| 2675 static_cast<double>(static_cast<int32_t>(value))); | |
| 2676 output_[frame_index]->SetFrameSlot(output_offset, kPlaceholder); | |
| 2677 } | |
| 2678 return; | |
| 2679 } | |
| 2680 | |
| 2681 case Translation::UINT32_STACK_SLOT: { | |
| 2682 int input_slot_index = iterator->Next(); | |
| 2683 unsigned input_offset = input_->GetOffsetFromSlotIndex(input_slot_index); | |
| 2684 uintptr_t value = | |
| 2685 static_cast<uintptr_t>(input_->GetFrameSlot(input_offset)); | |
| 2686 bool is_smi = value <= static_cast<uintptr_t>(Smi::kMaxValue); | |
| 2687 if (trace_scope_ != NULL) { | |
| 2688 PrintF(trace_scope_->file(), | |
| 2689 " 0x%08" V8PRIxPTR ": ", | |
| 2690 output_[frame_index]->GetTop() + output_offset); | |
| 2691 PrintF(trace_scope_->file(), | |
| 2692 "[top + %d] <- %" V8PRIuPTR " ; [sp + %d] (uint32 %s)\n", | |
| 2693 output_offset, | |
| 2694 value, | |
| 2695 input_offset, | |
| 2696 TraceValueType(is_smi)); | |
| 2697 } | |
| 2698 if (is_smi) { | |
| 2699 intptr_t tagged_value = | |
| 2700 reinterpret_cast<intptr_t>(Smi::FromInt(static_cast<int>(value))); | |
| 2701 output_[frame_index]->SetFrameSlot(output_offset, tagged_value); | |
| 2702 } else { | |
| 2703 // We save the untagged value on the side and store a GC-safe | |
| 2704 // temporary placeholder in the frame. | |
| 2705 AddDoubleValue(output_[frame_index]->GetTop() + output_offset, | |
| 2706 static_cast<double>(static_cast<uint32_t>(value))); | |
| 2707 output_[frame_index]->SetFrameSlot(output_offset, kPlaceholder); | |
| 2708 } | |
| 2709 return; | |
| 2710 } | |
| 2711 | |
| 2712 case Translation::BOOL_STACK_SLOT: { | |
| 2713 int input_slot_index = iterator->Next(); | |
| 2714 unsigned input_offset = input_->GetOffsetFromSlotIndex(input_slot_index); | |
| 2715 uintptr_t value = | |
| 2716 static_cast<uintptr_t>(input_->GetFrameSlot(input_offset)); | |
| 2717 bool is_smi = value <= static_cast<uintptr_t>(Smi::kMaxValue); | |
| 2718 if (trace_scope_ != NULL) { | |
| 2719 PrintF(trace_scope_->file(), " 0x%08" V8PRIxPTR ": ", | |
| 2720 output_[frame_index]->GetTop() + output_offset); | |
| 2721 PrintF(trace_scope_->file(), | |
| 2722 "[top + %d] <- %" V8PRIuPTR " ; [sp + %d] (uint32 %s)\n", | |
| 2723 output_offset, value, input_offset, TraceValueType(is_smi)); | |
| 2724 } | |
| 2725 if (value == 0) { | |
| 2726 output_[frame_index]->SetFrameSlot( | |
| 2727 output_offset, | |
| 2728 reinterpret_cast<intptr_t>(isolate_->heap()->false_value())); | |
| 2729 } else { | |
| 2730 DCHECK_EQ(1U, value); | |
| 2731 output_[frame_index]->SetFrameSlot( | |
| 2732 output_offset, | |
| 2733 reinterpret_cast<intptr_t>(isolate_->heap()->true_value())); | |
| 2734 } | |
| 2735 return; | |
| 2736 } | |
| 2737 | |
| 2738 case Translation::DOUBLE_STACK_SLOT: { | |
| 2739 int input_slot_index = iterator->Next(); | |
| 2740 unsigned input_offset = input_->GetOffsetFromSlotIndex(input_slot_index); | |
| 2741 double value = input_->GetDoubleFrameSlot(input_offset); | |
| 2742 int int_value = FastD2IChecked(value); | |
| 2743 bool is_smi = | |
| 2744 !IsMinusZero(value) && value == int_value && Smi::IsValid(int_value); | |
| 2745 if (trace_scope_ != NULL) { | |
| 2746 PrintF(trace_scope_->file(), | |
| 2747 " 0x%08" V8PRIxPTR ": [top + %d] <- %e ; [sp + %d]\n", | |
| 2748 output_[frame_index]->GetTop() + output_offset, | |
| 2749 output_offset, | |
| 2750 value, | |
| 2751 input_offset); | |
| 2752 } | |
| 2753 if (is_smi) { | |
| 2754 intptr_t tagged_value = | |
| 2755 reinterpret_cast<intptr_t>(Smi::FromInt(int_value)); | |
| 2756 output_[frame_index]->SetFrameSlot(output_offset, tagged_value); | |
| 2757 } else { | |
| 2758 // We save the untagged value on the side and store a GC-safe | |
| 2759 // temporary placeholder in the frame. | |
| 2760 AddDoubleValue(output_[frame_index]->GetTop() + output_offset, value); | |
| 2761 output_[frame_index]->SetFrameSlot(output_offset, kPlaceholder); | |
| 2762 } | |
| 2763 return; | |
| 2764 } | |
| 2765 | |
| 2766 case Translation::LITERAL: { | |
| 2767 Object* literal = ComputeLiteral(iterator->Next()); | |
| 2768 if (trace_scope_ != NULL) { | |
| 2769 PrintF(trace_scope_->file(), | |
| 2770 " 0x%08" V8PRIxPTR ": [top + %d] <- ", | |
| 2771 output_[frame_index]->GetTop() + output_offset, | |
| 2772 output_offset); | |
| 2773 literal->ShortPrint(trace_scope_->file()); | |
| 2774 PrintF(trace_scope_->file(), " ; literal\n"); | |
| 2775 } | |
| 2776 intptr_t value = reinterpret_cast<intptr_t>(literal); | |
| 2777 output_[frame_index]->SetFrameSlot(output_offset, value); | |
| 2778 return; | |
| 2779 } | |
| 2780 | |
| 2781 case Translation::DUPLICATED_OBJECT: { | |
| 2782 int object_index = iterator->Next(); | |
| 2783 if (trace_scope_ != NULL) { | |
| 2784 PrintF(trace_scope_->file(), | |
| 2785 " 0x%08" V8PRIxPTR ": [top + %d] <- ", | |
| 2786 output_[frame_index]->GetTop() + output_offset, | |
| 2787 output_offset); | |
| 2788 isolate_->heap()->arguments_marker()->ShortPrint(trace_scope_->file()); | |
| 2789 PrintF(trace_scope_->file(), | |
| 2790 " ; duplicate of object #%d\n", object_index); | |
| 2791 } | |
| 2792 // Use the materialization marker value as a sentinel and fill in | |
| 2793 // the object after the deoptimized frame is built. | |
| 2794 intptr_t value = reinterpret_cast<intptr_t>( | |
| 2795 isolate_->heap()->arguments_marker()); | |
| 2796 AddObjectDuplication(output_[frame_index]->GetTop() + output_offset, | |
| 2797 object_index); | |
| 2798 output_[frame_index]->SetFrameSlot(output_offset, value); | |
| 2799 return; | |
| 2800 } | |
| 2801 | |
| 2802 case Translation::ARGUMENTS_OBJECT: | |
| 2803 case Translation::CAPTURED_OBJECT: { | |
| 2804 int length = iterator->Next(); | |
| 2805 bool is_args = opcode == Translation::ARGUMENTS_OBJECT; | |
| 2806 if (trace_scope_ != NULL) { | |
| 2807 PrintF(trace_scope_->file(), | |
| 2808 " 0x%08" V8PRIxPTR ": [top + %d] <- ", | |
| 2809 output_[frame_index]->GetTop() + output_offset, | |
| 2810 output_offset); | |
| 2811 isolate_->heap()->arguments_marker()->ShortPrint(trace_scope_->file()); | |
| 2812 PrintF(trace_scope_->file(), | |
| 2813 " ; object (length = %d, is_args = %d)\n", length, is_args); | |
| 2814 } | |
| 2815 // Use the materialization marker value as a sentinel and fill in | |
| 2816 // the object after the deoptimized frame is built. | |
| 2817 intptr_t value = reinterpret_cast<intptr_t>( | |
| 2818 isolate_->heap()->arguments_marker()); | |
| 2819 AddObjectStart(output_[frame_index]->GetTop() + output_offset, | |
| 2820 length, is_args); | |
| 2821 output_[frame_index]->SetFrameSlot(output_offset, value); | |
| 2822 // We save the object values on the side and materialize the actual | |
| 2823 // object after the deoptimized frame is built. | |
| 2824 int object_index = deferred_objects_.length() - 1; | |
| 2825 for (int i = 0; i < length; i++) { | |
| 2826 DoTranslateObject(iterator, object_index, i); | |
| 2827 } | |
| 2828 return; | |
| 2829 } | |
| 2830 } | |
| 2831 } | 1897 } |
| 2832 | 1898 |
| 2833 | 1899 |
| 2834 unsigned Deoptimizer::ComputeInputFrameSize() const { | 1900 unsigned Deoptimizer::ComputeInputFrameSize() const { |
| 2835 unsigned fixed_size = ComputeFixedSize(function_); | 1901 unsigned fixed_size = ComputeFixedSize(function_); |
| 2836 // The fp-to-sp delta already takes the context, constant pool pointer and the | 1902 // The fp-to-sp delta already takes the context, constant pool pointer and the |
| 2837 // function into account so we have to avoid double counting them. | 1903 // function into account so we have to avoid double counting them. |
| 2838 unsigned result = fixed_size + fp_to_sp_delta_ - | 1904 unsigned result = fixed_size + fp_to_sp_delta_ - |
| 2839 StandardFrameConstants::kFixedFrameSizeFromFp; | 1905 StandardFrameConstants::kFixedFrameSizeFromFp; |
| 2840 if (compiled_code_->kind() == Code::OPTIMIZED_FUNCTION) { | 1906 if (compiled_code_->kind() == Code::OPTIMIZED_FUNCTION) { |
| 2841 unsigned stack_slots = compiled_code_->stack_slots(); | 1907 unsigned stack_slots = compiled_code_->stack_slots(); |
| 2842 unsigned outgoing_size = ComputeOutgoingArgumentSize(); | 1908 unsigned outgoing_size = ComputeOutgoingArgumentSize(); |
| 2843 CHECK(result == fixed_size + (stack_slots * kPointerSize) + outgoing_size); | 1909 CHECK(result == fixed_size + (stack_slots * kPointerSize) + outgoing_size); |
| 2844 } | 1910 } |
| 2845 return result; | 1911 return result; |
| 2846 } | 1912 } |
| 2847 | 1913 |
| 2848 | 1914 |
| 2849 unsigned Deoptimizer::ComputeFixedSize(JSFunction* function) const { | 1915 unsigned Deoptimizer::ComputeFixedSize(JSFunction* function) const { |
| 2850 // The fixed part of the frame consists of the return address, frame | 1916 // The fixed part of the frame consists of the return address, frame |
| 2851 // pointer, function, context, and all the incoming arguments. | 1917 // pointer, function, context, and all the incoming arguments. |
| 2852 return ComputeIncomingArgumentSize(function) + | 1918 return ComputeIncomingArgumentSize(function) + |
| 2853 StandardFrameConstants::kFixedFrameSize; | 1919 StandardFrameConstants::kFixedFrameSize; |
| 2854 } | 1920 } |
| 2855 | 1921 |
| 2856 | 1922 |
| 2857 unsigned Deoptimizer::ComputeIncomingArgumentSize(JSFunction* function) const { | 1923 unsigned Deoptimizer::ComputeIncomingArgumentSize(JSFunction* function) const { |
| 2858 // The incoming arguments is the values for formal parameters and | 1924 // The incoming arguments is the values for formal parameters and |
| 2859 // the receiver. Every slot contains a pointer. | 1925 // the receiver. Every slot contains a pointer. |
| 2860 if (function->IsSmi()) { | 1926 if (function->IsSmi()) { |
| 2861 CHECK_EQ(Smi::cast(function), Smi::FromInt(StackFrame::STUB)); | 1927 CHECK_EQ(Smi::cast(function), Smi::FromInt(StackFrame::STUB)); |
| 2862 return 0; | 1928 return 0; |
| 2863 } | 1929 } |
| 2864 unsigned arguments = | 1930 unsigned arguments = |
| 2865 function->shared()->internal_formal_parameter_count() + 1; | 1931 function->shared()->internal_formal_parameter_count() + 1; |
| 2866 return arguments * kPointerSize; | 1932 return arguments * kPointerSize; |
| 2867 } | 1933 } |
| 2868 | 1934 |
| 2869 | 1935 |
| 2870 unsigned Deoptimizer::ComputeOutgoingArgumentSize() const { | 1936 unsigned Deoptimizer::ComputeOutgoingArgumentSize() const { |
| 2871 DeoptimizationInputData* data = DeoptimizationInputData::cast( | 1937 DeoptimizationInputData* data = |
| 2872 compiled_code_->deoptimization_data()); | 1938 DeoptimizationInputData::cast(compiled_code_->deoptimization_data()); |
| 2873 unsigned height = data->ArgumentsStackHeight(bailout_id_)->value(); | 1939 unsigned height = data->ArgumentsStackHeight(bailout_id_)->value(); |
| 2874 return height * kPointerSize; | 1940 return height * kPointerSize; |
| 2875 } | 1941 } |
| 2876 | 1942 |
| 2877 | 1943 |
| 2878 Object* Deoptimizer::ComputeLiteral(int index) const { | 1944 Object* Deoptimizer::ComputeLiteral(int index) const { |
| 2879 DeoptimizationInputData* data = DeoptimizationInputData::cast( | 1945 DeoptimizationInputData* data = |
| 2880 compiled_code_->deoptimization_data()); | 1946 DeoptimizationInputData::cast(compiled_code_->deoptimization_data()); |
| 2881 FixedArray* literals = data->LiteralArray(); | 1947 FixedArray* literals = data->LiteralArray(); |
| 2882 return literals->get(index); | 1948 return literals->get(index); |
| 2883 } | 1949 } |
| 2884 | 1950 |
| 2885 | 1951 |
| 2886 void Deoptimizer::AddObjectStart(intptr_t slot, int length, bool is_args) { | |
| 2887 ObjectMaterializationDescriptor object_desc( | |
| 2888 reinterpret_cast<Address>(slot), jsframe_count_, length, -1, is_args); | |
| 2889 deferred_objects_.Add(object_desc); | |
| 2890 } | |
| 2891 | |
| 2892 | |
| 2893 void Deoptimizer::AddObjectDuplication(intptr_t slot, int object_index) { | |
| 2894 ObjectMaterializationDescriptor object_desc( | |
| 2895 reinterpret_cast<Address>(slot), jsframe_count_, -1, object_index, false); | |
| 2896 deferred_objects_.Add(object_desc); | |
| 2897 } | |
| 2898 | |
| 2899 | |
| 2900 void Deoptimizer::AddObjectTaggedValue(intptr_t value) { | |
| 2901 deferred_objects_tagged_values_.Add(reinterpret_cast<Object*>(value)); | |
| 2902 } | |
| 2903 | |
| 2904 | |
| 2905 void Deoptimizer::AddObjectDoubleValue(double value) { | |
| 2906 deferred_objects_tagged_values_.Add(isolate()->heap()->the_hole_value()); | |
| 2907 HeapNumberMaterializationDescriptor<int> value_desc( | |
| 2908 deferred_objects_tagged_values_.length() - 1, value); | |
| 2909 deferred_objects_double_values_.Add(value_desc); | |
| 2910 } | |
| 2911 | |
| 2912 | |
| 2913 void Deoptimizer::AddDoubleValue(intptr_t slot_address, double value) { | |
| 2914 HeapNumberMaterializationDescriptor<Address> value_desc( | |
| 2915 reinterpret_cast<Address>(slot_address), value); | |
| 2916 deferred_heap_numbers_.Add(value_desc); | |
| 2917 } | |
| 2918 | |
| 2919 | |
| 2920 void Deoptimizer::EnsureCodeForDeoptimizationEntry(Isolate* isolate, | 1952 void Deoptimizer::EnsureCodeForDeoptimizationEntry(Isolate* isolate, |
| 2921 BailoutType type, | 1953 BailoutType type, |
| 2922 int max_entry_id) { | 1954 int max_entry_id) { |
| 2923 // We cannot run this if the serializer is enabled because this will | 1955 // We cannot run this if the serializer is enabled because this will |
| 2924 // cause us to emit relocation information for the external | 1956 // cause us to emit relocation information for the external |
| 2925 // references. This is fine because the deoptimizer's code section | 1957 // references. This is fine because the deoptimizer's code section |
| 2926 // isn't meant to be serialized at all. | 1958 // isn't meant to be serialized at all. |
| 2927 CHECK(type == EAGER || type == SOFT || type == LAZY); | 1959 CHECK(type == EAGER || type == SOFT || type == LAZY); |
| 2928 DeoptimizerData* data = isolate->deoptimizer_data(); | 1960 DeoptimizerData* data = isolate->deoptimizer_data(); |
| 2929 int entry_count = data->deopt_entry_code_entries_[type]; | 1961 int entry_count = data->deopt_entry_code_entries_[type]; |
| (...skipping 10 matching lines...) Expand all Loading... | |
| 2940 DCHECK(!RelocInfo::RequiresRelocation(desc)); | 1972 DCHECK(!RelocInfo::RequiresRelocation(desc)); |
| 2941 | 1973 |
| 2942 MemoryChunk* chunk = data->deopt_entry_code_[type]; | 1974 MemoryChunk* chunk = data->deopt_entry_code_[type]; |
| 2943 CHECK(static_cast<int>(Deoptimizer::GetMaxDeoptTableSize()) >= | 1975 CHECK(static_cast<int>(Deoptimizer::GetMaxDeoptTableSize()) >= |
| 2944 desc.instr_size); | 1976 desc.instr_size); |
| 2945 if (!chunk->CommitArea(desc.instr_size)) { | 1977 if (!chunk->CommitArea(desc.instr_size)) { |
| 2946 V8::FatalProcessOutOfMemory( | 1978 V8::FatalProcessOutOfMemory( |
| 2947 "Deoptimizer::EnsureCodeForDeoptimizationEntry"); | 1979 "Deoptimizer::EnsureCodeForDeoptimizationEntry"); |
| 2948 } | 1980 } |
| 2949 CopyBytes(chunk->area_start(), desc.buffer, | 1981 CopyBytes(chunk->area_start(), desc.buffer, |
| 2950 static_cast<size_t>(desc.instr_size)); | 1982 static_cast<size_t>(desc.instr_size)); |
| 2951 CpuFeatures::FlushICache(chunk->area_start(), desc.instr_size); | 1983 CpuFeatures::FlushICache(chunk->area_start(), desc.instr_size); |
| 2952 | 1984 |
| 2953 data->deopt_entry_code_entries_[type] = entry_count; | 1985 data->deopt_entry_code_entries_[type] = entry_count; |
| 2954 } | 1986 } |
| 2955 | 1987 |
| 2956 | 1988 |
| 2957 FrameDescription::FrameDescription(uint32_t frame_size, | 1989 FrameDescription::FrameDescription(uint32_t frame_size, |
| 2958 JSFunction* function) | 1990 JSFunction* function) |
| 2959 : frame_size_(frame_size), | 1991 : frame_size_(frame_size), |
| 2960 function_(function), | 1992 function_(function), |
| (...skipping 150 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 3111 void Translation::BeginJSFrame(BailoutId node_id, | 2143 void Translation::BeginJSFrame(BailoutId node_id, |
| 3112 int literal_id, | 2144 int literal_id, |
| 3113 unsigned height) { | 2145 unsigned height) { |
| 3114 buffer_->Add(JS_FRAME, zone()); | 2146 buffer_->Add(JS_FRAME, zone()); |
| 3115 buffer_->Add(node_id.ToInt(), zone()); | 2147 buffer_->Add(node_id.ToInt(), zone()); |
| 3116 buffer_->Add(literal_id, zone()); | 2148 buffer_->Add(literal_id, zone()); |
| 3117 buffer_->Add(height, zone()); | 2149 buffer_->Add(height, zone()); |
| 3118 } | 2150 } |
| 3119 | 2151 |
| 3120 | 2152 |
| 3121 void Translation::BeginCompiledStubFrame() { | 2153 void Translation::BeginCompiledStubFrame(int height) { |
| 3122 buffer_->Add(COMPILED_STUB_FRAME, zone()); | 2154 buffer_->Add(COMPILED_STUB_FRAME, zone()); |
| 2155 buffer_->Add(height, zone()); | |
| 3123 } | 2156 } |
| 3124 | 2157 |
| 3125 | 2158 |
| 3126 void Translation::BeginArgumentsObject(int args_length) { | 2159 void Translation::BeginArgumentsObject(int args_length) { |
| 3127 buffer_->Add(ARGUMENTS_OBJECT, zone()); | 2160 buffer_->Add(ARGUMENTS_OBJECT, zone()); |
| 3128 buffer_->Add(args_length, zone()); | 2161 buffer_->Add(args_length, zone()); |
| 3129 } | 2162 } |
| 3130 | 2163 |
| 3131 | 2164 |
| 3132 void Translation::BeginCapturedObject(int length) { | 2165 void Translation::BeginCapturedObject(int length) { |
| (...skipping 124 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 3257 TRANSLATION_OPCODE_LIST(TRANSLATION_OPCODE_CASE) | 2290 TRANSLATION_OPCODE_LIST(TRANSLATION_OPCODE_CASE) |
| 3258 } | 2291 } |
| 3259 #undef TRANSLATION_OPCODE_CASE | 2292 #undef TRANSLATION_OPCODE_CASE |
| 3260 UNREACHABLE(); | 2293 UNREACHABLE(); |
| 3261 return ""; | 2294 return ""; |
| 3262 } | 2295 } |
| 3263 | 2296 |
| 3264 #endif | 2297 #endif |
| 3265 | 2298 |
| 3266 | 2299 |
| 2300 Handle<FixedArray> MaterializedObjectStore::Get(Address fp) { | |
| 2301 int index = StackIdToIndex(fp); | |
| 2302 if (index == -1) { | |
| 2303 return Handle<FixedArray>::null(); | |
| 2304 } | |
| 2305 Handle<FixedArray> array = GetStackEntries(); | |
| 2306 CHECK_GT(array->length(), index); | |
| 2307 return Handle<FixedArray>::cast(Handle<Object>(array->get(index), isolate())); | |
| 2308 } | |
| 2309 | |
| 2310 | |
| 2311 void MaterializedObjectStore::Set(Address fp, | |
| 2312 Handle<FixedArray> materialized_objects) { | |
| 2313 int index = StackIdToIndex(fp); | |
| 2314 if (index == -1) { | |
| 2315 index = frame_fps_.length(); | |
| 2316 frame_fps_.Add(fp); | |
| 2317 } | |
| 2318 | |
| 2319 Handle<FixedArray> array = EnsureStackEntries(index + 1); | |
| 2320 array->set(index, *materialized_objects); | |
| 2321 } | |
| 2322 | |
| 2323 | |
| 2324 bool MaterializedObjectStore::Remove(Address fp) { | |
| 2325 int index = StackIdToIndex(fp); | |
| 2326 if (index == -1) { | |
| 2327 return false; | |
| 2328 } | |
| 2329 CHECK_GE(index, 0); | |
| 2330 | |
| 2331 frame_fps_.Remove(index); | |
| 2332 FixedArray* array = isolate()->heap()->materialized_objects(); | |
| 2333 CHECK_LT(index, array->length()); | |
| 2334 for (int i = index; i < frame_fps_.length(); i++) { | |
| 2335 array->set(i, array->get(i + 1)); | |
| 2336 } | |
| 2337 array->set(frame_fps_.length(), isolate()->heap()->undefined_value()); | |
| 2338 return true; | |
| 2339 } | |
| 2340 | |
| 2341 | |
| 2342 int MaterializedObjectStore::StackIdToIndex(Address fp) { | |
| 2343 for (int i = 0; i < frame_fps_.length(); i++) { | |
| 2344 if (frame_fps_[i] == fp) { | |
| 2345 return i; | |
| 2346 } | |
| 2347 } | |
| 2348 return -1; | |
| 2349 } | |
| 2350 | |
| 2351 | |
| 2352 Handle<FixedArray> MaterializedObjectStore::GetStackEntries() { | |
| 2353 return Handle<FixedArray>(isolate()->heap()->materialized_objects()); | |
| 2354 } | |
| 2355 | |
| 2356 | |
| 2357 Handle<FixedArray> MaterializedObjectStore::EnsureStackEntries(int length) { | |
| 2358 Handle<FixedArray> array = GetStackEntries(); | |
| 2359 if (array->length() >= length) { | |
| 2360 return array; | |
| 2361 } | |
| 2362 | |
| 2363 int new_length = length > 10 ? length : 10; | |
| 2364 if (new_length < 2 * array->length()) { | |
| 2365 new_length = 2 * array->length(); | |
| 2366 } | |
| 2367 | |
| 2368 Handle<FixedArray> new_array = | |
| 2369 isolate()->factory()->NewFixedArray(new_length, TENURED); | |
| 2370 for (int i = 0; i < array->length(); i++) { | |
| 2371 new_array->set(i, array->get(i)); | |
| 2372 } | |
| 2373 for (int i = array->length(); i < length; i++) { | |
| 2374 new_array->set(i, isolate()->heap()->undefined_value()); | |
| 2375 } | |
| 2376 isolate()->heap()->public_set_materialized_objects(*new_array); | |
| 2377 return new_array; | |
| 2378 } | |
| 2379 | |
| 2380 | |
| 2381 DeoptimizedFrameInfo::DeoptimizedFrameInfo(Deoptimizer* deoptimizer, | |
| 2382 int frame_index, | |
| 2383 bool has_arguments_adaptor, | |
| 2384 bool has_construct_stub) { | |
| 2385 FrameDescription* output_frame = deoptimizer->output_[frame_index]; | |
| 2386 function_ = output_frame->GetFunction(); | |
| 2387 context_ = reinterpret_cast<Object*>(output_frame->GetContext()); | |
| 2388 has_construct_stub_ = has_construct_stub; | |
| 2389 expression_count_ = output_frame->GetExpressionCount(); | |
| 2390 expression_stack_ = new Object* [expression_count_]; | |
| 2391 // Get the source position using the unoptimized code. | |
| 2392 Address pc = reinterpret_cast<Address>(output_frame->GetPc()); | |
| 2393 Code* code = Code::cast(deoptimizer->isolate()->FindCodeObject(pc)); | |
| 2394 source_position_ = code->SourcePosition(pc); | |
| 2395 | |
| 2396 for (int i = 0; i < expression_count_; i++) { | |
| 2397 SetExpression(i, output_frame->GetExpression(i)); | |
| 2398 } | |
| 2399 | |
| 2400 if (has_arguments_adaptor) { | |
| 2401 output_frame = deoptimizer->output_[frame_index - 1]; | |
| 2402 CHECK_EQ(output_frame->GetFrameType(), StackFrame::ARGUMENTS_ADAPTOR); | |
| 2403 } | |
| 2404 | |
| 2405 parameters_count_ = output_frame->ComputeParametersCount(); | |
| 2406 parameters_ = new Object* [parameters_count_]; | |
| 2407 for (int i = 0; i < parameters_count_; i++) { | |
| 2408 SetParameter(i, output_frame->GetParameter(i)); | |
| 2409 } | |
| 2410 } | |
| 2411 | |
| 2412 | |
| 2413 DeoptimizedFrameInfo::~DeoptimizedFrameInfo() { | |
| 2414 delete[] expression_stack_; | |
| 2415 delete[] parameters_; | |
| 2416 } | |
| 2417 | |
| 2418 | |
| 2419 void DeoptimizedFrameInfo::Iterate(ObjectVisitor* v) { | |
| 2420 v->VisitPointer(bit_cast<Object**>(&function_)); | |
| 2421 v->VisitPointer(&context_); | |
| 2422 v->VisitPointers(parameters_, parameters_ + parameters_count_); | |
| 2423 v->VisitPointers(expression_stack_, expression_stack_ + expression_count_); | |
| 2424 } | |
| 2425 | |
| 2426 | |
| 2427 const char* Deoptimizer::GetDeoptReason(DeoptReason deopt_reason) { | |
| 2428 DCHECK(deopt_reason < kLastDeoptReason); | |
| 2429 #define DEOPT_MESSAGES_TEXTS(C, T) T, | |
| 2430 static const char* deopt_messages_[] = { | |
| 2431 DEOPT_MESSAGES_LIST(DEOPT_MESSAGES_TEXTS)}; | |
| 2432 #undef DEOPT_MESSAGES_TEXTS | |
| 2433 return deopt_messages_[deopt_reason]; | |
| 2434 } | |
| 2435 | |
| 2436 | |
| 2437 Deoptimizer::DeoptInfo Deoptimizer::GetDeoptInfo(Code* code, Address pc) { | |
| 2438 SourcePosition last_position = SourcePosition::Unknown(); | |
| 2439 Deoptimizer::DeoptReason last_reason = Deoptimizer::kNoReason; | |
| 2440 int mask = RelocInfo::ModeMask(RelocInfo::DEOPT_REASON) | | |
| 2441 RelocInfo::ModeMask(RelocInfo::POSITION); | |
| 2442 for (RelocIterator it(code, mask); !it.done(); it.next()) { | |
| 2443 RelocInfo* info = it.rinfo(); | |
| 2444 if (info->pc() >= pc) return DeoptInfo(last_position, NULL, last_reason); | |
| 2445 if (info->rmode() == RelocInfo::POSITION) { | |
| 2446 int raw_position = static_cast<int>(info->data()); | |
| 2447 last_position = raw_position ? SourcePosition::FromRaw(raw_position) | |
| 2448 : SourcePosition::Unknown(); | |
| 2449 } else if (info->rmode() == RelocInfo::DEOPT_REASON) { | |
| 2450 last_reason = static_cast<Deoptimizer::DeoptReason>(info->data()); | |
| 2451 } | |
| 2452 } | |
| 2453 return DeoptInfo(SourcePosition::Unknown(), NULL, Deoptimizer::kNoReason); | |
| 2454 } | |
| 2455 | |
| 2456 | |
| 2457 // static | |
| 2458 TranslatedValue TranslatedValue::NewArgumentsObject(TranslatedState* container, | |
| 2459 int length, | |
| 2460 int object_index) { | |
| 2461 TranslatedValue slot(container, ARGUMENTS_OBJECT); | |
| 2462 slot.deferred_object_length_ = length; | |
| 2463 slot.object_index_ = object_index; | |
| 2464 return slot; | |
| 2465 } | |
| 2466 | |
| 2467 | |
| 2468 // static | |
| 2469 TranslatedValue TranslatedValue::NewDeferredObject(TranslatedState* container, | |
| 2470 int length, | |
| 2471 int object_index) { | |
| 2472 TranslatedValue slot(container, DEFERRED_OBJECT); | |
| 2473 slot.deferred_object_length_ = length; | |
| 2474 slot.object_index_ = object_index; | |
| 2475 return slot; | |
| 2476 } | |
| 2477 | |
| 2478 | |
| 2479 // static | |
| 2480 TranslatedValue TranslatedValue::NewDuplicateObject(TranslatedState* container, | |
| 2481 int id) { | |
| 2482 TranslatedValue slot(container, DUPLICATE_OBJECT); | |
| 2483 slot.object_index_ = id; | |
| 2484 return slot; | |
| 2485 } | |
| 2486 | |
| 2487 | |
| 2488 // static | |
| 2489 TranslatedValue TranslatedValue::NewDouble(TranslatedState* container, | |
| 2490 double value) { | |
| 2491 TranslatedValue slot(container, DOUBLE); | |
| 2492 slot.double_value_ = value; | |
| 2493 return slot; | |
| 2494 } | |
| 2495 | |
| 2496 | |
| 2497 // static | |
| 2498 TranslatedValue TranslatedValue::NewInt32(TranslatedState* container, | |
| 2499 uint32_t value) { | |
| 2500 TranslatedValue slot(container, INT32); | |
| 2501 slot.integral_value_ = value; | |
| 2502 return slot; | |
| 2503 } | |
| 2504 | |
| 2505 | |
| 2506 // static | |
| 2507 TranslatedValue TranslatedValue::NewUInt32(TranslatedState* container, | |
| 2508 uint32_t value) { | |
| 2509 TranslatedValue slot(container, UINT32); | |
| 2510 slot.integral_value_ = value; | |
| 2511 return slot; | |
| 2512 } | |
| 2513 | |
| 2514 | |
| 2515 // static | |
| 2516 TranslatedValue TranslatedValue::NewBool(TranslatedState* container, | |
| 2517 uint32_t value) { | |
| 2518 TranslatedValue slot(container, BOOLBIT); | |
| 2519 slot.integral_value_ = value; | |
| 2520 return slot; | |
| 2521 } | |
| 2522 | |
| 2523 | |
| 2524 // static | |
| 2525 TranslatedValue TranslatedValue::NewTagged(TranslatedState* container, | |
| 2526 Object* literal) { | |
| 2527 TranslatedValue slot(container, TAGGED); | |
| 2528 slot.raw_literal_ = literal; | |
| 2529 return slot; | |
| 2530 } | |
| 2531 | |
| 2532 | |
| 2533 // static | |
| 2534 TranslatedValue TranslatedValue::NewInvalid() { | |
| 2535 return TranslatedValue(nullptr, UNKNOWN); | |
| 2536 } | |
| 2537 | |
| 2538 | |
| 2539 Isolate* TranslatedValue::isolate() { return container_->isolate(); } | |
| 2540 | |
| 2541 | |
| 2542 Object* TranslatedValue::GetRawValue() { | |
| 2543 // If we have a value, return it. | |
| 2544 Handle<Object> result_handle; | |
| 2545 if (value_.ToHandle(&result_handle)) { | |
| 2546 return *result_handle; | |
| 2547 } | |
| 2548 | |
| 2549 // Otherwise, do a best effort to get the value without allocation. | |
| 2550 switch (kind()) { | |
| 2551 case TAGGED: | |
| 2552 return raw_literal_; | |
| 2553 | |
| 2554 case INT32: { | |
| 2555 int32_t i32_value = static_cast<int32_t>(integral_value_); | |
| 2556 bool is_smi = Smi::IsValid(i32_value); | |
| 2557 if (is_smi) { | |
| 2558 return Smi::FromInt(i32_value); | |
| 2559 } | |
| 2560 break; | |
| 2561 } | |
| 2562 | |
| 2563 case UINT32: { | |
| 2564 bool is_smi = (integral_value_ <= static_cast<uintptr_t>(Smi::kMaxValue)); | |
| 2565 if (is_smi) { | |
| 2566 return Smi::FromInt(static_cast<int32_t>(integral_value_)); | |
| 2567 } | |
| 2568 break; | |
| 2569 } | |
| 2570 | |
| 2571 case DOUBLE: { | |
| 2572 int int_value = FastD2IChecked(double_value_); | |
| 2573 bool is_smi = !IsMinusZero(double_value_) && double_value_ == int_value && | |
| 2574 Smi::IsValid(int_value); | |
| 2575 if (is_smi) { | |
| 2576 return Smi::FromInt(static_cast<int32_t>(int_value)); | |
| 2577 } | |
| 2578 break; | |
| 2579 } | |
| 2580 | |
| 2581 case BOOLBIT: { | |
| 2582 if (integral_value_ == 0) { | |
| 2583 return isolate()->heap()->false_value(); | |
| 2584 } else { | |
| 2585 CHECK_EQ(1, integral_value_); | |
| 2586 return isolate()->heap()->true_value(); | |
| 2587 } | |
| 2588 } | |
| 2589 | |
| 2590 default: | |
| 2591 break; | |
| 2592 } | |
| 2593 | |
| 2594 // If we could not get the value without allocation, return the arguments | |
| 2595 // marker. | |
| 2596 return isolate()->heap()->arguments_marker(); | |
| 2597 } | |
| 2598 | |
| 2599 | |
| 2600 Handle<Object> TranslatedValue::GetValue() { | |
| 2601 Handle<Object> result; | |
| 2602 // If we already have a value, then get it. | |
| 2603 if (value_.ToHandle(&result)) return result; | |
| 2604 | |
| 2605 // Otherwise we have to materialize. | |
| 2606 switch (kind()) { | |
| 2607 case TranslatedValue::TAGGED: | |
| 2608 case TranslatedValue::INT32: | |
| 2609 case TranslatedValue::UINT32: | |
| 2610 case TranslatedValue::BOOLBIT: | |
| 2611 case TranslatedValue::DOUBLE: { | |
| 2612 MaterializeSimple(); | |
| 2613 return value_.ToHandleChecked(); | |
| 2614 } | |
| 2615 | |
| 2616 case TranslatedValue::ARGUMENTS_OBJECT: | |
| 2617 case TranslatedValue::DEFERRED_OBJECT: | |
| 2618 case TranslatedValue::DUPLICATE_OBJECT: | |
| 2619 return container_->MaterializeObjectAt(ObjectIndex()); | |
| 2620 | |
| 2621 case TranslatedValue::UNKNOWN: | |
| 2622 FATAL("unexpected case"); | |
| 2623 return Handle<Object>::null(); | |
| 2624 } | |
| 2625 | |
| 2626 FATAL("internal error: value missing"); | |
| 2627 return Handle<Object>::null(); | |
| 2628 } | |
| 2629 | |
| 2630 | |
| 2631 void TranslatedValue::MaterializeSimple() { | |
| 2632 // If we already have materialized, return. | |
| 2633 if (!value_.is_null()) return; | |
| 2634 | |
| 2635 Object* raw_value = GetRawValue(); | |
| 2636 if (raw_value != isolate()->heap()->arguments_marker()) { | |
| 2637 // We can get the value without allocation, just return it here. | |
| 2638 value_ = Handle<Object>(raw_value, isolate()); | |
| 2639 return; | |
| 2640 } | |
| 2641 | |
| 2642 switch (kind()) { | |
| 2643 case INT32: { | |
| 2644 int32_t i32_value = static_cast<int32_t>(integral_value_); | |
|
Benedikt Meurer
2015/06/03 04:02:17
Use a bit_cast<int32_t> here, or if you turn the f
Jarin
2015/06/03 09:28:07
Turned to unions.
| |
| 2645 value_ = Handle<Object>(isolate()->factory()->NewNumber(i32_value)); | |
| 2646 return; | |
| 2647 } | |
| 2648 | |
| 2649 case UINT32: | |
| 2650 value_ = Handle<Object>(isolate()->factory()->NewNumber(integral_value_)); | |
| 2651 return; | |
| 2652 | |
| 2653 case DOUBLE: | |
| 2654 value_ = Handle<Object>(isolate()->factory()->NewNumber(double_value_)); | |
| 2655 return; | |
| 2656 | |
| 2657 case DEFERRED_OBJECT: | |
| 2658 case DUPLICATE_OBJECT: | |
| 2659 case ARGUMENTS_OBJECT: | |
| 2660 case UNKNOWN: | |
| 2661 case TAGGED: | |
| 2662 case BOOLBIT: | |
| 2663 FATAL("internal error: unexpected materialization."); | |
| 2664 break; | |
| 2665 } | |
| 2666 } | |
| 2667 | |
| 2668 | |
| 2669 bool TranslatedValue::IsMaterializedObject() { | |
| 2670 switch (kind()) { | |
| 2671 case DEFERRED_OBJECT: | |
| 2672 case DUPLICATE_OBJECT: | |
| 2673 case ARGUMENTS_OBJECT: | |
| 2674 return true; | |
| 2675 default: | |
| 2676 return false; | |
| 2677 } | |
| 2678 } | |
| 2679 | |
| 2680 | |
| 2681 int TranslatedState::SlotOffsetFp(int slot_index) { | |
| 2682 if (slot_index >= 0) { | |
| 2683 const int offset = StandardFrameConstants::kExpressionsOffset; | |
| 2684 return offset - (slot_index * kPointerSize); | |
| 2685 } else { | |
| 2686 const int offset = StandardFrameConstants::kCallerSPOffset; | |
| 2687 return offset - ((slot_index + 1) * kPointerSize); | |
| 2688 } | |
| 2689 } | |
| 2690 | |
| 2691 | |
| 2692 Address TranslatedState::SlotAddress(Address fp, int slot_index) { | |
| 2693 return fp + SlotOffsetFp(slot_index); | |
| 2694 } | |
| 2695 | |
| 2696 | |
| 2697 uint32_t TranslatedState::GetUInt32Slot(Address fp, int slot_offset) { | |
| 2698 Address address = fp + slot_offset; | |
| 2699 #if V8_TARGET_BIG_ENDIAN && V8_HOST_ARCH_64_BIT | |
| 2700 return Memory::uint32_at(address + kIntSize); | |
| 2701 #else | |
| 2702 return Memory::uint32_at(address); | |
| 2703 #endif | |
| 2704 } | |
| 2705 | |
| 2706 | |
| 2707 void TranslatedValue::Handlify() { | |
| 2708 if (raw_literal_ != nullptr) { | |
| 2709 value_ = Handle<Object>(raw_literal_, isolate()); | |
| 2710 raw_literal_ = nullptr; | |
| 2711 } | |
| 2712 } | |
| 2713 | |
| 2714 | |
| 2715 TranslatedFrame TranslatedFrame::JSFrame(BailoutId node_id, | |
| 2716 JSFunction* function, int height) { | |
| 2717 TranslatedFrame frame(Function, function->GetIsolate(), function, height); | |
| 2718 frame.node_id_ = node_id; | |
| 2719 return frame; | |
| 2720 } | |
| 2721 | |
| 2722 | |
| 2723 TranslatedFrame TranslatedFrame::AccessorFrame(Kind kind, | |
| 2724 JSFunction* function) { | |
| 2725 DCHECK(kind == Setter || kind == Getter); | |
| 2726 return TranslatedFrame(kind, function->GetIsolate(), function); | |
| 2727 } | |
| 2728 | |
| 2729 | |
| 2730 TranslatedFrame TranslatedFrame::ArgumentsAdaptorFrame(JSFunction* function, | |
| 2731 int height) { | |
| 2732 return TranslatedFrame(ArgumentsAdaptor, function->GetIsolate(), function, | |
| 2733 height); | |
| 2734 } | |
| 2735 | |
| 2736 | |
| 2737 TranslatedFrame TranslatedFrame::ConstructStubFrame(JSFunction* function, | |
| 2738 int height) { | |
| 2739 return TranslatedFrame(ConstructStub, function->GetIsolate(), function, | |
| 2740 height); | |
| 2741 } | |
| 2742 | |
| 2743 | |
| 2744 int TranslatedFrame::GetValueCount() { | |
| 2745 switch (kind()) { | |
| 2746 case Function: { | |
| 2747 int parameter_count = | |
| 2748 (raw_function_ == nullptr | |
| 2749 ? function_->shared()->internal_formal_parameter_count() | |
| 2750 : raw_function_->shared()->internal_formal_parameter_count()) + | |
| 2751 1; | |
| 2752 return height_ + parameter_count; | |
| 2753 } | |
| 2754 | |
| 2755 case Getter: | |
| 2756 return 1; // Receiver. | |
| 2757 | |
| 2758 case Setter: | |
| 2759 return 2; // Receiver and the value to set. | |
| 2760 | |
| 2761 case ArgumentsAdaptor: | |
| 2762 case ConstructStub: | |
| 2763 case CompiledStub: | |
| 2764 return height_; | |
| 2765 | |
| 2766 case Invalid: | |
| 2767 UNREACHABLE(); | |
| 2768 break; | |
| 2769 } | |
| 2770 UNREACHABLE(); | |
| 2771 return -1; | |
| 2772 } | |
| 2773 | |
| 2774 | |
| 2775 void TranslatedFrame::Handlify(Isolate* isolate) { | |
| 2776 if (raw_function_ != nullptr) { | |
| 2777 function_ = Handle<JSFunction>(raw_function_, isolate); | |
| 2778 raw_function_ = nullptr; | |
| 2779 } | |
| 2780 for (auto& value : values_) { | |
| 2781 value.Handlify(); | |
| 2782 } | |
| 2783 } | |
| 2784 | |
| 2785 | |
| 2786 TranslatedFrame TranslatedState::CreateNextTranslatedFrame( | |
| 2787 TranslationIterator* iterator, FixedArray* literal_array, Address fp, | |
| 2788 JSFunction* frame_function, FILE* trace_file) { | |
| 2789 Translation::Opcode opcode = | |
| 2790 static_cast<Translation::Opcode>(iterator->Next()); | |
| 2791 switch (opcode) { | |
| 2792 case Translation::JS_FRAME: { | |
| 2793 BailoutId node_id = BailoutId(iterator->Next()); | |
| 2794 int closure_id = iterator->Next(); | |
| 2795 JSFunction* function = | |
| 2796 (closure_id == Translation::kSelfLiteralId) | |
| 2797 ? frame_function | |
| 2798 : JSFunction::cast(literal_array->get(closure_id)); | |
| 2799 int height = iterator->Next(); | |
| 2800 if (trace_file != nullptr) { | |
| 2801 PrintF(trace_file, " reading input frame "); | |
| 2802 function->PrintName(trace_file); | |
| 2803 int arg_count = | |
| 2804 function->shared()->internal_formal_parameter_count() + 1; | |
| 2805 PrintF(trace_file, " => node=%d, args=%d, height=%d; inputs:\n", | |
| 2806 arg_count, node_id.ToInt(), height); | |
| 2807 } | |
| 2808 return TranslatedFrame::JSFrame(node_id, function, height); | |
| 2809 } | |
| 2810 | |
| 2811 case Translation::ARGUMENTS_ADAPTOR_FRAME: { | |
| 2812 JSFunction* function = | |
| 2813 JSFunction::cast(literal_array->get(iterator->Next())); | |
| 2814 int height = iterator->Next(); | |
| 2815 if (trace_file != nullptr) { | |
| 2816 PrintF(trace_file, " reading arguments adaptor frame"); | |
| 2817 function->PrintName(trace_file); | |
| 2818 PrintF(trace_file, " => height=%d; inputs:\n", height); | |
| 2819 } | |
| 2820 return TranslatedFrame::ArgumentsAdaptorFrame(function, height); | |
| 2821 } | |
| 2822 | |
| 2823 case Translation::CONSTRUCT_STUB_FRAME: { | |
| 2824 JSFunction* function = | |
| 2825 JSFunction::cast(literal_array->get(iterator->Next())); | |
| 2826 int height = iterator->Next(); | |
| 2827 if (trace_file != nullptr) { | |
| 2828 PrintF(trace_file, " reading construct stub frame "); | |
| 2829 function->PrintName(trace_file); | |
| 2830 PrintF(trace_file, " => height=%d; inputs:\n", height); | |
| 2831 } | |
| 2832 return TranslatedFrame::ConstructStubFrame(function, height); | |
| 2833 } | |
| 2834 | |
| 2835 case Translation::GETTER_STUB_FRAME: { | |
| 2836 JSFunction* function = | |
| 2837 JSFunction::cast(literal_array->get(iterator->Next())); | |
| 2838 if (trace_file != nullptr) { | |
| 2839 PrintF(trace_file, " reading getter frame "); | |
| 2840 function->PrintName(trace_file); | |
| 2841 PrintF(trace_file, "; inputs:\n"); | |
| 2842 } | |
| 2843 return TranslatedFrame::AccessorFrame(TranslatedFrame::Getter, function); | |
| 2844 } | |
| 2845 | |
| 2846 case Translation::SETTER_STUB_FRAME: { | |
| 2847 JSFunction* function = | |
| 2848 JSFunction::cast(literal_array->get(iterator->Next())); | |
| 2849 if (trace_file != nullptr) { | |
| 2850 PrintF(trace_file, " reading setter frame "); | |
| 2851 function->PrintName(trace_file); | |
| 2852 PrintF(trace_file, "; inputs:\n"); | |
| 2853 } | |
| 2854 return TranslatedFrame::AccessorFrame(TranslatedFrame::Setter, function); | |
| 2855 } | |
| 2856 | |
| 2857 case Translation::COMPILED_STUB_FRAME: { | |
| 2858 int height = iterator->Next(); | |
| 2859 if (trace_file != nullptr) { | |
| 2860 PrintF(trace_file, | |
| 2861 " reading compiler stub frame => height=%d; inputs:\n", height); | |
| 2862 } | |
| 2863 return TranslatedFrame::CompiledStubFrame(height, | |
| 2864 literal_array->GetIsolate()); | |
| 2865 } | |
| 2866 | |
| 2867 case Translation::BEGIN: | |
| 2868 case Translation::DUPLICATED_OBJECT: | |
| 2869 case Translation::ARGUMENTS_OBJECT: | |
| 2870 case Translation::CAPTURED_OBJECT: | |
| 2871 case Translation::REGISTER: | |
| 2872 case Translation::INT32_REGISTER: | |
| 2873 case Translation::UINT32_REGISTER: | |
| 2874 case Translation::BOOL_REGISTER: | |
| 2875 case Translation::DOUBLE_REGISTER: | |
| 2876 case Translation::STACK_SLOT: | |
| 2877 case Translation::INT32_STACK_SLOT: | |
| 2878 case Translation::UINT32_STACK_SLOT: | |
| 2879 case Translation::BOOL_STACK_SLOT: | |
| 2880 case Translation::DOUBLE_STACK_SLOT: | |
| 2881 case Translation::LITERAL: | |
| 2882 break; | |
| 2883 } | |
| 2884 FATAL("We should never get here - unexpected deopt info."); | |
| 2885 return TranslatedFrame::InvalidFrame(); | |
| 2886 } | |
| 2887 | |
| 2888 | |
| 2889 // static | |
| 2890 void TranslatedFrame::AdvanceIterator( | |
| 2891 std::deque<TranslatedValue>::iterator* iter) { | |
| 2892 int values_to_skip = 1; | |
| 2893 while (values_to_skip > 0) { | |
| 2894 // Consume the current element. | |
| 2895 values_to_skip--; | |
| 2896 // Add all the children. | |
| 2897 values_to_skip += (*iter)->GetChildrenCount(); | |
| 2898 | |
| 2899 (*iter)++; | |
| 2900 } | |
| 2901 } | |
| 2902 | |
| 2903 | |
| 3267 // We can't intermix stack decoding and allocations because | 2904 // We can't intermix stack decoding and allocations because |
| 3268 // deoptimization infrastracture is not GC safe. | 2905 // deoptimization infrastracture is not GC safe. |
| 3269 // Thus we build a temporary structure in malloced space. | 2906 // Thus we build a temporary structure in malloced space. |
| 3270 SlotRef SlotRefValueBuilder::ComputeSlotForNextArgument( | 2907 TranslatedValue TranslatedState::CreateNextTranslatedValue( |
| 3271 Translation::Opcode opcode, | 2908 int frame_index, int value_index, TranslationIterator* iterator, |
| 3272 TranslationIterator* iterator, | 2909 FixedArray* literal_array, Address fp, RegisterValues* registers, |
| 3273 DeoptimizationInputData* data, | 2910 FILE* trace_file) { |
| 3274 JavaScriptFrame* frame) { | 2911 disasm::NameConverter converter; |
| 2912 | |
| 2913 Translation::Opcode opcode = | |
| 2914 static_cast<Translation::Opcode>(iterator->Next()); | |
| 3275 switch (opcode) { | 2915 switch (opcode) { |
| 3276 case Translation::BEGIN: | 2916 case Translation::BEGIN: |
| 3277 case Translation::JS_FRAME: | 2917 case Translation::JS_FRAME: |
| 3278 case Translation::ARGUMENTS_ADAPTOR_FRAME: | 2918 case Translation::ARGUMENTS_ADAPTOR_FRAME: |
| 3279 case Translation::CONSTRUCT_STUB_FRAME: | 2919 case Translation::CONSTRUCT_STUB_FRAME: |
| 3280 case Translation::GETTER_STUB_FRAME: | 2920 case Translation::GETTER_STUB_FRAME: |
| 3281 case Translation::SETTER_STUB_FRAME: | 2921 case Translation::SETTER_STUB_FRAME: |
| 2922 case Translation::COMPILED_STUB_FRAME: | |
| 3282 // Peeled off before getting here. | 2923 // Peeled off before getting here. |
| 3283 break; | 2924 break; |
| 3284 | 2925 |
| 3285 case Translation::DUPLICATED_OBJECT: { | 2926 case Translation::DUPLICATED_OBJECT: { |
| 3286 return SlotRef::NewDuplicateObject(iterator->Next()); | 2927 int object_id = iterator->Next(); |
| 3287 } | 2928 if (trace_file != nullptr) { |
| 3288 | 2929 PrintF(trace_file, "duplicated object #%d", object_id); |
| 3289 case Translation::ARGUMENTS_OBJECT: | 2930 } |
| 3290 return SlotRef::NewArgumentsObject(iterator->Next()); | 2931 object_positions_.push_back(object_positions_[object_id]); |
| 2932 return TranslatedValue::NewDuplicateObject(this, object_id); | |
| 2933 } | |
| 2934 | |
| 2935 case Translation::ARGUMENTS_OBJECT: { | |
| 2936 int arg_count = iterator->Next(); | |
| 2937 int object_index = static_cast<int>(object_positions_.size()); | |
| 2938 if (trace_file != nullptr) { | |
| 2939 PrintF(trace_file, "argumets object #%d (length = %d)", object_index, | |
| 2940 arg_count); | |
| 2941 } | |
| 2942 object_positions_.push_back({frame_index, value_index}); | |
| 2943 return TranslatedValue::NewArgumentsObject(this, arg_count, object_index); | |
| 2944 } | |
| 3291 | 2945 |
| 3292 case Translation::CAPTURED_OBJECT: { | 2946 case Translation::CAPTURED_OBJECT: { |
| 3293 return SlotRef::NewDeferredObject(iterator->Next()); | 2947 int field_count = iterator->Next(); |
| 3294 } | 2948 int object_index = static_cast<int>(object_positions_.size()); |
| 3295 | 2949 if (trace_file != nullptr) { |
| 3296 case Translation::REGISTER: | 2950 PrintF(trace_file, "captured object #%d (length = %d)", object_index, |
| 3297 case Translation::INT32_REGISTER: | 2951 field_count); |
| 3298 case Translation::UINT32_REGISTER: | 2952 } |
| 3299 case Translation::BOOL_REGISTER: | 2953 object_positions_.push_back({frame_index, value_index}); |
| 3300 case Translation::DOUBLE_REGISTER: | 2954 return TranslatedValue::NewDeferredObject(this, field_count, |
| 3301 // We are at safepoint which corresponds to call. All registers are | 2955 object_index); |
| 3302 // saved by caller so there would be no live registers at this | 2956 } |
| 3303 // point. Thus these translation commands should not be used. | 2957 |
| 3304 break; | 2958 case Translation::REGISTER: { |
| 2959 int input_reg = iterator->Next(); | |
| 2960 if (registers == nullptr) return TranslatedValue::NewInvalid(); | |
| 2961 intptr_t value = registers->GetRegister(input_reg); | |
| 2962 if (trace_file != nullptr) { | |
| 2963 PrintF(trace_file, "0x%08" V8PRIxPTR " ; %s ", value, | |
| 2964 converter.NameOfCPURegister(input_reg)); | |
| 2965 reinterpret_cast<Object*>(value)->ShortPrint(trace_file); | |
| 2966 } | |
| 2967 return TranslatedValue::NewTagged(this, reinterpret_cast<Object*>(value)); | |
| 2968 } | |
| 2969 | |
| 2970 case Translation::INT32_REGISTER: { | |
| 2971 int input_reg = iterator->Next(); | |
| 2972 if (registers == nullptr) return TranslatedValue::NewInvalid(); | |
| 2973 intptr_t value = registers->GetRegister(input_reg); | |
| 2974 if (trace_file != nullptr) { | |
| 2975 PrintF(trace_file, "%" V8PRIdPTR " ; %s ", value, | |
| 2976 converter.NameOfCPURegister(input_reg)); | |
| 2977 } | |
| 2978 return TranslatedValue::NewInt32(this, static_cast<uint32_t>(value)); | |
| 2979 } | |
| 2980 | |
| 2981 case Translation::UINT32_REGISTER: { | |
| 2982 int input_reg = iterator->Next(); | |
| 2983 if (registers == nullptr) return TranslatedValue::NewInvalid(); | |
| 2984 intptr_t value = registers->GetRegister(input_reg); | |
| 2985 if (trace_file != nullptr) { | |
| 2986 PrintF(trace_file, "%" V8PRIuPTR " ; %s (uint)", value, | |
| 2987 converter.NameOfCPURegister(input_reg)); | |
| 2988 reinterpret_cast<Object*>(value)->ShortPrint(trace_file); | |
| 2989 } | |
| 2990 return TranslatedValue::NewUInt32(this, static_cast<uint32_t>(value)); | |
| 2991 } | |
| 2992 | |
| 2993 case Translation::BOOL_REGISTER: { | |
| 2994 int input_reg = iterator->Next(); | |
| 2995 if (registers == nullptr) return TranslatedValue::NewInvalid(); | |
| 2996 intptr_t value = registers->GetRegister(input_reg); | |
| 2997 if (trace_file != nullptr) { | |
| 2998 PrintF(trace_file, "%" V8PRIdPTR " ; %s (bool)", value, | |
| 2999 converter.NameOfCPURegister(input_reg)); | |
| 3000 } | |
| 3001 return TranslatedValue::NewBool(this, static_cast<uint32_t>(value)); | |
| 3002 } | |
| 3003 | |
| 3004 case Translation::DOUBLE_REGISTER: { | |
| 3005 int input_reg = iterator->Next(); | |
| 3006 if (registers == nullptr) return TranslatedValue::NewInvalid(); | |
| 3007 double value = registers->GetDoubleRegister(input_reg); | |
| 3008 if (trace_file != nullptr) { | |
| 3009 PrintF(trace_file, "%e ; %s (bool)", value, | |
| 3010 DoubleRegister::AllocationIndexToString(input_reg)); | |
| 3011 } | |
| 3012 return TranslatedValue::NewDouble(this, value); | |
| 3013 } | |
| 3305 | 3014 |
| 3306 case Translation::STACK_SLOT: { | 3015 case Translation::STACK_SLOT: { |
| 3307 int slot_index = iterator->Next(); | 3016 int slot_offset = SlotOffsetFp(iterator->Next()); |
| 3308 Address slot_addr = SlotAddress(frame, slot_index); | 3017 intptr_t value = *(reinterpret_cast<intptr_t*>(fp + slot_offset)); |
| 3309 return SlotRef(slot_addr, SlotRef::TAGGED); | 3018 if (trace_file != nullptr) { |
| 3019 PrintF(trace_file, "0x%08" V8PRIxPTR " ; [fp %c %d] ", value, | |
| 3020 slot_offset < 0 ? '-' : '+', std::abs(slot_offset)); | |
| 3021 reinterpret_cast<Object*>(value)->ShortPrint(trace_file); | |
| 3022 } | |
| 3023 return TranslatedValue::NewTagged(this, reinterpret_cast<Object*>(value)); | |
| 3310 } | 3024 } |
| 3311 | 3025 |
| 3312 case Translation::INT32_STACK_SLOT: { | 3026 case Translation::INT32_STACK_SLOT: { |
| 3313 int slot_index = iterator->Next(); | 3027 int slot_offset = SlotOffsetFp(iterator->Next()); |
| 3314 Address slot_addr = SlotAddress(frame, slot_index); | 3028 uint32_t value = GetUInt32Slot(fp, slot_offset); |
| 3315 return SlotRef(slot_addr, SlotRef::INT32); | 3029 if (trace_file != nullptr) { |
| 3030 PrintF(trace_file, "%d ; (int) [fp %c %d] ", | |
| 3031 static_cast<int32_t>(value), slot_offset < 0 ? '-' : '+', | |
| 3032 std::abs(slot_offset)); | |
| 3033 } | |
| 3034 return TranslatedValue::NewInt32(this, value); | |
| 3316 } | 3035 } |
| 3317 | 3036 |
| 3318 case Translation::UINT32_STACK_SLOT: { | 3037 case Translation::UINT32_STACK_SLOT: { |
| 3319 int slot_index = iterator->Next(); | 3038 int slot_offset = SlotOffsetFp(iterator->Next()); |
| 3320 Address slot_addr = SlotAddress(frame, slot_index); | 3039 uint32_t value = GetUInt32Slot(fp, slot_offset); |
| 3321 return SlotRef(slot_addr, SlotRef::UINT32); | 3040 if (trace_file != nullptr) { |
| 3041 PrintF(trace_file, "%u ; (uint) [fp %c %d] ", value, | |
| 3042 slot_offset < 0 ? '-' : '+', std::abs(slot_offset)); | |
| 3043 } | |
| 3044 return TranslatedValue::NewUInt32(this, value); | |
| 3322 } | 3045 } |
| 3323 | 3046 |
| 3324 case Translation::BOOL_STACK_SLOT: { | 3047 case Translation::BOOL_STACK_SLOT: { |
| 3325 int slot_index = iterator->Next(); | 3048 int slot_offset = SlotOffsetFp(iterator->Next()); |
| 3326 Address slot_addr = SlotAddress(frame, slot_index); | 3049 uint32_t value = GetUInt32Slot(fp, slot_offset); |
| 3327 return SlotRef(slot_addr, SlotRef::BOOLBIT); | 3050 if (trace_file != nullptr) { |
| 3051 PrintF(trace_file, "%u ; (bool) [fp %c %d] ", value, | |
| 3052 slot_offset < 0 ? '-' : '+', std::abs(slot_offset)); | |
| 3053 } | |
| 3054 return TranslatedValue::NewBool(this, value); | |
| 3328 } | 3055 } |
| 3329 | 3056 |
| 3330 case Translation::DOUBLE_STACK_SLOT: { | 3057 case Translation::DOUBLE_STACK_SLOT: { |
| 3331 int slot_index = iterator->Next(); | 3058 int slot_offset = SlotOffsetFp(iterator->Next()); |
| 3332 Address slot_addr = SlotAddress(frame, slot_index); | 3059 double value = *(reinterpret_cast<double*>(fp + slot_offset)); |
| 3333 return SlotRef(slot_addr, SlotRef::DOUBLE); | 3060 if (trace_file != nullptr) { |
| 3061 PrintF(trace_file, "%e ; (double) [fp %c %d] ", value, | |
| 3062 slot_offset < 0 ? '-' : '+', std::abs(slot_offset)); | |
| 3063 } | |
| 3064 return TranslatedValue::NewDouble(this, value); | |
| 3334 } | 3065 } |
| 3335 | 3066 |
| 3336 case Translation::LITERAL: { | 3067 case Translation::LITERAL: { |
| 3337 int literal_index = iterator->Next(); | 3068 int literal_index = iterator->Next(); |
| 3338 return SlotRef(data->GetIsolate(), | 3069 Object* value = literal_array->get(literal_index); |
| 3339 data->LiteralArray()->get(literal_index)); | 3070 if (trace_file != nullptr) { |
| 3340 } | 3071 PrintF(trace_file, "0x%08" V8PRIxPTR " ; (literal %d) ", |
| 3341 | 3072 reinterpret_cast<intptr_t>(value), literal_index); |
| 3342 case Translation::COMPILED_STUB_FRAME: | 3073 reinterpret_cast<Object*>(value)->ShortPrint(trace_file); |
| 3343 UNREACHABLE(); | 3074 } |
| 3344 break; | 3075 |
| 3076 return TranslatedValue::NewTagged(this, value); | |
| 3077 } | |
| 3345 } | 3078 } |
| 3346 | 3079 |
| 3347 FATAL("We should never get here - unexpected deopt info."); | 3080 FATAL("We should never get here - unexpected deopt info."); |
| 3348 return SlotRef(); | 3081 return TranslatedValue(nullptr, TranslatedValue::UNKNOWN); |
| 3349 } | 3082 } |
| 3350 | 3083 |
| 3351 | 3084 |
| 3352 SlotRefValueBuilder::SlotRefValueBuilder(JavaScriptFrame* frame, | 3085 TranslatedState::TranslatedState(JavaScriptFrame* frame) |
| 3353 int inlined_jsframe_index, | 3086 : isolate_(nullptr), |
| 3354 int formal_parameter_count) | 3087 stack_frame_pointer_(nullptr), |
| 3355 : current_slot_(0), | 3088 has_adapted_arguments_(false) { |
| 3356 args_length_(-1), | |
| 3357 first_slot_index_(-1), | |
| 3358 should_deoptimize_(false) { | |
| 3359 DisallowHeapAllocation no_gc; | |
| 3360 | |
| 3361 int deopt_index = Safepoint::kNoDeoptimizationIndex; | 3089 int deopt_index = Safepoint::kNoDeoptimizationIndex; |
| 3362 DeoptimizationInputData* data = | 3090 DeoptimizationInputData* data = |
| 3363 static_cast<OptimizedFrame*>(frame)->GetDeoptimizationData(&deopt_index); | 3091 static_cast<OptimizedFrame*>(frame)->GetDeoptimizationData(&deopt_index); |
| 3364 TranslationIterator it(data->TranslationByteArray(), | 3092 TranslationIterator it(data->TranslationByteArray(), |
| 3365 data->TranslationIndex(deopt_index)->value()); | 3093 data->TranslationIndex(deopt_index)->value()); |
| 3366 Translation::Opcode opcode = static_cast<Translation::Opcode>(it.Next()); | 3094 Init(frame->fp(), frame->function(), &it, data->LiteralArray(), |
| 3367 CHECK_EQ(opcode, Translation::BEGIN); | 3095 nullptr /* registers */, nullptr /* trace file */); |
| 3368 it.Next(); // Drop frame count. | 3096 } |
| 3369 | 3097 |
| 3370 stack_frame_id_ = frame->fp(); | 3098 |
| 3371 | 3099 TranslatedState::TranslatedState() |
| 3372 int jsframe_count = it.Next(); | 3100 : isolate_(nullptr), |
| 3373 CHECK_GT(jsframe_count, inlined_jsframe_index); | 3101 stack_frame_pointer_(nullptr), |
| 3374 int jsframes_to_skip = inlined_jsframe_index; | 3102 has_adapted_arguments_(false) {} |
| 3375 int number_of_slots = -1; // Number of slots inside our frame (yet unknown) | 3103 |
| 3376 while (number_of_slots != 0) { | 3104 |
| 3377 opcode = static_cast<Translation::Opcode>(it.Next()); | 3105 void TranslatedState::Init(Address input_frame_pointer, |
| 3378 bool processed = false; | 3106 JSFunction* input_frame_function, |
| 3379 if (opcode == Translation::ARGUMENTS_ADAPTOR_FRAME) { | 3107 TranslationIterator* iterator, |
| 3380 if (jsframes_to_skip == 0) { | 3108 FixedArray* literal_array, RegisterValues* registers, |
| 3381 CHECK_EQ(Translation::NumberOfOperandsFor(opcode), 2); | 3109 FILE* trace_file) { |
| 3382 | 3110 DCHECK(frames_.empty()); |
| 3383 it.Skip(1); // literal id | 3111 |
| 3384 int height = it.Next(); | 3112 isolate_ = literal_array->GetIsolate(); |
| 3385 | 3113 // Read out the 'header' translation. |
| 3386 // Skip the translation command for the receiver. | 3114 Translation::Opcode opcode = |
| 3387 it.Skip(Translation::NumberOfOperandsFor( | 3115 static_cast<Translation::Opcode>(iterator->Next()); |
| 3388 static_cast<Translation::Opcode>(it.Next()))); | 3116 CHECK(opcode == Translation::BEGIN); |
| 3389 | 3117 |
| 3390 // We reached the arguments adaptor frame corresponding to the | 3118 int count = iterator->Next(); |
| 3391 // inlined function in question. Number of arguments is height - 1. | 3119 iterator->Next(); // Drop JS frames count. |
| 3392 first_slot_index_ = slot_refs_.length(); | 3120 |
| 3393 args_length_ = height - 1; | 3121 frames_.reserve(count); |
| 3394 number_of_slots = height - 1; | 3122 |
| 3395 processed = true; | 3123 std::stack<int> nested_counts; |
| 3396 } | 3124 |
| 3397 } else if (opcode == Translation::JS_FRAME) { | 3125 // Read the frames |
| 3398 if (jsframes_to_skip == 0) { | 3126 for (int i = 0; i < count; i++) { |
| 3399 // Skip over operands to advance to the next opcode. | 3127 // Read the frame descriptor. |
| 3400 it.Skip(Translation::NumberOfOperandsFor(opcode)); | 3128 frames_.push_back( |
| 3401 | 3129 CreateNextTranslatedFrame(iterator, literal_array, input_frame_pointer, |
| 3402 // Skip the translation command for the receiver. | 3130 input_frame_function, trace_file)); |
| 3403 it.Skip(Translation::NumberOfOperandsFor( | 3131 TranslatedFrame& frame = frames_.back(); |
| 3404 static_cast<Translation::Opcode>(it.Next()))); | 3132 |
| 3405 | 3133 // Read the values. |
| 3406 // We reached the frame corresponding to the inlined function | 3134 int values_to_process = frame.GetValueCount(); |
| 3407 // in question. Process the translation commands for the | 3135 while (values_to_process > 0 || !nested_counts.empty()) { |
| 3408 // arguments. Number of arguments is equal to the number of | 3136 if (trace_file != nullptr) { |
| 3409 // format parameter count. | 3137 if (nested_counts.empty()) { |
| 3410 first_slot_index_ = slot_refs_.length(); | 3138 // For top level values, print the value number. |
| 3411 args_length_ = formal_parameter_count; | 3139 PrintF(trace_file, " %3i: ", |
| 3412 number_of_slots = formal_parameter_count; | 3140 frame.GetValueCount() - values_to_process); |
| 3413 processed = true; | 3141 } else { |
| 3414 } | 3142 // Take care of indenting for nested values. |
| 3415 jsframes_to_skip--; | 3143 PrintF(trace_file, " "); |
| 3416 } else if (opcode != Translation::BEGIN && | 3144 for (size_t j = 0; j < nested_counts.size(); j++) { |
| 3417 opcode != Translation::CONSTRUCT_STUB_FRAME && | 3145 PrintF(trace_file, " "); |
| 3418 opcode != Translation::GETTER_STUB_FRAME && | 3146 } |
| 3419 opcode != Translation::SETTER_STUB_FRAME && | |
| 3420 opcode != Translation::COMPILED_STUB_FRAME) { | |
| 3421 slot_refs_.Add(ComputeSlotForNextArgument(opcode, &it, data, frame)); | |
| 3422 | |
| 3423 if (first_slot_index_ >= 0) { | |
| 3424 // We have found the beginning of our frame -> make sure we count | |
| 3425 // the nested slots of captured objects | |
| 3426 number_of_slots--; | |
| 3427 SlotRef& slot = slot_refs_.last(); | |
| 3428 CHECK_NE(slot.Representation(), SlotRef::ARGUMENTS_OBJECT); | |
| 3429 number_of_slots += slot.GetChildrenCount(); | |
| 3430 if (slot.Representation() == SlotRef::DEFERRED_OBJECT || | |
| 3431 slot.Representation() == SlotRef::DUPLICATE_OBJECT) { | |
| 3432 should_deoptimize_ = true; | |
| 3433 } | 3147 } |
| 3434 } | 3148 } |
| 3435 | 3149 |
| 3436 processed = true; | 3150 TranslatedValue value = CreateNextTranslatedValue( |
| 3437 } | 3151 i, static_cast<int>(frame.values_.size()), iterator, literal_array, |
| 3438 if (!processed) { | 3152 input_frame_pointer, registers, trace_file); |
| 3439 // Skip over operands to advance to the next opcode. | 3153 frame.Add(value); |
| 3440 it.Skip(Translation::NumberOfOperandsFor(opcode)); | 3154 |
| 3441 } | 3155 if (trace_file != nullptr) { |
| 3442 } | 3156 PrintF(trace_file, "\n"); |
| 3443 if (should_deoptimize_) { | 3157 } |
| 3444 List<JSFunction*> functions(2); | 3158 |
| 3445 frame->GetFunctions(&functions); | 3159 // Update the value count and resolve the nesting. |
| 3446 Deoptimizer::DeoptimizeFunction(functions[0]); | 3160 values_to_process--; |
| 3447 } | 3161 int children_count = value.GetChildrenCount(); |
| 3448 } | 3162 if (children_count > 0) { |
| 3449 | 3163 nested_counts.push(values_to_process); |
| 3450 | 3164 values_to_process = children_count; |
| 3451 Handle<Object> SlotRef::GetValue(Isolate* isolate) { | 3165 } else { |
| 3452 switch (representation_) { | 3166 while (values_to_process == 0 && !nested_counts.empty()) { |
| 3453 case TAGGED: { | 3167 values_to_process = nested_counts.top(); |
| 3454 Handle<Object> value(Memory::Object_at(addr_), isolate); | 3168 nested_counts.pop(); |
| 3169 } | |
| 3170 } | |
| 3171 } | |
| 3172 } | |
| 3173 | |
| 3174 CHECK(!iterator->HasNext() || | |
| 3175 static_cast<Translation::Opcode>(iterator->Next()) == | |
| 3176 Translation::BEGIN); | |
| 3177 } | |
| 3178 | |
| 3179 | |
| 3180 void TranslatedState::Prepare(bool has_adapted_arguments, | |
| 3181 Address stack_frame_pointer) { | |
| 3182 for (auto& frame : frames_) { | |
| 3183 frame.Handlify(isolate_); | |
| 3184 } | |
| 3185 | |
| 3186 stack_frame_pointer_ = stack_frame_pointer; | |
| 3187 has_adapted_arguments_ = has_adapted_arguments; | |
| 3188 | |
| 3189 UpdateFromPreviouslyMaterializedObjects(); | |
| 3190 } | |
| 3191 | |
| 3192 | |
| 3193 Handle<Object> TranslatedState::MaterializeAt(int frame_index, | |
| 3194 int* value_index) { | |
| 3195 TranslatedFrame* frame = &(frames_[frame_index]); | |
| 3196 DCHECK(static_cast<size_t>(*value_index) < frame->values_.size()); | |
| 3197 | |
| 3198 TranslatedValue* slot = &(frame->values_[*value_index]); | |
| 3199 (*value_index)++; | |
| 3200 | |
| 3201 switch (slot->kind()) { | |
| 3202 case TranslatedValue::TAGGED: | |
| 3203 case TranslatedValue::INT32: | |
| 3204 case TranslatedValue::UINT32: | |
| 3205 case TranslatedValue::BOOLBIT: | |
| 3206 case TranslatedValue::DOUBLE: { | |
| 3207 slot->MaterializeSimple(); | |
| 3208 Handle<Object> value = slot->GetValue(); | |
| 3455 if (value->IsMutableHeapNumber()) { | 3209 if (value->IsMutableHeapNumber()) { |
| 3456 HeapNumber::cast(*value)->set_map(isolate->heap()->heap_number_map()); | 3210 HeapNumber::cast(*value)->set_map(isolate()->heap()->heap_number_map()); |
| 3457 } | 3211 } |
| 3458 return value; | 3212 return value; |
| 3459 } | 3213 } |
| 3460 | 3214 |
| 3461 case INT32: { | 3215 case TranslatedValue::ARGUMENTS_OBJECT: { |
| 3462 #if V8_TARGET_BIG_ENDIAN && V8_HOST_ARCH_64_BIT | 3216 int length = slot->GetChildrenCount(); |
| 3463 int value = Memory::int32_at(addr_ + kIntSize); | 3217 Handle<JSObject> arguments; |
| 3464 #else | 3218 if (GetAdaptedArguments(&arguments, frame_index)) { |
| 3465 int value = Memory::int32_at(addr_); | 3219 // Store the materialized object and consume the nested values. |
| 3466 #endif | 3220 for (int i = 0; i < length; ++i) { |
| 3467 if (Smi::IsValid(value)) { | 3221 MaterializeAt(frame_index, value_index); |
| 3468 return Handle<Object>(Smi::FromInt(value), isolate); | 3222 } |
| 3469 } else { | 3223 } else { |
| 3470 return isolate->factory()->NewNumberFromInt(value); | 3224 Handle<JSFunction> function = frame->function(); |
| 3471 } | 3225 arguments = isolate_->factory()->NewArgumentsObject(function, length); |
| 3472 } | 3226 Handle<FixedArray> array = isolate_->factory()->NewFixedArray(length); |
| 3473 | 3227 DCHECK_EQ(array->length(), length); |
| 3474 case UINT32: { | 3228 arguments->set_elements(*array); |
| 3475 #if V8_TARGET_BIG_ENDIAN && V8_HOST_ARCH_64_BIT | 3229 for (int i = 0; i < length; ++i) { |
| 3476 uint32_t value = Memory::uint32_at(addr_ + kIntSize); | 3230 Handle<Object> value = MaterializeAt(frame_index, value_index); |
| 3477 #else | 3231 array->set(i, *value); |
| 3478 uint32_t value = Memory::uint32_at(addr_); | 3232 } |
| 3479 #endif | 3233 } |
| 3480 if (value <= static_cast<uint32_t>(Smi::kMaxValue)) { | 3234 slot->value_ = arguments; |
| 3481 return Handle<Object>(Smi::FromInt(static_cast<int>(value)), isolate); | 3235 return arguments; |
| 3482 } else { | 3236 } |
| 3483 return isolate->factory()->NewNumber(static_cast<double>(value)); | 3237 case TranslatedValue::DEFERRED_OBJECT: { |
| 3484 } | 3238 int length = slot->GetChildrenCount(); |
| 3485 } | 3239 |
| 3486 | 3240 // The map must be a tagged object. |
| 3487 case BOOLBIT: { | 3241 CHECK(frame->values_[*value_index].kind() == TranslatedValue::TAGGED); |
| 3488 #if V8_TARGET_BIG_ENDIAN && V8_HOST_ARCH_64_BIT | 3242 |
| 3489 uint32_t value = Memory::uint32_at(addr_ + kIntSize); | 3243 Handle<Object> result; |
| 3490 #else | 3244 if (slot->value_.ToHandle(&result)) { |
| 3491 uint32_t value = Memory::uint32_at(addr_); | 3245 // This has been previously materialized, return the previous value. |
| 3492 #endif | 3246 // We still need to skip all the nested objects. |
| 3493 if (value == 0) { | 3247 for (int i = 0; i < length; i++) { |
| 3494 return isolate->factory()->false_value(); | 3248 MaterializeAt(frame_index, value_index); |
| 3495 } else { | 3249 } |
| 3496 DCHECK_EQ(1U, value); | 3250 |
| 3497 return isolate->factory()->true_value(); | 3251 return result; |
| 3498 } | 3252 } |
| 3499 } | 3253 |
| 3500 | 3254 Handle<Object> map_object = MaterializeAt(frame_index, value_index); |
| 3501 case DOUBLE: { | 3255 Handle<Map> map = |
| 3502 double value = read_double_value(addr_); | 3256 Map::GeneralizeAllFieldRepresentations(Handle<Map>::cast(map_object)); |
| 3503 return isolate->factory()->NewNumber(value); | |
| 3504 } | |
| 3505 | |
| 3506 case LITERAL: | |
| 3507 return literal_; | |
| 3508 | |
| 3509 default: | |
| 3510 FATAL("We should never get here - unexpected deopt info."); | |
| 3511 return Handle<Object>::null(); | |
| 3512 } | |
| 3513 } | |
| 3514 | |
| 3515 | |
| 3516 void SlotRefValueBuilder::Prepare(Isolate* isolate) { | |
| 3517 MaterializedObjectStore* materialized_store = | |
| 3518 isolate->materialized_object_store(); | |
| 3519 previously_materialized_objects_ = materialized_store->Get(stack_frame_id_); | |
| 3520 prev_materialized_count_ = previously_materialized_objects_.is_null() | |
| 3521 ? 0 : previously_materialized_objects_->length(); | |
| 3522 | |
| 3523 // Skip any materialized objects of the inlined "parent" frames. | |
| 3524 // (Note that we still need to materialize them because they might be | |
| 3525 // referred to as duplicated objects.) | |
| 3526 while (current_slot_ < first_slot_index_) { | |
| 3527 GetNext(isolate, 0); | |
| 3528 } | |
| 3529 CHECK_EQ(current_slot_, first_slot_index_); | |
| 3530 } | |
| 3531 | |
| 3532 | |
| 3533 Handle<Object> SlotRefValueBuilder::GetPreviouslyMaterialized( | |
| 3534 Isolate* isolate, int length) { | |
| 3535 int object_index = materialized_objects_.length(); | |
| 3536 Handle<Object> return_value = Handle<Object>( | |
| 3537 previously_materialized_objects_->get(object_index), isolate); | |
| 3538 materialized_objects_.Add(return_value); | |
| 3539 | |
| 3540 // Now need to skip all the nested objects (and possibly read them from | |
| 3541 // the materialization store, too). | |
| 3542 for (int i = 0; i < length; i++) { | |
| 3543 SlotRef& slot = slot_refs_[current_slot_]; | |
| 3544 current_slot_++; | |
| 3545 | |
| 3546 // We need to read all the nested objects - add them to the | |
| 3547 // number of objects we need to process. | |
| 3548 length += slot.GetChildrenCount(); | |
| 3549 | |
| 3550 // Put the nested deferred/duplicate objects into our materialization | |
| 3551 // array. | |
| 3552 if (slot.Representation() == SlotRef::DEFERRED_OBJECT || | |
| 3553 slot.Representation() == SlotRef::DUPLICATE_OBJECT) { | |
| 3554 int nested_object_index = materialized_objects_.length(); | |
| 3555 Handle<Object> nested_object = Handle<Object>( | |
| 3556 previously_materialized_objects_->get(nested_object_index), | |
| 3557 isolate); | |
| 3558 materialized_objects_.Add(nested_object); | |
| 3559 } | |
| 3560 } | |
| 3561 | |
| 3562 return return_value; | |
| 3563 } | |
| 3564 | |
| 3565 | |
| 3566 Handle<Object> SlotRefValueBuilder::GetNext(Isolate* isolate, int lvl) { | |
| 3567 SlotRef& slot = slot_refs_[current_slot_]; | |
| 3568 current_slot_++; | |
| 3569 switch (slot.Representation()) { | |
| 3570 case SlotRef::TAGGED: | |
| 3571 case SlotRef::INT32: | |
| 3572 case SlotRef::UINT32: | |
| 3573 case SlotRef::BOOLBIT: | |
| 3574 case SlotRef::DOUBLE: | |
| 3575 case SlotRef::LITERAL: | |
| 3576 return slot.GetValue(isolate); | |
| 3577 | |
| 3578 case SlotRef::ARGUMENTS_OBJECT: { | |
| 3579 // We should never need to materialize an arguments object, | |
| 3580 // but we still need to put something into the array | |
| 3581 // so that the indexing is consistent. | |
| 3582 materialized_objects_.Add(isolate->factory()->undefined_value()); | |
| 3583 int length = slot.GetChildrenCount(); | |
| 3584 for (int i = 0; i < length; ++i) { | |
| 3585 // We don't need the argument, just ignore it | |
| 3586 GetNext(isolate, lvl + 1); | |
| 3587 } | |
| 3588 return isolate->factory()->undefined_value(); | |
| 3589 } | |
| 3590 case SlotRef::DEFERRED_OBJECT: { | |
| 3591 int length = slot.GetChildrenCount(); | |
| 3592 CHECK(slot_refs_[current_slot_].Representation() == SlotRef::LITERAL || | |
| 3593 slot_refs_[current_slot_].Representation() == SlotRef::TAGGED); | |
| 3594 | |
| 3595 int object_index = materialized_objects_.length(); | |
| 3596 if (object_index < prev_materialized_count_) { | |
| 3597 return GetPreviouslyMaterialized(isolate, length); | |
| 3598 } | |
| 3599 | |
| 3600 Handle<Object> map_object = slot_refs_[current_slot_].GetValue(isolate); | |
| 3601 Handle<Map> map = Map::GeneralizeAllFieldRepresentations( | |
| 3602 Handle<Map>::cast(map_object)); | |
| 3603 current_slot_++; | |
| 3604 // TODO(jarin) this should be unified with the code in | |
| 3605 // Deoptimizer::MaterializeNextHeapObject() | |
| 3606 switch (map->instance_type()) { | 3257 switch (map->instance_type()) { |
| 3607 case MUTABLE_HEAP_NUMBER_TYPE: | 3258 case MUTABLE_HEAP_NUMBER_TYPE: |
| 3608 case HEAP_NUMBER_TYPE: { | 3259 case HEAP_NUMBER_TYPE: { |
| 3609 // Reuse the HeapNumber value directly as it is already properly | 3260 // Reuse the HeapNumber value directly as it is already properly |
| 3610 // tagged and skip materializing the HeapNumber explicitly. | 3261 // tagged and skip materializing the HeapNumber explicitly. |
| 3611 Handle<Object> object = GetNext(isolate, lvl + 1); | 3262 Handle<Object> object = MaterializeAt(frame_index, value_index); |
| 3612 materialized_objects_.Add(object); | 3263 slot->value_ = object; |
| 3613 // On 32-bit architectures, there is an extra slot there because | 3264 // On 32-bit architectures, there is an extra slot there because |
| 3614 // the escape analysis calculates the number of slots as | 3265 // the escape analysis calculates the number of slots as |
| 3615 // object-size/pointer-size. To account for this, we read out | 3266 // object-size/pointer-size. To account for this, we read out |
| 3616 // any extra slots. | 3267 // any extra slots. |
| 3617 for (int i = 0; i < length - 2; i++) { | 3268 for (int i = 0; i < length - 2; i++) { |
| 3618 GetNext(isolate, lvl + 1); | 3269 MaterializeAt(frame_index, value_index); |
| 3619 } | 3270 } |
| 3620 return object; | 3271 return object; |
| 3621 } | 3272 } |
| 3622 case JS_OBJECT_TYPE: { | 3273 case JS_OBJECT_TYPE: { |
| 3623 Handle<JSObject> object = | 3274 Handle<JSObject> object = |
| 3624 isolate->factory()->NewJSObjectFromMap(map, NOT_TENURED, false); | 3275 isolate_->factory()->NewJSObjectFromMap(map, NOT_TENURED, false); |
| 3625 materialized_objects_.Add(object); | 3276 slot->value_ = object; |
| 3626 Handle<Object> properties = GetNext(isolate, lvl + 1); | 3277 Handle<Object> properties = MaterializeAt(frame_index, value_index); |
| 3627 Handle<Object> elements = GetNext(isolate, lvl + 1); | 3278 Handle<Object> elements = MaterializeAt(frame_index, value_index); |
| 3628 object->set_properties(FixedArray::cast(*properties)); | 3279 object->set_properties(FixedArray::cast(*properties)); |
| 3629 object->set_elements(FixedArrayBase::cast(*elements)); | 3280 object->set_elements(FixedArrayBase::cast(*elements)); |
| 3630 for (int i = 0; i < length - 3; ++i) { | 3281 for (int i = 0; i < length - 3; ++i) { |
| 3631 Handle<Object> value = GetNext(isolate, lvl + 1); | 3282 Handle<Object> value = MaterializeAt(frame_index, value_index); |
| 3632 FieldIndex index = FieldIndex::ForPropertyIndex(object->map(), i); | 3283 FieldIndex index = FieldIndex::ForPropertyIndex(object->map(), i); |
| 3633 object->FastPropertyAtPut(index, *value); | 3284 object->FastPropertyAtPut(index, *value); |
| 3634 } | 3285 } |
| 3635 return object; | 3286 return object; |
| 3636 } | 3287 } |
| 3637 case JS_ARRAY_TYPE: { | 3288 case JS_ARRAY_TYPE: { |
| 3638 Handle<JSArray> object = | 3289 Handle<JSArray> object = |
| 3639 isolate->factory()->NewJSArray(0, map->elements_kind()); | 3290 isolate_->factory()->NewJSArray(0, map->elements_kind()); |
| 3640 materialized_objects_.Add(object); | 3291 slot->value_ = object; |
| 3641 Handle<Object> properties = GetNext(isolate, lvl + 1); | 3292 Handle<Object> properties = MaterializeAt(frame_index, value_index); |
| 3642 Handle<Object> elements = GetNext(isolate, lvl + 1); | 3293 Handle<Object> elements = MaterializeAt(frame_index, value_index); |
| 3643 Handle<Object> length = GetNext(isolate, lvl + 1); | 3294 Handle<Object> length = MaterializeAt(frame_index, value_index); |
| 3644 object->set_properties(FixedArray::cast(*properties)); | 3295 object->set_properties(FixedArray::cast(*properties)); |
| 3645 object->set_elements(FixedArrayBase::cast(*elements)); | 3296 object->set_elements(FixedArrayBase::cast(*elements)); |
| 3646 object->set_length(*length); | 3297 object->set_length(*length); |
| 3647 return object; | 3298 return object; |
| 3648 } | 3299 } |
| 3649 default: | 3300 default: |
| 3650 PrintF(stderr, | 3301 PrintF(stderr, "[couldn't handle instance type %d]\n", |
| 3651 "[couldn't handle instance type %d]\n", map->instance_type()); | 3302 map->instance_type()); |
| 3652 UNREACHABLE(); | 3303 FATAL("unreachable"); |
| 3653 break; | 3304 return Handle<Object>::null(); |
| 3654 } | 3305 } |
| 3655 UNREACHABLE(); | 3306 UNREACHABLE(); |
| 3656 break; | 3307 break; |
| 3657 } | 3308 } |
| 3658 | 3309 |
| 3659 case SlotRef::DUPLICATE_OBJECT: { | 3310 case TranslatedValue::DUPLICATE_OBJECT: { |
| 3660 int object_index = slot.DuplicateObjectId(); | 3311 int object_index = slot->ObjectIndex(); |
| 3661 Handle<Object> object = materialized_objects_[object_index]; | 3312 TranslatedState::ObjectPosition pos = object_positions_[object_index]; |
| 3662 materialized_objects_.Add(object); | 3313 |
| 3314 // Make sure the duplicate is refering to a previous object. | |
| 3315 DCHECK(pos.frame_index_ < frame_index || | |
| 3316 (pos.frame_index_ == frame_index && | |
| 3317 pos.value_index_ < *value_index - 1)); | |
| 3318 | |
| 3319 Handle<Object> object = | |
| 3320 frames_[pos.frame_index_].values_[pos.value_index_].GetValue(); | |
| 3321 | |
| 3322 // The object should have a (non-sentinel) value. | |
| 3323 DCHECK(!object.is_null() && | |
| 3324 !object.is_identical_to(isolate_->factory()->arguments_marker())); | |
| 3325 | |
| 3326 slot->value_ = object; | |
| 3663 return object; | 3327 return object; |
| 3664 } | 3328 } |
| 3665 default: | 3329 |
| 3330 case TranslatedValue::UNKNOWN: | |
| 3666 UNREACHABLE(); | 3331 UNREACHABLE(); |
| 3667 break; | 3332 break; |
| 3668 } | 3333 } |
| 3669 | 3334 |
| 3670 FATAL("We should never get here - unexpected deopt slot kind."); | 3335 FATAL("We should never get here - unexpected deopt slot kind."); |
| 3671 return Handle<Object>::null(); | 3336 return Handle<Object>::null(); |
| 3672 } | 3337 } |
| 3673 | 3338 |
| 3674 | 3339 |
| 3675 void SlotRefValueBuilder::Finish(Isolate* isolate) { | 3340 Handle<Object> TranslatedState::MaterializeObjectAt(int object_index) { |
| 3676 // We should have processed all the slots | 3341 TranslatedState::ObjectPosition pos = object_positions_[object_index]; |
| 3677 CHECK_EQ(slot_refs_.length(), current_slot_); | 3342 return MaterializeAt(pos.frame_index_, &(pos.value_index_)); |
| 3343 } | |
| 3678 | 3344 |
| 3679 if (should_deoptimize_ && | 3345 |
| 3680 materialized_objects_.length() > prev_materialized_count_) { | 3346 bool TranslatedState::GetAdaptedArguments(Handle<JSObject>* result, |
| 3681 // We have materialized some new objects and they might be accessible | 3347 int frame_index) { |
| 3682 // from the arguments object, so we have to store them | 3348 if (frame_index == 0) { |
| 3683 // to prevent duplicate materialization. | 3349 // Top level frame -> we need to go to the parent frame on the stack. |
| 3684 Handle<FixedArray> array = isolate->factory()->NewFixedArray( | 3350 if (!has_adapted_arguments_) return false; |
| 3685 materialized_objects_.length()); | 3351 |
| 3686 for (int i = 0; i < materialized_objects_.length(); i++) { | 3352 // This is top level frame, so we need to go to the stack to get |
| 3687 array->set(i, *(materialized_objects_.at(i))); | 3353 // this function's argument. (Note that this relies on not inlining |
| 3354 // recursive functions!) | |
| 3355 Handle<JSFunction> function = frames_[frame_index].function(); | |
| 3356 *result = Handle<JSObject>::cast(Accessors::FunctionGetArguments(function)); | |
| 3357 return true; | |
| 3358 } else { | |
| 3359 TranslatedFrame* previous_frame = &(frames_[frame_index]); | |
| 3360 if (previous_frame->kind() != TranslatedFrame::ArgumentsAdaptor) { | |
| 3361 return false; | |
| 3688 } | 3362 } |
| 3689 isolate->materialized_object_store()->Set(stack_frame_id_, array); | 3363 // We get the adapted arguments from the parent translation. |
| 3364 int length = previous_frame->GetValueCount(); | |
| 3365 Handle<JSFunction> function = previous_frame->function(); | |
| 3366 Handle<JSObject> arguments = | |
| 3367 isolate_->factory()->NewArgumentsObject(function, length); | |
| 3368 Handle<FixedArray> array = isolate_->factory()->NewFixedArray(length); | |
| 3369 arguments->set_elements(*array); | |
| 3370 TranslatedFrame::ValueIterator arg_iterator = previous_frame->begin(); | |
| 3371 for (int i = 0; i < length; ++i) { | |
| 3372 Handle<Object> value = arg_iterator->GetValue(); | |
| 3373 array->set(i, *value); | |
| 3374 arg_iterator++; | |
| 3375 } | |
| 3376 CHECK(arg_iterator == previous_frame->end()); | |
| 3377 *result = arguments; | |
| 3378 return true; | |
| 3690 } | 3379 } |
| 3691 } | 3380 } |
| 3692 | 3381 |
| 3693 | 3382 |
| 3694 Handle<FixedArray> MaterializedObjectStore::Get(Address fp) { | 3383 TranslatedFrame* TranslatedState::GetArgumentsInfoFromJSFrameIndex( |
| 3695 int index = StackIdToIndex(fp); | 3384 int jsframe_index, int* args_count) { |
| 3696 if (index == -1) { | 3385 for (size_t i = 0; i < frames_.size(); i++) { |
| 3697 return Handle<FixedArray>::null(); | 3386 if (frames_[i].kind() == TranslatedFrame::Function) { |
| 3387 if (jsframe_index > 0) { | |
| 3388 jsframe_index--; | |
| 3389 } else { | |
| 3390 // We have the JS function frame, now check if it has arguments adaptor. | |
| 3391 if (i > 0 && | |
| 3392 frames_[i - 1].kind() == TranslatedFrame::ArgumentsAdaptor) { | |
| 3393 *args_count = frames_[i - 1].height(); | |
| 3394 return &(frames_[i - 1]); | |
| 3395 } | |
| 3396 *args_count = | |
| 3397 frames_[i].function()->shared()->internal_formal_parameter_count() + | |
| 3398 1; | |
| 3399 return &(frames_[i]); | |
| 3400 } | |
| 3401 } | |
| 3698 } | 3402 } |
| 3699 Handle<FixedArray> array = GetStackEntries(); | 3403 return nullptr; |
| 3700 CHECK_GT(array->length(), index); | |
| 3701 return Handle<FixedArray>::cast(Handle<Object>(array->get(index), | |
| 3702 isolate())); | |
| 3703 } | 3404 } |
| 3704 | 3405 |
| 3705 | 3406 |
| 3706 void MaterializedObjectStore::Set(Address fp, | 3407 void TranslatedState::StoreMaterializedValuesAndDeopt() { |
| 3707 Handle<FixedArray> materialized_objects) { | 3408 MaterializedObjectStore* materialized_store = |
| 3708 int index = StackIdToIndex(fp); | 3409 isolate_->materialized_object_store(); |
| 3709 if (index == -1) { | 3410 Handle<FixedArray> previously_materialized_objects = |
| 3710 index = frame_fps_.length(); | 3411 materialized_store->Get(stack_frame_pointer_); |
| 3711 frame_fps_.Add(fp); | 3412 |
| 3413 Handle<Object> marker = isolate_->factory()->arguments_marker(); | |
| 3414 | |
| 3415 int length = static_cast<int>(object_positions_.size()); | |
| 3416 bool new_store = false; | |
| 3417 if (previously_materialized_objects.is_null()) { | |
| 3418 previously_materialized_objects = | |
| 3419 isolate_->factory()->NewFixedArray(length); | |
| 3420 for (int i = 0; i < length; i++) { | |
| 3421 previously_materialized_objects->set(i, *marker); | |
| 3422 } | |
| 3423 new_store = true; | |
| 3712 } | 3424 } |
| 3713 | 3425 |
| 3714 Handle<FixedArray> array = EnsureStackEntries(index + 1); | 3426 DCHECK_EQ(length, previously_materialized_objects->length()); |
| 3715 array->set(index, *materialized_objects); | |
| 3716 } | |
| 3717 | 3427 |
| 3428 bool value_changed = false; | |
| 3429 for (int i = 0; i < length; i++) { | |
| 3430 TranslatedState::ObjectPosition pos = object_positions_[i]; | |
| 3431 TranslatedValue* value_info = | |
| 3432 &(frames_[pos.frame_index_].values_[pos.value_index_]); | |
| 3718 | 3433 |
| 3719 bool MaterializedObjectStore::Remove(Address fp) { | 3434 DCHECK(value_info->IsMaterializedObject()); |
| 3720 int index = StackIdToIndex(fp); | |
| 3721 if (index == -1) { | |
| 3722 return false; | |
| 3723 } | |
| 3724 CHECK_GE(index, 0); | |
| 3725 | 3435 |
| 3726 frame_fps_.Remove(index); | 3436 Handle<Object> value(value_info->GetRawValue(), isolate_); |
| 3727 FixedArray* array = isolate()->heap()->materialized_objects(); | |
| 3728 CHECK_LT(index, array->length()); | |
| 3729 for (int i = index; i < frame_fps_.length(); i++) { | |
| 3730 array->set(i, array->get(i + 1)); | |
| 3731 } | |
| 3732 array->set(frame_fps_.length(), isolate()->heap()->undefined_value()); | |
| 3733 return true; | |
| 3734 } | |
| 3735 | 3437 |
| 3736 | 3438 if (!value.is_identical_to(marker)) { |
| 3737 int MaterializedObjectStore::StackIdToIndex(Address fp) { | 3439 if (previously_materialized_objects->get(i) == *marker) { |
| 3738 for (int i = 0; i < frame_fps_.length(); i++) { | 3440 previously_materialized_objects->set(i, *value); |
| 3739 if (frame_fps_[i] == fp) { | 3441 value_changed = true; |
| 3740 return i; | 3442 } else { |
| 3443 DCHECK(previously_materialized_objects->get(i) == *value); | |
| 3444 } | |
| 3741 } | 3445 } |
| 3742 } | 3446 } |
| 3743 return -1; | 3447 if (new_store && value_changed) { |
| 3744 } | 3448 materialized_store->Set(stack_frame_pointer_, |
| 3745 | 3449 previously_materialized_objects); |
| 3746 | 3450 DCHECK(frames_[0].kind() == TranslatedFrame::Function); |
| 3747 Handle<FixedArray> MaterializedObjectStore::GetStackEntries() { | 3451 Deoptimizer::DeoptimizeFunction(*(frames_[0].function())); |
| 3748 return Handle<FixedArray>(isolate()->heap()->materialized_objects()); | |
| 3749 } | |
| 3750 | |
| 3751 | |
| 3752 Handle<FixedArray> MaterializedObjectStore::EnsureStackEntries(int length) { | |
| 3753 Handle<FixedArray> array = GetStackEntries(); | |
| 3754 if (array->length() >= length) { | |
| 3755 return array; | |
| 3756 } | |
| 3757 | |
| 3758 int new_length = length > 10 ? length : 10; | |
| 3759 if (new_length < 2 * array->length()) { | |
| 3760 new_length = 2 * array->length(); | |
| 3761 } | |
| 3762 | |
| 3763 Handle<FixedArray> new_array = | |
| 3764 isolate()->factory()->NewFixedArray(new_length, TENURED); | |
| 3765 for (int i = 0; i < array->length(); i++) { | |
| 3766 new_array->set(i, array->get(i)); | |
| 3767 } | |
| 3768 for (int i = array->length(); i < length; i++) { | |
| 3769 new_array->set(i, isolate()->heap()->undefined_value()); | |
| 3770 } | |
| 3771 isolate()->heap()->public_set_materialized_objects(*new_array); | |
| 3772 return new_array; | |
| 3773 } | |
| 3774 | |
| 3775 | |
| 3776 DeoptimizedFrameInfo::DeoptimizedFrameInfo(Deoptimizer* deoptimizer, | |
| 3777 int frame_index, | |
| 3778 bool has_arguments_adaptor, | |
| 3779 bool has_construct_stub) { | |
| 3780 FrameDescription* output_frame = deoptimizer->output_[frame_index]; | |
| 3781 function_ = output_frame->GetFunction(); | |
| 3782 context_ = reinterpret_cast<Object*>(output_frame->GetContext()); | |
| 3783 has_construct_stub_ = has_construct_stub; | |
| 3784 expression_count_ = output_frame->GetExpressionCount(); | |
| 3785 expression_stack_ = new Object*[expression_count_]; | |
| 3786 // Get the source position using the unoptimized code. | |
| 3787 Address pc = reinterpret_cast<Address>(output_frame->GetPc()); | |
| 3788 Code* code = Code::cast(deoptimizer->isolate()->FindCodeObject(pc)); | |
| 3789 source_position_ = code->SourcePosition(pc); | |
| 3790 | |
| 3791 for (int i = 0; i < expression_count_; i++) { | |
| 3792 SetExpression(i, output_frame->GetExpression(i)); | |
| 3793 } | |
| 3794 | |
| 3795 if (has_arguments_adaptor) { | |
| 3796 output_frame = deoptimizer->output_[frame_index - 1]; | |
| 3797 CHECK_EQ(output_frame->GetFrameType(), StackFrame::ARGUMENTS_ADAPTOR); | |
| 3798 } | |
| 3799 | |
| 3800 parameters_count_ = output_frame->ComputeParametersCount(); | |
| 3801 parameters_ = new Object*[parameters_count_]; | |
| 3802 for (int i = 0; i < parameters_count_; i++) { | |
| 3803 SetParameter(i, output_frame->GetParameter(i)); | |
| 3804 } | 3452 } |
| 3805 } | 3453 } |
| 3806 | 3454 |
| 3807 | 3455 |
| 3808 DeoptimizedFrameInfo::~DeoptimizedFrameInfo() { | 3456 void TranslatedState::UpdateFromPreviouslyMaterializedObjects() { |
| 3809 delete[] expression_stack_; | 3457 MaterializedObjectStore* materialized_store = |
| 3810 delete[] parameters_; | 3458 isolate_->materialized_object_store(); |
| 3459 Handle<FixedArray> previously_materialized_objects = | |
| 3460 materialized_store->Get(stack_frame_pointer_); | |
| 3461 | |
| 3462 // If we have no previously materialized objects, there is nothing to do. | |
| 3463 if (previously_materialized_objects.is_null()) return; | |
| 3464 | |
| 3465 Handle<Object> marker = isolate_->factory()->arguments_marker(); | |
| 3466 | |
| 3467 int length = static_cast<int>(object_positions_.size()); | |
| 3468 DCHECK_EQ(length, previously_materialized_objects->length()); | |
| 3469 | |
| 3470 for (int i = 0; i < length; i++) { | |
| 3471 // For a previously materialized objects, inject their value into the | |
| 3472 // translated values. | |
| 3473 if (previously_materialized_objects->get(i) != *marker) { | |
| 3474 TranslatedState::ObjectPosition pos = object_positions_[i]; | |
| 3475 TranslatedValue* value_info = | |
| 3476 &(frames_[pos.frame_index_].values_[pos.value_index_]); | |
| 3477 DCHECK(value_info->IsMaterializedObject()); | |
| 3478 | |
| 3479 value_info->value_ = | |
| 3480 Handle<Object>(previously_materialized_objects->get(i), isolate_); | |
| 3481 } | |
| 3482 } | |
| 3811 } | 3483 } |
| 3812 | 3484 |
| 3813 | |
| 3814 void DeoptimizedFrameInfo::Iterate(ObjectVisitor* v) { | |
| 3815 v->VisitPointer(bit_cast<Object**>(&function_)); | |
| 3816 v->VisitPointer(&context_); | |
| 3817 v->VisitPointers(parameters_, parameters_ + parameters_count_); | |
| 3818 v->VisitPointers(expression_stack_, expression_stack_ + expression_count_); | |
| 3819 } | |
| 3820 | |
| 3821 | |
| 3822 const char* Deoptimizer::GetDeoptReason(DeoptReason deopt_reason) { | |
| 3823 DCHECK(deopt_reason < kLastDeoptReason); | |
| 3824 #define DEOPT_MESSAGES_TEXTS(C, T) T, | |
| 3825 static const char* deopt_messages_[] = { | |
| 3826 DEOPT_MESSAGES_LIST(DEOPT_MESSAGES_TEXTS)}; | |
| 3827 #undef DEOPT_MESSAGES_TEXTS | |
| 3828 return deopt_messages_[deopt_reason]; | |
| 3829 } | |
| 3830 | |
| 3831 | |
| 3832 Deoptimizer::DeoptInfo Deoptimizer::GetDeoptInfo(Code* code, Address pc) { | |
| 3833 SourcePosition last_position = SourcePosition::Unknown(); | |
| 3834 Deoptimizer::DeoptReason last_reason = Deoptimizer::kNoReason; | |
| 3835 int mask = RelocInfo::ModeMask(RelocInfo::DEOPT_REASON) | | |
| 3836 RelocInfo::ModeMask(RelocInfo::POSITION); | |
| 3837 for (RelocIterator it(code, mask); !it.done(); it.next()) { | |
| 3838 RelocInfo* info = it.rinfo(); | |
| 3839 if (info->pc() >= pc) return DeoptInfo(last_position, NULL, last_reason); | |
| 3840 if (info->rmode() == RelocInfo::POSITION) { | |
| 3841 int raw_position = static_cast<int>(info->data()); | |
| 3842 last_position = raw_position ? SourcePosition::FromRaw(raw_position) | |
| 3843 : SourcePosition::Unknown(); | |
| 3844 } else if (info->rmode() == RelocInfo::DEOPT_REASON) { | |
| 3845 last_reason = static_cast<Deoptimizer::DeoptReason>(info->data()); | |
| 3846 } | |
| 3847 } | |
| 3848 return DeoptInfo(SourcePosition::Unknown(), NULL, Deoptimizer::kNoReason); | |
| 3849 } | |
| 3850 } // namespace internal | 3485 } // namespace internal |
| 3851 } // namespace v8 | 3486 } // namespace v8 |
| OLD | NEW |