Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(216)

Side by Side Diff: src/compiler/js-typed-lowering.cc

Issue 1447323005: Fix argument allocation dangling effect chains (Closed) Base URL: https://chromium.googlesource.com/v8/v8.git@master
Patch Set: Account for zero arguments Created 5 years, 1 month ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
« no previous file with comments | « src/compiler/js-typed-lowering.h ('k') | no next file » | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
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
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
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
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
OLDNEW
« no previous file with comments | « src/compiler/js-typed-lowering.h ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698