2014-07-30 13:54:45 +00:00
|
|
|
// 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.
|
|
|
|
|
2015-10-28 13:09:46 +00:00
|
|
|
// TODO(jochen): Remove this after the setting is turned on globally.
|
|
|
|
#define V8_IMMINENT_DEPRECATION_WARNINGS
|
|
|
|
|
2014-07-30 13:54:45 +00:00
|
|
|
#include <limits>
|
|
|
|
|
|
|
|
#include "test/cctest/cctest.h"
|
2015-01-30 09:29:25 +00:00
|
|
|
#include "test/cctest/compiler/codegen-tester.h"
|
2014-07-30 13:54:45 +00:00
|
|
|
#include "test/cctest/compiler/graph-builder-tester.h"
|
2014-09-24 16:04:05 +00:00
|
|
|
#include "test/cctest/compiler/value-helper.h"
|
2014-07-30 13:54:45 +00:00
|
|
|
|
|
|
|
#include "src/compiler/node-matchers.h"
|
|
|
|
#include "src/compiler/representation-change.h"
|
|
|
|
|
2015-10-30 09:16:26 +00:00
|
|
|
namespace v8 {
|
2014-07-30 13:54:45 +00:00
|
|
|
namespace internal {
|
|
|
|
namespace compiler {
|
|
|
|
|
|
|
|
class RepresentationChangerTester : public HandleAndZoneScope,
|
|
|
|
public GraphAndBuilders {
|
|
|
|
public:
|
2014-08-05 08:47:39 +00:00
|
|
|
explicit RepresentationChangerTester(int num_parameters = 0)
|
2014-07-30 13:54:45 +00:00
|
|
|
: GraphAndBuilders(main_zone()),
|
2014-09-12 11:06:37 +00:00
|
|
|
javascript_(main_zone()),
|
2015-01-23 15:19:34 +00:00
|
|
|
jsgraph_(main_isolate(), main_graph_, &main_common_, &javascript_,
|
2015-10-26 14:29:15 +00:00
|
|
|
&main_simplified_, &main_machine_),
|
|
|
|
changer_(&jsgraph_, main_isolate()) {
|
2014-08-05 08:47:39 +00:00
|
|
|
Node* s = graph()->NewNode(common()->Start(num_parameters));
|
|
|
|
graph()->SetStart(s);
|
2014-07-30 13:54:45 +00:00
|
|
|
}
|
|
|
|
|
2014-09-12 11:06:37 +00:00
|
|
|
JSOperatorBuilder javascript_;
|
2014-07-30 13:54:45 +00:00
|
|
|
JSGraph jsgraph_;
|
|
|
|
RepresentationChanger changer_;
|
|
|
|
|
|
|
|
Isolate* isolate() { return main_isolate(); }
|
|
|
|
Graph* graph() { return main_graph_; }
|
|
|
|
CommonOperatorBuilder* common() { return &main_common_; }
|
|
|
|
JSGraph* jsgraph() { return &jsgraph_; }
|
|
|
|
RepresentationChanger* changer() { return &changer_; }
|
|
|
|
|
|
|
|
// TODO(titzer): use ValueChecker / ValueUtil
|
|
|
|
void CheckInt32Constant(Node* n, int32_t expected) {
|
2014-09-08 09:16:11 +00:00
|
|
|
Int32Matcher m(n);
|
2014-07-30 13:54:45 +00:00
|
|
|
CHECK(m.HasValue());
|
|
|
|
CHECK_EQ(expected, m.Value());
|
|
|
|
}
|
|
|
|
|
2014-09-24 16:04:05 +00:00
|
|
|
void CheckUint32Constant(Node* n, uint32_t expected) {
|
|
|
|
Uint32Matcher m(n);
|
|
|
|
CHECK(m.HasValue());
|
|
|
|
CHECK_EQ(static_cast<int>(expected), static_cast<int>(m.Value()));
|
|
|
|
}
|
|
|
|
|
|
|
|
void CheckFloat64Constant(Node* n, double expected) {
|
|
|
|
Float64Matcher m(n);
|
|
|
|
CHECK(m.HasValue());
|
2015-01-30 09:29:25 +00:00
|
|
|
CheckDoubleEq(expected, m.Value());
|
2014-09-24 16:04:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CheckFloat32Constant(Node* n, float expected) {
|
|
|
|
CHECK_EQ(IrOpcode::kFloat32Constant, n->opcode());
|
|
|
|
float fval = OpParameter<float>(n->op());
|
|
|
|
CHECK_EQ(expected, fval);
|
|
|
|
}
|
|
|
|
|
2014-09-08 09:16:11 +00:00
|
|
|
void CheckHeapConstant(Node* n, HeapObject* expected) {
|
2015-06-19 12:48:58 +00:00
|
|
|
HeapObjectMatcher m(n);
|
2014-07-30 13:54:45 +00:00
|
|
|
CHECK(m.HasValue());
|
2015-08-31 08:24:52 +00:00
|
|
|
CHECK_EQ(expected, *m.Value());
|
2014-07-30 13:54:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CheckNumberConstant(Node* n, double expected) {
|
2014-09-08 09:16:11 +00:00
|
|
|
NumberMatcher m(n);
|
2014-07-30 13:54:45 +00:00
|
|
|
CHECK_EQ(IrOpcode::kNumberConstant, n->opcode());
|
|
|
|
CHECK(m.HasValue());
|
2015-01-30 09:29:25 +00:00
|
|
|
CheckDoubleEq(expected, m.Value());
|
2014-07-30 13:54:45 +00:00
|
|
|
}
|
|
|
|
|
2015-01-14 12:06:41 +00:00
|
|
|
Node* Parameter(int index = 0) {
|
|
|
|
return graph()->NewNode(common()->Parameter(index), graph()->start());
|
2014-07-30 13:54:45 +00:00
|
|
|
}
|
|
|
|
|
2014-08-14 09:19:54 +00:00
|
|
|
void CheckTypeError(MachineTypeUnion from, MachineTypeUnion to) {
|
2014-07-30 13:54:45 +00:00
|
|
|
changer()->testing_type_errors_ = true;
|
|
|
|
changer()->type_error_ = false;
|
|
|
|
Node* n = Parameter(0);
|
|
|
|
Node* c = changer()->GetRepresentationFor(n, from, to);
|
|
|
|
CHECK(changer()->type_error_);
|
2014-08-14 09:19:54 +00:00
|
|
|
CHECK_EQ(n, c);
|
2014-07-30 13:54:45 +00:00
|
|
|
}
|
|
|
|
|
2014-08-14 09:19:54 +00:00
|
|
|
void CheckNop(MachineTypeUnion from, MachineTypeUnion to) {
|
2014-07-30 13:54:45 +00:00
|
|
|
Node* n = Parameter(0);
|
|
|
|
Node* c = changer()->GetRepresentationFor(n, from, to);
|
|
|
|
CHECK_EQ(n, c);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2014-09-24 11:08:35 +00:00
|
|
|
static const MachineType all_reps[] = {kRepBit, kRepWord32, kRepWord64,
|
|
|
|
kRepFloat32, kRepFloat64, kRepTagged};
|
2014-07-30 13:54:45 +00:00
|
|
|
|
|
|
|
|
2015-01-14 12:06:41 +00:00
|
|
|
TEST(BoolToBit_constant) {
|
2014-07-30 13:54:45 +00:00
|
|
|
RepresentationChangerTester r;
|
|
|
|
|
|
|
|
Node* true_node = r.jsgraph()->TrueConstant();
|
2014-08-14 09:19:54 +00:00
|
|
|
Node* true_bit =
|
|
|
|
r.changer()->GetRepresentationFor(true_node, kRepTagged, kRepBit);
|
2014-07-30 13:54:45 +00:00
|
|
|
r.CheckInt32Constant(true_bit, 1);
|
|
|
|
|
|
|
|
Node* false_node = r.jsgraph()->FalseConstant();
|
|
|
|
Node* false_bit =
|
2014-08-14 09:19:54 +00:00
|
|
|
r.changer()->GetRepresentationFor(false_node, kRepTagged, kRepBit);
|
2014-07-30 13:54:45 +00:00
|
|
|
r.CheckInt32Constant(false_bit, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
TEST(BitToBool_constant) {
|
|
|
|
RepresentationChangerTester r;
|
|
|
|
|
|
|
|
for (int i = -5; i < 5; i++) {
|
|
|
|
Node* node = r.jsgraph()->Int32Constant(i);
|
2014-08-14 09:19:54 +00:00
|
|
|
Node* val = r.changer()->GetRepresentationFor(node, kRepBit, kRepTagged);
|
2014-07-30 13:54:45 +00:00
|
|
|
r.CheckHeapConstant(val, i == 0 ? r.isolate()->heap()->false_value()
|
|
|
|
: r.isolate()->heap()->true_value());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
TEST(ToTagged_constant) {
|
|
|
|
RepresentationChangerTester r;
|
|
|
|
|
2014-09-24 16:04:05 +00:00
|
|
|
{
|
|
|
|
FOR_FLOAT64_INPUTS(i) {
|
|
|
|
Node* n = r.jsgraph()->Float64Constant(*i);
|
|
|
|
Node* c = r.changer()->GetRepresentationFor(n, kRepFloat64, kRepTagged);
|
|
|
|
r.CheckNumberConstant(c, *i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
FOR_FLOAT64_INPUTS(i) {
|
|
|
|
Node* n = r.jsgraph()->Constant(*i);
|
|
|
|
Node* c = r.changer()->GetRepresentationFor(n, kRepFloat64, kRepTagged);
|
|
|
|
r.CheckNumberConstant(c, *i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
FOR_FLOAT32_INPUTS(i) {
|
|
|
|
Node* n = r.jsgraph()->Float32Constant(*i);
|
|
|
|
Node* c = r.changer()->GetRepresentationFor(n, kRepFloat32, kRepTagged);
|
|
|
|
r.CheckNumberConstant(c, *i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
FOR_INT32_INPUTS(i) {
|
|
|
|
Node* n = r.jsgraph()->Int32Constant(*i);
|
|
|
|
Node* c = r.changer()->GetRepresentationFor(n, kRepWord32 | kTypeInt32,
|
|
|
|
kRepTagged);
|
|
|
|
r.CheckNumberConstant(c, *i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
FOR_UINT32_INPUTS(i) {
|
|
|
|
Node* n = r.jsgraph()->Int32Constant(*i);
|
|
|
|
Node* c = r.changer()->GetRepresentationFor(n, kRepWord32 | kTypeUint32,
|
|
|
|
kRepTagged);
|
|
|
|
r.CheckNumberConstant(c, *i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
TEST(ToFloat64_constant) {
|
|
|
|
RepresentationChangerTester r;
|
|
|
|
|
|
|
|
{
|
|
|
|
FOR_FLOAT64_INPUTS(i) {
|
|
|
|
Node* n = r.jsgraph()->Float64Constant(*i);
|
|
|
|
Node* c = r.changer()->GetRepresentationFor(n, kRepFloat64, kRepFloat64);
|
|
|
|
CHECK_EQ(n, c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
FOR_FLOAT64_INPUTS(i) {
|
|
|
|
Node* n = r.jsgraph()->Constant(*i);
|
|
|
|
Node* c = r.changer()->GetRepresentationFor(n, kRepTagged, kRepFloat64);
|
|
|
|
r.CheckFloat64Constant(c, *i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
FOR_FLOAT32_INPUTS(i) {
|
|
|
|
Node* n = r.jsgraph()->Float32Constant(*i);
|
|
|
|
Node* c = r.changer()->GetRepresentationFor(n, kRepFloat32, kRepFloat64);
|
|
|
|
r.CheckFloat64Constant(c, *i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
FOR_INT32_INPUTS(i) {
|
|
|
|
Node* n = r.jsgraph()->Int32Constant(*i);
|
|
|
|
Node* c = r.changer()->GetRepresentationFor(n, kRepWord32 | kTypeInt32,
|
|
|
|
kRepFloat64);
|
|
|
|
r.CheckFloat64Constant(c, *i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
FOR_UINT32_INPUTS(i) {
|
|
|
|
Node* n = r.jsgraph()->Int32Constant(*i);
|
|
|
|
Node* c = r.changer()->GetRepresentationFor(n, kRepWord32 | kTypeUint32,
|
|
|
|
kRepFloat64);
|
|
|
|
r.CheckFloat64Constant(c, *i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static bool IsFloat32Int32(int32_t val) {
|
|
|
|
return val >= -(1 << 23) && val <= (1 << 23);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static bool IsFloat32Uint32(uint32_t val) { return val <= (1 << 23); }
|
|
|
|
|
|
|
|
|
|
|
|
TEST(ToFloat32_constant) {
|
|
|
|
RepresentationChangerTester r;
|
|
|
|
|
|
|
|
{
|
|
|
|
FOR_FLOAT32_INPUTS(i) {
|
|
|
|
Node* n = r.jsgraph()->Float32Constant(*i);
|
|
|
|
Node* c = r.changer()->GetRepresentationFor(n, kRepFloat32, kRepFloat32);
|
|
|
|
CHECK_EQ(n, c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
FOR_FLOAT32_INPUTS(i) {
|
|
|
|
Node* n = r.jsgraph()->Constant(*i);
|
|
|
|
Node* c = r.changer()->GetRepresentationFor(n, kRepTagged, kRepFloat32);
|
|
|
|
r.CheckFloat32Constant(c, *i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
FOR_FLOAT32_INPUTS(i) {
|
|
|
|
Node* n = r.jsgraph()->Float64Constant(*i);
|
|
|
|
Node* c = r.changer()->GetRepresentationFor(n, kRepFloat64, kRepFloat32);
|
|
|
|
r.CheckFloat32Constant(c, *i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
FOR_INT32_INPUTS(i) {
|
|
|
|
if (!IsFloat32Int32(*i)) continue;
|
|
|
|
Node* n = r.jsgraph()->Int32Constant(*i);
|
|
|
|
Node* c = r.changer()->GetRepresentationFor(n, kRepWord32 | kTypeInt32,
|
|
|
|
kRepFloat32);
|
2014-09-25 05:17:38 +00:00
|
|
|
r.CheckFloat32Constant(c, static_cast<float>(*i));
|
2014-09-24 16:04:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
FOR_UINT32_INPUTS(i) {
|
|
|
|
if (!IsFloat32Uint32(*i)) continue;
|
|
|
|
Node* n = r.jsgraph()->Int32Constant(*i);
|
|
|
|
Node* c = r.changer()->GetRepresentationFor(n, kRepWord32 | kTypeUint32,
|
|
|
|
kRepFloat32);
|
2014-09-25 05:17:38 +00:00
|
|
|
r.CheckFloat32Constant(c, static_cast<float>(*i));
|
2014-09-24 16:04:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
TEST(ToInt32_constant) {
|
|
|
|
RepresentationChangerTester r;
|
|
|
|
|
|
|
|
{
|
|
|
|
FOR_INT32_INPUTS(i) {
|
|
|
|
Node* n = r.jsgraph()->Int32Constant(*i);
|
|
|
|
Node* c = r.changer()->GetRepresentationFor(n, kRepWord32 | kTypeInt32,
|
|
|
|
kRepWord32);
|
|
|
|
r.CheckInt32Constant(c, *i);
|
|
|
|
}
|
2014-07-30 13:54:45 +00:00
|
|
|
}
|
|
|
|
|
2014-09-24 16:04:05 +00:00
|
|
|
{
|
|
|
|
FOR_INT32_INPUTS(i) {
|
|
|
|
if (!IsFloat32Int32(*i)) continue;
|
2014-09-25 05:17:38 +00:00
|
|
|
Node* n = r.jsgraph()->Float32Constant(static_cast<float>(*i));
|
2014-09-24 16:04:05 +00:00
|
|
|
Node* c = r.changer()->GetRepresentationFor(n, kRepFloat32 | kTypeInt32,
|
|
|
|
kRepWord32);
|
|
|
|
r.CheckInt32Constant(c, *i);
|
|
|
|
}
|
2014-09-24 11:55:07 +00:00
|
|
|
}
|
|
|
|
|
2014-09-24 16:04:05 +00:00
|
|
|
{
|
|
|
|
FOR_INT32_INPUTS(i) {
|
|
|
|
Node* n = r.jsgraph()->Float64Constant(*i);
|
|
|
|
Node* c = r.changer()->GetRepresentationFor(n, kRepFloat64 | kTypeInt32,
|
|
|
|
kRepWord32);
|
|
|
|
r.CheckInt32Constant(c, *i);
|
|
|
|
}
|
2014-07-30 13:54:45 +00:00
|
|
|
}
|
|
|
|
|
2014-09-24 16:04:05 +00:00
|
|
|
{
|
|
|
|
FOR_INT32_INPUTS(i) {
|
|
|
|
Node* n = r.jsgraph()->Constant(*i);
|
|
|
|
Node* c = r.changer()->GetRepresentationFor(n, kRepTagged | kTypeInt32,
|
|
|
|
kRepWord32);
|
|
|
|
r.CheckInt32Constant(c, *i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
TEST(ToUint32_constant) {
|
|
|
|
RepresentationChangerTester r;
|
|
|
|
|
|
|
|
{
|
|
|
|
FOR_UINT32_INPUTS(i) {
|
|
|
|
Node* n = r.jsgraph()->Int32Constant(*i);
|
|
|
|
Node* c = r.changer()->GetRepresentationFor(n, kRepWord32 | kTypeUint32,
|
|
|
|
kRepWord32);
|
|
|
|
r.CheckUint32Constant(c, *i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
FOR_UINT32_INPUTS(i) {
|
|
|
|
if (!IsFloat32Uint32(*i)) continue;
|
2014-09-25 05:17:38 +00:00
|
|
|
Node* n = r.jsgraph()->Float32Constant(static_cast<float>(*i));
|
2014-09-24 16:04:05 +00:00
|
|
|
Node* c = r.changer()->GetRepresentationFor(n, kRepFloat32 | kTypeUint32,
|
|
|
|
kRepWord32);
|
|
|
|
r.CheckUint32Constant(c, *i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
FOR_UINT32_INPUTS(i) {
|
|
|
|
Node* n = r.jsgraph()->Float64Constant(*i);
|
|
|
|
Node* c = r.changer()->GetRepresentationFor(n, kRepFloat64 | kTypeUint32,
|
|
|
|
kRepWord32);
|
|
|
|
r.CheckUint32Constant(c, *i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
FOR_UINT32_INPUTS(i) {
|
|
|
|
Node* n = r.jsgraph()->Constant(static_cast<double>(*i));
|
|
|
|
Node* c = r.changer()->GetRepresentationFor(n, kRepTagged | kTypeUint32,
|
|
|
|
kRepWord32);
|
|
|
|
r.CheckUint32Constant(c, *i);
|
|
|
|
}
|
2014-07-30 13:54:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-08-14 09:19:54 +00:00
|
|
|
static void CheckChange(IrOpcode::Value expected, MachineTypeUnion from,
|
2015-01-14 12:06:41 +00:00
|
|
|
MachineTypeUnion to) {
|
2014-07-30 13:54:45 +00:00
|
|
|
RepresentationChangerTester r;
|
|
|
|
|
2015-01-14 12:06:41 +00:00
|
|
|
Node* n = r.Parameter();
|
2014-07-30 13:54:45 +00:00
|
|
|
Node* c = r.changer()->GetRepresentationFor(n, from, to);
|
|
|
|
|
|
|
|
CHECK_NE(c, n);
|
|
|
|
CHECK_EQ(expected, c->opcode());
|
|
|
|
CHECK_EQ(n, c->InputAt(0));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-09-24 11:55:07 +00:00
|
|
|
static void CheckTwoChanges(IrOpcode::Value expected2,
|
|
|
|
IrOpcode::Value expected1, MachineTypeUnion from,
|
2015-01-14 12:06:41 +00:00
|
|
|
MachineTypeUnion to) {
|
2014-09-24 11:55:07 +00:00
|
|
|
RepresentationChangerTester r;
|
|
|
|
|
2015-01-14 12:06:41 +00:00
|
|
|
Node* n = r.Parameter();
|
2014-09-24 11:55:07 +00:00
|
|
|
Node* c1 = r.changer()->GetRepresentationFor(n, from, to);
|
|
|
|
|
|
|
|
CHECK_NE(c1, n);
|
|
|
|
CHECK_EQ(expected1, c1->opcode());
|
|
|
|
Node* c2 = c1->InputAt(0);
|
|
|
|
CHECK_NE(c2, n);
|
|
|
|
CHECK_EQ(expected2, c2->opcode());
|
|
|
|
CHECK_EQ(n, c2->InputAt(0));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-07-30 13:54:45 +00:00
|
|
|
TEST(SingleChanges) {
|
2015-01-14 12:06:41 +00:00
|
|
|
CheckChange(IrOpcode::kChangeBoolToBit, kRepTagged, kRepBit);
|
2014-08-14 09:19:54 +00:00
|
|
|
CheckChange(IrOpcode::kChangeBitToBool, kRepBit, kRepTagged);
|
2014-07-30 13:54:45 +00:00
|
|
|
|
2014-08-14 09:19:54 +00:00
|
|
|
CheckChange(IrOpcode::kChangeInt32ToTagged, kRepWord32 | kTypeInt32,
|
|
|
|
kRepTagged);
|
|
|
|
CheckChange(IrOpcode::kChangeUint32ToTagged, kRepWord32 | kTypeUint32,
|
|
|
|
kRepTagged);
|
|
|
|
CheckChange(IrOpcode::kChangeFloat64ToTagged, kRepFloat64, kRepTagged);
|
2014-07-30 13:54:45 +00:00
|
|
|
|
2014-08-14 09:19:54 +00:00
|
|
|
CheckChange(IrOpcode::kChangeTaggedToInt32, kRepTagged | kTypeInt32,
|
|
|
|
kRepWord32);
|
|
|
|
CheckChange(IrOpcode::kChangeTaggedToUint32, kRepTagged | kTypeUint32,
|
|
|
|
kRepWord32);
|
|
|
|
CheckChange(IrOpcode::kChangeTaggedToFloat64, kRepTagged, kRepFloat64);
|
2014-07-30 13:54:45 +00:00
|
|
|
|
|
|
|
// Int32,Uint32 <-> Float64 are actually machine conversions.
|
2014-08-14 09:19:54 +00:00
|
|
|
CheckChange(IrOpcode::kChangeInt32ToFloat64, kRepWord32 | kTypeInt32,
|
|
|
|
kRepFloat64);
|
|
|
|
CheckChange(IrOpcode::kChangeUint32ToFloat64, kRepWord32 | kTypeUint32,
|
|
|
|
kRepFloat64);
|
|
|
|
CheckChange(IrOpcode::kChangeFloat64ToInt32, kRepFloat64 | kTypeInt32,
|
|
|
|
kRepWord32);
|
|
|
|
CheckChange(IrOpcode::kChangeFloat64ToUint32, kRepFloat64 | kTypeUint32,
|
|
|
|
kRepWord32);
|
2014-09-24 11:55:07 +00:00
|
|
|
|
2014-10-08 11:16:45 +00:00
|
|
|
CheckChange(IrOpcode::kTruncateFloat64ToFloat32, kRepFloat64, kRepFloat32);
|
|
|
|
|
2014-09-24 11:55:07 +00:00
|
|
|
// Int32,Uint32 <-> Float32 require two changes.
|
|
|
|
CheckTwoChanges(IrOpcode::kChangeInt32ToFloat64,
|
|
|
|
IrOpcode::kTruncateFloat64ToFloat32, kRepWord32 | kTypeInt32,
|
|
|
|
kRepFloat32);
|
|
|
|
CheckTwoChanges(IrOpcode::kChangeUint32ToFloat64,
|
|
|
|
IrOpcode::kTruncateFloat64ToFloat32, kRepWord32 | kTypeUint32,
|
|
|
|
kRepFloat32);
|
|
|
|
CheckTwoChanges(IrOpcode::kChangeFloat32ToFloat64,
|
|
|
|
IrOpcode::kChangeFloat64ToInt32, kRepFloat32 | kTypeInt32,
|
|
|
|
kRepWord32);
|
|
|
|
CheckTwoChanges(IrOpcode::kChangeFloat32ToFloat64,
|
|
|
|
IrOpcode::kChangeFloat64ToUint32, kRepFloat32 | kTypeUint32,
|
|
|
|
kRepWord32);
|
|
|
|
|
|
|
|
// Float32 <-> Tagged require two changes.
|
|
|
|
CheckTwoChanges(IrOpcode::kChangeFloat32ToFloat64,
|
|
|
|
IrOpcode::kChangeFloat64ToTagged, kRepFloat32, kRepTagged);
|
|
|
|
CheckTwoChanges(IrOpcode::kChangeTaggedToFloat64,
|
|
|
|
IrOpcode::kTruncateFloat64ToFloat32, kRepTagged, kRepFloat32);
|
2014-07-30 13:54:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
TEST(SignednessInWord32) {
|
|
|
|
RepresentationChangerTester r;
|
|
|
|
|
2014-08-14 09:19:54 +00:00
|
|
|
// TODO(titzer): assume that uses of a word32 without a sign mean kTypeInt32.
|
|
|
|
CheckChange(IrOpcode::kChangeTaggedToInt32, kRepTagged,
|
|
|
|
kRepWord32 | kTypeInt32);
|
|
|
|
CheckChange(IrOpcode::kChangeTaggedToUint32, kRepTagged,
|
|
|
|
kRepWord32 | kTypeUint32);
|
|
|
|
CheckChange(IrOpcode::kChangeInt32ToFloat64, kRepWord32, kRepFloat64);
|
|
|
|
CheckChange(IrOpcode::kChangeFloat64ToInt32, kRepFloat64, kRepWord32);
|
2014-09-24 11:55:07 +00:00
|
|
|
|
|
|
|
CheckTwoChanges(IrOpcode::kChangeInt32ToFloat64,
|
|
|
|
IrOpcode::kTruncateFloat64ToFloat32, kRepWord32, kRepFloat32);
|
|
|
|
CheckTwoChanges(IrOpcode::kChangeFloat32ToFloat64,
|
|
|
|
IrOpcode::kChangeFloat64ToInt32, kRepFloat32, kRepWord32);
|
2014-07-30 13:54:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
TEST(Nops) {
|
|
|
|
RepresentationChangerTester r;
|
|
|
|
|
|
|
|
// X -> X is always a nop for any single representation X.
|
2014-08-26 09:19:24 +00:00
|
|
|
for (size_t i = 0; i < arraysize(all_reps); i++) {
|
2014-07-30 13:54:45 +00:00
|
|
|
r.CheckNop(all_reps[i], all_reps[i]);
|
|
|
|
}
|
|
|
|
|
2014-09-19 09:56:12 +00:00
|
|
|
// 32-bit floats.
|
|
|
|
r.CheckNop(kRepFloat32, kRepFloat32);
|
|
|
|
r.CheckNop(kRepFloat32 | kTypeNumber, kRepFloat32);
|
|
|
|
r.CheckNop(kRepFloat32, kRepFloat32 | kTypeNumber);
|
|
|
|
|
2014-08-14 09:19:54 +00:00
|
|
|
// 32-bit words can be used as smaller word sizes and vice versa, because
|
|
|
|
// loads from memory implicitly sign or zero extend the value to the
|
|
|
|
// full machine word size, and stores implicitly truncate.
|
|
|
|
r.CheckNop(kRepWord32, kRepWord8);
|
|
|
|
r.CheckNop(kRepWord32, kRepWord16);
|
|
|
|
r.CheckNop(kRepWord32, kRepWord32);
|
|
|
|
r.CheckNop(kRepWord8, kRepWord32);
|
|
|
|
r.CheckNop(kRepWord16, kRepWord32);
|
|
|
|
|
|
|
|
// kRepBit (result of comparison) is implicitly a wordish thing.
|
|
|
|
r.CheckNop(kRepBit, kRepWord8);
|
|
|
|
r.CheckNop(kRepBit | kTypeBool, kRepWord8);
|
|
|
|
r.CheckNop(kRepBit, kRepWord16);
|
|
|
|
r.CheckNop(kRepBit | kTypeBool, kRepWord16);
|
|
|
|
r.CheckNop(kRepBit, kRepWord32);
|
|
|
|
r.CheckNop(kRepBit | kTypeBool, kRepWord32);
|
|
|
|
r.CheckNop(kRepBit, kRepWord64);
|
|
|
|
r.CheckNop(kRepBit | kTypeBool, kRepWord64);
|
2014-07-30 13:54:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
TEST(TypeErrors) {
|
|
|
|
RepresentationChangerTester r;
|
|
|
|
|
2015-01-14 12:06:41 +00:00
|
|
|
// Wordish cannot be implicitly converted to/from comparison conditions.
|
|
|
|
r.CheckTypeError(kRepWord8, kRepBit);
|
|
|
|
r.CheckTypeError(kRepWord8, kRepBit | kTypeBool);
|
|
|
|
r.CheckTypeError(kRepWord16, kRepBit);
|
|
|
|
r.CheckTypeError(kRepWord16, kRepBit | kTypeBool);
|
|
|
|
r.CheckTypeError(kRepWord32, kRepBit);
|
|
|
|
r.CheckTypeError(kRepWord32, kRepBit | kTypeBool);
|
|
|
|
r.CheckTypeError(kRepWord64, kRepBit);
|
|
|
|
r.CheckTypeError(kRepWord64, kRepBit | kTypeBool);
|
|
|
|
|
2014-07-30 13:54:45 +00:00
|
|
|
// Floats cannot be implicitly converted to/from comparison conditions.
|
2014-08-14 09:19:54 +00:00
|
|
|
r.CheckTypeError(kRepFloat64, kRepBit);
|
|
|
|
r.CheckTypeError(kRepFloat64, kRepBit | kTypeBool);
|
|
|
|
r.CheckTypeError(kRepBit, kRepFloat64);
|
|
|
|
r.CheckTypeError(kRepBit | kTypeBool, kRepFloat64);
|
2014-07-30 13:54:45 +00:00
|
|
|
|
2014-09-19 09:56:12 +00:00
|
|
|
// Floats cannot be implicitly converted to/from comparison conditions.
|
|
|
|
r.CheckTypeError(kRepFloat32, kRepBit);
|
|
|
|
r.CheckTypeError(kRepFloat32, kRepBit | kTypeBool);
|
|
|
|
r.CheckTypeError(kRepBit, kRepFloat32);
|
|
|
|
r.CheckTypeError(kRepBit | kTypeBool, kRepFloat32);
|
|
|
|
|
2014-07-30 13:54:45 +00:00
|
|
|
// Word64 is internal and shouldn't be implicitly converted.
|
2014-08-14 09:19:54 +00:00
|
|
|
r.CheckTypeError(kRepWord64, kRepTagged | kTypeBool);
|
|
|
|
r.CheckTypeError(kRepWord64, kRepTagged);
|
|
|
|
r.CheckTypeError(kRepWord64, kRepTagged | kTypeBool);
|
|
|
|
r.CheckTypeError(kRepTagged, kRepWord64);
|
|
|
|
r.CheckTypeError(kRepTagged | kTypeBool, kRepWord64);
|
2014-07-30 13:54:45 +00:00
|
|
|
|
|
|
|
// Word64 / Word32 shouldn't be implicitly converted.
|
2014-08-14 09:19:54 +00:00
|
|
|
r.CheckTypeError(kRepWord64, kRepWord32);
|
|
|
|
r.CheckTypeError(kRepWord32, kRepWord64);
|
|
|
|
r.CheckTypeError(kRepWord64, kRepWord32 | kTypeInt32);
|
|
|
|
r.CheckTypeError(kRepWord32 | kTypeInt32, kRepWord64);
|
|
|
|
r.CheckTypeError(kRepWord64, kRepWord32 | kTypeUint32);
|
|
|
|
r.CheckTypeError(kRepWord32 | kTypeUint32, kRepWord64);
|
2014-07-30 13:54:45 +00:00
|
|
|
|
2014-08-26 09:19:24 +00:00
|
|
|
for (size_t i = 0; i < arraysize(all_reps); i++) {
|
|
|
|
for (size_t j = 0; j < arraysize(all_reps); j++) {
|
2014-07-30 13:54:45 +00:00
|
|
|
if (i == j) continue;
|
|
|
|
// Only a single from representation is allowed.
|
2014-08-14 09:19:54 +00:00
|
|
|
r.CheckTypeError(all_reps[i] | all_reps[j], kRepTagged);
|
2014-07-30 13:54:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-10-30 09:16:26 +00:00
|
|
|
|
|
|
|
} // namespace compiler
|
|
|
|
} // namespace internal
|
|
|
|
} // namespace v8
|