Revert of [turbofan] ARM: Improve AND instruction selection (patchset #1 id:1 of https://codereview.chromium.org/1677023002/ )
Reason for revert: Possibly causing Mozilla test failures - will investigate. Original issue's description: > [turbofan] ARM: Improve AND instruction selection > > Improve instruction selector for mask and shift operations by using cheaper > instructions where possible, in preference to UBFX. > > BUG= > > Committed: https://crrev.com/53d9c12977f07f55b6f2a72128b8d02c4c857845 > Cr-Commit-Position: refs/heads/master@{#33843} TBR=bmeurer@chromium.org,jarin@chromium.org,danno@chromium.org # Skipping CQ checks because original CL landed less than 1 days ago. NOPRESUBMIT=true NOTREECHECKS=true NOTRY=true BUG= Review URL: https://codereview.chromium.org/1681953003 Cr-Commit-Position: refs/heads/master@{#33850}
This commit is contained in:
parent
8902513e4d
commit
5ac796b553
@ -551,67 +551,43 @@ void InstructionSelector::VisitWord32And(Node* node) {
|
||||
if (m.right().HasValue()) {
|
||||
uint32_t const value = m.right().Value();
|
||||
uint32_t width = base::bits::CountPopulation32(value);
|
||||
uint32_t leading_zeros = base::bits::CountLeadingZeros32(value);
|
||||
|
||||
// Try to merge SHR operations on the left hand input into this AND.
|
||||
if (m.left().IsWord32Shr()) {
|
||||
Int32BinopMatcher mshr(m.left().node());
|
||||
if (mshr.right().HasValue()) {
|
||||
uint32_t const shift = mshr.right().Value();
|
||||
|
||||
if (((shift == 8) || (shift == 16) || (shift == 24)) &&
|
||||
((value == 0xff) || (value == 0xffff))) {
|
||||
// Merge SHR into AND by emitting a UXTB or UXTH instruction with a
|
||||
// bytewise rotation.
|
||||
Emit((value == 0xff) ? kArmUxtb : kArmUxth,
|
||||
g.DefineAsRegister(m.node()), g.UseRegister(mshr.left().node()),
|
||||
g.TempImmediate(mshr.right().Value()));
|
||||
return;
|
||||
} else if (IsSupported(ARMv7) && (width != 0) &&
|
||||
((leading_zeros + width) == 32)) {
|
||||
// Merge Shr into And by emitting a UBFX instruction.
|
||||
DCHECK_EQ(0u, base::bits::CountTrailingZeros32(value));
|
||||
if ((1 <= shift) && (shift <= 31)) {
|
||||
// UBFX cannot extract bits past the register size, however since
|
||||
// shifting the original value would have introduced some zeros we
|
||||
// can still use UBFX with a smaller mask and the remaining bits
|
||||
// will be zeros.
|
||||
EmitUbfx(this, node, mshr.left().node(), shift,
|
||||
std::min(width, 32 - shift));
|
||||
return;
|
||||
}
|
||||
uint32_t msb = base::bits::CountLeadingZeros32(value);
|
||||
// Try to interpret this AND as UBFX.
|
||||
if (IsSupported(ARMv7) && width != 0 && msb + width == 32) {
|
||||
DCHECK_EQ(0u, base::bits::CountTrailingZeros32(value));
|
||||
if (m.left().IsWord32Shr()) {
|
||||
Int32BinopMatcher mleft(m.left().node());
|
||||
if (mleft.right().IsInRange(0, 31)) {
|
||||
// UBFX cannot extract bits past the register size, however since
|
||||
// shifting the original value would have introduced some zeros we can
|
||||
// still use UBFX with a smaller mask and the remaining bits will be
|
||||
// zeros.
|
||||
uint32_t const lsb = mleft.right().Value();
|
||||
return EmitUbfx(this, node, mleft.left().node(), lsb,
|
||||
std::min(width, 32 - lsb));
|
||||
}
|
||||
}
|
||||
} else if (value == 0xffff) {
|
||||
// Emit UXTH for this AND. We don't bother testing for UXTB, as it's no
|
||||
// better than AND 0xff for this operation.
|
||||
Emit(kArmUxth, g.DefineAsRegister(m.node()),
|
||||
g.UseRegister(m.left().node()), g.TempImmediate(0));
|
||||
return;
|
||||
return EmitUbfx(this, node, m.left().node(), 0, width);
|
||||
}
|
||||
// Try to interpret this AND as BIC.
|
||||
if (g.CanBeImmediate(~value)) {
|
||||
// Emit BIC for this AND by inverting the immediate value first.
|
||||
Emit(kArmBic | AddressingModeField::encode(kMode_Operand2_I),
|
||||
g.DefineAsRegister(node), g.UseRegister(m.left().node()),
|
||||
g.TempImmediate(~value));
|
||||
return;
|
||||
}
|
||||
if (!g.CanBeImmediate(value) && IsSupported(ARMv7)) {
|
||||
// If value has 9 to 23 contiguous set bits, and has the lsb set, we can
|
||||
// replace this AND with UBFX. Other contiguous bit patterns have already
|
||||
// been handled by BIC or will be handled by AND.
|
||||
if ((width != 0) && ((leading_zeros + width) == 32) &&
|
||||
(9 <= leading_zeros) && (leading_zeros <= 23)) {
|
||||
DCHECK_EQ(0u, base::bits::CountTrailingZeros32(value));
|
||||
EmitUbfx(this, node, m.left().node(), 0, width);
|
||||
return;
|
||||
}
|
||||
|
||||
// Try to interpret this AND as UXTH.
|
||||
if (value == 0xffff) {
|
||||
Emit(kArmUxth, g.DefineAsRegister(m.node()),
|
||||
g.UseRegister(m.left().node()), g.TempImmediate(0));
|
||||
return;
|
||||
}
|
||||
// Try to interpret this AND as BFC.
|
||||
if (IsSupported(ARMv7)) {
|
||||
width = 32 - width;
|
||||
leading_zeros = base::bits::CountLeadingZeros32(~value);
|
||||
msb = base::bits::CountLeadingZeros32(~value);
|
||||
uint32_t lsb = base::bits::CountTrailingZeros32(~value);
|
||||
if ((leading_zeros + width + lsb) == 32) {
|
||||
// This AND can be replaced with BFC.
|
||||
if (msb + width + lsb == 32) {
|
||||
Emit(kArmBfc, g.DefineSameAsFirst(node), g.UseRegister(m.left().node()),
|
||||
g.TempImmediate(lsb), g.TempImmediate(width));
|
||||
return;
|
||||
|
@ -2545,8 +2545,7 @@ TEST_F(InstructionSelectorTest, Uint32ModWithParametersForSUDIVAndMLS) {
|
||||
|
||||
|
||||
TEST_F(InstructionSelectorTest, Word32AndWithUbfxImmediateForARMv7) {
|
||||
TRACED_FORRANGE(int32_t, width, 9, 23) {
|
||||
if (width == 16) continue; // Uxth.
|
||||
TRACED_FORRANGE(int32_t, width, 1, 32) {
|
||||
StreamBuilder m(this, MachineType::Int32(), MachineType::Int32());
|
||||
m.Return(m.Word32And(m.Parameter(0),
|
||||
m.Int32Constant(0xffffffffu >> (32 - width))));
|
||||
@ -2557,8 +2556,7 @@ TEST_F(InstructionSelectorTest, Word32AndWithUbfxImmediateForARMv7) {
|
||||
EXPECT_EQ(0, s.ToInt32(s[0]->InputAt(1)));
|
||||
EXPECT_EQ(width, s.ToInt32(s[0]->InputAt(2)));
|
||||
}
|
||||
TRACED_FORRANGE(int32_t, width, 9, 23) {
|
||||
if (width == 16) continue; // Uxth.
|
||||
TRACED_FORRANGE(int32_t, width, 1, 32) {
|
||||
StreamBuilder m(this, MachineType::Int32(), MachineType::Int32());
|
||||
m.Return(m.Word32And(m.Int32Constant(0xffffffffu >> (32 - width)),
|
||||
m.Parameter(0)));
|
||||
@ -2574,7 +2572,7 @@ TEST_F(InstructionSelectorTest, Word32AndWithUbfxImmediateForARMv7) {
|
||||
|
||||
TEST_F(InstructionSelectorTest, Word32AndWithBfcImmediateForARMv7) {
|
||||
TRACED_FORRANGE(int32_t, lsb, 0, 31) {
|
||||
TRACED_FORRANGE(int32_t, width, 9, (24 - lsb) - 1) {
|
||||
TRACED_FORRANGE(int32_t, width, 9, (32 - lsb) - 1) {
|
||||
StreamBuilder m(this, MachineType::Int32(), MachineType::Int32());
|
||||
m.Return(m.Word32And(
|
||||
m.Parameter(0),
|
||||
@ -2591,7 +2589,7 @@ TEST_F(InstructionSelectorTest, Word32AndWithBfcImmediateForARMv7) {
|
||||
}
|
||||
}
|
||||
TRACED_FORRANGE(int32_t, lsb, 0, 31) {
|
||||
TRACED_FORRANGE(int32_t, width, 9, (24 - lsb) - 1) {
|
||||
TRACED_FORRANGE(int32_t, width, 9, (32 - lsb) - 1) {
|
||||
StreamBuilder m(this, MachineType::Int32(), MachineType::Int32());
|
||||
m.Return(
|
||||
m.Word32And(m.Int32Constant(~((0xffffffffu >> (32 - width)) << lsb)),
|
||||
@ -2830,11 +2828,8 @@ TEST_F(InstructionSelectorTest, Word32NotWithParameter) {
|
||||
|
||||
|
||||
TEST_F(InstructionSelectorTest, Word32AndWithWord32ShrWithImmediateForARMv7) {
|
||||
TRACED_FORRANGE(int32_t, lsb, 1, 31) {
|
||||
TRACED_FORRANGE(int32_t, lsb, 0, 31) {
|
||||
TRACED_FORRANGE(int32_t, width, 1, 32 - lsb) {
|
||||
if (((width == 8) || (width == 16)) &&
|
||||
((lsb == 8) || (lsb == 16) || (lsb == 24)))
|
||||
continue; // Uxtb/h ror.
|
||||
StreamBuilder m(this, MachineType::Int32(), MachineType::Int32());
|
||||
m.Return(m.Word32And(m.Word32Shr(m.Parameter(0), m.Int32Constant(lsb)),
|
||||
m.Int32Constant(0xffffffffu >> (32 - width))));
|
||||
@ -2846,11 +2841,8 @@ TEST_F(InstructionSelectorTest, Word32AndWithWord32ShrWithImmediateForARMv7) {
|
||||
EXPECT_EQ(width, s.ToInt32(s[0]->InputAt(2)));
|
||||
}
|
||||
}
|
||||
TRACED_FORRANGE(int32_t, lsb, 1, 31) {
|
||||
TRACED_FORRANGE(int32_t, lsb, 0, 31) {
|
||||
TRACED_FORRANGE(int32_t, width, 1, 32 - lsb) {
|
||||
if (((width == 8) || (width == 16)) &&
|
||||
((lsb == 8) || (lsb == 16) || (lsb == 24)))
|
||||
continue; // Uxtb/h ror.
|
||||
StreamBuilder m(this, MachineType::Int32(), MachineType::Int32());
|
||||
m.Return(m.Word32And(m.Int32Constant(0xffffffffu >> (32 - width)),
|
||||
m.Word32Shr(m.Parameter(0), m.Int32Constant(lsb))));
|
||||
@ -2865,62 +2857,6 @@ TEST_F(InstructionSelectorTest, Word32AndWithWord32ShrWithImmediateForARMv7) {
|
||||
}
|
||||
|
||||
|
||||
TEST_F(InstructionSelectorTest, Word32AndWithWord32ShrAnd0xff) {
|
||||
TRACED_FORRANGE(int32_t, shr, 1, 3) {
|
||||
StreamBuilder m(this, MachineType::Int32(), MachineType::Int32());
|
||||
Node* const p0 = m.Parameter(0);
|
||||
Node* const r = m.Word32And(m.Word32Shr(p0, m.Int32Constant(shr * 8)),
|
||||
m.Int32Constant(0xff));
|
||||
m.Return(r);
|
||||
Stream s = m.Build();
|
||||
ASSERT_EQ(1U, s.size());
|
||||
EXPECT_EQ(kArmUxtb, s[0]->arch_opcode());
|
||||
ASSERT_EQ(2U, s[0]->InputCount());
|
||||
EXPECT_EQ(shr * 8, s.ToInt32(s[0]->InputAt(1)));
|
||||
}
|
||||
TRACED_FORRANGE(int32_t, shr, 1, 3) {
|
||||
StreamBuilder m(this, MachineType::Int32(), MachineType::Int32());
|
||||
Node* const p0 = m.Parameter(0);
|
||||
Node* const r = m.Word32And(m.Int32Constant(0xff),
|
||||
m.Word32Shr(p0, m.Int32Constant(shr * 8)));
|
||||
m.Return(r);
|
||||
Stream s = m.Build();
|
||||
ASSERT_EQ(1U, s.size());
|
||||
EXPECT_EQ(kArmUxtb, s[0]->arch_opcode());
|
||||
ASSERT_EQ(2U, s[0]->InputCount());
|
||||
EXPECT_EQ(shr * 8, s.ToInt32(s[0]->InputAt(1)));
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
TEST_F(InstructionSelectorTest, Word32AndWithWord32ShrAnd0xffff) {
|
||||
TRACED_FORRANGE(int32_t, shr, 1, 3) {
|
||||
StreamBuilder m(this, MachineType::Int32(), MachineType::Int32());
|
||||
Node* const p0 = m.Parameter(0);
|
||||
Node* const r = m.Word32And(m.Word32Shr(p0, m.Int32Constant(shr * 8)),
|
||||
m.Int32Constant(0xffff));
|
||||
m.Return(r);
|
||||
Stream s = m.Build();
|
||||
ASSERT_EQ(1U, s.size());
|
||||
EXPECT_EQ(kArmUxth, s[0]->arch_opcode());
|
||||
ASSERT_EQ(2U, s[0]->InputCount());
|
||||
EXPECT_EQ(shr * 8, s.ToInt32(s[0]->InputAt(1)));
|
||||
}
|
||||
TRACED_FORRANGE(int32_t, shr, 1, 3) {
|
||||
StreamBuilder m(this, MachineType::Int32(), MachineType::Int32());
|
||||
Node* const p0 = m.Parameter(0);
|
||||
Node* const r = m.Word32And(m.Int32Constant(0xffff),
|
||||
m.Word32Shr(p0, m.Int32Constant(shr * 8)));
|
||||
m.Return(r);
|
||||
Stream s = m.Build();
|
||||
ASSERT_EQ(1U, s.size());
|
||||
EXPECT_EQ(kArmUxth, s[0]->arch_opcode());
|
||||
ASSERT_EQ(2U, s[0]->InputCount());
|
||||
EXPECT_EQ(shr * 8, s.ToInt32(s[0]->InputAt(1)));
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
TEST_F(InstructionSelectorTest, Word32Clz) {
|
||||
StreamBuilder m(this, MachineType::Uint32(), MachineType::Uint32());
|
||||
Node* const p0 = m.Parameter(0);
|
||||
|
Loading…
Reference in New Issue
Block a user