| Index: test/unittests/compiler/mips/instruction-selector-mips-unittest.cc
|
| diff --git a/test/unittests/compiler/mips/instruction-selector-mips-unittest.cc b/test/unittests/compiler/mips/instruction-selector-mips-unittest.cc
|
| index a16ad7a31fc789c30939a9d787a860fb83389790..64f380cebeb11002c74c5fc3f2add9272f77661e 100644
|
| --- a/test/unittests/compiler/mips/instruction-selector-mips-unittest.cc
|
| +++ b/test/unittests/compiler/mips/instruction-selector-mips-unittest.cc
|
| @@ -296,6 +296,46 @@ INSTANTIATE_TEST_CASE_P(InstructionSelectorTest, InstructionSelectorShiftTest,
|
| ::testing::ValuesIn(kShiftInstructions));
|
|
|
|
|
| +TEST_F(InstructionSelectorTest, Word32ShrWithWord32AndWithImmediate) {
|
| + // The available shift operand range is `0 <= imm < 32`, but we also test
|
| + // that immediates outside this range are handled properly (modulo-32).
|
| + TRACED_FORRANGE(int32_t, shift, -32, 63) {
|
| + int32_t lsb = shift & 0x1f;
|
| + TRACED_FORRANGE(int32_t, width, 1, 32 - lsb) {
|
| + uint32_t jnk = rng()->NextInt();
|
| + jnk = (lsb > 0) ? (jnk >> (32 - lsb)) : 0;
|
| + uint32_t msk = ((0xffffffffu >> (32 - width)) << lsb) | jnk;
|
| + StreamBuilder m(this, kMachInt32, kMachInt32);
|
| + m.Return(m.Word32Shr(m.Word32And(m.Parameter(0), m.Int32Constant(msk)),
|
| + m.Int32Constant(shift)));
|
| + Stream s = m.Build();
|
| + ASSERT_EQ(1U, s.size());
|
| + EXPECT_EQ(kMipsExt, s[0]->arch_opcode());
|
| + ASSERT_EQ(3U, s[0]->InputCount());
|
| + EXPECT_EQ(lsb, s.ToInt32(s[0]->InputAt(1)));
|
| + EXPECT_EQ(width, s.ToInt32(s[0]->InputAt(2)));
|
| + }
|
| + }
|
| + TRACED_FORRANGE(int32_t, shift, -32, 63) {
|
| + int32_t lsb = shift & 0x1f;
|
| + TRACED_FORRANGE(int32_t, width, 1, 32 - lsb) {
|
| + uint32_t jnk = rng()->NextInt();
|
| + jnk = (lsb > 0) ? (jnk >> (32 - lsb)) : 0;
|
| + uint32_t msk = ((0xffffffffu >> (32 - width)) << lsb) | jnk;
|
| + StreamBuilder m(this, kMachInt32, kMachInt32);
|
| + m.Return(m.Word32Shr(m.Word32And(m.Int32Constant(msk), m.Parameter(0)),
|
| + m.Int32Constant(shift)));
|
| + Stream s = m.Build();
|
| + ASSERT_EQ(1U, s.size());
|
| + EXPECT_EQ(kMipsExt, s[0]->arch_opcode());
|
| + ASSERT_EQ(3U, s[0]->InputCount());
|
| + EXPECT_EQ(lsb, s.ToInt32(s[0]->InputAt(1)));
|
| + EXPECT_EQ(width, s.ToInt32(s[0]->InputAt(2)));
|
| + }
|
| + }
|
| +}
|
| +
|
| +
|
| // ----------------------------------------------------------------------------
|
| // Logical instructions.
|
| // ----------------------------------------------------------------------------
|
| @@ -322,6 +362,45 @@ INSTANTIATE_TEST_CASE_P(InstructionSelectorTest, InstructionSelectorLogicalTest,
|
| ::testing::ValuesIn(kLogicalInstructions));
|
|
|
|
|
| +TEST_F(InstructionSelectorTest, Word32AndWithImmediateWithWord32Shr) {
|
| + // The available shift operand range is `0 <= imm < 32`, but we also test
|
| + // that immediates outside this range are handled properly (modulo-32).
|
| + TRACED_FORRANGE(int32_t, shift, -32, 63) {
|
| + int32_t lsb = shift & 0x1f;
|
| + TRACED_FORRANGE(int32_t, width, 1, 31) {
|
| + uint32_t msk = (1 << width) - 1;
|
| + StreamBuilder m(this, kMachInt32, kMachInt32);
|
| + m.Return(m.Word32And(m.Word32Shr(m.Parameter(0), m.Int32Constant(shift)),
|
| + m.Int32Constant(msk)));
|
| + Stream s = m.Build();
|
| + ASSERT_EQ(1U, s.size());
|
| + EXPECT_EQ(kMipsExt, s[0]->arch_opcode());
|
| + ASSERT_EQ(3U, s[0]->InputCount());
|
| + EXPECT_EQ(lsb, s.ToInt32(s[0]->InputAt(1)));
|
| + int32_t actual_width = (lsb + width > 32) ? (32 - lsb) : width;
|
| + EXPECT_EQ(actual_width, s.ToInt32(s[0]->InputAt(2)));
|
| + }
|
| + }
|
| + TRACED_FORRANGE(int32_t, shift, -32, 63) {
|
| + int32_t lsb = shift & 0x1f;
|
| + TRACED_FORRANGE(int32_t, width, 1, 31) {
|
| + uint32_t msk = (1 << width) - 1;
|
| + StreamBuilder m(this, kMachInt32, kMachInt32);
|
| + m.Return(
|
| + m.Word32And(m.Int32Constant(msk),
|
| + m.Word32Shr(m.Parameter(0), m.Int32Constant(shift))));
|
| + Stream s = m.Build();
|
| + ASSERT_EQ(1U, s.size());
|
| + EXPECT_EQ(kMipsExt, s[0]->arch_opcode());
|
| + ASSERT_EQ(3U, s[0]->InputCount());
|
| + EXPECT_EQ(lsb, s.ToInt32(s[0]->InputAt(1)));
|
| + int32_t actual_width = (lsb + width > 32) ? (32 - lsb) : width;
|
| + EXPECT_EQ(actual_width, s.ToInt32(s[0]->InputAt(2)));
|
| + }
|
| + }
|
| +}
|
| +
|
| +
|
| // ----------------------------------------------------------------------------
|
| // MUL/DIV instructions.
|
| // ----------------------------------------------------------------------------
|
|
|