2016-02-08 12:33:44 +00:00
|
|
|
// Copyright 2016 the V8 project authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
|
|
|
#include "src/compiler/js-create-lowering.h"
|
2019-05-21 09:30:15 +00:00
|
|
|
#include "src/codegen/code-factory.h"
|
2019-07-16 21:46:08 +00:00
|
|
|
#include "src/codegen/tick-counter.h"
|
2016-02-08 12:33:44 +00:00
|
|
|
#include "src/compiler/access-builder.h"
|
2018-07-06 11:05:49 +00:00
|
|
|
#include "src/compiler/compilation-dependencies.h"
|
2016-02-08 12:33:44 +00:00
|
|
|
#include "src/compiler/js-graph.h"
|
|
|
|
#include "src/compiler/js-operator.h"
|
|
|
|
#include "src/compiler/machine-operator.h"
|
|
|
|
#include "src/compiler/node-properties.h"
|
|
|
|
#include "src/compiler/operator-properties.h"
|
2019-05-22 07:55:37 +00:00
|
|
|
#include "src/execution/isolate-inl.h"
|
2018-05-23 13:29:02 +00:00
|
|
|
#include "src/objects/arguments.h"
|
2019-05-23 08:51:46 +00:00
|
|
|
#include "src/objects/feedback-vector.h"
|
2016-02-08 12:33:44 +00:00
|
|
|
#include "test/unittests/compiler/compiler-test-utils.h"
|
|
|
|
#include "test/unittests/compiler/graph-unittest.h"
|
|
|
|
#include "test/unittests/compiler/node-test-utils.h"
|
|
|
|
#include "testing/gmock-support.h"
|
|
|
|
|
|
|
|
using testing::_;
|
|
|
|
using testing::BitEq;
|
|
|
|
using testing::IsNaN;
|
|
|
|
|
|
|
|
namespace v8 {
|
|
|
|
namespace internal {
|
|
|
|
namespace compiler {
|
|
|
|
|
|
|
|
class JSCreateLoweringTest : public TypedGraphTest {
|
|
|
|
public:
|
|
|
|
JSCreateLoweringTest()
|
2018-07-09 08:22:14 +00:00
|
|
|
: TypedGraphTest(3),
|
|
|
|
javascript_(zone()),
|
2019-02-27 18:10:29 +00:00
|
|
|
deps_(broker(), zone()),
|
|
|
|
handle_scope_(isolate()) {}
|
2018-09-17 11:30:48 +00:00
|
|
|
~JSCreateLoweringTest() override = default;
|
2016-02-08 12:33:44 +00:00
|
|
|
|
|
|
|
protected:
|
|
|
|
Reduction Reduce(Node* node) {
|
|
|
|
MachineOperatorBuilder machine(zone());
|
|
|
|
SimplifiedOperatorBuilder simplified(zone());
|
|
|
|
JSGraph jsgraph(isolate(), graph(), common(), javascript(), &simplified,
|
|
|
|
&machine);
|
|
|
|
// TODO(titzer): mock the GraphReducer here for better unit testing.
|
2019-07-16 21:46:08 +00:00
|
|
|
GraphReducer graph_reducer(zone(), graph(), tick_counter());
|
2018-10-12 09:18:57 +00:00
|
|
|
JSCreateLowering reducer(&graph_reducer, &deps_, &jsgraph, broker(),
|
2018-09-19 09:23:17 +00:00
|
|
|
zone());
|
2016-02-08 12:33:44 +00:00
|
|
|
return reducer.Reduce(node);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* FrameState(Handle<SharedFunctionInfo> shared, Node* outer_frame_state) {
|
2017-01-05 10:44:44 +00:00
|
|
|
Node* state_values =
|
|
|
|
graph()->NewNode(common()->StateValues(0, SparseInputMask::Dense()));
|
2016-02-08 12:33:44 +00:00
|
|
|
return graph()->NewNode(
|
[turbofan] Remove the JSContextRelaxation reducer.
This reducer doesn't really add value, because:
(a) it is only concerned with JSCallFunction and JSToNumber, but when
we get to it, all JSCallFunction nodes will have been replaced by
Call nodes, and in the not so far future, we will also have
replaced almost all JSToNumber nodes with better code,
(b) and the reducer tries to be smart and use one of the outermost
contexts, but that might not be beneficial always; actually it
might even create longer live ranges and lead to more spilling
in some cases.
But most importantly, the JSContextRelaxation currently blocks inlining
based on SharedFunctionInfo, because it requires the inliner to check
the native context, which in turn requires JSFunction knowledge. So I'm
removing this reducer for now to unblock the more important inliner
changes.
R=jarin@chromium.org
Review URL: https://codereview.chromium.org/1715633002
Cr-Commit-Position: refs/heads/master@{#34139}
2016-02-19 07:55:26 +00:00
|
|
|
common()->FrameState(
|
|
|
|
BailoutId::None(), OutputFrameStateCombine::Ignore(),
|
|
|
|
common()->CreateFrameStateFunctionInfo(
|
2017-06-14 11:21:43 +00:00
|
|
|
FrameStateType::kInterpretedFunction, 1, 0, shared)),
|
2016-02-08 12:33:44 +00:00
|
|
|
state_values, state_values, state_values, NumberConstant(0),
|
|
|
|
UndefinedConstant(), outer_frame_state);
|
|
|
|
}
|
|
|
|
|
|
|
|
JSOperatorBuilder* javascript() { return &javascript_; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
JSOperatorBuilder javascript_;
|
|
|
|
CompilationDependencies deps_;
|
2018-07-09 08:22:14 +00:00
|
|
|
CanonicalHandleScope handle_scope_;
|
2016-02-08 12:33:44 +00:00
|
|
|
};
|
|
|
|
|
2017-02-07 09:00:18 +00:00
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
// JSCreate
|
|
|
|
|
2016-02-08 12:33:44 +00:00
|
|
|
TEST_F(JSCreateLoweringTest, JSCreate) {
|
|
|
|
Handle<JSFunction> function = isolate()->object_function();
|
2019-03-18 14:34:05 +00:00
|
|
|
Node* const target = graph()->NewNode(common()->HeapConstant(function));
|
2016-02-08 12:33:44 +00:00
|
|
|
Node* const context = Parameter(Type::Any());
|
|
|
|
Node* const effect = graph()->start();
|
2017-02-07 09:00:18 +00:00
|
|
|
Node* const control = graph()->start();
|
|
|
|
Reduction r =
|
|
|
|
Reduce(graph()->NewNode(javascript()->Create(), target, target, context,
|
|
|
|
EmptyFrameState(), effect, control));
|
2016-02-08 12:33:44 +00:00
|
|
|
ASSERT_TRUE(r.Changed());
|
|
|
|
EXPECT_THAT(
|
|
|
|
r.replacement(),
|
|
|
|
IsFinishRegion(
|
|
|
|
IsAllocate(IsNumberConstant(function->initial_map().instance_size()),
|
2017-02-07 09:00:18 +00:00
|
|
|
IsBeginRegion(effect), control),
|
2016-02-08 12:33:44 +00:00
|
|
|
_));
|
|
|
|
}
|
|
|
|
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
// JSCreateArguments
|
|
|
|
|
|
|
|
TEST_F(JSCreateLoweringTest, JSCreateArgumentsInlinedMapped) {
|
|
|
|
Node* const closure = Parameter(Type::Any());
|
|
|
|
Node* const context = UndefinedConstant();
|
|
|
|
Node* const effect = graph()->start();
|
2018-07-05 20:34:14 +00:00
|
|
|
Handle<SharedFunctionInfo> shared(isolate()->regexp_function()->shared(),
|
2018-06-26 10:14:12 +00:00
|
|
|
isolate());
|
2016-02-08 12:33:44 +00:00
|
|
|
Node* const frame_state_outer = FrameState(shared, graph()->start());
|
|
|
|
Node* const frame_state_inner = FrameState(shared, frame_state_outer);
|
|
|
|
Reduction r = Reduce(graph()->NewNode(
|
|
|
|
javascript()->CreateArguments(CreateArgumentsType::kMappedArguments),
|
2016-05-11 06:10:46 +00:00
|
|
|
closure, context, frame_state_inner, effect));
|
2016-02-08 12:33:44 +00:00
|
|
|
ASSERT_TRUE(r.Changed());
|
2016-05-11 06:10:46 +00:00
|
|
|
EXPECT_THAT(
|
|
|
|
r.replacement(),
|
|
|
|
IsFinishRegion(
|
|
|
|
IsAllocate(IsNumberConstant(JSSloppyArgumentsObject::kSize), _, _),
|
|
|
|
_));
|
2016-02-08 12:33:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(JSCreateLoweringTest, JSCreateArgumentsInlinedUnmapped) {
|
|
|
|
Node* const closure = Parameter(Type::Any());
|
|
|
|
Node* const context = UndefinedConstant();
|
|
|
|
Node* const effect = graph()->start();
|
2018-07-05 20:34:14 +00:00
|
|
|
Handle<SharedFunctionInfo> shared(isolate()->regexp_function()->shared(),
|
2018-06-26 10:14:12 +00:00
|
|
|
isolate());
|
2016-02-08 12:33:44 +00:00
|
|
|
Node* const frame_state_outer = FrameState(shared, graph()->start());
|
|
|
|
Node* const frame_state_inner = FrameState(shared, frame_state_outer);
|
|
|
|
Reduction r = Reduce(graph()->NewNode(
|
|
|
|
javascript()->CreateArguments(CreateArgumentsType::kUnmappedArguments),
|
2016-05-11 06:10:46 +00:00
|
|
|
closure, context, frame_state_inner, effect));
|
2016-02-08 12:33:44 +00:00
|
|
|
ASSERT_TRUE(r.Changed());
|
2016-05-11 06:10:46 +00:00
|
|
|
EXPECT_THAT(
|
|
|
|
r.replacement(),
|
|
|
|
IsFinishRegion(
|
|
|
|
IsAllocate(IsNumberConstant(JSStrictArgumentsObject::kSize), _, _),
|
|
|
|
_));
|
2016-02-08 12:33:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(JSCreateLoweringTest, JSCreateArgumentsInlinedRestArray) {
|
|
|
|
Node* const closure = Parameter(Type::Any());
|
|
|
|
Node* const context = UndefinedConstant();
|
|
|
|
Node* const effect = graph()->start();
|
2018-07-05 20:34:14 +00:00
|
|
|
Handle<SharedFunctionInfo> shared(isolate()->regexp_function()->shared(),
|
2018-06-26 10:14:12 +00:00
|
|
|
isolate());
|
2016-02-08 12:33:44 +00:00
|
|
|
Node* const frame_state_outer = FrameState(shared, graph()->start());
|
|
|
|
Node* const frame_state_inner = FrameState(shared, frame_state_outer);
|
|
|
|
Reduction r = Reduce(graph()->NewNode(
|
|
|
|
javascript()->CreateArguments(CreateArgumentsType::kRestParameter),
|
2016-05-11 06:10:46 +00:00
|
|
|
closure, context, frame_state_inner, effect));
|
2016-02-08 12:33:44 +00:00
|
|
|
ASSERT_TRUE(r.Changed());
|
2019-11-15 20:37:49 +00:00
|
|
|
EXPECT_THAT(r.replacement(),
|
|
|
|
IsFinishRegion(
|
|
|
|
IsAllocate(IsNumberConstant(JSArray::kHeaderSize), _, _), _));
|
2016-02-08 12:33:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
// JSCreateFunctionContext
|
|
|
|
|
|
|
|
TEST_F(JSCreateLoweringTest, JSCreateFunctionContextViaInlinedAllocation) {
|
|
|
|
Node* const context = Parameter(Type::Any());
|
|
|
|
Node* const effect = graph()->start();
|
|
|
|
Node* const control = graph()->start();
|
2018-05-02 13:57:26 +00:00
|
|
|
Reduction const r = Reduce(graph()->NewNode(
|
2018-06-26 10:14:12 +00:00
|
|
|
javascript()->CreateFunctionContext(
|
|
|
|
handle(ScopeInfo::Empty(isolate()), isolate()), 8, FUNCTION_SCOPE),
|
2018-05-02 13:57:26 +00:00
|
|
|
context, effect, control));
|
2016-02-08 12:33:44 +00:00
|
|
|
ASSERT_TRUE(r.Changed());
|
|
|
|
EXPECT_THAT(r.replacement(),
|
|
|
|
IsFinishRegion(IsAllocate(IsNumberConstant(Context::SizeFor(
|
|
|
|
8 + Context::MIN_CONTEXT_SLOTS)),
|
|
|
|
IsBeginRegion(_), control),
|
|
|
|
_));
|
|
|
|
}
|
|
|
|
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
// JSCreateWithContext
|
|
|
|
|
|
|
|
TEST_F(JSCreateLoweringTest, JSCreateWithContext) {
|
2019-08-12 21:07:13 +00:00
|
|
|
Handle<ScopeInfo> scope_info =
|
|
|
|
ReadOnlyRoots(isolate()).empty_function_scope_info_handle();
|
2016-02-08 12:33:44 +00:00
|
|
|
Node* const object = Parameter(Type::Receiver());
|
|
|
|
Node* const context = Parameter(Type::Any());
|
|
|
|
Node* const effect = graph()->start();
|
|
|
|
Node* const control = graph()->start();
|
|
|
|
Reduction r =
|
2016-09-06 08:27:36 +00:00
|
|
|
Reduce(graph()->NewNode(javascript()->CreateWithContext(scope_info),
|
2018-05-02 13:57:26 +00:00
|
|
|
object, context, effect, control));
|
2016-02-08 12:33:44 +00:00
|
|
|
ASSERT_TRUE(r.Changed());
|
Reland x3 "[runtime] Remove extension slots from context objects"
Original change's description:
> [runtime] Remove extension slots from context objects
>
> Context objects have an extension slot, which contains further
> additional data that depends on the type of the context.
>
> This CL removes the extension slot from contexts that don't need
> them, hence reducing memory.
>
> The following contexts will still have an extension slot: native,
> module, await, block and with contexts. See objects/contexts.h for
> what the slot is used for.
> The following contexts will not have an extension slot anymore (they
> were not used before): script, catch and builtin contexts.
> Eval and function contexts only have the extension slot if they
> contain a sloppy eval.
>
> Bug: v8:9744
> Change-Id: I8ca56c22fa02437bbac392ea72174ebfca80e030
> Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/1863191
> Commit-Queue: Victor Gomes <victorgomes@google.com>
> Reviewed-by: Toon Verwaest <verwaest@chromium.org>
> Reviewed-by: Jakob Gruber <jgruber@chromium.org>
> Reviewed-by: Ulan Degenbaev <ulan@chromium.org>
> Reviewed-by: Leszek Swirski <leszeks@chromium.org>
> Reviewed-by: Peter Marshall <petermarshall@chromium.org>
> Auto-Submit: Victor Gomes <victorgomes@google.com>
> Cr-Commit-Position: refs/heads/master@{#64372}
TBR=verwaest@chromium.org,jgruber@chromium.org,ulan@chromium.org,leszeks@chromium.org,petermarshall@chromium.org
Bug: v8:9744
Change-Id: I8700ed2fa62c89e86c39bb16ac3167f38ea8d63f
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/1873695
Commit-Queue: Victor Gomes <victorgomes@chromium.org>
Reviewed-by: Leszek Swirski <leszeks@chromium.org>
Reviewed-by: Ulan Degenbaev <ulan@chromium.org>
Reviewed-by: Peter Marshall <petermarshall@chromium.org>
Cr-Commit-Position: refs/heads/master@{#64477}
2019-10-22 12:59:24 +00:00
|
|
|
EXPECT_THAT(
|
|
|
|
r.replacement(),
|
|
|
|
IsFinishRegion(IsAllocate(IsNumberConstant(Context::SizeFor(
|
|
|
|
Context::MIN_CONTEXT_EXTENDED_SLOTS)),
|
|
|
|
IsBeginRegion(_), control),
|
|
|
|
_));
|
2016-02-08 12:33:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
// JSCreateCatchContext
|
|
|
|
|
|
|
|
TEST_F(JSCreateLoweringTest, JSCreateCatchContext) {
|
2019-08-12 21:07:13 +00:00
|
|
|
Handle<ScopeInfo> scope_info =
|
|
|
|
ReadOnlyRoots(isolate()).empty_function_scope_info_handle();
|
2016-02-08 12:33:44 +00:00
|
|
|
Node* const exception = Parameter(Type::Receiver());
|
|
|
|
Node* const context = Parameter(Type::Any());
|
|
|
|
Node* const effect = graph()->start();
|
|
|
|
Node* const control = graph()->start();
|
2018-05-04 09:29:20 +00:00
|
|
|
Reduction r =
|
|
|
|
Reduce(graph()->NewNode(javascript()->CreateCatchContext(scope_info),
|
|
|
|
exception, context, effect, control));
|
2016-02-08 12:33:44 +00:00
|
|
|
ASSERT_TRUE(r.Changed());
|
|
|
|
EXPECT_THAT(r.replacement(),
|
|
|
|
IsFinishRegion(IsAllocate(IsNumberConstant(Context::SizeFor(
|
|
|
|
Context::MIN_CONTEXT_SLOTS + 1)),
|
|
|
|
IsBeginRegion(_), control),
|
|
|
|
_));
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace compiler
|
|
|
|
} // namespace internal
|
|
|
|
} // namespace v8
|