blob: cd1602f674606fed9fef986767bb075608b6fc0e [file] [log] [blame]
buzbee67bf8852011-08-17 17:51:35 -07001/*
2 * Copyright (C) 2011 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
buzbeeeaf09bc2012-11-15 14:51:41 -080017#include "compiler_internals.h"
Ian Rogers8d3a1172013-06-04 01:13:28 -070018#include "dataflow_iterator-inl.h"
buzbee67bf8852011-08-17 17:51:35 -070019
buzbee1fd33462013-03-25 13:40:45 -070020#define NOTVISITED (-1)
21
Elliott Hughes11d1b0c2012-01-23 16:57:47 -080022namespace art {
23
buzbeea5abf702013-04-12 14:39:29 -070024void MIRGraph::ClearAllVisitedFlags() {
25 AllNodesIterator iter(this, false /* not iterative */);
26 for (BasicBlock* bb = iter.Next(); bb != NULL; bb = iter.Next()) {
27 bb->visited = false;
28 }
29}
30
buzbee311ca162013-02-28 15:56:43 -080031BasicBlock* MIRGraph::NeedsVisit(BasicBlock* bb) {
buzbeee5f01222012-06-14 15:19:35 -070032 if (bb != NULL) {
33 if (bb->visited || bb->hidden) {
34 bb = NULL;
35 }
36 }
37 return bb;
38}
39
Brian Carlstrom2ce745c2013-07-17 17:44:30 -070040BasicBlock* MIRGraph::NextUnvisitedSuccessor(BasicBlock* bb) {
buzbeefa57c472012-11-21 12:06:18 -080041 BasicBlock* res = NeedsVisit(bb->fall_through);
buzbeee5f01222012-06-14 15:19:35 -070042 if (res == NULL) {
buzbee52a77fc2012-11-20 19:50:46 -080043 res = NeedsVisit(bb->taken);
buzbeee5f01222012-06-14 15:19:35 -070044 if (res == NULL) {
buzbeefa57c472012-11-21 12:06:18 -080045 if (bb->successor_block_list.block_list_type != kNotUsed) {
buzbee862a7602013-04-05 10:58:54 -070046 GrowableArray<SuccessorBlockInfo*>::Iterator iterator(bb->successor_block_list.blocks);
buzbeee5f01222012-06-14 15:19:35 -070047 while (true) {
buzbee862a7602013-04-05 10:58:54 -070048 SuccessorBlockInfo *sbi = iterator.Next();
Brian Carlstrom0cd7ec22013-07-17 23:40:20 -070049 if (sbi == NULL) {
50 break;
51 }
buzbee52a77fc2012-11-20 19:50:46 -080052 res = NeedsVisit(sbi->block);
Brian Carlstrom0cd7ec22013-07-17 23:40:20 -070053 if (res != NULL) {
54 break;
55 }
buzbeee5f01222012-06-14 15:19:35 -070056 }
57 }
58 }
59 }
60 return res;
61}
62
Brian Carlstrom2ce745c2013-07-17 17:44:30 -070063void MIRGraph::MarkPreOrder(BasicBlock* block) {
buzbeee5f01222012-06-14 15:19:35 -070064 block->visited = true;
buzbeefa57c472012-11-21 12:06:18 -080065 /* Enqueue the pre_order block id */
buzbee862a7602013-04-05 10:58:54 -070066 dfs_order_->Insert(block->id);
buzbeee5f01222012-06-14 15:19:35 -070067}
68
Brian Carlstrom2ce745c2013-07-17 17:44:30 -070069void MIRGraph::RecordDFSOrders(BasicBlock* block) {
buzbeee5f01222012-06-14 15:19:35 -070070 std::vector<BasicBlock*> succ;
buzbee311ca162013-02-28 15:56:43 -080071 MarkPreOrder(block);
buzbeee5f01222012-06-14 15:19:35 -070072 succ.push_back(block);
73 while (!succ.empty()) {
74 BasicBlock* curr = succ.back();
buzbeefa57c472012-11-21 12:06:18 -080075 BasicBlock* next_successor = NextUnvisitedSuccessor(curr);
76 if (next_successor != NULL) {
buzbee311ca162013-02-28 15:56:43 -080077 MarkPreOrder(next_successor);
buzbeefa57c472012-11-21 12:06:18 -080078 succ.push_back(next_successor);
buzbeee5f01222012-06-14 15:19:35 -070079 continue;
80 }
buzbee862a7602013-04-05 10:58:54 -070081 curr->dfs_id = dfs_post_order_->Size();
82 dfs_post_order_->Insert(curr->id);
buzbeee5f01222012-06-14 15:19:35 -070083 succ.pop_back();
84 }
85}
86
buzbee5b537102012-01-17 17:33:47 -080087/* Sort the blocks by the Depth-First-Search */
Brian Carlstrom2ce745c2013-07-17 17:44:30 -070088void MIRGraph::ComputeDFSOrders() {
buzbeefa57c472012-11-21 12:06:18 -080089 /* Initialize or reset the DFS pre_order list */
buzbee862a7602013-04-05 10:58:54 -070090 if (dfs_order_ == NULL) {
91 dfs_order_ = new (arena_) GrowableArray<int>(arena_, GetNumBlocks(), kGrowableArrayDfsOrder);
Bill Buzbeea114add2012-05-03 15:00:40 -070092 } else {
93 /* Just reset the used length on the counter */
buzbee862a7602013-04-05 10:58:54 -070094 dfs_order_->Reset();
Bill Buzbeea114add2012-05-03 15:00:40 -070095 }
buzbee67bf8852011-08-17 17:51:35 -070096
buzbeefa57c472012-11-21 12:06:18 -080097 /* Initialize or reset the DFS post_order list */
buzbee862a7602013-04-05 10:58:54 -070098 if (dfs_post_order_ == NULL) {
99 dfs_post_order_ = new (arena_) GrowableArray<int>(arena_, GetNumBlocks(), kGrowableArrayDfsPostOrder);
Bill Buzbeea114add2012-05-03 15:00:40 -0700100 } else {
101 /* Just reset the used length on the counter */
buzbee862a7602013-04-05 10:58:54 -0700102 dfs_post_order_->Reset();
Bill Buzbeea114add2012-05-03 15:00:40 -0700103 }
buzbee5b537102012-01-17 17:33:47 -0800104
buzbeee5f01222012-06-14 15:19:35 -0700105 // Reset visited flags from all nodes
buzbeea5abf702013-04-12 14:39:29 -0700106 ClearAllVisitedFlags();
107
buzbeee5f01222012-06-14 15:19:35 -0700108 // Record dfs orders
buzbee311ca162013-02-28 15:56:43 -0800109 RecordDFSOrders(GetEntryBlock());
buzbeee5f01222012-06-14 15:19:35 -0700110
buzbee862a7602013-04-05 10:58:54 -0700111 num_reachable_blocks_ = dfs_order_->Size();
buzbee67bf8852011-08-17 17:51:35 -0700112}
113
114/*
115 * Mark block bit on the per-Dalvik register vector to denote that Dalvik
116 * register idx is defined in BasicBlock bb.
117 */
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700118bool MIRGraph::FillDefBlockMatrix(BasicBlock* bb) {
Brian Carlstrom0cd7ec22013-07-17 23:40:20 -0700119 if (bb->data_flow_info == NULL) {
120 return false;
121 }
buzbee67bf8852011-08-17 17:51:35 -0700122
buzbee862a7602013-04-05 10:58:54 -0700123 ArenaBitVector::Iterator iterator(bb->data_flow_info->def_v);
Bill Buzbeea114add2012-05-03 15:00:40 -0700124 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700125 int idx = iterator.Next();
Brian Carlstrom0cd7ec22013-07-17 23:40:20 -0700126 if (idx == -1) {
127 break;
128 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700129 /* Block bb defines register idx */
buzbee862a7602013-04-05 10:58:54 -0700130 def_block_matrix_[idx]->SetBit(bb->id);
Bill Buzbeea114add2012-05-03 15:00:40 -0700131 }
132 return true;
buzbee67bf8852011-08-17 17:51:35 -0700133}
134
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700135void MIRGraph::ComputeDefBlockMatrix() {
buzbee311ca162013-02-28 15:56:43 -0800136 int num_registers = cu_->num_dalvik_registers;
buzbeefa57c472012-11-21 12:06:18 -0800137 /* Allocate num_dalvik_registers bit vector pointers */
buzbee311ca162013-02-28 15:56:43 -0800138 def_block_matrix_ = static_cast<ArenaBitVector**>
Mathieu Chartierf6c4b3b2013-08-24 16:11:37 -0700139 (arena_->Alloc(sizeof(ArenaBitVector *) * num_registers,
140 ArenaAllocator::kAllocDFInfo));
Bill Buzbeea114add2012-05-03 15:00:40 -0700141 int i;
buzbee67bf8852011-08-17 17:51:35 -0700142
buzbeefa57c472012-11-21 12:06:18 -0800143 /* Initialize num_register vectors with num_blocks bits each */
144 for (i = 0; i < num_registers; i++) {
buzbee862a7602013-04-05 10:58:54 -0700145 def_block_matrix_[i] =
146 new (arena_) ArenaBitVector(arena_, GetNumBlocks(), false, kBitMapBMatrix);
Bill Buzbeea114add2012-05-03 15:00:40 -0700147 }
buzbee0665fe02013-03-21 12:32:21 -0700148 AllNodesIterator iter(this, false /* not iterative */);
buzbee311ca162013-02-28 15:56:43 -0800149 for (BasicBlock* bb = iter.Next(); bb != NULL; bb = iter.Next()) {
150 FindLocalLiveIn(bb);
151 }
buzbee0665fe02013-03-21 12:32:21 -0700152 AllNodesIterator iter2(this, false /* not iterative */);
buzbee311ca162013-02-28 15:56:43 -0800153 for (BasicBlock* bb = iter2.Next(); bb != NULL; bb = iter2.Next()) {
154 FillDefBlockMatrix(bb);
155 }
buzbee67bf8852011-08-17 17:51:35 -0700156
Bill Buzbeea114add2012-05-03 15:00:40 -0700157 /*
158 * Also set the incoming parameters as defs in the entry block.
159 * Only need to handle the parameters for the outer method.
160 */
buzbee311ca162013-02-28 15:56:43 -0800161 int num_regs = cu_->num_dalvik_registers;
162 int in_reg = num_regs - cu_->num_ins;
buzbeefa57c472012-11-21 12:06:18 -0800163 for (; in_reg < num_regs; in_reg++) {
buzbee862a7602013-04-05 10:58:54 -0700164 def_block_matrix_[in_reg]->SetBit(GetEntryBlock()->id);
Bill Buzbeea114add2012-05-03 15:00:40 -0700165 }
buzbee67bf8852011-08-17 17:51:35 -0700166}
167
buzbeea5abf702013-04-12 14:39:29 -0700168void MIRGraph::ComputeDomPostOrderTraversal(BasicBlock* bb) {
169 if (dom_post_order_traversal_ == NULL) {
170 // First time - create the array.
171 dom_post_order_traversal_ =
172 new (arena_) GrowableArray<int>(arena_, num_reachable_blocks_,
173 kGrowableArrayDomPostOrderTraversal);
174 } else {
175 dom_post_order_traversal_->Reset();
Bill Buzbeea114add2012-05-03 15:00:40 -0700176 }
buzbeea5abf702013-04-12 14:39:29 -0700177 ClearAllVisitedFlags();
178 std::vector<std::pair<BasicBlock*, ArenaBitVector::Iterator*> > work_stack;
179 bb->visited = true;
180 work_stack.push_back(std::make_pair(bb, new (arena_) ArenaBitVector::Iterator(bb->i_dominated)));
181 while (!work_stack.empty()) {
182 std::pair<BasicBlock*, ArenaBitVector::Iterator*> curr = work_stack.back();
183 BasicBlock* curr_bb = curr.first;
184 ArenaBitVector::Iterator* curr_idom_iter = curr.second;
185 int bb_idx = curr_idom_iter->Next();
186 while ((bb_idx != -1) && (NeedsVisit(GetBasicBlock(bb_idx)) == NULL)) {
187 bb_idx = curr_idom_iter->Next();
188 }
189 if (bb_idx != -1) {
190 BasicBlock* new_bb = GetBasicBlock(bb_idx);
191 new_bb->visited = true;
192 work_stack.push_back(
193 std::make_pair(new_bb, new (arena_) ArenaBitVector::Iterator(new_bb->i_dominated)));
194 } else {
195 // no successor/next
196 dom_post_order_traversal_->Insert(curr_bb->id);
197 work_stack.pop_back();
buzbee67bf8852011-08-17 17:51:35 -0700198
buzbeea5abf702013-04-12 14:39:29 -0700199 /* hacky loop detection */
200 if (curr_bb->taken && curr_bb->dominators->IsBitSet(curr_bb->taken->id)) {
201 attributes_ |= METHOD_HAS_LOOP;
202 }
203 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700204 }
buzbee67bf8852011-08-17 17:51:35 -0700205}
206
buzbee311ca162013-02-28 15:56:43 -0800207void MIRGraph::CheckForDominanceFrontier(BasicBlock* dom_bb,
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700208 const BasicBlock* succ_bb) {
Bill Buzbeea114add2012-05-03 15:00:40 -0700209 /*
210 * TODO - evaluate whether phi will ever need to be inserted into exit
211 * blocks.
212 */
buzbeefa57c472012-11-21 12:06:18 -0800213 if (succ_bb->i_dom != dom_bb &&
214 succ_bb->block_type == kDalvikByteCode &&
215 succ_bb->hidden == false) {
buzbee862a7602013-04-05 10:58:54 -0700216 dom_bb->dom_frontier->SetBit(succ_bb->id);
Bill Buzbeea114add2012-05-03 15:00:40 -0700217 }
buzbee67bf8852011-08-17 17:51:35 -0700218}
219
220/* Worker function to compute the dominance frontier */
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700221bool MIRGraph::ComputeDominanceFrontier(BasicBlock* bb) {
Bill Buzbeea114add2012-05-03 15:00:40 -0700222 /* Calculate DF_local */
223 if (bb->taken) {
buzbee311ca162013-02-28 15:56:43 -0800224 CheckForDominanceFrontier(bb, bb->taken);
Bill Buzbeea114add2012-05-03 15:00:40 -0700225 }
buzbeefa57c472012-11-21 12:06:18 -0800226 if (bb->fall_through) {
buzbee311ca162013-02-28 15:56:43 -0800227 CheckForDominanceFrontier(bb, bb->fall_through);
Bill Buzbeea114add2012-05-03 15:00:40 -0700228 }
buzbeefa57c472012-11-21 12:06:18 -0800229 if (bb->successor_block_list.block_list_type != kNotUsed) {
buzbee862a7602013-04-05 10:58:54 -0700230 GrowableArray<SuccessorBlockInfo*>::Iterator iterator(bb->successor_block_list.blocks);
Bill Buzbeea114add2012-05-03 15:00:40 -0700231 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700232 SuccessorBlockInfo *successor_block_info = iterator.Next();
Brian Carlstrom0cd7ec22013-07-17 23:40:20 -0700233 if (successor_block_info == NULL) {
234 break;
235 }
buzbeefa57c472012-11-21 12:06:18 -0800236 BasicBlock* succ_bb = successor_block_info->block;
buzbee311ca162013-02-28 15:56:43 -0800237 CheckForDominanceFrontier(bb, succ_bb);
Bill Buzbeea114add2012-05-03 15:00:40 -0700238 }
239 }
buzbee67bf8852011-08-17 17:51:35 -0700240
Bill Buzbeea114add2012-05-03 15:00:40 -0700241 /* Calculate DF_up */
buzbee862a7602013-04-05 10:58:54 -0700242 ArenaBitVector::Iterator bv_iterator(bb->i_dominated);
Bill Buzbeea114add2012-05-03 15:00:40 -0700243 while (true) {
Brian Carlstrom7934ac22013-07-26 10:54:15 -0700244 // TUNING: hot call to BitVectorIteratorNext
buzbee862a7602013-04-05 10:58:54 -0700245 int dominated_idx = bv_iterator.Next();
Brian Carlstrom0cd7ec22013-07-17 23:40:20 -0700246 if (dominated_idx == -1) {
247 break;
248 }
buzbee311ca162013-02-28 15:56:43 -0800249 BasicBlock* dominated_bb = GetBasicBlock(dominated_idx);
buzbee862a7602013-04-05 10:58:54 -0700250 ArenaBitVector::Iterator df_iterator(dominated_bb->dom_frontier);
buzbee67bf8852011-08-17 17:51:35 -0700251 while (true) {
Brian Carlstrom7934ac22013-07-26 10:54:15 -0700252 // TUNING: hot call to BitVectorIteratorNext
buzbee862a7602013-04-05 10:58:54 -0700253 int df_up_idx = df_iterator.Next();
Brian Carlstrom0cd7ec22013-07-17 23:40:20 -0700254 if (df_up_idx == -1) {
255 break;
256 }
buzbee311ca162013-02-28 15:56:43 -0800257 BasicBlock* df_up_block = GetBasicBlock(df_up_idx);
258 CheckForDominanceFrontier(bb, df_up_block);
buzbee67bf8852011-08-17 17:51:35 -0700259 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700260 }
buzbee67bf8852011-08-17 17:51:35 -0700261
Bill Buzbeea114add2012-05-03 15:00:40 -0700262 return true;
buzbee67bf8852011-08-17 17:51:35 -0700263}
264
265/* Worker function for initializing domination-related data structures */
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700266void MIRGraph::InitializeDominationInfo(BasicBlock* bb) {
buzbee311ca162013-02-28 15:56:43 -0800267 int num_total_blocks = GetBasicBlockListCount();
buzbee67bf8852011-08-17 17:51:35 -0700268
Brian Carlstromdf629502013-07-17 22:39:56 -0700269 if (bb->dominators == NULL) {
buzbee862a7602013-04-05 10:58:54 -0700270 bb->dominators = new (arena_) ArenaBitVector(arena_, num_total_blocks,
271 false /* expandable */, kBitMapDominators);
272 bb->i_dominated = new (arena_) ArenaBitVector(arena_, num_total_blocks,
273 false /* expandable */, kBitMapIDominated);
274 bb->dom_frontier = new (arena_) ArenaBitVector(arena_, num_total_blocks,
275 false /* expandable */, kBitMapDomFrontier);
Bill Buzbeea114add2012-05-03 15:00:40 -0700276 } else {
buzbee862a7602013-04-05 10:58:54 -0700277 bb->dominators->ClearAllBits();
278 bb->i_dominated->ClearAllBits();
279 bb->dom_frontier->ClearAllBits();
Bill Buzbeea114add2012-05-03 15:00:40 -0700280 }
281 /* Set all bits in the dominator vector */
buzbee862a7602013-04-05 10:58:54 -0700282 bb->dominators->SetInitialBits(num_total_blocks);
buzbee67bf8852011-08-17 17:51:35 -0700283
buzbee862a7602013-04-05 10:58:54 -0700284 return;
buzbee67bf8852011-08-17 17:51:35 -0700285}
286
buzbee5b537102012-01-17 17:33:47 -0800287/*
buzbeefa57c472012-11-21 12:06:18 -0800288 * Walk through the ordered i_dom list until we reach common parent.
289 * Given the ordering of i_dom_list, this common parent represents the
buzbee5b537102012-01-17 17:33:47 -0800290 * last element of the intersection of block1 and block2 dominators.
291 */
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700292int MIRGraph::FindCommonParent(int block1, int block2) {
Bill Buzbeea114add2012-05-03 15:00:40 -0700293 while (block1 != block2) {
294 while (block1 < block2) {
buzbee311ca162013-02-28 15:56:43 -0800295 block1 = i_dom_list_[block1];
Bill Buzbeea114add2012-05-03 15:00:40 -0700296 DCHECK_NE(block1, NOTVISITED);
buzbee5b537102012-01-17 17:33:47 -0800297 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700298 while (block2 < block1) {
buzbee311ca162013-02-28 15:56:43 -0800299 block2 = i_dom_list_[block2];
Bill Buzbeea114add2012-05-03 15:00:40 -0700300 DCHECK_NE(block2, NOTVISITED);
301 }
302 }
303 return block1;
buzbee5b537102012-01-17 17:33:47 -0800304}
305
306/* Worker function to compute each block's immediate dominator */
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700307bool MIRGraph::ComputeblockIDom(BasicBlock* bb) {
Bill Buzbeea114add2012-05-03 15:00:40 -0700308 /* Special-case entry block */
buzbee311ca162013-02-28 15:56:43 -0800309 if (bb == GetEntryBlock()) {
buzbee5b537102012-01-17 17:33:47 -0800310 return false;
Bill Buzbeea114add2012-05-03 15:00:40 -0700311 }
312
313 /* Iterate through the predecessors */
buzbee862a7602013-04-05 10:58:54 -0700314 GrowableArray<BasicBlock*>::Iterator iter(bb->predecessors);
Bill Buzbeea114add2012-05-03 15:00:40 -0700315
316 /* Find the first processed predecessor */
buzbee862a7602013-04-05 10:58:54 -0700317 int idom = -1;
Bill Buzbeea114add2012-05-03 15:00:40 -0700318 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700319 BasicBlock* pred_bb = iter.Next();
buzbeefa57c472012-11-21 12:06:18 -0800320 CHECK(pred_bb != NULL);
buzbee311ca162013-02-28 15:56:43 -0800321 if (i_dom_list_[pred_bb->dfs_id] != NOTVISITED) {
buzbeefa57c472012-11-21 12:06:18 -0800322 idom = pred_bb->dfs_id;
Bill Buzbeea114add2012-05-03 15:00:40 -0700323 break;
324 }
325 }
326
327 /* Scan the rest of the predecessors */
328 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700329 BasicBlock* pred_bb = iter.Next();
Brian Carlstrom0cd7ec22013-07-17 23:40:20 -0700330 if (!pred_bb) {
331 break;
332 }
buzbee311ca162013-02-28 15:56:43 -0800333 if (i_dom_list_[pred_bb->dfs_id] == NOTVISITED) {
Bill Buzbeea114add2012-05-03 15:00:40 -0700334 continue;
335 } else {
buzbee311ca162013-02-28 15:56:43 -0800336 idom = FindCommonParent(pred_bb->dfs_id, idom);
Bill Buzbeea114add2012-05-03 15:00:40 -0700337 }
338 }
339
340 DCHECK_NE(idom, NOTVISITED);
341
342 /* Did something change? */
buzbee311ca162013-02-28 15:56:43 -0800343 if (i_dom_list_[bb->dfs_id] != idom) {
344 i_dom_list_[bb->dfs_id] = idom;
Bill Buzbeea114add2012-05-03 15:00:40 -0700345 return true;
346 }
347 return false;
buzbee5b537102012-01-17 17:33:47 -0800348}
349
350/* Worker function to compute each block's domintors */
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700351bool MIRGraph::ComputeBlockDominators(BasicBlock* bb) {
buzbee311ca162013-02-28 15:56:43 -0800352 if (bb == GetEntryBlock()) {
buzbee862a7602013-04-05 10:58:54 -0700353 bb->dominators->ClearAllBits();
Bill Buzbeea114add2012-05-03 15:00:40 -0700354 } else {
buzbee862a7602013-04-05 10:58:54 -0700355 bb->dominators->Copy(bb->i_dom->dominators);
Bill Buzbeea114add2012-05-03 15:00:40 -0700356 }
buzbee862a7602013-04-05 10:58:54 -0700357 bb->dominators->SetBit(bb->id);
Bill Buzbeea114add2012-05-03 15:00:40 -0700358 return false;
buzbee5b537102012-01-17 17:33:47 -0800359}
360
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700361bool MIRGraph::SetDominators(BasicBlock* bb) {
buzbee311ca162013-02-28 15:56:43 -0800362 if (bb != GetEntryBlock()) {
363 int idom_dfs_idx = i_dom_list_[bb->dfs_id];
buzbeefa57c472012-11-21 12:06:18 -0800364 DCHECK_NE(idom_dfs_idx, NOTVISITED);
buzbee862a7602013-04-05 10:58:54 -0700365 int i_dom_idx = dfs_post_order_->Get(idom_dfs_idx);
buzbee311ca162013-02-28 15:56:43 -0800366 BasicBlock* i_dom = GetBasicBlock(i_dom_idx);
buzbeefa57c472012-11-21 12:06:18 -0800367 bb->i_dom = i_dom;
368 /* Add bb to the i_dominated set of the immediate dominator block */
buzbee862a7602013-04-05 10:58:54 -0700369 i_dom->i_dominated->SetBit(bb->id);
Bill Buzbeea114add2012-05-03 15:00:40 -0700370 }
371 return false;
buzbee5b537102012-01-17 17:33:47 -0800372}
373
buzbee67bf8852011-08-17 17:51:35 -0700374/* Compute dominators, immediate dominator, and dominance fronter */
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700375void MIRGraph::ComputeDominators() {
buzbee311ca162013-02-28 15:56:43 -0800376 int num_reachable_blocks = num_reachable_blocks_;
377 int num_total_blocks = GetBasicBlockListCount();
buzbee67bf8852011-08-17 17:51:35 -0700378
Bill Buzbeea114add2012-05-03 15:00:40 -0700379 /* Initialize domination-related data structures */
buzbee0665fe02013-03-21 12:32:21 -0700380 ReachableNodesIterator iter(this, false /* not iterative */);
buzbee311ca162013-02-28 15:56:43 -0800381 for (BasicBlock* bb = iter.Next(); bb != NULL; bb = iter.Next()) {
382 InitializeDominationInfo(bb);
383 }
buzbee67bf8852011-08-17 17:51:35 -0700384
buzbeefa57c472012-11-21 12:06:18 -0800385 /* Initalize & Clear i_dom_list */
buzbee311ca162013-02-28 15:56:43 -0800386 if (i_dom_list_ == NULL) {
Mathieu Chartierf6c4b3b2013-08-24 16:11:37 -0700387 i_dom_list_ = static_cast<int*>(arena_->Alloc(sizeof(int) * num_reachable_blocks,
388 ArenaAllocator::kAllocDFInfo));
Bill Buzbeea114add2012-05-03 15:00:40 -0700389 }
buzbeefa57c472012-11-21 12:06:18 -0800390 for (int i = 0; i < num_reachable_blocks; i++) {
buzbee311ca162013-02-28 15:56:43 -0800391 i_dom_list_[i] = NOTVISITED;
Bill Buzbeea114add2012-05-03 15:00:40 -0700392 }
buzbee5b537102012-01-17 17:33:47 -0800393
buzbeefa57c472012-11-21 12:06:18 -0800394 /* For post-order, last block is entry block. Set its i_dom to istelf */
buzbee311ca162013-02-28 15:56:43 -0800395 DCHECK_EQ(GetEntryBlock()->dfs_id, num_reachable_blocks-1);
396 i_dom_list_[GetEntryBlock()->dfs_id] = GetEntryBlock()->dfs_id;
buzbee5b537102012-01-17 17:33:47 -0800397
Bill Buzbeea114add2012-05-03 15:00:40 -0700398 /* Compute the immediate dominators */
buzbee0665fe02013-03-21 12:32:21 -0700399 ReversePostOrderDfsIterator iter2(this, true /* iterative */);
buzbee311ca162013-02-28 15:56:43 -0800400 bool change = false;
401 for (BasicBlock* bb = iter2.Next(false); bb != NULL; bb = iter2.Next(change)) {
402 change = ComputeblockIDom(bb);
403 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700404
405 /* Set the dominator for the root node */
buzbee862a7602013-04-05 10:58:54 -0700406 GetEntryBlock()->dominators->ClearAllBits();
407 GetEntryBlock()->dominators->SetBit(GetEntryBlock()->id);
Bill Buzbeea114add2012-05-03 15:00:40 -0700408
buzbee311ca162013-02-28 15:56:43 -0800409 if (temp_block_v_ == NULL) {
buzbee862a7602013-04-05 10:58:54 -0700410 temp_block_v_ = new (arena_) ArenaBitVector(arena_, num_total_blocks,
411 false /* expandable */, kBitMapTmpBlockV);
Bill Buzbeea114add2012-05-03 15:00:40 -0700412 } else {
buzbee862a7602013-04-05 10:58:54 -0700413 temp_block_v_->ClearAllBits();
Bill Buzbeea114add2012-05-03 15:00:40 -0700414 }
buzbee311ca162013-02-28 15:56:43 -0800415 GetEntryBlock()->i_dom = NULL;
Bill Buzbeea114add2012-05-03 15:00:40 -0700416
buzbee0665fe02013-03-21 12:32:21 -0700417 ReachableNodesIterator iter3(this, false /* not iterative */);
buzbee311ca162013-02-28 15:56:43 -0800418 for (BasicBlock* bb = iter3.Next(); bb != NULL; bb = iter3.Next()) {
419 SetDominators(bb);
Bill Buzbeea114add2012-05-03 15:00:40 -0700420 }
buzbee67bf8852011-08-17 17:51:35 -0700421
buzbee0665fe02013-03-21 12:32:21 -0700422 ReversePostOrderDfsIterator iter4(this, false /* not iterative */);
buzbee311ca162013-02-28 15:56:43 -0800423 for (BasicBlock* bb = iter4.Next(); bb != NULL; bb = iter4.Next()) {
424 ComputeBlockDominators(bb);
425 }
buzbee5b537102012-01-17 17:33:47 -0800426
buzbeea5abf702013-04-12 14:39:29 -0700427 // Compute the dominance frontier for each block.
buzbee311ca162013-02-28 15:56:43 -0800428 ComputeDomPostOrderTraversal(GetEntryBlock());
buzbee0665fe02013-03-21 12:32:21 -0700429 PostOrderDOMIterator iter5(this, false /* not iterative */);
buzbee311ca162013-02-28 15:56:43 -0800430 for (BasicBlock* bb = iter5.Next(); bb != NULL; bb = iter5.Next()) {
431 ComputeDominanceFrontier(bb);
432 }
buzbee67bf8852011-08-17 17:51:35 -0700433}
434
435/*
436 * Perform dest U= src1 ^ ~src2
437 * This is probably not general enough to be placed in BitVector.[ch].
438 */
buzbee311ca162013-02-28 15:56:43 -0800439void MIRGraph::ComputeSuccLineIn(ArenaBitVector* dest, const ArenaBitVector* src1,
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700440 const ArenaBitVector* src2) {
buzbee862a7602013-04-05 10:58:54 -0700441 if (dest->GetStorageSize() != src1->GetStorageSize() ||
442 dest->GetStorageSize() != src2->GetStorageSize() ||
443 dest->IsExpandable() != src1->IsExpandable() ||
444 dest->IsExpandable() != src2->IsExpandable()) {
Bill Buzbeea114add2012-05-03 15:00:40 -0700445 LOG(FATAL) << "Incompatible set properties";
446 }
buzbee67bf8852011-08-17 17:51:35 -0700447
Bill Buzbeea114add2012-05-03 15:00:40 -0700448 unsigned int idx;
buzbee862a7602013-04-05 10:58:54 -0700449 for (idx = 0; idx < dest->GetStorageSize(); idx++) {
450 dest->GetRawStorage()[idx] |= src1->GetRawStorageWord(idx) & ~(src2->GetRawStorageWord(idx));
Bill Buzbeea114add2012-05-03 15:00:40 -0700451 }
buzbee67bf8852011-08-17 17:51:35 -0700452}
453
454/*
455 * Iterate through all successor blocks and propagate up the live-in sets.
456 * The calculated result is used for phi-node pruning - where we only need to
457 * insert a phi node if the variable is live-in to the block.
458 */
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700459bool MIRGraph::ComputeBlockLiveIns(BasicBlock* bb) {
buzbee311ca162013-02-28 15:56:43 -0800460 ArenaBitVector* temp_dalvik_register_v = temp_dalvik_register_v_;
buzbee67bf8852011-08-17 17:51:35 -0700461
Brian Carlstrom0cd7ec22013-07-17 23:40:20 -0700462 if (bb->data_flow_info == NULL) {
463 return false;
464 }
buzbee862a7602013-04-05 10:58:54 -0700465 temp_dalvik_register_v->Copy(bb->data_flow_info->live_in_v);
buzbeefa57c472012-11-21 12:06:18 -0800466 if (bb->taken && bb->taken->data_flow_info)
467 ComputeSuccLineIn(temp_dalvik_register_v, bb->taken->data_flow_info->live_in_v,
468 bb->data_flow_info->def_v);
469 if (bb->fall_through && bb->fall_through->data_flow_info)
buzbee311ca162013-02-28 15:56:43 -0800470 ComputeSuccLineIn(temp_dalvik_register_v, bb->fall_through->data_flow_info->live_in_v,
buzbeefa57c472012-11-21 12:06:18 -0800471 bb->data_flow_info->def_v);
472 if (bb->successor_block_list.block_list_type != kNotUsed) {
buzbee862a7602013-04-05 10:58:54 -0700473 GrowableArray<SuccessorBlockInfo*>::Iterator iterator(bb->successor_block_list.blocks);
Bill Buzbeea114add2012-05-03 15:00:40 -0700474 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700475 SuccessorBlockInfo *successor_block_info = iterator.Next();
Brian Carlstrom0cd7ec22013-07-17 23:40:20 -0700476 if (successor_block_info == NULL) {
477 break;
478 }
buzbeefa57c472012-11-21 12:06:18 -0800479 BasicBlock* succ_bb = successor_block_info->block;
480 if (succ_bb->data_flow_info) {
buzbee311ca162013-02-28 15:56:43 -0800481 ComputeSuccLineIn(temp_dalvik_register_v, succ_bb->data_flow_info->live_in_v,
buzbeefa57c472012-11-21 12:06:18 -0800482 bb->data_flow_info->def_v);
Bill Buzbeea114add2012-05-03 15:00:40 -0700483 }
buzbee67bf8852011-08-17 17:51:35 -0700484 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700485 }
buzbee862a7602013-04-05 10:58:54 -0700486 if (!temp_dalvik_register_v->Equal(bb->data_flow_info->live_in_v)) {
487 bb->data_flow_info->live_in_v->Copy(temp_dalvik_register_v);
Bill Buzbeea114add2012-05-03 15:00:40 -0700488 return true;
489 }
490 return false;
buzbee67bf8852011-08-17 17:51:35 -0700491}
492
493/* Insert phi nodes to for each variable to the dominance frontiers */
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700494void MIRGraph::InsertPhiNodes() {
buzbeefa57c472012-11-21 12:06:18 -0800495 int dalvik_reg;
buzbee862a7602013-04-05 10:58:54 -0700496 ArenaBitVector* phi_blocks =
497 new (arena_) ArenaBitVector(arena_, GetNumBlocks(), false, kBitMapPhi);
498 ArenaBitVector* tmp_blocks =
499 new (arena_) ArenaBitVector(arena_, GetNumBlocks(), false, kBitMapTmpBlocks);
500 ArenaBitVector* input_blocks =
501 new (arena_) ArenaBitVector(arena_, GetNumBlocks(), false, kBitMapInputBlocks);
buzbee67bf8852011-08-17 17:51:35 -0700502
buzbee311ca162013-02-28 15:56:43 -0800503 temp_dalvik_register_v_ =
buzbee862a7602013-04-05 10:58:54 -0700504 new (arena_) ArenaBitVector(arena_, cu_->num_dalvik_registers, false, kBitMapRegisterV);
buzbee67bf8852011-08-17 17:51:35 -0700505
buzbee0665fe02013-03-21 12:32:21 -0700506 PostOrderDfsIterator iter(this, true /* iterative */);
buzbee311ca162013-02-28 15:56:43 -0800507 bool change = false;
508 for (BasicBlock* bb = iter.Next(false); bb != NULL; bb = iter.Next(change)) {
509 change = ComputeBlockLiveIns(bb);
510 }
buzbee67bf8852011-08-17 17:51:35 -0700511
Bill Buzbeea114add2012-05-03 15:00:40 -0700512 /* Iterate through each Dalvik register */
buzbee311ca162013-02-28 15:56:43 -0800513 for (dalvik_reg = cu_->num_dalvik_registers - 1; dalvik_reg >= 0; dalvik_reg--) {
Bill Buzbeea114add2012-05-03 15:00:40 -0700514 bool change;
buzbee67bf8852011-08-17 17:51:35 -0700515
buzbee862a7602013-04-05 10:58:54 -0700516 input_blocks->Copy(def_block_matrix_[dalvik_reg]);
517 phi_blocks->ClearAllBits();
buzbee67bf8852011-08-17 17:51:35 -0700518
Bill Buzbeea114add2012-05-03 15:00:40 -0700519 /* Calculate the phi blocks for each Dalvik register */
520 do {
521 change = false;
buzbee862a7602013-04-05 10:58:54 -0700522 tmp_blocks->ClearAllBits();
523 ArenaBitVector::Iterator iterator(input_blocks);
buzbee67bf8852011-08-17 17:51:35 -0700524
Bill Buzbeea114add2012-05-03 15:00:40 -0700525 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700526 int idx = iterator.Next();
Brian Carlstrom0cd7ec22013-07-17 23:40:20 -0700527 if (idx == -1) {
528 break;
buzbee67bf8852011-08-17 17:51:35 -0700529 }
Brian Carlstrom0cd7ec22013-07-17 23:40:20 -0700530 BasicBlock* def_bb = GetBasicBlock(idx);
Bill Buzbeea114add2012-05-03 15:00:40 -0700531
Brian Carlstrom0cd7ec22013-07-17 23:40:20 -0700532 /* Merge the dominance frontier to tmp_blocks */
Brian Carlstrom7934ac22013-07-26 10:54:15 -0700533 // TUNING: hot call to Union().
Brian Carlstrom0cd7ec22013-07-17 23:40:20 -0700534 if (def_bb->dom_frontier != NULL) {
535 tmp_blocks->Union(def_bb->dom_frontier);
536 }
537 }
538 if (!phi_blocks->Equal(tmp_blocks)) {
539 change = true;
540 phi_blocks->Copy(tmp_blocks);
541
542 /*
543 * Iterate through the original blocks plus the new ones in
544 * the dominance frontier.
545 */
546 input_blocks->Copy(phi_blocks);
547 input_blocks->Union(def_block_matrix_[dalvik_reg]);
Bill Buzbeea114add2012-05-03 15:00:40 -0700548 }
549 } while (change);
550
551 /*
buzbeefa57c472012-11-21 12:06:18 -0800552 * Insert a phi node for dalvik_reg in the phi_blocks if the Dalvik
Bill Buzbeea114add2012-05-03 15:00:40 -0700553 * register is in the live-in set.
554 */
buzbee862a7602013-04-05 10:58:54 -0700555 ArenaBitVector::Iterator iterator(phi_blocks);
Bill Buzbeea114add2012-05-03 15:00:40 -0700556 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700557 int idx = iterator.Next();
Brian Carlstrom0cd7ec22013-07-17 23:40:20 -0700558 if (idx == -1) {
559 break;
560 }
buzbee311ca162013-02-28 15:56:43 -0800561 BasicBlock* phi_bb = GetBasicBlock(idx);
Bill Buzbeea114add2012-05-03 15:00:40 -0700562 /* Variable will be clobbered before being used - no need for phi */
Brian Carlstrom0cd7ec22013-07-17 23:40:20 -0700563 if (!phi_bb->data_flow_info->live_in_v->IsBitSet(dalvik_reg)) {
564 continue;
565 }
buzbee862a7602013-04-05 10:58:54 -0700566 MIR *phi =
Mathieu Chartierf6c4b3b2013-08-24 16:11:37 -0700567 static_cast<MIR*>(arena_->Alloc(sizeof(MIR), ArenaAllocator::kAllocDFInfo));
buzbeecbd6d442012-11-17 14:11:25 -0800568 phi->dalvikInsn.opcode = static_cast<Instruction::Code>(kMirOpPhi);
buzbeefa57c472012-11-21 12:06:18 -0800569 phi->dalvikInsn.vA = dalvik_reg;
570 phi->offset = phi_bb->start_offset;
Brian Carlstrom7934ac22013-07-26 10:54:15 -0700571 phi->m_unit_index = 0; // Arbitrarily assign all Phi nodes to outermost method.
buzbeefa57c472012-11-21 12:06:18 -0800572 PrependMIR(phi_bb, phi);
buzbee67bf8852011-08-17 17:51:35 -0700573 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700574 }
buzbee67bf8852011-08-17 17:51:35 -0700575}
576
577/*
578 * Worker function to insert phi-operands with latest SSA names from
579 * predecessor blocks
580 */
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700581bool MIRGraph::InsertPhiNodeOperands(BasicBlock* bb) {
Bill Buzbeea114add2012-05-03 15:00:40 -0700582 MIR *mir;
buzbee6969d502012-06-15 16:40:31 -0700583 std::vector<int> uses;
buzbeefa57c472012-11-21 12:06:18 -0800584 std::vector<int> incoming_arc;
buzbee67bf8852011-08-17 17:51:35 -0700585
Bill Buzbeea114add2012-05-03 15:00:40 -0700586 /* Phi nodes are at the beginning of each block */
buzbee28c9a832012-11-21 15:39:13 -0800587 for (mir = bb->first_mir_insn; mir != NULL; mir = mir->next) {
buzbeecbd6d442012-11-17 14:11:25 -0800588 if (mir->dalvikInsn.opcode != static_cast<Instruction::Code>(kMirOpPhi))
Bill Buzbeea114add2012-05-03 15:00:40 -0700589 return true;
buzbeefa57c472012-11-21 12:06:18 -0800590 int ssa_reg = mir->ssa_rep->defs[0];
591 DCHECK_GE(ssa_reg, 0); // Shouldn't see compiler temps here
buzbee311ca162013-02-28 15:56:43 -0800592 int v_reg = SRegToVReg(ssa_reg);
buzbee67bf8852011-08-17 17:51:35 -0700593
buzbee6969d502012-06-15 16:40:31 -0700594 uses.clear();
buzbeefa57c472012-11-21 12:06:18 -0800595 incoming_arc.clear();
buzbee67bf8852011-08-17 17:51:35 -0700596
Bill Buzbeea114add2012-05-03 15:00:40 -0700597 /* Iterate through the predecessors */
buzbee862a7602013-04-05 10:58:54 -0700598 GrowableArray<BasicBlock*>::Iterator iter(bb->predecessors);
Bill Buzbeea114add2012-05-03 15:00:40 -0700599 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700600 BasicBlock* pred_bb = iter.Next();
Brian Carlstrom0cd7ec22013-07-17 23:40:20 -0700601 if (!pred_bb) {
602 break;
603 }
buzbeefa57c472012-11-21 12:06:18 -0800604 int ssa_reg = pred_bb->data_flow_info->vreg_to_ssa_map[v_reg];
605 uses.push_back(ssa_reg);
606 incoming_arc.push_back(pred_bb->id);
buzbee67bf8852011-08-17 17:51:35 -0700607 }
608
Bill Buzbeea114add2012-05-03 15:00:40 -0700609 /* Count the number of SSA registers for a Dalvik register */
buzbeefa57c472012-11-21 12:06:18 -0800610 int num_uses = uses.size();
611 mir->ssa_rep->num_uses = num_uses;
612 mir->ssa_rep->uses =
Mathieu Chartierf6c4b3b2013-08-24 16:11:37 -0700613 static_cast<int*>(arena_->Alloc(sizeof(int) * num_uses, ArenaAllocator::kAllocDFInfo));
buzbeefa57c472012-11-21 12:06:18 -0800614 mir->ssa_rep->fp_use =
Mathieu Chartierf6c4b3b2013-08-24 16:11:37 -0700615 static_cast<bool*>(arena_->Alloc(sizeof(bool) * num_uses, ArenaAllocator::kAllocDFInfo));
buzbee2cfc6392012-05-07 14:51:40 -0700616 int* incoming =
Mathieu Chartierf6c4b3b2013-08-24 16:11:37 -0700617 static_cast<int*>(arena_->Alloc(sizeof(int) * num_uses, ArenaAllocator::kAllocDFInfo));
buzbee2cfc6392012-05-07 14:51:40 -0700618 // TODO: Ugly, rework (but don't burden each MIR/LIR for Phi-only needs)
buzbeecbd6d442012-11-17 14:11:25 -0800619 mir->dalvikInsn.vB = reinterpret_cast<uintptr_t>(incoming);
Bill Buzbeea114add2012-05-03 15:00:40 -0700620
Bill Buzbeea114add2012-05-03 15:00:40 -0700621 /* Set the uses array for the phi node */
buzbeefa57c472012-11-21 12:06:18 -0800622 int *use_ptr = mir->ssa_rep->uses;
623 for (int i = 0; i < num_uses; i++) {
624 *use_ptr++ = uses[i];
625 *incoming++ = incoming_arc[i];
Bill Buzbeea114add2012-05-03 15:00:40 -0700626 }
627 }
628
629 return true;
buzbee67bf8852011-08-17 17:51:35 -0700630}
631
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700632void MIRGraph::DoDFSPreOrderSSARename(BasicBlock* block) {
Brian Carlstrom0cd7ec22013-07-17 23:40:20 -0700633 if (block->visited || block->hidden) {
634 return;
635 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700636 block->visited = true;
buzbeef0cde542011-09-13 14:55:02 -0700637
Bill Buzbeea114add2012-05-03 15:00:40 -0700638 /* Process this block */
buzbee311ca162013-02-28 15:56:43 -0800639 DoSSAConversion(block);
640 int map_size = sizeof(int) * cu_->num_dalvik_registers;
buzbeef0cde542011-09-13 14:55:02 -0700641
Bill Buzbeea114add2012-05-03 15:00:40 -0700642 /* Save SSA map snapshot */
buzbee862a7602013-04-05 10:58:54 -0700643 int* saved_ssa_map =
Mathieu Chartierf6c4b3b2013-08-24 16:11:37 -0700644 static_cast<int*>(arena_->Alloc(map_size, ArenaAllocator::kAllocDalvikToSSAMap));
buzbee311ca162013-02-28 15:56:43 -0800645 memcpy(saved_ssa_map, vreg_to_ssa_map_, map_size);
buzbeef0cde542011-09-13 14:55:02 -0700646
buzbeefa57c472012-11-21 12:06:18 -0800647 if (block->fall_through) {
buzbee311ca162013-02-28 15:56:43 -0800648 DoDFSPreOrderSSARename(block->fall_through);
Bill Buzbeea114add2012-05-03 15:00:40 -0700649 /* Restore SSA map snapshot */
buzbee311ca162013-02-28 15:56:43 -0800650 memcpy(vreg_to_ssa_map_, saved_ssa_map, map_size);
Bill Buzbeea114add2012-05-03 15:00:40 -0700651 }
652 if (block->taken) {
buzbee311ca162013-02-28 15:56:43 -0800653 DoDFSPreOrderSSARename(block->taken);
Bill Buzbeea114add2012-05-03 15:00:40 -0700654 /* Restore SSA map snapshot */
buzbee311ca162013-02-28 15:56:43 -0800655 memcpy(vreg_to_ssa_map_, saved_ssa_map, map_size);
Bill Buzbeea114add2012-05-03 15:00:40 -0700656 }
buzbeefa57c472012-11-21 12:06:18 -0800657 if (block->successor_block_list.block_list_type != kNotUsed) {
buzbee862a7602013-04-05 10:58:54 -0700658 GrowableArray<SuccessorBlockInfo*>::Iterator iterator(block->successor_block_list.blocks);
Bill Buzbeea114add2012-05-03 15:00:40 -0700659 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700660 SuccessorBlockInfo *successor_block_info = iterator.Next();
Brian Carlstrom0cd7ec22013-07-17 23:40:20 -0700661 if (successor_block_info == NULL) {
662 break;
663 }
buzbeefa57c472012-11-21 12:06:18 -0800664 BasicBlock* succ_bb = successor_block_info->block;
buzbee311ca162013-02-28 15:56:43 -0800665 DoDFSPreOrderSSARename(succ_bb);
Bill Buzbeea114add2012-05-03 15:00:40 -0700666 /* Restore SSA map snapshot */
buzbee311ca162013-02-28 15:56:43 -0800667 memcpy(vreg_to_ssa_map_, saved_ssa_map, map_size);
buzbeef0cde542011-09-13 14:55:02 -0700668 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700669 }
buzbee311ca162013-02-28 15:56:43 -0800670 vreg_to_ssa_map_ = saved_ssa_map;
Bill Buzbeea114add2012-05-03 15:00:40 -0700671 return;
buzbeef0cde542011-09-13 14:55:02 -0700672}
673
buzbee67bf8852011-08-17 17:51:35 -0700674/* Perform SSA transformation for the whole method */
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700675void MIRGraph::SSATransformation() {
Bill Buzbeea114add2012-05-03 15:00:40 -0700676 /* Compute the DFS order */
buzbee311ca162013-02-28 15:56:43 -0800677 ComputeDFSOrders();
buzbee67bf8852011-08-17 17:51:35 -0700678
buzbee1fd33462013-03-25 13:40:45 -0700679 /* Compute the dominator info */
680 ComputeDominators();
buzbee67bf8852011-08-17 17:51:35 -0700681
Bill Buzbeea114add2012-05-03 15:00:40 -0700682 /* Allocate data structures in preparation for SSA conversion */
buzbee311ca162013-02-28 15:56:43 -0800683 CompilerInitializeSSAConversion();
buzbee67bf8852011-08-17 17:51:35 -0700684
buzbee1fd33462013-03-25 13:40:45 -0700685 /* Find out the "Dalvik reg def x block" relation */
686 ComputeDefBlockMatrix();
buzbee67bf8852011-08-17 17:51:35 -0700687
buzbee1fd33462013-03-25 13:40:45 -0700688 /* Insert phi nodes to dominance frontiers for all variables */
689 InsertPhiNodes();
buzbee67bf8852011-08-17 17:51:35 -0700690
Bill Buzbeea114add2012-05-03 15:00:40 -0700691 /* Rename register names by local defs and phi nodes */
buzbeea5abf702013-04-12 14:39:29 -0700692 ClearAllVisitedFlags();
buzbee311ca162013-02-28 15:56:43 -0800693 DoDFSPreOrderSSARename(GetEntryBlock());
buzbee67bf8852011-08-17 17:51:35 -0700694
buzbee1fd33462013-03-25 13:40:45 -0700695 /*
696 * Shared temp bit vector used by each block to count the number of defs
697 * from all the predecessor blocks.
698 */
buzbee862a7602013-04-05 10:58:54 -0700699 temp_ssa_register_v_ =
700 new (arena_) ArenaBitVector(arena_, GetNumSSARegs(), false, kBitMapTempSSARegisterV);
buzbee2cfc6392012-05-07 14:51:40 -0700701
buzbee1fd33462013-03-25 13:40:45 -0700702 /* Insert phi-operands with latest SSA names from predecessor blocks */
703 ReachableNodesIterator iter2(this, false /* not iterative */);
704 for (BasicBlock* bb = iter2.Next(); bb != NULL; bb = iter2.Next()) {
705 InsertPhiNodeOperands(bb);
buzbee311ca162013-02-28 15:56:43 -0800706 }
buzbee1fd33462013-03-25 13:40:45 -0700707
buzbee311ca162013-02-28 15:56:43 -0800708 if (cu_->enable_debug & (1 << kDebugDumpCFG)) {
709 DumpCFG("/sdcard/3_post_ssa_cfg/", false);
Bill Buzbeea114add2012-05-03 15:00:40 -0700710 }
buzbee1fd33462013-03-25 13:40:45 -0700711 if (cu_->enable_debug & (1 << kDebugVerifyDataflow)) {
712 VerifyDataflow();
713 }
buzbee67bf8852011-08-17 17:51:35 -0700714}
Elliott Hughes11d1b0c2012-01-23 16:57:47 -0800715
716} // namespace art