OLD | NEW |
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 3163 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
3174 | 3174 |
3175 | 3175 |
3176 void CEntryStub::Generate(MacroAssembler* masm) { | 3176 void CEntryStub::Generate(MacroAssembler* masm) { |
3177 // Called from JavaScript; parameters are on stack as if calling JS function | 3177 // Called from JavaScript; parameters are on stack as if calling JS function |
3178 // r0: number of arguments including receiver | 3178 // r0: number of arguments including receiver |
3179 // r1: pointer to builtin function | 3179 // r1: pointer to builtin function |
3180 // fp: frame pointer (restored after C call) | 3180 // fp: frame pointer (restored after C call) |
3181 // sp: stack pointer (restored as callee's sp after C call) | 3181 // sp: stack pointer (restored as callee's sp after C call) |
3182 // cp: current context (C callee-saved) | 3182 // cp: current context (C callee-saved) |
3183 | 3183 |
| 3184 ProfileEntryHookStub::MaybeCallEntryHook(masm); |
| 3185 |
3184 // Result returned in r0 or r0+r1 by default. | 3186 // Result returned in r0 or r0+r1 by default. |
3185 | 3187 |
3186 // NOTE: Invocations of builtins may return failure objects | 3188 // NOTE: Invocations of builtins may return failure objects |
3187 // instead of a proper result. The builtin entry handles | 3189 // instead of a proper result. The builtin entry handles |
3188 // this by performing a garbage collection and retrying the | 3190 // this by performing a garbage collection and retrying the |
3189 // builtin once. | 3191 // builtin once. |
3190 | 3192 |
3191 // Compute the argv pointer in a callee-saved register. | 3193 // Compute the argv pointer in a callee-saved register. |
3192 __ add(r6, sp, Operand(r0, LSL, kPointerSizeLog2)); | 3194 __ add(r6, sp, Operand(r0, LSL, kPointerSizeLog2)); |
3193 __ sub(r6, r6, Operand(kPointerSize)); | 3195 __ sub(r6, r6, Operand(kPointerSize)); |
(...skipping 70 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
3264 | 3266 |
3265 void JSEntryStub::GenerateBody(MacroAssembler* masm, bool is_construct) { | 3267 void JSEntryStub::GenerateBody(MacroAssembler* masm, bool is_construct) { |
3266 // r0: code entry | 3268 // r0: code entry |
3267 // r1: function | 3269 // r1: function |
3268 // r2: receiver | 3270 // r2: receiver |
3269 // r3: argc | 3271 // r3: argc |
3270 // [sp+0]: argv | 3272 // [sp+0]: argv |
3271 | 3273 |
3272 Label invoke, handler_entry, exit; | 3274 Label invoke, handler_entry, exit; |
3273 | 3275 |
| 3276 ProfileEntryHookStub::MaybeCallEntryHook(masm); |
| 3277 |
3274 // Called from C, so do not pop argc and args on exit (preserve sp) | 3278 // Called from C, so do not pop argc and args on exit (preserve sp) |
3275 // No need to save register-passed args | 3279 // No need to save register-passed args |
3276 // Save callee-saved registers (incl. cp and fp), sp, and lr | 3280 // Save callee-saved registers (incl. cp and fp), sp, and lr |
3277 __ stm(db_w, sp, kCalleeSaved | lr.bit()); | 3281 __ stm(db_w, sp, kCalleeSaved | lr.bit()); |
3278 | 3282 |
3279 // Save callee-saved vfp registers. | 3283 // Save callee-saved vfp registers. |
3280 __ vstm(db_w, sp, kFirstCalleeSavedDoubleReg, kLastCalleeSavedDoubleReg); | 3284 __ vstm(db_w, sp, kFirstCalleeSavedDoubleReg, kLastCalleeSavedDoubleReg); |
3281 // Set up the reserved register for 0.0. | 3285 // Set up the reserved register for 0.0. |
3282 __ vmov(kDoubleRegZero, 0.0); | 3286 __ vmov(kDoubleRegZero, 0.0); |
3283 __ VFPEnsureFPSCRState(r4); | 3287 __ VFPEnsureFPSCRState(r4); |
(...skipping 3780 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
7064 __ add(r1, r1, Operand(1)); | 7068 __ add(r1, r1, Operand(1)); |
7065 } | 7069 } |
7066 masm->LeaveFrame(StackFrame::STUB_FAILURE_TRAMPOLINE); | 7070 masm->LeaveFrame(StackFrame::STUB_FAILURE_TRAMPOLINE); |
7067 __ mov(r1, Operand(r1, LSL, kPointerSizeLog2)); | 7071 __ mov(r1, Operand(r1, LSL, kPointerSizeLog2)); |
7068 __ add(sp, sp, r1); | 7072 __ add(sp, sp, r1); |
7069 __ Ret(); | 7073 __ Ret(); |
7070 } | 7074 } |
7071 | 7075 |
7072 | 7076 |
7073 void ProfileEntryHookStub::MaybeCallEntryHook(MacroAssembler* masm) { | 7077 void ProfileEntryHookStub::MaybeCallEntryHook(MacroAssembler* masm) { |
7074 if (entry_hook_ != NULL) { | 7078 if (masm->isolate()->function_entry_hook() != NULL) { |
7075 PredictableCodeSizeScope predictable(masm, 4 * Assembler::kInstrSize); | 7079 PredictableCodeSizeScope predictable(masm, 4 * Assembler::kInstrSize); |
| 7080 AllowStubCallsScope allow_stub_calls(masm, true); |
7076 ProfileEntryHookStub stub; | 7081 ProfileEntryHookStub stub; |
7077 __ push(lr); | 7082 __ push(lr); |
7078 __ CallStub(&stub); | 7083 __ CallStub(&stub); |
7079 __ pop(lr); | 7084 __ pop(lr); |
7080 } | 7085 } |
7081 } | 7086 } |
7082 | 7087 |
7083 | 7088 |
7084 void ProfileEntryHookStub::Generate(MacroAssembler* masm) { | 7089 void ProfileEntryHookStub::Generate(MacroAssembler* masm) { |
7085 // The entry hook is a "push lr" instruction, followed by a call. | 7090 // The entry hook is a "push lr" instruction, followed by a call. |
7086 const int32_t kReturnAddressDistanceFromFunctionStart = | 7091 const int32_t kReturnAddressDistanceFromFunctionStart = |
7087 3 * Assembler::kInstrSize; | 7092 3 * Assembler::kInstrSize; |
7088 | 7093 |
7089 // Save live volatile registers. | 7094 // This should contain all kCallerSaved registers. |
7090 __ Push(lr, r5, r1); | 7095 const RegList kSavedRegs = |
7091 const int32_t kNumSavedRegs = 3; | 7096 1 << 0 | // r0 |
| 7097 1 << 1 | // r1 |
| 7098 1 << 2 | // r2 |
| 7099 1 << 3 | // r3 |
| 7100 1 << 5 | // r5 |
| 7101 1 << 9; // r9 |
| 7102 // We also save lr, so the count here is one higher than the mask indicates. |
| 7103 const int32_t kNumSavedRegs = 7; |
| 7104 |
| 7105 ASSERT((kCallerSaved & kSavedRegs) == kCallerSaved); |
| 7106 |
| 7107 // Save all caller-save registers as this may be called from anywhere. |
| 7108 __ stm(db_w, sp, kSavedRegs | lr.bit()); |
7092 | 7109 |
7093 // Compute the function's address for the first argument. | 7110 // Compute the function's address for the first argument. |
7094 __ sub(r0, lr, Operand(kReturnAddressDistanceFromFunctionStart)); | 7111 __ sub(r0, lr, Operand(kReturnAddressDistanceFromFunctionStart)); |
7095 | 7112 |
7096 // The caller's return address is above the saved temporaries. | 7113 // The caller's return address is above the saved temporaries. |
7097 // Grab that for the second argument to the hook. | 7114 // Grab that for the second argument to the hook. |
7098 __ add(r1, sp, Operand(kNumSavedRegs * kPointerSize)); | 7115 __ add(r1, sp, Operand(kNumSavedRegs * kPointerSize)); |
7099 | 7116 |
7100 // Align the stack if necessary. | 7117 // Align the stack if necessary. |
7101 int frame_alignment = masm->ActivationFrameAlignment(); | 7118 int frame_alignment = masm->ActivationFrameAlignment(); |
7102 if (frame_alignment > kPointerSize) { | 7119 if (frame_alignment > kPointerSize) { |
7103 __ mov(r5, sp); | 7120 __ mov(r5, sp); |
7104 ASSERT(IsPowerOf2(frame_alignment)); | 7121 ASSERT(IsPowerOf2(frame_alignment)); |
7105 __ and_(sp, sp, Operand(-frame_alignment)); | 7122 __ and_(sp, sp, Operand(-frame_alignment)); |
7106 } | 7123 } |
7107 | 7124 |
7108 #if defined(V8_HOST_ARCH_ARM) | 7125 #if defined(V8_HOST_ARCH_ARM) |
7109 __ mov(ip, Operand(reinterpret_cast<int32_t>(&entry_hook_))); | 7126 int32_t entry_hook = |
7110 __ ldr(ip, MemOperand(ip)); | 7127 reinterpret_cast<int32_t>(masm->isolate()->function_entry_hook()); |
| 7128 __ mov(ip, Operand(entry_hook)); |
7111 #else | 7129 #else |
7112 // Under the simulator we need to indirect the entry hook through a | 7130 // Under the simulator we need to indirect the entry hook through a |
7113 // trampoline function at a known address. | 7131 // trampoline function at a known address. |
7114 Address trampoline_address = reinterpret_cast<Address>( | 7132 ApiFunction dispatcher(FUNCTION_ADDR(EntryHookTrampoline)); |
7115 reinterpret_cast<intptr_t>(EntryHookTrampoline)); | |
7116 ApiFunction dispatcher(trampoline_address); | |
7117 __ mov(ip, Operand(ExternalReference(&dispatcher, | 7133 __ mov(ip, Operand(ExternalReference(&dispatcher, |
7118 ExternalReference::BUILTIN_CALL, | 7134 ExternalReference::BUILTIN_CALL, |
7119 masm->isolate()))); | 7135 masm->isolate()))); |
7120 #endif | 7136 #endif |
7121 __ Call(ip); | 7137 __ Call(ip); |
7122 | 7138 |
7123 // Restore the stack pointer if needed. | 7139 // Restore the stack pointer if needed. |
7124 if (frame_alignment > kPointerSize) { | 7140 if (frame_alignment > kPointerSize) { |
7125 __ mov(sp, r5); | 7141 __ mov(sp, r5); |
7126 } | 7142 } |
7127 | 7143 |
7128 __ Pop(lr, r5, r1); | 7144 // Also pop pc to get Ret(0). |
7129 __ Ret(); | 7145 __ ldm(ia_w, sp, kSavedRegs | pc.bit()); |
7130 } | 7146 } |
7131 | 7147 |
7132 | 7148 |
7133 template<class T> | 7149 template<class T> |
7134 static void CreateArrayDispatch(MacroAssembler* masm) { | 7150 static void CreateArrayDispatch(MacroAssembler* masm) { |
7135 int last_index = GetSequenceIndexFromFastElementsKind( | 7151 int last_index = GetSequenceIndexFromFastElementsKind( |
7136 TERMINAL_FAST_ELEMENTS_KIND); | 7152 TERMINAL_FAST_ELEMENTS_KIND); |
7137 for (int i = 0; i <= last_index; ++i) { | 7153 for (int i = 0; i <= last_index; ++i) { |
7138 Label next; | 7154 Label next; |
7139 ElementsKind kind = GetFastElementsKindFromSequenceIndex(i); | 7155 ElementsKind kind = GetFastElementsKindFromSequenceIndex(i); |
(...skipping 261 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
7401 __ bind(&fast_elements_case); | 7417 __ bind(&fast_elements_case); |
7402 GenerateCase(masm, FAST_ELEMENTS); | 7418 GenerateCase(masm, FAST_ELEMENTS); |
7403 } | 7419 } |
7404 | 7420 |
7405 | 7421 |
7406 #undef __ | 7422 #undef __ |
7407 | 7423 |
7408 } } // namespace v8::internal | 7424 } } // namespace v8::internal |
7409 | 7425 |
7410 #endif // V8_TARGET_ARCH_ARM | 7426 #endif // V8_TARGET_ARCH_ARM |
OLD | NEW |