| OLD | NEW |
| 1 // Copyright 2013 the V8 project authors. All rights reserved. | 1 // Copyright 2013 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 977 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 988 } | 988 } |
| 989 __ Jump(masm->isolate()->builtins()->OnStackReplacement(), | 989 __ Jump(masm->isolate()->builtins()->OnStackReplacement(), |
| 990 RelocInfo::CODE_TARGET); | 990 RelocInfo::CODE_TARGET); |
| 991 | 991 |
| 992 __ Bind(&ok); | 992 __ Bind(&ok); |
| 993 __ Ret(); | 993 __ Ret(); |
| 994 } | 994 } |
| 995 | 995 |
| 996 | 996 |
| 997 void Builtins::Generate_FunctionCall(MacroAssembler* masm) { | 997 void Builtins::Generate_FunctionCall(MacroAssembler* masm) { |
| 998 enum { |
| 999 call_type_JS_func = 0, |
| 1000 call_type_func_proxy = 1, |
| 1001 call_type_non_func = 2 |
| 1002 }; |
| 1003 Register argc = x0; |
| 1004 Register function = x1; |
| 1005 Register call_type = x4; |
| 1006 Register scratch1 = x10; |
| 1007 Register scratch2 = x11; |
| 998 Register receiver_type = x13; | 1008 Register receiver_type = x13; |
| 999 | 1009 |
| 1000 ASM_LOCATION("Builtins::Generate_FunctionCall"); | 1010 ASM_LOCATION("Builtins::Generate_FunctionCall"); |
| 1001 // TODO(all/rames): Optimize and use named registers. | |
| 1002 // 1. Make sure we have at least one argument. | 1011 // 1. Make sure we have at least one argument. |
| 1003 // x0: actual number of arguments | |
| 1004 { Label done; | 1012 { Label done; |
| 1005 __ Cbnz(x0, &done); | 1013 __ Cbnz(argc, &done); |
| 1006 __ LoadRoot(x10, Heap::kUndefinedValueRootIndex); | 1014 __ LoadRoot(scratch1, Heap::kUndefinedValueRootIndex); |
| 1007 __ Push(x10); | 1015 __ Push(scratch1); |
| 1008 __ Mov(x0, 1); | 1016 __ Mov(argc, 1); |
| 1009 __ Bind(&done); | 1017 __ Bind(&done); |
| 1010 } | 1018 } |
| 1011 | 1019 |
| 1012 // 2. Get the function to call (passed as receiver) from the stack, check | 1020 // 2. Get the function to call (passed as receiver) from the stack, check |
| 1013 // if it is a function. | 1021 // if it is a function. |
| 1014 // x0: actual number of arguments | |
| 1015 Label slow, non_function; | 1022 Label slow, non_function; |
| 1016 // TODO(jbramley): Consider giving Peek a unit_size parameter, like Claim and | 1023 __ Peek(function, Operand(argc, LSL, kXRegSizeInBytesLog2)); |
| 1017 // Drop. This usage pattern is very common. | 1024 __ JumpIfSmi(function, &non_function); |
| 1018 __ Peek(x1, Operand(x0, LSL, kXRegSizeInBytesLog2)); | 1025 __ JumpIfNotObjectType(function, scratch1, receiver_type, |
| 1019 __ JumpIfSmi(x1, &non_function); | 1026 JS_FUNCTION_TYPE, &slow); |
| 1020 __ JumpIfNotObjectType(x1, x10, receiver_type, JS_FUNCTION_TYPE, &slow); | |
| 1021 | 1027 |
| 1022 // 3a. Patch the first argument if necessary when calling a function. | 1028 // 3a. Patch the first argument if necessary when calling a function. |
| 1023 // x0: actual number of arguments | |
| 1024 // x1: function | |
| 1025 Label shift_arguments; | 1029 Label shift_arguments; |
| 1026 __ Mov(x4, 0); // Indicates a regular JS_FUNCTION. | 1030 __ Mov(call_type, call_type_JS_func); |
| 1027 { Label convert_to_object, use_global_receiver, patch_receiver; | 1031 { Label convert_to_object, use_global_receiver, patch_receiver; |
| 1028 // Change context eagerly in case we need the global receiver. | 1032 // Change context eagerly in case we need the global receiver. |
| 1029 __ Ldr(cp, FieldMemOperand(x1, JSFunction::kContextOffset)); | 1033 __ Ldr(cp, FieldMemOperand(function, JSFunction::kContextOffset)); |
| 1030 | 1034 |
| 1031 // Do not transform the receiver for strict mode functions. | 1035 // Do not transform the receiver for strict mode functions. |
| 1032 __ Ldr(x10, FieldMemOperand(x1, JSFunction::kSharedFunctionInfoOffset)); | 1036 // Also do not transform the receiver for native (Compilerhints already in |
| 1033 __ Ldr(w11, FieldMemOperand(x10, SharedFunctionInfo::kCompilerHintsOffset)); | 1037 // x3). |
| 1034 __ Tbnz(x11, SharedFunctionInfo::kStrictModeFunction, &shift_arguments); | 1038 __ Ldr(scratch1, |
| 1035 | 1039 FieldMemOperand(function, JSFunction::kSharedFunctionInfoOffset)); |
| 1036 // TODO(all): Shoudld we insert space to avoid BTAC collisions? | 1040 __ Ldr(scratch2.W(), |
| 1037 // Do not transform the receiver for native (Compilerhints already in x3). | 1041 FieldMemOperand(scratch1, SharedFunctionInfo::kCompilerHintsOffset)); |
| 1038 __ Tbnz(x11, SharedFunctionInfo::kNative, &shift_arguments); | 1042 __ TestAndBranchIfAnySet( |
| 1043 scratch2.W(), |
| 1044 (1 << SharedFunctionInfo::kStrictModeFunction) | |
| 1045 (1 << SharedFunctionInfo::kNative), |
| 1046 &shift_arguments); |
| 1039 | 1047 |
| 1040 // Compute the receiver in non-strict mode. | 1048 // Compute the receiver in non-strict mode. |
| 1041 __ Sub(x10, x0, 1); | 1049 Register receiver = x2; |
| 1042 __ Peek(x2, Operand(x10, LSL, kXRegSizeInBytesLog2)); | 1050 __ Sub(scratch1, argc, 1); |
| 1043 // x0: actual number of arguments | 1051 __ Peek(receiver, Operand(scratch1, LSL, kXRegSizeInBytesLog2)); |
| 1044 // x1: function | 1052 __ JumpIfSmi(receiver, &convert_to_object); |
| 1045 // x2: first argument | |
| 1046 __ JumpIfSmi(x2, &convert_to_object); | |
| 1047 | 1053 |
| 1048 // TODO(all): We could potentially work to optimize loads of root values. | 1054 __ JumpIfRoot(receiver, Heap::kUndefinedValueRootIndex, |
| 1049 // TODO(all): If the indexes are successive we can use 'ldp'. | 1055 &use_global_receiver); |
| 1050 __ JumpIfRoot(x2, Heap::kUndefinedValueRootIndex, &use_global_receiver); | 1056 __ JumpIfRoot(receiver, Heap::kNullValueRootIndex, &use_global_receiver); |
| 1051 __ JumpIfRoot(x2, Heap::kNullValueRootIndex, &use_global_receiver); | |
| 1052 | 1057 |
| 1053 STATIC_ASSERT(LAST_SPEC_OBJECT_TYPE == LAST_TYPE); | 1058 STATIC_ASSERT(LAST_SPEC_OBJECT_TYPE == LAST_TYPE); |
| 1054 __ JumpIfObjectType(x2, x10, x11, FIRST_SPEC_OBJECT_TYPE, &shift_arguments, | 1059 __ JumpIfObjectType(receiver, scratch1, scratch2, |
| 1055 ge); | 1060 FIRST_SPEC_OBJECT_TYPE, &shift_arguments, ge); |
| 1056 | 1061 |
| 1057 __ Bind(&convert_to_object); | 1062 __ Bind(&convert_to_object); |
| 1058 | 1063 |
| 1059 { | 1064 { |
| 1060 // Enter an internal frame in order to preserve argument count. | 1065 // Enter an internal frame in order to preserve argument count. |
| 1061 FrameScope scope(masm, StackFrame::INTERNAL); | 1066 FrameScope scope(masm, StackFrame::INTERNAL); |
| 1062 __ SmiTag(x0); | 1067 __ SmiTag(argc); |
| 1063 | 1068 |
| 1064 __ Push(x0, x2); | 1069 __ Push(argc, receiver); |
| 1065 __ InvokeBuiltin(Builtins::TO_OBJECT, CALL_FUNCTION); | 1070 __ InvokeBuiltin(Builtins::TO_OBJECT, CALL_FUNCTION); |
| 1066 __ Mov(x2, x0); | 1071 __ Mov(receiver, x0); |
| 1067 | 1072 |
| 1068 __ Pop(x0); | 1073 __ Pop(argc); |
| 1069 __ SmiUntag(x0); | 1074 __ SmiUntag(argc); |
| 1070 | 1075 |
| 1071 // Exit the internal frame. | 1076 // Exit the internal frame. |
| 1072 } | 1077 } |
| 1073 | 1078 |
| 1074 // Restore the function to x1, and the flag to x4. | 1079 // Restore the function and flag in the registers. |
| 1075 __ Peek(x1, Operand(x0, LSL, kXRegSizeInBytesLog2)); | 1080 __ Peek(function, Operand(argc, LSL, kXRegSizeInBytesLog2)); |
| 1076 __ Mov(x4, 0); | 1081 __ Mov(call_type, call_type_JS_func); |
| 1077 __ B(&patch_receiver); | 1082 __ B(&patch_receiver); |
| 1078 | 1083 |
| 1079 __ Bind(&use_global_receiver); | 1084 __ Bind(&use_global_receiver); |
| 1080 __ Ldr(x2, GlobalObjectMemOperand()); | 1085 __ Ldr(receiver, GlobalObjectMemOperand()); |
| 1081 __ Ldr(x2, FieldMemOperand(x2, GlobalObject::kGlobalReceiverOffset)); | 1086 __ Ldr(receiver, |
| 1087 FieldMemOperand(receiver, GlobalObject::kGlobalReceiverOffset)); |
| 1088 |
| 1082 | 1089 |
| 1083 __ Bind(&patch_receiver); | 1090 __ Bind(&patch_receiver); |
| 1084 __ Sub(x10, x0, 1); | 1091 __ Sub(scratch1, argc, 1); |
| 1085 __ Poke(x2, Operand(x10, LSL, kXRegSizeInBytesLog2)); | 1092 __ Poke(receiver, Operand(scratch1, LSL, kXRegSizeInBytesLog2)); |
| 1086 | 1093 |
| 1087 __ B(&shift_arguments); | 1094 __ B(&shift_arguments); |
| 1088 } | 1095 } |
| 1089 | 1096 |
| 1090 // 3b. Check for function proxy. | 1097 // 3b. Check for function proxy. |
| 1091 __ Bind(&slow); | 1098 __ Bind(&slow); |
| 1092 __ Mov(x4, 1); // Indicate function proxy. | 1099 __ Mov(call_type, call_type_func_proxy); |
| 1093 __ Cmp(receiver_type, JS_FUNCTION_PROXY_TYPE); | 1100 __ Cmp(receiver_type, JS_FUNCTION_PROXY_TYPE); |
| 1094 __ B(eq, &shift_arguments); | 1101 __ B(eq, &shift_arguments); |
| 1095 __ Bind(&non_function); | 1102 __ Bind(&non_function); |
| 1096 __ Mov(x4, 2); // Indicate non-function. | 1103 __ Mov(call_type, call_type_non_func); |
| 1097 | 1104 |
| 1098 // 3c. Patch the first argument when calling a non-function. The | 1105 // 3c. Patch the first argument when calling a non-function. The |
| 1099 // CALL_NON_FUNCTION builtin expects the non-function callee as | 1106 // CALL_NON_FUNCTION builtin expects the non-function callee as |
| 1100 // receiver, so overwrite the first argument which will ultimately | 1107 // receiver, so overwrite the first argument which will ultimately |
| 1101 // become the receiver. | 1108 // become the receiver. |
| 1102 // x0: actual number of arguments | 1109 // call type (0: JS function, 1: function proxy, 2: non-function) |
| 1103 // x1: function | 1110 __ Sub(scratch1, argc, 1); |
| 1104 // x4: call type (0: JS function, 1: function proxy, 2: non-function) | 1111 __ Poke(function, Operand(scratch1, LSL, kXRegSizeInBytesLog2)); |
| 1105 __ Sub(x10, x0, 1); | |
| 1106 __ Poke(x1, Operand(x10, LSL, kXRegSizeInBytesLog2)); | |
| 1107 | 1112 |
| 1108 // 4. Shift arguments and return address one slot down on the stack | 1113 // 4. Shift arguments and return address one slot down on the stack |
| 1109 // (overwriting the original receiver). Adjust argument count to make | 1114 // (overwriting the original receiver). Adjust argument count to make |
| 1110 // the original first argument the new receiver. | 1115 // the original first argument the new receiver. |
| 1111 // x0: actual number of arguments | 1116 // call type (0: JS function, 1: function proxy, 2: non-function) |
| 1112 // x1: function | |
| 1113 // x4: call type (0: JS function, 1: function proxy, 2: non-function) | |
| 1114 __ Bind(&shift_arguments); | 1117 __ Bind(&shift_arguments); |
| 1115 { Label loop; | 1118 { Label loop; |
| 1116 // Calculate the copy start address (destination). Copy end address is jssp. | 1119 // Calculate the copy start address (destination). Copy end address is jssp. |
| 1117 __ Add(x11, jssp, Operand(x0, LSL, kPointerSizeLog2)); | 1120 __ Add(scratch2, jssp, Operand(argc, LSL, kPointerSizeLog2)); |
| 1118 __ Sub(x10, x11, kPointerSize); | 1121 __ Sub(scratch1, scratch2, kPointerSize); |
| 1119 | 1122 |
| 1120 // TODO(all): Optimize to copy values 2 by 2? | |
| 1121 __ Bind(&loop); | 1123 __ Bind(&loop); |
| 1122 __ Ldr(x12, MemOperand(x10, -kPointerSize, PostIndex)); | 1124 __ Ldr(x12, MemOperand(scratch1, -kPointerSize, PostIndex)); |
| 1123 __ Str(x12, MemOperand(x11, -kPointerSize, PostIndex)); | 1125 __ Str(x12, MemOperand(scratch2, -kPointerSize, PostIndex)); |
| 1124 __ Cmp(x10, jssp); | 1126 __ Cmp(scratch1, jssp); |
| 1125 __ B(ge, &loop); | 1127 __ B(ge, &loop); |
| 1126 // Adjust the actual number of arguments and remove the top element | 1128 // Adjust the actual number of arguments and remove the top element |
| 1127 // (which is a copy of the last argument). | 1129 // (which is a copy of the last argument). |
| 1128 __ Sub(x0, x0, 1); | 1130 __ Sub(argc, argc, 1); |
| 1129 __ Drop(1); | 1131 __ Drop(1); |
| 1130 } | 1132 } |
| 1131 | 1133 |
| 1132 // 5a. Call non-function via tail call to CALL_NON_FUNCTION builtin, | 1134 // 5a. Call non-function via tail call to CALL_NON_FUNCTION builtin, |
| 1133 // or a function proxy via CALL_FUNCTION_PROXY. | 1135 // or a function proxy via CALL_FUNCTION_PROXY. |
| 1134 // x0: actual number of arguments | 1136 // call type (0: JS function, 1: function proxy, 2: non-function) |
| 1135 // x1: function | 1137 { Label js_function, non_proxy; |
| 1136 // x4: call type (0: JS function, 1: function proxy, 2: non-function) | 1138 __ Cbz(call_type, &js_function); |
| 1137 { Label function, non_proxy; | |
| 1138 __ Cbz(x4, &function); | |
| 1139 // Expected number of arguments is 0 for CALL_NON_FUNCTION. | 1139 // Expected number of arguments is 0 for CALL_NON_FUNCTION. |
| 1140 __ Mov(x2, 0); | 1140 __ Mov(x2, 0); |
| 1141 __ Cmp(x4, 1); | 1141 __ Cmp(call_type, call_type_func_proxy); |
| 1142 __ B(ne, &non_proxy); | 1142 __ B(ne, &non_proxy); |
| 1143 | 1143 |
| 1144 __ Push(x1); // Re-add proxy object as additional argument. | 1144 __ Push(function); // Re-add proxy object as additional argument. |
| 1145 __ Add(x0, x0, 1); | 1145 __ Add(argc, argc, 1); |
| 1146 __ GetBuiltinFunction(x1, Builtins::CALL_FUNCTION_PROXY); | 1146 __ GetBuiltinFunction(function, Builtins::CALL_FUNCTION_PROXY); |
| 1147 __ Jump(masm->isolate()->builtins()->ArgumentsAdaptorTrampoline(), | 1147 __ Jump(masm->isolate()->builtins()->ArgumentsAdaptorTrampoline(), |
| 1148 RelocInfo::CODE_TARGET); | 1148 RelocInfo::CODE_TARGET); |
| 1149 | 1149 |
| 1150 __ Bind(&non_proxy); | 1150 __ Bind(&non_proxy); |
| 1151 __ GetBuiltinFunction(x1, Builtins::CALL_NON_FUNCTION); | 1151 __ GetBuiltinFunction(function, Builtins::CALL_NON_FUNCTION); |
| 1152 __ Jump(masm->isolate()->builtins()->ArgumentsAdaptorTrampoline(), | 1152 __ Jump(masm->isolate()->builtins()->ArgumentsAdaptorTrampoline(), |
| 1153 RelocInfo::CODE_TARGET); | 1153 RelocInfo::CODE_TARGET); |
| 1154 __ Bind(&function); | 1154 __ Bind(&js_function); |
| 1155 } | 1155 } |
| 1156 | 1156 |
| 1157 // 5b. Get the code to call from the function and check that the number of | 1157 // 5b. Get the code to call from the function and check that the number of |
| 1158 // expected arguments matches what we're providing. If so, jump | 1158 // expected arguments matches what we're providing. If so, jump |
| 1159 // (tail-call) to the code in register edx without checking arguments. | 1159 // (tail-call) to the code in register edx without checking arguments. |
| 1160 // x0: actual number of arguments | 1160 __ Ldr(x3, FieldMemOperand(function, JSFunction::kSharedFunctionInfoOffset)); |
| 1161 // x1: function | |
| 1162 __ Ldr(x3, FieldMemOperand(x1, JSFunction::kSharedFunctionInfoOffset)); | |
| 1163 __ Ldrsw(x2, | 1161 __ Ldrsw(x2, |
| 1164 FieldMemOperand(x3, | 1162 FieldMemOperand(x3, |
| 1165 SharedFunctionInfo::kFormalParameterCountOffset)); | 1163 SharedFunctionInfo::kFormalParameterCountOffset)); |
| 1166 Label dont_adapt_args; | 1164 Label dont_adapt_args; |
| 1167 __ Cmp(x2, x0); // Check formal and actual parameter counts. | 1165 __ Cmp(x2, argc); // Check formal and actual parameter counts. |
| 1168 __ B(eq, &dont_adapt_args); | 1166 __ B(eq, &dont_adapt_args); |
| 1169 __ Jump(masm->isolate()->builtins()->ArgumentsAdaptorTrampoline(), | 1167 __ Jump(masm->isolate()->builtins()->ArgumentsAdaptorTrampoline(), |
| 1170 RelocInfo::CODE_TARGET); | 1168 RelocInfo::CODE_TARGET); |
| 1171 __ Bind(&dont_adapt_args); | 1169 __ Bind(&dont_adapt_args); |
| 1172 | 1170 |
| 1173 __ Ldr(x3, FieldMemOperand(x1, JSFunction::kCodeEntryOffset)); | 1171 __ Ldr(x3, FieldMemOperand(function, JSFunction::kCodeEntryOffset)); |
| 1174 ParameterCount expected(0); | 1172 ParameterCount expected(0); |
| 1175 __ InvokeCode(x3, expected, expected, JUMP_FUNCTION, NullCallWrapper()); | 1173 __ InvokeCode(x3, expected, expected, JUMP_FUNCTION, NullCallWrapper()); |
| 1176 } | 1174 } |
| 1177 | 1175 |
| 1178 | 1176 |
| 1179 void Builtins::Generate_FunctionApply(MacroAssembler* masm) { | 1177 void Builtins::Generate_FunctionApply(MacroAssembler* masm) { |
| 1180 ASM_LOCATION("Builtins::Generate_FunctionApply"); | 1178 ASM_LOCATION("Builtins::Generate_FunctionApply"); |
| 1181 const int kIndexOffset = | 1179 const int kIndexOffset = |
| 1182 StandardFrameConstants::kExpressionsOffset - (2 * kPointerSize); | 1180 StandardFrameConstants::kExpressionsOffset - (2 * kPointerSize); |
| 1183 const int kLimitOffset = | 1181 const int kLimitOffset = |
| (...skipping 288 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1472 __ Bind(&dont_adapt_arguments); | 1470 __ Bind(&dont_adapt_arguments); |
| 1473 __ Jump(x3); | 1471 __ Jump(x3); |
| 1474 } | 1472 } |
| 1475 | 1473 |
| 1476 | 1474 |
| 1477 #undef __ | 1475 #undef __ |
| 1478 | 1476 |
| 1479 } } // namespace v8::internal | 1477 } } // namespace v8::internal |
| 1480 | 1478 |
| 1481 #endif // V8_TARGET_ARCH_ARM | 1479 #endif // V8_TARGET_ARCH_ARM |
| OLD | NEW |