Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(1008)

Side by Side Diff: src/x64/deoptimizer-x64.cc

Issue 10701054: Enable stub generation using Hydrogen/Lithium (again) (Closed) Base URL: https://v8.googlecode.com/svn/branches/bleeding_edge
Patch Set: First pass at pre-VFP2 RA Created 8 years, 1 month ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
OLDNEW
1 // Copyright 2012 the V8 project authors. All rights reserved. 1 // Copyright 2012 the V8 project authors. All rights reserved.
2 // Redistribution and use in source and binary forms, with or without 2 // Redistribution and use in source and binary forms, with or without
3 // modification, are permitted provided that the following conditions are 3 // modification, are permitted provided that the following conditions are
4 // met: 4 // met:
5 // 5 //
6 // * Redistributions of source code must retain the above copyright 6 // * Redistributions of source code must retain the above copyright
7 // notice, this list of conditions and the following disclaimer. 7 // notice, this list of conditions and the following disclaimer.
8 // * Redistributions in binary form must reproduce the above 8 // * Redistributions in binary form must reproduce the above
9 // copyright notice, this list of conditions and the following 9 // copyright notice, this list of conditions and the following
10 // disclaimer in the documentation and/or other materials provided 10 // disclaimer in the documentation and/or other materials provided
(...skipping 441 matching lines...) Expand 10 before | Expand all | Expand 10 after
452 Builtins* builtins = isolate_->builtins(); 452 Builtins* builtins = isolate_->builtins();
453 Code* adaptor_trampoline = 453 Code* adaptor_trampoline =
454 builtins->builtin(Builtins::kArgumentsAdaptorTrampoline); 454 builtins->builtin(Builtins::kArgumentsAdaptorTrampoline);
455 intptr_t pc_value = reinterpret_cast<intptr_t>( 455 intptr_t pc_value = reinterpret_cast<intptr_t>(
456 adaptor_trampoline->instruction_start() + 456 adaptor_trampoline->instruction_start() +
457 isolate_->heap()->arguments_adaptor_deopt_pc_offset()->value()); 457 isolate_->heap()->arguments_adaptor_deopt_pc_offset()->value());
458 output_frame->SetPc(pc_value); 458 output_frame->SetPc(pc_value);
459 } 459 }
460 460
461 461
462 void Deoptimizer::DoCompiledStubPseudoFrame(TranslationIterator* iterator,
463 int frame_index) {
464 // Builtins* builtins = isolate_->builtins();
465 // Code::Kind stub_kind = static_cast<Code::Kind>(iterator->Next());
466 FrameDescription* output_frame = new(0) FrameDescription(0, 0);
467 Code* continuation =
468 isolate_->builtins()->builtin(Builtins::kNotifyICMiss);
469 output_frame->SetState(Smi::FromInt(FullCodeGenerator::NO_REGISTERS));
470 output_frame->SetContinuation(
471 reinterpret_cast<uintptr_t>(continuation->entry()));
472 Handle<Code> miss_ic = isolate_->builtins()->KeyedLoadIC_Miss();
473 output_frame->SetPc(
474 reinterpret_cast<intptr_t>(miss_ic->instruction_start()));
475
476 Code::Kind stub_kind = static_cast<Code::Kind>(iterator->Next());
477 ASSERT(stub_kind == Code::KEYED_LOAD_IC);
478 USE(stub_kind);
479
480 Translation::Opcode opcode =
481 static_cast<Translation::Opcode>(iterator->Next());
482 ASSERT(opcode == Translation::REGISTER);
483 USE(opcode);
484 int input_reg = iterator->Next();
485 intptr_t input_value = input_->GetRegister(input_reg);
486 output_frame->SetRegister(rdx.code(), input_value);
487
488 int32_t next = iterator->Next();
489 opcode = static_cast<Translation::Opcode>(next);
490 ASSERT(opcode == Translation::REGISTER);
491 input_reg = iterator->Next();
492 input_value = input_->GetRegister(input_reg);
493 output_frame->SetRegister(rax.code(), input_value);
494
495 Address frame_ptr =
496 reinterpret_cast<Address>(input_->GetRegister(rbp.code()));
497 Address context_ptr_raw = frame_ptr + StandardFrameConstants::kContextOffset;
498 Context** context_ptr =
499 reinterpret_cast<Context**>(context_ptr_raw);
500 output_frame->SetRegister(rsi.code(),
501 reinterpret_cast<intptr_t>(*context_ptr));
502 output_frame->SetRegister(rbp.code(), reinterpret_cast<intptr_t>(frame_ptr));
503
504 ASSERT(frame_index == 0);
505 output_[frame_index] = output_frame;
506 }
507
508
462 void Deoptimizer::DoComputeConstructStubFrame(TranslationIterator* iterator, 509 void Deoptimizer::DoComputeConstructStubFrame(TranslationIterator* iterator,
463 int frame_index) { 510 int frame_index) {
464 Builtins* builtins = isolate_->builtins(); 511 Builtins* builtins = isolate_->builtins();
465 Code* construct_stub = builtins->builtin(Builtins::kJSConstructStubGeneric); 512 Code* construct_stub = builtins->builtin(Builtins::kJSConstructStubGeneric);
466 JSFunction* function = JSFunction::cast(ComputeLiteral(iterator->Next())); 513 JSFunction* function = JSFunction::cast(ComputeLiteral(iterator->Next()));
467 unsigned height = iterator->Next(); 514 unsigned height = iterator->Next();
468 unsigned height_in_bytes = height * kPointerSize; 515 unsigned height_in_bytes = height * kPointerSize;
469 if (FLAG_trace_deopt) { 516 if (FLAG_trace_deopt) {
470 PrintF(" translating construct stub => height=%d\n", height_in_bytes); 517 PrintF(" translating construct stub => height=%d\n", height_in_bytes);
471 } 518 }
(...skipping 399 matching lines...) Expand 10 before | Expand all | Expand 10 after
871 918
872 void Deoptimizer::FillInputFrame(Address tos, JavaScriptFrame* frame) { 919 void Deoptimizer::FillInputFrame(Address tos, JavaScriptFrame* frame) {
873 // Set the register values. The values are not important as there are no 920 // Set the register values. The values are not important as there are no
874 // callee saved registers in JavaScript frames, so all registers are 921 // callee saved registers in JavaScript frames, so all registers are
875 // spilled. Registers rbp and rsp are set to the correct values though. 922 // spilled. Registers rbp and rsp are set to the correct values though.
876 for (int i = 0; i < Register::kNumRegisters; i++) { 923 for (int i = 0; i < Register::kNumRegisters; i++) {
877 input_->SetRegister(i, i * 4); 924 input_->SetRegister(i, i * 4);
878 } 925 }
879 input_->SetRegister(rsp.code(), reinterpret_cast<intptr_t>(frame->sp())); 926 input_->SetRegister(rsp.code(), reinterpret_cast<intptr_t>(frame->sp()));
880 input_->SetRegister(rbp.code(), reinterpret_cast<intptr_t>(frame->fp())); 927 input_->SetRegister(rbp.code(), reinterpret_cast<intptr_t>(frame->fp()));
881 for (int i = 0; i < DoubleRegister::kNumAllocatableRegisters; i++) { 928 for (int i = 0; i < DoubleRegister::NumAllocatableRegisters(); i++) {
882 input_->SetDoubleRegister(i, 0.0); 929 input_->SetDoubleRegister(i, 0.0);
883 } 930 }
884 931
885 // Fill the frame content from the actual data on the frame. 932 // Fill the frame content from the actual data on the frame.
886 for (unsigned i = 0; i < input_->GetFrameSize(); i += kPointerSize) { 933 for (unsigned i = 0; i < input_->GetFrameSize(); i += kPointerSize) {
887 input_->SetFrameSlot(i, Memory::uint64_at(tos + i)); 934 input_->SetFrameSlot(i, Memory::uint64_at(tos + i));
888 } 935 }
889 } 936 }
890 937
891 938
892 #define __ masm()-> 939 #define __ masm()->
893 940
894 void Deoptimizer::EntryGenerator::Generate() { 941 void Deoptimizer::EntryGenerator::Generate() {
895 GeneratePrologue(); 942 GeneratePrologue();
896 943
897 // Save all general purpose registers before messing with them. 944 // Save all general purpose registers before messing with them.
898 const int kNumberOfRegisters = Register::kNumRegisters; 945 const int kNumberOfRegisters = Register::kNumRegisters;
899 946
900 const int kDoubleRegsSize = kDoubleSize * 947 const int kDoubleRegsSize = kDoubleSize *
901 XMMRegister::kNumAllocatableRegisters; 948 XMMRegister::NumAllocatableRegisters();
902 __ subq(rsp, Immediate(kDoubleRegsSize)); 949 __ subq(rsp, Immediate(kDoubleRegsSize));
903 950
904 for (int i = 0; i < XMMRegister::kNumAllocatableRegisters; ++i) { 951 for (int i = 0; i < XMMRegister::NumAllocatableRegisters(); ++i) {
905 XMMRegister xmm_reg = XMMRegister::FromAllocationIndex(i); 952 XMMRegister xmm_reg = XMMRegister::FromAllocationIndex(i);
906 int offset = i * kDoubleSize; 953 int offset = i * kDoubleSize;
907 __ movsd(Operand(rsp, offset), xmm_reg); 954 __ movsd(Operand(rsp, offset), xmm_reg);
908 } 955 }
909 956
910 // We push all registers onto the stack, even though we do not need 957 // We push all registers onto the stack, even though we do not need
911 // to restore all later. 958 // to restore all later.
912 for (int i = 0; i < kNumberOfRegisters; i++) { 959 for (int i = 0; i < kNumberOfRegisters; i++) {
913 Register r = Register::from_code(i); 960 Register r = Register::from_code(i);
914 __ push(r); 961 __ push(r);
(...skipping 68 matching lines...) Expand 10 before | Expand all | Expand 10 after
983 __ movq(rbx, Operand(rax, Deoptimizer::input_offset())); 1030 __ movq(rbx, Operand(rax, Deoptimizer::input_offset()));
984 1031
985 // Fill in the input registers. 1032 // Fill in the input registers.
986 for (int i = kNumberOfRegisters -1; i >= 0; i--) { 1033 for (int i = kNumberOfRegisters -1; i >= 0; i--) {
987 int offset = (i * kPointerSize) + FrameDescription::registers_offset(); 1034 int offset = (i * kPointerSize) + FrameDescription::registers_offset();
988 __ pop(Operand(rbx, offset)); 1035 __ pop(Operand(rbx, offset));
989 } 1036 }
990 1037
991 // Fill in the double input registers. 1038 // Fill in the double input registers.
992 int double_regs_offset = FrameDescription::double_registers_offset(); 1039 int double_regs_offset = FrameDescription::double_registers_offset();
993 for (int i = 0; i < XMMRegister::kNumAllocatableRegisters; i++) { 1040 for (int i = 0; i < XMMRegister::NumAllocatableRegisters(); i++) {
994 int dst_offset = i * kDoubleSize + double_regs_offset; 1041 int dst_offset = i * kDoubleSize + double_regs_offset;
995 __ pop(Operand(rbx, dst_offset)); 1042 __ pop(Operand(rbx, dst_offset));
996 } 1043 }
997 1044
998 // Remove the bailout id from the stack. 1045 // Remove the bailout id from the stack.
999 if (type() == EAGER) { 1046 if (type() == EAGER) {
1000 __ addq(rsp, Immediate(kPointerSize)); 1047 __ addq(rsp, Immediate(kPointerSize));
1001 } else { 1048 } else {
1002 __ addq(rsp, Immediate(2 * kPointerSize)); 1049 __ addq(rsp, Immediate(2 * kPointerSize));
1003 } 1050 }
1004 1051
1005 // Compute a pointer to the unwinding limit in register rcx; that is 1052 // Compute a pointer to the unwinding limit in register rcx; that is
1006 // the first stack slot not part of the input frame. 1053 // the first stack slot not part of the input frame.
1007 __ movq(rcx, Operand(rbx, FrameDescription::frame_size_offset())); 1054 __ movq(rcx, Operand(rbx, FrameDescription::frame_size_offset()));
1008 __ addq(rcx, rsp); 1055 __ addq(rcx, rsp);
1009 1056
1010 // Unwind the stack down to - but not including - the unwinding 1057 // Unwind the stack down to - but not including - the unwinding
1011 // limit and copy the contents of the activation frame to the input 1058 // limit and copy the contents of the activation frame to the input
1012 // frame description. 1059 // frame description.
1013 __ lea(rdx, Operand(rbx, FrameDescription::frame_content_offset())); 1060 __ lea(rdx, Operand(rbx, FrameDescription::frame_content_offset()));
1061 Label pop_loop_header;
1062 __ jmp(&pop_loop_header);
1014 Label pop_loop; 1063 Label pop_loop;
1015 __ bind(&pop_loop); 1064 __ bind(&pop_loop);
1016 __ pop(Operand(rdx, 0)); 1065 __ pop(Operand(rdx, 0));
1017 __ addq(rdx, Immediate(sizeof(intptr_t))); 1066 __ addq(rdx, Immediate(sizeof(intptr_t)));
1067 __ bind(&pop_loop_header);
1018 __ cmpq(rcx, rsp); 1068 __ cmpq(rcx, rsp);
1019 __ j(not_equal, &pop_loop); 1069 __ j(not_equal, &pop_loop);
1020 1070
1021 // Compute the output frame in the deoptimizer. 1071 // Compute the output frame in the deoptimizer.
1022 __ push(rax); 1072 __ push(rax);
1023 __ PrepareCallCFunction(2); 1073 __ PrepareCallCFunction(2);
1024 __ movq(arg1, rax); 1074 __ movq(arg1, rax);
1025 __ LoadAddress(arg2, ExternalReference::isolate_address()); 1075 __ LoadAddress(arg2, ExternalReference::isolate_address());
1026 { 1076 {
1027 AllowExternalCallThatCantCauseGC scope(masm()); 1077 AllowExternalCallThatCantCauseGC scope(masm());
1028 __ CallCFunction( 1078 __ CallCFunction(
1029 ExternalReference::compute_output_frames_function(isolate), 2); 1079 ExternalReference::compute_output_frames_function(isolate), 2);
1030 } 1080 }
1031 __ pop(rax); 1081 __ pop(rax);
1032 1082
1033 // Replace the current frame with the output frames. 1083 // Replace the current frame with the output frames.
1034 Label outer_push_loop, inner_push_loop; 1084 Label outer_push_loop, inner_push_loop,
1085 outer_loop_header, inner_loop_header;
1035 // Outer loop state: rax = current FrameDescription**, rdx = one past the 1086 // Outer loop state: rax = current FrameDescription**, rdx = one past the
1036 // last FrameDescription**. 1087 // last FrameDescription**.
1037 __ movl(rdx, Operand(rax, Deoptimizer::output_count_offset())); 1088 __ movl(rdx, Operand(rax, Deoptimizer::output_count_offset()));
1038 __ movq(rax, Operand(rax, Deoptimizer::output_offset())); 1089 __ movq(rax, Operand(rax, Deoptimizer::output_offset()));
1039 __ lea(rdx, Operand(rax, rdx, times_8, 0)); 1090 __ lea(rdx, Operand(rax, rdx, times_8, 0));
1091 __ jmp(&outer_loop_header);
1040 __ bind(&outer_push_loop); 1092 __ bind(&outer_push_loop);
1041 // Inner loop state: rbx = current FrameDescription*, rcx = loop index. 1093 // Inner loop state: rbx = current FrameDescription*, rcx = loop index.
1042 __ movq(rbx, Operand(rax, 0)); 1094 __ movq(rbx, Operand(rax, 0));
1043 __ movq(rcx, Operand(rbx, FrameDescription::frame_size_offset())); 1095 __ movq(rcx, Operand(rbx, FrameDescription::frame_size_offset()));
1096 __ jmp(&inner_loop_header);
1044 __ bind(&inner_push_loop); 1097 __ bind(&inner_push_loop);
1045 __ subq(rcx, Immediate(sizeof(intptr_t))); 1098 __ subq(rcx, Immediate(sizeof(intptr_t)));
1046 __ push(Operand(rbx, rcx, times_1, FrameDescription::frame_content_offset())); 1099 __ push(Operand(rbx, rcx, times_1, FrameDescription::frame_content_offset()));
1100 __ bind(&inner_loop_header);
1047 __ testq(rcx, rcx); 1101 __ testq(rcx, rcx);
1048 __ j(not_zero, &inner_push_loop); 1102 __ j(not_zero, &inner_push_loop);
1049 __ addq(rax, Immediate(kPointerSize)); 1103 __ addq(rax, Immediate(kPointerSize));
1104 __ bind(&outer_loop_header);
1050 __ cmpq(rax, rdx); 1105 __ cmpq(rax, rdx);
1051 __ j(below, &outer_push_loop); 1106 __ j(below, &outer_push_loop);
1052 1107
1053 // In case of OSR, we have to restore the XMM registers. 1108 // In case of OSR, we have to restore the XMM registers.
1054 if (type() == OSR) { 1109 if (type() == OSR) {
1055 for (int i = 0; i < XMMRegister::kNumAllocatableRegisters; ++i) { 1110 for (int i = 0; i < XMMRegister::NumAllocatableRegisters(); ++i) {
1056 XMMRegister xmm_reg = XMMRegister::FromAllocationIndex(i); 1111 XMMRegister xmm_reg = XMMRegister::FromAllocationIndex(i);
1057 int src_offset = i * kDoubleSize + double_regs_offset; 1112 int src_offset = i * kDoubleSize + double_regs_offset;
1058 __ movsd(xmm_reg, Operand(rbx, src_offset)); 1113 __ movsd(xmm_reg, Operand(rbx, src_offset));
1059 } 1114 }
1060 } 1115 }
1061 1116
1062 // Push state, pc, and continuation from the last output frame. 1117 // Push state, pc, and continuation from the last output frame.
1063 if (type() != OSR) { 1118 if (type() != OSR) {
1064 __ push(Operand(rbx, FrameDescription::state_offset())); 1119 __ push(Operand(rbx, FrameDescription::state_offset()));
1065 } 1120 }
(...skipping 39 matching lines...) Expand 10 before | Expand all | Expand 10 after
1105 } 1160 }
1106 __ bind(&done); 1161 __ bind(&done);
1107 } 1162 }
1108 1163
1109 #undef __ 1164 #undef __
1110 1165
1111 1166
1112 } } // namespace v8::internal 1167 } } // namespace v8::internal
1113 1168
1114 #endif // V8_TARGET_ARCH_X64 1169 #endif // V8_TARGET_ARCH_X64
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698