OLD | NEW |
1 // Copyright 2011 the V8 project authors. All rights reserved. | 1 // Copyright 2011 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 2081 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
2092 CallFunctionStub stub(arg_count, flags); | 2092 CallFunctionStub stub(arg_count, flags); |
2093 __ CallStub(&stub); | 2093 __ CallStub(&stub); |
2094 RecordJSReturnSite(expr); | 2094 RecordJSReturnSite(expr); |
2095 // Restore context register. | 2095 // Restore context register. |
2096 __ movq(rsi, Operand(rbp, StandardFrameConstants::kContextOffset)); | 2096 __ movq(rsi, Operand(rbp, StandardFrameConstants::kContextOffset)); |
2097 // Discard the function left on TOS. | 2097 // Discard the function left on TOS. |
2098 context()->DropAndPlug(1, rax); | 2098 context()->DropAndPlug(1, rax); |
2099 } | 2099 } |
2100 | 2100 |
2101 | 2101 |
2102 void FullCodeGenerator::EmitResolvePossiblyDirectEval(ResolveEvalFlag flag, | 2102 void FullCodeGenerator::EmitResolvePossiblyDirectEval(int arg_count) { |
2103 int arg_count) { | |
2104 // Push copy of the first argument or undefined if it doesn't exist. | 2103 // Push copy of the first argument or undefined if it doesn't exist. |
2105 if (arg_count > 0) { | 2104 if (arg_count > 0) { |
2106 __ push(Operand(rsp, arg_count * kPointerSize)); | 2105 __ push(Operand(rsp, arg_count * kPointerSize)); |
2107 } else { | 2106 } else { |
2108 __ PushRoot(Heap::kUndefinedValueRootIndex); | 2107 __ PushRoot(Heap::kUndefinedValueRootIndex); |
2109 } | 2108 } |
2110 | 2109 |
2111 // Push the receiver of the enclosing function and do runtime call. | 2110 // Push the receiver of the enclosing function and do runtime call. |
2112 __ push(Operand(rbp, (2 + info_->scope()->num_parameters()) * kPointerSize)); | 2111 __ push(Operand(rbp, (2 + info_->scope()->num_parameters()) * kPointerSize)); |
2113 | 2112 |
2114 // Push the strict mode flag. In harmony mode every eval call | 2113 // Push the strict mode flag. In harmony mode every eval call |
2115 // is a strict mode eval call. | 2114 // is a strict mode eval call. |
2116 StrictModeFlag strict_mode = | 2115 StrictModeFlag strict_mode = |
2117 FLAG_harmony_scoping ? kStrictMode : strict_mode_flag(); | 2116 FLAG_harmony_scoping ? kStrictMode : strict_mode_flag(); |
2118 __ Push(Smi::FromInt(strict_mode)); | 2117 __ Push(Smi::FromInt(strict_mode)); |
2119 | 2118 |
2120 __ CallRuntime(flag == SKIP_CONTEXT_LOOKUP | 2119 __ CallRuntime(Runtime::kResolvePossiblyDirectEval, 4); |
2121 ? Runtime::kResolvePossiblyDirectEvalNoLookup | |
2122 : Runtime::kResolvePossiblyDirectEval, 4); | |
2123 } | 2120 } |
2124 | 2121 |
2125 | 2122 |
2126 void FullCodeGenerator::VisitCall(Call* expr) { | 2123 void FullCodeGenerator::VisitCall(Call* expr) { |
2127 #ifdef DEBUG | 2124 #ifdef DEBUG |
2128 // We want to verify that RecordJSReturnSite gets called on all paths | 2125 // We want to verify that RecordJSReturnSite gets called on all paths |
2129 // through this function. Avoid early returns. | 2126 // through this function. Avoid early returns. |
2130 expr->return_is_recorded_ = false; | 2127 expr->return_is_recorded_ = false; |
2131 #endif | 2128 #endif |
2132 | 2129 |
(...skipping 10 matching lines...) Expand all Loading... |
2143 int arg_count = args->length(); | 2140 int arg_count = args->length(); |
2144 { PreservePositionScope pos_scope(masm()->positions_recorder()); | 2141 { PreservePositionScope pos_scope(masm()->positions_recorder()); |
2145 VisitForStackValue(callee); | 2142 VisitForStackValue(callee); |
2146 __ PushRoot(Heap::kUndefinedValueRootIndex); // Reserved receiver slot. | 2143 __ PushRoot(Heap::kUndefinedValueRootIndex); // Reserved receiver slot. |
2147 | 2144 |
2148 // Push the arguments. | 2145 // Push the arguments. |
2149 for (int i = 0; i < arg_count; i++) { | 2146 for (int i = 0; i < arg_count; i++) { |
2150 VisitForStackValue(args->at(i)); | 2147 VisitForStackValue(args->at(i)); |
2151 } | 2148 } |
2152 | 2149 |
2153 // If we know that eval can only be shadowed by eval-introduced | |
2154 // variables we attempt to load the global eval function directly in | |
2155 // generated code. If we succeed, there is no need to perform a | |
2156 // context lookup in the runtime system. | |
2157 Label done; | |
2158 Variable* var = proxy->var(); | |
2159 if (!var->IsUnallocated() && var->mode() == DYNAMIC_GLOBAL) { | |
2160 Label slow; | |
2161 EmitLoadGlobalCheckExtensions(var, NOT_INSIDE_TYPEOF, &slow); | |
2162 // Push the function and resolve eval. | |
2163 __ push(rax); | |
2164 EmitResolvePossiblyDirectEval(SKIP_CONTEXT_LOOKUP, arg_count); | |
2165 __ jmp(&done); | |
2166 __ bind(&slow); | |
2167 } | |
2168 | |
2169 // Push a copy of the function (found below the arguments) and resolve | 2150 // Push a copy of the function (found below the arguments) and resolve |
2170 // eval. | 2151 // eval. |
2171 __ push(Operand(rsp, (arg_count + 1) * kPointerSize)); | 2152 __ push(Operand(rsp, (arg_count + 1) * kPointerSize)); |
2172 EmitResolvePossiblyDirectEval(PERFORM_CONTEXT_LOOKUP, arg_count); | 2153 EmitResolvePossiblyDirectEval(arg_count); |
2173 __ bind(&done); | |
2174 | 2154 |
2175 // The runtime call returns a pair of values in rax (function) and | 2155 // The runtime call returns a pair of values in rax (function) and |
2176 // rdx (receiver). Touch up the stack with the right values. | 2156 // rdx (receiver). Touch up the stack with the right values. |
2177 __ movq(Operand(rsp, (arg_count + 0) * kPointerSize), rdx); | 2157 __ movq(Operand(rsp, (arg_count + 0) * kPointerSize), rdx); |
2178 __ movq(Operand(rsp, (arg_count + 1) * kPointerSize), rax); | 2158 __ movq(Operand(rsp, (arg_count + 1) * kPointerSize), rax); |
2179 } | 2159 } |
2180 // Record source position for debugger. | 2160 // Record source position for debugger. |
2181 SetSourcePosition(expr->position()); | 2161 SetSourcePosition(expr->position()); |
2182 CallFunctionStub stub(arg_count, RECEIVER_MIGHT_BE_IMPLICIT); | 2162 CallFunctionStub stub(arg_count, RECEIVER_MIGHT_BE_IMPLICIT); |
2183 __ CallStub(&stub); | 2163 __ CallStub(&stub); |
(...skipping 2090 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
4274 *context_length = 0; | 4254 *context_length = 0; |
4275 return previous_; | 4255 return previous_; |
4276 } | 4256 } |
4277 | 4257 |
4278 | 4258 |
4279 #undef __ | 4259 #undef __ |
4280 | 4260 |
4281 } } // namespace v8::internal | 4261 } } // namespace v8::internal |
4282 | 4262 |
4283 #endif // V8_TARGET_ARCH_X64 | 4263 #endif // V8_TARGET_ARCH_X64 |
OLD | NEW |