Commit 1514287a authored by Benedikt Meurer's avatar Benedikt Meurer

[arm] Fix UBFX detection.

R=svenpanne@chromium.org

Review URL: https://codereview.chromium.org/725063002

Cr-Commit-Position: refs/heads/master@{#25344}
parent 05e92143
...@@ -368,6 +368,16 @@ void EmitBic(InstructionSelector* selector, Node* node, Node* left, ...@@ -368,6 +368,16 @@ void EmitBic(InstructionSelector* selector, Node* node, Node* left,
g.UseRegister(right)); g.UseRegister(right));
} }
void EmitUbfx(InstructionSelector* selector, Node* node, Node* left,
uint32_t lsb, uint32_t width) {
DCHECK_LE(1, width);
DCHECK_LE(width, 32 - lsb);
ArmOperandGenerator g(selector);
selector->Emit(kArmUbfx, g.DefineAsRegister(node), g.UseRegister(left),
g.TempImmediate(lsb), g.TempImmediate(width));
}
} // namespace } // namespace
...@@ -398,15 +408,16 @@ void InstructionSelector::VisitWord32And(Node* node) { ...@@ -398,15 +408,16 @@ void InstructionSelector::VisitWord32And(Node* node) {
if (m.left().IsWord32Shr()) { if (m.left().IsWord32Shr()) {
Int32BinopMatcher mleft(m.left().node()); Int32BinopMatcher mleft(m.left().node());
if (mleft.right().IsInRange(0, 31)) { if (mleft.right().IsInRange(0, 31)) {
Emit(kArmUbfx, g.DefineAsRegister(node), // UBFX cannot extract bits past the register size, however since
g.UseRegister(mleft.left().node()), // shifting the original value would have introduced some zeros we can
g.UseImmediate(mleft.right().node()), g.TempImmediate(width)); // still use UBFX with a smaller mask and the remaining bits will be
return; // zeros.
uint32_t const lsb = mleft.right().Value();
return EmitUbfx(this, node, mleft.left().node(), lsb,
std::min(width, 32 - lsb));
} }
} }
Emit(kArmUbfx, g.DefineAsRegister(node), g.UseRegister(m.left().node()), return EmitUbfx(this, node, m.left().node(), 0, width);
g.TempImmediate(0), g.TempImmediate(width));
return;
} }
// Try to interpret this AND as BIC. // Try to interpret this AND as BIC.
if (g.CanBeImmediate(~value)) { if (g.CanBeImmediate(~value)) {
...@@ -523,10 +534,7 @@ void InstructionSelector::VisitWord32Shr(Node* node) { ...@@ -523,10 +534,7 @@ void InstructionSelector::VisitWord32Shr(Node* node) {
uint32_t msb = base::bits::CountLeadingZeros32(value); uint32_t msb = base::bits::CountLeadingZeros32(value);
if (msb + width + lsb == 32) { if (msb + width + lsb == 32) {
DCHECK_EQ(lsb, base::bits::CountTrailingZeros32(value)); DCHECK_EQ(lsb, base::bits::CountTrailingZeros32(value));
Emit(kArmUbfx, g.DefineAsRegister(node), return EmitUbfx(this, node, mleft.left().node(), lsb, width);
g.UseRegister(mleft.left().node()), g.TempImmediate(lsb),
g.TempImmediate(width));
return;
} }
} }
} }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment