| Index: src/compiler/ia32/instruction-selector-ia32.cc
|
| diff --git a/src/compiler/ia32/instruction-selector-ia32.cc b/src/compiler/ia32/instruction-selector-ia32.cc
|
| index a5e99c68a8168329b55c2fba5f064c42faa56854..ca33ab07c95d197010af01c4cafcb5dc434fad46 100644
|
| --- a/src/compiler/ia32/instruction-selector-ia32.cc
|
| +++ b/src/compiler/ia32/instruction-selector-ia32.cc
|
| @@ -53,6 +53,178 @@ static AddressingMode AdjustAddressingMode(AddressingMode base_mode,
|
| }
|
|
|
|
|
| +// Fairly intel-specify node matcher used for matching scale factors in
|
| +// addressing modes.
|
| +// Matches nodes of form [x * N] for N in {1,2,4,8}
|
| +class ScaleFactorMatcher : public NodeMatcher {
|
| + public:
|
| + static const int kMatchedFactors[4];
|
| +
|
| + explicit ScaleFactorMatcher(Node* node);
|
| +
|
| + bool Matches() const { return left_ != NULL; }
|
| + int Power() const {
|
| + DCHECK(Matches());
|
| + return power_;
|
| + }
|
| + Node* Left() const {
|
| + DCHECK(Matches());
|
| + return left_;
|
| + }
|
| +
|
| + private:
|
| + Node* left_;
|
| + int power_;
|
| +};
|
| +
|
| +
|
| +// Fairly intel-specify node matcher used for matching index and displacement
|
| +// operands in addressing modes.
|
| +// Matches nodes of form:
|
| +// [x * N]
|
| +// [x * N + K]
|
| +// [x + K]
|
| +// [x] -- fallback case
|
| +// for N in {1,2,4,8} and K int32_t
|
| +class IndexAndDisplacementMatcher : public NodeMatcher {
|
| + public:
|
| + explicit IndexAndDisplacementMatcher(Node* node);
|
| +
|
| + Node* index_node() const { return index_node_; }
|
| + int displacement() const { return displacement_; }
|
| + int power() const { return power_; }
|
| +
|
| + private:
|
| + Node* index_node_;
|
| + int displacement_;
|
| + int power_;
|
| +};
|
| +
|
| +
|
| +// Fairly intel-specify node matcher used for matching multiplies that can be
|
| +// transformed to lea instructions.
|
| +// Matches nodes of form:
|
| +// [x * N]
|
| +// for N in {1,2,3,4,5,8,9}
|
| +class LeaMultiplyMatcher : public NodeMatcher {
|
| + public:
|
| + static const int kMatchedFactors[7];
|
| +
|
| + explicit LeaMultiplyMatcher(Node* node);
|
| +
|
| + bool Matches() const { return left_ != NULL; }
|
| + int Power() const {
|
| + DCHECK(Matches());
|
| + return power_;
|
| + }
|
| + Node* Left() const {
|
| + DCHECK(Matches());
|
| + return left_;
|
| + }
|
| + // Displacement will be either 0 or 1.
|
| + int32_t Displacement() const {
|
| + DCHECK(Matches());
|
| + return displacement_;
|
| + }
|
| +
|
| + private:
|
| + Node* left_;
|
| + int power_;
|
| + int displacement_;
|
| +};
|
| +
|
| +
|
| +const int ScaleFactorMatcher::kMatchedFactors[] = {1, 2, 4, 8};
|
| +
|
| +
|
| +ScaleFactorMatcher::ScaleFactorMatcher(Node* node)
|
| + : NodeMatcher(node), left_(NULL), power_(0) {
|
| + if (opcode() != IrOpcode::kInt32Mul) return;
|
| + // TODO(dcarney): should test 64 bit ints as well.
|
| + Int32BinopMatcher m(this->node());
|
| + if (!m.right().HasValue()) return;
|
| + int32_t value = m.right().Value();
|
| + switch (value) {
|
| + case 8:
|
| + power_++; // Fall through.
|
| + case 4:
|
| + power_++; // Fall through.
|
| + case 2:
|
| + power_++; // Fall through.
|
| + case 1:
|
| + break;
|
| + default:
|
| + return;
|
| + }
|
| + left_ = m.left().node();
|
| +}
|
| +
|
| +
|
| +IndexAndDisplacementMatcher::IndexAndDisplacementMatcher(Node* node)
|
| + : NodeMatcher(node), index_node_(node), displacement_(0), power_(0) {
|
| + if (opcode() == IrOpcode::kInt32Add) {
|
| + Int32BinopMatcher m(this->node());
|
| + if (m.right().HasValue()) {
|
| + displacement_ = m.right().Value();
|
| + index_node_ = m.left().node();
|
| + }
|
| + }
|
| + // Test scale factor.
|
| + ScaleFactorMatcher scale_matcher(index_node_);
|
| + if (scale_matcher.Matches()) {
|
| + index_node_ = scale_matcher.Left();
|
| + power_ = scale_matcher.Power();
|
| + }
|
| +}
|
| +
|
| +
|
| +const int LeaMultiplyMatcher::kMatchedFactors[7] = {1, 2, 3, 4, 5, 8, 9};
|
| +
|
| +
|
| +LeaMultiplyMatcher::LeaMultiplyMatcher(Node* node)
|
| + : NodeMatcher(node), left_(NULL), power_(0), displacement_(0) {
|
| + if (opcode() != IrOpcode::kInt32Mul && opcode() != IrOpcode::kInt64Mul) {
|
| + return;
|
| + }
|
| + int64_t value;
|
| + Node* left = NULL;
|
| + {
|
| + Int32BinopMatcher m(this->node());
|
| + if (m.right().HasValue()) {
|
| + value = m.right().Value();
|
| + left = m.left().node();
|
| + } else {
|
| + Int64BinopMatcher m(this->node());
|
| + if (m.right().HasValue()) {
|
| + value = m.right().Value();
|
| + left = m.left().node();
|
| + } else {
|
| + return;
|
| + }
|
| + }
|
| + }
|
| + switch (value) {
|
| + case 9:
|
| + case 8:
|
| + power_++; // Fall through.
|
| + case 5:
|
| + case 4:
|
| + power_++; // Fall through.
|
| + case 3:
|
| + case 2:
|
| + power_++; // Fall through.
|
| + case 1:
|
| + break;
|
| + default:
|
| + return;
|
| + }
|
| + if (!base::bits::IsPowerOfTwo64(value)) {
|
| + displacement_ = 1;
|
| + }
|
| + left_ = left;
|
| +}
|
| +
|
| +
|
| class AddressingModeMatcher {
|
| public:
|
| AddressingModeMatcher(IA32OperandGenerator* g, Node* base, Node* index)
|
|
|