1 /*
2  * Copyright (C) 2007 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 
17 package com.android.dx.ssa;
18 
19 import com.android.dx.util.IntSet;
20 import java.util.ArrayList;
21 import java.util.BitSet;
22 
23 /**
24  * Calculates the dominance-frontiers of a method's basic blocks.
25  * Algorithm from "A Simple, Fast Dominance Algorithm" by Cooper,
26  * Harvey, and Kennedy; transliterated to Java.
27  */
28 public class DomFront {
29     /** local debug flag */
30     private static final boolean DEBUG = false;
31 
32     /** {@code non-null;} method being processed */
33     private final SsaMethod meth;
34 
35     private final ArrayList<SsaBasicBlock> nodes;
36 
37     private final DomInfo[] domInfos;
38 
39     /**
40      * Dominance-frontier information for a single basic block.
41      */
42     public static class DomInfo {
43         /**
44          * {@code null-ok;} the dominance frontier set indexed by
45          * block index
46          */
47         public IntSet dominanceFrontiers;
48 
49         /** {@code >= 0 after run();} the index of the immediate dominator */
50         public int idom = -1;
51     }
52 
53     /**
54      * Constructs instance. Call {@link DomFront#run} to process.
55      *
56      * @param meth {@code non-null;} method to process
57      */
DomFront(SsaMethod meth)58     public DomFront(SsaMethod meth) {
59         this.meth = meth;
60         nodes = meth.getBlocks();
61 
62         int szNodes = nodes.size();
63         domInfos = new DomInfo[szNodes];
64 
65         for (int i = 0; i < szNodes; i++) {
66             domInfos[i] = new DomInfo();
67         }
68     }
69 
70     /**
71      * Calculates the dominance frontier information for the method.
72      *
73      * @return {@code non-null;} an array of DomInfo structures
74      */
run()75     public DomInfo[] run() {
76         int szNodes = nodes.size();
77 
78         if (DEBUG) {
79             for (int i = 0; i < szNodes; i++) {
80                 SsaBasicBlock node = nodes.get(i);
81                 System.out.println("pred[" + i + "]: "
82                         + node.getPredecessors());
83             }
84         }
85 
86         Dominators methDom = Dominators.make(meth, domInfos, false);
87 
88         if (DEBUG) {
89             for (int i = 0; i < szNodes; i++) {
90                 DomInfo info = domInfos[i];
91                 System.out.println("idom[" + i + "]: "
92                         + info.idom);
93             }
94         }
95 
96         buildDomTree();
97 
98         if (DEBUG) {
99             debugPrintDomChildren();
100         }
101 
102         for (int i = 0; i < szNodes; i++) {
103             domInfos[i].dominanceFrontiers
104                     = SetFactory.makeDomFrontSet(szNodes);
105         }
106 
107         calcDomFronts();
108 
109         if (DEBUG) {
110             for (int i = 0; i < szNodes; i++) {
111                 System.out.println("df[" + i + "]: "
112                         + domInfos[i].dominanceFrontiers);
113             }
114         }
115 
116         return domInfos;
117     }
118 
debugPrintDomChildren()119     private void debugPrintDomChildren() {
120         int szNodes = nodes.size();
121 
122         for (int i = 0; i < szNodes; i++) {
123             SsaBasicBlock node = nodes.get(i);
124             StringBuffer sb = new StringBuffer();
125 
126             sb.append('{');
127             boolean comma = false;
128             for (SsaBasicBlock child : node.getDomChildren()) {
129                 if (comma) {
130                     sb.append(',');
131                 }
132                 sb.append(child);
133                 comma = true;
134             }
135             sb.append('}');
136 
137             System.out.println("domChildren[" + node + "]: "
138                     + sb);
139         }
140     }
141 
142     /**
143      * The dominators algorithm leaves us knowing who the immediate dominator
144      * is for each node. This sweeps the node list and builds the proper
145      * dominance tree.
146      */
buildDomTree()147     private void buildDomTree() {
148         int szNodes = nodes.size();
149 
150         for (int i = 0; i < szNodes; i++) {
151             DomInfo info = domInfos[i];
152 
153             if (info.idom == -1) continue;
154 
155             SsaBasicBlock domParent = nodes.get(info.idom);
156             domParent.addDomChild(nodes.get(i));
157         }
158     }
159 
160     /**
161      * Calculates the dominance-frontier set.
162      * from "A Simple, Fast Dominance Algorithm" by Cooper,
163      * Harvey, and Kennedy; transliterated to Java.
164      */
calcDomFronts()165     private void calcDomFronts() {
166         int szNodes = nodes.size();
167 
168         for (int b = 0; b < szNodes; b++) {
169             SsaBasicBlock nb = nodes.get(b);
170             DomInfo nbInfo = domInfos[b];
171             BitSet pred = nb.getPredecessors();
172 
173             if (pred.cardinality() > 1) {
174                 for (int i = pred.nextSetBit(0); i >= 0;
175                      i = pred.nextSetBit(i + 1)) {
176 
177                     for (int runnerIndex = i;
178                          runnerIndex != nbInfo.idom; /* empty */) {
179                         /*
180                          * We can stop if we hit a block we already
181                          * added label to, since we must be at a part
182                          * of the dom tree we have seen before.
183                          */
184                         if (runnerIndex == -1) break;
185 
186                         DomInfo runnerInfo = domInfos[runnerIndex];
187 
188                         if (runnerInfo.dominanceFrontiers.has(b)) {
189                             break;
190                         }
191 
192                         // Add b to runner's dominance frontier set.
193                         runnerInfo.dominanceFrontiers.add(b);
194                         runnerIndex = runnerInfo.idom;
195                     }
196                 }
197             }
198         }
199     }
200 }
201