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.dex.code;
18 
19 import com.android.dx.rop.code.BasicBlock;
20 import com.android.dx.rop.code.BasicBlockList;
21 import com.android.dx.rop.code.Insn;
22 import com.android.dx.rop.code.RopMethod;
23 import com.android.dx.rop.code.SourcePosition;
24 
25 /**
26  * Container for the set of {@link CodeAddress} instances associated with
27  * the blocks of a particular method. Each block has a corresponding
28  * start address, end address, and last instruction address.
29  */
30 public final class BlockAddresses {
31     /** {@code non-null;} array containing addresses for the start of each basic
32      * block (indexed by basic block label) */
33     private final CodeAddress[] starts;
34 
35     /** {@code non-null;} array containing addresses for the final instruction
36      * of each basic block (indexed by basic block label) */
37     private final CodeAddress[] lasts;
38 
39     /** {@code non-null;} array containing addresses for the end (just past the
40      * final instruction) of each basic block (indexed by basic block
41      * label) */
42     private final CodeAddress[] ends;
43 
44     /**
45      * Constructs an instance.
46      *
47      * @param method {@code non-null;} the method to have block addresses for
48      */
BlockAddresses(RopMethod method)49     public BlockAddresses(RopMethod method) {
50         BasicBlockList blocks = method.getBlocks();
51         int maxLabel = blocks.getMaxLabel();
52 
53         this.starts = new CodeAddress[maxLabel];
54         this.lasts = new CodeAddress[maxLabel];
55         this.ends = new CodeAddress[maxLabel];
56 
57         setupArrays(method);
58     }
59 
60     /**
61      * Gets the instance for the start of the given block.
62      *
63      * @param block {@code non-null;} the block in question
64      * @return {@code non-null;} the appropriate instance
65      */
getStart(BasicBlock block)66     public CodeAddress getStart(BasicBlock block) {
67         return starts[block.getLabel()];
68     }
69 
70     /**
71      * Gets the instance for the start of the block with the given label.
72      *
73      * @param label {@code non-null;} the label of the block in question
74      * @return {@code non-null;} the appropriate instance
75      */
getStart(int label)76     public CodeAddress getStart(int label) {
77         return starts[label];
78     }
79 
80     /**
81      * Gets the instance for the final instruction of the given block.
82      *
83      * @param block {@code non-null;} the block in question
84      * @return {@code non-null;} the appropriate instance
85      */
getLast(BasicBlock block)86     public CodeAddress getLast(BasicBlock block) {
87         return lasts[block.getLabel()];
88     }
89 
90     /**
91      * Gets the instance for the final instruction of the block with
92      * the given label.
93      *
94      * @param label {@code non-null;} the label of the block in question
95      * @return {@code non-null;} the appropriate instance
96      */
getLast(int label)97     public CodeAddress getLast(int label) {
98         return lasts[label];
99     }
100 
101     /**
102      * Gets the instance for the end (address after the final instruction)
103      * of the given block.
104      *
105      * @param block {@code non-null;} the block in question
106      * @return {@code non-null;} the appropriate instance
107      */
getEnd(BasicBlock block)108     public CodeAddress getEnd(BasicBlock block) {
109         return ends[block.getLabel()];
110     }
111 
112     /**
113      * Gets the instance for the end (address after the final instruction)
114      * of the block with the given label.
115      *
116      * @param label {@code non-null;} the label of the block in question
117      * @return {@code non-null;} the appropriate instance
118      */
getEnd(int label)119     public CodeAddress getEnd(int label) {
120         return ends[label];
121     }
122 
123     /**
124      * Sets up the address arrays.
125      */
setupArrays(RopMethod method)126     private void setupArrays(RopMethod method) {
127         BasicBlockList blocks = method.getBlocks();
128         int sz = blocks.size();
129 
130         for (int i = 0; i < sz; i++) {
131             BasicBlock one = blocks.get(i);
132             int label = one.getLabel();
133             Insn insn = one.getInsns().get(0);
134 
135             starts[label] = new CodeAddress(insn.getPosition());
136 
137             SourcePosition pos = one.getLastInsn().getPosition();
138 
139             lasts[label] = new CodeAddress(pos);
140             ends[label] = new CodeAddress(pos);
141         }
142     }
143 }
144