blob: ccd2454a49598bec08d40059ad17531f05ec32d7 [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();
buzbeee5f01222012-06-14 15:19:35 -070049 if (sbi == NULL) break;
buzbee52a77fc2012-11-20 19:50:46 -080050 res = NeedsVisit(sbi->block);
buzbeee5f01222012-06-14 15:19:35 -070051 if (res != NULL) break;
52 }
53 }
54 }
55 }
56 return res;
57}
58
Brian Carlstrom2ce745c2013-07-17 17:44:30 -070059void MIRGraph::MarkPreOrder(BasicBlock* block) {
buzbeee5f01222012-06-14 15:19:35 -070060 block->visited = true;
buzbeefa57c472012-11-21 12:06:18 -080061 /* Enqueue the pre_order block id */
buzbee862a7602013-04-05 10:58:54 -070062 dfs_order_->Insert(block->id);
buzbeee5f01222012-06-14 15:19:35 -070063}
64
Brian Carlstrom2ce745c2013-07-17 17:44:30 -070065void MIRGraph::RecordDFSOrders(BasicBlock* block) {
buzbeee5f01222012-06-14 15:19:35 -070066 std::vector<BasicBlock*> succ;
buzbee311ca162013-02-28 15:56:43 -080067 MarkPreOrder(block);
buzbeee5f01222012-06-14 15:19:35 -070068 succ.push_back(block);
69 while (!succ.empty()) {
70 BasicBlock* curr = succ.back();
buzbeefa57c472012-11-21 12:06:18 -080071 BasicBlock* next_successor = NextUnvisitedSuccessor(curr);
72 if (next_successor != NULL) {
buzbee311ca162013-02-28 15:56:43 -080073 MarkPreOrder(next_successor);
buzbeefa57c472012-11-21 12:06:18 -080074 succ.push_back(next_successor);
buzbeee5f01222012-06-14 15:19:35 -070075 continue;
76 }
buzbee862a7602013-04-05 10:58:54 -070077 curr->dfs_id = dfs_post_order_->Size();
78 dfs_post_order_->Insert(curr->id);
buzbeee5f01222012-06-14 15:19:35 -070079 succ.pop_back();
80 }
81}
82
buzbee5b537102012-01-17 17:33:47 -080083/* Sort the blocks by the Depth-First-Search */
Brian Carlstrom2ce745c2013-07-17 17:44:30 -070084void MIRGraph::ComputeDFSOrders() {
buzbeefa57c472012-11-21 12:06:18 -080085 /* Initialize or reset the DFS pre_order list */
buzbee862a7602013-04-05 10:58:54 -070086 if (dfs_order_ == NULL) {
87 dfs_order_ = new (arena_) GrowableArray<int>(arena_, GetNumBlocks(), kGrowableArrayDfsOrder);
Bill Buzbeea114add2012-05-03 15:00:40 -070088 } else {
89 /* Just reset the used length on the counter */
buzbee862a7602013-04-05 10:58:54 -070090 dfs_order_->Reset();
Bill Buzbeea114add2012-05-03 15:00:40 -070091 }
buzbee67bf8852011-08-17 17:51:35 -070092
buzbeefa57c472012-11-21 12:06:18 -080093 /* Initialize or reset the DFS post_order list */
buzbee862a7602013-04-05 10:58:54 -070094 if (dfs_post_order_ == NULL) {
95 dfs_post_order_ = new (arena_) GrowableArray<int>(arena_, GetNumBlocks(), kGrowableArrayDfsPostOrder);
Bill Buzbeea114add2012-05-03 15:00:40 -070096 } else {
97 /* Just reset the used length on the counter */
buzbee862a7602013-04-05 10:58:54 -070098 dfs_post_order_->Reset();
Bill Buzbeea114add2012-05-03 15:00:40 -070099 }
buzbee5b537102012-01-17 17:33:47 -0800100
buzbeee5f01222012-06-14 15:19:35 -0700101 // Reset visited flags from all nodes
buzbeea5abf702013-04-12 14:39:29 -0700102 ClearAllVisitedFlags();
103
buzbeee5f01222012-06-14 15:19:35 -0700104 // Record dfs orders
buzbee311ca162013-02-28 15:56:43 -0800105 RecordDFSOrders(GetEntryBlock());
buzbeee5f01222012-06-14 15:19:35 -0700106
buzbee862a7602013-04-05 10:58:54 -0700107 num_reachable_blocks_ = dfs_order_->Size();
buzbee67bf8852011-08-17 17:51:35 -0700108}
109
110/*
111 * Mark block bit on the per-Dalvik register vector to denote that Dalvik
112 * register idx is defined in BasicBlock bb.
113 */
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700114bool MIRGraph::FillDefBlockMatrix(BasicBlock* bb) {
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
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700127void MIRGraph::ComputeDefBlockMatrix() {
buzbee311ca162013-02-28 15:56:43 -0800128 int num_registers = cu_->num_dalvik_registers;
buzbeefa57c472012-11-21 12:06:18 -0800129 /* Allocate num_dalvik_registers bit vector pointers */
buzbee311ca162013-02-28 15:56:43 -0800130 def_block_matrix_ = static_cast<ArenaBitVector**>
buzbee862a7602013-04-05 10:58:54 -0700131 (arena_->NewMem(sizeof(ArenaBitVector *) * num_registers, true,
132 ArenaAllocator::kAllocDFInfo));
Bill Buzbeea114add2012-05-03 15:00:40 -0700133 int i;
buzbee67bf8852011-08-17 17:51:35 -0700134
buzbeefa57c472012-11-21 12:06:18 -0800135 /* Initialize num_register vectors with num_blocks bits each */
136 for (i = 0; i < num_registers; i++) {
buzbee862a7602013-04-05 10:58:54 -0700137 def_block_matrix_[i] =
138 new (arena_) ArenaBitVector(arena_, GetNumBlocks(), false, kBitMapBMatrix);
Bill Buzbeea114add2012-05-03 15:00:40 -0700139 }
buzbee0665fe02013-03-21 12:32:21 -0700140 AllNodesIterator iter(this, false /* not iterative */);
buzbee311ca162013-02-28 15:56:43 -0800141 for (BasicBlock* bb = iter.Next(); bb != NULL; bb = iter.Next()) {
142 FindLocalLiveIn(bb);
143 }
buzbee0665fe02013-03-21 12:32:21 -0700144 AllNodesIterator iter2(this, false /* not iterative */);
buzbee311ca162013-02-28 15:56:43 -0800145 for (BasicBlock* bb = iter2.Next(); bb != NULL; bb = iter2.Next()) {
146 FillDefBlockMatrix(bb);
147 }
buzbee67bf8852011-08-17 17:51:35 -0700148
Bill Buzbeea114add2012-05-03 15:00:40 -0700149 /*
150 * Also set the incoming parameters as defs in the entry block.
151 * Only need to handle the parameters for the outer method.
152 */
buzbee311ca162013-02-28 15:56:43 -0800153 int num_regs = cu_->num_dalvik_registers;
154 int in_reg = num_regs - cu_->num_ins;
buzbeefa57c472012-11-21 12:06:18 -0800155 for (; in_reg < num_regs; in_reg++) {
buzbee862a7602013-04-05 10:58:54 -0700156 def_block_matrix_[in_reg]->SetBit(GetEntryBlock()->id);
Bill Buzbeea114add2012-05-03 15:00:40 -0700157 }
buzbee67bf8852011-08-17 17:51:35 -0700158}
159
buzbeea5abf702013-04-12 14:39:29 -0700160void MIRGraph::ComputeDomPostOrderTraversal(BasicBlock* bb) {
161 if (dom_post_order_traversal_ == NULL) {
162 // First time - create the array.
163 dom_post_order_traversal_ =
164 new (arena_) GrowableArray<int>(arena_, num_reachable_blocks_,
165 kGrowableArrayDomPostOrderTraversal);
166 } else {
167 dom_post_order_traversal_->Reset();
Bill Buzbeea114add2012-05-03 15:00:40 -0700168 }
buzbeea5abf702013-04-12 14:39:29 -0700169 ClearAllVisitedFlags();
170 std::vector<std::pair<BasicBlock*, ArenaBitVector::Iterator*> > work_stack;
171 bb->visited = true;
172 work_stack.push_back(std::make_pair(bb, new (arena_) ArenaBitVector::Iterator(bb->i_dominated)));
173 while (!work_stack.empty()) {
174 std::pair<BasicBlock*, ArenaBitVector::Iterator*> curr = work_stack.back();
175 BasicBlock* curr_bb = curr.first;
176 ArenaBitVector::Iterator* curr_idom_iter = curr.second;
177 int bb_idx = curr_idom_iter->Next();
178 while ((bb_idx != -1) && (NeedsVisit(GetBasicBlock(bb_idx)) == NULL)) {
179 bb_idx = curr_idom_iter->Next();
180 }
181 if (bb_idx != -1) {
182 BasicBlock* new_bb = GetBasicBlock(bb_idx);
183 new_bb->visited = true;
184 work_stack.push_back(
185 std::make_pair(new_bb, new (arena_) ArenaBitVector::Iterator(new_bb->i_dominated)));
186 } else {
187 // no successor/next
188 dom_post_order_traversal_->Insert(curr_bb->id);
189 work_stack.pop_back();
buzbee67bf8852011-08-17 17:51:35 -0700190
buzbeea5abf702013-04-12 14:39:29 -0700191 /* hacky loop detection */
192 if (curr_bb->taken && curr_bb->dominators->IsBitSet(curr_bb->taken->id)) {
193 attributes_ |= METHOD_HAS_LOOP;
194 }
195 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700196 }
buzbee67bf8852011-08-17 17:51:35 -0700197}
198
buzbee311ca162013-02-28 15:56:43 -0800199void MIRGraph::CheckForDominanceFrontier(BasicBlock* dom_bb,
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700200 const BasicBlock* succ_bb) {
Bill Buzbeea114add2012-05-03 15:00:40 -0700201 /*
202 * TODO - evaluate whether phi will ever need to be inserted into exit
203 * blocks.
204 */
buzbeefa57c472012-11-21 12:06:18 -0800205 if (succ_bb->i_dom != dom_bb &&
206 succ_bb->block_type == kDalvikByteCode &&
207 succ_bb->hidden == false) {
buzbee862a7602013-04-05 10:58:54 -0700208 dom_bb->dom_frontier->SetBit(succ_bb->id);
Bill Buzbeea114add2012-05-03 15:00:40 -0700209 }
buzbee67bf8852011-08-17 17:51:35 -0700210}
211
212/* Worker function to compute the dominance frontier */
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700213bool MIRGraph::ComputeDominanceFrontier(BasicBlock* bb) {
Bill Buzbeea114add2012-05-03 15:00:40 -0700214 /* Calculate DF_local */
215 if (bb->taken) {
buzbee311ca162013-02-28 15:56:43 -0800216 CheckForDominanceFrontier(bb, bb->taken);
Bill Buzbeea114add2012-05-03 15:00:40 -0700217 }
buzbeefa57c472012-11-21 12:06:18 -0800218 if (bb->fall_through) {
buzbee311ca162013-02-28 15:56:43 -0800219 CheckForDominanceFrontier(bb, bb->fall_through);
Bill Buzbeea114add2012-05-03 15:00:40 -0700220 }
buzbeefa57c472012-11-21 12:06:18 -0800221 if (bb->successor_block_list.block_list_type != kNotUsed) {
buzbee862a7602013-04-05 10:58:54 -0700222 GrowableArray<SuccessorBlockInfo*>::Iterator iterator(bb->successor_block_list.blocks);
Bill Buzbeea114add2012-05-03 15:00:40 -0700223 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700224 SuccessorBlockInfo *successor_block_info = iterator.Next();
buzbeefa57c472012-11-21 12:06:18 -0800225 if (successor_block_info == NULL) break;
226 BasicBlock* succ_bb = successor_block_info->block;
buzbee311ca162013-02-28 15:56:43 -0800227 CheckForDominanceFrontier(bb, succ_bb);
Bill Buzbeea114add2012-05-03 15:00:40 -0700228 }
229 }
buzbee67bf8852011-08-17 17:51:35 -0700230
Bill Buzbeea114add2012-05-03 15:00:40 -0700231 /* Calculate DF_up */
buzbee862a7602013-04-05 10:58:54 -0700232 ArenaBitVector::Iterator bv_iterator(bb->i_dominated);
Bill Buzbeea114add2012-05-03 15:00:40 -0700233 while (true) {
buzbee52a77fc2012-11-20 19:50:46 -0800234 //TUNING: hot call to BitVectorIteratorNext
buzbee862a7602013-04-05 10:58:54 -0700235 int dominated_idx = bv_iterator.Next();
buzbeefa57c472012-11-21 12:06:18 -0800236 if (dominated_idx == -1) break;
buzbee311ca162013-02-28 15:56:43 -0800237 BasicBlock* dominated_bb = GetBasicBlock(dominated_idx);
buzbee862a7602013-04-05 10:58:54 -0700238 ArenaBitVector::Iterator df_iterator(dominated_bb->dom_frontier);
buzbee67bf8852011-08-17 17:51:35 -0700239 while (true) {
buzbee52a77fc2012-11-20 19:50:46 -0800240 //TUNING: hot call to BitVectorIteratorNext
buzbee862a7602013-04-05 10:58:54 -0700241 int df_up_idx = df_iterator.Next();
buzbeefa57c472012-11-21 12:06:18 -0800242 if (df_up_idx == -1) break;
buzbee311ca162013-02-28 15:56:43 -0800243 BasicBlock* df_up_block = GetBasicBlock(df_up_idx);
244 CheckForDominanceFrontier(bb, df_up_block);
buzbee67bf8852011-08-17 17:51:35 -0700245 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700246 }
buzbee67bf8852011-08-17 17:51:35 -0700247
Bill Buzbeea114add2012-05-03 15:00:40 -0700248 return true;
buzbee67bf8852011-08-17 17:51:35 -0700249}
250
251/* Worker function for initializing domination-related data structures */
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700252void MIRGraph::InitializeDominationInfo(BasicBlock* bb) {
buzbee311ca162013-02-28 15:56:43 -0800253 int num_total_blocks = GetBasicBlockListCount();
buzbee67bf8852011-08-17 17:51:35 -0700254
Bill Buzbeea114add2012-05-03 15:00:40 -0700255 if (bb->dominators == NULL ) {
buzbee862a7602013-04-05 10:58:54 -0700256 bb->dominators = new (arena_) ArenaBitVector(arena_, num_total_blocks,
257 false /* expandable */, kBitMapDominators);
258 bb->i_dominated = new (arena_) ArenaBitVector(arena_, num_total_blocks,
259 false /* expandable */, kBitMapIDominated);
260 bb->dom_frontier = new (arena_) ArenaBitVector(arena_, num_total_blocks,
261 false /* expandable */, kBitMapDomFrontier);
Bill Buzbeea114add2012-05-03 15:00:40 -0700262 } else {
buzbee862a7602013-04-05 10:58:54 -0700263 bb->dominators->ClearAllBits();
264 bb->i_dominated->ClearAllBits();
265 bb->dom_frontier->ClearAllBits();
Bill Buzbeea114add2012-05-03 15:00:40 -0700266 }
267 /* Set all bits in the dominator vector */
buzbee862a7602013-04-05 10:58:54 -0700268 bb->dominators->SetInitialBits(num_total_blocks);
buzbee67bf8852011-08-17 17:51:35 -0700269
buzbee862a7602013-04-05 10:58:54 -0700270 return;
buzbee67bf8852011-08-17 17:51:35 -0700271}
272
buzbee5b537102012-01-17 17:33:47 -0800273/*
buzbeefa57c472012-11-21 12:06:18 -0800274 * Walk through the ordered i_dom list until we reach common parent.
275 * Given the ordering of i_dom_list, this common parent represents the
buzbee5b537102012-01-17 17:33:47 -0800276 * last element of the intersection of block1 and block2 dominators.
277 */
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700278int MIRGraph::FindCommonParent(int block1, int block2) {
Bill Buzbeea114add2012-05-03 15:00:40 -0700279 while (block1 != block2) {
280 while (block1 < block2) {
buzbee311ca162013-02-28 15:56:43 -0800281 block1 = i_dom_list_[block1];
Bill Buzbeea114add2012-05-03 15:00:40 -0700282 DCHECK_NE(block1, NOTVISITED);
buzbee5b537102012-01-17 17:33:47 -0800283 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700284 while (block2 < block1) {
buzbee311ca162013-02-28 15:56:43 -0800285 block2 = i_dom_list_[block2];
Bill Buzbeea114add2012-05-03 15:00:40 -0700286 DCHECK_NE(block2, NOTVISITED);
287 }
288 }
289 return block1;
buzbee5b537102012-01-17 17:33:47 -0800290}
291
292/* Worker function to compute each block's immediate dominator */
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700293bool MIRGraph::ComputeblockIDom(BasicBlock* bb) {
Bill Buzbeea114add2012-05-03 15:00:40 -0700294 /* Special-case entry block */
buzbee311ca162013-02-28 15:56:43 -0800295 if (bb == GetEntryBlock()) {
buzbee5b537102012-01-17 17:33:47 -0800296 return false;
Bill Buzbeea114add2012-05-03 15:00:40 -0700297 }
298
299 /* Iterate through the predecessors */
buzbee862a7602013-04-05 10:58:54 -0700300 GrowableArray<BasicBlock*>::Iterator iter(bb->predecessors);
Bill Buzbeea114add2012-05-03 15:00:40 -0700301
302 /* Find the first processed predecessor */
buzbee862a7602013-04-05 10:58:54 -0700303 int idom = -1;
Bill Buzbeea114add2012-05-03 15:00:40 -0700304 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700305 BasicBlock* pred_bb = iter.Next();
buzbeefa57c472012-11-21 12:06:18 -0800306 CHECK(pred_bb != NULL);
buzbee311ca162013-02-28 15:56:43 -0800307 if (i_dom_list_[pred_bb->dfs_id] != NOTVISITED) {
buzbeefa57c472012-11-21 12:06:18 -0800308 idom = pred_bb->dfs_id;
Bill Buzbeea114add2012-05-03 15:00:40 -0700309 break;
310 }
311 }
312
313 /* Scan the rest of the predecessors */
314 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700315 BasicBlock* pred_bb = iter.Next();
buzbeefa57c472012-11-21 12:06:18 -0800316 if (!pred_bb) break;
buzbee311ca162013-02-28 15:56:43 -0800317 if (i_dom_list_[pred_bb->dfs_id] == NOTVISITED) {
Bill Buzbeea114add2012-05-03 15:00:40 -0700318 continue;
319 } else {
buzbee311ca162013-02-28 15:56:43 -0800320 idom = FindCommonParent(pred_bb->dfs_id, idom);
Bill Buzbeea114add2012-05-03 15:00:40 -0700321 }
322 }
323
324 DCHECK_NE(idom, NOTVISITED);
325
326 /* Did something change? */
buzbee311ca162013-02-28 15:56:43 -0800327 if (i_dom_list_[bb->dfs_id] != idom) {
328 i_dom_list_[bb->dfs_id] = idom;
Bill Buzbeea114add2012-05-03 15:00:40 -0700329 return true;
330 }
331 return false;
buzbee5b537102012-01-17 17:33:47 -0800332}
333
334/* Worker function to compute each block's domintors */
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700335bool MIRGraph::ComputeBlockDominators(BasicBlock* bb) {
buzbee311ca162013-02-28 15:56:43 -0800336 if (bb == GetEntryBlock()) {
buzbee862a7602013-04-05 10:58:54 -0700337 bb->dominators->ClearAllBits();
Bill Buzbeea114add2012-05-03 15:00:40 -0700338 } else {
buzbee862a7602013-04-05 10:58:54 -0700339 bb->dominators->Copy(bb->i_dom->dominators);
Bill Buzbeea114add2012-05-03 15:00:40 -0700340 }
buzbee862a7602013-04-05 10:58:54 -0700341 bb->dominators->SetBit(bb->id);
Bill Buzbeea114add2012-05-03 15:00:40 -0700342 return false;
buzbee5b537102012-01-17 17:33:47 -0800343}
344
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700345bool MIRGraph::SetDominators(BasicBlock* bb) {
buzbee311ca162013-02-28 15:56:43 -0800346 if (bb != GetEntryBlock()) {
347 int idom_dfs_idx = i_dom_list_[bb->dfs_id];
buzbeefa57c472012-11-21 12:06:18 -0800348 DCHECK_NE(idom_dfs_idx, NOTVISITED);
buzbee862a7602013-04-05 10:58:54 -0700349 int i_dom_idx = dfs_post_order_->Get(idom_dfs_idx);
buzbee311ca162013-02-28 15:56:43 -0800350 BasicBlock* i_dom = GetBasicBlock(i_dom_idx);
buzbeefa57c472012-11-21 12:06:18 -0800351 bb->i_dom = i_dom;
352 /* Add bb to the i_dominated set of the immediate dominator block */
buzbee862a7602013-04-05 10:58:54 -0700353 i_dom->i_dominated->SetBit(bb->id);
Bill Buzbeea114add2012-05-03 15:00:40 -0700354 }
355 return false;
buzbee5b537102012-01-17 17:33:47 -0800356}
357
buzbee67bf8852011-08-17 17:51:35 -0700358/* Compute dominators, immediate dominator, and dominance fronter */
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700359void MIRGraph::ComputeDominators() {
buzbee311ca162013-02-28 15:56:43 -0800360 int num_reachable_blocks = num_reachable_blocks_;
361 int num_total_blocks = GetBasicBlockListCount();
buzbee67bf8852011-08-17 17:51:35 -0700362
Bill Buzbeea114add2012-05-03 15:00:40 -0700363 /* Initialize domination-related data structures */
buzbee0665fe02013-03-21 12:32:21 -0700364 ReachableNodesIterator iter(this, false /* not iterative */);
buzbee311ca162013-02-28 15:56:43 -0800365 for (BasicBlock* bb = iter.Next(); bb != NULL; bb = iter.Next()) {
366 InitializeDominationInfo(bb);
367 }
buzbee67bf8852011-08-17 17:51:35 -0700368
buzbeefa57c472012-11-21 12:06:18 -0800369 /* Initalize & Clear i_dom_list */
buzbee311ca162013-02-28 15:56:43 -0800370 if (i_dom_list_ == NULL) {
buzbee862a7602013-04-05 10:58:54 -0700371 i_dom_list_ = static_cast<int*>(arena_->NewMem(sizeof(int) * num_reachable_blocks, false,
372 ArenaAllocator::kAllocDFInfo));
Bill Buzbeea114add2012-05-03 15:00:40 -0700373 }
buzbeefa57c472012-11-21 12:06:18 -0800374 for (int i = 0; i < num_reachable_blocks; i++) {
buzbee311ca162013-02-28 15:56:43 -0800375 i_dom_list_[i] = NOTVISITED;
Bill Buzbeea114add2012-05-03 15:00:40 -0700376 }
buzbee5b537102012-01-17 17:33:47 -0800377
buzbeefa57c472012-11-21 12:06:18 -0800378 /* For post-order, last block is entry block. Set its i_dom to istelf */
buzbee311ca162013-02-28 15:56:43 -0800379 DCHECK_EQ(GetEntryBlock()->dfs_id, num_reachable_blocks-1);
380 i_dom_list_[GetEntryBlock()->dfs_id] = GetEntryBlock()->dfs_id;
buzbee5b537102012-01-17 17:33:47 -0800381
Bill Buzbeea114add2012-05-03 15:00:40 -0700382 /* Compute the immediate dominators */
buzbee0665fe02013-03-21 12:32:21 -0700383 ReversePostOrderDfsIterator iter2(this, true /* iterative */);
buzbee311ca162013-02-28 15:56:43 -0800384 bool change = false;
385 for (BasicBlock* bb = iter2.Next(false); bb != NULL; bb = iter2.Next(change)) {
386 change = ComputeblockIDom(bb);
387 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700388
389 /* Set the dominator for the root node */
buzbee862a7602013-04-05 10:58:54 -0700390 GetEntryBlock()->dominators->ClearAllBits();
391 GetEntryBlock()->dominators->SetBit(GetEntryBlock()->id);
Bill Buzbeea114add2012-05-03 15:00:40 -0700392
buzbee311ca162013-02-28 15:56:43 -0800393 if (temp_block_v_ == NULL) {
buzbee862a7602013-04-05 10:58:54 -0700394 temp_block_v_ = new (arena_) ArenaBitVector(arena_, num_total_blocks,
395 false /* expandable */, kBitMapTmpBlockV);
Bill Buzbeea114add2012-05-03 15:00:40 -0700396 } else {
buzbee862a7602013-04-05 10:58:54 -0700397 temp_block_v_->ClearAllBits();
Bill Buzbeea114add2012-05-03 15:00:40 -0700398 }
buzbee311ca162013-02-28 15:56:43 -0800399 GetEntryBlock()->i_dom = NULL;
Bill Buzbeea114add2012-05-03 15:00:40 -0700400
buzbee0665fe02013-03-21 12:32:21 -0700401 ReachableNodesIterator iter3(this, false /* not iterative */);
buzbee311ca162013-02-28 15:56:43 -0800402 for (BasicBlock* bb = iter3.Next(); bb != NULL; bb = iter3.Next()) {
403 SetDominators(bb);
Bill Buzbeea114add2012-05-03 15:00:40 -0700404 }
buzbee67bf8852011-08-17 17:51:35 -0700405
buzbee0665fe02013-03-21 12:32:21 -0700406 ReversePostOrderDfsIterator iter4(this, false /* not iterative */);
buzbee311ca162013-02-28 15:56:43 -0800407 for (BasicBlock* bb = iter4.Next(); bb != NULL; bb = iter4.Next()) {
408 ComputeBlockDominators(bb);
409 }
buzbee5b537102012-01-17 17:33:47 -0800410
buzbeea5abf702013-04-12 14:39:29 -0700411 // Compute the dominance frontier for each block.
buzbee311ca162013-02-28 15:56:43 -0800412 ComputeDomPostOrderTraversal(GetEntryBlock());
buzbee0665fe02013-03-21 12:32:21 -0700413 PostOrderDOMIterator iter5(this, false /* not iterative */);
buzbee311ca162013-02-28 15:56:43 -0800414 for (BasicBlock* bb = iter5.Next(); bb != NULL; bb = iter5.Next()) {
415 ComputeDominanceFrontier(bb);
416 }
buzbee67bf8852011-08-17 17:51:35 -0700417}
418
419/*
420 * Perform dest U= src1 ^ ~src2
421 * This is probably not general enough to be placed in BitVector.[ch].
422 */
buzbee311ca162013-02-28 15:56:43 -0800423void MIRGraph::ComputeSuccLineIn(ArenaBitVector* dest, const ArenaBitVector* src1,
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700424 const ArenaBitVector* src2) {
buzbee862a7602013-04-05 10:58:54 -0700425 if (dest->GetStorageSize() != src1->GetStorageSize() ||
426 dest->GetStorageSize() != src2->GetStorageSize() ||
427 dest->IsExpandable() != src1->IsExpandable() ||
428 dest->IsExpandable() != src2->IsExpandable()) {
Bill Buzbeea114add2012-05-03 15:00:40 -0700429 LOG(FATAL) << "Incompatible set properties";
430 }
buzbee67bf8852011-08-17 17:51:35 -0700431
Bill Buzbeea114add2012-05-03 15:00:40 -0700432 unsigned int idx;
buzbee862a7602013-04-05 10:58:54 -0700433 for (idx = 0; idx < dest->GetStorageSize(); idx++) {
434 dest->GetRawStorage()[idx] |= src1->GetRawStorageWord(idx) & ~(src2->GetRawStorageWord(idx));
Bill Buzbeea114add2012-05-03 15:00:40 -0700435 }
buzbee67bf8852011-08-17 17:51:35 -0700436}
437
438/*
439 * Iterate through all successor blocks and propagate up the live-in sets.
440 * The calculated result is used for phi-node pruning - where we only need to
441 * insert a phi node if the variable is live-in to the block.
442 */
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700443bool MIRGraph::ComputeBlockLiveIns(BasicBlock* bb) {
buzbee311ca162013-02-28 15:56:43 -0800444 ArenaBitVector* temp_dalvik_register_v = temp_dalvik_register_v_;
buzbee67bf8852011-08-17 17:51:35 -0700445
buzbeefa57c472012-11-21 12:06:18 -0800446 if (bb->data_flow_info == NULL) return false;
buzbee862a7602013-04-05 10:58:54 -0700447 temp_dalvik_register_v->Copy(bb->data_flow_info->live_in_v);
buzbeefa57c472012-11-21 12:06:18 -0800448 if (bb->taken && bb->taken->data_flow_info)
449 ComputeSuccLineIn(temp_dalvik_register_v, bb->taken->data_flow_info->live_in_v,
450 bb->data_flow_info->def_v);
451 if (bb->fall_through && bb->fall_through->data_flow_info)
buzbee311ca162013-02-28 15:56:43 -0800452 ComputeSuccLineIn(temp_dalvik_register_v, bb->fall_through->data_flow_info->live_in_v,
buzbeefa57c472012-11-21 12:06:18 -0800453 bb->data_flow_info->def_v);
454 if (bb->successor_block_list.block_list_type != kNotUsed) {
buzbee862a7602013-04-05 10:58:54 -0700455 GrowableArray<SuccessorBlockInfo*>::Iterator iterator(bb->successor_block_list.blocks);
Bill Buzbeea114add2012-05-03 15:00:40 -0700456 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700457 SuccessorBlockInfo *successor_block_info = iterator.Next();
buzbeefa57c472012-11-21 12:06:18 -0800458 if (successor_block_info == NULL) break;
459 BasicBlock* succ_bb = successor_block_info->block;
460 if (succ_bb->data_flow_info) {
buzbee311ca162013-02-28 15:56:43 -0800461 ComputeSuccLineIn(temp_dalvik_register_v, succ_bb->data_flow_info->live_in_v,
buzbeefa57c472012-11-21 12:06:18 -0800462 bb->data_flow_info->def_v);
Bill Buzbeea114add2012-05-03 15:00:40 -0700463 }
buzbee67bf8852011-08-17 17:51:35 -0700464 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700465 }
buzbee862a7602013-04-05 10:58:54 -0700466 if (!temp_dalvik_register_v->Equal(bb->data_flow_info->live_in_v)) {
467 bb->data_flow_info->live_in_v->Copy(temp_dalvik_register_v);
Bill Buzbeea114add2012-05-03 15:00:40 -0700468 return true;
469 }
470 return false;
buzbee67bf8852011-08-17 17:51:35 -0700471}
472
473/* Insert phi nodes to for each variable to the dominance frontiers */
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700474void MIRGraph::InsertPhiNodes() {
buzbeefa57c472012-11-21 12:06:18 -0800475 int dalvik_reg;
buzbee862a7602013-04-05 10:58:54 -0700476 ArenaBitVector* phi_blocks =
477 new (arena_) ArenaBitVector(arena_, GetNumBlocks(), false, kBitMapPhi);
478 ArenaBitVector* tmp_blocks =
479 new (arena_) ArenaBitVector(arena_, GetNumBlocks(), false, kBitMapTmpBlocks);
480 ArenaBitVector* input_blocks =
481 new (arena_) ArenaBitVector(arena_, GetNumBlocks(), false, kBitMapInputBlocks);
buzbee67bf8852011-08-17 17:51:35 -0700482
buzbee311ca162013-02-28 15:56:43 -0800483 temp_dalvik_register_v_ =
buzbee862a7602013-04-05 10:58:54 -0700484 new (arena_) ArenaBitVector(arena_, cu_->num_dalvik_registers, false, kBitMapRegisterV);
buzbee67bf8852011-08-17 17:51:35 -0700485
buzbee0665fe02013-03-21 12:32:21 -0700486 PostOrderDfsIterator iter(this, true /* iterative */);
buzbee311ca162013-02-28 15:56:43 -0800487 bool change = false;
488 for (BasicBlock* bb = iter.Next(false); bb != NULL; bb = iter.Next(change)) {
489 change = ComputeBlockLiveIns(bb);
490 }
buzbee67bf8852011-08-17 17:51:35 -0700491
Bill Buzbeea114add2012-05-03 15:00:40 -0700492 /* Iterate through each Dalvik register */
buzbee311ca162013-02-28 15:56:43 -0800493 for (dalvik_reg = cu_->num_dalvik_registers - 1; dalvik_reg >= 0; dalvik_reg--) {
Bill Buzbeea114add2012-05-03 15:00:40 -0700494 bool change;
buzbee67bf8852011-08-17 17:51:35 -0700495
buzbee862a7602013-04-05 10:58:54 -0700496 input_blocks->Copy(def_block_matrix_[dalvik_reg]);
497 phi_blocks->ClearAllBits();
buzbee67bf8852011-08-17 17:51:35 -0700498
Bill Buzbeea114add2012-05-03 15:00:40 -0700499 /* Calculate the phi blocks for each Dalvik register */
500 do {
501 change = false;
buzbee862a7602013-04-05 10:58:54 -0700502 tmp_blocks->ClearAllBits();
503 ArenaBitVector::Iterator iterator(input_blocks);
buzbee67bf8852011-08-17 17:51:35 -0700504
Bill Buzbeea114add2012-05-03 15:00:40 -0700505 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700506 int idx = iterator.Next();
Bill Buzbeea114add2012-05-03 15:00:40 -0700507 if (idx == -1) break;
buzbee311ca162013-02-28 15:56:43 -0800508 BasicBlock* def_bb = GetBasicBlock(idx);
buzbee67bf8852011-08-17 17:51:35 -0700509
buzbeefa57c472012-11-21 12:06:18 -0800510 /* Merge the dominance frontier to tmp_blocks */
buzbee862a7602013-04-05 10:58:54 -0700511 //TUNING: hot call to Union().
buzbeefa57c472012-11-21 12:06:18 -0800512 if (def_bb->dom_frontier != NULL) {
buzbee862a7602013-04-05 10:58:54 -0700513 tmp_blocks->Union(def_bb->dom_frontier);
Bill Buzbeea114add2012-05-03 15:00:40 -0700514 }
buzbee67bf8852011-08-17 17:51:35 -0700515 }
buzbee862a7602013-04-05 10:58:54 -0700516 if (!phi_blocks->Equal(tmp_blocks)) {
Bill Buzbeea114add2012-05-03 15:00:40 -0700517 change = true;
buzbee862a7602013-04-05 10:58:54 -0700518 phi_blocks->Copy(tmp_blocks);
Bill Buzbeea114add2012-05-03 15:00:40 -0700519
520 /*
521 * Iterate through the original blocks plus the new ones in
522 * the dominance frontier.
523 */
buzbee862a7602013-04-05 10:58:54 -0700524 input_blocks->Copy(phi_blocks);
525 input_blocks->Union(def_block_matrix_[dalvik_reg]);
Bill Buzbeea114add2012-05-03 15:00:40 -0700526 }
527 } while (change);
528
529 /*
buzbeefa57c472012-11-21 12:06:18 -0800530 * Insert a phi node for dalvik_reg in the phi_blocks if the Dalvik
Bill Buzbeea114add2012-05-03 15:00:40 -0700531 * register is in the live-in set.
532 */
buzbee862a7602013-04-05 10:58:54 -0700533 ArenaBitVector::Iterator iterator(phi_blocks);
Bill Buzbeea114add2012-05-03 15:00:40 -0700534 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700535 int idx = iterator.Next();
Bill Buzbeea114add2012-05-03 15:00:40 -0700536 if (idx == -1) break;
buzbee311ca162013-02-28 15:56:43 -0800537 BasicBlock* phi_bb = GetBasicBlock(idx);
Bill Buzbeea114add2012-05-03 15:00:40 -0700538 /* Variable will be clobbered before being used - no need for phi */
buzbee862a7602013-04-05 10:58:54 -0700539 if (!phi_bb->data_flow_info->live_in_v->IsBitSet(dalvik_reg)) continue;
540 MIR *phi =
541 static_cast<MIR*>(arena_->NewMem(sizeof(MIR), true, ArenaAllocator::kAllocDFInfo));
buzbeecbd6d442012-11-17 14:11:25 -0800542 phi->dalvikInsn.opcode = static_cast<Instruction::Code>(kMirOpPhi);
buzbeefa57c472012-11-21 12:06:18 -0800543 phi->dalvikInsn.vA = dalvik_reg;
544 phi->offset = phi_bb->start_offset;
buzbee311ca162013-02-28 15:56:43 -0800545 phi->m_unit_index = 0; // Arbitrarily assign all Phi nodes to outermost method.
buzbeefa57c472012-11-21 12:06:18 -0800546 PrependMIR(phi_bb, phi);
buzbee67bf8852011-08-17 17:51:35 -0700547 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700548 }
buzbee67bf8852011-08-17 17:51:35 -0700549}
550
551/*
552 * Worker function to insert phi-operands with latest SSA names from
553 * predecessor blocks
554 */
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700555bool MIRGraph::InsertPhiNodeOperands(BasicBlock* bb) {
Bill Buzbeea114add2012-05-03 15:00:40 -0700556 MIR *mir;
buzbee6969d502012-06-15 16:40:31 -0700557 std::vector<int> uses;
buzbeefa57c472012-11-21 12:06:18 -0800558 std::vector<int> incoming_arc;
buzbee67bf8852011-08-17 17:51:35 -0700559
Bill Buzbeea114add2012-05-03 15:00:40 -0700560 /* Phi nodes are at the beginning of each block */
buzbee28c9a832012-11-21 15:39:13 -0800561 for (mir = bb->first_mir_insn; mir != NULL; mir = mir->next) {
buzbeecbd6d442012-11-17 14:11:25 -0800562 if (mir->dalvikInsn.opcode != static_cast<Instruction::Code>(kMirOpPhi))
Bill Buzbeea114add2012-05-03 15:00:40 -0700563 return true;
buzbeefa57c472012-11-21 12:06:18 -0800564 int ssa_reg = mir->ssa_rep->defs[0];
565 DCHECK_GE(ssa_reg, 0); // Shouldn't see compiler temps here
buzbee311ca162013-02-28 15:56:43 -0800566 int v_reg = SRegToVReg(ssa_reg);
buzbee67bf8852011-08-17 17:51:35 -0700567
buzbee6969d502012-06-15 16:40:31 -0700568 uses.clear();
buzbeefa57c472012-11-21 12:06:18 -0800569 incoming_arc.clear();
buzbee67bf8852011-08-17 17:51:35 -0700570
Bill Buzbeea114add2012-05-03 15:00:40 -0700571 /* Iterate through the predecessors */
buzbee862a7602013-04-05 10:58:54 -0700572 GrowableArray<BasicBlock*>::Iterator iter(bb->predecessors);
Bill Buzbeea114add2012-05-03 15:00:40 -0700573 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700574 BasicBlock* pred_bb = iter.Next();
buzbeefa57c472012-11-21 12:06:18 -0800575 if (!pred_bb) break;
576 int ssa_reg = pred_bb->data_flow_info->vreg_to_ssa_map[v_reg];
577 uses.push_back(ssa_reg);
578 incoming_arc.push_back(pred_bb->id);
buzbee67bf8852011-08-17 17:51:35 -0700579 }
580
Bill Buzbeea114add2012-05-03 15:00:40 -0700581 /* Count the number of SSA registers for a Dalvik register */
buzbeefa57c472012-11-21 12:06:18 -0800582 int num_uses = uses.size();
583 mir->ssa_rep->num_uses = num_uses;
584 mir->ssa_rep->uses =
buzbee862a7602013-04-05 10:58:54 -0700585 static_cast<int*>(arena_->NewMem(sizeof(int) * num_uses, false,
586 ArenaAllocator::kAllocDFInfo));
buzbeefa57c472012-11-21 12:06:18 -0800587 mir->ssa_rep->fp_use =
buzbee862a7602013-04-05 10:58:54 -0700588 static_cast<bool*>(arena_->NewMem(sizeof(bool) * num_uses, true,
589 ArenaAllocator::kAllocDFInfo));
buzbee2cfc6392012-05-07 14:51:40 -0700590 int* incoming =
buzbee862a7602013-04-05 10:58:54 -0700591 static_cast<int*>(arena_->NewMem(sizeof(int) * num_uses, false,
592 ArenaAllocator::kAllocDFInfo));
buzbee2cfc6392012-05-07 14:51:40 -0700593 // TODO: Ugly, rework (but don't burden each MIR/LIR for Phi-only needs)
buzbeecbd6d442012-11-17 14:11:25 -0800594 mir->dalvikInsn.vB = reinterpret_cast<uintptr_t>(incoming);
Bill Buzbeea114add2012-05-03 15:00:40 -0700595
Bill Buzbeea114add2012-05-03 15:00:40 -0700596 /* Set the uses array for the phi node */
buzbeefa57c472012-11-21 12:06:18 -0800597 int *use_ptr = mir->ssa_rep->uses;
598 for (int i = 0; i < num_uses; i++) {
599 *use_ptr++ = uses[i];
600 *incoming++ = incoming_arc[i];
Bill Buzbeea114add2012-05-03 15:00:40 -0700601 }
602 }
603
604 return true;
buzbee67bf8852011-08-17 17:51:35 -0700605}
606
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700607void MIRGraph::DoDFSPreOrderSSARename(BasicBlock* block) {
buzbeef0cde542011-09-13 14:55:02 -0700608
Bill Buzbeea114add2012-05-03 15:00:40 -0700609 if (block->visited || block->hidden) return;
610 block->visited = true;
buzbeef0cde542011-09-13 14:55:02 -0700611
Bill Buzbeea114add2012-05-03 15:00:40 -0700612 /* Process this block */
buzbee311ca162013-02-28 15:56:43 -0800613 DoSSAConversion(block);
614 int map_size = sizeof(int) * cu_->num_dalvik_registers;
buzbeef0cde542011-09-13 14:55:02 -0700615
Bill Buzbeea114add2012-05-03 15:00:40 -0700616 /* Save SSA map snapshot */
buzbee862a7602013-04-05 10:58:54 -0700617 int* saved_ssa_map =
618 static_cast<int*>(arena_->NewMem(map_size, false, ArenaAllocator::kAllocDalvikToSSAMap));
buzbee311ca162013-02-28 15:56:43 -0800619 memcpy(saved_ssa_map, vreg_to_ssa_map_, map_size);
buzbeef0cde542011-09-13 14:55:02 -0700620
buzbeefa57c472012-11-21 12:06:18 -0800621 if (block->fall_through) {
buzbee311ca162013-02-28 15:56:43 -0800622 DoDFSPreOrderSSARename(block->fall_through);
Bill Buzbeea114add2012-05-03 15:00:40 -0700623 /* Restore SSA map snapshot */
buzbee311ca162013-02-28 15:56:43 -0800624 memcpy(vreg_to_ssa_map_, saved_ssa_map, map_size);
Bill Buzbeea114add2012-05-03 15:00:40 -0700625 }
626 if (block->taken) {
buzbee311ca162013-02-28 15:56:43 -0800627 DoDFSPreOrderSSARename(block->taken);
Bill Buzbeea114add2012-05-03 15:00:40 -0700628 /* Restore SSA map snapshot */
buzbee311ca162013-02-28 15:56:43 -0800629 memcpy(vreg_to_ssa_map_, saved_ssa_map, map_size);
Bill Buzbeea114add2012-05-03 15:00:40 -0700630 }
buzbeefa57c472012-11-21 12:06:18 -0800631 if (block->successor_block_list.block_list_type != kNotUsed) {
buzbee862a7602013-04-05 10:58:54 -0700632 GrowableArray<SuccessorBlockInfo*>::Iterator iterator(block->successor_block_list.blocks);
Bill Buzbeea114add2012-05-03 15:00:40 -0700633 while (true) {
buzbee862a7602013-04-05 10:58:54 -0700634 SuccessorBlockInfo *successor_block_info = iterator.Next();
buzbeefa57c472012-11-21 12:06:18 -0800635 if (successor_block_info == NULL) break;
636 BasicBlock* succ_bb = successor_block_info->block;
buzbee311ca162013-02-28 15:56:43 -0800637 DoDFSPreOrderSSARename(succ_bb);
Bill Buzbeea114add2012-05-03 15:00:40 -0700638 /* Restore SSA map snapshot */
buzbee311ca162013-02-28 15:56:43 -0800639 memcpy(vreg_to_ssa_map_, saved_ssa_map, map_size);
buzbeef0cde542011-09-13 14:55:02 -0700640 }
Bill Buzbeea114add2012-05-03 15:00:40 -0700641 }
buzbee311ca162013-02-28 15:56:43 -0800642 vreg_to_ssa_map_ = saved_ssa_map;
Bill Buzbeea114add2012-05-03 15:00:40 -0700643 return;
buzbeef0cde542011-09-13 14:55:02 -0700644}
645
buzbee67bf8852011-08-17 17:51:35 -0700646/* Perform SSA transformation for the whole method */
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700647void MIRGraph::SSATransformation() {
Bill Buzbeea114add2012-05-03 15:00:40 -0700648 /* Compute the DFS order */
buzbee311ca162013-02-28 15:56:43 -0800649 ComputeDFSOrders();
buzbee67bf8852011-08-17 17:51:35 -0700650
buzbee1fd33462013-03-25 13:40:45 -0700651 /* Compute the dominator info */
652 ComputeDominators();
buzbee67bf8852011-08-17 17:51:35 -0700653
Bill Buzbeea114add2012-05-03 15:00:40 -0700654 /* Allocate data structures in preparation for SSA conversion */
buzbee311ca162013-02-28 15:56:43 -0800655 CompilerInitializeSSAConversion();
buzbee67bf8852011-08-17 17:51:35 -0700656
buzbee1fd33462013-03-25 13:40:45 -0700657 /* Find out the "Dalvik reg def x block" relation */
658 ComputeDefBlockMatrix();
buzbee67bf8852011-08-17 17:51:35 -0700659
buzbee1fd33462013-03-25 13:40:45 -0700660 /* Insert phi nodes to dominance frontiers for all variables */
661 InsertPhiNodes();
buzbee67bf8852011-08-17 17:51:35 -0700662
Bill Buzbeea114add2012-05-03 15:00:40 -0700663 /* Rename register names by local defs and phi nodes */
buzbeea5abf702013-04-12 14:39:29 -0700664 ClearAllVisitedFlags();
buzbee311ca162013-02-28 15:56:43 -0800665 DoDFSPreOrderSSARename(GetEntryBlock());
buzbee67bf8852011-08-17 17:51:35 -0700666
buzbee1fd33462013-03-25 13:40:45 -0700667 /*
668 * Shared temp bit vector used by each block to count the number of defs
669 * from all the predecessor blocks.
670 */
buzbee862a7602013-04-05 10:58:54 -0700671 temp_ssa_register_v_ =
672 new (arena_) ArenaBitVector(arena_, GetNumSSARegs(), false, kBitMapTempSSARegisterV);
buzbee2cfc6392012-05-07 14:51:40 -0700673
buzbee1fd33462013-03-25 13:40:45 -0700674 /* Insert phi-operands with latest SSA names from predecessor blocks */
675 ReachableNodesIterator iter2(this, false /* not iterative */);
676 for (BasicBlock* bb = iter2.Next(); bb != NULL; bb = iter2.Next()) {
677 InsertPhiNodeOperands(bb);
buzbee311ca162013-02-28 15:56:43 -0800678 }
buzbee1fd33462013-03-25 13:40:45 -0700679
buzbee311ca162013-02-28 15:56:43 -0800680 if (cu_->enable_debug & (1 << kDebugDumpCFG)) {
681 DumpCFG("/sdcard/3_post_ssa_cfg/", false);
Bill Buzbeea114add2012-05-03 15:00:40 -0700682 }
buzbee1fd33462013-03-25 13:40:45 -0700683 if (cu_->enable_debug & (1 << kDebugVerifyDataflow)) {
684 VerifyDataflow();
685 }
buzbee67bf8852011-08-17 17:51:35 -0700686}
Elliott Hughes11d1b0c2012-01-23 16:57:47 -0800687
688} // namespace art