OLD | NEW |
1 // Copyright 2013 the V8 project authors. All rights reserved. | 1 // Copyright 2013 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_ARM64 | 5 #if V8_TARGET_ARCH_ARM64 |
6 | 6 |
7 #include "src/arm64/frames-arm64.h" | 7 #include "src/arm64/frames-arm64.h" |
8 #include "src/codegen.h" | 8 #include "src/codegen.h" |
9 #include "src/debug/debug.h" | 9 #include "src/debug/debug.h" |
10 #include "src/deoptimizer.h" | 10 #include "src/deoptimizer.h" |
(...skipping 1974 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1985 { | 1985 { |
1986 Label no_interpreter_frame; | 1986 Label no_interpreter_frame; |
1987 __ Ldr(scratch3, MemOperand(fp, StandardFrameConstants::kMarkerOffset)); | 1987 __ Ldr(scratch3, MemOperand(fp, StandardFrameConstants::kMarkerOffset)); |
1988 __ Cmp(scratch3, Operand(Smi::FromInt(StackFrame::STUB))); | 1988 __ Cmp(scratch3, Operand(Smi::FromInt(StackFrame::STUB))); |
1989 __ B(ne, &no_interpreter_frame); | 1989 __ B(ne, &no_interpreter_frame); |
1990 __ Ldr(fp, MemOperand(fp, StandardFrameConstants::kCallerFPOffset)); | 1990 __ Ldr(fp, MemOperand(fp, StandardFrameConstants::kCallerFPOffset)); |
1991 __ bind(&no_interpreter_frame); | 1991 __ bind(&no_interpreter_frame); |
1992 } | 1992 } |
1993 | 1993 |
1994 // Check if next frame is an arguments adaptor frame. | 1994 // Check if next frame is an arguments adaptor frame. |
| 1995 Register caller_args_count_reg = scratch1; |
1995 Label no_arguments_adaptor, formal_parameter_count_loaded; | 1996 Label no_arguments_adaptor, formal_parameter_count_loaded; |
1996 __ Ldr(scratch2, MemOperand(fp, StandardFrameConstants::kCallerFPOffset)); | 1997 __ Ldr(scratch2, MemOperand(fp, StandardFrameConstants::kCallerFPOffset)); |
1997 __ Ldr(scratch3, | 1998 __ Ldr(scratch3, |
1998 MemOperand(scratch2, StandardFrameConstants::kContextOffset)); | 1999 MemOperand(scratch2, StandardFrameConstants::kContextOffset)); |
1999 __ Cmp(scratch3, Operand(Smi::FromInt(StackFrame::ARGUMENTS_ADAPTOR))); | 2000 __ Cmp(scratch3, Operand(Smi::FromInt(StackFrame::ARGUMENTS_ADAPTOR))); |
2000 __ B(ne, &no_arguments_adaptor); | 2001 __ B(ne, &no_arguments_adaptor); |
2001 | 2002 |
2002 // Drop arguments adaptor frame and load arguments count. | 2003 // Drop current frame and load arguments count from arguments adaptor frame. |
2003 __ mov(fp, scratch2); | 2004 __ mov(fp, scratch2); |
2004 __ Ldr(scratch1, | 2005 __ Ldr(caller_args_count_reg, |
2005 MemOperand(fp, ArgumentsAdaptorFrameConstants::kLengthOffset)); | 2006 MemOperand(fp, ArgumentsAdaptorFrameConstants::kLengthOffset)); |
2006 __ SmiUntag(scratch1); | 2007 __ SmiUntag(caller_args_count_reg); |
2007 __ B(&formal_parameter_count_loaded); | 2008 __ B(&formal_parameter_count_loaded); |
2008 | 2009 |
2009 __ bind(&no_arguments_adaptor); | 2010 __ bind(&no_arguments_adaptor); |
2010 // Load caller's formal parameter count | 2011 // Load caller's formal parameter count |
2011 __ Ldr(scratch1, MemOperand(fp, JavaScriptFrameConstants::kFunctionOffset)); | 2012 __ Ldr(scratch1, MemOperand(fp, JavaScriptFrameConstants::kFunctionOffset)); |
2012 __ Ldr(scratch1, | 2013 __ Ldr(scratch1, |
2013 FieldMemOperand(scratch1, JSFunction::kSharedFunctionInfoOffset)); | 2014 FieldMemOperand(scratch1, JSFunction::kSharedFunctionInfoOffset)); |
2014 __ Ldrsw(scratch1, | 2015 __ Ldrsw(caller_args_count_reg, |
2015 FieldMemOperand(scratch1, | 2016 FieldMemOperand(scratch1, |
2016 SharedFunctionInfo::kFormalParameterCountOffset)); | 2017 SharedFunctionInfo::kFormalParameterCountOffset)); |
2017 __ bind(&formal_parameter_count_loaded); | 2018 __ bind(&formal_parameter_count_loaded); |
2018 | 2019 |
2019 // Calculate the end of destination area where we will put the arguments | 2020 ParameterCount callee_args_count(args_reg); |
2020 // after we drop current frame. We add kPointerSize to count the receiver | 2021 __ PrepareForTailCall(callee_args_count, caller_args_count_reg, scratch2, |
2021 // argument which is not included into formal parameters count. | 2022 scratch3); |
2022 Register dst_reg = scratch2; | |
2023 __ add(dst_reg, fp, Operand(scratch1, LSL, kPointerSizeLog2)); | |
2024 __ add(dst_reg, dst_reg, | |
2025 Operand(StandardFrameConstants::kCallerSPOffset + kPointerSize)); | |
2026 | |
2027 Register src_reg = scratch1; | |
2028 __ add(src_reg, jssp, Operand(args_reg, LSL, kPointerSizeLog2)); | |
2029 // Count receiver argument as well (not included in args_reg). | |
2030 __ add(src_reg, src_reg, Operand(kPointerSize)); | |
2031 | |
2032 if (FLAG_debug_code) { | |
2033 __ Cmp(src_reg, dst_reg); | |
2034 __ Check(lo, kStackAccessBelowStackPointer); | |
2035 } | |
2036 | |
2037 // Restore caller's frame pointer and return address now as they will be | |
2038 // overwritten by the copying loop. | |
2039 __ Ldr(lr, MemOperand(fp, StandardFrameConstants::kCallerPCOffset)); | |
2040 __ Ldr(fp, MemOperand(fp, StandardFrameConstants::kCallerFPOffset)); | |
2041 | |
2042 // Now copy callee arguments to the caller frame going backwards to avoid | |
2043 // callee arguments corruption (source and destination areas could overlap). | |
2044 | |
2045 // Both src_reg and dst_reg are pointing to the word after the one to copy, | |
2046 // so they must be pre-decremented in the loop. | |
2047 Register tmp_reg = scratch3; | |
2048 Label loop, entry; | |
2049 __ B(&entry); | |
2050 __ bind(&loop); | |
2051 __ Ldr(tmp_reg, MemOperand(src_reg, -kPointerSize, PreIndex)); | |
2052 __ Str(tmp_reg, MemOperand(dst_reg, -kPointerSize, PreIndex)); | |
2053 __ bind(&entry); | |
2054 __ Cmp(jssp, src_reg); | |
2055 __ B(ne, &loop); | |
2056 | |
2057 // Leave current frame. | |
2058 __ Mov(jssp, dst_reg); | |
2059 __ SetStackPointer(jssp); | |
2060 __ AssertStackConsistency(); | |
2061 | |
2062 __ bind(&done); | 2023 __ bind(&done); |
2063 } | 2024 } |
2064 } // namespace | 2025 } // namespace |
2065 | 2026 |
2066 // static | 2027 // static |
2067 void Builtins::Generate_CallFunction(MacroAssembler* masm, | 2028 void Builtins::Generate_CallFunction(MacroAssembler* masm, |
2068 ConvertReceiverMode mode, | 2029 ConvertReceiverMode mode, |
2069 TailCallMode tail_call_mode) { | 2030 TailCallMode tail_call_mode) { |
2070 ASM_LOCATION("Builtins::Generate_CallFunction"); | 2031 ASM_LOCATION("Builtins::Generate_CallFunction"); |
2071 // ----------- S t a t e ------------- | 2032 // ----------- S t a t e ------------- |
(...skipping 637 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
2709 } | 2670 } |
2710 } | 2671 } |
2711 | 2672 |
2712 | 2673 |
2713 #undef __ | 2674 #undef __ |
2714 | 2675 |
2715 } // namespace internal | 2676 } // namespace internal |
2716 } // namespace v8 | 2677 } // namespace v8 |
2717 | 2678 |
2718 #endif // V8_TARGET_ARCH_ARM | 2679 #endif // V8_TARGET_ARCH_ARM |
OLD | NEW |