blob: 02982e55f98397b9ec663255fe36e49eb8c50d8d [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
buzbee311ca162013-02-28 15:56:43 -080024BasicBlock* MIRGraph::NeedsVisit(BasicBlock* bb) {
buzbeee5f01222012-06-14 15:19:35 -070025 if (bb != NULL) {
26 if (bb->visited || bb->hidden) {
27 bb = NULL;
28 }
29 }
30 return bb;
31}
32
buzbee311ca162013-02-28 15:56:43 -080033BasicBlock* MIRGraph::NextUnvisitedSuccessor(BasicBlock* bb)
buzbeee5f01222012-06-14 15:19:35 -070034{
buzbeefa57c472012-11-21 12:06:18 -080035 BasicBlock* res = NeedsVisit(bb->fall_through);
buzbeee5f01222012-06-14 15:19:35 -070036 if (res == NULL) {
buzbee52a77fc2012-11-20 19:50:46 -080037 res = NeedsVisit(bb->taken);
buzbeee5f01222012-06-14 15:19:35 -070038 if (res == NULL) {
buzbeefa57c472012-11-21 12:06:18 -080039 if (bb->successor_block_list.block_list_type != kNotUsed) {
buzbee862a7602013-04-05 10:58:54 -070040 GrowableArray<SuccessorBlockInfo*>::Iterator iterator(bb->successor_block_list.blocks);
buzbeee5f01222012-06-14 15:19:35 -070041 while (true) {
buzbee862a7602013-04-05 10:58:54 -070042 SuccessorBlockInfo *sbi = iterator.Next();
buzbeee5f01222012-06-14 15:19:35 -070043 if (sbi == NULL) break;
buzbee52a77fc2012-11-20 19:50:46 -080044 res = NeedsVisit(sbi->block);
buzbeee5f01222012-06-14 15:19:35 -070045 if (res != NULL) break;
46 }
47 }
48 }
49 }
50 return res;
51}
52
buzbee311ca162013-02-28 15:56:43 -080053void MIRGraph::MarkPreOrder(BasicBlock* block)
buzbeee5f01222012-06-14 15:19:35 -070054{
55 block->visited = true;
buzbeefa57c472012-11-21 12:06:18 -080056 /* Enqueue the pre_order block id */
buzbee862a7602013-04-05 10:58:54 -070057 dfs_order_->Insert(block->id);
buzbeee5f01222012-06-14 15:19:35 -070058}
59
buzbee311ca162013-02-28 15:56:43 -080060void MIRGraph::RecordDFSOrders(BasicBlock* block)
buzbee67bf8852011-08-17 17:51:35 -070061{
buzbeee5f01222012-06-14 15:19:35 -070062 std::vector<BasicBlock*> succ;
buzbee311ca162013-02-28 15:56:43 -080063 MarkPreOrder(block);
buzbeee5f01222012-06-14 15:19:35 -070064 succ.push_back(block);
65 while (!succ.empty()) {
66 BasicBlock* curr = succ.back();
buzbeefa57c472012-11-21 12:06:18 -080067 BasicBlock* next_successor = NextUnvisitedSuccessor(curr);
68 if (next_successor != NULL) {
buzbee311ca162013-02-28 15:56:43 -080069 MarkPreOrder(next_successor);
buzbeefa57c472012-11-21 12:06:18 -080070 succ.push_back(next_successor);
buzbeee5f01222012-06-14 15:19:35 -070071 continue;
72 }
buzbee862a7602013-04-05 10:58:54 -070073 curr->dfs_id = dfs_post_order_->Size();
74 dfs_post_order_->Insert(curr->id);
buzbeee5f01222012-06-14 15:19:35 -070075 succ.pop_back();
76 }
77}
78
buzbee5b537102012-01-17 17:33:47 -080079/* Sort the blocks by the Depth-First-Search */
buzbee311ca162013-02-28 15:56:43 -080080void MIRGraph::ComputeDFSOrders()
buzbee67bf8852011-08-17 17:51:35 -070081{
buzbeefa57c472012-11-21 12:06:18 -080082 /* Initialize or reset the DFS pre_order list */
buzbee862a7602013-04-05 10:58:54 -070083 if (dfs_order_ == NULL) {
84 dfs_order_ = new (arena_) GrowableArray<int>(arena_, GetNumBlocks(), kGrowableArrayDfsOrder);
Bill Buzbeea114add2012-05-03 15:00:40 -070085 } else {
86 /* Just reset the used length on the counter */
buzbee862a7602013-04-05 10:58:54 -070087 dfs_order_->Reset();
Bill Buzbeea114add2012-05-03 15:00:40 -070088 }
buzbee67bf8852011-08-17 17:51:35 -070089
buzbeefa57c472012-11-21 12:06:18 -080090 /* Initialize or reset the DFS post_order list */
buzbee862a7602013-04-05 10:58:54 -070091 if (dfs_post_order_ == NULL) {
92 dfs_post_order_ = new (arena_) GrowableArray<int>(arena_, GetNumBlocks(), kGrowableArrayDfsPostOrder);
Bill Buzbeea114add2012-05-03 15:00:40 -070093 } else {
94 /* Just reset the used length on the counter */
buzbee862a7602013-04-05 10:58:54 -070095 dfs_post_order_->Reset();
Bill Buzbeea114add2012-05-03 15:00:40 -070096 }
buzbee5b537102012-01-17 17:33:47 -080097
buzbeee5f01222012-06-14 15:19:35 -070098 // Reset visited flags from all nodes
buzbee0665fe02013-03-21 12:32:21 -070099 AllNodesIterator iter(this, false /* not iterative */);
buzbee311ca162013-02-28 15:56:43 -0800100 for (BasicBlock* bb = iter.Next(); bb != NULL; bb = iter.Next()) {
101 ClearVisitedFlag(bb);
102 }
buzbeee5f01222012-06-14 15:19:35 -0700103 // Record dfs orders
buzbee311ca162013-02-28 15:56:43 -0800104 RecordDFSOrders(GetEntryBlock());
buzbeee5f01222012-06-14 15:19:35 -0700105
buzbee862a7602013-04-05 10:58:54 -0700106 num_reachable_blocks_ = dfs_order_->Size();
buzbee67bf8852011-08-17 17:51:35 -0700107}
108
109/*
110 * Mark block bit on the per-Dalvik register vector to denote that Dalvik
111 * register idx is defined in BasicBlock bb.
112 */
buzbee311ca162013-02-28 15:56:43 -0800113bool MIRGraph::FillDefBlockMatrix(BasicBlock* bb)
buzbee67bf8852011-08-17 17:51:35 -0700114{
buzbeefa57c472012-11-21 12:06:18 -0800115 if (bb->data_flow_info == NULL) return false;
buzbee67bf8852011-08-17 17:51:35 -0700116
buzbee862a7602013-04-05 10:58:54 -0700117 ArenaBitVector::Iterator iterator(bb->data_flow_info->def_v);
Bill Buzbeea114add2012-05-03 15:00:40 -0700118 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700119 int idx = iterator.Next();
Bill Buzbeea114add2012-05-03 15:00:40 -0700120 if (idx == -1) break;
121 /* Block bb defines register idx */
buzbee862a7602013-04-05 10:58:54 -0700122 def_block_matrix_[idx]->SetBit(bb->id);
Bill Buzbeea114add2012-05-03 15:00:40 -0700123 }
124 return true;
buzbee67bf8852011-08-17 17:51:35 -0700125}
126
buzbee311ca162013-02-28 15:56:43 -0800127void MIRGraph::ComputeDefBlockMatrix()
buzbee67bf8852011-08-17 17:51:35 -0700128{
buzbee311ca162013-02-28 15:56:43 -0800129 int num_registers = cu_->num_dalvik_registers;
buzbeefa57c472012-11-21 12:06:18 -0800130 /* Allocate num_dalvik_registers bit vector pointers */
buzbee311ca162013-02-28 15:56:43 -0800131 def_block_matrix_ = static_cast<ArenaBitVector**>
buzbee862a7602013-04-05 10:58:54 -0700132 (arena_->NewMem(sizeof(ArenaBitVector *) * num_registers, true,
133 ArenaAllocator::kAllocDFInfo));
Bill Buzbeea114add2012-05-03 15:00:40 -0700134 int i;
buzbee67bf8852011-08-17 17:51:35 -0700135
buzbeefa57c472012-11-21 12:06:18 -0800136 /* Initialize num_register vectors with num_blocks bits each */
137 for (i = 0; i < num_registers; i++) {
buzbee862a7602013-04-05 10:58:54 -0700138 def_block_matrix_[i] =
139 new (arena_) ArenaBitVector(arena_, GetNumBlocks(), false, kBitMapBMatrix);
Bill Buzbeea114add2012-05-03 15:00:40 -0700140 }
buzbee0665fe02013-03-21 12:32:21 -0700141 AllNodesIterator iter(this, false /* not iterative */);
buzbee311ca162013-02-28 15:56:43 -0800142 for (BasicBlock* bb = iter.Next(); bb != NULL; bb = iter.Next()) {
143 FindLocalLiveIn(bb);
144 }
buzbee0665fe02013-03-21 12:32:21 -0700145 AllNodesIterator iter2(this, false /* not iterative */);
buzbee311ca162013-02-28 15:56:43 -0800146 for (BasicBlock* bb = iter2.Next(); bb != NULL; bb = iter2.Next()) {
147 FillDefBlockMatrix(bb);
148 }
buzbee67bf8852011-08-17 17:51:35 -0700149
Bill Buzbeea114add2012-05-03 15:00:40 -0700150 /*
151 * Also set the incoming parameters as defs in the entry block.
152 * Only need to handle the parameters for the outer method.
153 */
buzbee311ca162013-02-28 15:56:43 -0800154 int num_regs = cu_->num_dalvik_registers;
155 int in_reg = num_regs - cu_->num_ins;
buzbeefa57c472012-11-21 12:06:18 -0800156 for (; in_reg < num_regs; in_reg++) {
buzbee862a7602013-04-05 10:58:54 -0700157 def_block_matrix_[in_reg]->SetBit(GetEntryBlock()->id);
Bill Buzbeea114add2012-05-03 15:00:40 -0700158 }
buzbee67bf8852011-08-17 17:51:35 -0700159}
160
161/* Compute the post-order traversal of the CFG */
buzbee311ca162013-02-28 15:56:43 -0800162void MIRGraph::ComputeDomPostOrderTraversal(BasicBlock* bb)
buzbee67bf8852011-08-17 17:51:35 -0700163{
buzbee862a7602013-04-05 10:58:54 -0700164 ArenaBitVector::Iterator bv_iterator(bb->i_dominated);
buzbee67bf8852011-08-17 17:51:35 -0700165
Bill Buzbeea114add2012-05-03 15:00:40 -0700166 /* Iterate through the dominated blocks first */
167 while (true) {
buzbee52a77fc2012-11-20 19:50:46 -0800168 //TUNING: hot call to BitVectorIteratorNext
buzbee862a7602013-04-05 10:58:54 -0700169 int bb_idx = bv_iterator.Next();
buzbeefa57c472012-11-21 12:06:18 -0800170 if (bb_idx == -1) break;
buzbee311ca162013-02-28 15:56:43 -0800171 BasicBlock* dominated_bb = GetBasicBlock(bb_idx);
172 ComputeDomPostOrderTraversal(dominated_bb);
Bill Buzbeea114add2012-05-03 15:00:40 -0700173 }
buzbee67bf8852011-08-17 17:51:35 -0700174
Bill Buzbeea114add2012-05-03 15:00:40 -0700175 /* Enter the current block id */
buzbee862a7602013-04-05 10:58:54 -0700176 dom_post_order_traversal_->Insert(bb->id);
buzbee67bf8852011-08-17 17:51:35 -0700177
Bill Buzbeea114add2012-05-03 15:00:40 -0700178 /* hacky loop detection */
buzbee862a7602013-04-05 10:58:54 -0700179 if (bb->taken && bb->dominators->IsBitSet(bb->taken->id)) {
buzbee1fd33462013-03-25 13:40:45 -0700180 attributes_ |= METHOD_HAS_LOOP;
Bill Buzbeea114add2012-05-03 15:00:40 -0700181 }
buzbee67bf8852011-08-17 17:51:35 -0700182}
183
buzbee311ca162013-02-28 15:56:43 -0800184void MIRGraph::CheckForDominanceFrontier(BasicBlock* dom_bb,
185 const BasicBlock* succ_bb)
buzbee67bf8852011-08-17 17:51:35 -0700186{
Bill Buzbeea114add2012-05-03 15:00:40 -0700187 /*
188 * TODO - evaluate whether phi will ever need to be inserted into exit
189 * blocks.
190 */
buzbeefa57c472012-11-21 12:06:18 -0800191 if (succ_bb->i_dom != dom_bb &&
192 succ_bb->block_type == kDalvikByteCode &&
193 succ_bb->hidden == false) {
buzbee862a7602013-04-05 10:58:54 -0700194 dom_bb->dom_frontier->SetBit(succ_bb->id);
Bill Buzbeea114add2012-05-03 15:00:40 -0700195 }
buzbee67bf8852011-08-17 17:51:35 -0700196}
197
198/* Worker function to compute the dominance frontier */
buzbee311ca162013-02-28 15:56:43 -0800199bool MIRGraph::ComputeDominanceFrontier(BasicBlock* bb)
buzbee67bf8852011-08-17 17:51:35 -0700200{
Bill Buzbeea114add2012-05-03 15:00:40 -0700201 /* Calculate DF_local */
202 if (bb->taken) {
buzbee311ca162013-02-28 15:56:43 -0800203 CheckForDominanceFrontier(bb, bb->taken);
Bill Buzbeea114add2012-05-03 15:00:40 -0700204 }
buzbeefa57c472012-11-21 12:06:18 -0800205 if (bb->fall_through) {
buzbee311ca162013-02-28 15:56:43 -0800206 CheckForDominanceFrontier(bb, bb->fall_through);
Bill Buzbeea114add2012-05-03 15:00:40 -0700207 }
buzbeefa57c472012-11-21 12:06:18 -0800208 if (bb->successor_block_list.block_list_type != kNotUsed) {
buzbee862a7602013-04-05 10:58:54 -0700209 GrowableArray<SuccessorBlockInfo*>::Iterator iterator(bb->successor_block_list.blocks);
Bill Buzbeea114add2012-05-03 15:00:40 -0700210 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700211 SuccessorBlockInfo *successor_block_info = iterator.Next();
buzbeefa57c472012-11-21 12:06:18 -0800212 if (successor_block_info == NULL) break;
213 BasicBlock* succ_bb = successor_block_info->block;
buzbee311ca162013-02-28 15:56:43 -0800214 CheckForDominanceFrontier(bb, succ_bb);
Bill Buzbeea114add2012-05-03 15:00:40 -0700215 }
216 }
buzbee67bf8852011-08-17 17:51:35 -0700217
Bill Buzbeea114add2012-05-03 15:00:40 -0700218 /* Calculate DF_up */
buzbee862a7602013-04-05 10:58:54 -0700219 ArenaBitVector::Iterator bv_iterator(bb->i_dominated);
Bill Buzbeea114add2012-05-03 15:00:40 -0700220 while (true) {
buzbee52a77fc2012-11-20 19:50:46 -0800221 //TUNING: hot call to BitVectorIteratorNext
buzbee862a7602013-04-05 10:58:54 -0700222 int dominated_idx = bv_iterator.Next();
buzbeefa57c472012-11-21 12:06:18 -0800223 if (dominated_idx == -1) break;
buzbee311ca162013-02-28 15:56:43 -0800224 BasicBlock* dominated_bb = GetBasicBlock(dominated_idx);
buzbee862a7602013-04-05 10:58:54 -0700225 ArenaBitVector::Iterator df_iterator(dominated_bb->dom_frontier);
buzbee67bf8852011-08-17 17:51:35 -0700226 while (true) {
buzbee52a77fc2012-11-20 19:50:46 -0800227 //TUNING: hot call to BitVectorIteratorNext
buzbee862a7602013-04-05 10:58:54 -0700228 int df_up_idx = df_iterator.Next();
buzbeefa57c472012-11-21 12:06:18 -0800229 if (df_up_idx == -1) break;
buzbee311ca162013-02-28 15:56:43 -0800230 BasicBlock* df_up_block = GetBasicBlock(df_up_idx);
231 CheckForDominanceFrontier(bb, df_up_block);
buzbee67bf8852011-08-17 17:51:35 -0700232 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700233 }
buzbee67bf8852011-08-17 17:51:35 -0700234
Bill Buzbeea114add2012-05-03 15:00:40 -0700235 return true;
buzbee67bf8852011-08-17 17:51:35 -0700236}
237
238/* Worker function for initializing domination-related data structures */
buzbee862a7602013-04-05 10:58:54 -0700239void MIRGraph::InitializeDominationInfo(BasicBlock* bb)
buzbee67bf8852011-08-17 17:51:35 -0700240{
buzbee311ca162013-02-28 15:56:43 -0800241 int num_total_blocks = GetBasicBlockListCount();
buzbee67bf8852011-08-17 17:51:35 -0700242
Bill Buzbeea114add2012-05-03 15:00:40 -0700243 if (bb->dominators == NULL ) {
buzbee862a7602013-04-05 10:58:54 -0700244 bb->dominators = new (arena_) ArenaBitVector(arena_, num_total_blocks,
245 false /* expandable */, kBitMapDominators);
246 bb->i_dominated = new (arena_) ArenaBitVector(arena_, num_total_blocks,
247 false /* expandable */, kBitMapIDominated);
248 bb->dom_frontier = new (arena_) ArenaBitVector(arena_, num_total_blocks,
249 false /* expandable */, kBitMapDomFrontier);
Bill Buzbeea114add2012-05-03 15:00:40 -0700250 } else {
buzbee862a7602013-04-05 10:58:54 -0700251 bb->dominators->ClearAllBits();
252 bb->i_dominated->ClearAllBits();
253 bb->dom_frontier->ClearAllBits();
Bill Buzbeea114add2012-05-03 15:00:40 -0700254 }
255 /* Set all bits in the dominator vector */
buzbee862a7602013-04-05 10:58:54 -0700256 bb->dominators->SetInitialBits(num_total_blocks);
buzbee67bf8852011-08-17 17:51:35 -0700257
buzbee862a7602013-04-05 10:58:54 -0700258 return;
buzbee67bf8852011-08-17 17:51:35 -0700259}
260
buzbee5b537102012-01-17 17:33:47 -0800261/*
buzbeefa57c472012-11-21 12:06:18 -0800262 * Walk through the ordered i_dom list until we reach common parent.
263 * Given the ordering of i_dom_list, this common parent represents the
buzbee5b537102012-01-17 17:33:47 -0800264 * last element of the intersection of block1 and block2 dominators.
265 */
buzbee311ca162013-02-28 15:56:43 -0800266int MIRGraph::FindCommonParent(int block1, int block2)
buzbee5b537102012-01-17 17:33:47 -0800267{
Bill Buzbeea114add2012-05-03 15:00:40 -0700268 while (block1 != block2) {
269 while (block1 < block2) {
buzbee311ca162013-02-28 15:56:43 -0800270 block1 = i_dom_list_[block1];
Bill Buzbeea114add2012-05-03 15:00:40 -0700271 DCHECK_NE(block1, NOTVISITED);
buzbee5b537102012-01-17 17:33:47 -0800272 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700273 while (block2 < block1) {
buzbee311ca162013-02-28 15:56:43 -0800274 block2 = i_dom_list_[block2];
Bill Buzbeea114add2012-05-03 15:00:40 -0700275 DCHECK_NE(block2, NOTVISITED);
276 }
277 }
278 return block1;
buzbee5b537102012-01-17 17:33:47 -0800279}
280
281/* Worker function to compute each block's immediate dominator */
buzbee311ca162013-02-28 15:56:43 -0800282bool MIRGraph::ComputeblockIDom(BasicBlock* bb)
buzbee5b537102012-01-17 17:33:47 -0800283{
Bill Buzbeea114add2012-05-03 15:00:40 -0700284 /* Special-case entry block */
buzbee311ca162013-02-28 15:56:43 -0800285 if (bb == GetEntryBlock()) {
buzbee5b537102012-01-17 17:33:47 -0800286 return false;
Bill Buzbeea114add2012-05-03 15:00:40 -0700287 }
288
289 /* Iterate through the predecessors */
buzbee862a7602013-04-05 10:58:54 -0700290 GrowableArray<BasicBlock*>::Iterator iter(bb->predecessors);
Bill Buzbeea114add2012-05-03 15:00:40 -0700291
292 /* Find the first processed predecessor */
buzbee862a7602013-04-05 10:58:54 -0700293 int idom = -1;
Bill Buzbeea114add2012-05-03 15:00:40 -0700294 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700295 BasicBlock* pred_bb = iter.Next();
buzbeefa57c472012-11-21 12:06:18 -0800296 CHECK(pred_bb != NULL);
buzbee311ca162013-02-28 15:56:43 -0800297 if (i_dom_list_[pred_bb->dfs_id] != NOTVISITED) {
buzbeefa57c472012-11-21 12:06:18 -0800298 idom = pred_bb->dfs_id;
Bill Buzbeea114add2012-05-03 15:00:40 -0700299 break;
300 }
301 }
302
303 /* Scan the rest of the predecessors */
304 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700305 BasicBlock* pred_bb = iter.Next();
buzbeefa57c472012-11-21 12:06:18 -0800306 if (!pred_bb) break;
buzbee311ca162013-02-28 15:56:43 -0800307 if (i_dom_list_[pred_bb->dfs_id] == NOTVISITED) {
Bill Buzbeea114add2012-05-03 15:00:40 -0700308 continue;
309 } else {
buzbee311ca162013-02-28 15:56:43 -0800310 idom = FindCommonParent(pred_bb->dfs_id, idom);
Bill Buzbeea114add2012-05-03 15:00:40 -0700311 }
312 }
313
314 DCHECK_NE(idom, NOTVISITED);
315
316 /* Did something change? */
buzbee311ca162013-02-28 15:56:43 -0800317 if (i_dom_list_[bb->dfs_id] != idom) {
318 i_dom_list_[bb->dfs_id] = idom;
Bill Buzbeea114add2012-05-03 15:00:40 -0700319 return true;
320 }
321 return false;
buzbee5b537102012-01-17 17:33:47 -0800322}
323
324/* Worker function to compute each block's domintors */
buzbee311ca162013-02-28 15:56:43 -0800325bool MIRGraph::ComputeBlockDominators(BasicBlock* bb)
buzbee5b537102012-01-17 17:33:47 -0800326{
buzbee311ca162013-02-28 15:56:43 -0800327 if (bb == GetEntryBlock()) {
buzbee862a7602013-04-05 10:58:54 -0700328 bb->dominators->ClearAllBits();
Bill Buzbeea114add2012-05-03 15:00:40 -0700329 } else {
buzbee862a7602013-04-05 10:58:54 -0700330 bb->dominators->Copy(bb->i_dom->dominators);
Bill Buzbeea114add2012-05-03 15:00:40 -0700331 }
buzbee862a7602013-04-05 10:58:54 -0700332 bb->dominators->SetBit(bb->id);
Bill Buzbeea114add2012-05-03 15:00:40 -0700333 return false;
buzbee5b537102012-01-17 17:33:47 -0800334}
335
buzbee311ca162013-02-28 15:56:43 -0800336bool MIRGraph::SetDominators(BasicBlock* bb)
buzbee5b537102012-01-17 17:33:47 -0800337{
buzbee311ca162013-02-28 15:56:43 -0800338 if (bb != GetEntryBlock()) {
339 int idom_dfs_idx = i_dom_list_[bb->dfs_id];
buzbeefa57c472012-11-21 12:06:18 -0800340 DCHECK_NE(idom_dfs_idx, NOTVISITED);
buzbee862a7602013-04-05 10:58:54 -0700341 int i_dom_idx = dfs_post_order_->Get(idom_dfs_idx);
buzbee311ca162013-02-28 15:56:43 -0800342 BasicBlock* i_dom = GetBasicBlock(i_dom_idx);
buzbeefa57c472012-11-21 12:06:18 -0800343 bb->i_dom = i_dom;
344 /* Add bb to the i_dominated set of the immediate dominator block */
buzbee862a7602013-04-05 10:58:54 -0700345 i_dom->i_dominated->SetBit(bb->id);
Bill Buzbeea114add2012-05-03 15:00:40 -0700346 }
347 return false;
buzbee5b537102012-01-17 17:33:47 -0800348}
349
buzbee67bf8852011-08-17 17:51:35 -0700350/* Compute dominators, immediate dominator, and dominance fronter */
buzbee311ca162013-02-28 15:56:43 -0800351void MIRGraph::ComputeDominators()
buzbee67bf8852011-08-17 17:51:35 -0700352{
buzbee311ca162013-02-28 15:56:43 -0800353 int num_reachable_blocks = num_reachable_blocks_;
354 int num_total_blocks = GetBasicBlockListCount();
buzbee67bf8852011-08-17 17:51:35 -0700355
Bill Buzbeea114add2012-05-03 15:00:40 -0700356 /* Initialize domination-related data structures */
buzbee0665fe02013-03-21 12:32:21 -0700357 ReachableNodesIterator iter(this, false /* not iterative */);
buzbee311ca162013-02-28 15:56:43 -0800358 for (BasicBlock* bb = iter.Next(); bb != NULL; bb = iter.Next()) {
359 InitializeDominationInfo(bb);
360 }
buzbee67bf8852011-08-17 17:51:35 -0700361
buzbeefa57c472012-11-21 12:06:18 -0800362 /* Initalize & Clear i_dom_list */
buzbee311ca162013-02-28 15:56:43 -0800363 if (i_dom_list_ == NULL) {
buzbee862a7602013-04-05 10:58:54 -0700364 i_dom_list_ = static_cast<int*>(arena_->NewMem(sizeof(int) * num_reachable_blocks, false,
365 ArenaAllocator::kAllocDFInfo));
Bill Buzbeea114add2012-05-03 15:00:40 -0700366 }
buzbeefa57c472012-11-21 12:06:18 -0800367 for (int i = 0; i < num_reachable_blocks; i++) {
buzbee311ca162013-02-28 15:56:43 -0800368 i_dom_list_[i] = NOTVISITED;
Bill Buzbeea114add2012-05-03 15:00:40 -0700369 }
buzbee5b537102012-01-17 17:33:47 -0800370
buzbeefa57c472012-11-21 12:06:18 -0800371 /* For post-order, last block is entry block. Set its i_dom to istelf */
buzbee311ca162013-02-28 15:56:43 -0800372 DCHECK_EQ(GetEntryBlock()->dfs_id, num_reachable_blocks-1);
373 i_dom_list_[GetEntryBlock()->dfs_id] = GetEntryBlock()->dfs_id;
buzbee5b537102012-01-17 17:33:47 -0800374
Bill Buzbeea114add2012-05-03 15:00:40 -0700375 /* Compute the immediate dominators */
buzbee0665fe02013-03-21 12:32:21 -0700376 ReversePostOrderDfsIterator iter2(this, true /* iterative */);
buzbee311ca162013-02-28 15:56:43 -0800377 bool change = false;
378 for (BasicBlock* bb = iter2.Next(false); bb != NULL; bb = iter2.Next(change)) {
379 change = ComputeblockIDom(bb);
380 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700381
382 /* Set the dominator for the root node */
buzbee862a7602013-04-05 10:58:54 -0700383 GetEntryBlock()->dominators->ClearAllBits();
384 GetEntryBlock()->dominators->SetBit(GetEntryBlock()->id);
Bill Buzbeea114add2012-05-03 15:00:40 -0700385
buzbee311ca162013-02-28 15:56:43 -0800386 if (temp_block_v_ == NULL) {
buzbee862a7602013-04-05 10:58:54 -0700387 temp_block_v_ = new (arena_) ArenaBitVector(arena_, num_total_blocks,
388 false /* expandable */, kBitMapTmpBlockV);
Bill Buzbeea114add2012-05-03 15:00:40 -0700389 } else {
buzbee862a7602013-04-05 10:58:54 -0700390 temp_block_v_->ClearAllBits();
Bill Buzbeea114add2012-05-03 15:00:40 -0700391 }
buzbee311ca162013-02-28 15:56:43 -0800392 GetEntryBlock()->i_dom = NULL;
Bill Buzbeea114add2012-05-03 15:00:40 -0700393
buzbee0665fe02013-03-21 12:32:21 -0700394 ReachableNodesIterator iter3(this, false /* not iterative */);
buzbee311ca162013-02-28 15:56:43 -0800395 for (BasicBlock* bb = iter3.Next(); bb != NULL; bb = iter3.Next()) {
396 SetDominators(bb);
Bill Buzbeea114add2012-05-03 15:00:40 -0700397 }
buzbee67bf8852011-08-17 17:51:35 -0700398
buzbee0665fe02013-03-21 12:32:21 -0700399 ReversePostOrderDfsIterator iter4(this, false /* not iterative */);
buzbee311ca162013-02-28 15:56:43 -0800400 for (BasicBlock* bb = iter4.Next(); bb != NULL; bb = iter4.Next()) {
401 ComputeBlockDominators(bb);
402 }
buzbee5b537102012-01-17 17:33:47 -0800403
Bill Buzbeea114add2012-05-03 15:00:40 -0700404 /*
405 * Now go ahead and compute the post order traversal based on the
buzbeefa57c472012-11-21 12:06:18 -0800406 * i_dominated sets.
Bill Buzbeea114add2012-05-03 15:00:40 -0700407 */
buzbee862a7602013-04-05 10:58:54 -0700408 if (dom_post_order_traversal_ == NULL) {
409 dom_post_order_traversal_ =
410 new (arena_) GrowableArray<int>(arena_, num_reachable_blocks, kGrowableArrayDomPostOrderTraversal);
Bill Buzbeea114add2012-05-03 15:00:40 -0700411 } else {
buzbee862a7602013-04-05 10:58:54 -0700412 dom_post_order_traversal_->Reset();
Bill Buzbeea114add2012-05-03 15:00:40 -0700413 }
buzbee5b537102012-01-17 17:33:47 -0800414
buzbee311ca162013-02-28 15:56:43 -0800415 ComputeDomPostOrderTraversal(GetEntryBlock());
buzbee862a7602013-04-05 10:58:54 -0700416 DCHECK_EQ(dom_post_order_traversal_->Size(), num_reachable_blocks_);
buzbee5b537102012-01-17 17:33:47 -0800417
Bill Buzbeea114add2012-05-03 15:00:40 -0700418 /* Now compute the dominance frontier for each block */
buzbee0665fe02013-03-21 12:32:21 -0700419 PostOrderDOMIterator iter5(this, false /* not iterative */);
buzbee311ca162013-02-28 15:56:43 -0800420 for (BasicBlock* bb = iter5.Next(); bb != NULL; bb = iter5.Next()) {
421 ComputeDominanceFrontier(bb);
422 }
buzbee67bf8852011-08-17 17:51:35 -0700423}
424
425/*
426 * Perform dest U= src1 ^ ~src2
427 * This is probably not general enough to be placed in BitVector.[ch].
428 */
buzbee311ca162013-02-28 15:56:43 -0800429void MIRGraph::ComputeSuccLineIn(ArenaBitVector* dest, const ArenaBitVector* src1,
430 const ArenaBitVector* src2)
buzbee67bf8852011-08-17 17:51:35 -0700431{
buzbee862a7602013-04-05 10:58:54 -0700432 if (dest->GetStorageSize() != src1->GetStorageSize() ||
433 dest->GetStorageSize() != src2->GetStorageSize() ||
434 dest->IsExpandable() != src1->IsExpandable() ||
435 dest->IsExpandable() != src2->IsExpandable()) {
Bill Buzbeea114add2012-05-03 15:00:40 -0700436 LOG(FATAL) << "Incompatible set properties";
437 }
buzbee67bf8852011-08-17 17:51:35 -0700438
Bill Buzbeea114add2012-05-03 15:00:40 -0700439 unsigned int idx;
buzbee862a7602013-04-05 10:58:54 -0700440 for (idx = 0; idx < dest->GetStorageSize(); idx++) {
441 dest->GetRawStorage()[idx] |= src1->GetRawStorageWord(idx) & ~(src2->GetRawStorageWord(idx));
Bill Buzbeea114add2012-05-03 15:00:40 -0700442 }
buzbee67bf8852011-08-17 17:51:35 -0700443}
444
445/*
446 * Iterate through all successor blocks and propagate up the live-in sets.
447 * The calculated result is used for phi-node pruning - where we only need to
448 * insert a phi node if the variable is live-in to the block.
449 */
buzbee311ca162013-02-28 15:56:43 -0800450bool MIRGraph::ComputeBlockLiveIns(BasicBlock* bb)
buzbee67bf8852011-08-17 17:51:35 -0700451{
buzbee311ca162013-02-28 15:56:43 -0800452 ArenaBitVector* temp_dalvik_register_v = temp_dalvik_register_v_;
buzbee67bf8852011-08-17 17:51:35 -0700453
buzbeefa57c472012-11-21 12:06:18 -0800454 if (bb->data_flow_info == NULL) return false;
buzbee862a7602013-04-05 10:58:54 -0700455 temp_dalvik_register_v->Copy(bb->data_flow_info->live_in_v);
buzbeefa57c472012-11-21 12:06:18 -0800456 if (bb->taken && bb->taken->data_flow_info)
457 ComputeSuccLineIn(temp_dalvik_register_v, bb->taken->data_flow_info->live_in_v,
458 bb->data_flow_info->def_v);
459 if (bb->fall_through && bb->fall_through->data_flow_info)
buzbee311ca162013-02-28 15:56:43 -0800460 ComputeSuccLineIn(temp_dalvik_register_v, bb->fall_through->data_flow_info->live_in_v,
buzbeefa57c472012-11-21 12:06:18 -0800461 bb->data_flow_info->def_v);
462 if (bb->successor_block_list.block_list_type != kNotUsed) {
buzbee862a7602013-04-05 10:58:54 -0700463 GrowableArray<SuccessorBlockInfo*>::Iterator iterator(bb->successor_block_list.blocks);
Bill Buzbeea114add2012-05-03 15:00:40 -0700464 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700465 SuccessorBlockInfo *successor_block_info = iterator.Next();
buzbeefa57c472012-11-21 12:06:18 -0800466 if (successor_block_info == NULL) break;
467 BasicBlock* succ_bb = successor_block_info->block;
468 if (succ_bb->data_flow_info) {
buzbee311ca162013-02-28 15:56:43 -0800469 ComputeSuccLineIn(temp_dalvik_register_v, succ_bb->data_flow_info->live_in_v,
buzbeefa57c472012-11-21 12:06:18 -0800470 bb->data_flow_info->def_v);
Bill Buzbeea114add2012-05-03 15:00:40 -0700471 }
buzbee67bf8852011-08-17 17:51:35 -0700472 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700473 }
buzbee862a7602013-04-05 10:58:54 -0700474 if (!temp_dalvik_register_v->Equal(bb->data_flow_info->live_in_v)) {
475 bb->data_flow_info->live_in_v->Copy(temp_dalvik_register_v);
Bill Buzbeea114add2012-05-03 15:00:40 -0700476 return true;
477 }
478 return false;
buzbee67bf8852011-08-17 17:51:35 -0700479}
480
481/* Insert phi nodes to for each variable to the dominance frontiers */
buzbee311ca162013-02-28 15:56:43 -0800482void MIRGraph::InsertPhiNodes()
buzbee67bf8852011-08-17 17:51:35 -0700483{
buzbeefa57c472012-11-21 12:06:18 -0800484 int dalvik_reg;
buzbee862a7602013-04-05 10:58:54 -0700485 ArenaBitVector* phi_blocks =
486 new (arena_) ArenaBitVector(arena_, GetNumBlocks(), false, kBitMapPhi);
487 ArenaBitVector* tmp_blocks =
488 new (arena_) ArenaBitVector(arena_, GetNumBlocks(), false, kBitMapTmpBlocks);
489 ArenaBitVector* input_blocks =
490 new (arena_) ArenaBitVector(arena_, GetNumBlocks(), false, kBitMapInputBlocks);
buzbee67bf8852011-08-17 17:51:35 -0700491
buzbee311ca162013-02-28 15:56:43 -0800492 temp_dalvik_register_v_ =
buzbee862a7602013-04-05 10:58:54 -0700493 new (arena_) ArenaBitVector(arena_, cu_->num_dalvik_registers, false, kBitMapRegisterV);
buzbee67bf8852011-08-17 17:51:35 -0700494
buzbee0665fe02013-03-21 12:32:21 -0700495 PostOrderDfsIterator iter(this, true /* iterative */);
buzbee311ca162013-02-28 15:56:43 -0800496 bool change = false;
497 for (BasicBlock* bb = iter.Next(false); bb != NULL; bb = iter.Next(change)) {
498 change = ComputeBlockLiveIns(bb);
499 }
buzbee67bf8852011-08-17 17:51:35 -0700500
Bill Buzbeea114add2012-05-03 15:00:40 -0700501 /* Iterate through each Dalvik register */
buzbee311ca162013-02-28 15:56:43 -0800502 for (dalvik_reg = cu_->num_dalvik_registers - 1; dalvik_reg >= 0; dalvik_reg--) {
Bill Buzbeea114add2012-05-03 15:00:40 -0700503 bool change;
buzbee67bf8852011-08-17 17:51:35 -0700504
buzbee862a7602013-04-05 10:58:54 -0700505 input_blocks->Copy(def_block_matrix_[dalvik_reg]);
506 phi_blocks->ClearAllBits();
buzbee67bf8852011-08-17 17:51:35 -0700507
Bill Buzbeea114add2012-05-03 15:00:40 -0700508 /* Calculate the phi blocks for each Dalvik register */
509 do {
510 change = false;
buzbee862a7602013-04-05 10:58:54 -0700511 tmp_blocks->ClearAllBits();
512 ArenaBitVector::Iterator iterator(input_blocks);
buzbee67bf8852011-08-17 17:51:35 -0700513
Bill Buzbeea114add2012-05-03 15:00:40 -0700514 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700515 int idx = iterator.Next();
Bill Buzbeea114add2012-05-03 15:00:40 -0700516 if (idx == -1) break;
buzbee311ca162013-02-28 15:56:43 -0800517 BasicBlock* def_bb = GetBasicBlock(idx);
buzbee67bf8852011-08-17 17:51:35 -0700518
buzbeefa57c472012-11-21 12:06:18 -0800519 /* Merge the dominance frontier to tmp_blocks */
buzbee862a7602013-04-05 10:58:54 -0700520 //TUNING: hot call to Union().
buzbeefa57c472012-11-21 12:06:18 -0800521 if (def_bb->dom_frontier != NULL) {
buzbee862a7602013-04-05 10:58:54 -0700522 tmp_blocks->Union(def_bb->dom_frontier);
Bill Buzbeea114add2012-05-03 15:00:40 -0700523 }
buzbee67bf8852011-08-17 17:51:35 -0700524 }
buzbee862a7602013-04-05 10:58:54 -0700525 if (!phi_blocks->Equal(tmp_blocks)) {
Bill Buzbeea114add2012-05-03 15:00:40 -0700526 change = true;
buzbee862a7602013-04-05 10:58:54 -0700527 phi_blocks->Copy(tmp_blocks);
Bill Buzbeea114add2012-05-03 15:00:40 -0700528
529 /*
530 * Iterate through the original blocks plus the new ones in
531 * the dominance frontier.
532 */
buzbee862a7602013-04-05 10:58:54 -0700533 input_blocks->Copy(phi_blocks);
534 input_blocks->Union(def_block_matrix_[dalvik_reg]);
Bill Buzbeea114add2012-05-03 15:00:40 -0700535 }
536 } while (change);
537
538 /*
buzbeefa57c472012-11-21 12:06:18 -0800539 * Insert a phi node for dalvik_reg in the phi_blocks if the Dalvik
Bill Buzbeea114add2012-05-03 15:00:40 -0700540 * register is in the live-in set.
541 */
buzbee862a7602013-04-05 10:58:54 -0700542 ArenaBitVector::Iterator iterator(phi_blocks);
Bill Buzbeea114add2012-05-03 15:00:40 -0700543 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700544 int idx = iterator.Next();
Bill Buzbeea114add2012-05-03 15:00:40 -0700545 if (idx == -1) break;
buzbee311ca162013-02-28 15:56:43 -0800546 BasicBlock* phi_bb = GetBasicBlock(idx);
Bill Buzbeea114add2012-05-03 15:00:40 -0700547 /* Variable will be clobbered before being used - no need for phi */
buzbee862a7602013-04-05 10:58:54 -0700548 if (!phi_bb->data_flow_info->live_in_v->IsBitSet(dalvik_reg)) continue;
549 MIR *phi =
550 static_cast<MIR*>(arena_->NewMem(sizeof(MIR), true, ArenaAllocator::kAllocDFInfo));
buzbeecbd6d442012-11-17 14:11:25 -0800551 phi->dalvikInsn.opcode = static_cast<Instruction::Code>(kMirOpPhi);
buzbeefa57c472012-11-21 12:06:18 -0800552 phi->dalvikInsn.vA = dalvik_reg;
553 phi->offset = phi_bb->start_offset;
buzbee311ca162013-02-28 15:56:43 -0800554 phi->m_unit_index = 0; // Arbitrarily assign all Phi nodes to outermost method.
buzbeefa57c472012-11-21 12:06:18 -0800555 PrependMIR(phi_bb, phi);
buzbee67bf8852011-08-17 17:51:35 -0700556 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700557 }
buzbee67bf8852011-08-17 17:51:35 -0700558}
559
560/*
561 * Worker function to insert phi-operands with latest SSA names from
562 * predecessor blocks
563 */
buzbee311ca162013-02-28 15:56:43 -0800564bool MIRGraph::InsertPhiNodeOperands(BasicBlock* bb)
buzbee67bf8852011-08-17 17:51:35 -0700565{
Bill Buzbeea114add2012-05-03 15:00:40 -0700566 MIR *mir;
buzbee6969d502012-06-15 16:40:31 -0700567 std::vector<int> uses;
buzbeefa57c472012-11-21 12:06:18 -0800568 std::vector<int> incoming_arc;
buzbee67bf8852011-08-17 17:51:35 -0700569
Bill Buzbeea114add2012-05-03 15:00:40 -0700570 /* Phi nodes are at the beginning of each block */
buzbee28c9a832012-11-21 15:39:13 -0800571 for (mir = bb->first_mir_insn; mir != NULL; mir = mir->next) {
buzbeecbd6d442012-11-17 14:11:25 -0800572 if (mir->dalvikInsn.opcode != static_cast<Instruction::Code>(kMirOpPhi))
Bill Buzbeea114add2012-05-03 15:00:40 -0700573 return true;
buzbeefa57c472012-11-21 12:06:18 -0800574 int ssa_reg = mir->ssa_rep->defs[0];
575 DCHECK_GE(ssa_reg, 0); // Shouldn't see compiler temps here
buzbee311ca162013-02-28 15:56:43 -0800576 int v_reg = SRegToVReg(ssa_reg);
buzbee67bf8852011-08-17 17:51:35 -0700577
buzbee6969d502012-06-15 16:40:31 -0700578 uses.clear();
buzbeefa57c472012-11-21 12:06:18 -0800579 incoming_arc.clear();
buzbee67bf8852011-08-17 17:51:35 -0700580
Bill Buzbeea114add2012-05-03 15:00:40 -0700581 /* Iterate through the predecessors */
buzbee862a7602013-04-05 10:58:54 -0700582 GrowableArray<BasicBlock*>::Iterator iter(bb->predecessors);
Bill Buzbeea114add2012-05-03 15:00:40 -0700583 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700584 BasicBlock* pred_bb = iter.Next();
buzbeefa57c472012-11-21 12:06:18 -0800585 if (!pred_bb) break;
586 int ssa_reg = pred_bb->data_flow_info->vreg_to_ssa_map[v_reg];
587 uses.push_back(ssa_reg);
588 incoming_arc.push_back(pred_bb->id);
buzbee67bf8852011-08-17 17:51:35 -0700589 }
590
Bill Buzbeea114add2012-05-03 15:00:40 -0700591 /* Count the number of SSA registers for a Dalvik register */
buzbeefa57c472012-11-21 12:06:18 -0800592 int num_uses = uses.size();
593 mir->ssa_rep->num_uses = num_uses;
594 mir->ssa_rep->uses =
buzbee862a7602013-04-05 10:58:54 -0700595 static_cast<int*>(arena_->NewMem(sizeof(int) * num_uses, false,
596 ArenaAllocator::kAllocDFInfo));
buzbeefa57c472012-11-21 12:06:18 -0800597 mir->ssa_rep->fp_use =
buzbee862a7602013-04-05 10:58:54 -0700598 static_cast<bool*>(arena_->NewMem(sizeof(bool) * num_uses, true,
599 ArenaAllocator::kAllocDFInfo));
buzbee2cfc6392012-05-07 14:51:40 -0700600 int* incoming =
buzbee862a7602013-04-05 10:58:54 -0700601 static_cast<int*>(arena_->NewMem(sizeof(int) * num_uses, false,
602 ArenaAllocator::kAllocDFInfo));
buzbee2cfc6392012-05-07 14:51:40 -0700603 // TODO: Ugly, rework (but don't burden each MIR/LIR for Phi-only needs)
buzbeecbd6d442012-11-17 14:11:25 -0800604 mir->dalvikInsn.vB = reinterpret_cast<uintptr_t>(incoming);
Bill Buzbeea114add2012-05-03 15:00:40 -0700605
Bill Buzbeea114add2012-05-03 15:00:40 -0700606 /* Set the uses array for the phi node */
buzbeefa57c472012-11-21 12:06:18 -0800607 int *use_ptr = mir->ssa_rep->uses;
608 for (int i = 0; i < num_uses; i++) {
609 *use_ptr++ = uses[i];
610 *incoming++ = incoming_arc[i];
Bill Buzbeea114add2012-05-03 15:00:40 -0700611 }
612 }
613
614 return true;
buzbee67bf8852011-08-17 17:51:35 -0700615}
616
buzbee311ca162013-02-28 15:56:43 -0800617void MIRGraph::DoDFSPreOrderSSARename(BasicBlock* block)
buzbeef0cde542011-09-13 14:55:02 -0700618{
619
Bill Buzbeea114add2012-05-03 15:00:40 -0700620 if (block->visited || block->hidden) return;
621 block->visited = true;
buzbeef0cde542011-09-13 14:55:02 -0700622
Bill Buzbeea114add2012-05-03 15:00:40 -0700623 /* Process this block */
buzbee311ca162013-02-28 15:56:43 -0800624 DoSSAConversion(block);
625 int map_size = sizeof(int) * cu_->num_dalvik_registers;
buzbeef0cde542011-09-13 14:55:02 -0700626
Bill Buzbeea114add2012-05-03 15:00:40 -0700627 /* Save SSA map snapshot */
buzbee862a7602013-04-05 10:58:54 -0700628 int* saved_ssa_map =
629 static_cast<int*>(arena_->NewMem(map_size, false, ArenaAllocator::kAllocDalvikToSSAMap));
buzbee311ca162013-02-28 15:56:43 -0800630 memcpy(saved_ssa_map, vreg_to_ssa_map_, map_size);
buzbeef0cde542011-09-13 14:55:02 -0700631
buzbeefa57c472012-11-21 12:06:18 -0800632 if (block->fall_through) {
buzbee311ca162013-02-28 15:56:43 -0800633 DoDFSPreOrderSSARename(block->fall_through);
Bill Buzbeea114add2012-05-03 15:00:40 -0700634 /* Restore SSA map snapshot */
buzbee311ca162013-02-28 15:56:43 -0800635 memcpy(vreg_to_ssa_map_, saved_ssa_map, map_size);
Bill Buzbeea114add2012-05-03 15:00:40 -0700636 }
637 if (block->taken) {
buzbee311ca162013-02-28 15:56:43 -0800638 DoDFSPreOrderSSARename(block->taken);
Bill Buzbeea114add2012-05-03 15:00:40 -0700639 /* Restore SSA map snapshot */
buzbee311ca162013-02-28 15:56:43 -0800640 memcpy(vreg_to_ssa_map_, saved_ssa_map, map_size);
Bill Buzbeea114add2012-05-03 15:00:40 -0700641 }
buzbeefa57c472012-11-21 12:06:18 -0800642 if (block->successor_block_list.block_list_type != kNotUsed) {
buzbee862a7602013-04-05 10:58:54 -0700643 GrowableArray<SuccessorBlockInfo*>::Iterator iterator(block->successor_block_list.blocks);
Bill Buzbeea114add2012-05-03 15:00:40 -0700644 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700645 SuccessorBlockInfo *successor_block_info = iterator.Next();
buzbeefa57c472012-11-21 12:06:18 -0800646 if (successor_block_info == NULL) break;
647 BasicBlock* succ_bb = successor_block_info->block;
buzbee311ca162013-02-28 15:56:43 -0800648 DoDFSPreOrderSSARename(succ_bb);
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);
buzbeef0cde542011-09-13 14:55:02 -0700651 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700652 }
buzbee311ca162013-02-28 15:56:43 -0800653 vreg_to_ssa_map_ = saved_ssa_map;
Bill Buzbeea114add2012-05-03 15:00:40 -0700654 return;
buzbeef0cde542011-09-13 14:55:02 -0700655}
656
buzbee67bf8852011-08-17 17:51:35 -0700657/* Perform SSA transformation for the whole method */
buzbee311ca162013-02-28 15:56:43 -0800658void MIRGraph::SSATransformation()
buzbee67bf8852011-08-17 17:51:35 -0700659{
Bill Buzbeea114add2012-05-03 15:00:40 -0700660 /* Compute the DFS order */
buzbee311ca162013-02-28 15:56:43 -0800661 ComputeDFSOrders();
buzbee67bf8852011-08-17 17:51:35 -0700662
buzbee1fd33462013-03-25 13:40:45 -0700663 /* Compute the dominator info */
664 ComputeDominators();
buzbee67bf8852011-08-17 17:51:35 -0700665
Bill Buzbeea114add2012-05-03 15:00:40 -0700666 /* Allocate data structures in preparation for SSA conversion */
buzbee311ca162013-02-28 15:56:43 -0800667 CompilerInitializeSSAConversion();
buzbee67bf8852011-08-17 17:51:35 -0700668
buzbee1fd33462013-03-25 13:40:45 -0700669 /* Find out the "Dalvik reg def x block" relation */
670 ComputeDefBlockMatrix();
buzbee67bf8852011-08-17 17:51:35 -0700671
buzbee1fd33462013-03-25 13:40:45 -0700672 /* Insert phi nodes to dominance frontiers for all variables */
673 InsertPhiNodes();
buzbee67bf8852011-08-17 17:51:35 -0700674
Bill Buzbeea114add2012-05-03 15:00:40 -0700675 /* Rename register names by local defs and phi nodes */
buzbee1fd33462013-03-25 13:40:45 -0700676 AllNodesIterator iter1(this, false /* not iterative */);
677 for (BasicBlock* bb = iter1.Next(); bb != NULL; bb = iter1.Next()) {
buzbee311ca162013-02-28 15:56:43 -0800678 ClearVisitedFlag(bb);
679 }
680 DoDFSPreOrderSSARename(GetEntryBlock());
buzbee67bf8852011-08-17 17:51:35 -0700681
buzbee1fd33462013-03-25 13:40:45 -0700682 /*
683 * Shared temp bit vector used by each block to count the number of defs
684 * from all the predecessor blocks.
685 */
buzbee862a7602013-04-05 10:58:54 -0700686 temp_ssa_register_v_ =
687 new (arena_) ArenaBitVector(arena_, GetNumSSARegs(), false, kBitMapTempSSARegisterV);
buzbee2cfc6392012-05-07 14:51:40 -0700688
buzbee1fd33462013-03-25 13:40:45 -0700689 /* Insert phi-operands with latest SSA names from predecessor blocks */
690 ReachableNodesIterator iter2(this, false /* not iterative */);
691 for (BasicBlock* bb = iter2.Next(); bb != NULL; bb = iter2.Next()) {
692 InsertPhiNodeOperands(bb);
buzbee311ca162013-02-28 15:56:43 -0800693 }
buzbee1fd33462013-03-25 13:40:45 -0700694
buzbee311ca162013-02-28 15:56:43 -0800695 if (cu_->enable_debug & (1 << kDebugDumpCFG)) {
696 DumpCFG("/sdcard/3_post_ssa_cfg/", false);
Bill Buzbeea114add2012-05-03 15:00:40 -0700697 }
buzbee1fd33462013-03-25 13:40:45 -0700698 if (cu_->enable_debug & (1 << kDebugVerifyDataflow)) {
699 VerifyDataflow();
700 }
buzbee67bf8852011-08-17 17:51:35 -0700701}
Elliott Hughes11d1b0c2012-01-23 16:57:47 -0800702
703} // namespace art