| Index: src/compiler/machine-operator-unittest.cc
|
| diff --git a/src/compiler/machine-operator-unittest.cc b/src/compiler/machine-operator-unittest.cc
|
| index dbf0b5b2dd83edd1f96dda7bc77a93d052b21f01..5b34000b58610165467e62c204fb3a6224f84565 100644
|
| --- a/src/compiler/machine-operator-unittest.cc
|
| +++ b/src/compiler/machine-operator-unittest.cc
|
| @@ -4,82 +4,320 @@
|
|
|
| #include "src/compiler/machine-operator.h"
|
| #include "src/compiler/operator-properties-inl.h"
|
| -#include "src/test/test-utils.h"
|
| +#include "testing/gtest-support.h"
|
|
|
| namespace v8 {
|
| namespace internal {
|
| namespace compiler {
|
|
|
| -class MachineOperatorCommonTest
|
| - : public TestWithZone,
|
| - public ::testing::WithParamInterface<MachineType> {
|
| - public:
|
| - MachineOperatorCommonTest() : machine_(NULL) {}
|
| - virtual ~MachineOperatorCommonTest() { EXPECT_EQ(NULL, machine_); }
|
| -
|
| - virtual void SetUp() OVERRIDE {
|
| - TestWithZone::SetUp();
|
| - EXPECT_EQ(NULL, machine_);
|
| - machine_ = new MachineOperatorBuilder(zone(), GetParam());
|
| - }
|
| +#if GTEST_HAS_COMBINE
|
| +
|
| +// TODO(bmeurer): Find a new home for these.
|
| +inline std::ostream& operator<<(std::ostream& os, const MachineType& type) {
|
| + OStringStream ost;
|
| + ost << type;
|
| + return os << ost.c_str();
|
| +}
|
| +inline std::ostream& operator<<(std::ostream& os,
|
| + const WriteBarrierKind& write_barrier_kind) {
|
| + OStringStream ost;
|
| + ost << write_barrier_kind;
|
| + return os << ost.c_str();
|
| +}
|
|
|
| - virtual void TearDown() OVERRIDE {
|
| - ASSERT_TRUE(machine_ != NULL);
|
| - delete machine_;
|
| - machine_ = NULL;
|
| - TestWithZone::TearDown();
|
| - }
|
|
|
| +template <typename T>
|
| +class MachineOperatorTestWithParam
|
| + : public ::testing::TestWithParam< ::testing::tuple<MachineType, T> > {
|
| protected:
|
| - MachineOperatorBuilder* machine() const { return machine_; }
|
| + MachineType type() const { return ::testing::get<0>(B::GetParam()); }
|
| + const T& GetParam() const { return ::testing::get<1>(B::GetParam()); }
|
|
|
| private:
|
| - MachineOperatorBuilder* machine_;
|
| + typedef ::testing::TestWithParam< ::testing::tuple<MachineType, T> > B;
|
| };
|
|
|
|
|
| -TEST_P(MachineOperatorCommonTest, ChangeInt32ToInt64) {
|
| - const Operator* op = machine()->ChangeInt32ToInt64();
|
| - EXPECT_EQ(1, OperatorProperties::GetValueInputCount(op));
|
| - EXPECT_EQ(1, OperatorProperties::GetTotalInputCount(op));
|
| - EXPECT_EQ(0, OperatorProperties::GetControlOutputCount(op));
|
| - EXPECT_EQ(0, OperatorProperties::GetEffectOutputCount(op));
|
| - EXPECT_EQ(1, OperatorProperties::GetValueOutputCount(op));
|
| +namespace {
|
| +
|
| +const MachineType kMachineReps[] = {kRepWord32, kRepWord64};
|
| +
|
| +
|
| +const MachineType kMachineTypes[] = {
|
| + kMachFloat32, kMachFloat64, kMachInt8, kMachUint8, kMachInt16,
|
| + kMachUint16, kMachInt32, kMachUint32, kMachInt64, kMachUint64,
|
| + kMachPtr, kMachAnyTagged, kRepBit, kRepWord8, kRepWord16,
|
| + kRepWord32, kRepWord64, kRepFloat32, kRepFloat64, kRepTagged};
|
| +
|
| +} // namespace
|
| +
|
| +
|
| +// -----------------------------------------------------------------------------
|
| +// Load operator.
|
| +
|
| +
|
| +typedef MachineOperatorTestWithParam<LoadRepresentation>
|
| + MachineLoadOperatorTest;
|
| +
|
| +
|
| +TEST_P(MachineLoadOperatorTest, InstancesAreGloballyShared) {
|
| + MachineOperatorBuilder machine1(type());
|
| + MachineOperatorBuilder machine2(type());
|
| + EXPECT_EQ(machine1.Load(GetParam()), machine2.Load(GetParam()));
|
| }
|
|
|
|
|
| -TEST_P(MachineOperatorCommonTest, ChangeUint32ToUint64) {
|
| - const Operator* op = machine()->ChangeUint32ToUint64();
|
| - EXPECT_EQ(1, OperatorProperties::GetValueInputCount(op));
|
| - EXPECT_EQ(1, OperatorProperties::GetTotalInputCount(op));
|
| - EXPECT_EQ(0, OperatorProperties::GetControlOutputCount(op));
|
| - EXPECT_EQ(0, OperatorProperties::GetEffectOutputCount(op));
|
| +TEST_P(MachineLoadOperatorTest, NumberOfInputsAndOutputs) {
|
| + MachineOperatorBuilder machine(type());
|
| + const Operator* op = machine.Load(GetParam());
|
| +
|
| + EXPECT_EQ(2, OperatorProperties::GetValueInputCount(op));
|
| + EXPECT_EQ(1, OperatorProperties::GetEffectInputCount(op));
|
| + EXPECT_EQ(0, OperatorProperties::GetControlInputCount(op));
|
| + EXPECT_EQ(3, OperatorProperties::GetTotalInputCount(op));
|
| +
|
| EXPECT_EQ(1, OperatorProperties::GetValueOutputCount(op));
|
| + EXPECT_EQ(1, OperatorProperties::GetEffectOutputCount(op));
|
| + EXPECT_EQ(0, OperatorProperties::GetControlOutputCount(op));
|
| }
|
|
|
|
|
| -TEST_P(MachineOperatorCommonTest, TruncateFloat64ToInt32) {
|
| - const Operator* op = machine()->TruncateFloat64ToInt32();
|
| - EXPECT_EQ(1, OperatorProperties::GetValueInputCount(op));
|
| - EXPECT_EQ(1, OperatorProperties::GetTotalInputCount(op));
|
| - EXPECT_EQ(0, OperatorProperties::GetControlOutputCount(op));
|
| - EXPECT_EQ(0, OperatorProperties::GetEffectOutputCount(op));
|
| - EXPECT_EQ(1, OperatorProperties::GetValueOutputCount(op));
|
| +TEST_P(MachineLoadOperatorTest, OpcodeIsCorrect) {
|
| + MachineOperatorBuilder machine(type());
|
| + EXPECT_EQ(IrOpcode::kLoad, machine.Load(GetParam())->opcode());
|
| +}
|
| +
|
| +
|
| +TEST_P(MachineLoadOperatorTest, ParameterIsCorrect) {
|
| + MachineOperatorBuilder machine(type());
|
| + EXPECT_EQ(GetParam(),
|
| + OpParameter<LoadRepresentation>(machine.Load(GetParam())));
|
| }
|
|
|
|
|
| -TEST_P(MachineOperatorCommonTest, TruncateInt64ToInt32) {
|
| - const Operator* op = machine()->TruncateInt64ToInt32();
|
| - EXPECT_EQ(1, OperatorProperties::GetValueInputCount(op));
|
| - EXPECT_EQ(1, OperatorProperties::GetTotalInputCount(op));
|
| +INSTANTIATE_TEST_CASE_P(MachineOperatorTest, MachineLoadOperatorTest,
|
| + ::testing::Combine(::testing::ValuesIn(kMachineReps),
|
| + ::testing::ValuesIn(kMachineTypes)));
|
| +
|
| +
|
| +// -----------------------------------------------------------------------------
|
| +// Store operator.
|
| +
|
| +
|
| +class MachineStoreOperatorTest
|
| + : public MachineOperatorTestWithParam<
|
| + ::testing::tuple<MachineType, WriteBarrierKind> > {
|
| + protected:
|
| + StoreRepresentation GetParam() const {
|
| + return StoreRepresentation(
|
| + ::testing::get<0>(MachineOperatorTestWithParam<
|
| + ::testing::tuple<MachineType, WriteBarrierKind> >::GetParam()),
|
| + ::testing::get<1>(MachineOperatorTestWithParam<
|
| + ::testing::tuple<MachineType, WriteBarrierKind> >::GetParam()));
|
| + }
|
| +};
|
| +
|
| +
|
| +TEST_P(MachineStoreOperatorTest, InstancesAreGloballyShared) {
|
| + MachineOperatorBuilder machine1(type());
|
| + MachineOperatorBuilder machine2(type());
|
| + EXPECT_EQ(machine1.Store(GetParam()), machine2.Store(GetParam()));
|
| +}
|
| +
|
| +
|
| +TEST_P(MachineStoreOperatorTest, NumberOfInputsAndOutputs) {
|
| + MachineOperatorBuilder machine(type());
|
| + const Operator* op = machine.Store(GetParam());
|
| +
|
| + EXPECT_EQ(3, OperatorProperties::GetValueInputCount(op));
|
| + EXPECT_EQ(1, OperatorProperties::GetEffectInputCount(op));
|
| + EXPECT_EQ(1, OperatorProperties::GetControlInputCount(op));
|
| + EXPECT_EQ(5, OperatorProperties::GetTotalInputCount(op));
|
| +
|
| + EXPECT_EQ(0, OperatorProperties::GetValueOutputCount(op));
|
| + EXPECT_EQ(1, OperatorProperties::GetEffectOutputCount(op));
|
| EXPECT_EQ(0, OperatorProperties::GetControlOutputCount(op));
|
| +}
|
| +
|
| +
|
| +TEST_P(MachineStoreOperatorTest, OpcodeIsCorrect) {
|
| + MachineOperatorBuilder machine(type());
|
| + EXPECT_EQ(IrOpcode::kStore, machine.Store(GetParam())->opcode());
|
| +}
|
| +
|
| +
|
| +TEST_P(MachineStoreOperatorTest, ParameterIsCorrect) {
|
| + MachineOperatorBuilder machine(type());
|
| + EXPECT_EQ(GetParam(),
|
| + OpParameter<StoreRepresentation>(machine.Store(GetParam())));
|
| +}
|
| +
|
| +
|
| +INSTANTIATE_TEST_CASE_P(
|
| + MachineOperatorTest, MachineStoreOperatorTest,
|
| + ::testing::Combine(
|
| + ::testing::ValuesIn(kMachineReps),
|
| + ::testing::Combine(::testing::ValuesIn(kMachineTypes),
|
| + ::testing::Values(kNoWriteBarrier,
|
| + kFullWriteBarrier))));
|
| +
|
| +
|
| +// -----------------------------------------------------------------------------
|
| +// Pure operators.
|
| +
|
| +
|
| +namespace {
|
| +
|
| +struct PureOperator {
|
| + const Operator* (MachineOperatorBuilder::*constructor)();
|
| + IrOpcode::Value opcode;
|
| + int value_input_count;
|
| + int value_output_count;
|
| +};
|
| +
|
| +
|
| +std::ostream& operator<<(std::ostream& os, const PureOperator& pop) {
|
| + return os << IrOpcode::Mnemonic(pop.opcode);
|
| +}
|
| +
|
| +
|
| +const PureOperator kPureOperators[] = {
|
| +#define PURE(Name, input_count, output_count) \
|
| + { \
|
| + &MachineOperatorBuilder::Name, IrOpcode::k##Name, input_count, \
|
| + output_count \
|
| + }
|
| + PURE(Word32And, 2, 1), PURE(Word32Or, 2, 1),
|
| + PURE(Word32Xor, 2, 1), PURE(Word32Shl, 2, 1),
|
| + PURE(Word32Shr, 2, 1), PURE(Word32Sar, 2, 1),
|
| + PURE(Word32Ror, 2, 1), PURE(Word32Equal, 2, 1),
|
| + PURE(Word64And, 2, 1), PURE(Word64Or, 2, 1),
|
| + PURE(Word64Xor, 2, 1), PURE(Word64Shl, 2, 1),
|
| + PURE(Word64Shr, 2, 1), PURE(Word64Sar, 2, 1),
|
| + PURE(Word64Ror, 2, 1), PURE(Word64Equal, 2, 1),
|
| + PURE(Int32Add, 2, 1), PURE(Int32AddWithOverflow, 2, 2),
|
| + PURE(Int32Sub, 2, 1), PURE(Int32SubWithOverflow, 2, 2),
|
| + PURE(Int32Mul, 2, 1), PURE(Int32Div, 2, 1),
|
| + PURE(Int32UDiv, 2, 1), PURE(Int32Mod, 2, 1),
|
| + PURE(Int32UMod, 2, 1), PURE(Int32LessThan, 2, 1),
|
| + PURE(Int32LessThanOrEqual, 2, 1), PURE(Uint32LessThan, 2, 1),
|
| + PURE(Uint32LessThanOrEqual, 2, 1), PURE(Int64Add, 2, 1),
|
| + PURE(Int64Sub, 2, 1), PURE(Int64Mul, 2, 1),
|
| + PURE(Int64Div, 2, 1), PURE(Int64UDiv, 2, 1),
|
| + PURE(Int64Mod, 2, 1), PURE(Int64UMod, 2, 1),
|
| + PURE(Int64LessThan, 2, 1), PURE(Int64LessThanOrEqual, 2, 1),
|
| + PURE(ChangeFloat64ToInt32, 1, 1), PURE(ChangeFloat64ToUint32, 1, 1),
|
| + PURE(ChangeInt32ToInt64, 1, 1), PURE(ChangeUint32ToFloat64, 1, 1),
|
| + PURE(ChangeUint32ToUint64, 1, 1), PURE(TruncateFloat64ToInt32, 1, 1),
|
| + PURE(TruncateInt64ToInt32, 1, 1), PURE(Float64Add, 2, 1),
|
| + PURE(Float64Sub, 2, 1), PURE(Float64Mul, 2, 1),
|
| + PURE(Float64Div, 2, 1), PURE(Float64Mod, 2, 1),
|
| + PURE(Float64Equal, 2, 1), PURE(Float64LessThan, 2, 1),
|
| + PURE(Float64LessThanOrEqual, 2, 1)
|
| +#undef PURE
|
| +};
|
| +
|
| +
|
| +typedef MachineOperatorTestWithParam<PureOperator> MachinePureOperatorTest;
|
| +
|
| +} // namespace
|
| +
|
| +
|
| +TEST_P(MachinePureOperatorTest, InstancesAreGloballyShared) {
|
| + const PureOperator& pop = GetParam();
|
| + MachineOperatorBuilder machine1(type());
|
| + MachineOperatorBuilder machine2(type());
|
| + EXPECT_EQ((machine1.*pop.constructor)(), (machine2.*pop.constructor)());
|
| +}
|
| +
|
| +
|
| +TEST_P(MachinePureOperatorTest, NumberOfInputsAndOutputs) {
|
| + MachineOperatorBuilder machine(type());
|
| + const PureOperator& pop = GetParam();
|
| + const Operator* op = (machine.*pop.constructor)();
|
| +
|
| + EXPECT_EQ(pop.value_input_count, OperatorProperties::GetValueInputCount(op));
|
| + EXPECT_EQ(0, OperatorProperties::GetEffectInputCount(op));
|
| + EXPECT_EQ(0, OperatorProperties::GetControlInputCount(op));
|
| + EXPECT_EQ(pop.value_input_count, OperatorProperties::GetTotalInputCount(op));
|
| +
|
| + EXPECT_EQ(pop.value_output_count,
|
| + OperatorProperties::GetValueOutputCount(op));
|
| EXPECT_EQ(0, OperatorProperties::GetEffectOutputCount(op));
|
| - EXPECT_EQ(1, OperatorProperties::GetValueOutputCount(op));
|
| + EXPECT_EQ(0, OperatorProperties::GetControlOutputCount(op));
|
| +}
|
| +
|
| +
|
| +TEST_P(MachinePureOperatorTest, MarkedAsPure) {
|
| + MachineOperatorBuilder machine(type());
|
| + const PureOperator& pop = GetParam();
|
| + const Operator* op = (machine.*pop.constructor)();
|
| + EXPECT_TRUE(op->HasProperty(Operator::kPure));
|
| }
|
|
|
|
|
| -INSTANTIATE_TEST_CASE_P(MachineOperatorTest, MachineOperatorCommonTest,
|
| - ::testing::Values(kRepWord32, kRepWord64));
|
| +TEST_P(MachinePureOperatorTest, OpcodeIsCorrect) {
|
| + MachineOperatorBuilder machine(type());
|
| + const PureOperator& pop = GetParam();
|
| + const Operator* op = (machine.*pop.constructor)();
|
| + EXPECT_EQ(pop.opcode, op->opcode());
|
| +}
|
| +
|
| +
|
| +INSTANTIATE_TEST_CASE_P(
|
| + MachineOperatorTest, MachinePureOperatorTest,
|
| + ::testing::Combine(::testing::ValuesIn(kMachineReps),
|
| + ::testing::ValuesIn(kPureOperators)));
|
| +
|
| +#endif // GTEST_HAS_COMBINE
|
| +
|
| +
|
| +// -----------------------------------------------------------------------------
|
| +// Pseudo operators.
|
| +
|
| +
|
| +TEST(MachineOperatorTest, PseudoOperatorsWhenWordSizeIs32Bit) {
|
| + MachineOperatorBuilder machine(kRepWord32);
|
| + EXPECT_EQ(machine.Word32And(), machine.WordAnd());
|
| + EXPECT_EQ(machine.Word32Or(), machine.WordOr());
|
| + EXPECT_EQ(machine.Word32Xor(), machine.WordXor());
|
| + EXPECT_EQ(machine.Word32Shl(), machine.WordShl());
|
| + EXPECT_EQ(machine.Word32Shr(), machine.WordShr());
|
| + EXPECT_EQ(machine.Word32Sar(), machine.WordSar());
|
| + EXPECT_EQ(machine.Word32Ror(), machine.WordRor());
|
| + EXPECT_EQ(machine.Word32Equal(), machine.WordEqual());
|
| + EXPECT_EQ(machine.Int32Add(), machine.IntAdd());
|
| + EXPECT_EQ(machine.Int32Sub(), machine.IntSub());
|
| + EXPECT_EQ(machine.Int32Mul(), machine.IntMul());
|
| + EXPECT_EQ(machine.Int32Div(), machine.IntDiv());
|
| + EXPECT_EQ(machine.Int32UDiv(), machine.IntUDiv());
|
| + EXPECT_EQ(machine.Int32Mod(), machine.IntMod());
|
| + EXPECT_EQ(machine.Int32UMod(), machine.IntUMod());
|
| + EXPECT_EQ(machine.Int32LessThan(), machine.IntLessThan());
|
| + EXPECT_EQ(machine.Int32LessThanOrEqual(), machine.IntLessThanOrEqual());
|
| +}
|
| +
|
| +
|
| +TEST(MachineOperatorTest, PseudoOperatorsWhenWordSizeIs64Bit) {
|
| + MachineOperatorBuilder machine(kRepWord64);
|
| + EXPECT_EQ(machine.Word64And(), machine.WordAnd());
|
| + EXPECT_EQ(machine.Word64Or(), machine.WordOr());
|
| + EXPECT_EQ(machine.Word64Xor(), machine.WordXor());
|
| + EXPECT_EQ(machine.Word64Shl(), machine.WordShl());
|
| + EXPECT_EQ(machine.Word64Shr(), machine.WordShr());
|
| + EXPECT_EQ(machine.Word64Sar(), machine.WordSar());
|
| + EXPECT_EQ(machine.Word64Ror(), machine.WordRor());
|
| + EXPECT_EQ(machine.Word64Equal(), machine.WordEqual());
|
| + EXPECT_EQ(machine.Int64Add(), machine.IntAdd());
|
| + EXPECT_EQ(machine.Int64Sub(), machine.IntSub());
|
| + EXPECT_EQ(machine.Int64Mul(), machine.IntMul());
|
| + EXPECT_EQ(machine.Int64Div(), machine.IntDiv());
|
| + EXPECT_EQ(machine.Int64UDiv(), machine.IntUDiv());
|
| + EXPECT_EQ(machine.Int64Mod(), machine.IntMod());
|
| + EXPECT_EQ(machine.Int64UMod(), machine.IntUMod());
|
| + EXPECT_EQ(machine.Int64LessThan(), machine.IntLessThan());
|
| + EXPECT_EQ(machine.Int64LessThanOrEqual(), machine.IntLessThanOrEqual());
|
| +}
|
|
|
| } // namespace compiler
|
| } // namespace internal
|
|
|