4ec63ff97e
- "Switch inlining to use simplified instead of machine loads." - "Add copy support in inliner." Reland fixes: - size_t conversion for 64bit arches - Don't call front() on empty vector (triggers assertion on windows) - turbo_inlining now implies turbo_types, as it requires simplified lowering. R=mstarzinger@chromium.org Review URL: https://codereview.chromium.org/559843004 git-svn-id: https://v8.googlecode.com/svn/branches/bleeding_edge@23911 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
292 lines
8.0 KiB
C++
292 lines
8.0 KiB
C++
// Copyright 2014 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/v8.h"
|
|
|
|
#include "src/compiler/js-graph.h"
|
|
#include "src/compiler/node-properties-inl.h"
|
|
#include "src/compiler/typer.h"
|
|
#include "src/types.h"
|
|
#include "test/cctest/cctest.h"
|
|
#include "test/cctest/compiler/value-helper.h"
|
|
|
|
using namespace v8::internal;
|
|
using namespace v8::internal::compiler;
|
|
|
|
class JSCacheTesterHelper {
|
|
protected:
|
|
explicit JSCacheTesterHelper(Zone* zone)
|
|
: main_graph_(zone),
|
|
main_common_(zone),
|
|
main_javascript_(zone),
|
|
main_typer_(zone),
|
|
main_machine_() {}
|
|
Graph main_graph_;
|
|
CommonOperatorBuilder main_common_;
|
|
JSOperatorBuilder main_javascript_;
|
|
Typer main_typer_;
|
|
MachineOperatorBuilder main_machine_;
|
|
};
|
|
|
|
|
|
class JSConstantCacheTester : public HandleAndZoneScope,
|
|
public JSCacheTesterHelper,
|
|
public JSGraph {
|
|
public:
|
|
JSConstantCacheTester()
|
|
: JSCacheTesterHelper(main_zone()),
|
|
JSGraph(&main_graph_, &main_common_, &main_javascript_, &main_typer_,
|
|
&main_machine_) {}
|
|
|
|
Type* upper(Node* node) { return NodeProperties::GetBounds(node).upper; }
|
|
|
|
Handle<Object> handle(Node* node) {
|
|
CHECK_EQ(IrOpcode::kHeapConstant, node->opcode());
|
|
return OpParameter<Unique<Object> >(node).handle();
|
|
}
|
|
|
|
Factory* factory() { return main_isolate()->factory(); }
|
|
};
|
|
|
|
|
|
TEST(ZeroConstant1) {
|
|
JSConstantCacheTester T;
|
|
|
|
Node* zero = T.ZeroConstant();
|
|
|
|
CHECK_EQ(IrOpcode::kNumberConstant, zero->opcode());
|
|
CHECK_EQ(zero, T.Constant(0));
|
|
CHECK_NE(zero, T.Constant(-0.0));
|
|
CHECK_NE(zero, T.Constant(1.0));
|
|
CHECK_NE(zero, T.Constant(v8::base::OS::nan_value()));
|
|
CHECK_NE(zero, T.Float64Constant(0));
|
|
CHECK_NE(zero, T.Int32Constant(0));
|
|
|
|
Type* t = T.upper(zero);
|
|
|
|
CHECK(t->Is(Type::Number()));
|
|
CHECK(t->Is(Type::Integral32()));
|
|
CHECK(t->Is(Type::Signed32()));
|
|
CHECK(t->Is(Type::Unsigned32()));
|
|
CHECK(t->Is(Type::SignedSmall()));
|
|
CHECK(t->Is(Type::UnsignedSmall()));
|
|
}
|
|
|
|
|
|
TEST(MinusZeroConstant) {
|
|
JSConstantCacheTester T;
|
|
|
|
Node* minus_zero = T.Constant(-0.0);
|
|
Node* zero = T.ZeroConstant();
|
|
|
|
CHECK_EQ(IrOpcode::kNumberConstant, minus_zero->opcode());
|
|
CHECK_EQ(minus_zero, T.Constant(-0.0));
|
|
CHECK_NE(zero, minus_zero);
|
|
|
|
Type* t = T.upper(minus_zero);
|
|
|
|
CHECK(t->Is(Type::Number()));
|
|
CHECK(t->Is(Type::MinusZero()));
|
|
CHECK(!t->Is(Type::Integral32()));
|
|
CHECK(!t->Is(Type::Signed32()));
|
|
CHECK(!t->Is(Type::Unsigned32()));
|
|
CHECK(!t->Is(Type::SignedSmall()));
|
|
CHECK(!t->Is(Type::UnsignedSmall()));
|
|
|
|
double zero_value = OpParameter<double>(zero);
|
|
double minus_zero_value = OpParameter<double>(minus_zero);
|
|
|
|
CHECK_EQ(0.0, zero_value);
|
|
CHECK_NE(-0.0, zero_value);
|
|
CHECK_EQ(-0.0, minus_zero_value);
|
|
CHECK_NE(0.0, minus_zero_value);
|
|
}
|
|
|
|
|
|
TEST(ZeroConstant2) {
|
|
JSConstantCacheTester T;
|
|
|
|
Node* zero = T.Constant(0);
|
|
|
|
CHECK_EQ(IrOpcode::kNumberConstant, zero->opcode());
|
|
CHECK_EQ(zero, T.ZeroConstant());
|
|
CHECK_NE(zero, T.Constant(-0.0));
|
|
CHECK_NE(zero, T.Constant(1.0));
|
|
CHECK_NE(zero, T.Constant(v8::base::OS::nan_value()));
|
|
CHECK_NE(zero, T.Float64Constant(0));
|
|
CHECK_NE(zero, T.Int32Constant(0));
|
|
|
|
Type* t = T.upper(zero);
|
|
|
|
CHECK(t->Is(Type::Number()));
|
|
CHECK(t->Is(Type::Integral32()));
|
|
CHECK(t->Is(Type::Signed32()));
|
|
CHECK(t->Is(Type::Unsigned32()));
|
|
CHECK(t->Is(Type::SignedSmall()));
|
|
CHECK(t->Is(Type::UnsignedSmall()));
|
|
}
|
|
|
|
|
|
TEST(OneConstant1) {
|
|
JSConstantCacheTester T;
|
|
|
|
Node* one = T.OneConstant();
|
|
|
|
CHECK_EQ(IrOpcode::kNumberConstant, one->opcode());
|
|
CHECK_EQ(one, T.Constant(1));
|
|
CHECK_EQ(one, T.Constant(1.0));
|
|
CHECK_NE(one, T.Constant(1.01));
|
|
CHECK_NE(one, T.Constant(-1.01));
|
|
CHECK_NE(one, T.Constant(v8::base::OS::nan_value()));
|
|
CHECK_NE(one, T.Float64Constant(1.0));
|
|
CHECK_NE(one, T.Int32Constant(1));
|
|
|
|
Type* t = T.upper(one);
|
|
|
|
CHECK(t->Is(Type::Number()));
|
|
CHECK(t->Is(Type::Integral32()));
|
|
CHECK(t->Is(Type::Signed32()));
|
|
CHECK(t->Is(Type::Unsigned32()));
|
|
CHECK(t->Is(Type::SignedSmall()));
|
|
CHECK(t->Is(Type::UnsignedSmall()));
|
|
}
|
|
|
|
|
|
TEST(OneConstant2) {
|
|
JSConstantCacheTester T;
|
|
|
|
Node* one = T.Constant(1);
|
|
|
|
CHECK_EQ(IrOpcode::kNumberConstant, one->opcode());
|
|
CHECK_EQ(one, T.OneConstant());
|
|
CHECK_EQ(one, T.Constant(1.0));
|
|
CHECK_NE(one, T.Constant(1.01));
|
|
CHECK_NE(one, T.Constant(-1.01));
|
|
CHECK_NE(one, T.Constant(v8::base::OS::nan_value()));
|
|
CHECK_NE(one, T.Float64Constant(1.0));
|
|
CHECK_NE(one, T.Int32Constant(1));
|
|
|
|
Type* t = T.upper(one);
|
|
|
|
CHECK(t->Is(Type::Number()));
|
|
CHECK(t->Is(Type::Integral32()));
|
|
CHECK(t->Is(Type::Signed32()));
|
|
CHECK(t->Is(Type::Unsigned32()));
|
|
CHECK(t->Is(Type::SignedSmall()));
|
|
CHECK(t->Is(Type::UnsignedSmall()));
|
|
}
|
|
|
|
|
|
TEST(Canonicalizations) {
|
|
JSConstantCacheTester T;
|
|
|
|
CHECK_EQ(T.ZeroConstant(), T.ZeroConstant());
|
|
CHECK_EQ(T.UndefinedConstant(), T.UndefinedConstant());
|
|
CHECK_EQ(T.TheHoleConstant(), T.TheHoleConstant());
|
|
CHECK_EQ(T.TrueConstant(), T.TrueConstant());
|
|
CHECK_EQ(T.FalseConstant(), T.FalseConstant());
|
|
CHECK_EQ(T.NullConstant(), T.NullConstant());
|
|
CHECK_EQ(T.ZeroConstant(), T.ZeroConstant());
|
|
CHECK_EQ(T.OneConstant(), T.OneConstant());
|
|
CHECK_EQ(T.NaNConstant(), T.NaNConstant());
|
|
}
|
|
|
|
|
|
TEST(NoAliasing) {
|
|
JSConstantCacheTester T;
|
|
|
|
Node* nodes[] = {T.UndefinedConstant(), T.TheHoleConstant(), T.TrueConstant(),
|
|
T.FalseConstant(), T.NullConstant(), T.ZeroConstant(),
|
|
T.OneConstant(), T.NaNConstant(), T.Constant(21),
|
|
T.Constant(22.2)};
|
|
|
|
for (size_t i = 0; i < arraysize(nodes); i++) {
|
|
for (size_t j = 0; j < arraysize(nodes); j++) {
|
|
if (i != j) CHECK_NE(nodes[i], nodes[j]);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
TEST(CanonicalizingNumbers) {
|
|
JSConstantCacheTester T;
|
|
|
|
FOR_FLOAT64_INPUTS(i) {
|
|
Node* node = T.Constant(*i);
|
|
for (int j = 0; j < 5; j++) {
|
|
CHECK_EQ(node, T.Constant(*i));
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
TEST(NumberTypes) {
|
|
JSConstantCacheTester T;
|
|
|
|
FOR_FLOAT64_INPUTS(i) {
|
|
double value = *i;
|
|
Node* node = T.Constant(value);
|
|
CHECK(T.upper(node)->Equals(Type::Of(value, T.main_zone())));
|
|
}
|
|
}
|
|
|
|
|
|
TEST(HeapNumbers) {
|
|
JSConstantCacheTester T;
|
|
|
|
FOR_FLOAT64_INPUTS(i) {
|
|
double value = *i;
|
|
Handle<Object> num = T.factory()->NewNumber(value);
|
|
Handle<HeapNumber> heap = T.factory()->NewHeapNumber(value);
|
|
Node* node1 = T.Constant(value);
|
|
Node* node2 = T.Constant(num);
|
|
Node* node3 = T.Constant(heap);
|
|
CHECK_EQ(node1, node2);
|
|
CHECK_EQ(node1, node3);
|
|
}
|
|
}
|
|
|
|
|
|
TEST(OddballHandle) {
|
|
JSConstantCacheTester T;
|
|
|
|
CHECK_EQ(T.UndefinedConstant(), T.Constant(T.factory()->undefined_value()));
|
|
CHECK_EQ(T.TheHoleConstant(), T.Constant(T.factory()->the_hole_value()));
|
|
CHECK_EQ(T.TrueConstant(), T.Constant(T.factory()->true_value()));
|
|
CHECK_EQ(T.FalseConstant(), T.Constant(T.factory()->false_value()));
|
|
CHECK_EQ(T.NullConstant(), T.Constant(T.factory()->null_value()));
|
|
CHECK_EQ(T.NaNConstant(), T.Constant(T.factory()->nan_value()));
|
|
}
|
|
|
|
|
|
TEST(OddballValues) {
|
|
JSConstantCacheTester T;
|
|
|
|
CHECK_EQ(*T.factory()->undefined_value(), *T.handle(T.UndefinedConstant()));
|
|
CHECK_EQ(*T.factory()->the_hole_value(), *T.handle(T.TheHoleConstant()));
|
|
CHECK_EQ(*T.factory()->true_value(), *T.handle(T.TrueConstant()));
|
|
CHECK_EQ(*T.factory()->false_value(), *T.handle(T.FalseConstant()));
|
|
CHECK_EQ(*T.factory()->null_value(), *T.handle(T.NullConstant()));
|
|
}
|
|
|
|
|
|
TEST(OddballTypes) {
|
|
JSConstantCacheTester T;
|
|
|
|
CHECK(T.upper(T.UndefinedConstant())->Is(Type::Undefined()));
|
|
// TODO(dcarney): figure this out.
|
|
// CHECK(T.upper(T.TheHoleConstant())->Is(Type::Internal()));
|
|
CHECK(T.upper(T.TrueConstant())->Is(Type::Boolean()));
|
|
CHECK(T.upper(T.FalseConstant())->Is(Type::Boolean()));
|
|
CHECK(T.upper(T.NullConstant())->Is(Type::Null()));
|
|
CHECK(T.upper(T.ZeroConstant())->Is(Type::Number()));
|
|
CHECK(T.upper(T.OneConstant())->Is(Type::Number()));
|
|
CHECK(T.upper(T.NaNConstant())->Is(Type::NaN()));
|
|
}
|
|
|
|
|
|
TEST(ExternalReferences) {
|
|
// TODO(titzer): test canonicalization of external references.
|
|
}
|