Index: test/unittests/compiler/js-type-feedback-unittest.cc |
diff --git a/test/unittests/compiler/js-type-feedback-unittest.cc b/test/unittests/compiler/js-type-feedback-unittest.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..08fe68a8baa037141046ab748394342c131c1540 |
--- /dev/null |
+++ b/test/unittests/compiler/js-type-feedback-unittest.cc |
@@ -0,0 +1,277 @@ |
+// Copyright 2015 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.h" |
+ |
+#include "src/compiler/access-builder.h" |
+#include "src/compiler/js-graph.h" |
+#include "src/compiler/js-operator.h" |
+#include "src/compiler/js-type-feedback.h" |
+#include "src/compiler/machine-operator.h" |
+#include "src/compiler/node-matchers.h" |
+#include "src/compiler/node-properties.h" |
+#include "src/compiler/operator-properties.h" |
+ |
+#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::Capture; |
+ |
+ |
+namespace v8 { |
+namespace internal { |
+namespace compiler { |
+ |
+class JSTypeFeedbackTest : public TypedGraphTest { |
+ public: |
+ JSTypeFeedbackTest() |
+ : TypedGraphTest(3), |
+ javascript_(zone()), |
+ dependencies_(isolate(), zone()) {} |
+ ~JSTypeFeedbackTest() override { dependencies_.Rollback(); } |
+ |
+ protected: |
+ Reduction Reduce(Node* node) { |
+ Handle<GlobalObject> global_object( |
+ isolate()->native_context()->global_object(), isolate()); |
+ |
+ MachineOperatorBuilder machine(zone()); |
+ JSGraph jsgraph(isolate(), graph(), common(), javascript(), &machine); |
+ JSTypeFeedbackTable table(zone()); |
+ JSTypeFeedbackSpecializer reducer(&jsgraph, &table, nullptr, global_object, |
+ &dependencies_); |
+ return reducer.Reduce(node); |
+ } |
+ |
+ Node* EmptyFrameState() { |
+ MachineOperatorBuilder machine(zone()); |
+ JSGraph jsgraph(isolate(), graph(), common(), javascript(), &machine); |
+ return jsgraph.EmptyFrameState(); |
+ } |
+ |
+ JSOperatorBuilder* javascript() { return &javascript_; } |
+ |
+ void SetGlobalProperty(const char* string, int value) { |
+ SetGlobalProperty(string, Handle<Smi>(Smi::FromInt(value), isolate())); |
+ } |
+ |
+ void SetGlobalProperty(const char* string, double value) { |
+ SetGlobalProperty(string, isolate()->factory()->NewNumber(value)); |
+ } |
+ |
+ void SetGlobalProperty(const char* string, Handle<Object> value) { |
+ Handle<JSObject> global(isolate()->context()->global_object(), isolate()); |
+ Handle<String> name = |
+ isolate()->factory()->NewStringFromAsciiChecked(string); |
+ MaybeHandle<Object> result = |
+ JSReceiver::SetProperty(global, name, value, SLOPPY); |
+ result.Assert(); |
+ } |
+ |
+ Node* ReturnLoadNamedFromGlobal(const char* string, Node* effect, |
+ Node* control) { |
+ VectorSlotPair feedback(Handle<TypeFeedbackVector>::null(), |
+ FeedbackVectorICSlot::Invalid()); |
+ Node* global = Parameter(Type::GlobalObject()); |
+ Node* context = UndefinedConstant(); |
+ |
+ Unique<Name> name = Unique<Name>::CreateUninitialized( |
+ isolate()->factory()->NewStringFromAsciiChecked(string)); |
+ Node* load = graph()->NewNode(javascript()->LoadNamed(name, feedback), |
+ global, context); |
+ if (FLAG_turbo_deoptimization) { |
+ load->AppendInput(zone(), EmptyFrameState()); |
+ } |
+ load->AppendInput(zone(), effect); |
+ load->AppendInput(zone(), control); |
+ Node* if_success = graph()->NewNode(common()->IfSuccess(), load); |
+ return graph()->NewNode(common()->Return(), load, load, if_success); |
+ } |
+ |
+ CompilationDependencies* dependencies() { return &dependencies_; } |
+ |
+ private: |
+ JSOperatorBuilder javascript_; |
+ CompilationDependencies dependencies_; |
+}; |
+ |
+#define WITH_AND_WITHOUT_TURBO_DEOPTIMIZATION \ |
+ for (int i = FLAG_turbo_deoptimization = 0; i < 2; \ |
+ FLAG_turbo_deoptimization = ++i) |
+ |
+ |
+TEST_F(JSTypeFeedbackTest, JSLoadNamedGlobalConst_smi) { |
+ const int const_value = 111; |
+ const char* property_name = "banana"; |
+ SetGlobalProperty(property_name, const_value); |
+ |
+ WITH_AND_WITHOUT_TURBO_DEOPTIMIZATION { |
+ Node* ret = ReturnLoadNamedFromGlobal(property_name, graph()->start(), |
+ graph()->start()); |
+ graph()->SetEnd(graph()->NewNode(common()->End(), ret)); |
+ |
+ Reduction r = Reduce(ret->InputAt(0)); |
+ |
+ if (FLAG_turbo_deoptimization) { |
+ // Check LoadNamed(global) => HeapConstant[const_value] |
+ ASSERT_TRUE(r.Changed()); |
+ EXPECT_THAT(r.replacement(), IsNumberConstant(const_value)); |
+ |
+ EXPECT_THAT(ret, IsReturn(IsNumberConstant(const_value), graph()->start(), |
+ graph()->start())); |
+ EXPECT_THAT(graph()->end(), IsEnd(ret)); |
+ |
+ EXPECT_FALSE(dependencies()->IsEmpty()); |
+ dependencies()->Rollback(); |
+ } else { |
+ ASSERT_FALSE(r.Changed()); |
+ EXPECT_TRUE(dependencies()->IsEmpty()); |
+ } |
+ } |
+} |
+ |
+ |
+TEST_F(JSTypeFeedbackTest, JSLoadNamedGlobalConst_derble) { |
+ const double const_value = -11.25; |
+ const char* property_name = "kiwi"; |
+ SetGlobalProperty(property_name, const_value); |
+ |
+ WITH_AND_WITHOUT_TURBO_DEOPTIMIZATION { |
+ Node* ret = ReturnLoadNamedFromGlobal(property_name, graph()->start(), |
+ graph()->start()); |
+ graph()->SetEnd(graph()->NewNode(common()->End(), ret)); |
+ |
+ Reduction r = Reduce(ret->InputAt(0)); |
+ |
+ if (FLAG_turbo_deoptimization) { |
+ // Check LoadNamed(global) => HeapConstant[const_value] |
+ ASSERT_TRUE(r.Changed()); |
+ EXPECT_THAT(r.replacement(), IsNumberConstant(const_value)); |
+ |
+ EXPECT_THAT(ret, IsReturn(IsNumberConstant(const_value), graph()->start(), |
+ graph()->start())); |
+ EXPECT_THAT(graph()->end(), IsEnd(ret)); |
+ |
+ EXPECT_FALSE(dependencies()->IsEmpty()); |
+ } else { |
+ ASSERT_FALSE(r.Changed()); |
+ EXPECT_TRUE(dependencies()->IsEmpty()); |
+ } |
+ } |
+} |
+ |
+ |
+TEST_F(JSTypeFeedbackTest, JSLoadNamedGlobalConst_string) { |
+ Unique<HeapObject> const_value = Unique<HeapObject>::CreateImmovable( |
+ isolate()->factory()->undefined_string()); |
+ const char* property_name = "mango"; |
+ SetGlobalProperty(property_name, const_value.handle()); |
+ |
+ WITH_AND_WITHOUT_TURBO_DEOPTIMIZATION { |
+ Node* ret = ReturnLoadNamedFromGlobal(property_name, graph()->start(), |
+ graph()->start()); |
+ graph()->SetEnd(graph()->NewNode(common()->End(), ret)); |
+ |
+ Reduction r = Reduce(ret->InputAt(0)); |
+ |
+ if (FLAG_turbo_deoptimization) { |
+ // Check LoadNamed(global) => HeapConstant[const_value] |
+ ASSERT_TRUE(r.Changed()); |
+ EXPECT_THAT(r.replacement(), IsHeapConstant(const_value)); |
+ |
+ EXPECT_THAT(ret, IsReturn(IsHeapConstant(const_value), graph()->start(), |
+ graph()->start())); |
+ EXPECT_THAT(graph()->end(), IsEnd(ret)); |
+ |
+ EXPECT_FALSE(dependencies()->IsEmpty()); |
+ dependencies()->Rollback(); |
+ } else { |
+ ASSERT_FALSE(r.Changed()); |
+ EXPECT_TRUE(dependencies()->IsEmpty()); |
+ } |
+ } |
+} |
+ |
+ |
+TEST_F(JSTypeFeedbackTest, JSLoadNamedGlobalPropertyCell_smi) { |
+ const char* property_name = "melon"; |
+ SetGlobalProperty(property_name, 123); |
+ SetGlobalProperty(property_name, 124); |
+ |
+ WITH_AND_WITHOUT_TURBO_DEOPTIMIZATION { |
+ Node* ret = ReturnLoadNamedFromGlobal(property_name, graph()->start(), |
+ graph()->start()); |
+ graph()->SetEnd(graph()->NewNode(common()->End(), ret)); |
+ |
+ Reduction r = Reduce(ret->InputAt(0)); |
+ |
+ if (FLAG_turbo_deoptimization) { |
+ // Check LoadNamed(global) => LoadField[PropertyCell::value](cell) |
+ ASSERT_TRUE(r.Changed()); |
+ FieldAccess access = AccessBuilder::ForPropertyCellValue(); |
+ Capture<Node*> cell_capture; |
+ Matcher<Node*> load_field_match = IsLoadField( |
+ access, CaptureEq(&cell_capture), graph()->start(), graph()->start()); |
+ EXPECT_THAT(r.replacement(), load_field_match); |
+ |
+ HeapObjectMatcher<PropertyCell> cell(cell_capture.value()); |
+ EXPECT_TRUE(cell.HasValue()); |
+ EXPECT_TRUE(cell.Value().handle()->IsPropertyCell()); |
+ |
+ EXPECT_THAT( |
+ ret, IsReturn(load_field_match, load_field_match, graph()->start())); |
+ EXPECT_THAT(graph()->end(), IsEnd(ret)); |
+ |
+ EXPECT_FALSE(dependencies()->IsEmpty()); |
+ dependencies()->Rollback(); |
+ } else { |
+ ASSERT_FALSE(r.Changed()); |
+ EXPECT_TRUE(dependencies()->IsEmpty()); |
+ } |
+ } |
+} |
+ |
+ |
+TEST_F(JSTypeFeedbackTest, JSLoadNamedGlobalPropertyCell_string) { |
+ const char* property_name = "pineapple"; |
+ SetGlobalProperty(property_name, isolate()->factory()->undefined_string()); |
+ SetGlobalProperty(property_name, isolate()->factory()->undefined_value()); |
+ |
+ WITH_AND_WITHOUT_TURBO_DEOPTIMIZATION { |
+ Node* ret = ReturnLoadNamedFromGlobal(property_name, graph()->start(), |
+ graph()->start()); |
+ graph()->SetEnd(graph()->NewNode(common()->End(), ret)); |
+ |
+ Reduction r = Reduce(ret->InputAt(0)); |
+ |
+ if (FLAG_turbo_deoptimization) { |
+ // Check LoadNamed(global) => LoadField[PropertyCell::value](cell) |
+ ASSERT_TRUE(r.Changed()); |
+ FieldAccess access = AccessBuilder::ForPropertyCellValue(); |
+ Capture<Node*> cell_capture; |
+ Matcher<Node*> load_field_match = IsLoadField( |
+ access, CaptureEq(&cell_capture), graph()->start(), graph()->start()); |
+ EXPECT_THAT(r.replacement(), load_field_match); |
+ |
+ HeapObjectMatcher<PropertyCell> cell(cell_capture.value()); |
+ EXPECT_TRUE(cell.HasValue()); |
+ EXPECT_TRUE(cell.Value().handle()->IsPropertyCell()); |
+ |
+ EXPECT_THAT( |
+ ret, IsReturn(load_field_match, load_field_match, graph()->start())); |
+ EXPECT_THAT(graph()->end(), IsEnd(ret)); |
+ |
+ EXPECT_FALSE(dependencies()->IsEmpty()); |
+ dependencies()->Rollback(); |
+ } else { |
+ ASSERT_FALSE(r.Changed()); |
+ EXPECT_TRUE(dependencies()->IsEmpty()); |
+ } |
+ } |
+} |
+} |
+} |
+} |