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. |
// ---------------------------------------------------------------------------- |