OLD | NEW |
---|---|
1 // Copyright 2012 the V8 project authors. All rights reserved. | 1 // Copyright 2012 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/frames.h" | 5 #include "src/frames.h" |
6 | 6 |
7 #include <sstream> | 7 #include <sstream> |
8 | 8 |
9 #include "src/v8.h" | 9 #include "src/v8.h" |
10 | 10 |
(...skipping 866 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
877 DCHECK(frames->length() == 0); | 877 DCHECK(frames->length() == 0); |
878 DCHECK(is_optimized()); | 878 DCHECK(is_optimized()); |
879 | 879 |
880 // Delegate to JS frame in absence of turbofan deoptimization. | 880 // Delegate to JS frame in absence of turbofan deoptimization. |
881 // TODO(turbofan): Revisit once we support deoptimization across the board. | 881 // TODO(turbofan): Revisit once we support deoptimization across the board. |
882 if (LookupCode()->is_turbofanned() && function()->shared()->asm_function() && | 882 if (LookupCode()->is_turbofanned() && function()->shared()->asm_function() && |
883 !FLAG_turbo_asm_deoptimization) { | 883 !FLAG_turbo_asm_deoptimization) { |
884 return JavaScriptFrame::Summarize(frames); | 884 return JavaScriptFrame::Summarize(frames); |
885 } | 885 } |
886 | 886 |
887 int deopt_index = Safepoint::kNoDeoptimizationIndex; | |
888 DeoptimizationInputData* data = GetDeoptimizationData(&deopt_index); | |
889 FixedArray* literal_array = data->LiteralArray(); | |
890 | |
891 // BUG(3243555): Since we don't have a lazy-deopt registered at | |
892 // throw-statements, we can't use the translation at the call-site of | |
893 // throw. An entry with no deoptimization index indicates a call-site | |
894 // without a lazy-deopt. As a consequence we are not allowed to inline | |
895 // functions containing throw. | |
896 DCHECK(deopt_index != Safepoint::kNoDeoptimizationIndex); | |
897 | |
898 TranslationIterator it(data->TranslationByteArray(), | |
899 data->TranslationIndex(deopt_index)->value()); | |
900 Translation::Opcode opcode = static_cast<Translation::Opcode>(it.Next()); | |
901 DCHECK(opcode == Translation::BEGIN); | |
902 it.Next(); // Drop frame count. | |
903 int jsframe_count = it.Next(); | |
904 | |
905 // We create the summary in reverse order because the frames | 887 // We create the summary in reverse order because the frames |
906 // in the deoptimization translation are ordered bottom-to-top. | 888 // in the deoptimization translation are ordered bottom-to-top. |
889 TranslatedState state(this); | |
Jarin
2015/06/09 12:32:53
I am wondering whether we should put this in no-al
| |
907 bool is_constructor = IsConstructor(); | 890 bool is_constructor = IsConstructor(); |
908 int i = jsframe_count; | 891 for (TranslatedFrame const& frame : state) { |
909 while (i > 0) { | 892 switch (frame.kind()) { |
910 opcode = static_cast<Translation::Opcode>(it.Next()); | 893 case TranslatedFrame::kFunction: { |
911 if (opcode == Translation::JS_FRAME) { | 894 BailoutId const ast_id = frame.node_id(); |
912 i--; | 895 JSFunction* const function = frame.raw_function(); |
913 BailoutId ast_id = BailoutId(it.Next()); | |
914 JSFunction* function = LiteralAt(literal_array, it.Next()); | |
915 it.Next(); // Skip height. | |
916 | 896 |
917 // The translation commands are ordered and the receiver is always | 897 // Get the correct receiver in the optimized frame. |
918 // at the first position. | 898 Object* receiver = frame.front().GetRawValue(); |
919 // If we are at a call, the receiver is always in a stack slot. | 899 if (receiver == isolate()->heap()->arguments_marker()) { |
920 // Otherwise we are not guaranteed to get the receiver value. | 900 // TODO(jarin): Materializing a captured object (or duplicated |
921 opcode = static_cast<Translation::Opcode>(it.Next()); | 901 // object) is hard, we return undefined for now. This breaks the |
922 int index = it.Next(); | 902 // produced stack trace, as constructor frames aren't marked as |
903 // such anymore. | |
904 receiver = isolate()->heap()->undefined_value(); | |
905 } | |
923 | 906 |
924 // Get the correct receiver in the optimized frame. | 907 Code* code = function->shared()->code(); |
925 Object* receiver = NULL; | 908 DeoptimizationOutputData* output_data = |
926 if (opcode == Translation::LITERAL) { | 909 DeoptimizationOutputData::cast(code->deoptimization_data()); |
927 receiver = data->LiteralArray()->get(index); | 910 unsigned entry = |
928 } else if (opcode == Translation::STACK_SLOT) { | 911 Deoptimizer::GetOutputInfo(output_data, ast_id, function->shared()); |
929 // Positive index means the value is spilled to the locals | 912 unsigned pc_offset = |
930 // area. Negative means it is stored in the incoming parameter | 913 FullCodeGenerator::PcField::decode(entry) + Code::kHeaderSize; |
931 // area. | 914 DCHECK(pc_offset > 0); |
932 if (index >= 0) { | 915 |
933 receiver = GetExpression(index); | 916 FrameSummary summary(receiver, function, code, pc_offset, |
934 } else { | 917 is_constructor); |
935 // Index -1 overlaps with last parameter, -n with the first parameter, | 918 frames->Add(summary); |
936 // (-n - 1) with the receiver with n being the number of parameters | 919 is_constructor = false; |
937 // of the outermost, optimized frame. | 920 break; |
938 int parameter_count = ComputeParametersCount(); | |
939 int parameter_index = index + parameter_count; | |
940 receiver = (parameter_index == -1) | |
941 ? this->receiver() | |
942 : this->GetParameter(parameter_index); | |
943 } | |
944 } else { | |
945 // The receiver is not in a stack slot nor in a literal. We give up. | |
946 // TODO(3029): Materializing a captured object (or duplicated | |
947 // object) is hard, we return undefined for now. This breaks the | |
948 // produced stack trace, as constructor frames aren't marked as | |
949 // such anymore. | |
950 receiver = isolate()->heap()->undefined_value(); | |
951 } | 921 } |
952 | 922 |
953 Code* code = function->shared()->code(); | 923 case TranslatedFrame::kConstructStub: { |
954 DeoptimizationOutputData* output_data = | 924 // The next encountered JS_FRAME will be marked as a constructor call. |
955 DeoptimizationOutputData::cast(code->deoptimization_data()); | 925 DCHECK(!is_constructor); |
956 unsigned entry = Deoptimizer::GetOutputInfo(output_data, | 926 is_constructor = true; |
957 ast_id, | 927 break; |
958 function->shared()); | 928 } |
959 unsigned pc_offset = | |
960 FullCodeGenerator::PcField::decode(entry) + Code::kHeaderSize; | |
961 DCHECK(pc_offset > 0); | |
962 | 929 |
963 FrameSummary summary(receiver, function, code, pc_offset, is_constructor); | 930 case TranslatedFrame::kInvalid: |
964 frames->Add(summary); | 931 UNREACHABLE(); |
965 is_constructor = false; | 932 case TranslatedFrame::kArgumentsAdaptor: |
966 } else if (opcode == Translation::CONSTRUCT_STUB_FRAME) { | 933 case TranslatedFrame::kCompiledStub: |
967 // The next encountered JS_FRAME will be marked as a constructor call. | 934 case TranslatedFrame::kGetter: |
968 it.Skip(Translation::NumberOfOperandsFor(opcode)); | 935 case TranslatedFrame::kSetter: |
969 DCHECK(!is_constructor); | 936 break; |
970 is_constructor = true; | |
971 } else { | |
972 // Skip over operands to advance to the next opcode. | |
973 it.Skip(Translation::NumberOfOperandsFor(opcode)); | |
974 } | 937 } |
975 } | 938 } |
976 DCHECK(!is_constructor); | 939 DCHECK(!is_constructor); |
977 } | 940 } |
978 | 941 |
979 | 942 |
980 int OptimizedFrame::LookupExceptionHandlerInTable( | 943 int OptimizedFrame::LookupExceptionHandlerInTable( |
981 int* stack_slots, HandlerTable::CatchPrediction* prediction) { | 944 int* stack_slots, HandlerTable::CatchPrediction* prediction) { |
982 Code* code = LookupCode(); | 945 Code* code = LookupCode(); |
983 DCHECK(code->is_optimized_code()); | 946 DCHECK(code->is_optimized_code()); |
(...skipping 33 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
1017 DCHECK(functions->length() == 0); | 980 DCHECK(functions->length() == 0); |
1018 DCHECK(is_optimized()); | 981 DCHECK(is_optimized()); |
1019 | 982 |
1020 // Delegate to JS frame in absence of turbofan deoptimization. | 983 // Delegate to JS frame in absence of turbofan deoptimization. |
1021 // TODO(turbofan): Revisit once we support deoptimization across the board. | 984 // TODO(turbofan): Revisit once we support deoptimization across the board. |
1022 if (LookupCode()->is_turbofanned() && function()->shared()->asm_function() && | 985 if (LookupCode()->is_turbofanned() && function()->shared()->asm_function() && |
1023 !FLAG_turbo_asm_deoptimization) { | 986 !FLAG_turbo_asm_deoptimization) { |
1024 return JavaScriptFrame::GetFunctions(functions); | 987 return JavaScriptFrame::GetFunctions(functions); |
1025 } | 988 } |
1026 | 989 |
1027 int deopt_index = Safepoint::kNoDeoptimizationIndex; | 990 TranslatedState state(this); |
1028 DeoptimizationInputData* data = GetDeoptimizationData(&deopt_index); | 991 for (TranslatedFrame const& frame : state) { |
1029 FixedArray* literal_array = data->LiteralArray(); | 992 if (frame.kind() == TranslatedFrame::kFunction) { |
1030 | 993 functions->Add(frame.raw_function()); |
1031 TranslationIterator it(data->TranslationByteArray(), | |
1032 data->TranslationIndex(deopt_index)->value()); | |
1033 Translation::Opcode opcode = static_cast<Translation::Opcode>(it.Next()); | |
1034 DCHECK(opcode == Translation::BEGIN); | |
1035 it.Next(); // Drop frame count. | |
1036 int jsframe_count = it.Next(); | |
1037 | |
1038 // We insert the frames in reverse order because the frames | |
1039 // in the deoptimization translation are ordered bottom-to-top. | |
1040 while (jsframe_count > 0) { | |
1041 opcode = static_cast<Translation::Opcode>(it.Next()); | |
1042 if (opcode == Translation::JS_FRAME) { | |
1043 jsframe_count--; | |
1044 it.Next(); // Skip ast id. | |
1045 JSFunction* function = LiteralAt(literal_array, it.Next()); | |
1046 it.Next(); // Skip height. | |
1047 functions->Add(function); | |
1048 } else { | |
1049 // Skip over operands to advance to the next opcode. | |
1050 it.Skip(Translation::NumberOfOperandsFor(opcode)); | |
1051 } | 994 } |
1052 } | 995 } |
1053 } | 996 } |
1054 | 997 |
1055 | 998 |
1056 int ArgumentsAdaptorFrame::GetNumberOfIncomingArguments() const { | 999 int ArgumentsAdaptorFrame::GetNumberOfIncomingArguments() const { |
1057 return Smi::cast(GetExpression(0))->value(); | 1000 return Smi::cast(GetExpression(0))->value(); |
1058 } | 1001 } |
1059 | 1002 |
1060 | 1003 |
(...skipping 434 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
1495 for (StackFrameIterator it(isolate); !it.done(); it.Advance()) { | 1438 for (StackFrameIterator it(isolate); !it.done(); it.Advance()) { |
1496 StackFrame* frame = AllocateFrameCopy(it.frame(), zone); | 1439 StackFrame* frame = AllocateFrameCopy(it.frame(), zone); |
1497 list.Add(frame, zone); | 1440 list.Add(frame, zone); |
1498 } | 1441 } |
1499 return list.ToVector(); | 1442 return list.ToVector(); |
1500 } | 1443 } |
1501 | 1444 |
1502 | 1445 |
1503 } // namespace internal | 1446 } // namespace internal |
1504 } // namespace v8 | 1447 } // namespace v8 |
OLD | NEW |