OLD | NEW |
---|---|
1 // Copyright 2014 the V8 project authors. All rights reserved. | 1 // Copyright 2014 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 #include "src/code-factory.h" | 5 #include "src/code-factory.h" |
6 #include "src/compilation-dependencies.h" | 6 #include "src/compilation-dependencies.h" |
7 #include "src/compiler/access-builder.h" | 7 #include "src/compiler/access-builder.h" |
8 #include "src/compiler/js-graph.h" | 8 #include "src/compiler/js-graph.h" |
9 #include "src/compiler/js-typed-lowering.h" | 9 #include "src/compiler/js-typed-lowering.h" |
10 #include "src/compiler/linkage.h" | 10 #include "src/compiler/linkage.h" |
(...skipping 1449 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
1460 Node* const control = NodeProperties::GetControlInput(node); | 1460 Node* const control = NodeProperties::GetControlInput(node); |
1461 Node* const context = NodeProperties::GetContextInput(node); | 1461 Node* const context = NodeProperties::GetContextInput(node); |
1462 // TODO(mstarzinger): Duplicate parameters are not handled yet. | 1462 // TODO(mstarzinger): Duplicate parameters are not handled yet. |
1463 if (shared->has_duplicate_parameters()) return NoChange(); | 1463 if (shared->has_duplicate_parameters()) return NoChange(); |
1464 // Choose the correct frame state and frame state info depending on whether | 1464 // Choose the correct frame state and frame state info depending on whether |
1465 // there conceptually is an arguments adaptor frame in the call chain. | 1465 // there conceptually is an arguments adaptor frame in the call chain. |
1466 Node* const args_state = GetArgumentsFrameState(frame_state); | 1466 Node* const args_state = GetArgumentsFrameState(frame_state); |
1467 FrameStateInfo args_state_info = OpParameter<FrameStateInfo>(args_state); | 1467 FrameStateInfo args_state_info = OpParameter<FrameStateInfo>(args_state); |
1468 // Prepare element backing store to be used by arguments object. | 1468 // Prepare element backing store to be used by arguments object. |
1469 bool has_aliased_arguments = false; | 1469 bool has_aliased_arguments = false; |
1470 Node* const elements = AllocateAliasedArguments( | 1470 Node* allocate_effect = effect; |
1471 effect, control, args_state, context, shared, &has_aliased_arguments); | 1471 Node* const elements = |
1472 AllocateAliasedArguments(&allocate_effect, control, args_state, context, | |
1473 shared, &has_aliased_arguments); | |
1472 // Load the arguments object map from the current native context. | 1474 // Load the arguments object map from the current native context. |
1473 Node* const load_global_object = graph()->NewNode( | 1475 Node* const load_global_object = graph()->NewNode( |
1474 simplified()->LoadField( | 1476 simplified()->LoadField( |
1475 AccessBuilder::ForContextSlot(Context::GLOBAL_OBJECT_INDEX)), | 1477 AccessBuilder::ForContextSlot(Context::GLOBAL_OBJECT_INDEX)), |
1476 context, effect, control); | 1478 context, effect, control); |
1477 Node* const load_native_context = | 1479 Node* const load_native_context = |
1478 graph()->NewNode(simplified()->LoadField( | 1480 graph()->NewNode(simplified()->LoadField( |
1479 AccessBuilder::ForJSGlobalObjectNativeContext()), | 1481 AccessBuilder::ForJSGlobalObjectNativeContext()), |
1480 load_global_object, effect, control); | 1482 load_global_object, effect, control); |
1481 Node* const load_arguments_map = graph()->NewNode( | 1483 Node* const load_arguments_map = graph()->NewNode( |
1482 simplified()->LoadField(AccessBuilder::ForContextSlot( | 1484 simplified()->LoadField(AccessBuilder::ForContextSlot( |
1483 has_aliased_arguments ? Context::FAST_ALIASED_ARGUMENTS_MAP_INDEX | 1485 has_aliased_arguments ? Context::FAST_ALIASED_ARGUMENTS_MAP_INDEX |
1484 : Context::SLOPPY_ARGUMENTS_MAP_INDEX)), | 1486 : Context::SLOPPY_ARGUMENTS_MAP_INDEX)), |
1485 load_native_context, effect, control); | 1487 load_native_context, effect, control); |
1486 // Actually allocate and initialize the arguments object. | 1488 // Actually allocate and initialize the arguments object. |
1487 AllocationBuilder a(jsgraph(), effect, control); | 1489 AllocationBuilder a(jsgraph(), allocate_effect, control); |
Michael Starzinger
2015/11/20 09:15:56
Instead of having the out-parameter passing, could
sigurds
2015/11/20 10:24:39
Done.
| |
1488 Node* properties = jsgraph()->EmptyFixedArrayConstant(); | 1490 Node* properties = jsgraph()->EmptyFixedArrayConstant(); |
1489 int length = args_state_info.parameter_count() - 1; // Minus receiver. | 1491 int length = args_state_info.parameter_count() - 1; // Minus receiver. |
1490 STATIC_ASSERT(Heap::kSloppyArgumentsObjectSize == 5 * kPointerSize); | 1492 STATIC_ASSERT(Heap::kSloppyArgumentsObjectSize == 5 * kPointerSize); |
1491 a.Allocate(Heap::kSloppyArgumentsObjectSize); | 1493 a.Allocate(Heap::kSloppyArgumentsObjectSize); |
1492 a.Store(AccessBuilder::ForMap(), load_arguments_map); | 1494 a.Store(AccessBuilder::ForMap(), load_arguments_map); |
1493 a.Store(AccessBuilder::ForJSObjectProperties(), properties); | 1495 a.Store(AccessBuilder::ForJSObjectProperties(), properties); |
1494 a.Store(AccessBuilder::ForJSObjectElements(), elements); | 1496 a.Store(AccessBuilder::ForJSObjectElements(), elements); |
1495 a.Store(AccessBuilder::ForArgumentsLength(), jsgraph()->Constant(length)); | 1497 a.Store(AccessBuilder::ForArgumentsLength(), jsgraph()->Constant(length)); |
1496 a.Store(AccessBuilder::ForArgumentsCallee(), callee); | 1498 a.Store(AccessBuilder::ForArgumentsCallee(), callee); |
1497 RelaxControls(node); | 1499 RelaxControls(node); |
1498 a.FinishAndChange(node); | 1500 a.FinishAndChange(node); |
1499 return Changed(node); | 1501 return Changed(node); |
1500 } | 1502 } |
1501 | 1503 |
1502 // Use inline allocation for all unmapped arguments objects within inlined | 1504 // Use inline allocation for all unmapped arguments objects within inlined |
1503 // (i.e. non-outermost) frames, independent of the object size. | 1505 // (i.e. non-outermost) frames, independent of the object size. |
1504 if (p.type() == CreateArgumentsParameters::kUnmappedArguments && | 1506 if (p.type() == CreateArgumentsParameters::kUnmappedArguments && |
1505 outer_state->opcode() == IrOpcode::kFrameState) { | 1507 outer_state->opcode() == IrOpcode::kFrameState) { |
1506 Node* const effect = NodeProperties::GetEffectInput(node); | 1508 Node* const effect = NodeProperties::GetEffectInput(node); |
1507 Node* const control = NodeProperties::GetControlInput(node); | 1509 Node* const control = NodeProperties::GetControlInput(node); |
1508 Node* const context = NodeProperties::GetContextInput(node); | 1510 Node* const context = NodeProperties::GetContextInput(node); |
1509 // Choose the correct frame state and frame state info depending on whether | 1511 // Choose the correct frame state and frame state info depending on whether |
1510 // there conceptually is an arguments adaptor frame in the call chain. | 1512 // there conceptually is an arguments adaptor frame in the call chain. |
1511 Node* const args_state = GetArgumentsFrameState(frame_state); | 1513 Node* const args_state = GetArgumentsFrameState(frame_state); |
1512 FrameStateInfo args_state_info = OpParameter<FrameStateInfo>(args_state); | 1514 FrameStateInfo args_state_info = OpParameter<FrameStateInfo>(args_state); |
1513 // Prepare element backing store to be used by arguments object. | 1515 // Prepare element backing store to be used by arguments object. |
1514 Node* const elements = AllocateArguments(effect, control, args_state); | 1516 Node* allocate_effect = effect; |
1517 Node* const elements = | |
1518 AllocateArguments(&allocate_effect, control, args_state); | |
1515 // Load the arguments object map from the current native context. | 1519 // Load the arguments object map from the current native context. |
1516 Node* const load_global_object = graph()->NewNode( | 1520 Node* const load_global_object = graph()->NewNode( |
1517 simplified()->LoadField( | 1521 simplified()->LoadField( |
1518 AccessBuilder::ForContextSlot(Context::GLOBAL_OBJECT_INDEX)), | 1522 AccessBuilder::ForContextSlot(Context::GLOBAL_OBJECT_INDEX)), |
1519 context, effect, control); | 1523 context, effect, control); |
1520 Node* const load_native_context = | 1524 Node* const load_native_context = |
1521 graph()->NewNode(simplified()->LoadField( | 1525 graph()->NewNode(simplified()->LoadField( |
1522 AccessBuilder::ForJSGlobalObjectNativeContext()), | 1526 AccessBuilder::ForJSGlobalObjectNativeContext()), |
1523 load_global_object, effect, control); | 1527 load_global_object, effect, control); |
1524 Node* const load_arguments_map = graph()->NewNode( | 1528 Node* const load_arguments_map = graph()->NewNode( |
1525 simplified()->LoadField( | 1529 simplified()->LoadField( |
1526 AccessBuilder::ForContextSlot(Context::STRICT_ARGUMENTS_MAP_INDEX)), | 1530 AccessBuilder::ForContextSlot(Context::STRICT_ARGUMENTS_MAP_INDEX)), |
1527 load_native_context, effect, control); | 1531 load_native_context, effect, control); |
1528 // Actually allocate and initialize the arguments object. | 1532 // Actually allocate and initialize the arguments object. |
1529 AllocationBuilder a(jsgraph(), effect, control); | 1533 AllocationBuilder a(jsgraph(), allocate_effect, control); |
Michael Starzinger
2015/11/20 09:15:56
Likewise.
sigurds
2015/11/20 10:24:39
Done.
| |
1530 Node* properties = jsgraph()->EmptyFixedArrayConstant(); | 1534 Node* properties = jsgraph()->EmptyFixedArrayConstant(); |
1531 int length = args_state_info.parameter_count() - 1; // Minus receiver. | 1535 int length = args_state_info.parameter_count() - 1; // Minus receiver. |
1532 STATIC_ASSERT(Heap::kStrictArgumentsObjectSize == 4 * kPointerSize); | 1536 STATIC_ASSERT(Heap::kStrictArgumentsObjectSize == 4 * kPointerSize); |
1533 a.Allocate(Heap::kStrictArgumentsObjectSize); | 1537 a.Allocate(Heap::kStrictArgumentsObjectSize); |
1534 a.Store(AccessBuilder::ForMap(), load_arguments_map); | 1538 a.Store(AccessBuilder::ForMap(), load_arguments_map); |
1535 a.Store(AccessBuilder::ForJSObjectProperties(), properties); | 1539 a.Store(AccessBuilder::ForJSObjectProperties(), properties); |
1536 a.Store(AccessBuilder::ForJSObjectElements(), elements); | 1540 a.Store(AccessBuilder::ForJSObjectElements(), elements); |
1537 a.Store(AccessBuilder::ForArgumentsLength(), jsgraph()->Constant(length)); | 1541 a.Store(AccessBuilder::ForArgumentsLength(), jsgraph()->Constant(length)); |
1538 RelaxControls(node); | 1542 RelaxControls(node); |
1539 a.FinishAndChange(node); | 1543 a.FinishAndChange(node); |
(...skipping 734 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
2274 | 2278 |
2275 Node* JSTypedLowering::Word32Shl(Node* const lhs, int32_t const rhs) { | 2279 Node* JSTypedLowering::Word32Shl(Node* const lhs, int32_t const rhs) { |
2276 if (rhs == 0) return lhs; | 2280 if (rhs == 0) return lhs; |
2277 return graph()->NewNode(machine()->Word32Shl(), lhs, | 2281 return graph()->NewNode(machine()->Word32Shl(), lhs, |
2278 jsgraph()->Int32Constant(rhs)); | 2282 jsgraph()->Int32Constant(rhs)); |
2279 } | 2283 } |
2280 | 2284 |
2281 | 2285 |
2282 // Helper that allocates a FixedArray holding argument values recorded in the | 2286 // Helper that allocates a FixedArray holding argument values recorded in the |
2283 // given {frame_state}. Serves as backing store for JSCreateArguments nodes. | 2287 // given {frame_state}. Serves as backing store for JSCreateArguments nodes. |
2284 Node* JSTypedLowering::AllocateArguments(Node* effect, Node* control, | 2288 Node* JSTypedLowering::AllocateArguments(Node** effect, Node* control, |
2285 Node* frame_state) { | 2289 Node* frame_state) { |
2286 FrameStateInfo state_info = OpParameter<FrameStateInfo>(frame_state); | 2290 FrameStateInfo state_info = OpParameter<FrameStateInfo>(frame_state); |
2287 int argument_count = state_info.parameter_count() - 1; // Minus receiver. | 2291 int argument_count = state_info.parameter_count() - 1; // Minus receiver. |
2288 if (argument_count == 0) return jsgraph()->EmptyFixedArrayConstant(); | 2292 if (argument_count == 0) return jsgraph()->EmptyFixedArrayConstant(); |
2289 | 2293 |
2290 // Prepare an iterator over argument values recorded in the frame state. | 2294 // Prepare an iterator over argument values recorded in the frame state. |
2291 Node* const parameters = frame_state->InputAt(kFrameStateParametersInput); | 2295 Node* const parameters = frame_state->InputAt(kFrameStateParametersInput); |
2292 StateValuesAccess parameters_access(parameters); | 2296 StateValuesAccess parameters_access(parameters); |
2293 auto paratemers_it = ++parameters_access.begin(); | 2297 auto paratemers_it = ++parameters_access.begin(); |
2294 | 2298 |
2295 // Actually allocate the backing store. | 2299 // Actually allocate the backing store. |
2296 AllocationBuilder a(jsgraph(), effect, control); | 2300 AllocationBuilder a(jsgraph(), *effect, control); |
2297 a.AllocateArray(argument_count, factory()->fixed_array_map()); | 2301 a.AllocateArray(argument_count, factory()->fixed_array_map()); |
2298 for (int i = 0; i < argument_count; ++i, ++paratemers_it) { | 2302 for (int i = 0; i < argument_count; ++i, ++paratemers_it) { |
2299 a.Store(AccessBuilder::ForFixedArraySlot(i), (*paratemers_it).node); | 2303 a.Store(AccessBuilder::ForFixedArraySlot(i), (*paratemers_it).node); |
2300 } | 2304 } |
2301 return a.Finish(); | 2305 return * effect = a.Finish(); |
2302 } | 2306 } |
2303 | 2307 |
2304 | 2308 |
2305 // Helper that allocates a FixedArray serving as a parameter map for values | 2309 // Helper that allocates a FixedArray serving as a parameter map for values |
2306 // recorded in the given {frame_state}. Some elements map to slots within the | 2310 // recorded in the given {frame_state}. Some elements map to slots within the |
2307 // given {context}. Serves as backing store for JSCreateArguments nodes. | 2311 // given {context}. Serves as backing store for JSCreateArguments nodes. |
2308 Node* JSTypedLowering::AllocateAliasedArguments( | 2312 Node* JSTypedLowering::AllocateAliasedArguments( |
2309 Node* effect, Node* control, Node* frame_state, Node* context, | 2313 Node** effect, Node* control, Node* frame_state, Node* context, |
2310 Handle<SharedFunctionInfo> shared, bool* has_aliased_arguments) { | 2314 Handle<SharedFunctionInfo> shared, bool* has_aliased_arguments) { |
2311 FrameStateInfo state_info = OpParameter<FrameStateInfo>(frame_state); | 2315 FrameStateInfo state_info = OpParameter<FrameStateInfo>(frame_state); |
2312 int argument_count = state_info.parameter_count() - 1; // Minus receiver. | 2316 int argument_count = state_info.parameter_count() - 1; // Minus receiver. |
2313 if (argument_count == 0) return jsgraph()->EmptyFixedArrayConstant(); | 2317 if (argument_count == 0) return jsgraph()->EmptyFixedArrayConstant(); |
2314 | 2318 |
2315 // If there is no aliasing, the arguments object elements are not special in | 2319 // If there is no aliasing, the arguments object elements are not special in |
2316 // any way, we can just return an unmapped backing store instead. | 2320 // any way, we can just return an unmapped backing store instead. |
2317 int parameter_count = shared->internal_formal_parameter_count(); | 2321 int parameter_count = shared->internal_formal_parameter_count(); |
2318 if (parameter_count == 0) { | 2322 if (parameter_count == 0) { |
2319 return AllocateArguments(effect, control, frame_state); | 2323 return AllocateArguments(effect, control, frame_state); |
2320 } | 2324 } |
2321 | 2325 |
2322 // Calculate number of argument values being aliased/mapped. | 2326 // Calculate number of argument values being aliased/mapped. |
2323 int mapped_count = Min(argument_count, parameter_count); | 2327 int mapped_count = Min(argument_count, parameter_count); |
2324 *has_aliased_arguments = true; | 2328 *has_aliased_arguments = true; |
2325 | 2329 |
2326 // Prepare an iterator over argument values recorded in the frame state. | 2330 // Prepare an iterator over argument values recorded in the frame state. |
2327 Node* const parameters = frame_state->InputAt(kFrameStateParametersInput); | 2331 Node* const parameters = frame_state->InputAt(kFrameStateParametersInput); |
2328 StateValuesAccess parameters_access(parameters); | 2332 StateValuesAccess parameters_access(parameters); |
2329 auto paratemers_it = ++parameters_access.begin(); | 2333 auto paratemers_it = ++parameters_access.begin(); |
2330 | 2334 |
2331 // The unmapped argument values recorded in the frame state are stored yet | 2335 // The unmapped argument values recorded in the frame state are stored yet |
2332 // another indirection away and then linked into the parameter map below, | 2336 // another indirection away and then linked into the parameter map below, |
2333 // whereas mapped argument values are replaced with a hole instead. | 2337 // whereas mapped argument values are replaced with a hole instead. |
2334 AllocationBuilder aa(jsgraph(), effect, control); | 2338 AllocationBuilder aa(jsgraph(), *effect, control); |
2335 aa.AllocateArray(argument_count, factory()->fixed_array_map()); | 2339 aa.AllocateArray(argument_count, factory()->fixed_array_map()); |
2336 for (int i = 0; i < mapped_count; ++i, ++paratemers_it) { | 2340 for (int i = 0; i < mapped_count; ++i, ++paratemers_it) { |
2337 aa.Store(AccessBuilder::ForFixedArraySlot(i), jsgraph()->TheHoleConstant()); | 2341 aa.Store(AccessBuilder::ForFixedArraySlot(i), jsgraph()->TheHoleConstant()); |
2338 } | 2342 } |
2339 for (int i = mapped_count; i < argument_count; ++i, ++paratemers_it) { | 2343 for (int i = mapped_count; i < argument_count; ++i, ++paratemers_it) { |
2340 aa.Store(AccessBuilder::ForFixedArraySlot(i), (*paratemers_it).node); | 2344 aa.Store(AccessBuilder::ForFixedArraySlot(i), (*paratemers_it).node); |
2341 } | 2345 } |
2342 Node* arguments = aa.Finish(); | 2346 Node* arguments = aa.Finish(); |
2343 | 2347 |
2344 // Actually allocate the backing store. | 2348 // Actually allocate the backing store. |
2345 AllocationBuilder a(jsgraph(), effect, control); | 2349 AllocationBuilder a(jsgraph(), arguments, control); |
2346 a.AllocateArray(mapped_count + 2, factory()->sloppy_arguments_elements_map()); | 2350 a.AllocateArray(mapped_count + 2, factory()->sloppy_arguments_elements_map()); |
2347 a.Store(AccessBuilder::ForFixedArraySlot(0), context); | 2351 a.Store(AccessBuilder::ForFixedArraySlot(0), context); |
2348 a.Store(AccessBuilder::ForFixedArraySlot(1), arguments); | 2352 a.Store(AccessBuilder::ForFixedArraySlot(1), arguments); |
2349 for (int i = 0; i < mapped_count; ++i) { | 2353 for (int i = 0; i < mapped_count; ++i) { |
2350 int idx = Context::MIN_CONTEXT_SLOTS + parameter_count - 1 - i; | 2354 int idx = Context::MIN_CONTEXT_SLOTS + parameter_count - 1 - i; |
2351 a.Store(AccessBuilder::ForFixedArraySlot(i + 2), jsgraph()->Constant(idx)); | 2355 a.Store(AccessBuilder::ForFixedArraySlot(i + 2), jsgraph()->Constant(idx)); |
2352 } | 2356 } |
2353 return a.Finish(); | 2357 return * effect = a.Finish(); |
2354 } | 2358 } |
2355 | 2359 |
2356 | 2360 |
2357 Factory* JSTypedLowering::factory() const { return jsgraph()->factory(); } | 2361 Factory* JSTypedLowering::factory() const { return jsgraph()->factory(); } |
2358 | 2362 |
2359 | 2363 |
2360 Graph* JSTypedLowering::graph() const { return jsgraph()->graph(); } | 2364 Graph* JSTypedLowering::graph() const { return jsgraph()->graph(); } |
2361 | 2365 |
2362 | 2366 |
2363 Isolate* JSTypedLowering::isolate() const { return jsgraph()->isolate(); } | 2367 Isolate* JSTypedLowering::isolate() const { return jsgraph()->isolate(); } |
(...skipping 19 matching lines...) Expand all Loading... | |
2383 } | 2387 } |
2384 | 2388 |
2385 | 2389 |
2386 CompilationDependencies* JSTypedLowering::dependencies() const { | 2390 CompilationDependencies* JSTypedLowering::dependencies() const { |
2387 return dependencies_; | 2391 return dependencies_; |
2388 } | 2392 } |
2389 | 2393 |
2390 } // namespace compiler | 2394 } // namespace compiler |
2391 } // namespace internal | 2395 } // namespace internal |
2392 } // namespace v8 | 2396 } // namespace v8 |
OLD | NEW |