| 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 "v8.h" | 5 #include "v8.h" |
| 6 | 6 |
| 7 #if V8_TARGET_ARCH_IA32 | 7 #if V8_TARGET_ARCH_IA32 |
| 8 | 8 |
| 9 #include "codegen.h" | 9 #include "codegen.h" |
| 10 #include "heap.h" | 10 #include "heap.h" |
| (...skipping 1032 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1043 __ movsd(double_scratch, Operand::StaticArray( | 1043 __ movsd(double_scratch, Operand::StaticArray( |
| 1044 temp2, times_8, ExternalReference::math_exp_log_table())); | 1044 temp2, times_8, ExternalReference::math_exp_log_table())); |
| 1045 __ orps(input, double_scratch); | 1045 __ orps(input, double_scratch); |
| 1046 __ mulsd(result, input); | 1046 __ mulsd(result, input); |
| 1047 __ bind(&done); | 1047 __ bind(&done); |
| 1048 } | 1048 } |
| 1049 | 1049 |
| 1050 #undef __ | 1050 #undef __ |
| 1051 | 1051 |
| 1052 | 1052 |
| 1053 static byte* GetNoCodeAgeSequence(uint32_t* length) { | 1053 CodeAgingHelper::CodeAgingHelper() { |
| 1054 static bool initialized = false; | 1054 ASSERT(young_sequence_.length() == kNoCodeAgeSequenceLength); |
| 1055 static byte sequence[kNoCodeAgeSequenceLength]; | 1055 CodePatcher patcher(young_sequence_.start(), young_sequence_.length()); |
| 1056 *length = kNoCodeAgeSequenceLength; | 1056 patcher.masm()->push(ebp); |
| 1057 if (!initialized) { | 1057 patcher.masm()->mov(ebp, esp); |
| 1058 // The sequence of instructions that is patched out for aging code is the | 1058 patcher.masm()->push(esi); |
| 1059 // following boilerplate stack-building prologue that is found both in | 1059 patcher.masm()->push(edi); |
| 1060 // FUNCTION and OPTIMIZED_FUNCTION code: | |
| 1061 CodePatcher patcher(sequence, kNoCodeAgeSequenceLength); | |
| 1062 patcher.masm()->push(ebp); | |
| 1063 patcher.masm()->mov(ebp, esp); | |
| 1064 patcher.masm()->push(esi); | |
| 1065 patcher.masm()->push(edi); | |
| 1066 initialized = true; | |
| 1067 } | |
| 1068 return sequence; | |
| 1069 } | 1060 } |
| 1070 | 1061 |
| 1071 | 1062 |
| 1072 bool Code::IsYoungSequence(byte* sequence) { | 1063 #ifdef DEBUG |
| 1073 uint32_t young_length; | 1064 bool CodeAgingHelper::IsOld(byte* candidate) const { |
| 1074 byte* young_sequence = GetNoCodeAgeSequence(&young_length); | 1065 return *candidate == kCallOpcode; |
| 1075 bool result = (!memcmp(sequence, young_sequence, young_length)); | 1066 } |
| 1076 ASSERT(result || *sequence == kCallOpcode); | 1067 #endif |
| 1068 |
| 1069 |
| 1070 bool Code::IsYoungSequence(Isolate* isolate, byte* sequence) { |
| 1071 bool result = isolate->code_aging_helper()->IsYoung(sequence); |
| 1072 ASSERT(result || isolate->code_aging_helper()->IsOld(sequence)); |
| 1077 return result; | 1073 return result; |
| 1078 } | 1074 } |
| 1079 | 1075 |
| 1080 | 1076 |
| 1081 void Code::GetCodeAgeAndParity(byte* sequence, Age* age, | 1077 void Code::GetCodeAgeAndParity(Isolate* isolate, byte* sequence, Age* age, |
| 1082 MarkingParity* parity) { | 1078 MarkingParity* parity) { |
| 1083 if (IsYoungSequence(sequence)) { | 1079 if (IsYoungSequence(isolate, sequence)) { |
| 1084 *age = kNoAgeCodeAge; | 1080 *age = kNoAgeCodeAge; |
| 1085 *parity = NO_MARKING_PARITY; | 1081 *parity = NO_MARKING_PARITY; |
| 1086 } else { | 1082 } else { |
| 1087 sequence++; // Skip the kCallOpcode byte | 1083 sequence++; // Skip the kCallOpcode byte |
| 1088 Address target_address = sequence + *reinterpret_cast<int*>(sequence) + | 1084 Address target_address = sequence + *reinterpret_cast<int*>(sequence) + |
| 1089 Assembler::kCallTargetAddressOffset; | 1085 Assembler::kCallTargetAddressOffset; |
| 1090 Code* stub = GetCodeFromTargetAddress(target_address); | 1086 Code* stub = GetCodeFromTargetAddress(target_address); |
| 1091 GetCodeAgeAndParity(stub, age, parity); | 1087 GetCodeAgeAndParity(stub, age, parity); |
| 1092 } | 1088 } |
| 1093 } | 1089 } |
| 1094 | 1090 |
| 1095 | 1091 |
| 1096 void Code::PatchPlatformCodeAge(Isolate* isolate, | 1092 void Code::PatchPlatformCodeAge(Isolate* isolate, |
| 1097 byte* sequence, | 1093 byte* sequence, |
| 1098 Code::Age age, | 1094 Code::Age age, |
| 1099 MarkingParity parity) { | 1095 MarkingParity parity) { |
| 1100 uint32_t young_length; | 1096 uint32_t young_length = isolate->code_aging_helper()->young_sequence_length(); |
| 1101 byte* young_sequence = GetNoCodeAgeSequence(&young_length); | |
| 1102 if (age == kNoAgeCodeAge) { | 1097 if (age == kNoAgeCodeAge) { |
| 1103 CopyBytes(sequence, young_sequence, young_length); | 1098 isolate->code_aging_helper()->CopyYoungSequenceTo(sequence); |
| 1104 CPU::FlushICache(sequence, young_length); | 1099 CPU::FlushICache(sequence, young_length); |
| 1105 } else { | 1100 } else { |
| 1106 Code* stub = GetCodeAgeStub(isolate, age, parity); | 1101 Code* stub = GetCodeAgeStub(isolate, age, parity); |
| 1107 CodePatcher patcher(sequence, young_length); | 1102 CodePatcher patcher(sequence, young_length); |
| 1108 patcher.masm()->call(stub->instruction_start(), RelocInfo::NONE32); | 1103 patcher.masm()->call(stub->instruction_start(), RelocInfo::NONE32); |
| 1109 } | 1104 } |
| 1110 } | 1105 } |
| 1111 | 1106 |
| 1112 | 1107 |
| 1113 } } // namespace v8::internal | 1108 } } // namespace v8::internal |
| 1114 | 1109 |
| 1115 #endif // V8_TARGET_ARCH_IA32 | 1110 #endif // V8_TARGET_ARCH_IA32 |
| OLD | NEW |