blob: a90d705c6f2b88380d1f4fec9da41160f01f8ead [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"
buzbee311ca162013-02-28 15:56:43 -080018#include "dataflow_iterator.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
buzbee311ca162013-02-28 15:56:43 -080040BasicBlock* MIRGraph::NextUnvisitedSuccessor(BasicBlock* bb)
buzbeee5f01222012-06-14 15:19:35 -070041{
buzbeefa57c472012-11-21 12:06:18 -080042 BasicBlock* res = NeedsVisit(bb->fall_through);
buzbeee5f01222012-06-14 15:19:35 -070043 if (res == NULL) {
buzbee52a77fc2012-11-20 19:50:46 -080044 res = NeedsVisit(bb->taken);
buzbeee5f01222012-06-14 15:19:35 -070045 if (res == NULL) {
buzbeefa57c472012-11-21 12:06:18 -080046 if (bb->successor_block_list.block_list_type != kNotUsed) {
buzbee862a7602013-04-05 10:58:54 -070047 GrowableArray<SuccessorBlockInfo*>::Iterator iterator(bb->successor_block_list.blocks);
buzbeee5f01222012-06-14 15:19:35 -070048 while (true) {
buzbee862a7602013-04-05 10:58:54 -070049 SuccessorBlockInfo *sbi = iterator.Next();
buzbeee5f01222012-06-14 15:19:35 -070050 if (sbi == NULL) break;
buzbee52a77fc2012-11-20 19:50:46 -080051 res = NeedsVisit(sbi->block);
buzbeee5f01222012-06-14 15:19:35 -070052 if (res != NULL) break;
53 }
54 }
55 }
56 }
57 return res;
58}
59
buzbee311ca162013-02-28 15:56:43 -080060void MIRGraph::MarkPreOrder(BasicBlock* block)
buzbeee5f01222012-06-14 15:19:35 -070061{
62 block->visited = true;
buzbeefa57c472012-11-21 12:06:18 -080063 /* Enqueue the pre_order block id */
buzbee862a7602013-04-05 10:58:54 -070064 dfs_order_->Insert(block->id);
buzbeee5f01222012-06-14 15:19:35 -070065}
66
buzbee311ca162013-02-28 15:56:43 -080067void MIRGraph::RecordDFSOrders(BasicBlock* block)
buzbee67bf8852011-08-17 17:51:35 -070068{
buzbeee5f01222012-06-14 15:19:35 -070069 std::vector<BasicBlock*> succ;
buzbee311ca162013-02-28 15:56:43 -080070 MarkPreOrder(block);
buzbeee5f01222012-06-14 15:19:35 -070071 succ.push_back(block);
72 while (!succ.empty()) {
73 BasicBlock* curr = succ.back();
buzbeefa57c472012-11-21 12:06:18 -080074 BasicBlock* next_successor = NextUnvisitedSuccessor(curr);
75 if (next_successor != NULL) {
buzbee311ca162013-02-28 15:56:43 -080076 MarkPreOrder(next_successor);
buzbeefa57c472012-11-21 12:06:18 -080077 succ.push_back(next_successor);
buzbeee5f01222012-06-14 15:19:35 -070078 continue;
79 }
buzbee862a7602013-04-05 10:58:54 -070080 curr->dfs_id = dfs_post_order_->Size();
81 dfs_post_order_->Insert(curr->id);
buzbeee5f01222012-06-14 15:19:35 -070082 succ.pop_back();
83 }
84}
85
buzbee5b537102012-01-17 17:33:47 -080086/* Sort the blocks by the Depth-First-Search */
buzbee311ca162013-02-28 15:56:43 -080087void MIRGraph::ComputeDFSOrders()
buzbee67bf8852011-08-17 17:51:35 -070088{
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 */
buzbee311ca162013-02-28 15:56:43 -0800118bool MIRGraph::FillDefBlockMatrix(BasicBlock* bb)
buzbee67bf8852011-08-17 17:51:35 -0700119{
buzbeefa57c472012-11-21 12:06:18 -0800120 if (bb->data_flow_info == NULL) return false;
buzbee67bf8852011-08-17 17:51:35 -0700121
buzbee862a7602013-04-05 10:58:54 -0700122 ArenaBitVector::Iterator iterator(bb->data_flow_info->def_v);
Bill Buzbeea114add2012-05-03 15:00:40 -0700123 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700124 int idx = iterator.Next();
Bill Buzbeea114add2012-05-03 15:00:40 -0700125 if (idx == -1) break;
126 /* Block bb defines register idx */
buzbee862a7602013-04-05 10:58:54 -0700127 def_block_matrix_[idx]->SetBit(bb->id);
Bill Buzbeea114add2012-05-03 15:00:40 -0700128 }
129 return true;
buzbee67bf8852011-08-17 17:51:35 -0700130}
131
buzbee311ca162013-02-28 15:56:43 -0800132void MIRGraph::ComputeDefBlockMatrix()
buzbee67bf8852011-08-17 17:51:35 -0700133{
buzbee311ca162013-02-28 15:56:43 -0800134 int num_registers = cu_->num_dalvik_registers;
buzbeefa57c472012-11-21 12:06:18 -0800135 /* Allocate num_dalvik_registers bit vector pointers */
buzbee311ca162013-02-28 15:56:43 -0800136 def_block_matrix_ = static_cast<ArenaBitVector**>
buzbee862a7602013-04-05 10:58:54 -0700137 (arena_->NewMem(sizeof(ArenaBitVector *) * num_registers, true,
138 ArenaAllocator::kAllocDFInfo));
Bill Buzbeea114add2012-05-03 15:00:40 -0700139 int i;
buzbee67bf8852011-08-17 17:51:35 -0700140
buzbeefa57c472012-11-21 12:06:18 -0800141 /* Initialize num_register vectors with num_blocks bits each */
142 for (i = 0; i < num_registers; i++) {
buzbee862a7602013-04-05 10:58:54 -0700143 def_block_matrix_[i] =
144 new (arena_) ArenaBitVector(arena_, GetNumBlocks(), false, kBitMapBMatrix);
Bill Buzbeea114add2012-05-03 15:00:40 -0700145 }
buzbee0665fe02013-03-21 12:32:21 -0700146 AllNodesIterator iter(this, false /* not iterative */);
buzbee311ca162013-02-28 15:56:43 -0800147 for (BasicBlock* bb = iter.Next(); bb != NULL; bb = iter.Next()) {
148 FindLocalLiveIn(bb);
149 }
buzbee0665fe02013-03-21 12:32:21 -0700150 AllNodesIterator iter2(this, false /* not iterative */);
buzbee311ca162013-02-28 15:56:43 -0800151 for (BasicBlock* bb = iter2.Next(); bb != NULL; bb = iter2.Next()) {
152 FillDefBlockMatrix(bb);
153 }
buzbee67bf8852011-08-17 17:51:35 -0700154
Bill Buzbeea114add2012-05-03 15:00:40 -0700155 /*
156 * Also set the incoming parameters as defs in the entry block.
157 * Only need to handle the parameters for the outer method.
158 */
buzbee311ca162013-02-28 15:56:43 -0800159 int num_regs = cu_->num_dalvik_registers;
160 int in_reg = num_regs - cu_->num_ins;
buzbeefa57c472012-11-21 12:06:18 -0800161 for (; in_reg < num_regs; in_reg++) {
buzbee862a7602013-04-05 10:58:54 -0700162 def_block_matrix_[in_reg]->SetBit(GetEntryBlock()->id);
Bill Buzbeea114add2012-05-03 15:00:40 -0700163 }
buzbee67bf8852011-08-17 17:51:35 -0700164}
165
buzbeea5abf702013-04-12 14:39:29 -0700166void MIRGraph::ComputeDomPostOrderTraversal(BasicBlock* bb) {
167 if (dom_post_order_traversal_ == NULL) {
168 // First time - create the array.
169 dom_post_order_traversal_ =
170 new (arena_) GrowableArray<int>(arena_, num_reachable_blocks_,
171 kGrowableArrayDomPostOrderTraversal);
172 } else {
173 dom_post_order_traversal_->Reset();
Bill Buzbeea114add2012-05-03 15:00:40 -0700174 }
buzbeea5abf702013-04-12 14:39:29 -0700175 ClearAllVisitedFlags();
176 std::vector<std::pair<BasicBlock*, ArenaBitVector::Iterator*> > work_stack;
177 bb->visited = true;
178 work_stack.push_back(std::make_pair(bb, new (arena_) ArenaBitVector::Iterator(bb->i_dominated)));
179 while (!work_stack.empty()) {
180 std::pair<BasicBlock*, ArenaBitVector::Iterator*> curr = work_stack.back();
181 BasicBlock* curr_bb = curr.first;
182 ArenaBitVector::Iterator* curr_idom_iter = curr.second;
183 int bb_idx = curr_idom_iter->Next();
184 while ((bb_idx != -1) && (NeedsVisit(GetBasicBlock(bb_idx)) == NULL)) {
185 bb_idx = curr_idom_iter->Next();
186 }
187 if (bb_idx != -1) {
188 BasicBlock* new_bb = GetBasicBlock(bb_idx);
189 new_bb->visited = true;
190 work_stack.push_back(
191 std::make_pair(new_bb, new (arena_) ArenaBitVector::Iterator(new_bb->i_dominated)));
192 } else {
193 // no successor/next
194 dom_post_order_traversal_->Insert(curr_bb->id);
195 work_stack.pop_back();
buzbee67bf8852011-08-17 17:51:35 -0700196
buzbeea5abf702013-04-12 14:39:29 -0700197 /* hacky loop detection */
198 if (curr_bb->taken && curr_bb->dominators->IsBitSet(curr_bb->taken->id)) {
199 attributes_ |= METHOD_HAS_LOOP;
200 }
201 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700202 }
buzbee67bf8852011-08-17 17:51:35 -0700203}
204
buzbee311ca162013-02-28 15:56:43 -0800205void MIRGraph::CheckForDominanceFrontier(BasicBlock* dom_bb,
206 const BasicBlock* succ_bb)
buzbee67bf8852011-08-17 17:51:35 -0700207{
Bill Buzbeea114add2012-05-03 15:00:40 -0700208 /*
209 * TODO - evaluate whether phi will ever need to be inserted into exit
210 * blocks.
211 */
buzbeefa57c472012-11-21 12:06:18 -0800212 if (succ_bb->i_dom != dom_bb &&
213 succ_bb->block_type == kDalvikByteCode &&
214 succ_bb->hidden == false) {
buzbee862a7602013-04-05 10:58:54 -0700215 dom_bb->dom_frontier->SetBit(succ_bb->id);
Bill Buzbeea114add2012-05-03 15:00:40 -0700216 }
buzbee67bf8852011-08-17 17:51:35 -0700217}
218
219/* Worker function to compute the dominance frontier */
buzbee311ca162013-02-28 15:56:43 -0800220bool MIRGraph::ComputeDominanceFrontier(BasicBlock* bb)
buzbee67bf8852011-08-17 17:51:35 -0700221{
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();
buzbeefa57c472012-11-21 12:06:18 -0800233 if (successor_block_info == NULL) break;
234 BasicBlock* succ_bb = successor_block_info->block;
buzbee311ca162013-02-28 15:56:43 -0800235 CheckForDominanceFrontier(bb, succ_bb);
Bill Buzbeea114add2012-05-03 15:00:40 -0700236 }
237 }
buzbee67bf8852011-08-17 17:51:35 -0700238
Bill Buzbeea114add2012-05-03 15:00:40 -0700239 /* Calculate DF_up */
buzbee862a7602013-04-05 10:58:54 -0700240 ArenaBitVector::Iterator bv_iterator(bb->i_dominated);
Bill Buzbeea114add2012-05-03 15:00:40 -0700241 while (true) {
buzbee52a77fc2012-11-20 19:50:46 -0800242 //TUNING: hot call to BitVectorIteratorNext
buzbee862a7602013-04-05 10:58:54 -0700243 int dominated_idx = bv_iterator.Next();
buzbeefa57c472012-11-21 12:06:18 -0800244 if (dominated_idx == -1) break;
buzbee311ca162013-02-28 15:56:43 -0800245 BasicBlock* dominated_bb = GetBasicBlock(dominated_idx);
buzbee862a7602013-04-05 10:58:54 -0700246 ArenaBitVector::Iterator df_iterator(dominated_bb->dom_frontier);
buzbee67bf8852011-08-17 17:51:35 -0700247 while (true) {
buzbee52a77fc2012-11-20 19:50:46 -0800248 //TUNING: hot call to BitVectorIteratorNext
buzbee862a7602013-04-05 10:58:54 -0700249 int df_up_idx = df_iterator.Next();
buzbeefa57c472012-11-21 12:06:18 -0800250 if (df_up_idx == -1) break;
buzbee311ca162013-02-28 15:56:43 -0800251 BasicBlock* df_up_block = GetBasicBlock(df_up_idx);
252 CheckForDominanceFrontier(bb, df_up_block);
buzbee67bf8852011-08-17 17:51:35 -0700253 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700254 }
buzbee67bf8852011-08-17 17:51:35 -0700255
Bill Buzbeea114add2012-05-03 15:00:40 -0700256 return true;
buzbee67bf8852011-08-17 17:51:35 -0700257}
258
259/* Worker function for initializing domination-related data structures */
buzbee862a7602013-04-05 10:58:54 -0700260void MIRGraph::InitializeDominationInfo(BasicBlock* bb)
buzbee67bf8852011-08-17 17:51:35 -0700261{
buzbee311ca162013-02-28 15:56:43 -0800262 int num_total_blocks = GetBasicBlockListCount();
buzbee67bf8852011-08-17 17:51:35 -0700263
Bill Buzbeea114add2012-05-03 15:00:40 -0700264 if (bb->dominators == NULL ) {
buzbee862a7602013-04-05 10:58:54 -0700265 bb->dominators = new (arena_) ArenaBitVector(arena_, num_total_blocks,
266 false /* expandable */, kBitMapDominators);
267 bb->i_dominated = new (arena_) ArenaBitVector(arena_, num_total_blocks,
268 false /* expandable */, kBitMapIDominated);
269 bb->dom_frontier = new (arena_) ArenaBitVector(arena_, num_total_blocks,
270 false /* expandable */, kBitMapDomFrontier);
Bill Buzbeea114add2012-05-03 15:00:40 -0700271 } else {
buzbee862a7602013-04-05 10:58:54 -0700272 bb->dominators->ClearAllBits();
273 bb->i_dominated->ClearAllBits();
274 bb->dom_frontier->ClearAllBits();
Bill Buzbeea114add2012-05-03 15:00:40 -0700275 }
276 /* Set all bits in the dominator vector */
buzbee862a7602013-04-05 10:58:54 -0700277 bb->dominators->SetInitialBits(num_total_blocks);
buzbee67bf8852011-08-17 17:51:35 -0700278
buzbee862a7602013-04-05 10:58:54 -0700279 return;
buzbee67bf8852011-08-17 17:51:35 -0700280}
281
buzbee5b537102012-01-17 17:33:47 -0800282/*
buzbeefa57c472012-11-21 12:06:18 -0800283 * Walk through the ordered i_dom list until we reach common parent.
284 * Given the ordering of i_dom_list, this common parent represents the
buzbee5b537102012-01-17 17:33:47 -0800285 * last element of the intersection of block1 and block2 dominators.
286 */
buzbee311ca162013-02-28 15:56:43 -0800287int MIRGraph::FindCommonParent(int block1, int block2)
buzbee5b537102012-01-17 17:33:47 -0800288{
Bill Buzbeea114add2012-05-03 15:00:40 -0700289 while (block1 != block2) {
290 while (block1 < block2) {
buzbee311ca162013-02-28 15:56:43 -0800291 block1 = i_dom_list_[block1];
Bill Buzbeea114add2012-05-03 15:00:40 -0700292 DCHECK_NE(block1, NOTVISITED);
buzbee5b537102012-01-17 17:33:47 -0800293 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700294 while (block2 < block1) {
buzbee311ca162013-02-28 15:56:43 -0800295 block2 = i_dom_list_[block2];
Bill Buzbeea114add2012-05-03 15:00:40 -0700296 DCHECK_NE(block2, NOTVISITED);
297 }
298 }
299 return block1;
buzbee5b537102012-01-17 17:33:47 -0800300}
301
302/* Worker function to compute each block's immediate dominator */
buzbee311ca162013-02-28 15:56:43 -0800303bool MIRGraph::ComputeblockIDom(BasicBlock* bb)
buzbee5b537102012-01-17 17:33:47 -0800304{
Bill Buzbeea114add2012-05-03 15:00:40 -0700305 /* Special-case entry block */
buzbee311ca162013-02-28 15:56:43 -0800306 if (bb == GetEntryBlock()) {
buzbee5b537102012-01-17 17:33:47 -0800307 return false;
Bill Buzbeea114add2012-05-03 15:00:40 -0700308 }
309
310 /* Iterate through the predecessors */
buzbee862a7602013-04-05 10:58:54 -0700311 GrowableArray<BasicBlock*>::Iterator iter(bb->predecessors);
Bill Buzbeea114add2012-05-03 15:00:40 -0700312
313 /* Find the first processed predecessor */
buzbee862a7602013-04-05 10:58:54 -0700314 int idom = -1;
Bill Buzbeea114add2012-05-03 15:00:40 -0700315 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700316 BasicBlock* pred_bb = iter.Next();
buzbeefa57c472012-11-21 12:06:18 -0800317 CHECK(pred_bb != NULL);
buzbee311ca162013-02-28 15:56:43 -0800318 if (i_dom_list_[pred_bb->dfs_id] != NOTVISITED) {
buzbeefa57c472012-11-21 12:06:18 -0800319 idom = pred_bb->dfs_id;
Bill Buzbeea114add2012-05-03 15:00:40 -0700320 break;
321 }
322 }
323
324 /* Scan the rest of the predecessors */
325 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700326 BasicBlock* pred_bb = iter.Next();
buzbeefa57c472012-11-21 12:06:18 -0800327 if (!pred_bb) break;
buzbee311ca162013-02-28 15:56:43 -0800328 if (i_dom_list_[pred_bb->dfs_id] == NOTVISITED) {
Bill Buzbeea114add2012-05-03 15:00:40 -0700329 continue;
330 } else {
buzbee311ca162013-02-28 15:56:43 -0800331 idom = FindCommonParent(pred_bb->dfs_id, idom);
Bill Buzbeea114add2012-05-03 15:00:40 -0700332 }
333 }
334
335 DCHECK_NE(idom, NOTVISITED);
336
337 /* Did something change? */
buzbee311ca162013-02-28 15:56:43 -0800338 if (i_dom_list_[bb->dfs_id] != idom) {
339 i_dom_list_[bb->dfs_id] = idom;
Bill Buzbeea114add2012-05-03 15:00:40 -0700340 return true;
341 }
342 return false;
buzbee5b537102012-01-17 17:33:47 -0800343}
344
345/* Worker function to compute each block's domintors */
buzbee311ca162013-02-28 15:56:43 -0800346bool MIRGraph::ComputeBlockDominators(BasicBlock* bb)
buzbee5b537102012-01-17 17:33:47 -0800347{
buzbee311ca162013-02-28 15:56:43 -0800348 if (bb == GetEntryBlock()) {
buzbee862a7602013-04-05 10:58:54 -0700349 bb->dominators->ClearAllBits();
Bill Buzbeea114add2012-05-03 15:00:40 -0700350 } else {
buzbee862a7602013-04-05 10:58:54 -0700351 bb->dominators->Copy(bb->i_dom->dominators);
Bill Buzbeea114add2012-05-03 15:00:40 -0700352 }
buzbee862a7602013-04-05 10:58:54 -0700353 bb->dominators->SetBit(bb->id);
Bill Buzbeea114add2012-05-03 15:00:40 -0700354 return false;
buzbee5b537102012-01-17 17:33:47 -0800355}
356
buzbee311ca162013-02-28 15:56:43 -0800357bool MIRGraph::SetDominators(BasicBlock* bb)
buzbee5b537102012-01-17 17:33:47 -0800358{
buzbee311ca162013-02-28 15:56:43 -0800359 if (bb != GetEntryBlock()) {
360 int idom_dfs_idx = i_dom_list_[bb->dfs_id];
buzbeefa57c472012-11-21 12:06:18 -0800361 DCHECK_NE(idom_dfs_idx, NOTVISITED);
buzbee862a7602013-04-05 10:58:54 -0700362 int i_dom_idx = dfs_post_order_->Get(idom_dfs_idx);
buzbee311ca162013-02-28 15:56:43 -0800363 BasicBlock* i_dom = GetBasicBlock(i_dom_idx);
buzbeefa57c472012-11-21 12:06:18 -0800364 bb->i_dom = i_dom;
365 /* Add bb to the i_dominated set of the immediate dominator block */
buzbee862a7602013-04-05 10:58:54 -0700366 i_dom->i_dominated->SetBit(bb->id);
Bill Buzbeea114add2012-05-03 15:00:40 -0700367 }
368 return false;
buzbee5b537102012-01-17 17:33:47 -0800369}
370
buzbee67bf8852011-08-17 17:51:35 -0700371/* Compute dominators, immediate dominator, and dominance fronter */
buzbee311ca162013-02-28 15:56:43 -0800372void MIRGraph::ComputeDominators()
buzbee67bf8852011-08-17 17:51:35 -0700373{
buzbee311ca162013-02-28 15:56:43 -0800374 int num_reachable_blocks = num_reachable_blocks_;
375 int num_total_blocks = GetBasicBlockListCount();
buzbee67bf8852011-08-17 17:51:35 -0700376
Bill Buzbeea114add2012-05-03 15:00:40 -0700377 /* Initialize domination-related data structures */
buzbee0665fe02013-03-21 12:32:21 -0700378 ReachableNodesIterator iter(this, false /* not iterative */);
buzbee311ca162013-02-28 15:56:43 -0800379 for (BasicBlock* bb = iter.Next(); bb != NULL; bb = iter.Next()) {
380 InitializeDominationInfo(bb);
381 }
buzbee67bf8852011-08-17 17:51:35 -0700382
buzbeefa57c472012-11-21 12:06:18 -0800383 /* Initalize & Clear i_dom_list */
buzbee311ca162013-02-28 15:56:43 -0800384 if (i_dom_list_ == NULL) {
buzbee862a7602013-04-05 10:58:54 -0700385 i_dom_list_ = static_cast<int*>(arena_->NewMem(sizeof(int) * num_reachable_blocks, false,
386 ArenaAllocator::kAllocDFInfo));
Bill Buzbeea114add2012-05-03 15:00:40 -0700387 }
buzbeefa57c472012-11-21 12:06:18 -0800388 for (int i = 0; i < num_reachable_blocks; i++) {
buzbee311ca162013-02-28 15:56:43 -0800389 i_dom_list_[i] = NOTVISITED;
Bill Buzbeea114add2012-05-03 15:00:40 -0700390 }
buzbee5b537102012-01-17 17:33:47 -0800391
buzbeefa57c472012-11-21 12:06:18 -0800392 /* For post-order, last block is entry block. Set its i_dom to istelf */
buzbee311ca162013-02-28 15:56:43 -0800393 DCHECK_EQ(GetEntryBlock()->dfs_id, num_reachable_blocks-1);
394 i_dom_list_[GetEntryBlock()->dfs_id] = GetEntryBlock()->dfs_id;
buzbee5b537102012-01-17 17:33:47 -0800395
Bill Buzbeea114add2012-05-03 15:00:40 -0700396 /* Compute the immediate dominators */
buzbee0665fe02013-03-21 12:32:21 -0700397 ReversePostOrderDfsIterator iter2(this, true /* iterative */);
buzbee311ca162013-02-28 15:56:43 -0800398 bool change = false;
399 for (BasicBlock* bb = iter2.Next(false); bb != NULL; bb = iter2.Next(change)) {
400 change = ComputeblockIDom(bb);
401 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700402
403 /* Set the dominator for the root node */
buzbee862a7602013-04-05 10:58:54 -0700404 GetEntryBlock()->dominators->ClearAllBits();
405 GetEntryBlock()->dominators->SetBit(GetEntryBlock()->id);
Bill Buzbeea114add2012-05-03 15:00:40 -0700406
buzbee311ca162013-02-28 15:56:43 -0800407 if (temp_block_v_ == NULL) {
buzbee862a7602013-04-05 10:58:54 -0700408 temp_block_v_ = new (arena_) ArenaBitVector(arena_, num_total_blocks,
409 false /* expandable */, kBitMapTmpBlockV);
Bill Buzbeea114add2012-05-03 15:00:40 -0700410 } else {
buzbee862a7602013-04-05 10:58:54 -0700411 temp_block_v_->ClearAllBits();
Bill Buzbeea114add2012-05-03 15:00:40 -0700412 }
buzbee311ca162013-02-28 15:56:43 -0800413 GetEntryBlock()->i_dom = NULL;
Bill Buzbeea114add2012-05-03 15:00:40 -0700414
buzbee0665fe02013-03-21 12:32:21 -0700415 ReachableNodesIterator iter3(this, false /* not iterative */);
buzbee311ca162013-02-28 15:56:43 -0800416 for (BasicBlock* bb = iter3.Next(); bb != NULL; bb = iter3.Next()) {
417 SetDominators(bb);
Bill Buzbeea114add2012-05-03 15:00:40 -0700418 }
buzbee67bf8852011-08-17 17:51:35 -0700419
buzbee0665fe02013-03-21 12:32:21 -0700420 ReversePostOrderDfsIterator iter4(this, false /* not iterative */);
buzbee311ca162013-02-28 15:56:43 -0800421 for (BasicBlock* bb = iter4.Next(); bb != NULL; bb = iter4.Next()) {
422 ComputeBlockDominators(bb);
423 }
buzbee5b537102012-01-17 17:33:47 -0800424
buzbeea5abf702013-04-12 14:39:29 -0700425 // Compute the dominance frontier for each block.
buzbee311ca162013-02-28 15:56:43 -0800426 ComputeDomPostOrderTraversal(GetEntryBlock());
buzbee0665fe02013-03-21 12:32:21 -0700427 PostOrderDOMIterator iter5(this, false /* not iterative */);
buzbee311ca162013-02-28 15:56:43 -0800428 for (BasicBlock* bb = iter5.Next(); bb != NULL; bb = iter5.Next()) {
429 ComputeDominanceFrontier(bb);
430 }
buzbee67bf8852011-08-17 17:51:35 -0700431}
432
433/*
434 * Perform dest U= src1 ^ ~src2
435 * This is probably not general enough to be placed in BitVector.[ch].
436 */
buzbee311ca162013-02-28 15:56:43 -0800437void MIRGraph::ComputeSuccLineIn(ArenaBitVector* dest, const ArenaBitVector* src1,
438 const ArenaBitVector* src2)
buzbee67bf8852011-08-17 17:51:35 -0700439{
buzbee862a7602013-04-05 10:58:54 -0700440 if (dest->GetStorageSize() != src1->GetStorageSize() ||
441 dest->GetStorageSize() != src2->GetStorageSize() ||
442 dest->IsExpandable() != src1->IsExpandable() ||
443 dest->IsExpandable() != src2->IsExpandable()) {
Bill Buzbeea114add2012-05-03 15:00:40 -0700444 LOG(FATAL) << "Incompatible set properties";
445 }
buzbee67bf8852011-08-17 17:51:35 -0700446
Bill Buzbeea114add2012-05-03 15:00:40 -0700447 unsigned int idx;
buzbee862a7602013-04-05 10:58:54 -0700448 for (idx = 0; idx < dest->GetStorageSize(); idx++) {
449 dest->GetRawStorage()[idx] |= src1->GetRawStorageWord(idx) & ~(src2->GetRawStorageWord(idx));
Bill Buzbeea114add2012-05-03 15:00:40 -0700450 }
buzbee67bf8852011-08-17 17:51:35 -0700451}
452
453/*
454 * Iterate through all successor blocks and propagate up the live-in sets.
455 * The calculated result is used for phi-node pruning - where we only need to
456 * insert a phi node if the variable is live-in to the block.
457 */
buzbee311ca162013-02-28 15:56:43 -0800458bool MIRGraph::ComputeBlockLiveIns(BasicBlock* bb)
buzbee67bf8852011-08-17 17:51:35 -0700459{
buzbee311ca162013-02-28 15:56:43 -0800460 ArenaBitVector* temp_dalvik_register_v = temp_dalvik_register_v_;
buzbee67bf8852011-08-17 17:51:35 -0700461
buzbeefa57c472012-11-21 12:06:18 -0800462 if (bb->data_flow_info == NULL) return false;
buzbee862a7602013-04-05 10:58:54 -0700463 temp_dalvik_register_v->Copy(bb->data_flow_info->live_in_v);
buzbeefa57c472012-11-21 12:06:18 -0800464 if (bb->taken && bb->taken->data_flow_info)
465 ComputeSuccLineIn(temp_dalvik_register_v, bb->taken->data_flow_info->live_in_v,
466 bb->data_flow_info->def_v);
467 if (bb->fall_through && bb->fall_through->data_flow_info)
buzbee311ca162013-02-28 15:56:43 -0800468 ComputeSuccLineIn(temp_dalvik_register_v, bb->fall_through->data_flow_info->live_in_v,
buzbeefa57c472012-11-21 12:06:18 -0800469 bb->data_flow_info->def_v);
470 if (bb->successor_block_list.block_list_type != kNotUsed) {
buzbee862a7602013-04-05 10:58:54 -0700471 GrowableArray<SuccessorBlockInfo*>::Iterator iterator(bb->successor_block_list.blocks);
Bill Buzbeea114add2012-05-03 15:00:40 -0700472 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700473 SuccessorBlockInfo *successor_block_info = iterator.Next();
buzbeefa57c472012-11-21 12:06:18 -0800474 if (successor_block_info == NULL) break;
475 BasicBlock* succ_bb = successor_block_info->block;
476 if (succ_bb->data_flow_info) {
buzbee311ca162013-02-28 15:56:43 -0800477 ComputeSuccLineIn(temp_dalvik_register_v, succ_bb->data_flow_info->live_in_v,
buzbeefa57c472012-11-21 12:06:18 -0800478 bb->data_flow_info->def_v);
Bill Buzbeea114add2012-05-03 15:00:40 -0700479 }
buzbee67bf8852011-08-17 17:51:35 -0700480 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700481 }
buzbee862a7602013-04-05 10:58:54 -0700482 if (!temp_dalvik_register_v->Equal(bb->data_flow_info->live_in_v)) {
483 bb->data_flow_info->live_in_v->Copy(temp_dalvik_register_v);
Bill Buzbeea114add2012-05-03 15:00:40 -0700484 return true;
485 }
486 return false;
buzbee67bf8852011-08-17 17:51:35 -0700487}
488
489/* Insert phi nodes to for each variable to the dominance frontiers */
buzbee311ca162013-02-28 15:56:43 -0800490void MIRGraph::InsertPhiNodes()
buzbee67bf8852011-08-17 17:51:35 -0700491{
buzbeefa57c472012-11-21 12:06:18 -0800492 int dalvik_reg;
buzbee862a7602013-04-05 10:58:54 -0700493 ArenaBitVector* phi_blocks =
494 new (arena_) ArenaBitVector(arena_, GetNumBlocks(), false, kBitMapPhi);
495 ArenaBitVector* tmp_blocks =
496 new (arena_) ArenaBitVector(arena_, GetNumBlocks(), false, kBitMapTmpBlocks);
497 ArenaBitVector* input_blocks =
498 new (arena_) ArenaBitVector(arena_, GetNumBlocks(), false, kBitMapInputBlocks);
buzbee67bf8852011-08-17 17:51:35 -0700499
buzbee311ca162013-02-28 15:56:43 -0800500 temp_dalvik_register_v_ =
buzbee862a7602013-04-05 10:58:54 -0700501 new (arena_) ArenaBitVector(arena_, cu_->num_dalvik_registers, false, kBitMapRegisterV);
buzbee67bf8852011-08-17 17:51:35 -0700502
buzbee0665fe02013-03-21 12:32:21 -0700503 PostOrderDfsIterator iter(this, true /* iterative */);
buzbee311ca162013-02-28 15:56:43 -0800504 bool change = false;
505 for (BasicBlock* bb = iter.Next(false); bb != NULL; bb = iter.Next(change)) {
506 change = ComputeBlockLiveIns(bb);
507 }
buzbee67bf8852011-08-17 17:51:35 -0700508
Bill Buzbeea114add2012-05-03 15:00:40 -0700509 /* Iterate through each Dalvik register */
buzbee311ca162013-02-28 15:56:43 -0800510 for (dalvik_reg = cu_->num_dalvik_registers - 1; dalvik_reg >= 0; dalvik_reg--) {
Bill Buzbeea114add2012-05-03 15:00:40 -0700511 bool change;
buzbee67bf8852011-08-17 17:51:35 -0700512
buzbee862a7602013-04-05 10:58:54 -0700513 input_blocks->Copy(def_block_matrix_[dalvik_reg]);
514 phi_blocks->ClearAllBits();
buzbee67bf8852011-08-17 17:51:35 -0700515
Bill Buzbeea114add2012-05-03 15:00:40 -0700516 /* Calculate the phi blocks for each Dalvik register */
517 do {
518 change = false;
buzbee862a7602013-04-05 10:58:54 -0700519 tmp_blocks->ClearAllBits();
520 ArenaBitVector::Iterator iterator(input_blocks);
buzbee67bf8852011-08-17 17:51:35 -0700521
Bill Buzbeea114add2012-05-03 15:00:40 -0700522 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700523 int idx = iterator.Next();
Bill Buzbeea114add2012-05-03 15:00:40 -0700524 if (idx == -1) break;
buzbee311ca162013-02-28 15:56:43 -0800525 BasicBlock* def_bb = GetBasicBlock(idx);
buzbee67bf8852011-08-17 17:51:35 -0700526
buzbeefa57c472012-11-21 12:06:18 -0800527 /* Merge the dominance frontier to tmp_blocks */
buzbee862a7602013-04-05 10:58:54 -0700528 //TUNING: hot call to Union().
buzbeefa57c472012-11-21 12:06:18 -0800529 if (def_bb->dom_frontier != NULL) {
buzbee862a7602013-04-05 10:58:54 -0700530 tmp_blocks->Union(def_bb->dom_frontier);
Bill Buzbeea114add2012-05-03 15:00:40 -0700531 }
buzbee67bf8852011-08-17 17:51:35 -0700532 }
buzbee862a7602013-04-05 10:58:54 -0700533 if (!phi_blocks->Equal(tmp_blocks)) {
Bill Buzbeea114add2012-05-03 15:00:40 -0700534 change = true;
buzbee862a7602013-04-05 10:58:54 -0700535 phi_blocks->Copy(tmp_blocks);
Bill Buzbeea114add2012-05-03 15:00:40 -0700536
537 /*
538 * Iterate through the original blocks plus the new ones in
539 * the dominance frontier.
540 */
buzbee862a7602013-04-05 10:58:54 -0700541 input_blocks->Copy(phi_blocks);
542 input_blocks->Union(def_block_matrix_[dalvik_reg]);
Bill Buzbeea114add2012-05-03 15:00:40 -0700543 }
544 } while (change);
545
546 /*
buzbeefa57c472012-11-21 12:06:18 -0800547 * Insert a phi node for dalvik_reg in the phi_blocks if the Dalvik
Bill Buzbeea114add2012-05-03 15:00:40 -0700548 * register is in the live-in set.
549 */
buzbee862a7602013-04-05 10:58:54 -0700550 ArenaBitVector::Iterator iterator(phi_blocks);
Bill Buzbeea114add2012-05-03 15:00:40 -0700551 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700552 int idx = iterator.Next();
Bill Buzbeea114add2012-05-03 15:00:40 -0700553 if (idx == -1) break;
buzbee311ca162013-02-28 15:56:43 -0800554 BasicBlock* phi_bb = GetBasicBlock(idx);
Bill Buzbeea114add2012-05-03 15:00:40 -0700555 /* Variable will be clobbered before being used - no need for phi */
buzbee862a7602013-04-05 10:58:54 -0700556 if (!phi_bb->data_flow_info->live_in_v->IsBitSet(dalvik_reg)) continue;
557 MIR *phi =
558 static_cast<MIR*>(arena_->NewMem(sizeof(MIR), true, ArenaAllocator::kAllocDFInfo));
buzbeecbd6d442012-11-17 14:11:25 -0800559 phi->dalvikInsn.opcode = static_cast<Instruction::Code>(kMirOpPhi);
buzbeefa57c472012-11-21 12:06:18 -0800560 phi->dalvikInsn.vA = dalvik_reg;
561 phi->offset = phi_bb->start_offset;
buzbee311ca162013-02-28 15:56:43 -0800562 phi->m_unit_index = 0; // Arbitrarily assign all Phi nodes to outermost method.
buzbeefa57c472012-11-21 12:06:18 -0800563 PrependMIR(phi_bb, phi);
buzbee67bf8852011-08-17 17:51:35 -0700564 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700565 }
buzbee67bf8852011-08-17 17:51:35 -0700566}
567
568/*
569 * Worker function to insert phi-operands with latest SSA names from
570 * predecessor blocks
571 */
buzbee311ca162013-02-28 15:56:43 -0800572bool MIRGraph::InsertPhiNodeOperands(BasicBlock* bb)
buzbee67bf8852011-08-17 17:51:35 -0700573{
Bill Buzbeea114add2012-05-03 15:00:40 -0700574 MIR *mir;
buzbee6969d502012-06-15 16:40:31 -0700575 std::vector<int> uses;
buzbeefa57c472012-11-21 12:06:18 -0800576 std::vector<int> incoming_arc;
buzbee67bf8852011-08-17 17:51:35 -0700577
Bill Buzbeea114add2012-05-03 15:00:40 -0700578 /* Phi nodes are at the beginning of each block */
buzbee28c9a832012-11-21 15:39:13 -0800579 for (mir = bb->first_mir_insn; mir != NULL; mir = mir->next) {
buzbeecbd6d442012-11-17 14:11:25 -0800580 if (mir->dalvikInsn.opcode != static_cast<Instruction::Code>(kMirOpPhi))
Bill Buzbeea114add2012-05-03 15:00:40 -0700581 return true;
buzbeefa57c472012-11-21 12:06:18 -0800582 int ssa_reg = mir->ssa_rep->defs[0];
583 DCHECK_GE(ssa_reg, 0); // Shouldn't see compiler temps here
buzbee311ca162013-02-28 15:56:43 -0800584 int v_reg = SRegToVReg(ssa_reg);
buzbee67bf8852011-08-17 17:51:35 -0700585
buzbee6969d502012-06-15 16:40:31 -0700586 uses.clear();
buzbeefa57c472012-11-21 12:06:18 -0800587 incoming_arc.clear();
buzbee67bf8852011-08-17 17:51:35 -0700588
Bill Buzbeea114add2012-05-03 15:00:40 -0700589 /* Iterate through the predecessors */
buzbee862a7602013-04-05 10:58:54 -0700590 GrowableArray<BasicBlock*>::Iterator iter(bb->predecessors);
Bill Buzbeea114add2012-05-03 15:00:40 -0700591 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700592 BasicBlock* pred_bb = iter.Next();
buzbeefa57c472012-11-21 12:06:18 -0800593 if (!pred_bb) break;
594 int ssa_reg = pred_bb->data_flow_info->vreg_to_ssa_map[v_reg];
595 uses.push_back(ssa_reg);
596 incoming_arc.push_back(pred_bb->id);
buzbee67bf8852011-08-17 17:51:35 -0700597 }
598
Bill Buzbeea114add2012-05-03 15:00:40 -0700599 /* Count the number of SSA registers for a Dalvik register */
buzbeefa57c472012-11-21 12:06:18 -0800600 int num_uses = uses.size();
601 mir->ssa_rep->num_uses = num_uses;
602 mir->ssa_rep->uses =
buzbee862a7602013-04-05 10:58:54 -0700603 static_cast<int*>(arena_->NewMem(sizeof(int) * num_uses, false,
604 ArenaAllocator::kAllocDFInfo));
buzbeefa57c472012-11-21 12:06:18 -0800605 mir->ssa_rep->fp_use =
buzbee862a7602013-04-05 10:58:54 -0700606 static_cast<bool*>(arena_->NewMem(sizeof(bool) * num_uses, true,
607 ArenaAllocator::kAllocDFInfo));
buzbee2cfc6392012-05-07 14:51:40 -0700608 int* incoming =
buzbee862a7602013-04-05 10:58:54 -0700609 static_cast<int*>(arena_->NewMem(sizeof(int) * num_uses, false,
610 ArenaAllocator::kAllocDFInfo));
buzbee2cfc6392012-05-07 14:51:40 -0700611 // TODO: Ugly, rework (but don't burden each MIR/LIR for Phi-only needs)
buzbeecbd6d442012-11-17 14:11:25 -0800612 mir->dalvikInsn.vB = reinterpret_cast<uintptr_t>(incoming);
Bill Buzbeea114add2012-05-03 15:00:40 -0700613
Bill Buzbeea114add2012-05-03 15:00:40 -0700614 /* Set the uses array for the phi node */
buzbeefa57c472012-11-21 12:06:18 -0800615 int *use_ptr = mir->ssa_rep->uses;
616 for (int i = 0; i < num_uses; i++) {
617 *use_ptr++ = uses[i];
618 *incoming++ = incoming_arc[i];
Bill Buzbeea114add2012-05-03 15:00:40 -0700619 }
620 }
621
622 return true;
buzbee67bf8852011-08-17 17:51:35 -0700623}
624
buzbee311ca162013-02-28 15:56:43 -0800625void MIRGraph::DoDFSPreOrderSSARename(BasicBlock* block)
buzbeef0cde542011-09-13 14:55:02 -0700626{
627
Bill Buzbeea114add2012-05-03 15:00:40 -0700628 if (block->visited || block->hidden) return;
629 block->visited = true;
buzbeef0cde542011-09-13 14:55:02 -0700630
Bill Buzbeea114add2012-05-03 15:00:40 -0700631 /* Process this block */
buzbee311ca162013-02-28 15:56:43 -0800632 DoSSAConversion(block);
633 int map_size = sizeof(int) * cu_->num_dalvik_registers;
buzbeef0cde542011-09-13 14:55:02 -0700634
Bill Buzbeea114add2012-05-03 15:00:40 -0700635 /* Save SSA map snapshot */
buzbee862a7602013-04-05 10:58:54 -0700636 int* saved_ssa_map =
637 static_cast<int*>(arena_->NewMem(map_size, false, ArenaAllocator::kAllocDalvikToSSAMap));
buzbee311ca162013-02-28 15:56:43 -0800638 memcpy(saved_ssa_map, vreg_to_ssa_map_, map_size);
buzbeef0cde542011-09-13 14:55:02 -0700639
buzbeefa57c472012-11-21 12:06:18 -0800640 if (block->fall_through) {
buzbee311ca162013-02-28 15:56:43 -0800641 DoDFSPreOrderSSARename(block->fall_through);
Bill Buzbeea114add2012-05-03 15:00:40 -0700642 /* Restore SSA map snapshot */
buzbee311ca162013-02-28 15:56:43 -0800643 memcpy(vreg_to_ssa_map_, saved_ssa_map, map_size);
Bill Buzbeea114add2012-05-03 15:00:40 -0700644 }
645 if (block->taken) {
buzbee311ca162013-02-28 15:56:43 -0800646 DoDFSPreOrderSSARename(block->taken);
Bill Buzbeea114add2012-05-03 15:00:40 -0700647 /* Restore SSA map snapshot */
buzbee311ca162013-02-28 15:56:43 -0800648 memcpy(vreg_to_ssa_map_, saved_ssa_map, map_size);
Bill Buzbeea114add2012-05-03 15:00:40 -0700649 }
buzbeefa57c472012-11-21 12:06:18 -0800650 if (block->successor_block_list.block_list_type != kNotUsed) {
buzbee862a7602013-04-05 10:58:54 -0700651 GrowableArray<SuccessorBlockInfo*>::Iterator iterator(block->successor_block_list.blocks);
Bill Buzbeea114add2012-05-03 15:00:40 -0700652 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700653 SuccessorBlockInfo *successor_block_info = iterator.Next();
buzbeefa57c472012-11-21 12:06:18 -0800654 if (successor_block_info == NULL) break;
655 BasicBlock* succ_bb = successor_block_info->block;
buzbee311ca162013-02-28 15:56:43 -0800656 DoDFSPreOrderSSARename(succ_bb);
Bill Buzbeea114add2012-05-03 15:00:40 -0700657 /* Restore SSA map snapshot */
buzbee311ca162013-02-28 15:56:43 -0800658 memcpy(vreg_to_ssa_map_, saved_ssa_map, map_size);
buzbeef0cde542011-09-13 14:55:02 -0700659 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700660 }
buzbee311ca162013-02-28 15:56:43 -0800661 vreg_to_ssa_map_ = saved_ssa_map;
Bill Buzbeea114add2012-05-03 15:00:40 -0700662 return;
buzbeef0cde542011-09-13 14:55:02 -0700663}
664
buzbee67bf8852011-08-17 17:51:35 -0700665/* Perform SSA transformation for the whole method */
buzbee311ca162013-02-28 15:56:43 -0800666void MIRGraph::SSATransformation()
buzbee67bf8852011-08-17 17:51:35 -0700667{
Bill Buzbeea114add2012-05-03 15:00:40 -0700668 /* Compute the DFS order */
buzbee311ca162013-02-28 15:56:43 -0800669 ComputeDFSOrders();
buzbee67bf8852011-08-17 17:51:35 -0700670
buzbee1fd33462013-03-25 13:40:45 -0700671 /* Compute the dominator info */
672 ComputeDominators();
buzbee67bf8852011-08-17 17:51:35 -0700673
Bill Buzbeea114add2012-05-03 15:00:40 -0700674 /* Allocate data structures in preparation for SSA conversion */
buzbee311ca162013-02-28 15:56:43 -0800675 CompilerInitializeSSAConversion();
buzbee67bf8852011-08-17 17:51:35 -0700676
buzbee1fd33462013-03-25 13:40:45 -0700677 /* Find out the "Dalvik reg def x block" relation */
678 ComputeDefBlockMatrix();
buzbee67bf8852011-08-17 17:51:35 -0700679
buzbee1fd33462013-03-25 13:40:45 -0700680 /* Insert phi nodes to dominance frontiers for all variables */
681 InsertPhiNodes();
buzbee67bf8852011-08-17 17:51:35 -0700682
Bill Buzbeea114add2012-05-03 15:00:40 -0700683 /* Rename register names by local defs and phi nodes */
buzbeea5abf702013-04-12 14:39:29 -0700684 ClearAllVisitedFlags();
buzbee311ca162013-02-28 15:56:43 -0800685 DoDFSPreOrderSSARename(GetEntryBlock());
buzbee67bf8852011-08-17 17:51:35 -0700686
buzbee1fd33462013-03-25 13:40:45 -0700687 /*
688 * Shared temp bit vector used by each block to count the number of defs
689 * from all the predecessor blocks.
690 */
buzbee862a7602013-04-05 10:58:54 -0700691 temp_ssa_register_v_ =
692 new (arena_) ArenaBitVector(arena_, GetNumSSARegs(), false, kBitMapTempSSARegisterV);
buzbee2cfc6392012-05-07 14:51:40 -0700693
buzbee1fd33462013-03-25 13:40:45 -0700694 /* Insert phi-operands with latest SSA names from predecessor blocks */
695 ReachableNodesIterator iter2(this, false /* not iterative */);
696 for (BasicBlock* bb = iter2.Next(); bb != NULL; bb = iter2.Next()) {
697 InsertPhiNodeOperands(bb);
buzbee311ca162013-02-28 15:56:43 -0800698 }
buzbee1fd33462013-03-25 13:40:45 -0700699
buzbee311ca162013-02-28 15:56:43 -0800700 if (cu_->enable_debug & (1 << kDebugDumpCFG)) {
701 DumpCFG("/sdcard/3_post_ssa_cfg/", false);
Bill Buzbeea114add2012-05-03 15:00:40 -0700702 }
buzbee1fd33462013-03-25 13:40:45 -0700703 if (cu_->enable_debug & (1 << kDebugVerifyDataflow)) {
704 VerifyDataflow();
705 }
buzbee67bf8852011-08-17 17:51:35 -0700706}
Elliott Hughes11d1b0c2012-01-23 16:57:47 -0800707
708} // namespace art