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 #if V8_TARGET_ARCH_ARM | 5 #if V8_TARGET_ARCH_ARM |
6 | 6 |
7 #include "src/assembler-inl.h" | 7 #include "src/assembler-inl.h" |
8 #include "src/codegen.h" | 8 #include "src/codegen.h" |
9 #include "src/counters.h" | 9 #include "src/counters.h" |
10 #include "src/debug/debug.h" | 10 #include "src/debug/debug.h" |
(...skipping 2135 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
2146 | 2146 |
2147 // Dispatch to Call or Construct depending on whether new.target is undefined. | 2147 // Dispatch to Call or Construct depending on whether new.target is undefined. |
2148 { | 2148 { |
2149 __ CompareRoot(r3, Heap::kUndefinedValueRootIndex); | 2149 __ CompareRoot(r3, Heap::kUndefinedValueRootIndex); |
2150 __ Jump(masm->isolate()->builtins()->Call(), RelocInfo::CODE_TARGET, eq); | 2150 __ Jump(masm->isolate()->builtins()->Call(), RelocInfo::CODE_TARGET, eq); |
2151 __ Jump(masm->isolate()->builtins()->Construct(), RelocInfo::CODE_TARGET); | 2151 __ Jump(masm->isolate()->builtins()->Construct(), RelocInfo::CODE_TARGET); |
2152 } | 2152 } |
2153 } | 2153 } |
2154 | 2154 |
2155 // static | 2155 // static |
2156 void Builtins::Generate_CallForwardVarargs(MacroAssembler* masm, | 2156 void Builtins::Generate_ForwardVarargs(MacroAssembler* masm, |
2157 Handle<Code> code) { | 2157 Handle<Code> code) { |
2158 // ----------- S t a t e ------------- | 2158 // ----------- S t a t e ------------- |
2159 // -- r1 : the target to call (can be any Object) | 2159 // -- r0 : the number of arguments (not including the receiver) |
2160 // -- r2 : start index (to support rest parameters) | 2160 // -- r3 : the new.target (for [[Construct]] calls) |
2161 // -- lr : return address. | 2161 // -- r1 : the target to call (can be any Object) |
2162 // -- sp[0] : thisArgument | 2162 // -- r2 : start index (to support rest parameters) |
2163 // ----------------------------------- | 2163 // ----------------------------------- |
2164 | 2164 |
2165 // Check if we have an arguments adaptor frame below the function frame. | 2165 // Check if we have an arguments adaptor frame below the function frame. |
2166 Label arguments_adaptor, arguments_done; | 2166 Label arguments_adaptor, arguments_done; |
2167 __ ldr(r3, MemOperand(fp, StandardFrameConstants::kCallerFPOffset)); | 2167 __ ldr(r4, MemOperand(fp, StandardFrameConstants::kCallerFPOffset)); |
2168 __ ldr(ip, MemOperand(r3, CommonFrameConstants::kContextOrFrameTypeOffset)); | 2168 __ ldr(ip, MemOperand(r4, CommonFrameConstants::kContextOrFrameTypeOffset)); |
2169 __ cmp(ip, Operand(StackFrame::TypeToMarker(StackFrame::ARGUMENTS_ADAPTOR))); | 2169 __ cmp(ip, Operand(StackFrame::TypeToMarker(StackFrame::ARGUMENTS_ADAPTOR))); |
2170 __ b(eq, &arguments_adaptor); | 2170 __ b(eq, &arguments_adaptor); |
2171 { | 2171 { |
2172 __ ldr(r0, MemOperand(fp, JavaScriptFrameConstants::kFunctionOffset)); | 2172 __ ldr(r5, MemOperand(fp, JavaScriptFrameConstants::kFunctionOffset)); |
2173 __ ldr(r0, FieldMemOperand(r0, JSFunction::kSharedFunctionInfoOffset)); | 2173 __ ldr(r5, FieldMemOperand(r5, JSFunction::kSharedFunctionInfoOffset)); |
2174 __ ldr(r0, FieldMemOperand( | 2174 __ ldr(r5, FieldMemOperand( |
2175 r0, SharedFunctionInfo::kFormalParameterCountOffset)); | 2175 r5, SharedFunctionInfo::kFormalParameterCountOffset)); |
2176 __ mov(r3, fp); | 2176 __ mov(r4, fp); |
2177 } | 2177 } |
2178 __ b(&arguments_done); | 2178 __ b(&arguments_done); |
2179 __ bind(&arguments_adaptor); | 2179 __ bind(&arguments_adaptor); |
2180 { | 2180 { |
2181 // Load the length from the ArgumentsAdaptorFrame. | 2181 // Load the length from the ArgumentsAdaptorFrame. |
2182 __ ldr(r0, MemOperand(r3, ArgumentsAdaptorFrameConstants::kLengthOffset)); | 2182 __ ldr(r5, MemOperand(r4, ArgumentsAdaptorFrameConstants::kLengthOffset)); |
2183 } | 2183 } |
2184 __ bind(&arguments_done); | 2184 __ bind(&arguments_done); |
2185 | 2185 |
2186 Label stack_empty, stack_done, stack_overflow; | 2186 Label stack_done, stack_overflow; |
2187 __ SmiUntag(r0); | 2187 __ SmiUntag(r5); |
2188 __ sub(r0, r0, r2, SetCC); | 2188 __ sub(r5, r5, r2, SetCC); |
2189 __ b(le, &stack_empty); | 2189 __ b(le, &stack_done); |
2190 { | 2190 { |
2191 // Check for stack overflow. | 2191 // Check for stack overflow. |
2192 Generate_StackOverflowCheck(masm, r0, r2, &stack_overflow); | 2192 Generate_StackOverflowCheck(masm, r5, r2, &stack_overflow); |
2193 | 2193 |
2194 // Forward the arguments from the caller frame. | 2194 // Forward the arguments from the caller frame. |
2195 { | 2195 { |
2196 Label loop; | 2196 Label loop; |
2197 __ add(r3, r3, Operand(kPointerSize)); | 2197 __ add(r4, r4, Operand(kPointerSize)); |
2198 __ mov(r2, r0); | 2198 __ add(r0, r0, r5); |
2199 __ bind(&loop); | 2199 __ bind(&loop); |
2200 { | 2200 { |
2201 __ ldr(ip, MemOperand(r3, r2, LSL, kPointerSizeLog2)); | 2201 __ ldr(ip, MemOperand(r4, r5, LSL, kPointerSizeLog2)); |
2202 __ push(ip); | 2202 __ push(ip); |
2203 __ sub(r2, r2, Operand(1), SetCC); | 2203 __ sub(r5, r5, Operand(1), SetCC); |
2204 __ b(ne, &loop); | 2204 __ b(ne, &loop); |
2205 } | 2205 } |
2206 } | 2206 } |
2207 } | 2207 } |
2208 __ b(&stack_done); | 2208 __ b(&stack_done); |
2209 __ bind(&stack_overflow); | 2209 __ bind(&stack_overflow); |
2210 __ TailCallRuntime(Runtime::kThrowStackOverflow); | 2210 __ TailCallRuntime(Runtime::kThrowStackOverflow); |
2211 __ bind(&stack_empty); | |
2212 { | |
2213 // We just pass the receiver, which is already on the stack. | |
2214 __ mov(r0, Operand(0)); | |
2215 } | |
2216 __ bind(&stack_done); | 2211 __ bind(&stack_done); |
2217 | 2212 |
| 2213 // Tail-call to the {code} handler. |
2218 __ Jump(code, RelocInfo::CODE_TARGET); | 2214 __ Jump(code, RelocInfo::CODE_TARGET); |
2219 } | 2215 } |
2220 | 2216 |
2221 namespace { | 2217 namespace { |
2222 | 2218 |
2223 // Drops top JavaScript frame and an arguments adaptor frame below it (if | 2219 // Drops top JavaScript frame and an arguments adaptor frame below it (if |
2224 // present) preserving all the arguments prepared for current call. | 2220 // present) preserving all the arguments prepared for current call. |
2225 // Does nothing if debugger is currently active. | 2221 // Does nothing if debugger is currently active. |
2226 // ES6 14.6.3. PrepareForTailCall | 2222 // ES6 14.6.3. PrepareForTailCall |
2227 // | 2223 // |
(...skipping 829 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
3057 } | 3053 } |
3058 // Now jump to the instructions of the returned code object. | 3054 // Now jump to the instructions of the returned code object. |
3059 __ Jump(r8); | 3055 __ Jump(r8); |
3060 } | 3056 } |
3061 #undef __ | 3057 #undef __ |
3062 | 3058 |
3063 } // namespace internal | 3059 } // namespace internal |
3064 } // namespace v8 | 3060 } // namespace v8 |
3065 | 3061 |
3066 #endif // V8_TARGET_ARCH_ARM | 3062 #endif // V8_TARGET_ARCH_ARM |
OLD | NEW |