blob: e2ed25777753aca6bc085603fa2f0a562be6e40a [file] [log] [blame]
Artem Udovichenko4a0dad62016-01-26 12:28:31 +03001/*
2 * Copyright (C) 2015 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#ifndef ART_COMPILER_OPTIMIZING_INSTRUCTION_SIMPLIFIER_ARM_H_
18#define ART_COMPILER_OPTIMIZING_INSTRUCTION_SIMPLIFIER_ARM_H_
19
20#include "nodes.h"
21#include "optimization.h"
22
23namespace art {
24namespace arm {
25
26class InstructionSimplifierArmVisitor : public HGraphVisitor {
27 public:
28 InstructionSimplifierArmVisitor(HGraph* graph, OptimizingCompilerStats* stats)
29 : HGraphVisitor(graph), stats_(stats) {}
30
31 private:
32 void RecordSimplification() {
33 if (stats_ != nullptr) {
34 stats_->RecordStat(kInstructionSimplificationsArch);
35 }
36 }
37
Anton Kirilov74234da2017-01-13 14:42:47 +000038 bool TryMergeIntoUsersShifterOperand(HInstruction* instruction);
39 bool TryMergeIntoShifterOperand(HInstruction* use, HInstruction* bitfield_op, bool do_merge);
40 bool CanMergeIntoShifterOperand(HInstruction* use, HInstruction* bitfield_op) {
41 return TryMergeIntoShifterOperand(use, bitfield_op, /* do_merge */ false);
42 }
43 bool MergeIntoShifterOperand(HInstruction* use, HInstruction* bitfield_op) {
44 DCHECK(CanMergeIntoShifterOperand(use, bitfield_op));
45 return TryMergeIntoShifterOperand(use, bitfield_op, /* do_merge */ true);
46 }
47
48 /**
49 * This simplifier uses a special-purpose BB visitor.
50 * (1) No need to visit Phi nodes.
51 * (2) Since statements can be removed in a "forward" fashion,
52 * the visitor should test if each statement is still there.
53 */
54 void VisitBasicBlock(HBasicBlock* block) OVERRIDE {
55 // TODO: fragile iteration, provide more robust iterators?
56 for (HInstructionIterator it(block->GetInstructions()); !it.Done(); it.Advance()) {
57 HInstruction* instruction = it.Current();
58 if (instruction->IsInBlock()) {
59 instruction->Accept(this);
60 }
61 }
62 }
63
Artem Serov7fc63502016-02-09 17:15:29 +000064 void VisitAnd(HAnd* instruction) OVERRIDE;
Artem Serov328429f2016-07-06 16:23:04 +010065 void VisitArrayGet(HArrayGet* instruction) OVERRIDE;
66 void VisitArraySet(HArraySet* instruction) OVERRIDE;
Anton Kirilov74234da2017-01-13 14:42:47 +000067 void VisitMul(HMul* instruction) OVERRIDE;
68 void VisitOr(HOr* instruction) OVERRIDE;
69 void VisitShl(HShl* instruction) OVERRIDE;
70 void VisitShr(HShr* instruction) OVERRIDE;
71 void VisitTypeConversion(HTypeConversion* instruction) OVERRIDE;
72 void VisitUShr(HUShr* instruction) OVERRIDE;
Artem Udovichenko4a0dad62016-01-26 12:28:31 +030073
74 OptimizingCompilerStats* stats_;
75};
76
77
78class InstructionSimplifierArm : public HOptimization {
79 public:
80 InstructionSimplifierArm(HGraph* graph, OptimizingCompilerStats* stats)
Roland Levillain7fa7cf52016-11-04 14:10:29 +000081 : HOptimization(graph, kInstructionSimplifierArmPassName, stats) {}
Wojciech Staszkiewicz5319d3c2016-08-01 17:48:59 -070082
83 static constexpr const char* kInstructionSimplifierArmPassName = "instruction_simplifier_arm";
Artem Udovichenko4a0dad62016-01-26 12:28:31 +030084
85 void Run() OVERRIDE {
86 InstructionSimplifierArmVisitor visitor(graph_, stats_);
87 visitor.VisitReversePostOrder();
88 }
89};
90
91} // namespace arm
92} // namespace art
93
94#endif // ART_COMPILER_OPTIMIZING_INSTRUCTION_SIMPLIFIER_ARM_H_