xref: /aosp_15_r20/art/compiler/optimizing/loop_optimization_test.cc (revision 795d594fd825385562da6b089ea9b2033f3abf5a)
1*795d594fSAndroid Build Coastguard Worker /*
2*795d594fSAndroid Build Coastguard Worker  * Copyright (C) 2016 The Android Open Source Project
3*795d594fSAndroid Build Coastguard Worker  *
4*795d594fSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*795d594fSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*795d594fSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*795d594fSAndroid Build Coastguard Worker  *
8*795d594fSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*795d594fSAndroid Build Coastguard Worker  *
10*795d594fSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*795d594fSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*795d594fSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*795d594fSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*795d594fSAndroid Build Coastguard Worker  * limitations under the License.
15*795d594fSAndroid Build Coastguard Worker  */
16*795d594fSAndroid Build Coastguard Worker 
17*795d594fSAndroid Build Coastguard Worker #include "android-base/logging.h"
18*795d594fSAndroid Build Coastguard Worker #include "base/macros.h"
19*795d594fSAndroid Build Coastguard Worker #include "code_generator.h"
20*795d594fSAndroid Build Coastguard Worker #include "driver/compiler_options.h"
21*795d594fSAndroid Build Coastguard Worker #include "loop_optimization.h"
22*795d594fSAndroid Build Coastguard Worker #include "optimizing/data_type.h"
23*795d594fSAndroid Build Coastguard Worker #include "optimizing/nodes.h"
24*795d594fSAndroid Build Coastguard Worker #include "optimizing_unit_test.h"
25*795d594fSAndroid Build Coastguard Worker 
26*795d594fSAndroid Build Coastguard Worker namespace art HIDDEN {
27*795d594fSAndroid Build Coastguard Worker 
28*795d594fSAndroid Build Coastguard Worker // Base class for loop optimization tests.
29*795d594fSAndroid Build Coastguard Worker class LoopOptimizationTestBase : public OptimizingUnitTest {
30*795d594fSAndroid Build Coastguard Worker  protected:
SetUp()31*795d594fSAndroid Build Coastguard Worker   void SetUp() override {
32*795d594fSAndroid Build Coastguard Worker     OptimizingUnitTest::SetUp();
33*795d594fSAndroid Build Coastguard Worker 
34*795d594fSAndroid Build Coastguard Worker     BuildGraph();
35*795d594fSAndroid Build Coastguard Worker     iva_  = new (GetAllocator()) HInductionVarAnalysis(graph_);
36*795d594fSAndroid Build Coastguard Worker     if (compiler_options_ == nullptr) {
37*795d594fSAndroid Build Coastguard Worker       compiler_options_ = CommonCompilerTest::CreateCompilerOptions(kRuntimeISA, "default");
38*795d594fSAndroid Build Coastguard Worker       DCHECK(compiler_options_ != nullptr);
39*795d594fSAndroid Build Coastguard Worker     }
40*795d594fSAndroid Build Coastguard Worker     codegen_ = CodeGenerator::Create(graph_, *compiler_options_);
41*795d594fSAndroid Build Coastguard Worker     DCHECK(codegen_.get() != nullptr);
42*795d594fSAndroid Build Coastguard Worker     loop_opt_ = new (GetAllocator()) HLoopOptimization(
43*795d594fSAndroid Build Coastguard Worker         graph_, *codegen_.get(), iva_, /* stats= */ nullptr);
44*795d594fSAndroid Build Coastguard Worker   }
45*795d594fSAndroid Build Coastguard Worker 
TearDown()46*795d594fSAndroid Build Coastguard Worker   void TearDown() override {
47*795d594fSAndroid Build Coastguard Worker     codegen_.reset();
48*795d594fSAndroid Build Coastguard Worker     compiler_options_.reset();
49*795d594fSAndroid Build Coastguard Worker     graph_ = nullptr;
50*795d594fSAndroid Build Coastguard Worker     ResetPoolAndAllocator();
51*795d594fSAndroid Build Coastguard Worker     OptimizingUnitTest::TearDown();
52*795d594fSAndroid Build Coastguard Worker   }
53*795d594fSAndroid Build Coastguard Worker 
54*795d594fSAndroid Build Coastguard Worker   virtual void BuildGraph() = 0;
55*795d594fSAndroid Build Coastguard Worker 
56*795d594fSAndroid Build Coastguard Worker   // Run loop optimization and optionally check the graph.
PerformAnalysis(bool run_checker)57*795d594fSAndroid Build Coastguard Worker   void PerformAnalysis(bool run_checker) {
58*795d594fSAndroid Build Coastguard Worker     graph_->BuildDominatorTree();
59*795d594fSAndroid Build Coastguard Worker 
60*795d594fSAndroid Build Coastguard Worker     // Check the graph is valid before loop optimization.
61*795d594fSAndroid Build Coastguard Worker     std::ostringstream oss;
62*795d594fSAndroid Build Coastguard Worker     if (run_checker) {
63*795d594fSAndroid Build Coastguard Worker       ASSERT_TRUE(CheckGraph(oss)) << oss.str();
64*795d594fSAndroid Build Coastguard Worker     }
65*795d594fSAndroid Build Coastguard Worker 
66*795d594fSAndroid Build Coastguard Worker     iva_->Run();
67*795d594fSAndroid Build Coastguard Worker     loop_opt_->Run();
68*795d594fSAndroid Build Coastguard Worker 
69*795d594fSAndroid Build Coastguard Worker     // Check the graph is valid after loop optimization.
70*795d594fSAndroid Build Coastguard Worker     if (run_checker) {
71*795d594fSAndroid Build Coastguard Worker       ASSERT_TRUE(CheckGraph(oss)) << oss.str();
72*795d594fSAndroid Build Coastguard Worker     }
73*795d594fSAndroid Build Coastguard Worker   }
74*795d594fSAndroid Build Coastguard Worker 
75*795d594fSAndroid Build Coastguard Worker   // General building fields.
76*795d594fSAndroid Build Coastguard Worker   std::unique_ptr<CompilerOptions> compiler_options_;
77*795d594fSAndroid Build Coastguard Worker   std::unique_ptr<CodeGenerator> codegen_;
78*795d594fSAndroid Build Coastguard Worker   HInductionVarAnalysis* iva_;
79*795d594fSAndroid Build Coastguard Worker   HLoopOptimization* loop_opt_;
80*795d594fSAndroid Build Coastguard Worker 
81*795d594fSAndroid Build Coastguard Worker   HBasicBlock* return_block_;
82*795d594fSAndroid Build Coastguard Worker 
83*795d594fSAndroid Build Coastguard Worker   HInstruction* parameter_;
84*795d594fSAndroid Build Coastguard Worker };
85*795d594fSAndroid Build Coastguard Worker 
86*795d594fSAndroid Build Coastguard Worker /**
87*795d594fSAndroid Build Coastguard Worker  * Fixture class for the loop optimization tests. These unit tests mostly focus
88*795d594fSAndroid Build Coastguard Worker  * on constructing the loop hierarchy. Checker tests are also used to test
89*795d594fSAndroid Build Coastguard Worker  * specific optimizations.
90*795d594fSAndroid Build Coastguard Worker  */
91*795d594fSAndroid Build Coastguard Worker class LoopOptimizationTest : public LoopOptimizationTestBase {
92*795d594fSAndroid Build Coastguard Worker  protected:
~LoopOptimizationTest()93*795d594fSAndroid Build Coastguard Worker   virtual ~LoopOptimizationTest() {}
94*795d594fSAndroid Build Coastguard Worker 
95*795d594fSAndroid Build Coastguard Worker   /** Constructs bare minimum graph. */
BuildGraph()96*795d594fSAndroid Build Coastguard Worker   void BuildGraph() override {
97*795d594fSAndroid Build Coastguard Worker     return_block_ = InitEntryMainExitGraph();
98*795d594fSAndroid Build Coastguard Worker     graph_->SetNumberOfVRegs(1);
99*795d594fSAndroid Build Coastguard Worker     parameter_ = MakeParam(DataType::Type::kInt32);
100*795d594fSAndroid Build Coastguard Worker   }
101*795d594fSAndroid Build Coastguard Worker 
102*795d594fSAndroid Build Coastguard Worker   /** Adds a loop nest at given position before successor. */
AddLoop(HBasicBlock * position,HBasicBlock * successor)103*795d594fSAndroid Build Coastguard Worker   HBasicBlock* AddLoop(HBasicBlock* position, HBasicBlock* successor) {
104*795d594fSAndroid Build Coastguard Worker     HBasicBlock* header = new (GetAllocator()) HBasicBlock(graph_);
105*795d594fSAndroid Build Coastguard Worker     HBasicBlock* body = new (GetAllocator()) HBasicBlock(graph_);
106*795d594fSAndroid Build Coastguard Worker     graph_->AddBlock(header);
107*795d594fSAndroid Build Coastguard Worker     graph_->AddBlock(body);
108*795d594fSAndroid Build Coastguard Worker     // Control flow.
109*795d594fSAndroid Build Coastguard Worker     position->ReplaceSuccessor(successor, header);
110*795d594fSAndroid Build Coastguard Worker     header->AddSuccessor(body);
111*795d594fSAndroid Build Coastguard Worker     header->AddSuccessor(successor);
112*795d594fSAndroid Build Coastguard Worker     MakeIf(header, parameter_);
113*795d594fSAndroid Build Coastguard Worker     body->AddSuccessor(header);
114*795d594fSAndroid Build Coastguard Worker     MakeGoto(body);
115*795d594fSAndroid Build Coastguard Worker     return header;
116*795d594fSAndroid Build Coastguard Worker   }
117*795d594fSAndroid Build Coastguard Worker 
118*795d594fSAndroid Build Coastguard Worker   /** Constructs string representation of computed loop hierarchy. */
LoopStructure()119*795d594fSAndroid Build Coastguard Worker   std::string LoopStructure() {
120*795d594fSAndroid Build Coastguard Worker     return LoopStructureRecurse(loop_opt_->top_loop_);
121*795d594fSAndroid Build Coastguard Worker   }
122*795d594fSAndroid Build Coastguard Worker 
123*795d594fSAndroid Build Coastguard Worker   // Helper method
LoopStructureRecurse(HLoopOptimization::LoopNode * node)124*795d594fSAndroid Build Coastguard Worker   std::string LoopStructureRecurse(HLoopOptimization::LoopNode* node) {
125*795d594fSAndroid Build Coastguard Worker     std::string s;
126*795d594fSAndroid Build Coastguard Worker     for ( ; node != nullptr; node = node->next) {
127*795d594fSAndroid Build Coastguard Worker       s.append("[");
128*795d594fSAndroid Build Coastguard Worker       s.append(LoopStructureRecurse(node->inner));
129*795d594fSAndroid Build Coastguard Worker       s.append("]");
130*795d594fSAndroid Build Coastguard Worker     }
131*795d594fSAndroid Build Coastguard Worker     return s;
132*795d594fSAndroid Build Coastguard Worker   }
133*795d594fSAndroid Build Coastguard Worker };
134*795d594fSAndroid Build Coastguard Worker 
135*795d594fSAndroid Build Coastguard Worker #ifdef ART_ENABLE_CODEGEN_arm64
136*795d594fSAndroid Build Coastguard Worker // Unit tests for predicated vectorization.
137*795d594fSAndroid Build Coastguard Worker class PredicatedSimdLoopOptimizationTest : public LoopOptimizationTestBase {
138*795d594fSAndroid Build Coastguard Worker  protected:
SetUp()139*795d594fSAndroid Build Coastguard Worker   void SetUp() override {
140*795d594fSAndroid Build Coastguard Worker     // Predicated SIMD is only supported by SVE on Arm64.
141*795d594fSAndroid Build Coastguard Worker     compiler_options_ = CommonCompilerTest::CreateCompilerOptions(InstructionSet::kArm64,
142*795d594fSAndroid Build Coastguard Worker                                                                   "default",
143*795d594fSAndroid Build Coastguard Worker                                                                   "sve");
144*795d594fSAndroid Build Coastguard Worker     LoopOptimizationTestBase::SetUp();
145*795d594fSAndroid Build Coastguard Worker   }
146*795d594fSAndroid Build Coastguard Worker 
~PredicatedSimdLoopOptimizationTest()147*795d594fSAndroid Build Coastguard Worker   virtual ~PredicatedSimdLoopOptimizationTest() {}
148*795d594fSAndroid Build Coastguard Worker 
149*795d594fSAndroid Build Coastguard Worker   // Constructs a graph with a diamond loop which should be vectorizable with predicated
150*795d594fSAndroid Build Coastguard Worker   // vectorization. This graph includes a basic loop induction (consisting of Phi, Add, If and
151*795d594fSAndroid Build Coastguard Worker   // SuspendCheck instructions) to control the loop as well as an if comparison (consisting of
152*795d594fSAndroid Build Coastguard Worker   // Parameter, GreaterThanOrEqual and If instructions) to control the diamond loop.
153*795d594fSAndroid Build Coastguard Worker   //
154*795d594fSAndroid Build Coastguard Worker   //                       entry
155*795d594fSAndroid Build Coastguard Worker   //                         |
156*795d594fSAndroid Build Coastguard Worker   //                      preheader
157*795d594fSAndroid Build Coastguard Worker   //                         |
158*795d594fSAndroid Build Coastguard Worker   //  return <------------ header <----------------+
159*795d594fSAndroid Build Coastguard Worker   //     |                   |                     |
160*795d594fSAndroid Build Coastguard Worker   //   exit             diamond_top                |
161*795d594fSAndroid Build Coastguard Worker   //                       /   \                   |
162*795d594fSAndroid Build Coastguard Worker   //            diamond_true  diamond_false        |
163*795d594fSAndroid Build Coastguard Worker   //                       \   /                   |
164*795d594fSAndroid Build Coastguard Worker   //                     back_edge                 |
165*795d594fSAndroid Build Coastguard Worker   //                         |                     |
166*795d594fSAndroid Build Coastguard Worker   //                         +---------------------+
BuildGraph()167*795d594fSAndroid Build Coastguard Worker   void BuildGraph() override {
168*795d594fSAndroid Build Coastguard Worker     return_block_ = InitEntryMainExitGraphWithReturnVoid();
169*795d594fSAndroid Build Coastguard Worker     HBasicBlock* back_edge;
170*795d594fSAndroid Build Coastguard Worker     std::tie(std::ignore, header_, back_edge) = CreateWhileLoop(return_block_);
171*795d594fSAndroid Build Coastguard Worker     std::tie(diamond_top_, diamond_true_, std::ignore) = CreateDiamondPattern(back_edge);
172*795d594fSAndroid Build Coastguard Worker 
173*795d594fSAndroid Build Coastguard Worker     parameter_ = MakeParam(DataType::Type::kInt32);
174*795d594fSAndroid Build Coastguard Worker     std::tie(phi_, std::ignore) = MakeLinearLoopVar(header_, back_edge, 0, 1);
175*795d594fSAndroid Build Coastguard Worker     MakeSuspendCheck(header_);
176*795d594fSAndroid Build Coastguard Worker     HInstruction* trip = MakeCondition(header_,
177*795d594fSAndroid Build Coastguard Worker                                        kCondGE,
178*795d594fSAndroid Build Coastguard Worker                                        phi_,
179*795d594fSAndroid Build Coastguard Worker                                        graph_->GetIntConstant(kArm64DefaultSVEVectorLength));
180*795d594fSAndroid Build Coastguard Worker     MakeIf(header_, trip);
181*795d594fSAndroid Build Coastguard Worker     diamond_hif_ = MakeIf(diamond_top_, parameter_);
182*795d594fSAndroid Build Coastguard Worker   }
183*795d594fSAndroid Build Coastguard Worker 
184*795d594fSAndroid Build Coastguard Worker   // Add an ArraySet to the loop which will be vectorized, thus setting the type of vector
185*795d594fSAndroid Build Coastguard Worker   // instructions in the graph to the given vector_type. This needs to be called to ensure the loop
186*795d594fSAndroid Build Coastguard Worker   // is not simplified by SimplifyInduction or SimplifyBlocks before vectorization.
AddArraySetToLoop(DataType::Type vector_type)187*795d594fSAndroid Build Coastguard Worker   void AddArraySetToLoop(DataType::Type vector_type) {
188*795d594fSAndroid Build Coastguard Worker     // Ensure the data type is a java type so it can be stored in a TypeField. The actual type does
189*795d594fSAndroid Build Coastguard Worker     // not matter as long as the size is the same so it can still be vectorized.
190*795d594fSAndroid Build Coastguard Worker     DataType::Type new_type = DataType::SignedIntegralTypeFromSize(DataType::Size(vector_type));
191*795d594fSAndroid Build Coastguard Worker 
192*795d594fSAndroid Build Coastguard Worker     // Add an array set to prevent the loop from being optimized away before vectorization.
193*795d594fSAndroid Build Coastguard Worker     // Note: This uses an integer parameter and not an array reference to avoid the difficulties in
194*795d594fSAndroid Build Coastguard Worker     // allocating an array. The instruction is still treated as a valid ArraySet by loop
195*795d594fSAndroid Build Coastguard Worker     // optimization.
196*795d594fSAndroid Build Coastguard Worker     diamond_true_->AddInstruction(new (GetAllocator()) HArraySet(parameter_,
197*795d594fSAndroid Build Coastguard Worker                                                                  phi_,
198*795d594fSAndroid Build Coastguard Worker                                                                  graph_->GetIntConstant(1),
199*795d594fSAndroid Build Coastguard Worker                                                                  new_type,
200*795d594fSAndroid Build Coastguard Worker                                                                  /* dex_pc= */ 0));
201*795d594fSAndroid Build Coastguard Worker   }
202*795d594fSAndroid Build Coastguard Worker 
203*795d594fSAndroid Build Coastguard Worker   // Replace the input of diamond_hif_ with a new condition of the given types.
ReplaceIfCondition(DataType::Type l_type,DataType::Type r_type,HBasicBlock * condition_block,IfCondition cond)204*795d594fSAndroid Build Coastguard Worker   void ReplaceIfCondition(DataType::Type l_type,
205*795d594fSAndroid Build Coastguard Worker                           DataType::Type r_type,
206*795d594fSAndroid Build Coastguard Worker                           HBasicBlock* condition_block,
207*795d594fSAndroid Build Coastguard Worker                           IfCondition cond) {
208*795d594fSAndroid Build Coastguard Worker     AddArraySetToLoop(l_type);
209*795d594fSAndroid Build Coastguard Worker     HInstruction* l_param = MakeParam(l_type);
210*795d594fSAndroid Build Coastguard Worker     HInstruction* r_param = MakeParam(r_type);
211*795d594fSAndroid Build Coastguard Worker     HCondition* condition = MakeCondition(condition_block, cond, l_param, r_param);
212*795d594fSAndroid Build Coastguard Worker     diamond_hif_->ReplaceInput(condition, 0);
213*795d594fSAndroid Build Coastguard Worker   }
214*795d594fSAndroid Build Coastguard Worker 
215*795d594fSAndroid Build Coastguard Worker   // Is loop optimization able to vectorize predicated code?
IsPredicatedVectorizationSupported()216*795d594fSAndroid Build Coastguard Worker   bool IsPredicatedVectorizationSupported() {
217*795d594fSAndroid Build Coastguard Worker     // Mirror the check guarding TryVectorizePredicated in TryOptimizeInnerLoopFinite.
218*795d594fSAndroid Build Coastguard Worker     return kForceTryPredicatedSIMD && loop_opt_->IsInPredicatedVectorizationMode();
219*795d594fSAndroid Build Coastguard Worker   }
220*795d594fSAndroid Build Coastguard Worker 
221*795d594fSAndroid Build Coastguard Worker   HBasicBlock* header_;
222*795d594fSAndroid Build Coastguard Worker   HBasicBlock* diamond_top_;
223*795d594fSAndroid Build Coastguard Worker   HBasicBlock* diamond_true_;
224*795d594fSAndroid Build Coastguard Worker 
225*795d594fSAndroid Build Coastguard Worker   HPhi* phi_;
226*795d594fSAndroid Build Coastguard Worker   HIf* diamond_hif_;
227*795d594fSAndroid Build Coastguard Worker };
228*795d594fSAndroid Build Coastguard Worker 
229*795d594fSAndroid Build Coastguard Worker #endif  // ART_ENABLE_CODEGEN_arm64
230*795d594fSAndroid Build Coastguard Worker 
231*795d594fSAndroid Build Coastguard Worker //
232*795d594fSAndroid Build Coastguard Worker // The actual tests.
233*795d594fSAndroid Build Coastguard Worker //
234*795d594fSAndroid Build Coastguard Worker 
235*795d594fSAndroid Build Coastguard Worker // Loop structure tests can't run the graph checker because they don't create valid graphs.
236*795d594fSAndroid Build Coastguard Worker 
TEST_F(LoopOptimizationTest,NoLoops)237*795d594fSAndroid Build Coastguard Worker TEST_F(LoopOptimizationTest, NoLoops) {
238*795d594fSAndroid Build Coastguard Worker   PerformAnalysis(/*run_checker=*/ false);
239*795d594fSAndroid Build Coastguard Worker   EXPECT_EQ("", LoopStructure());
240*795d594fSAndroid Build Coastguard Worker }
241*795d594fSAndroid Build Coastguard Worker 
TEST_F(LoopOptimizationTest,SingleLoop)242*795d594fSAndroid Build Coastguard Worker TEST_F(LoopOptimizationTest, SingleLoop) {
243*795d594fSAndroid Build Coastguard Worker   AddLoop(entry_block_, return_block_);
244*795d594fSAndroid Build Coastguard Worker   PerformAnalysis(/*run_checker=*/ false);
245*795d594fSAndroid Build Coastguard Worker   EXPECT_EQ("[]", LoopStructure());
246*795d594fSAndroid Build Coastguard Worker }
247*795d594fSAndroid Build Coastguard Worker 
TEST_F(LoopOptimizationTest,LoopNest10)248*795d594fSAndroid Build Coastguard Worker TEST_F(LoopOptimizationTest, LoopNest10) {
249*795d594fSAndroid Build Coastguard Worker   HBasicBlock* b = entry_block_;
250*795d594fSAndroid Build Coastguard Worker   HBasicBlock* s = return_block_;
251*795d594fSAndroid Build Coastguard Worker   for (int i = 0; i < 10; i++) {
252*795d594fSAndroid Build Coastguard Worker     s = AddLoop(b, s);
253*795d594fSAndroid Build Coastguard Worker     b = s->GetSuccessors()[0];
254*795d594fSAndroid Build Coastguard Worker   }
255*795d594fSAndroid Build Coastguard Worker   PerformAnalysis(/*run_checker=*/ false);
256*795d594fSAndroid Build Coastguard Worker   EXPECT_EQ("[[[[[[[[[[]]]]]]]]]]", LoopStructure());
257*795d594fSAndroid Build Coastguard Worker }
258*795d594fSAndroid Build Coastguard Worker 
TEST_F(LoopOptimizationTest,LoopSequence10)259*795d594fSAndroid Build Coastguard Worker TEST_F(LoopOptimizationTest, LoopSequence10) {
260*795d594fSAndroid Build Coastguard Worker   HBasicBlock* b = entry_block_;
261*795d594fSAndroid Build Coastguard Worker   HBasicBlock* s = return_block_;
262*795d594fSAndroid Build Coastguard Worker   for (int i = 0; i < 10; i++) {
263*795d594fSAndroid Build Coastguard Worker     b = AddLoop(b, s);
264*795d594fSAndroid Build Coastguard Worker     s = b->GetSuccessors()[1];
265*795d594fSAndroid Build Coastguard Worker   }
266*795d594fSAndroid Build Coastguard Worker   PerformAnalysis(/*run_checker=*/ false);
267*795d594fSAndroid Build Coastguard Worker   EXPECT_EQ("[][][][][][][][][][]", LoopStructure());
268*795d594fSAndroid Build Coastguard Worker }
269*795d594fSAndroid Build Coastguard Worker 
TEST_F(LoopOptimizationTest,LoopSequenceOfNests)270*795d594fSAndroid Build Coastguard Worker TEST_F(LoopOptimizationTest, LoopSequenceOfNests) {
271*795d594fSAndroid Build Coastguard Worker   HBasicBlock* b = entry_block_;
272*795d594fSAndroid Build Coastguard Worker   HBasicBlock* s = return_block_;
273*795d594fSAndroid Build Coastguard Worker   for (int i = 0; i < 10; i++) {
274*795d594fSAndroid Build Coastguard Worker     b = AddLoop(b, s);
275*795d594fSAndroid Build Coastguard Worker     s = b->GetSuccessors()[1];
276*795d594fSAndroid Build Coastguard Worker     HBasicBlock* bi = b->GetSuccessors()[0];
277*795d594fSAndroid Build Coastguard Worker     HBasicBlock* si = b;
278*795d594fSAndroid Build Coastguard Worker     for (int j = 0; j < i; j++) {
279*795d594fSAndroid Build Coastguard Worker       si = AddLoop(bi, si);
280*795d594fSAndroid Build Coastguard Worker       bi = si->GetSuccessors()[0];
281*795d594fSAndroid Build Coastguard Worker     }
282*795d594fSAndroid Build Coastguard Worker   }
283*795d594fSAndroid Build Coastguard Worker   PerformAnalysis(/*run_checker=*/ false);
284*795d594fSAndroid Build Coastguard Worker   EXPECT_EQ("[]"
285*795d594fSAndroid Build Coastguard Worker             "[[]]"
286*795d594fSAndroid Build Coastguard Worker             "[[[]]]"
287*795d594fSAndroid Build Coastguard Worker             "[[[[]]]]"
288*795d594fSAndroid Build Coastguard Worker             "[[[[[]]]]]"
289*795d594fSAndroid Build Coastguard Worker             "[[[[[[]]]]]]"
290*795d594fSAndroid Build Coastguard Worker             "[[[[[[[]]]]]]]"
291*795d594fSAndroid Build Coastguard Worker             "[[[[[[[[]]]]]]]]"
292*795d594fSAndroid Build Coastguard Worker             "[[[[[[[[[]]]]]]]]]"
293*795d594fSAndroid Build Coastguard Worker             "[[[[[[[[[[]]]]]]]]]]",
294*795d594fSAndroid Build Coastguard Worker             LoopStructure());
295*795d594fSAndroid Build Coastguard Worker }
296*795d594fSAndroid Build Coastguard Worker 
TEST_F(LoopOptimizationTest,LoopNestWithSequence)297*795d594fSAndroid Build Coastguard Worker TEST_F(LoopOptimizationTest, LoopNestWithSequence) {
298*795d594fSAndroid Build Coastguard Worker   HBasicBlock* b = entry_block_;
299*795d594fSAndroid Build Coastguard Worker   HBasicBlock* s = return_block_;
300*795d594fSAndroid Build Coastguard Worker   for (int i = 0; i < 10; i++) {
301*795d594fSAndroid Build Coastguard Worker     s = AddLoop(b, s);
302*795d594fSAndroid Build Coastguard Worker     b = s->GetSuccessors()[0];
303*795d594fSAndroid Build Coastguard Worker   }
304*795d594fSAndroid Build Coastguard Worker   b = s;
305*795d594fSAndroid Build Coastguard Worker   s = b->GetSuccessors()[1];
306*795d594fSAndroid Build Coastguard Worker   for (int i = 0; i < 9; i++) {
307*795d594fSAndroid Build Coastguard Worker     b = AddLoop(b, s);
308*795d594fSAndroid Build Coastguard Worker     s = b->GetSuccessors()[1];
309*795d594fSAndroid Build Coastguard Worker   }
310*795d594fSAndroid Build Coastguard Worker   PerformAnalysis(/*run_checker=*/ false);
311*795d594fSAndroid Build Coastguard Worker   EXPECT_EQ("[[[[[[[[[[][][][][][][][][][]]]]]]]]]]", LoopStructure());
312*795d594fSAndroid Build Coastguard Worker }
313*795d594fSAndroid Build Coastguard Worker 
314*795d594fSAndroid Build Coastguard Worker // Check that SimplifyLoop() doesn't invalidate data flow when ordering loop headers'
315*795d594fSAndroid Build Coastguard Worker // predecessors.
316*795d594fSAndroid Build Coastguard Worker //
317*795d594fSAndroid Build Coastguard Worker // This is a test for nodes.cc functionality - HGraph::SimplifyLoop.
TEST_F(LoopOptimizationTest,SimplifyLoopReoderPredecessors)318*795d594fSAndroid Build Coastguard Worker TEST_F(LoopOptimizationTest, SimplifyLoopReoderPredecessors) {
319*795d594fSAndroid Build Coastguard Worker   // Can't use AddLoop as we want special order for blocks predecessors.
320*795d594fSAndroid Build Coastguard Worker   HBasicBlock* header = new (GetAllocator()) HBasicBlock(graph_);
321*795d594fSAndroid Build Coastguard Worker   HBasicBlock* body = new (GetAllocator()) HBasicBlock(graph_);
322*795d594fSAndroid Build Coastguard Worker   graph_->AddBlock(header);
323*795d594fSAndroid Build Coastguard Worker   graph_->AddBlock(body);
324*795d594fSAndroid Build Coastguard Worker 
325*795d594fSAndroid Build Coastguard Worker   // Control flow: make a loop back edge first in the list of predecessors.
326*795d594fSAndroid Build Coastguard Worker   entry_block_->RemoveSuccessor(return_block_);
327*795d594fSAndroid Build Coastguard Worker   body->AddSuccessor(header);
328*795d594fSAndroid Build Coastguard Worker   entry_block_->AddSuccessor(header);
329*795d594fSAndroid Build Coastguard Worker   header->AddSuccessor(body);
330*795d594fSAndroid Build Coastguard Worker   header->AddSuccessor(return_block_);
331*795d594fSAndroid Build Coastguard Worker   DCHECK(header->GetSuccessors()[1] == return_block_);
332*795d594fSAndroid Build Coastguard Worker 
333*795d594fSAndroid Build Coastguard Worker   // Data flow.
334*795d594fSAndroid Build Coastguard Worker   MakeIf(header, parameter_);
335*795d594fSAndroid Build Coastguard Worker   MakeGoto(body);
336*795d594fSAndroid Build Coastguard Worker 
337*795d594fSAndroid Build Coastguard Worker   HPhi* phi = new (GetAllocator()) HPhi(GetAllocator(), 0, 0, DataType::Type::kInt32);
338*795d594fSAndroid Build Coastguard Worker   header->AddPhi(phi);
339*795d594fSAndroid Build Coastguard Worker   HInstruction* add = MakeBinOp<HAdd>(body, DataType::Type::kInt32, phi, parameter_);
340*795d594fSAndroid Build Coastguard Worker 
341*795d594fSAndroid Build Coastguard Worker   phi->AddInput(add);
342*795d594fSAndroid Build Coastguard Worker   phi->AddInput(parameter_);
343*795d594fSAndroid Build Coastguard Worker 
344*795d594fSAndroid Build Coastguard Worker   graph_->ClearLoopInformation();
345*795d594fSAndroid Build Coastguard Worker   graph_->ClearDominanceInformation();
346*795d594fSAndroid Build Coastguard Worker   graph_->BuildDominatorTree();
347*795d594fSAndroid Build Coastguard Worker 
348*795d594fSAndroid Build Coastguard Worker   // BuildDominatorTree inserts a block beetween loop header and entry block.
349*795d594fSAndroid Build Coastguard Worker   EXPECT_EQ(header->GetPredecessors()[0]->GetSinglePredecessor(), entry_block_);
350*795d594fSAndroid Build Coastguard Worker 
351*795d594fSAndroid Build Coastguard Worker   // Check that after optimizations in BuildDominatorTree()/SimplifyCFG() phi inputs
352*795d594fSAndroid Build Coastguard Worker   // are still mapped correctly to the block predecessors.
353*795d594fSAndroid Build Coastguard Worker   for (size_t i = 0, e = phi->InputCount(); i < e; i++) {
354*795d594fSAndroid Build Coastguard Worker     HInstruction* input = phi->InputAt(i);
355*795d594fSAndroid Build Coastguard Worker     EXPECT_TRUE(input->GetBlock()->Dominates(header->GetPredecessors()[i]));
356*795d594fSAndroid Build Coastguard Worker   }
357*795d594fSAndroid Build Coastguard Worker }
358*795d594fSAndroid Build Coastguard Worker 
359*795d594fSAndroid Build Coastguard Worker // Test that SimplifyLoop() processes the multiple-preheaders loops correctly.
360*795d594fSAndroid Build Coastguard Worker //
361*795d594fSAndroid Build Coastguard Worker // This is a test for nodes.cc functionality - HGraph::SimplifyLoop.
TEST_F(LoopOptimizationTest,SimplifyLoopSinglePreheader)362*795d594fSAndroid Build Coastguard Worker TEST_F(LoopOptimizationTest, SimplifyLoopSinglePreheader) {
363*795d594fSAndroid Build Coastguard Worker   HBasicBlock* header = AddLoop(entry_block_, return_block_);
364*795d594fSAndroid Build Coastguard Worker 
365*795d594fSAndroid Build Coastguard Worker   header->InsertInstructionBefore(
366*795d594fSAndroid Build Coastguard Worker       new (GetAllocator()) HSuspendCheck(), header->GetLastInstruction());
367*795d594fSAndroid Build Coastguard Worker 
368*795d594fSAndroid Build Coastguard Worker   // Insert an if construct before the loop so it will have two preheaders.
369*795d594fSAndroid Build Coastguard Worker   HBasicBlock* if_block = new (GetAllocator()) HBasicBlock(graph_);
370*795d594fSAndroid Build Coastguard Worker   HBasicBlock* preheader0 = new (GetAllocator()) HBasicBlock(graph_);
371*795d594fSAndroid Build Coastguard Worker   HBasicBlock* preheader1 = new (GetAllocator()) HBasicBlock(graph_);
372*795d594fSAndroid Build Coastguard Worker 
373*795d594fSAndroid Build Coastguard Worker   graph_->AddBlock(if_block);
374*795d594fSAndroid Build Coastguard Worker   graph_->AddBlock(preheader0);
375*795d594fSAndroid Build Coastguard Worker   graph_->AddBlock(preheader1);
376*795d594fSAndroid Build Coastguard Worker 
377*795d594fSAndroid Build Coastguard Worker   // Fix successors/predecessors.
378*795d594fSAndroid Build Coastguard Worker   entry_block_->ReplaceSuccessor(header, if_block);
379*795d594fSAndroid Build Coastguard Worker   if_block->AddSuccessor(preheader0);
380*795d594fSAndroid Build Coastguard Worker   if_block->AddSuccessor(preheader1);
381*795d594fSAndroid Build Coastguard Worker   preheader0->AddSuccessor(header);
382*795d594fSAndroid Build Coastguard Worker   preheader1->AddSuccessor(header);
383*795d594fSAndroid Build Coastguard Worker 
384*795d594fSAndroid Build Coastguard Worker   MakeIf(if_block, parameter_);
385*795d594fSAndroid Build Coastguard Worker   MakeGoto(preheader0);
386*795d594fSAndroid Build Coastguard Worker   MakeGoto(preheader1);
387*795d594fSAndroid Build Coastguard Worker 
388*795d594fSAndroid Build Coastguard Worker   HBasicBlock* body = header->GetSuccessors()[0];
389*795d594fSAndroid Build Coastguard Worker   DCHECK(body != return_block_);
390*795d594fSAndroid Build Coastguard Worker 
391*795d594fSAndroid Build Coastguard Worker   // Add some data flow.
392*795d594fSAndroid Build Coastguard Worker   HIntConstant* const_0 = graph_->GetIntConstant(0);
393*795d594fSAndroid Build Coastguard Worker   HIntConstant* const_1 = graph_->GetIntConstant(1);
394*795d594fSAndroid Build Coastguard Worker   HIntConstant* const_2 = graph_->GetIntConstant(2);
395*795d594fSAndroid Build Coastguard Worker 
396*795d594fSAndroid Build Coastguard Worker   HAdd* preheader0_add = MakeBinOp<HAdd>(preheader0, DataType::Type::kInt32, parameter_, const_0);
397*795d594fSAndroid Build Coastguard Worker   HAdd* preheader1_add = MakeBinOp<HAdd>(preheader1, DataType::Type::kInt32, parameter_, const_1);
398*795d594fSAndroid Build Coastguard Worker 
399*795d594fSAndroid Build Coastguard Worker   HPhi* header_phi = new (GetAllocator()) HPhi(GetAllocator(), 0, 0, DataType::Type::kInt32);
400*795d594fSAndroid Build Coastguard Worker   header->AddPhi(header_phi);
401*795d594fSAndroid Build Coastguard Worker 
402*795d594fSAndroid Build Coastguard Worker   HAdd* body_add = MakeBinOp<HAdd>(body, DataType::Type::kInt32, parameter_, const_2);
403*795d594fSAndroid Build Coastguard Worker 
404*795d594fSAndroid Build Coastguard Worker   DCHECK(header->GetPredecessors()[0] == body);
405*795d594fSAndroid Build Coastguard Worker   DCHECK(header->GetPredecessors()[1] == preheader0);
406*795d594fSAndroid Build Coastguard Worker   DCHECK(header->GetPredecessors()[2] == preheader1);
407*795d594fSAndroid Build Coastguard Worker 
408*795d594fSAndroid Build Coastguard Worker   header_phi->AddInput(body_add);
409*795d594fSAndroid Build Coastguard Worker   header_phi->AddInput(preheader0_add);
410*795d594fSAndroid Build Coastguard Worker   header_phi->AddInput(preheader1_add);
411*795d594fSAndroid Build Coastguard Worker 
412*795d594fSAndroid Build Coastguard Worker   graph_->ClearLoopInformation();
413*795d594fSAndroid Build Coastguard Worker   graph_->ClearDominanceInformation();
414*795d594fSAndroid Build Coastguard Worker   graph_->BuildDominatorTree();
415*795d594fSAndroid Build Coastguard Worker 
416*795d594fSAndroid Build Coastguard Worker   EXPECT_EQ(header->GetPredecessors().size(), 2u);
417*795d594fSAndroid Build Coastguard Worker   EXPECT_EQ(header->GetPredecessors()[1], body);
418*795d594fSAndroid Build Coastguard Worker 
419*795d594fSAndroid Build Coastguard Worker   HBasicBlock* new_preheader = header->GetLoopInformation()->GetPreHeader();
420*795d594fSAndroid Build Coastguard Worker   EXPECT_EQ(preheader0->GetSingleSuccessor(), new_preheader);
421*795d594fSAndroid Build Coastguard Worker   EXPECT_EQ(preheader1->GetSingleSuccessor(), new_preheader);
422*795d594fSAndroid Build Coastguard Worker 
423*795d594fSAndroid Build Coastguard Worker   EXPECT_EQ(new_preheader->GetPhis().CountSize(), 1u);
424*795d594fSAndroid Build Coastguard Worker   HPhi* new_preheader_phi = new_preheader->GetFirstPhi()->AsPhi();
425*795d594fSAndroid Build Coastguard Worker   EXPECT_EQ(new_preheader_phi->InputCount(), 2u);
426*795d594fSAndroid Build Coastguard Worker   EXPECT_EQ(new_preheader_phi->InputAt(0), preheader0_add);
427*795d594fSAndroid Build Coastguard Worker   EXPECT_EQ(new_preheader_phi->InputAt(1), preheader1_add);
428*795d594fSAndroid Build Coastguard Worker 
429*795d594fSAndroid Build Coastguard Worker   EXPECT_EQ(header_phi->InputCount(), 2u);
430*795d594fSAndroid Build Coastguard Worker   EXPECT_EQ(header_phi->InputAt(0), new_preheader_phi);
431*795d594fSAndroid Build Coastguard Worker   EXPECT_EQ(header_phi->InputAt(1), body_add);
432*795d594fSAndroid Build Coastguard Worker }
433*795d594fSAndroid Build Coastguard Worker 
434*795d594fSAndroid Build Coastguard Worker #ifdef ART_ENABLE_CODEGEN_arm64
435*795d594fSAndroid Build Coastguard Worker #define FOR_EACH_CONDITION_INSTRUCTION(M, CondType) \
436*795d594fSAndroid Build Coastguard Worker   M(EQ, CondType)                                   \
437*795d594fSAndroid Build Coastguard Worker   M(NE, CondType)                                   \
438*795d594fSAndroid Build Coastguard Worker   M(LT, CondType)                                   \
439*795d594fSAndroid Build Coastguard Worker   M(LE, CondType)                                   \
440*795d594fSAndroid Build Coastguard Worker   M(GT, CondType)                                   \
441*795d594fSAndroid Build Coastguard Worker   M(GE, CondType)                                   \
442*795d594fSAndroid Build Coastguard Worker   M(B, CondType)                                    \
443*795d594fSAndroid Build Coastguard Worker   M(BE, CondType)                                   \
444*795d594fSAndroid Build Coastguard Worker   M(A, CondType)                                    \
445*795d594fSAndroid Build Coastguard Worker   M(AE, CondType)
446*795d594fSAndroid Build Coastguard Worker 
447*795d594fSAndroid Build Coastguard Worker // Define tests ensuring that all types of conditions can be handled in predicated vectorization
448*795d594fSAndroid Build Coastguard Worker // for diamond loops.
449*795d594fSAndroid Build Coastguard Worker #define DEFINE_CONDITION_TESTS(Name, CondType)                                                  \
450*795d594fSAndroid Build Coastguard Worker TEST_F(PredicatedSimdLoopOptimizationTest, VectorizeCondition##Name##CondType) {                \
451*795d594fSAndroid Build Coastguard Worker   if (!IsPredicatedVectorizationSupported()) {                                                  \
452*795d594fSAndroid Build Coastguard Worker     GTEST_SKIP() << "Predicated SIMD is not enabled.";                                          \
453*795d594fSAndroid Build Coastguard Worker   }                                                                                             \
454*795d594fSAndroid Build Coastguard Worker   ReplaceIfCondition(DataType::Type::k##CondType,                                               \
455*795d594fSAndroid Build Coastguard Worker                      DataType::Type::k##CondType,                                               \
456*795d594fSAndroid Build Coastguard Worker                      diamond_top_,                                                              \
457*795d594fSAndroid Build Coastguard Worker                      kCond##Name);                                                              \
458*795d594fSAndroid Build Coastguard Worker   PerformAnalysis(/*run_checker=*/ true);                                                       \
459*795d594fSAndroid Build Coastguard Worker   EXPECT_TRUE(graph_->HasPredicatedSIMD());                                                     \
460*795d594fSAndroid Build Coastguard Worker }
461*795d594fSAndroid Build Coastguard Worker FOR_EACH_CONDITION_INSTRUCTION(DEFINE_CONDITION_TESTS, Uint8)
462*795d594fSAndroid Build Coastguard Worker FOR_EACH_CONDITION_INSTRUCTION(DEFINE_CONDITION_TESTS, Int8)
463*795d594fSAndroid Build Coastguard Worker FOR_EACH_CONDITION_INSTRUCTION(DEFINE_CONDITION_TESTS, Uint16)
464*795d594fSAndroid Build Coastguard Worker FOR_EACH_CONDITION_INSTRUCTION(DEFINE_CONDITION_TESTS, Int16)
465*795d594fSAndroid Build Coastguard Worker FOR_EACH_CONDITION_INSTRUCTION(DEFINE_CONDITION_TESTS, Int32)
466*795d594fSAndroid Build Coastguard Worker #undef DEFINE_CONDITION_TESTS
467*795d594fSAndroid Build Coastguard Worker #undef FOR_EACH_CONDITION_INSTRUCTION
468*795d594fSAndroid Build Coastguard Worker #endif  // ART_ENABLE_CODEGEN_arm64
469*795d594fSAndroid Build Coastguard Worker 
470*795d594fSAndroid Build Coastguard Worker }  // namespace art
471