blob: 5f632c061d5770344c969734a9fac99354d4d487 [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
17#include "Dalvik.h"
18#include "Dataflow.h"
buzbee67bf8852011-08-17 17:51:35 -070019
Elliott Hughes11d1b0c2012-01-23 16:57:47 -080020namespace art {
21
buzbee67bf8852011-08-17 17:51:35 -070022/*
23 * Main table containing data flow attributes for each bytecode. The
24 * first kNumPackedOpcodes entries are for Dalvik bytecode
25 * instructions, where extended opcode at the MIR level are appended
26 * afterwards.
27 *
28 * TODO - many optimization flags are incomplete - they will only limit the
29 * scope of optimizations but will not cause mis-optimizations.
30 */
buzbeeba938cb2012-02-03 14:47:55 -080031const int oatDataFlowAttributes[kMirOpLast] = {
Elliott Hughesadb8c672012-03-06 16:49:32 -080032 // 00 NOP
buzbee67bf8852011-08-17 17:51:35 -070033 DF_NOP,
34
Elliott Hughesadb8c672012-03-06 16:49:32 -080035 // 01 MOVE vA, vB
buzbee67bf8852011-08-17 17:51:35 -070036 DF_DA | DF_UB | DF_IS_MOVE,
37
Elliott Hughesadb8c672012-03-06 16:49:32 -080038 // 02 MOVE_FROM16 vAA, vBBBB
buzbee67bf8852011-08-17 17:51:35 -070039 DF_DA | DF_UB | DF_IS_MOVE,
40
Elliott Hughesadb8c672012-03-06 16:49:32 -080041 // 03 MOVE_16 vAAAA, vBBBB
buzbee67bf8852011-08-17 17:51:35 -070042 DF_DA | DF_UB | DF_IS_MOVE,
43
Elliott Hughesadb8c672012-03-06 16:49:32 -080044 // 04 MOVE_WIDE vA, vB
buzbee67bf8852011-08-17 17:51:35 -070045 DF_DA_WIDE | DF_UB_WIDE | DF_IS_MOVE,
46
Elliott Hughesadb8c672012-03-06 16:49:32 -080047 // 05 MOVE_WIDE_FROM16 vAA, vBBBB
buzbee67bf8852011-08-17 17:51:35 -070048 DF_DA_WIDE | DF_UB_WIDE | DF_IS_MOVE,
49
Elliott Hughesadb8c672012-03-06 16:49:32 -080050 // 06 MOVE_WIDE_16 vAAAA, vBBBB
buzbee67bf8852011-08-17 17:51:35 -070051 DF_DA_WIDE | DF_UB_WIDE | DF_IS_MOVE,
52
Elliott Hughesadb8c672012-03-06 16:49:32 -080053 // 07 MOVE_OBJECT vA, vB
buzbee67bc2362011-10-11 18:08:40 -070054 DF_DA | DF_UB | DF_NULL_TRANSFER_0 | DF_IS_MOVE | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -070055
Elliott Hughesadb8c672012-03-06 16:49:32 -080056 // 08 MOVE_OBJECT_FROM16 vAA, vBBBB
buzbee67bc2362011-10-11 18:08:40 -070057 DF_DA | DF_UB | DF_NULL_TRANSFER_0 | DF_IS_MOVE | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -070058
Elliott Hughesadb8c672012-03-06 16:49:32 -080059 // 09 MOVE_OBJECT_16 vAAAA, vBBBB
buzbee67bc2362011-10-11 18:08:40 -070060 DF_DA | DF_UB | DF_NULL_TRANSFER_0 | DF_IS_MOVE | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -070061
Elliott Hughesadb8c672012-03-06 16:49:32 -080062 // 0A MOVE_RESULT vAA
buzbee67bf8852011-08-17 17:51:35 -070063 DF_DA,
64
Elliott Hughesadb8c672012-03-06 16:49:32 -080065 // 0B MOVE_RESULT_WIDE vAA
buzbee67bf8852011-08-17 17:51:35 -070066 DF_DA_WIDE,
67
Elliott Hughesadb8c672012-03-06 16:49:32 -080068 // 0C MOVE_RESULT_OBJECT vAA
buzbee67bc2362011-10-11 18:08:40 -070069 DF_DA | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -070070
Elliott Hughesadb8c672012-03-06 16:49:32 -080071 // 0D MOVE_EXCEPTION vAA
buzbee67bc2362011-10-11 18:08:40 -070072 DF_DA | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -070073
Elliott Hughesadb8c672012-03-06 16:49:32 -080074 // 0E RETURN_VOID
buzbee67bf8852011-08-17 17:51:35 -070075 DF_NOP,
76
Elliott Hughesadb8c672012-03-06 16:49:32 -080077 // 0F RETURN vAA
buzbee67bf8852011-08-17 17:51:35 -070078 DF_UA,
79
Elliott Hughesadb8c672012-03-06 16:49:32 -080080 // 10 RETURN_WIDE vAA
buzbee67bf8852011-08-17 17:51:35 -070081 DF_UA_WIDE,
82
Elliott Hughesadb8c672012-03-06 16:49:32 -080083 // 11 RETURN_OBJECT vAA
buzbee67bc2362011-10-11 18:08:40 -070084 DF_UA | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -070085
Elliott Hughesadb8c672012-03-06 16:49:32 -080086 // 12 CONST_4 vA, #+B
buzbee67bf8852011-08-17 17:51:35 -070087 DF_DA | DF_SETS_CONST,
88
Elliott Hughesadb8c672012-03-06 16:49:32 -080089 // 13 CONST_16 vAA, #+BBBB
buzbee67bf8852011-08-17 17:51:35 -070090 DF_DA | DF_SETS_CONST,
91
Elliott Hughesadb8c672012-03-06 16:49:32 -080092 // 14 CONST vAA, #+BBBBBBBB
buzbee67bf8852011-08-17 17:51:35 -070093 DF_DA | DF_SETS_CONST,
94
Elliott Hughesadb8c672012-03-06 16:49:32 -080095 // 15 CONST_HIGH16 VAA, #+BBBB0000
buzbee67bf8852011-08-17 17:51:35 -070096 DF_DA | DF_SETS_CONST,
97
Elliott Hughesadb8c672012-03-06 16:49:32 -080098 // 16 CONST_WIDE_16 vAA, #+BBBB
buzbee67bf8852011-08-17 17:51:35 -070099 DF_DA_WIDE | DF_SETS_CONST,
100
Elliott Hughesadb8c672012-03-06 16:49:32 -0800101 // 17 CONST_WIDE_32 vAA, #+BBBBBBBB
buzbee67bf8852011-08-17 17:51:35 -0700102 DF_DA_WIDE | DF_SETS_CONST,
103
Elliott Hughesadb8c672012-03-06 16:49:32 -0800104 // 18 CONST_WIDE vAA, #+BBBBBBBBBBBBBBBB
buzbee67bf8852011-08-17 17:51:35 -0700105 DF_DA_WIDE | DF_SETS_CONST,
106
Elliott Hughesadb8c672012-03-06 16:49:32 -0800107 // 19 CONST_WIDE_HIGH16 vAA, #+BBBB000000000000
buzbee67bf8852011-08-17 17:51:35 -0700108 DF_DA_WIDE | DF_SETS_CONST,
109
Elliott Hughesadb8c672012-03-06 16:49:32 -0800110 // 1A CONST_STRING vAA, string@BBBB
buzbee67bc2362011-10-11 18:08:40 -0700111 DF_DA | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -0700112
Elliott Hughesadb8c672012-03-06 16:49:32 -0800113 // 1B CONST_STRING_JUMBO vAA, string@BBBBBBBB
buzbee67bc2362011-10-11 18:08:40 -0700114 DF_DA | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -0700115
Elliott Hughesadb8c672012-03-06 16:49:32 -0800116 // 1C CONST_CLASS vAA, type@BBBB
buzbee67bc2362011-10-11 18:08:40 -0700117 DF_DA | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -0700118
Elliott Hughesadb8c672012-03-06 16:49:32 -0800119 // 1D MONITOR_ENTER vAA
buzbee67bc2362011-10-11 18:08:40 -0700120 DF_UA | DF_NULL_CHK_0 | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -0700121
Elliott Hughesadb8c672012-03-06 16:49:32 -0800122 // 1E MONITOR_EXIT vAA
buzbee67bc2362011-10-11 18:08:40 -0700123 DF_UA | DF_NULL_CHK_0 | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -0700124
Elliott Hughesadb8c672012-03-06 16:49:32 -0800125 // 1F CHK_CAST vAA, type@BBBB
buzbee9c044ce2012-03-18 13:24:07 -0700126 DF_UA | DF_CORE_A | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700127
Elliott Hughesadb8c672012-03-06 16:49:32 -0800128 // 20 INSTANCE_OF vA, vB, type@CCCC
buzbee9c044ce2012-03-18 13:24:07 -0700129 DF_DA | DF_UB | DF_CORE_A | DF_CORE_B | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700130
Elliott Hughesadb8c672012-03-06 16:49:32 -0800131 // 21 ARRAY_LENGTH vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700132 DF_DA | DF_UB | DF_NULL_CHK_0 | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700133
Elliott Hughesadb8c672012-03-06 16:49:32 -0800134 // 22 NEW_INSTANCE vAA, type@BBBB
buzbee9c044ce2012-03-18 13:24:07 -0700135 DF_DA | DF_NON_NULL_DST | DF_CORE_A | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700136
Elliott Hughesadb8c672012-03-06 16:49:32 -0800137 // 23 NEW_ARRAY vA, vB, type@CCCC
buzbee9c044ce2012-03-18 13:24:07 -0700138 DF_DA | DF_UB | DF_NON_NULL_DST | DF_CORE_A | DF_CORE_B | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700139
Elliott Hughesadb8c672012-03-06 16:49:32 -0800140 // 24 FILLED_NEW_ARRAY {vD, vE, vF, vG, vA}
buzbee9c044ce2012-03-18 13:24:07 -0700141 DF_FORMAT_35C | DF_NON_NULL_RET | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700142
Elliott Hughesadb8c672012-03-06 16:49:32 -0800143 // 25 FILLED_NEW_ARRAY_RANGE {vCCCC .. vNNNN}, type@BBBB
buzbee9c044ce2012-03-18 13:24:07 -0700144 DF_FORMAT_3RC | DF_NON_NULL_RET | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700145
Elliott Hughesadb8c672012-03-06 16:49:32 -0800146 // 26 FILL_ARRAY_DATA vAA, +BBBBBBBB
buzbee9c044ce2012-03-18 13:24:07 -0700147 DF_UA | DF_CORE_A | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700148
Elliott Hughesadb8c672012-03-06 16:49:32 -0800149 // 27 THROW vAA
buzbee9c044ce2012-03-18 13:24:07 -0700150 DF_UA | DF_CORE_A | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700151
Elliott Hughesadb8c672012-03-06 16:49:32 -0800152 // 28 GOTO
buzbee67bf8852011-08-17 17:51:35 -0700153 DF_NOP,
154
Elliott Hughesadb8c672012-03-06 16:49:32 -0800155 // 29 GOTO_16
buzbee67bf8852011-08-17 17:51:35 -0700156 DF_NOP,
157
Elliott Hughesadb8c672012-03-06 16:49:32 -0800158 // 2A GOTO_32
buzbee67bf8852011-08-17 17:51:35 -0700159 DF_NOP,
160
Elliott Hughesadb8c672012-03-06 16:49:32 -0800161 // 2B PACKED_SWITCH vAA, +BBBBBBBB
buzbee67bf8852011-08-17 17:51:35 -0700162 DF_UA,
163
Elliott Hughesadb8c672012-03-06 16:49:32 -0800164 // 2C SPARSE_SWITCH vAA, +BBBBBBBB
buzbee67bf8852011-08-17 17:51:35 -0700165 DF_UA,
166
Elliott Hughesadb8c672012-03-06 16:49:32 -0800167 // 2D CMPL_FLOAT vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700168 DF_DA | DF_UB | DF_UC | DF_FP_B | DF_FP_C | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -0700169
Elliott Hughesadb8c672012-03-06 16:49:32 -0800170 // 2E CMPG_FLOAT vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700171 DF_DA | DF_UB | DF_UC | DF_FP_B | DF_FP_C | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -0700172
Elliott Hughesadb8c672012-03-06 16:49:32 -0800173 // 2F CMPL_DOUBLE vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700174 DF_DA | DF_UB_WIDE | DF_UC_WIDE | DF_FP_B | DF_FP_C | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -0700175
Elliott Hughesadb8c672012-03-06 16:49:32 -0800176 // 30 CMPG_DOUBLE vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700177 DF_DA | DF_UB_WIDE | DF_UC_WIDE | DF_FP_B | DF_FP_C | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -0700178
Elliott Hughesadb8c672012-03-06 16:49:32 -0800179 // 31 CMP_LONG vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700180 DF_DA | DF_UB_WIDE | DF_UC_WIDE | DF_CORE_A | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700181
Elliott Hughesadb8c672012-03-06 16:49:32 -0800182 // 32 IF_EQ vA, vB, +CCCC
buzbee67bc2362011-10-11 18:08:40 -0700183 DF_UA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700184
Elliott Hughesadb8c672012-03-06 16:49:32 -0800185 // 33 IF_NE vA, vB, +CCCC
buzbee67bc2362011-10-11 18:08:40 -0700186 DF_UA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700187
Elliott Hughesadb8c672012-03-06 16:49:32 -0800188 // 34 IF_LT vA, vB, +CCCC
buzbee67bc2362011-10-11 18:08:40 -0700189 DF_UA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700190
Elliott Hughesadb8c672012-03-06 16:49:32 -0800191 // 35 IF_GE vA, vB, +CCCC
buzbee67bc2362011-10-11 18:08:40 -0700192 DF_UA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700193
Elliott Hughesadb8c672012-03-06 16:49:32 -0800194 // 36 IF_GT vA, vB, +CCCC
buzbee67bc2362011-10-11 18:08:40 -0700195 DF_UA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700196
Elliott Hughesadb8c672012-03-06 16:49:32 -0800197 // 37 IF_LE vA, vB, +CCCC
buzbee67bc2362011-10-11 18:08:40 -0700198 DF_UA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700199
200
Elliott Hughesadb8c672012-03-06 16:49:32 -0800201 // 38 IF_EQZ vAA, +BBBB
buzbee67bc2362011-10-11 18:08:40 -0700202 DF_UA | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -0700203
Elliott Hughesadb8c672012-03-06 16:49:32 -0800204 // 39 IF_NEZ vAA, +BBBB
buzbee67bc2362011-10-11 18:08:40 -0700205 DF_UA | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -0700206
Elliott Hughesadb8c672012-03-06 16:49:32 -0800207 // 3A IF_LTZ vAA, +BBBB
buzbee67bc2362011-10-11 18:08:40 -0700208 DF_UA | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -0700209
Elliott Hughesadb8c672012-03-06 16:49:32 -0800210 // 3B IF_GEZ vAA, +BBBB
buzbee67bc2362011-10-11 18:08:40 -0700211 DF_UA | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -0700212
Elliott Hughesadb8c672012-03-06 16:49:32 -0800213 // 3C IF_GTZ vAA, +BBBB
buzbee67bc2362011-10-11 18:08:40 -0700214 DF_UA | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -0700215
Elliott Hughesadb8c672012-03-06 16:49:32 -0800216 // 3D IF_LEZ vAA, +BBBB
buzbee67bc2362011-10-11 18:08:40 -0700217 DF_UA | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -0700218
Elliott Hughesadb8c672012-03-06 16:49:32 -0800219 // 3E UNUSED_3E
buzbee67bf8852011-08-17 17:51:35 -0700220 DF_NOP,
221
Elliott Hughesadb8c672012-03-06 16:49:32 -0800222 // 3F UNUSED_3F
buzbee67bf8852011-08-17 17:51:35 -0700223 DF_NOP,
224
Elliott Hughesadb8c672012-03-06 16:49:32 -0800225 // 40 UNUSED_40
buzbee67bf8852011-08-17 17:51:35 -0700226 DF_NOP,
227
Elliott Hughesadb8c672012-03-06 16:49:32 -0800228 // 41 UNUSED_41
buzbee67bf8852011-08-17 17:51:35 -0700229 DF_NOP,
230
Elliott Hughesadb8c672012-03-06 16:49:32 -0800231 // 42 UNUSED_42
buzbee67bf8852011-08-17 17:51:35 -0700232 DF_NOP,
233
Elliott Hughesadb8c672012-03-06 16:49:32 -0800234 // 43 UNUSED_43
buzbee67bf8852011-08-17 17:51:35 -0700235 DF_NOP,
236
Elliott Hughesadb8c672012-03-06 16:49:32 -0800237 // 44 AGET vAA, vBB, vCC
buzbee9c044ce2012-03-18 13:24:07 -0700238 DF_DA | DF_UB | DF_UC | DF_NULL_CHK_0 | DF_RANGE_CHK_1 | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700239
Elliott Hughesadb8c672012-03-06 16:49:32 -0800240 // 45 AGET_WIDE vAA, vBB, vCC
buzbee9c044ce2012-03-18 13:24:07 -0700241 DF_DA_WIDE | DF_UB | DF_UC | DF_NULL_CHK_0 | DF_RANGE_CHK_1 | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700242
Elliott Hughesadb8c672012-03-06 16:49:32 -0800243 // 46 AGET_OBJECT vAA, vBB, vCC
buzbee9c044ce2012-03-18 13:24:07 -0700244 DF_DA | DF_UB | DF_UC | DF_NULL_CHK_0 | DF_RANGE_CHK_1 | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700245
Elliott Hughesadb8c672012-03-06 16:49:32 -0800246 // 47 AGET_BOOLEAN vAA, vBB, vCC
buzbee9c044ce2012-03-18 13:24:07 -0700247 DF_DA | DF_UB | DF_UC | DF_NULL_CHK_0 | DF_RANGE_CHK_1 | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700248
Elliott Hughesadb8c672012-03-06 16:49:32 -0800249 // 48 AGET_BYTE vAA, vBB, vCC
buzbee9c044ce2012-03-18 13:24:07 -0700250 DF_DA | DF_UB | DF_UC | DF_NULL_CHK_0 | DF_RANGE_CHK_1 | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700251
Elliott Hughesadb8c672012-03-06 16:49:32 -0800252 // 49 AGET_CHAR vAA, vBB, vCC
buzbee9c044ce2012-03-18 13:24:07 -0700253 DF_DA | DF_UB | DF_UC | DF_NULL_CHK_0 | DF_RANGE_CHK_1 | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700254
Elliott Hughesadb8c672012-03-06 16:49:32 -0800255 // 4A AGET_SHORT vAA, vBB, vCC
buzbee9c044ce2012-03-18 13:24:07 -0700256 DF_DA | DF_UB | DF_UC | DF_NULL_CHK_0 | DF_RANGE_CHK_1 | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700257
Elliott Hughesadb8c672012-03-06 16:49:32 -0800258 // 4B APUT vAA, vBB, vCC
buzbee9c044ce2012-03-18 13:24:07 -0700259 DF_UA | DF_UB | DF_UC | DF_NULL_CHK_1 | DF_RANGE_CHK_2 | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700260
Elliott Hughesadb8c672012-03-06 16:49:32 -0800261 // 4C APUT_WIDE vAA, vBB, vCC
buzbee9c044ce2012-03-18 13:24:07 -0700262 DF_UA_WIDE | DF_UB | DF_UC | DF_NULL_CHK_2 | DF_RANGE_CHK_3 | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700263
Elliott Hughesadb8c672012-03-06 16:49:32 -0800264 // 4D APUT_OBJECT vAA, vBB, vCC
buzbee9c044ce2012-03-18 13:24:07 -0700265 DF_UA | DF_UB | DF_UC | DF_NULL_CHK_1 | DF_RANGE_CHK_2 | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700266
Elliott Hughesadb8c672012-03-06 16:49:32 -0800267 // 4E APUT_BOOLEAN vAA, vBB, vCC
buzbee9c044ce2012-03-18 13:24:07 -0700268 DF_UA | DF_UB | DF_UC | DF_NULL_CHK_1 | DF_RANGE_CHK_2 | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700269
Elliott Hughesadb8c672012-03-06 16:49:32 -0800270 // 4F APUT_BYTE vAA, vBB, vCC
buzbee9c044ce2012-03-18 13:24:07 -0700271 DF_UA | DF_UB | DF_UC | DF_NULL_CHK_1 | DF_RANGE_CHK_2 | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700272
Elliott Hughesadb8c672012-03-06 16:49:32 -0800273 // 50 APUT_CHAR vAA, vBB, vCC
buzbee9c044ce2012-03-18 13:24:07 -0700274 DF_UA | DF_UB | DF_UC | DF_NULL_CHK_1 | DF_RANGE_CHK_2 | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700275
Elliott Hughesadb8c672012-03-06 16:49:32 -0800276 // 51 APUT_SHORT vAA, vBB, vCC
buzbee9c044ce2012-03-18 13:24:07 -0700277 DF_UA | DF_UB | DF_UC | DF_NULL_CHK_1 | DF_RANGE_CHK_2 | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700278
Elliott Hughesadb8c672012-03-06 16:49:32 -0800279 // 52 IGET vA, vB, field@CCCC
buzbee9c044ce2012-03-18 13:24:07 -0700280 DF_DA | DF_UB | DF_NULL_CHK_0 | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700281
Elliott Hughesadb8c672012-03-06 16:49:32 -0800282 // 53 IGET_WIDE vA, vB, field@CCCC
buzbee9c044ce2012-03-18 13:24:07 -0700283 DF_DA_WIDE | DF_UB | DF_NULL_CHK_0 | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700284
Elliott Hughesadb8c672012-03-06 16:49:32 -0800285 // 54 IGET_OBJECT vA, vB, field@CCCC
buzbee9c044ce2012-03-18 13:24:07 -0700286 DF_DA | DF_UB | DF_NULL_CHK_0 | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700287
Elliott Hughesadb8c672012-03-06 16:49:32 -0800288 // 55 IGET_BOOLEAN vA, vB, field@CCCC
buzbee9c044ce2012-03-18 13:24:07 -0700289 DF_DA | DF_UB | DF_NULL_CHK_0 | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700290
Elliott Hughesadb8c672012-03-06 16:49:32 -0800291 // 56 IGET_BYTE vA, vB, field@CCCC
buzbee9c044ce2012-03-18 13:24:07 -0700292 DF_DA | DF_UB | DF_NULL_CHK_0 | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700293
Elliott Hughesadb8c672012-03-06 16:49:32 -0800294 // 57 IGET_CHAR vA, vB, field@CCCC
buzbee9c044ce2012-03-18 13:24:07 -0700295 DF_DA | DF_UB | DF_NULL_CHK_0 | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700296
Elliott Hughesadb8c672012-03-06 16:49:32 -0800297 // 58 IGET_SHORT vA, vB, field@CCCC
buzbee9c044ce2012-03-18 13:24:07 -0700298 DF_DA | DF_UB | DF_NULL_CHK_0 | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700299
Elliott Hughesadb8c672012-03-06 16:49:32 -0800300 // 59 IPUT vA, vB, field@CCCC
buzbee9c044ce2012-03-18 13:24:07 -0700301 DF_UA | DF_UB | DF_NULL_CHK_1 | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700302
Elliott Hughesadb8c672012-03-06 16:49:32 -0800303 // 5A IPUT_WIDE vA, vB, field@CCCC
buzbee9c044ce2012-03-18 13:24:07 -0700304 DF_UA_WIDE | DF_UB | DF_NULL_CHK_2 | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700305
Elliott Hughesadb8c672012-03-06 16:49:32 -0800306 // 5B IPUT_OBJECT vA, vB, field@CCCC
buzbee9c044ce2012-03-18 13:24:07 -0700307 DF_UA | DF_UB | DF_NULL_CHK_1 | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700308
Elliott Hughesadb8c672012-03-06 16:49:32 -0800309 // 5C IPUT_BOOLEAN vA, vB, field@CCCC
buzbee9c044ce2012-03-18 13:24:07 -0700310 DF_UA | DF_UB | DF_NULL_CHK_1 | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700311
Elliott Hughesadb8c672012-03-06 16:49:32 -0800312 // 5D IPUT_BYTE vA, vB, field@CCCC
buzbee9c044ce2012-03-18 13:24:07 -0700313 DF_UA | DF_UB | DF_NULL_CHK_1 | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700314
Elliott Hughesadb8c672012-03-06 16:49:32 -0800315 // 5E IPUT_CHAR vA, vB, field@CCCC
buzbee9c044ce2012-03-18 13:24:07 -0700316 DF_UA | DF_UB | DF_NULL_CHK_1 | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700317
Elliott Hughesadb8c672012-03-06 16:49:32 -0800318 // 5F IPUT_SHORT vA, vB, field@CCCC
buzbee9c044ce2012-03-18 13:24:07 -0700319 DF_UA | DF_UB | DF_NULL_CHK_1 | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700320
Elliott Hughesadb8c672012-03-06 16:49:32 -0800321 // 60 SGET vAA, field@BBBB
buzbee9c044ce2012-03-18 13:24:07 -0700322 DF_DA | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700323
Elliott Hughesadb8c672012-03-06 16:49:32 -0800324 // 61 SGET_WIDE vAA, field@BBBB
buzbee9c044ce2012-03-18 13:24:07 -0700325 DF_DA_WIDE | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700326
Elliott Hughesadb8c672012-03-06 16:49:32 -0800327 // 62 SGET_OBJECT vAA, field@BBBB
buzbee9c044ce2012-03-18 13:24:07 -0700328 DF_DA | DF_CORE_A | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700329
Elliott Hughesadb8c672012-03-06 16:49:32 -0800330 // 63 SGET_BOOLEAN vAA, field@BBBB
buzbee9c044ce2012-03-18 13:24:07 -0700331 DF_DA | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700332
Elliott Hughesadb8c672012-03-06 16:49:32 -0800333 // 64 SGET_BYTE vAA, field@BBBB
buzbee9c044ce2012-03-18 13:24:07 -0700334 DF_DA | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700335
Elliott Hughesadb8c672012-03-06 16:49:32 -0800336 // 65 SGET_CHAR vAA, field@BBBB
buzbee9c044ce2012-03-18 13:24:07 -0700337 DF_DA | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700338
Elliott Hughesadb8c672012-03-06 16:49:32 -0800339 // 66 SGET_SHORT vAA, field@BBBB
buzbee9c044ce2012-03-18 13:24:07 -0700340 DF_DA | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700341
Elliott Hughesadb8c672012-03-06 16:49:32 -0800342 // 67 SPUT vAA, field@BBBB
buzbee9c044ce2012-03-18 13:24:07 -0700343 DF_UA | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700344
Elliott Hughesadb8c672012-03-06 16:49:32 -0800345 // 68 SPUT_WIDE vAA, field@BBBB
buzbee9c044ce2012-03-18 13:24:07 -0700346 DF_UA_WIDE | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700347
Elliott Hughesadb8c672012-03-06 16:49:32 -0800348 // 69 SPUT_OBJECT vAA, field@BBBB
buzbee9c044ce2012-03-18 13:24:07 -0700349 DF_UA | DF_CORE_A | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700350
Elliott Hughesadb8c672012-03-06 16:49:32 -0800351 // 6A SPUT_BOOLEAN vAA, field@BBBB
buzbee9c044ce2012-03-18 13:24:07 -0700352 DF_UA | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700353
Elliott Hughesadb8c672012-03-06 16:49:32 -0800354 // 6B SPUT_BYTE vAA, field@BBBB
buzbee9c044ce2012-03-18 13:24:07 -0700355 DF_UA | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700356
Elliott Hughesadb8c672012-03-06 16:49:32 -0800357 // 6C SPUT_CHAR vAA, field@BBBB
buzbee9c044ce2012-03-18 13:24:07 -0700358 DF_UA | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700359
Elliott Hughesadb8c672012-03-06 16:49:32 -0800360 // 6D SPUT_SHORT vAA, field@BBBB
buzbee9c044ce2012-03-18 13:24:07 -0700361 DF_UA | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700362
Elliott Hughesadb8c672012-03-06 16:49:32 -0800363 // 6E INVOKE_VIRTUAL {vD, vE, vF, vG, vA}
buzbee9c044ce2012-03-18 13:24:07 -0700364 DF_FORMAT_35C | DF_NULL_CHK_OUT0 | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700365
Elliott Hughesadb8c672012-03-06 16:49:32 -0800366 // 6F INVOKE_SUPER {vD, vE, vF, vG, vA}
buzbee9c044ce2012-03-18 13:24:07 -0700367 DF_FORMAT_35C | DF_NULL_CHK_OUT0 | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700368
Elliott Hughesadb8c672012-03-06 16:49:32 -0800369 // 70 INVOKE_DIRECT {vD, vE, vF, vG, vA}
buzbee9c044ce2012-03-18 13:24:07 -0700370 DF_FORMAT_35C | DF_NULL_CHK_OUT0 | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700371
Elliott Hughesadb8c672012-03-06 16:49:32 -0800372 // 71 INVOKE_STATIC {vD, vE, vF, vG, vA}
buzbee9c044ce2012-03-18 13:24:07 -0700373 DF_FORMAT_35C | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700374
Elliott Hughesadb8c672012-03-06 16:49:32 -0800375 // 72 INVOKE_INTERFACE {vD, vE, vF, vG, vA}
buzbee9c044ce2012-03-18 13:24:07 -0700376 DF_FORMAT_35C | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700377
Elliott Hughesadb8c672012-03-06 16:49:32 -0800378 // 73 UNUSED_73
buzbee67bf8852011-08-17 17:51:35 -0700379 DF_NOP,
380
Elliott Hughesadb8c672012-03-06 16:49:32 -0800381 // 74 INVOKE_VIRTUAL_RANGE {vCCCC .. vNNNN}
buzbee9c044ce2012-03-18 13:24:07 -0700382 DF_FORMAT_3RC | DF_NULL_CHK_OUT0 | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700383
Elliott Hughesadb8c672012-03-06 16:49:32 -0800384 // 75 INVOKE_SUPER_RANGE {vCCCC .. vNNNN}
buzbee9c044ce2012-03-18 13:24:07 -0700385 DF_FORMAT_3RC | DF_NULL_CHK_OUT0 | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700386
Elliott Hughesadb8c672012-03-06 16:49:32 -0800387 // 76 INVOKE_DIRECT_RANGE {vCCCC .. vNNNN}
buzbee9c044ce2012-03-18 13:24:07 -0700388 DF_FORMAT_3RC | DF_NULL_CHK_OUT0 | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700389
Elliott Hughesadb8c672012-03-06 16:49:32 -0800390 // 77 INVOKE_STATIC_RANGE {vCCCC .. vNNNN}
buzbee9c044ce2012-03-18 13:24:07 -0700391 DF_FORMAT_3RC | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700392
Elliott Hughesadb8c672012-03-06 16:49:32 -0800393 // 78 INVOKE_INTERFACE_RANGE {vCCCC .. vNNNN}
buzbee9c044ce2012-03-18 13:24:07 -0700394 DF_FORMAT_3RC | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700395
Elliott Hughesadb8c672012-03-06 16:49:32 -0800396 // 79 UNUSED_79
buzbee67bf8852011-08-17 17:51:35 -0700397 DF_NOP,
398
Elliott Hughesadb8c672012-03-06 16:49:32 -0800399 // 7A UNUSED_7A
buzbee67bf8852011-08-17 17:51:35 -0700400 DF_NOP,
401
Elliott Hughesadb8c672012-03-06 16:49:32 -0800402 // 7B NEG_INT vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700403 DF_DA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700404
Elliott Hughesadb8c672012-03-06 16:49:32 -0800405 // 7C NOT_INT vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700406 DF_DA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700407
Elliott Hughesadb8c672012-03-06 16:49:32 -0800408 // 7D NEG_LONG vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700409 DF_DA_WIDE | DF_UB_WIDE | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700410
Elliott Hughesadb8c672012-03-06 16:49:32 -0800411 // 7E NOT_LONG vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700412 DF_DA_WIDE | DF_UB_WIDE | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700413
Elliott Hughesadb8c672012-03-06 16:49:32 -0800414 // 7F NEG_FLOAT vA, vB
buzbee67bf8852011-08-17 17:51:35 -0700415 DF_DA | DF_UB | DF_FP_A | DF_FP_B,
416
Elliott Hughesadb8c672012-03-06 16:49:32 -0800417 // 80 NEG_DOUBLE vA, vB
buzbee67bf8852011-08-17 17:51:35 -0700418 DF_DA_WIDE | DF_UB_WIDE | DF_FP_A | DF_FP_B,
419
Elliott Hughesadb8c672012-03-06 16:49:32 -0800420 // 81 INT_TO_LONG vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700421 DF_DA_WIDE | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700422
Elliott Hughesadb8c672012-03-06 16:49:32 -0800423 // 82 INT_TO_FLOAT vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700424 DF_DA | DF_UB | DF_FP_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700425
Elliott Hughesadb8c672012-03-06 16:49:32 -0800426 // 83 INT_TO_DOUBLE vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700427 DF_DA_WIDE | DF_UB | DF_FP_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700428
Elliott Hughesadb8c672012-03-06 16:49:32 -0800429 // 84 LONG_TO_INT vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700430 DF_DA | DF_UB_WIDE | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700431
Elliott Hughesadb8c672012-03-06 16:49:32 -0800432 // 85 LONG_TO_FLOAT vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700433 DF_DA | DF_UB_WIDE | DF_FP_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700434
Elliott Hughesadb8c672012-03-06 16:49:32 -0800435 // 86 LONG_TO_DOUBLE vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700436 DF_DA_WIDE | DF_UB_WIDE | DF_FP_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700437
Elliott Hughesadb8c672012-03-06 16:49:32 -0800438 // 87 FLOAT_TO_INT vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700439 DF_DA | DF_UB | DF_FP_B | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -0700440
Elliott Hughesadb8c672012-03-06 16:49:32 -0800441 // 88 FLOAT_TO_LONG vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700442 DF_DA_WIDE | DF_UB | DF_FP_B | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -0700443
Elliott Hughesadb8c672012-03-06 16:49:32 -0800444 // 89 FLOAT_TO_DOUBLE vA, vB
buzbee67bf8852011-08-17 17:51:35 -0700445 DF_DA_WIDE | DF_UB | DF_FP_A | DF_FP_B,
446
Elliott Hughesadb8c672012-03-06 16:49:32 -0800447 // 8A DOUBLE_TO_INT vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700448 DF_DA | DF_UB_WIDE | DF_FP_B | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -0700449
Elliott Hughesadb8c672012-03-06 16:49:32 -0800450 // 8B DOUBLE_TO_LONG vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700451 DF_DA_WIDE | DF_UB_WIDE | DF_FP_B | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -0700452
Elliott Hughesadb8c672012-03-06 16:49:32 -0800453 // 8C DOUBLE_TO_FLOAT vA, vB
buzbee67bf8852011-08-17 17:51:35 -0700454 DF_DA | DF_UB_WIDE | DF_FP_A | DF_FP_B,
455
Elliott Hughesadb8c672012-03-06 16:49:32 -0800456 // 8D INT_TO_BYTE vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700457 DF_DA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700458
Elliott Hughesadb8c672012-03-06 16:49:32 -0800459 // 8E INT_TO_CHAR vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700460 DF_DA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700461
Elliott Hughesadb8c672012-03-06 16:49:32 -0800462 // 8F INT_TO_SHORT vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700463 DF_DA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700464
Elliott Hughesadb8c672012-03-06 16:49:32 -0800465 // 90 ADD_INT vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700466 DF_DA | DF_UB | DF_UC | DF_IS_LINEAR | DF_CORE_A | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700467
Elliott Hughesadb8c672012-03-06 16:49:32 -0800468 // 91 SUB_INT vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700469 DF_DA | DF_UB | DF_UC | DF_IS_LINEAR | DF_CORE_A | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700470
Elliott Hughesadb8c672012-03-06 16:49:32 -0800471 // 92 MUL_INT vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700472 DF_DA | DF_UB | DF_UC | DF_CORE_A | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700473
Elliott Hughesadb8c672012-03-06 16:49:32 -0800474 // 93 DIV_INT vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700475 DF_DA | DF_UB | DF_UC | DF_CORE_A | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700476
Elliott Hughesadb8c672012-03-06 16:49:32 -0800477 // 94 REM_INT vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700478 DF_DA | DF_UB | DF_UC | DF_CORE_A | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700479
Elliott Hughesadb8c672012-03-06 16:49:32 -0800480 // 95 AND_INT vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700481 DF_DA | DF_UB | DF_UC | DF_CORE_A | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700482
Elliott Hughesadb8c672012-03-06 16:49:32 -0800483 // 96 OR_INT vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700484 DF_DA | DF_UB | DF_UC | DF_CORE_A | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700485
Elliott Hughesadb8c672012-03-06 16:49:32 -0800486 // 97 XOR_INT vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700487 DF_DA | DF_UB | DF_UC | DF_CORE_A | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700488
Elliott Hughesadb8c672012-03-06 16:49:32 -0800489 // 98 SHL_INT vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700490 DF_DA | DF_UB | DF_UC | DF_CORE_A | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700491
Elliott Hughesadb8c672012-03-06 16:49:32 -0800492 // 99 SHR_INT vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700493 DF_DA | DF_UB | DF_UC | DF_CORE_A | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700494
Elliott Hughesadb8c672012-03-06 16:49:32 -0800495 // 9A USHR_INT vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700496 DF_DA | DF_UB | DF_UC | DF_CORE_A | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700497
Elliott Hughesadb8c672012-03-06 16:49:32 -0800498 // 9B ADD_LONG vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700499 DF_DA_WIDE | DF_UB_WIDE | DF_UC_WIDE | DF_CORE_A | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700500
Elliott Hughesadb8c672012-03-06 16:49:32 -0800501 // 9C SUB_LONG vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700502 DF_DA_WIDE | DF_UB_WIDE | DF_UC_WIDE | DF_CORE_A | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700503
Elliott Hughesadb8c672012-03-06 16:49:32 -0800504 // 9D MUL_LONG vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700505 DF_DA_WIDE | DF_UB_WIDE | DF_UC_WIDE | DF_CORE_A | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700506
Elliott Hughesadb8c672012-03-06 16:49:32 -0800507 // 9E DIV_LONG vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700508 DF_DA_WIDE | DF_UB_WIDE | DF_UC_WIDE | DF_CORE_A | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700509
Elliott Hughesadb8c672012-03-06 16:49:32 -0800510 // 9F REM_LONG vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700511 DF_DA_WIDE | DF_UB_WIDE | DF_UC_WIDE | DF_CORE_A | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700512
Elliott Hughesadb8c672012-03-06 16:49:32 -0800513 // A0 AND_LONG vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700514 DF_DA_WIDE | DF_UB_WIDE | DF_UC_WIDE | DF_CORE_A | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700515
Elliott Hughesadb8c672012-03-06 16:49:32 -0800516 // A1 OR_LONG vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700517 DF_DA_WIDE | DF_UB_WIDE | DF_UC_WIDE | DF_CORE_A | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700518
Elliott Hughesadb8c672012-03-06 16:49:32 -0800519 // A2 XOR_LONG vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700520 DF_DA_WIDE | DF_UB_WIDE | DF_UC_WIDE | DF_CORE_A | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700521
Elliott Hughesadb8c672012-03-06 16:49:32 -0800522 // A3 SHL_LONG vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700523 DF_DA_WIDE | DF_UB_WIDE | DF_UC | DF_CORE_A | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700524
Elliott Hughesadb8c672012-03-06 16:49:32 -0800525 // A4 SHR_LONG vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700526 DF_DA_WIDE | DF_UB_WIDE | DF_UC | DF_CORE_A | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700527
Elliott Hughesadb8c672012-03-06 16:49:32 -0800528 // A5 USHR_LONG vAA, vBB, vCC
buzbee67bc2362011-10-11 18:08:40 -0700529 DF_DA_WIDE | DF_UB_WIDE | DF_UC | DF_CORE_A | DF_CORE_B | DF_CORE_C,
buzbee67bf8852011-08-17 17:51:35 -0700530
Elliott Hughesadb8c672012-03-06 16:49:32 -0800531 // A6 ADD_FLOAT vAA, vBB, vCC
buzbee67bf8852011-08-17 17:51:35 -0700532 DF_DA | DF_UB | DF_UC | DF_FP_A | DF_FP_B | DF_FP_C,
533
Elliott Hughesadb8c672012-03-06 16:49:32 -0800534 // A7 SUB_FLOAT vAA, vBB, vCC
buzbee67bf8852011-08-17 17:51:35 -0700535 DF_DA | DF_UB | DF_UC | DF_FP_A | DF_FP_B | DF_FP_C,
536
Elliott Hughesadb8c672012-03-06 16:49:32 -0800537 // A8 MUL_FLOAT vAA, vBB, vCC
buzbee67bf8852011-08-17 17:51:35 -0700538 DF_DA | DF_UB | DF_UC | DF_FP_A | DF_FP_B | DF_FP_C,
539
Elliott Hughesadb8c672012-03-06 16:49:32 -0800540 // A9 DIV_FLOAT vAA, vBB, vCC
buzbee67bf8852011-08-17 17:51:35 -0700541 DF_DA | DF_UB | DF_UC | DF_FP_A | DF_FP_B | DF_FP_C,
542
Elliott Hughesadb8c672012-03-06 16:49:32 -0800543 // AA REM_FLOAT vAA, vBB, vCC
buzbee67bf8852011-08-17 17:51:35 -0700544 DF_DA | DF_UB | DF_UC | DF_FP_A | DF_FP_B | DF_FP_C,
545
Elliott Hughesadb8c672012-03-06 16:49:32 -0800546 // AB ADD_DOUBLE vAA, vBB, vCC
buzbee67bf8852011-08-17 17:51:35 -0700547 DF_DA_WIDE | DF_UB_WIDE | DF_UC_WIDE | DF_FP_A | DF_FP_B | DF_FP_C,
548
Elliott Hughesadb8c672012-03-06 16:49:32 -0800549 // AC SUB_DOUBLE vAA, vBB, vCC
buzbee67bf8852011-08-17 17:51:35 -0700550 DF_DA_WIDE | DF_UB_WIDE | DF_UC_WIDE | DF_FP_A | DF_FP_B | DF_FP_C,
551
Elliott Hughesadb8c672012-03-06 16:49:32 -0800552 // AD MUL_DOUBLE vAA, vBB, vCC
buzbee67bf8852011-08-17 17:51:35 -0700553 DF_DA_WIDE | DF_UB_WIDE | DF_UC_WIDE | DF_FP_A | DF_FP_B | DF_FP_C,
554
Elliott Hughesadb8c672012-03-06 16:49:32 -0800555 // AE DIV_DOUBLE vAA, vBB, vCC
buzbee67bf8852011-08-17 17:51:35 -0700556 DF_DA_WIDE | DF_UB_WIDE | DF_UC_WIDE | DF_FP_A | DF_FP_B | DF_FP_C,
557
Elliott Hughesadb8c672012-03-06 16:49:32 -0800558 // AF REM_DOUBLE vAA, vBB, vCC
buzbee67bf8852011-08-17 17:51:35 -0700559 DF_DA_WIDE | DF_UB_WIDE | DF_UC_WIDE | DF_FP_A | DF_FP_B | DF_FP_C,
560
Elliott Hughesadb8c672012-03-06 16:49:32 -0800561 // B0 ADD_INT_2ADDR vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700562 DF_DA | DF_UA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700563
Elliott Hughesadb8c672012-03-06 16:49:32 -0800564 // B1 SUB_INT_2ADDR vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700565 DF_DA | DF_UA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700566
Elliott Hughesadb8c672012-03-06 16:49:32 -0800567 // B2 MUL_INT_2ADDR vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700568 DF_DA | DF_UA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700569
Elliott Hughesadb8c672012-03-06 16:49:32 -0800570 // B3 DIV_INT_2ADDR vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700571 DF_DA | DF_UA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700572
Elliott Hughesadb8c672012-03-06 16:49:32 -0800573 // B4 REM_INT_2ADDR vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700574 DF_DA | DF_UA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700575
Elliott Hughesadb8c672012-03-06 16:49:32 -0800576 // B5 AND_INT_2ADDR vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700577 DF_DA | DF_UA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700578
Elliott Hughesadb8c672012-03-06 16:49:32 -0800579 // B6 OR_INT_2ADDR vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700580 DF_DA | DF_UA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700581
Elliott Hughesadb8c672012-03-06 16:49:32 -0800582 // B7 XOR_INT_2ADDR vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700583 DF_DA | DF_UA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700584
Elliott Hughesadb8c672012-03-06 16:49:32 -0800585 // B8 SHL_INT_2ADDR vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700586 DF_DA | DF_UA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700587
Elliott Hughesadb8c672012-03-06 16:49:32 -0800588 // B9 SHR_INT_2ADDR vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700589 DF_DA | DF_UA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700590
Elliott Hughesadb8c672012-03-06 16:49:32 -0800591 // BA USHR_INT_2ADDR vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700592 DF_DA | DF_UA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700593
Elliott Hughesadb8c672012-03-06 16:49:32 -0800594 // BB ADD_LONG_2ADDR vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700595 DF_DA_WIDE | DF_UA_WIDE | DF_UB_WIDE | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700596
Elliott Hughesadb8c672012-03-06 16:49:32 -0800597 // BC SUB_LONG_2ADDR vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700598 DF_DA_WIDE | DF_UA_WIDE | DF_UB_WIDE | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700599
Elliott Hughesadb8c672012-03-06 16:49:32 -0800600 // BD MUL_LONG_2ADDR vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700601 DF_DA_WIDE | DF_UA_WIDE | DF_UB_WIDE | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700602
Elliott Hughesadb8c672012-03-06 16:49:32 -0800603 // BE DIV_LONG_2ADDR vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700604 DF_DA_WIDE | DF_UA_WIDE | DF_UB_WIDE | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700605
Elliott Hughesadb8c672012-03-06 16:49:32 -0800606 // BF REM_LONG_2ADDR vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700607 DF_DA_WIDE | DF_UA_WIDE | DF_UB_WIDE | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700608
Elliott Hughesadb8c672012-03-06 16:49:32 -0800609 // C0 AND_LONG_2ADDR vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700610 DF_DA_WIDE | DF_UA_WIDE | DF_UB_WIDE | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700611
Elliott Hughesadb8c672012-03-06 16:49:32 -0800612 // C1 OR_LONG_2ADDR vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700613 DF_DA_WIDE | DF_UA_WIDE | DF_UB_WIDE | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700614
Elliott Hughesadb8c672012-03-06 16:49:32 -0800615 // C2 XOR_LONG_2ADDR vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700616 DF_DA_WIDE | DF_UA_WIDE | DF_UB_WIDE | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700617
Elliott Hughesadb8c672012-03-06 16:49:32 -0800618 // C3 SHL_LONG_2ADDR vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700619 DF_DA_WIDE | DF_UA_WIDE | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700620
Elliott Hughesadb8c672012-03-06 16:49:32 -0800621 // C4 SHR_LONG_2ADDR vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700622 DF_DA_WIDE | DF_UA_WIDE | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700623
Elliott Hughesadb8c672012-03-06 16:49:32 -0800624 // C5 USHR_LONG_2ADDR vA, vB
buzbee67bc2362011-10-11 18:08:40 -0700625 DF_DA_WIDE | DF_UA_WIDE | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700626
Elliott Hughesadb8c672012-03-06 16:49:32 -0800627 // C6 ADD_FLOAT_2ADDR vA, vB
buzbee67bf8852011-08-17 17:51:35 -0700628 DF_DA | DF_UA | DF_UB | DF_FP_A | DF_FP_B,
629
Elliott Hughesadb8c672012-03-06 16:49:32 -0800630 // C7 SUB_FLOAT_2ADDR vA, vB
buzbee67bf8852011-08-17 17:51:35 -0700631 DF_DA | DF_UA | DF_UB | DF_FP_A | DF_FP_B,
632
Elliott Hughesadb8c672012-03-06 16:49:32 -0800633 // C8 MUL_FLOAT_2ADDR vA, vB
buzbee67bf8852011-08-17 17:51:35 -0700634 DF_DA | DF_UA | DF_UB | DF_FP_A | DF_FP_B,
635
Elliott Hughesadb8c672012-03-06 16:49:32 -0800636 // C9 DIV_FLOAT_2ADDR vA, vB
buzbee67bf8852011-08-17 17:51:35 -0700637 DF_DA | DF_UA | DF_UB | DF_FP_A | DF_FP_B,
638
Elliott Hughesadb8c672012-03-06 16:49:32 -0800639 // CA REM_FLOAT_2ADDR vA, vB
buzbee67bf8852011-08-17 17:51:35 -0700640 DF_DA | DF_UA | DF_UB | DF_FP_A | DF_FP_B,
641
Elliott Hughesadb8c672012-03-06 16:49:32 -0800642 // CB ADD_DOUBLE_2ADDR vA, vB
buzbee67bf8852011-08-17 17:51:35 -0700643 DF_DA_WIDE | DF_UA_WIDE | DF_UB_WIDE | DF_FP_A | DF_FP_B,
644
Elliott Hughesadb8c672012-03-06 16:49:32 -0800645 // CC SUB_DOUBLE_2ADDR vA, vB
buzbee67bf8852011-08-17 17:51:35 -0700646 DF_DA_WIDE | DF_UA_WIDE | DF_UB_WIDE | DF_FP_A | DF_FP_B,
647
Elliott Hughesadb8c672012-03-06 16:49:32 -0800648 // CD MUL_DOUBLE_2ADDR vA, vB
buzbee67bf8852011-08-17 17:51:35 -0700649 DF_DA_WIDE | DF_UA_WIDE | DF_UB_WIDE | DF_FP_A | DF_FP_B,
650
Elliott Hughesadb8c672012-03-06 16:49:32 -0800651 // CE DIV_DOUBLE_2ADDR vA, vB
buzbee67bf8852011-08-17 17:51:35 -0700652 DF_DA_WIDE | DF_UA_WIDE | DF_UB_WIDE | DF_FP_A | DF_FP_B,
653
Elliott Hughesadb8c672012-03-06 16:49:32 -0800654 // CF REM_DOUBLE_2ADDR vA, vB
buzbee67bf8852011-08-17 17:51:35 -0700655 DF_DA_WIDE | DF_UA_WIDE | DF_UB_WIDE | DF_FP_A | DF_FP_B,
656
Elliott Hughesadb8c672012-03-06 16:49:32 -0800657 // D0 ADD_INT_LIT16 vA, vB, #+CCCC
buzbee67bc2362011-10-11 18:08:40 -0700658 DF_DA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700659
Elliott Hughesadb8c672012-03-06 16:49:32 -0800660 // D1 RSUB_INT vA, vB, #+CCCC
buzbee67bc2362011-10-11 18:08:40 -0700661 DF_DA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700662
Elliott Hughesadb8c672012-03-06 16:49:32 -0800663 // D2 MUL_INT_LIT16 vA, vB, #+CCCC
buzbee67bc2362011-10-11 18:08:40 -0700664 DF_DA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700665
Elliott Hughesadb8c672012-03-06 16:49:32 -0800666 // D3 DIV_INT_LIT16 vA, vB, #+CCCC
buzbee67bc2362011-10-11 18:08:40 -0700667 DF_DA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700668
Elliott Hughesadb8c672012-03-06 16:49:32 -0800669 // D4 REM_INT_LIT16 vA, vB, #+CCCC
buzbee67bc2362011-10-11 18:08:40 -0700670 DF_DA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700671
Elliott Hughesadb8c672012-03-06 16:49:32 -0800672 // D5 AND_INT_LIT16 vA, vB, #+CCCC
buzbee67bc2362011-10-11 18:08:40 -0700673 DF_DA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700674
Elliott Hughesadb8c672012-03-06 16:49:32 -0800675 // D6 OR_INT_LIT16 vA, vB, #+CCCC
buzbee67bc2362011-10-11 18:08:40 -0700676 DF_DA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700677
Elliott Hughesadb8c672012-03-06 16:49:32 -0800678 // D7 XOR_INT_LIT16 vA, vB, #+CCCC
buzbee67bc2362011-10-11 18:08:40 -0700679 DF_DA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700680
Elliott Hughesadb8c672012-03-06 16:49:32 -0800681 // D8 ADD_INT_LIT8 vAA, vBB, #+CC
buzbee67bc2362011-10-11 18:08:40 -0700682 DF_DA | DF_UB | DF_IS_LINEAR | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700683
Elliott Hughesadb8c672012-03-06 16:49:32 -0800684 // D9 RSUB_INT_LIT8 vAA, vBB, #+CC
buzbee67bc2362011-10-11 18:08:40 -0700685 DF_DA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700686
Elliott Hughesadb8c672012-03-06 16:49:32 -0800687 // DA MUL_INT_LIT8 vAA, vBB, #+CC
buzbee67bc2362011-10-11 18:08:40 -0700688 DF_DA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700689
Elliott Hughesadb8c672012-03-06 16:49:32 -0800690 // DB DIV_INT_LIT8 vAA, vBB, #+CC
buzbee67bc2362011-10-11 18:08:40 -0700691 DF_DA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700692
Elliott Hughesadb8c672012-03-06 16:49:32 -0800693 // DC REM_INT_LIT8 vAA, vBB, #+CC
buzbee67bc2362011-10-11 18:08:40 -0700694 DF_DA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700695
Elliott Hughesadb8c672012-03-06 16:49:32 -0800696 // DD AND_INT_LIT8 vAA, vBB, #+CC
buzbee67bc2362011-10-11 18:08:40 -0700697 DF_DA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700698
Elliott Hughesadb8c672012-03-06 16:49:32 -0800699 // DE OR_INT_LIT8 vAA, vBB, #+CC
buzbee67bc2362011-10-11 18:08:40 -0700700 DF_DA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700701
Elliott Hughesadb8c672012-03-06 16:49:32 -0800702 // DF XOR_INT_LIT8 vAA, vBB, #+CC
buzbee67bc2362011-10-11 18:08:40 -0700703 DF_DA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700704
Elliott Hughesadb8c672012-03-06 16:49:32 -0800705 // E0 SHL_INT_LIT8 vAA, vBB, #+CC
buzbee67bc2362011-10-11 18:08:40 -0700706 DF_DA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700707
Elliott Hughesadb8c672012-03-06 16:49:32 -0800708 // E1 SHR_INT_LIT8 vAA, vBB, #+CC
buzbee67bc2362011-10-11 18:08:40 -0700709 DF_DA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700710
Elliott Hughesadb8c672012-03-06 16:49:32 -0800711 // E2 USHR_INT_LIT8 vAA, vBB, #+CC
buzbee67bc2362011-10-11 18:08:40 -0700712 DF_DA | DF_UB | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700713
Elliott Hughesadb8c672012-03-06 16:49:32 -0800714 // E3 IGET_VOLATILE
buzbee67bc2362011-10-11 18:08:40 -0700715 DF_DA | DF_UB | DF_NULL_CHK_0 | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700716
Elliott Hughesadb8c672012-03-06 16:49:32 -0800717 // E4 IPUT_VOLATILE
buzbee67bc2362011-10-11 18:08:40 -0700718 DF_UA | DF_UB | DF_NULL_CHK_1 | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700719
Elliott Hughesadb8c672012-03-06 16:49:32 -0800720 // E5 SGET_VOLATILE
buzbee9c044ce2012-03-18 13:24:07 -0700721 DF_DA | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700722
Elliott Hughesadb8c672012-03-06 16:49:32 -0800723 // E6 SPUT_VOLATILE
buzbee9c044ce2012-03-18 13:24:07 -0700724 DF_UA | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700725
Elliott Hughesadb8c672012-03-06 16:49:32 -0800726 // E7 IGET_OBJECT_VOLATILE
buzbee67bc2362011-10-11 18:08:40 -0700727 DF_DA | DF_UB | DF_NULL_CHK_0 | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700728
Elliott Hughesadb8c672012-03-06 16:49:32 -0800729 // E8 IGET_WIDE_VOLATILE
buzbee67bc2362011-10-11 18:08:40 -0700730 DF_DA_WIDE | DF_UB | DF_NULL_CHK_0 | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700731
Elliott Hughesadb8c672012-03-06 16:49:32 -0800732 // E9 IPUT_WIDE_VOLATILE
buzbee239c4e72012-03-16 08:42:29 -0700733 DF_UA_WIDE | DF_UB | DF_NULL_CHK_2 | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700734
Elliott Hughesadb8c672012-03-06 16:49:32 -0800735 // EA SGET_WIDE_VOLATILE
buzbee9c044ce2012-03-18 13:24:07 -0700736 DF_DA_WIDE | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700737
Elliott Hughesadb8c672012-03-06 16:49:32 -0800738 // EB SPUT_WIDE_VOLATILE
buzbee9c044ce2012-03-18 13:24:07 -0700739 DF_UA_WIDE | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700740
Elliott Hughesadb8c672012-03-06 16:49:32 -0800741 // EC BREAKPOINT
buzbee67bf8852011-08-17 17:51:35 -0700742 DF_NOP,
743
Elliott Hughesadb8c672012-03-06 16:49:32 -0800744 // ED THROW_VERIFICATION_ERROR
buzbee9c044ce2012-03-18 13:24:07 -0700745 DF_NOP | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700746
Elliott Hughesadb8c672012-03-06 16:49:32 -0800747 // EE EXECUTE_INLINE
buzbee67bf8852011-08-17 17:51:35 -0700748 DF_FORMAT_35C,
749
Elliott Hughesadb8c672012-03-06 16:49:32 -0800750 // EF EXECUTE_INLINE_RANGE
buzbee67bf8852011-08-17 17:51:35 -0700751 DF_FORMAT_3RC,
752
Elliott Hughesadb8c672012-03-06 16:49:32 -0800753 // F0 INVOKE_OBJECT_INIT_RANGE
buzbee43a36422011-09-14 14:00:13 -0700754 DF_NOP | DF_NULL_CHK_0,
buzbee67bf8852011-08-17 17:51:35 -0700755
Elliott Hughesadb8c672012-03-06 16:49:32 -0800756 // F1 RETURN_VOID_BARRIER
buzbee67bf8852011-08-17 17:51:35 -0700757 DF_NOP,
758
Elliott Hughesadb8c672012-03-06 16:49:32 -0800759 // F2 IGET_QUICK
buzbee9c044ce2012-03-18 13:24:07 -0700760 DF_DA | DF_UB | DF_NULL_CHK_0,
buzbee67bf8852011-08-17 17:51:35 -0700761
Elliott Hughesadb8c672012-03-06 16:49:32 -0800762 // F3 IGET_WIDE_QUICK
buzbee9c044ce2012-03-18 13:24:07 -0700763 DF_DA_WIDE | DF_UB | DF_NULL_CHK_0,
buzbee67bf8852011-08-17 17:51:35 -0700764
Elliott Hughesadb8c672012-03-06 16:49:32 -0800765 // F4 IGET_OBJECT_QUICK
buzbee9c044ce2012-03-18 13:24:07 -0700766 DF_DA | DF_UB | DF_NULL_CHK_0,
buzbee67bf8852011-08-17 17:51:35 -0700767
Elliott Hughesadb8c672012-03-06 16:49:32 -0800768 // F5 IPUT_QUICK
buzbee9c044ce2012-03-18 13:24:07 -0700769 DF_UA | DF_UB | DF_NULL_CHK_1,
buzbee67bf8852011-08-17 17:51:35 -0700770
Elliott Hughesadb8c672012-03-06 16:49:32 -0800771 // F6 IPUT_WIDE_QUICK
buzbee9c044ce2012-03-18 13:24:07 -0700772 DF_UA_WIDE | DF_UB | DF_NULL_CHK_2,
buzbee67bf8852011-08-17 17:51:35 -0700773
Elliott Hughesadb8c672012-03-06 16:49:32 -0800774 // F7 IPUT_OBJECT_QUICK
buzbee9c044ce2012-03-18 13:24:07 -0700775 DF_UA | DF_UB | DF_NULL_CHK_1,
buzbee67bf8852011-08-17 17:51:35 -0700776
Elliott Hughesadb8c672012-03-06 16:49:32 -0800777 // F8 INVOKE_VIRTUAL_QUICK
buzbee9c044ce2012-03-18 13:24:07 -0700778 DF_FORMAT_35C | DF_NULL_CHK_OUT0 | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700779
Elliott Hughesadb8c672012-03-06 16:49:32 -0800780 // F9 INVOKE_VIRTUAL_QUICK_RANGE
buzbee9c044ce2012-03-18 13:24:07 -0700781 DF_FORMAT_3RC | DF_NULL_CHK_OUT0 | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700782
Elliott Hughesadb8c672012-03-06 16:49:32 -0800783 // FA INVOKE_SUPER_QUICK
buzbee9c044ce2012-03-18 13:24:07 -0700784 DF_FORMAT_35C | DF_NULL_CHK_OUT0 | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700785
Elliott Hughesadb8c672012-03-06 16:49:32 -0800786 // FB INVOKE_SUPER_QUICK_RANGE
buzbee9c044ce2012-03-18 13:24:07 -0700787 DF_FORMAT_3RC | DF_NULL_CHK_OUT0 | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700788
Elliott Hughesadb8c672012-03-06 16:49:32 -0800789 // FC IPUT_OBJECT_VOLATILE
buzbee67bc2362011-10-11 18:08:40 -0700790 DF_UA | DF_UB | DF_NULL_CHK_1 | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700791
Elliott Hughesadb8c672012-03-06 16:49:32 -0800792 // FD SGET_OBJECT_VOLATILE
buzbee9c044ce2012-03-18 13:24:07 -0700793 DF_DA | DF_CORE_A | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700794
Elliott Hughesadb8c672012-03-06 16:49:32 -0800795 // FE SPUT_OBJECT_VOLATILE
buzbee9c044ce2012-03-18 13:24:07 -0700796 DF_UA | DF_CORE_A | DF_UMS,
buzbee67bf8852011-08-17 17:51:35 -0700797
Elliott Hughesadb8c672012-03-06 16:49:32 -0800798 // FF UNUSED_FF
buzbee67bf8852011-08-17 17:51:35 -0700799 DF_NOP,
800
801 // Beginning of extended MIR opcodes
Elliott Hughesadb8c672012-03-06 16:49:32 -0800802 // 100 MIR_PHI
buzbee43a36422011-09-14 14:00:13 -0700803 DF_PHI | DF_DA | DF_NULL_TRANSFER_N,
buzbee84fd6932012-03-29 16:44:16 -0700804
805 // 101 MIR_COPY
806 DF_DA | DF_UB | DF_IS_MOVE,
807
808 // 102 MIR_FUSED_CMPL_FLOAT
809 DF_UA | DF_UB | DF_FP_A | DF_FP_B,
810
811 // 103 MIR_FUSED_CMPG_FLOAT
812 DF_UA | DF_UB | DF_FP_A | DF_FP_B,
813
814 // 104 MIR_FUSED_CMPL_DOUBLE
815 DF_UA_WIDE | DF_UB_WIDE | DF_FP_A | DF_FP_B,
816
817 // 105 MIR_FUSED_CMPG_DOUBLE
818 DF_UA_WIDE | DF_UB_WIDE | DF_FP_A | DF_FP_B,
819
820 // 106 MIR_FUSED_CMP_LONG
821 DF_UA_WIDE | DF_UB_WIDE | DF_CORE_A | DF_CORE_B,
822
823 // 107 MIR_NOP
824 DF_NOP,
825
826 // 108 MIR_NULL_RANGE_UP_CHECK
827 0,
828
829 // 109 MIR_NULL_RANGE_DOWN_CHECK
830 0,
831
832 // 110 MIR_LOWER_BOUND
833 0,
buzbee67bf8852011-08-17 17:51:35 -0700834};
835
buzbeee1965672012-03-11 18:39:19 -0700836/* Return the base virtual register for a SSA name */
837int SRegToVReg(const CompilationUnit* cUnit, int ssaReg)
buzbee67bf8852011-08-17 17:51:35 -0700838{
buzbeee1965672012-03-11 18:39:19 -0700839 DCHECK_LT(ssaReg, (int)cUnit->ssaBaseVRegs->numUsed);
840 return GET_ELEM_N(cUnit->ssaBaseVRegs, int, ssaReg);
buzbee67bf8852011-08-17 17:51:35 -0700841}
842
buzbeee1965672012-03-11 18:39:19 -0700843int SRegToSubscript(const CompilationUnit* cUnit, int ssaReg)
844{
845 DCHECK(ssaReg < (int)cUnit->ssaSubscripts->numUsed);
846 return GET_ELEM_N(cUnit->ssaSubscripts, int, ssaReg);
847}
848
buzbee84fd6932012-03-29 16:44:16 -0700849int getSSAUseCount(CompilationUnit* cUnit, int sReg)
850{
851 DCHECK(sReg < (int)cUnit->rawUseCounts.numUsed);
852 return cUnit->rawUseCounts.elemList[sReg];
853}
854
855
buzbeeba938cb2012-02-03 14:47:55 -0800856char* oatGetDalvikDisassembly(CompilationUnit* cUnit,
Elliott Hughesadb8c672012-03-06 16:49:32 -0800857 const DecodedInstruction& insn, const char* note)
buzbee67bf8852011-08-17 17:51:35 -0700858{
859 char buffer[256];
Elliott Hughesadb8c672012-03-06 16:49:32 -0800860 Instruction::Code opcode = insn.opcode;
buzbee67bf8852011-08-17 17:51:35 -0700861 int dfAttributes = oatDataFlowAttributes[opcode];
862 int flags;
863 char* ret;
864
865 buffer[0] = 0;
866 if ((int)opcode >= (int)kMirOpFirst) {
867 if ((int)opcode == (int)kMirOpPhi) {
868 strcpy(buffer, "PHI");
Elliott Hughesadb8c672012-03-06 16:49:32 -0800869 } else {
buzbee67bf8852011-08-17 17:51:35 -0700870 sprintf(buffer, "Opcode %#x", opcode);
871 }
872 flags = 0;
873 } else {
Elliott Hughesadb8c672012-03-06 16:49:32 -0800874 strcpy(buffer, Instruction::Name(opcode));
875 flags = Instruction::Flags(opcode);
buzbee67bf8852011-08-17 17:51:35 -0700876 }
877
878 if (note)
879 strcat(buffer, note);
880
881 /* For branches, decode the instructions to print out the branch targets */
Elliott Hughesadb8c672012-03-06 16:49:32 -0800882 if (flags & Instruction::kBranch) {
883 Instruction::Format dalvikFormat = Instruction::FormatOf(insn.opcode);
buzbee67bf8852011-08-17 17:51:35 -0700884 int offset = 0;
885 switch (dalvikFormat) {
Elliott Hughesadb8c672012-03-06 16:49:32 -0800886 case Instruction::k21t:
887 snprintf(buffer + strlen(buffer), 256, " v%d,", insn.vA);
888 offset = (int) insn.vB;
buzbee67bf8852011-08-17 17:51:35 -0700889 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -0800890 case Instruction::k22t:
891 snprintf(buffer + strlen(buffer), 256, " v%d, v%d,", insn.vA, insn.vB);
892 offset = (int) insn.vC;
buzbee67bf8852011-08-17 17:51:35 -0700893 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -0800894 case Instruction::k10t:
895 case Instruction::k20t:
896 case Instruction::k30t:
897 offset = (int) insn.vA;
buzbee67bf8852011-08-17 17:51:35 -0700898 break;
899 default:
900 LOG(FATAL) << "Unexpected branch format " << (int)dalvikFormat
901 << " / opcode " << (int)opcode;
902 }
903 snprintf(buffer + strlen(buffer), 256, " (%c%x)",
904 offset > 0 ? '+' : '-',
905 offset > 0 ? offset : -offset);
906 } else if (dfAttributes & DF_FORMAT_35C) {
907 unsigned int i;
Elliott Hughesadb8c672012-03-06 16:49:32 -0800908 for (i = 0; i < insn.vA; i++) {
buzbee67bf8852011-08-17 17:51:35 -0700909 if (i != 0) strcat(buffer, ",");
Elliott Hughesadb8c672012-03-06 16:49:32 -0800910 snprintf(buffer + strlen(buffer), 256, " v%d", insn.arg[i]);
buzbee67bf8852011-08-17 17:51:35 -0700911 }
912 }
913 else if (dfAttributes & DF_FORMAT_3RC) {
914 snprintf(buffer + strlen(buffer), 256,
Elliott Hughesadb8c672012-03-06 16:49:32 -0800915 " v%d..v%d", insn.vC, insn.vC + insn.vA - 1);
buzbee67bf8852011-08-17 17:51:35 -0700916 }
917 else {
918 if (dfAttributes & DF_A_IS_REG) {
Elliott Hughesadb8c672012-03-06 16:49:32 -0800919 snprintf(buffer + strlen(buffer), 256, " v%d", insn.vA);
buzbee67bf8852011-08-17 17:51:35 -0700920 }
921 if (dfAttributes & DF_B_IS_REG) {
Elliott Hughesadb8c672012-03-06 16:49:32 -0800922 snprintf(buffer + strlen(buffer), 256, ", v%d", insn.vB);
buzbee67bf8852011-08-17 17:51:35 -0700923 }
924 else if ((int)opcode < (int)kMirOpFirst) {
Elliott Hughesadb8c672012-03-06 16:49:32 -0800925 snprintf(buffer + strlen(buffer), 256, ", (#%d)", insn.vB);
buzbee67bf8852011-08-17 17:51:35 -0700926 }
927 if (dfAttributes & DF_C_IS_REG) {
Elliott Hughesadb8c672012-03-06 16:49:32 -0800928 snprintf(buffer + strlen(buffer), 256, ", v%d", insn.vC);
buzbee67bf8852011-08-17 17:51:35 -0700929 }
930 else if ((int)opcode < (int)kMirOpFirst) {
Elliott Hughesadb8c672012-03-06 16:49:32 -0800931 snprintf(buffer + strlen(buffer), 256, ", (#%d)", insn.vC);
buzbee67bf8852011-08-17 17:51:35 -0700932 }
933 }
934 int length = strlen(buffer) + 1;
buzbeeba938cb2012-02-03 14:47:55 -0800935 ret = (char*)oatNew(cUnit, length, false, kAllocDFInfo);
buzbee67bf8852011-08-17 17:51:35 -0700936 memcpy(ret, buffer, length);
937 return ret;
938}
939
Elliott Hughesc1f143d2011-12-01 17:31:10 -0800940char* getSSAName(const CompilationUnit* cUnit, int ssaReg, char* name)
buzbee67bf8852011-08-17 17:51:35 -0700941{
buzbeee1965672012-03-11 18:39:19 -0700942 sprintf(name, "v%d_%d", SRegToVReg(cUnit, ssaReg),
943 SRegToSubscript(cUnit, ssaReg));
buzbee67bf8852011-08-17 17:51:35 -0700944 return name;
945}
946
947/*
948 * Dalvik instruction disassembler with optional SSA printing.
949 */
buzbee31a4a6f2012-02-28 15:36:15 -0800950char* oatFullDisassembler(CompilationUnit* cUnit, const MIR* mir)
buzbee67bf8852011-08-17 17:51:35 -0700951{
952 char buffer[256];
Elliott Hughes3b6baaa2011-10-14 19:13:56 -0700953 char operand0[32], operand1[32];
Elliott Hughesadb8c672012-03-06 16:49:32 -0800954 const DecodedInstruction* insn = &mir->dalvikInsn;
955 Instruction::Code opcode = insn->opcode;
buzbee67bf8852011-08-17 17:51:35 -0700956 int dfAttributes = oatDataFlowAttributes[opcode];
Elliott Hughesc1f143d2011-12-01 17:31:10 -0800957 char* ret;
buzbee67bf8852011-08-17 17:51:35 -0700958 int length;
buzbee67bf8852011-08-17 17:51:35 -0700959
960 buffer[0] = 0;
Elliott Hughesadb8c672012-03-06 16:49:32 -0800961 if (static_cast<int>(opcode) >= static_cast<int>(kMirOpFirst)) {
962 if (static_cast<int>(opcode) == static_cast<int>(kMirOpPhi)) {
buzbee67bf8852011-08-17 17:51:35 -0700963 snprintf(buffer, 256, "PHI %s = (%s",
964 getSSAName(cUnit, mir->ssaRep->defs[0], operand0),
965 getSSAName(cUnit, mir->ssaRep->uses[0], operand1));
966 int i;
967 for (i = 1; i < mir->ssaRep->numUses; i++) {
968 snprintf(buffer + strlen(buffer), 256, ", %s",
969 getSSAName(cUnit, mir->ssaRep->uses[i], operand0));
970 }
971 snprintf(buffer + strlen(buffer), 256, ")");
972 }
973 else {
974 sprintf(buffer, "Opcode %#x", opcode);
975 }
976 goto done;
977 } else {
Elliott Hughesadb8c672012-03-06 16:49:32 -0800978 strcpy(buffer, Instruction::Name(opcode));
buzbee67bf8852011-08-17 17:51:35 -0700979 }
980
buzbee67bf8852011-08-17 17:51:35 -0700981 /* For branches, decode the instructions to print out the branch targets */
Elliott Hughesadb8c672012-03-06 16:49:32 -0800982 if (Instruction::Flags(insn->opcode) & Instruction::kBranch) {
983 Instruction::Format dalvikFormat = Instruction::FormatOf(insn->opcode);
buzbee67bf8852011-08-17 17:51:35 -0700984 int delta = 0;
985 switch (dalvikFormat) {
Elliott Hughesadb8c672012-03-06 16:49:32 -0800986 case Instruction::k21t:
buzbee67bf8852011-08-17 17:51:35 -0700987 snprintf(buffer + strlen(buffer), 256, " %s, ",
988 getSSAName(cUnit, mir->ssaRep->uses[0], operand0));
989 delta = (int) insn->vB;
990 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -0800991 case Instruction::k22t:
buzbee67bf8852011-08-17 17:51:35 -0700992 snprintf(buffer + strlen(buffer), 256, " %s, %s, ",
993 getSSAName(cUnit, mir->ssaRep->uses[0], operand0),
994 getSSAName(cUnit, mir->ssaRep->uses[1], operand1));
995 delta = (int) insn->vC;
996 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -0800997 case Instruction::k10t:
998 case Instruction::k20t:
999 case Instruction::k30t:
buzbee67bf8852011-08-17 17:51:35 -07001000 delta = (int) insn->vA;
1001 break;
1002 default:
1003 LOG(FATAL) << "Unexpected branch format: " <<
1004 (int)dalvikFormat;
1005 }
1006 snprintf(buffer + strlen(buffer), 256, " %04x",
1007 mir->offset + delta);
1008 } else if (dfAttributes & (DF_FORMAT_35C | DF_FORMAT_3RC)) {
1009 unsigned int i;
1010 for (i = 0; i < insn->vA; i++) {
1011 if (i != 0) strcat(buffer, ",");
1012 snprintf(buffer + strlen(buffer), 256, " %s",
1013 getSSAName(cUnit, mir->ssaRep->uses[i], operand0));
1014 }
1015 } else {
1016 int udIdx;
1017 if (mir->ssaRep->numDefs) {
1018
1019 for (udIdx = 0; udIdx < mir->ssaRep->numDefs; udIdx++) {
1020 snprintf(buffer + strlen(buffer), 256, " %s",
1021 getSSAName(cUnit, mir->ssaRep->defs[udIdx], operand0));
1022 }
1023 strcat(buffer, ",");
1024 }
1025 if (mir->ssaRep->numUses) {
1026 /* No leading ',' for the first use */
1027 snprintf(buffer + strlen(buffer), 256, " %s",
1028 getSSAName(cUnit, mir->ssaRep->uses[0], operand0));
1029 for (udIdx = 1; udIdx < mir->ssaRep->numUses; udIdx++) {
1030 snprintf(buffer + strlen(buffer), 256, ", %s",
1031 getSSAName(cUnit, mir->ssaRep->uses[udIdx], operand0));
1032 }
1033 }
Elliott Hughesadb8c672012-03-06 16:49:32 -08001034 if (static_cast<int>(opcode) < static_cast<int>(kMirOpFirst)) {
1035 Instruction::Format dalvikFormat = Instruction::FormatOf(opcode);
buzbee67bf8852011-08-17 17:51:35 -07001036 switch (dalvikFormat) {
Elliott Hughesadb8c672012-03-06 16:49:32 -08001037 case Instruction::k11n: // op vA, #+B
1038 case Instruction::k21s: // op vAA, #+BBBB
1039 case Instruction::k21h: // op vAA, #+BBBB00000[00000000]
1040 case Instruction::k31i: // op vAA, #+BBBBBBBB
1041 case Instruction::k51l: // op vAA, #+BBBBBBBBBBBBBBBB
buzbee67bf8852011-08-17 17:51:35 -07001042 snprintf(buffer + strlen(buffer), 256, " #%#x", insn->vB);
1043 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001044 case Instruction::k21c: // op vAA, thing@BBBB
1045 case Instruction::k31c: // op vAA, thing@BBBBBBBB
buzbee67bf8852011-08-17 17:51:35 -07001046 snprintf(buffer + strlen(buffer), 256, " @%#x", insn->vB);
1047 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001048 case Instruction::k22b: // op vAA, vBB, #+CC
1049 case Instruction::k22s: // op vA, vB, #+CCCC
buzbee67bf8852011-08-17 17:51:35 -07001050 snprintf(buffer + strlen(buffer), 256, " #%#x", insn->vC);
1051 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001052 case Instruction::k22c: // op vA, vB, thing@CCCC
buzbee67bf8852011-08-17 17:51:35 -07001053 snprintf(buffer + strlen(buffer), 256, " @%#x", insn->vC);
1054 break;
1055 /* No need for special printing */
1056 default:
1057 break;
1058 }
1059 }
1060 }
1061
1062done:
1063 length = strlen(buffer) + 1;
buzbeeba938cb2012-02-03 14:47:55 -08001064 ret = (char*) oatNew(cUnit, length, false, kAllocDFInfo);
buzbee67bf8852011-08-17 17:51:35 -07001065 memcpy(ret, buffer, length);
1066 return ret;
1067}
1068
Elliott Hughesc1f143d2011-12-01 17:31:10 -08001069char* oatGetSSAString(CompilationUnit* cUnit, SSARepresentation* ssaRep)
buzbee67bf8852011-08-17 17:51:35 -07001070{
1071 char buffer[256];
1072 char* ret;
1073 int i;
1074
1075 buffer[0] = 0;
1076 for (i = 0; i < ssaRep->numDefs; i++) {
buzbeee1965672012-03-11 18:39:19 -07001077 int ssaReg = ssaRep->defs[i];
1078 sprintf(buffer + strlen(buffer), "s%d(v%d_%d) ", ssaReg,
1079 SRegToVReg(cUnit, ssaReg), SRegToSubscript(cUnit, ssaReg));
buzbee67bf8852011-08-17 17:51:35 -07001080 }
1081
1082 if (ssaRep->numDefs) {
1083 strcat(buffer, "<- ");
1084 }
1085
1086 for (i = 0; i < ssaRep->numUses; i++) {
buzbee67bf8852011-08-17 17:51:35 -07001087 int len = strlen(buffer);
buzbeee1965672012-03-11 18:39:19 -07001088 int ssaReg = ssaRep->uses[i];
buzbee67bf8852011-08-17 17:51:35 -07001089
buzbeee1965672012-03-11 18:39:19 -07001090 if (snprintf(buffer + len, 250 - len, "s%d(v%d_%d) ", ssaReg,
1091 SRegToVReg(cUnit, ssaReg),
1092 SRegToSubscript(cUnit, ssaReg))) {
buzbee67bf8852011-08-17 17:51:35 -07001093 strcat(buffer, "...");
1094 break;
1095 }
1096 }
1097
1098 int length = strlen(buffer) + 1;
buzbeeba938cb2012-02-03 14:47:55 -08001099 ret = (char*)oatNew(cUnit, length, false, kAllocDFInfo);
buzbee67bf8852011-08-17 17:51:35 -07001100 memcpy(ret, buffer, length);
1101 return ret;
1102}
1103
1104/* Any register that is used before being defined is considered live-in */
buzbee31a4a6f2012-02-28 15:36:15 -08001105inline void handleLiveInUse(CompilationUnit* cUnit, ArenaBitVector* useV,
1106 ArenaBitVector* defV, ArenaBitVector* liveInV,
1107 int dalvikRegId)
buzbee67bf8852011-08-17 17:51:35 -07001108{
buzbeeba938cb2012-02-03 14:47:55 -08001109 oatSetBit(cUnit, useV, dalvikRegId);
buzbee67bf8852011-08-17 17:51:35 -07001110 if (!oatIsBitSet(defV, dalvikRegId)) {
buzbeeba938cb2012-02-03 14:47:55 -08001111 oatSetBit(cUnit, liveInV, dalvikRegId);
buzbee67bf8852011-08-17 17:51:35 -07001112 }
1113}
1114
1115/* Mark a reg as being defined */
buzbee31a4a6f2012-02-28 15:36:15 -08001116inline void handleDef(CompilationUnit* cUnit, ArenaBitVector* defV,
1117 int dalvikRegId)
buzbee67bf8852011-08-17 17:51:35 -07001118{
buzbeeba938cb2012-02-03 14:47:55 -08001119 oatSetBit(cUnit, defV, dalvikRegId);
buzbee67bf8852011-08-17 17:51:35 -07001120}
1121
1122/*
1123 * Find out live-in variables for natural loops. Variables that are live-in in
1124 * the main loop body are considered to be defined in the entry block.
1125 */
1126bool oatFindLocalLiveIn(CompilationUnit* cUnit, BasicBlock* bb)
1127{
1128 MIR* mir;
1129 ArenaBitVector *useV, *defV, *liveInV;
1130
1131 if (bb->dataFlowInfo == NULL) return false;
1132
1133 useV = bb->dataFlowInfo->useV =
buzbeeba938cb2012-02-03 14:47:55 -08001134 oatAllocBitVector(cUnit, cUnit->numDalvikRegisters, false, kBitMapUse);
buzbee67bf8852011-08-17 17:51:35 -07001135 defV = bb->dataFlowInfo->defV =
buzbeeba938cb2012-02-03 14:47:55 -08001136 oatAllocBitVector(cUnit, cUnit->numDalvikRegisters, false, kBitMapDef);
buzbee67bf8852011-08-17 17:51:35 -07001137 liveInV = bb->dataFlowInfo->liveInV =
buzbeeba938cb2012-02-03 14:47:55 -08001138 oatAllocBitVector(cUnit, cUnit->numDalvikRegisters, false,
1139 kBitMapLiveIn);
buzbee67bf8852011-08-17 17:51:35 -07001140
1141 for (mir = bb->firstMIRInsn; mir; mir = mir->next) {
1142 int dfAttributes =
1143 oatDataFlowAttributes[mir->dalvikInsn.opcode];
1144 DecodedInstruction *dInsn = &mir->dalvikInsn;
1145
1146 if (dfAttributes & DF_HAS_USES) {
1147 if (dfAttributes & DF_UA) {
buzbeeba938cb2012-02-03 14:47:55 -08001148 handleLiveInUse(cUnit, useV, defV, liveInV, dInsn->vA);
buzbee67bf8852011-08-17 17:51:35 -07001149 } else if (dfAttributes & DF_UA_WIDE) {
buzbeeba938cb2012-02-03 14:47:55 -08001150 handleLiveInUse(cUnit, useV, defV, liveInV, dInsn->vA);
1151 handleLiveInUse(cUnit, useV, defV, liveInV, dInsn->vA+1);
buzbee67bf8852011-08-17 17:51:35 -07001152 }
1153 if (dfAttributes & DF_UB) {
buzbeeba938cb2012-02-03 14:47:55 -08001154 handleLiveInUse(cUnit, useV, defV, liveInV, dInsn->vB);
buzbee67bf8852011-08-17 17:51:35 -07001155 } else if (dfAttributes & DF_UB_WIDE) {
buzbeeba938cb2012-02-03 14:47:55 -08001156 handleLiveInUse(cUnit, useV, defV, liveInV, dInsn->vB);
1157 handleLiveInUse(cUnit, useV, defV, liveInV, dInsn->vB+1);
buzbee67bf8852011-08-17 17:51:35 -07001158 }
1159 if (dfAttributes & DF_UC) {
buzbeeba938cb2012-02-03 14:47:55 -08001160 handleLiveInUse(cUnit, useV, defV, liveInV, dInsn->vC);
buzbee67bf8852011-08-17 17:51:35 -07001161 } else if (dfAttributes & DF_UC_WIDE) {
buzbeeba938cb2012-02-03 14:47:55 -08001162 handleLiveInUse(cUnit, useV, defV, liveInV, dInsn->vC);
1163 handleLiveInUse(cUnit, useV, defV, liveInV, dInsn->vC+1);
buzbee67bf8852011-08-17 17:51:35 -07001164 }
1165 }
buzbee07ce1d72012-02-10 17:22:02 -08001166 if (dfAttributes & DF_FORMAT_35C) {
1167 for (unsigned int i = 0; i < dInsn->vA; i++) {
1168 handleLiveInUse(cUnit, useV, defV, liveInV, dInsn->arg[i]);
1169 }
1170 }
1171 if (dfAttributes & DF_FORMAT_3RC) {
1172 for (unsigned int i = 0; i < dInsn->vA; i++) {
1173 handleLiveInUse(cUnit, useV, defV, liveInV, dInsn->vC+i);
1174 }
1175 }
buzbee67bf8852011-08-17 17:51:35 -07001176 if (dfAttributes & DF_HAS_DEFS) {
buzbeeba938cb2012-02-03 14:47:55 -08001177 handleDef(cUnit, defV, dInsn->vA);
buzbee67bf8852011-08-17 17:51:35 -07001178 if (dfAttributes & DF_DA_WIDE) {
buzbeeba938cb2012-02-03 14:47:55 -08001179 handleDef(cUnit, defV, dInsn->vA+1);
buzbee67bf8852011-08-17 17:51:35 -07001180 }
1181 }
1182 }
1183 return true;
1184}
1185
buzbeee1965672012-03-11 18:39:19 -07001186int addNewSReg(CompilationUnit* cUnit, int vReg)
1187{
1188 // Compiler temps always have a subscript of 0
1189 int subscript = (vReg < 0) ? 0 : ++cUnit->SSALastDefs[vReg];
1190 int ssaReg = cUnit->numSSARegs++;
1191 oatInsertGrowableList(cUnit, cUnit->ssaBaseVRegs, vReg);
1192 oatInsertGrowableList(cUnit, cUnit->ssaSubscripts, subscript);
1193 DCHECK_EQ(cUnit->ssaBaseVRegs->numUsed, cUnit->ssaSubscripts->numUsed);
1194 return ssaReg;
1195}
1196
buzbee67bf8852011-08-17 17:51:35 -07001197/* Find out the latest SSA register for a given Dalvik register */
buzbee31a4a6f2012-02-28 15:36:15 -08001198void handleSSAUse(CompilationUnit* cUnit, int* uses, int dalvikReg,
1199 int regIndex)
buzbee67bf8852011-08-17 17:51:35 -07001200{
buzbeee1965672012-03-11 18:39:19 -07001201 DCHECK((dalvikReg >= 0) && (dalvikReg < cUnit->numDalvikRegisters));
1202 uses[regIndex] = cUnit->vRegToSSAMap[dalvikReg];
buzbee67bf8852011-08-17 17:51:35 -07001203}
1204
1205/* Setup a new SSA register for a given Dalvik register */
buzbee31a4a6f2012-02-28 15:36:15 -08001206void handleSSADef(CompilationUnit* cUnit, int* defs, int dalvikReg,
1207 int regIndex)
buzbee67bf8852011-08-17 17:51:35 -07001208{
buzbeee1965672012-03-11 18:39:19 -07001209 DCHECK((dalvikReg >= 0) && (dalvikReg < cUnit->numDalvikRegisters));
1210 int ssaReg = addNewSReg(cUnit, dalvikReg);
1211 cUnit->vRegToSSAMap[dalvikReg] = ssaReg;
buzbee67bf8852011-08-17 17:51:35 -07001212 defs[regIndex] = ssaReg;
1213}
1214
buzbeeec5adf32011-09-11 15:25:43 -07001215/* Look up new SSA names for format_35c instructions */
buzbee31a4a6f2012-02-28 15:36:15 -08001216void dataFlowSSAFormat35C(CompilationUnit* cUnit, MIR* mir)
buzbee67bf8852011-08-17 17:51:35 -07001217{
1218 DecodedInstruction *dInsn = &mir->dalvikInsn;
1219 int numUses = dInsn->vA;
1220 int i;
1221
1222 mir->ssaRep->numUses = numUses;
buzbeeba938cb2012-02-03 14:47:55 -08001223 mir->ssaRep->uses = (int *)oatNew(cUnit, sizeof(int) * numUses, true,
buzbee5abfa3e2012-01-31 17:01:43 -08001224 kAllocDFInfo);
buzbeeed3e9302011-09-23 17:34:19 -07001225 // NOTE: will be filled in during type & size inference pass
buzbeeba938cb2012-02-03 14:47:55 -08001226 mir->ssaRep->fpUse = (bool *)oatNew(cUnit, sizeof(bool) * numUses, true,
buzbee5abfa3e2012-01-31 17:01:43 -08001227 kAllocDFInfo);
buzbee67bf8852011-08-17 17:51:35 -07001228
1229 for (i = 0; i < numUses; i++) {
1230 handleSSAUse(cUnit, mir->ssaRep->uses, dInsn->arg[i], i);
1231 }
1232}
1233
buzbeeec5adf32011-09-11 15:25:43 -07001234/* Look up new SSA names for format_3rc instructions */
buzbee31a4a6f2012-02-28 15:36:15 -08001235void dataFlowSSAFormat3RC(CompilationUnit* cUnit, MIR* mir)
buzbee67bf8852011-08-17 17:51:35 -07001236{
1237 DecodedInstruction *dInsn = &mir->dalvikInsn;
1238 int numUses = dInsn->vA;
1239 int i;
1240
1241 mir->ssaRep->numUses = numUses;
buzbeeba938cb2012-02-03 14:47:55 -08001242 mir->ssaRep->uses = (int *)oatNew(cUnit, sizeof(int) * numUses, true,
buzbee5abfa3e2012-01-31 17:01:43 -08001243 kAllocDFInfo);
buzbeeed3e9302011-09-23 17:34:19 -07001244 // NOTE: will be filled in during type & size inference pass
buzbeeba938cb2012-02-03 14:47:55 -08001245 mir->ssaRep->fpUse = (bool *)oatNew(cUnit, sizeof(bool) * numUses, true,
buzbee5abfa3e2012-01-31 17:01:43 -08001246 kAllocDFInfo);
buzbee67bf8852011-08-17 17:51:35 -07001247
1248 for (i = 0; i < numUses; i++) {
1249 handleSSAUse(cUnit, mir->ssaRep->uses, dInsn->vC+i, i);
1250 }
1251}
1252
1253/* Entry function to convert a block into SSA representation */
1254bool oatDoSSAConversion(CompilationUnit* cUnit, BasicBlock* bb)
1255{
1256 MIR* mir;
1257
1258 if (bb->dataFlowInfo == NULL) return false;
1259
1260 for (mir = bb->firstMIRInsn; mir; mir = mir->next) {
1261 mir->ssaRep = (struct SSARepresentation *)
buzbeeba938cb2012-02-03 14:47:55 -08001262 oatNew(cUnit, sizeof(SSARepresentation), true, kAllocDFInfo);
buzbee67bf8852011-08-17 17:51:35 -07001263
1264 int dfAttributes =
1265 oatDataFlowAttributes[mir->dalvikInsn.opcode];
1266
buzbeef0cde542011-09-13 14:55:02 -07001267 // If not a pseudo-op, note non-leaf or can throw
Elliott Hughesadb8c672012-03-06 16:49:32 -08001268 if (static_cast<int>(mir->dalvikInsn.opcode) < static_cast<int>(kNumPackedOpcodes)) {
1269 int flags = Instruction::Flags(mir->dalvikInsn.opcode);
buzbeecefd1872011-09-09 09:59:52 -07001270
Elliott Hughesadb8c672012-03-06 16:49:32 -08001271 if (flags & Instruction::kThrow) {
buzbeef0cde542011-09-13 14:55:02 -07001272 cUnit->attrs &= ~METHOD_IS_THROW_FREE;
1273 }
buzbeecefd1872011-09-09 09:59:52 -07001274
Elliott Hughesadb8c672012-03-06 16:49:32 -08001275 if (flags & Instruction::kInvoke) {
buzbeef0cde542011-09-13 14:55:02 -07001276 cUnit->attrs &= ~METHOD_IS_LEAF;
1277 }
buzbeecefd1872011-09-09 09:59:52 -07001278 }
1279
buzbee67bf8852011-08-17 17:51:35 -07001280 int numUses = 0;
1281
1282 if (dfAttributes & DF_FORMAT_35C) {
1283 dataFlowSSAFormat35C(cUnit, mir);
1284 continue;
1285 }
1286
1287 if (dfAttributes & DF_FORMAT_3RC) {
1288 dataFlowSSAFormat3RC(cUnit, mir);
1289 continue;
1290 }
1291
1292 if (dfAttributes & DF_HAS_USES) {
1293 if (dfAttributes & DF_UA) {
1294 numUses++;
1295 } else if (dfAttributes & DF_UA_WIDE) {
1296 numUses += 2;
1297 }
1298 if (dfAttributes & DF_UB) {
1299 numUses++;
1300 } else if (dfAttributes & DF_UB_WIDE) {
1301 numUses += 2;
1302 }
1303 if (dfAttributes & DF_UC) {
1304 numUses++;
1305 } else if (dfAttributes & DF_UC_WIDE) {
1306 numUses += 2;
1307 }
1308 }
1309
1310 if (numUses) {
1311 mir->ssaRep->numUses = numUses;
buzbeeba938cb2012-02-03 14:47:55 -08001312 mir->ssaRep->uses = (int *)oatNew(cUnit, sizeof(int) * numUses,
buzbee5abfa3e2012-01-31 17:01:43 -08001313 false, kAllocDFInfo);
buzbeeba938cb2012-02-03 14:47:55 -08001314 mir->ssaRep->fpUse = (bool *)oatNew(cUnit, sizeof(bool) * numUses,
buzbee5abfa3e2012-01-31 17:01:43 -08001315 false, kAllocDFInfo);
buzbee67bf8852011-08-17 17:51:35 -07001316 }
1317
1318 int numDefs = 0;
1319
1320 if (dfAttributes & DF_HAS_DEFS) {
1321 numDefs++;
1322 if (dfAttributes & DF_DA_WIDE) {
1323 numDefs++;
1324 }
1325 }
1326
1327 if (numDefs) {
1328 mir->ssaRep->numDefs = numDefs;
buzbeeba938cb2012-02-03 14:47:55 -08001329 mir->ssaRep->defs = (int *)oatNew(cUnit, sizeof(int) * numDefs,
buzbee5abfa3e2012-01-31 17:01:43 -08001330 false, kAllocDFInfo);
buzbeeba938cb2012-02-03 14:47:55 -08001331 mir->ssaRep->fpDef = (bool *)oatNew(cUnit, sizeof(bool) * numDefs,
buzbee5abfa3e2012-01-31 17:01:43 -08001332 false, kAllocDFInfo);
buzbee67bf8852011-08-17 17:51:35 -07001333 }
1334
1335 DecodedInstruction *dInsn = &mir->dalvikInsn;
1336
1337 if (dfAttributes & DF_HAS_USES) {
1338 numUses = 0;
1339 if (dfAttributes & DF_UA) {
1340 mir->ssaRep->fpUse[numUses] = dfAttributes & DF_FP_A;
1341 handleSSAUse(cUnit, mir->ssaRep->uses, dInsn->vA, numUses++);
1342 } else if (dfAttributes & DF_UA_WIDE) {
1343 mir->ssaRep->fpUse[numUses] = dfAttributes & DF_FP_A;
1344 handleSSAUse(cUnit, mir->ssaRep->uses, dInsn->vA, numUses++);
1345 mir->ssaRep->fpUse[numUses] = dfAttributes & DF_FP_A;
1346 handleSSAUse(cUnit, mir->ssaRep->uses, dInsn->vA+1, numUses++);
1347 }
1348 if (dfAttributes & DF_UB) {
1349 mir->ssaRep->fpUse[numUses] = dfAttributes & DF_FP_B;
1350 handleSSAUse(cUnit, mir->ssaRep->uses, dInsn->vB, numUses++);
1351 } else if (dfAttributes & DF_UB_WIDE) {
1352 mir->ssaRep->fpUse[numUses] = dfAttributes & DF_FP_B;
1353 handleSSAUse(cUnit, mir->ssaRep->uses, dInsn->vB, numUses++);
1354 mir->ssaRep->fpUse[numUses] = dfAttributes & DF_FP_B;
1355 handleSSAUse(cUnit, mir->ssaRep->uses, dInsn->vB+1, numUses++);
1356 }
1357 if (dfAttributes & DF_UC) {
1358 mir->ssaRep->fpUse[numUses] = dfAttributes & DF_FP_C;
1359 handleSSAUse(cUnit, mir->ssaRep->uses, dInsn->vC, numUses++);
1360 } else if (dfAttributes & DF_UC_WIDE) {
1361 mir->ssaRep->fpUse[numUses] = dfAttributes & DF_FP_C;
1362 handleSSAUse(cUnit, mir->ssaRep->uses, dInsn->vC, numUses++);
1363 mir->ssaRep->fpUse[numUses] = dfAttributes & DF_FP_C;
1364 handleSSAUse(cUnit, mir->ssaRep->uses, dInsn->vC+1, numUses++);
1365 }
1366 }
1367 if (dfAttributes & DF_HAS_DEFS) {
1368 mir->ssaRep->fpDef[0] = dfAttributes & DF_FP_A;
1369 handleSSADef(cUnit, mir->ssaRep->defs, dInsn->vA, 0);
1370 if (dfAttributes & DF_DA_WIDE) {
1371 mir->ssaRep->fpDef[1] = dfAttributes & DF_FP_A;
1372 handleSSADef(cUnit, mir->ssaRep->defs, dInsn->vA+1, 1);
1373 }
1374 }
1375 }
1376
buzbee5abfa3e2012-01-31 17:01:43 -08001377 if (!cUnit->disableDataflow) {
1378 /*
1379 * Take a snapshot of Dalvik->SSA mapping at the end of each block. The
1380 * input to PHI nodes can be derived from the snapshot of all
1381 * predecessor blocks.
1382 */
buzbeee1965672012-03-11 18:39:19 -07001383 bb->dataFlowInfo->vRegToSSAMap =
buzbeeba938cb2012-02-03 14:47:55 -08001384 (int *)oatNew(cUnit, sizeof(int) * cUnit->numDalvikRegisters, false,
buzbee5abfa3e2012-01-31 17:01:43 -08001385 kAllocDFInfo);
buzbee67bf8852011-08-17 17:51:35 -07001386
buzbeee1965672012-03-11 18:39:19 -07001387 memcpy(bb->dataFlowInfo->vRegToSSAMap, cUnit->vRegToSSAMap,
buzbee5abfa3e2012-01-31 17:01:43 -08001388 sizeof(int) * cUnit->numDalvikRegisters);
1389 }
buzbee67bf8852011-08-17 17:51:35 -07001390 return true;
1391}
1392
1393/* Setup a constant value for opcodes thare have the DF_SETS_CONST attribute */
buzbee31a4a6f2012-02-28 15:36:15 -08001394void setConstant(CompilationUnit* cUnit, int ssaReg, int value)
buzbee67bf8852011-08-17 17:51:35 -07001395{
buzbeeba938cb2012-02-03 14:47:55 -08001396 oatSetBit(cUnit, cUnit->isConstantV, ssaReg);
buzbee67bf8852011-08-17 17:51:35 -07001397 cUnit->constantValues[ssaReg] = value;
1398}
1399
1400bool oatDoConstantPropagation(CompilationUnit* cUnit, BasicBlock* bb)
1401{
1402 MIR* mir;
1403 ArenaBitVector *isConstantV = cUnit->isConstantV;
1404
1405 for (mir = bb->firstMIRInsn; mir; mir = mir->next) {
1406 int dfAttributes =
1407 oatDataFlowAttributes[mir->dalvikInsn.opcode];
1408
1409 DecodedInstruction *dInsn = &mir->dalvikInsn;
1410
1411 if (!(dfAttributes & DF_HAS_DEFS)) continue;
1412
1413 /* Handle instructions that set up constants directly */
1414 if (dfAttributes & DF_SETS_CONST) {
1415 if (dfAttributes & DF_DA) {
1416 switch (dInsn->opcode) {
Elliott Hughesadb8c672012-03-06 16:49:32 -08001417 case Instruction::CONST_4:
1418 case Instruction::CONST_16:
1419 case Instruction::CONST:
buzbee67bf8852011-08-17 17:51:35 -07001420 setConstant(cUnit, mir->ssaRep->defs[0], dInsn->vB);
1421 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001422 case Instruction::CONST_HIGH16:
buzbee67bf8852011-08-17 17:51:35 -07001423 setConstant(cUnit, mir->ssaRep->defs[0],
1424 dInsn->vB << 16);
1425 break;
1426 default:
1427 break;
1428 }
1429 } else if (dfAttributes & DF_DA_WIDE) {
1430 switch (dInsn->opcode) {
Elliott Hughesadb8c672012-03-06 16:49:32 -08001431 case Instruction::CONST_WIDE_16:
1432 case Instruction::CONST_WIDE_32:
buzbee67bf8852011-08-17 17:51:35 -07001433 setConstant(cUnit, mir->ssaRep->defs[0], dInsn->vB);
1434 setConstant(cUnit, mir->ssaRep->defs[1], 0);
1435 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001436 case Instruction::CONST_WIDE:
buzbee67bf8852011-08-17 17:51:35 -07001437 setConstant(cUnit, mir->ssaRep->defs[0],
1438 (int) dInsn->vB_wide);
1439 setConstant(cUnit, mir->ssaRep->defs[1],
1440 (int) (dInsn->vB_wide >> 32));
1441 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001442 case Instruction::CONST_WIDE_HIGH16:
buzbee67bf8852011-08-17 17:51:35 -07001443 setConstant(cUnit, mir->ssaRep->defs[0], 0);
1444 setConstant(cUnit, mir->ssaRep->defs[1],
1445 dInsn->vB << 16);
1446 break;
1447 default:
1448 break;
1449 }
1450 }
1451 /* Handle instructions that set up constants directly */
1452 } else if (dfAttributes & DF_IS_MOVE) {
1453 int i;
1454
1455 for (i = 0; i < mir->ssaRep->numUses; i++) {
1456 if (!oatIsBitSet(isConstantV, mir->ssaRep->uses[i])) break;
1457 }
1458 /* Move a register holding a constant to another register */
1459 if (i == mir->ssaRep->numUses) {
1460 setConstant(cUnit, mir->ssaRep->defs[0],
1461 cUnit->constantValues[mir->ssaRep->uses[0]]);
1462 if (dfAttributes & DF_DA_WIDE) {
1463 setConstant(cUnit, mir->ssaRep->defs[1],
1464 cUnit->constantValues[mir->ssaRep->uses[1]]);
1465 }
1466 }
1467 }
1468 }
1469 /* TODO: implement code to handle arithmetic operations */
1470 return true;
1471}
1472
1473/* Setup the basic data structures for SSA conversion */
1474void oatInitializeSSAConversion(CompilationUnit* cUnit)
1475{
1476 int i;
Ian Rogersa3760aa2011-11-14 14:32:37 -08001477 int numDalvikReg = cUnit->numDalvikRegisters;
buzbee67bf8852011-08-17 17:51:35 -07001478
buzbeee1965672012-03-11 18:39:19 -07001479 cUnit->ssaBaseVRegs = (GrowableList *)oatNew(cUnit, sizeof(GrowableList),
1480 false, kAllocDFInfo);
1481 cUnit->ssaSubscripts = (GrowableList *)oatNew(cUnit, sizeof(GrowableList),
1482 false, kAllocDFInfo);
1483 // Create the ssa mappings, estimating the max size
1484 oatInitGrowableList(cUnit, cUnit->ssaBaseVRegs,
1485 numDalvikReg + cUnit->defCount + 128,
1486 kListSSAtoDalvikMap);
1487 oatInitGrowableList(cUnit, cUnit->ssaSubscripts,
buzbee5abfa3e2012-01-31 17:01:43 -08001488 numDalvikReg + cUnit->defCount + 128,
1489 kListSSAtoDalvikMap);
buzbee67bf8852011-08-17 17:51:35 -07001490 /*
1491 * Initial number of SSA registers is equal to the number of Dalvik
1492 * registers.
1493 */
1494 cUnit->numSSARegs = numDalvikReg;
1495
1496 /*
1497 * Initialize the SSA2Dalvik map list. For the first numDalvikReg elements,
1498 * the subscript is 0 so we use the ENCODE_REG_SUB macro to encode the value
1499 * into "(0 << 16) | i"
1500 */
1501 for (i = 0; i < numDalvikReg; i++) {
buzbeee1965672012-03-11 18:39:19 -07001502 oatInsertGrowableList(cUnit, cUnit->ssaBaseVRegs, i);
1503 oatInsertGrowableList(cUnit, cUnit->ssaSubscripts, 0);
buzbee67bf8852011-08-17 17:51:35 -07001504 }
1505
1506 /*
buzbeee1965672012-03-11 18:39:19 -07001507 * Initialize the DalvikToSSAMap map. There is one entry for each
1508 * Dalvik register, and the SSA names for those are the same.
buzbee67bf8852011-08-17 17:51:35 -07001509 */
buzbeee1965672012-03-11 18:39:19 -07001510 cUnit->vRegToSSAMap = (int *)oatNew(cUnit, sizeof(int) * numDalvikReg,
buzbee5abfa3e2012-01-31 17:01:43 -08001511 false, kAllocDFInfo);
buzbeef0cde542011-09-13 14:55:02 -07001512 /* Keep track of the higest def for each dalvik reg */
buzbeeba938cb2012-02-03 14:47:55 -08001513 cUnit->SSALastDefs = (int *)oatNew(cUnit, sizeof(int) * numDalvikReg,
buzbee5abfa3e2012-01-31 17:01:43 -08001514 false, kAllocDFInfo);
buzbeef0cde542011-09-13 14:55:02 -07001515
buzbee67bf8852011-08-17 17:51:35 -07001516 for (i = 0; i < numDalvikReg; i++) {
buzbeee1965672012-03-11 18:39:19 -07001517 cUnit->vRegToSSAMap[i] = i;
buzbeef0cde542011-09-13 14:55:02 -07001518 cUnit->SSALastDefs[i] = 0;
buzbee67bf8852011-08-17 17:51:35 -07001519 }
1520
buzbeee1965672012-03-11 18:39:19 -07001521 /* Add ssa reg for Method* */
1522 cUnit->methodSReg = addNewSReg(cUnit, SSA_METHOD_BASEREG);
1523
buzbee67bf8852011-08-17 17:51:35 -07001524 /*
1525 * Allocate the BasicBlockDataFlow structure for the entry and code blocks
1526 */
1527 GrowableListIterator iterator;
1528
1529 oatGrowableListIteratorInit(&cUnit->blockList, &iterator);
1530
1531 while (true) {
1532 BasicBlock* bb = (BasicBlock *) oatGrowableListIteratorNext(&iterator);
1533 if (bb == NULL) break;
1534 if (bb->hidden == true) continue;
1535 if (bb->blockType == kDalvikByteCode ||
1536 bb->blockType == kEntryBlock ||
1537 bb->blockType == kExitBlock) {
1538 bb->dataFlowInfo = (BasicBlockDataFlow *)
buzbeeba938cb2012-02-03 14:47:55 -08001539 oatNew(cUnit, sizeof(BasicBlockDataFlow),
buzbee5abfa3e2012-01-31 17:01:43 -08001540 true, kAllocDFInfo);
buzbee67bf8852011-08-17 17:51:35 -07001541 }
1542 }
1543}
1544
1545/* Clear the visited flag for each BB */
buzbee31a4a6f2012-02-28 15:36:15 -08001546bool oatClearVisitedFlag(struct CompilationUnit* cUnit, struct BasicBlock* bb)
buzbee67bf8852011-08-17 17:51:35 -07001547{
1548 bb->visited = false;
1549 return true;
1550}
1551
1552void oatDataFlowAnalysisDispatcher(CompilationUnit* cUnit,
1553 bool (*func)(CompilationUnit*, BasicBlock*),
1554 DataFlowAnalysisMode dfaMode,
1555 bool isIterative)
1556{
1557 bool change = true;
1558
1559 while (change) {
1560 change = false;
1561
buzbee5b537102012-01-17 17:33:47 -08001562 switch (dfaMode) {
buzbee67bf8852011-08-17 17:51:35 -07001563 /* Scan all blocks and perform the operations specified in func */
buzbee5b537102012-01-17 17:33:47 -08001564 case kAllNodes:
1565 {
1566 GrowableListIterator iterator;
1567 oatGrowableListIteratorInit(&cUnit->blockList, &iterator);
1568 while (true) {
1569 BasicBlock* bb =
1570 (BasicBlock *) oatGrowableListIteratorNext(&iterator);
1571 if (bb == NULL) break;
1572 if (bb->hidden == true) continue;
1573 change |= (*func)(cUnit, bb);
1574 }
buzbee67bf8852011-08-17 17:51:35 -07001575 }
buzbee5b537102012-01-17 17:33:47 -08001576 break;
1577 /* Scan reachable blocks and perform the ops specified in func. */
1578 case kReachableNodes:
1579 {
1580 int numReachableBlocks = cUnit->numReachableBlocks;
1581 int idx;
1582 const GrowableList *blockList = &cUnit->blockList;
buzbee67bf8852011-08-17 17:51:35 -07001583
buzbee5b537102012-01-17 17:33:47 -08001584 for (idx = 0; idx < numReachableBlocks; idx++) {
1585 int blockIdx = cUnit->dfsOrder.elemList[idx];
1586 BasicBlock* bb =
1587 (BasicBlock *) oatGrowableListGetElement(blockList,
1588 blockIdx);
1589 change |= (*func)(cUnit, bb);
1590 }
buzbee67bf8852011-08-17 17:51:35 -07001591 }
buzbee5b537102012-01-17 17:33:47 -08001592 break;
buzbee67bf8852011-08-17 17:51:35 -07001593
buzbee5b537102012-01-17 17:33:47 -08001594 /* Scan reachable blocks by pre-order dfs and invoke func on each. */
1595 case kPreOrderDFSTraversal:
1596 {
1597 int numReachableBlocks = cUnit->numReachableBlocks;
1598 int idx;
1599 const GrowableList *blockList = &cUnit->blockList;
buzbee67bf8852011-08-17 17:51:35 -07001600
buzbee5b537102012-01-17 17:33:47 -08001601 for (idx = 0; idx < numReachableBlocks; idx++) {
1602 int dfsIdx = cUnit->dfsOrder.elemList[idx];
1603 BasicBlock* bb =
1604 (BasicBlock *) oatGrowableListGetElement(blockList,
1605 dfsIdx);
1606 change |= (*func)(cUnit, bb);
1607 }
buzbee67bf8852011-08-17 17:51:35 -07001608 }
buzbee5b537102012-01-17 17:33:47 -08001609 break;
1610 /* Scan reachable blocks post-order dfs and invoke func on each. */
1611 case kPostOrderDFSTraversal:
1612 {
1613 int numReachableBlocks = cUnit->numReachableBlocks;
1614 int idx;
1615 const GrowableList *blockList = &cUnit->blockList;
buzbee67bf8852011-08-17 17:51:35 -07001616
buzbee5b537102012-01-17 17:33:47 -08001617 for (idx = numReachableBlocks - 1; idx >= 0; idx--) {
1618 int dfsIdx = cUnit->dfsOrder.elemList[idx];
1619 BasicBlock* bb =
1620 (BasicBlock *) oatGrowableListGetElement(blockList,
1621 dfsIdx);
1622 change |= (*func)(cUnit, bb);
1623 }
buzbee67bf8852011-08-17 17:51:35 -07001624 }
buzbee5b537102012-01-17 17:33:47 -08001625 break;
1626 /* Scan reachable post-order dom tree and invoke func on each. */
1627 case kPostOrderDOMTraversal:
1628 {
1629 int numReachableBlocks = cUnit->numReachableBlocks;
1630 int idx;
1631 const GrowableList *blockList = &cUnit->blockList;
1632
1633 for (idx = 0; idx < numReachableBlocks; idx++) {
1634 int domIdx = cUnit->domPostOrderTraversal.elemList[idx];
1635 BasicBlock* bb =
1636 (BasicBlock *) oatGrowableListGetElement(blockList,
1637 domIdx);
1638 change |= (*func)(cUnit, bb);
1639 }
1640 }
1641 break;
1642 /* Scan reachable blocks reverse post-order dfs, invoke func on each */
1643 case kReversePostOrderTraversal:
1644 {
1645 int numReachableBlocks = cUnit->numReachableBlocks;
1646 int idx;
1647 const GrowableList *blockList = &cUnit->blockList;
1648
1649 for (idx = numReachableBlocks - 1; idx >= 0; idx--) {
1650 int revIdx = cUnit->dfsPostOrder.elemList[idx];
1651 BasicBlock* bb =
1652 (BasicBlock *) oatGrowableListGetElement(blockList,
1653 revIdx);
1654 change |= (*func)(cUnit, bb);
1655 }
1656 }
1657 break;
1658 default:
1659 LOG(FATAL) << "Unknown traversal mode " << (int)dfaMode;
buzbee67bf8852011-08-17 17:51:35 -07001660 }
1661 /* If isIterative is false, exit the loop after the first iteration */
1662 change &= isIterative;
1663 }
1664}
buzbee43a36422011-09-14 14:00:13 -07001665
buzbeee1965672012-03-11 18:39:19 -07001666/* Advance to next strictly dominated MIR node in an extended basic block */
buzbee239c4e72012-03-16 08:42:29 -07001667MIR* advanceMIR(CompilationUnit* cUnit, BasicBlock** pBb, MIR* mir, ArenaBitVector* bv,
1668 bool clearMark) {
buzbeee1965672012-03-11 18:39:19 -07001669 BasicBlock* bb = *pBb;
1670 if (mir != NULL) {
1671 mir = mir->next;
1672 if (mir == NULL) {
1673 bb = bb->fallThrough;
1674 if ((bb == NULL) || bb->predecessors->numUsed != 1) {
1675 mir = NULL;
1676 } else {
1677 if (bv) {
1678 oatSetBit(cUnit, bv, bb->id);
1679 }
1680 *pBb = bb;
1681 mir = bb->firstMIRInsn;
1682 }
1683 }
1684 }
buzbee239c4e72012-03-16 08:42:29 -07001685 if (mir && clearMark) {
1686 mir->optimizationFlags &= ~MIR_MARK;
1687 }
buzbeee1965672012-03-11 18:39:19 -07001688 return mir;
1689}
1690
buzbeefc9e6fa2012-03-23 15:14:29 -07001691/*
1692 * To be used at an invoke mir. If the logically next mir node represents
1693 * a move-result, return it. Else, return NULL. If a move-result exists,
1694 * it is required to immediately follow the invoke with no intervening
1695 * opcodes or incoming arcs. However, if the result of the invoke is not
1696 * used, a move-result may not be present.
1697 */
1698MIR* oatFindMoveResult(CompilationUnit* cUnit, BasicBlock* bb, MIR* mir,
1699 bool wide)
1700{
1701 BasicBlock* tbb = bb;
1702 mir = advanceMIR(cUnit, &tbb, mir, NULL, false);
1703 while (mir != NULL) {
1704 if (!wide && mir->dalvikInsn.opcode == Instruction::MOVE_RESULT) {
1705 break;
1706 }
1707 if (wide && mir->dalvikInsn.opcode == Instruction::MOVE_RESULT_WIDE) {
1708 break;
1709 }
1710 // Keep going if pseudo op, otherwise terminate
1711 if (mir->dalvikInsn.opcode < static_cast<Instruction::Code>(kNumPackedOpcodes)) {
1712 mir = NULL;
1713 } else {
1714 mir = advanceMIR(cUnit, &tbb, mir, NULL, false);
1715 }
1716 }
1717 return mir;
1718}
1719
buzbee239c4e72012-03-16 08:42:29 -07001720void squashDupRangeChecks(CompilationUnit* cUnit, BasicBlock** pBp, MIR* mir,
1721 int arraySreg, int indexSreg)
1722{
1723 while (true) {
1724 mir = advanceMIR(cUnit, pBp, mir, NULL, false);
1725 if (!mir) {
1726 break;
1727 }
1728 if ((mir->ssaRep == NULL) ||
1729 (mir->optimizationFlags & MIR_IGNORE_RANGE_CHECK)) {
1730 continue;
1731 }
1732 int checkArray = INVALID_SREG;
1733 int checkIndex = INVALID_SREG;
Elliott Hughesb25c3f62012-03-26 16:35:06 -07001734 switch (mir->dalvikInsn.opcode) {
buzbee239c4e72012-03-16 08:42:29 -07001735 case Instruction::AGET:
1736 case Instruction::AGET_OBJECT:
1737 case Instruction::AGET_BOOLEAN:
1738 case Instruction::AGET_BYTE:
1739 case Instruction::AGET_CHAR:
1740 case Instruction::AGET_SHORT:
1741 case Instruction::AGET_WIDE:
1742 checkArray = mir->ssaRep->uses[0];
1743 checkIndex = mir->ssaRep->uses[1];
1744 break;
1745 break;
1746 case Instruction::APUT:
1747 case Instruction::APUT_OBJECT:
1748 case Instruction::APUT_SHORT:
1749 case Instruction::APUT_CHAR:
1750 case Instruction::APUT_BYTE:
1751 case Instruction::APUT_BOOLEAN:
1752 checkArray = mir->ssaRep->uses[1];
1753 checkIndex = mir->ssaRep->uses[2];
1754 break;
1755 case Instruction::APUT_WIDE:
1756 checkArray = mir->ssaRep->uses[2];
1757 checkIndex = mir->ssaRep->uses[3];
1758 default:
1759 break;
1760 }
1761 if (checkArray == INVALID_SREG) {
1762 continue;
1763 }
1764 if ((arraySreg == checkArray) && (indexSreg == checkIndex)) {
1765 if (cUnit->printMe) {
1766 LOG(INFO) << "Squashing range check @ 0x" << std::hex
1767 << mir->offset;
1768 }
1769 mir->optimizationFlags |= MIR_IGNORE_RANGE_CHECK;
1770 }
1771 }
1772}
1773
buzbeee1965672012-03-11 18:39:19 -07001774/* Allocate a compiler temp, return Sreg. Reuse existing if no conflict */
1775int allocCompilerTempSreg(CompilationUnit* cUnit, ArenaBitVector* bv)
1776{
1777 for (int i = 0; i < cUnit->numCompilerTemps; i++) {
1778 CompilerTemp* ct = (CompilerTemp*)cUnit->compilerTemps.elemList[i];
1779 ArenaBitVector* tBv = ct->bv;
1780 if (!oatTestBitVectors(bv, tBv)) {
1781 // Combine live maps and reuse existing temp
1782 oatUnifyBitVectors(tBv, tBv, bv);
1783 return ct->sReg;
1784 }
1785 }
1786
1787 // Create a new compiler temp & associated live bitmap
1788 CompilerTemp* ct = (CompilerTemp*)oatNew(cUnit, sizeof(CompilerTemp),
1789 true, kAllocMisc);
1790 ArenaBitVector *nBv = oatAllocBitVector(cUnit, cUnit->numBlocks, true,
1791 kBitMapMisc);
1792 oatCopyBitVector(nBv, bv);
1793 ct->bv = nBv;
1794 ct->sReg = addNewSReg(cUnit, SSA_CTEMP_BASEREG - cUnit->numCompilerTemps);
1795 cUnit->numCompilerTemps++;
1796 oatInsertGrowableList(cUnit, &cUnit->compilerTemps, (intptr_t)ct);
1797 DCHECK_EQ(cUnit->numCompilerTemps, (int)cUnit->compilerTemps.numUsed);
1798 return ct->sReg;
1799}
1800
1801/* Creata a new MIR node for a new pseudo op. */
1802MIR* rawMIR(CompilationUnit* cUnit, Instruction::Code opcode, int defs, int uses)
1803{
1804 MIR* res = (MIR*)oatNew( cUnit, sizeof(MIR), true, kAllocMIR);
1805 res->ssaRep =(struct SSARepresentation *)
1806 oatNew(cUnit, sizeof(SSARepresentation), true, kAllocDFInfo);
1807 if (uses) {
1808 res->ssaRep->numUses = uses;
1809 res->ssaRep->uses = (int*)oatNew(cUnit, sizeof(int) * uses, false, kAllocDFInfo);
1810 }
1811 if (defs) {
1812 res->ssaRep->numDefs = defs;
1813 res->ssaRep->defs = (int*)oatNew(cUnit, sizeof(int) * defs, false, kAllocDFInfo);
1814 res->ssaRep->fpDef = (bool*)oatNew(cUnit, sizeof(bool) * defs, true, kAllocDFInfo);
1815 }
1816 res->dalvikInsn.opcode = opcode;
1817 return res;
1818}
1819
1820/* Do some MIR-level basic block optimizations */
1821bool basicBlockOpt(CompilationUnit* cUnit, BasicBlock* bb)
1822{
1823 int numTemps = 0;
1824
1825 for (MIR* mir = bb->firstMIRInsn; mir; mir = mir->next) {
1826 // Look for interesting opcodes, skip otherwise
buzbee239c4e72012-03-16 08:42:29 -07001827 Instruction::Code opcode = mir->dalvikInsn.opcode;
Elliott Hughesb25c3f62012-03-26 16:35:06 -07001828 switch (opcode) {
buzbee239c4e72012-03-16 08:42:29 -07001829 case Instruction::AGET:
1830 case Instruction::AGET_OBJECT:
1831 case Instruction::AGET_BOOLEAN:
1832 case Instruction::AGET_BYTE:
1833 case Instruction::AGET_CHAR:
1834 case Instruction::AGET_SHORT:
1835 case Instruction::AGET_WIDE:
1836 if (!(mir->optimizationFlags & MIR_IGNORE_RANGE_CHECK)) {
1837 int arrSreg = mir->ssaRep->uses[0];
1838 int idxSreg = mir->ssaRep->uses[1];
1839 BasicBlock* tbb = bb;
1840 squashDupRangeChecks(cUnit, &tbb, mir, arrSreg, idxSreg);
1841 }
1842 break;
1843 case Instruction::APUT:
1844 case Instruction::APUT_OBJECT:
1845 case Instruction::APUT_SHORT:
1846 case Instruction::APUT_CHAR:
1847 case Instruction::APUT_BYTE:
1848 case Instruction::APUT_BOOLEAN:
1849 case Instruction::APUT_WIDE:
1850 if (!(mir->optimizationFlags & MIR_IGNORE_RANGE_CHECK)) {
1851 int start = (opcode == Instruction::APUT_WIDE) ? 2 : 1;
1852 int arrSreg = mir->ssaRep->uses[start];
1853 int idxSreg = mir->ssaRep->uses[start + 1];
1854 BasicBlock* tbb = bb;
1855 squashDupRangeChecks(cUnit, &tbb, mir, arrSreg, idxSreg);
1856 }
1857 break;
buzbee239c4e72012-03-16 08:42:29 -07001858 case Instruction::CMPL_FLOAT:
1859 case Instruction::CMPL_DOUBLE:
1860 case Instruction::CMPG_FLOAT:
1861 case Instruction::CMPG_DOUBLE:
1862 case Instruction::CMP_LONG:
buzbee84fd6932012-03-29 16:44:16 -07001863 if (mir->next != NULL) {
1864 MIR* mirNext = mir->next;
1865 Instruction::Code brOpcode = mirNext->dalvikInsn.opcode;
1866 ConditionCode ccode = kCondNv;
1867 switch(brOpcode) {
1868 case Instruction::IF_EQZ:
1869 ccode = kCondEq;
1870 break;
1871 case Instruction::IF_NEZ:
1872 // ccode = kCondNe;
1873 break;
1874 case Instruction::IF_LTZ:
1875 // ccode = kCondLt;
1876 break;
1877 case Instruction::IF_GEZ:
1878 // ccode = kCondGe;
1879 break;
1880 case Instruction::IF_GTZ:
1881 // ccode = kCondGt;
1882 break;
1883 case Instruction::IF_LEZ:
1884 // ccode = kCondLe;
1885 break;
1886 default:
1887 break;
1888 }
1889 // Make sure result of cmp is used by next insn and nowhere else
1890 if ((ccode != kCondNv) &&
1891 (mir->ssaRep->defs[0] == mirNext->ssaRep->uses[0]) &&
1892 (getSSAUseCount(cUnit, mir->ssaRep->defs[0]) == 1)) {
1893 mirNext->dalvikInsn.arg[0] = ccode;
1894 switch(opcode) {
1895 case Instruction::CMPL_FLOAT:
1896 mirNext->dalvikInsn.opcode =
1897 static_cast<Instruction::Code>(kMirOpFusedCmplFloat);
1898 break;
1899 case Instruction::CMPL_DOUBLE:
1900 mirNext->dalvikInsn.opcode =
1901 static_cast<Instruction::Code>(kMirOpFusedCmplDouble);
1902 break;
1903 case Instruction::CMPG_FLOAT:
1904 mirNext->dalvikInsn.opcode =
1905 static_cast<Instruction::Code>(kMirOpFusedCmpgFloat);
1906 break;
1907 case Instruction::CMPG_DOUBLE:
1908 mirNext->dalvikInsn.opcode =
1909 static_cast<Instruction::Code>(kMirOpFusedCmpgDouble);
1910 break;
1911 case Instruction::CMP_LONG:
1912 mirNext->dalvikInsn.opcode =
1913 static_cast<Instruction::Code>(kMirOpFusedCmpLong);
1914 break;
1915 default: LOG(ERROR) << "Unexpected opcode: " << (int)opcode;
1916 }
1917 mir->dalvikInsn.opcode = static_cast<Instruction::Code>(kMirOpNop);
1918 mirNext->ssaRep->numUses = mir->ssaRep->numUses;
1919 mirNext->ssaRep->uses = mir->ssaRep->uses;
1920 mirNext->ssaRep->fpUse = mir->ssaRep->fpUse;
1921 mirNext->ssaRep->numDefs = 0;
1922 mir->ssaRep->numUses = 0;
1923 mir->ssaRep->numDefs = 0;
1924 }
1925 }
buzbeee1965672012-03-11 18:39:19 -07001926 break;
1927 default:
1928 break;
1929 }
1930 }
1931
1932 if (numTemps > cUnit->numCompilerTemps) {
1933 cUnit->numCompilerTemps = numTemps;
1934 }
1935 return true;
1936}
1937
buzbee31a4a6f2012-02-28 15:36:15 -08001938bool nullCheckEliminationInit(struct CompilationUnit* cUnit,
1939 struct BasicBlock* bb)
buzbee43a36422011-09-14 14:00:13 -07001940{
1941 if (bb->dataFlowInfo == NULL) return false;
1942 bb->dataFlowInfo->endingNullCheckV =
buzbeeba938cb2012-02-03 14:47:55 -08001943 oatAllocBitVector(cUnit, cUnit->numSSARegs, false, kBitMapNullCheck);
buzbee43a36422011-09-14 14:00:13 -07001944 oatClearAllBits(bb->dataFlowInfo->endingNullCheckV);
1945 return true;
1946}
1947
1948/* Eliminate unnecessary null checks for a basic block. */
buzbee31a4a6f2012-02-28 15:36:15 -08001949bool eliminateNullChecks( struct CompilationUnit* cUnit, struct BasicBlock* bb)
buzbee43a36422011-09-14 14:00:13 -07001950{
1951 if (bb->dataFlowInfo == NULL) return false;
buzbee5abfa3e2012-01-31 17:01:43 -08001952
buzbee43a36422011-09-14 14:00:13 -07001953 /*
1954 * Set initial state. Be conservative with catch
1955 * blocks and start with no assumptions about null check
1956 * status (except for "this").
1957 */
buzbee43a36422011-09-14 14:00:13 -07001958 if ((bb->blockType == kEntryBlock) | bb->catchEntry) {
1959 oatClearAllBits(cUnit->tempSSARegisterV);
Elliott Hughes11d1b0c2012-01-23 16:57:47 -08001960 if ((cUnit->access_flags & kAccStatic) == 0) {
buzbee43a36422011-09-14 14:00:13 -07001961 // If non-static method, mark "this" as non-null
Ian Rogersa3760aa2011-11-14 14:32:37 -08001962 int thisReg = cUnit->numDalvikRegisters - cUnit->numIns;
buzbeeba938cb2012-02-03 14:47:55 -08001963 oatSetBit(cUnit, cUnit->tempSSARegisterV, thisReg);
buzbee43a36422011-09-14 14:00:13 -07001964 }
1965 } else {
1966 // Starting state is intesection of all incoming arcs
buzbee5abfa3e2012-01-31 17:01:43 -08001967 GrowableListIterator iter;
1968 oatGrowableListIteratorInit(bb->predecessors, &iter);
1969 BasicBlock* predBB = (BasicBlock*)oatGrowableListIteratorNext(&iter);
1970 DCHECK(predBB != NULL);
buzbee43a36422011-09-14 14:00:13 -07001971 oatCopyBitVector(cUnit->tempSSARegisterV,
1972 predBB->dataFlowInfo->endingNullCheckV);
1973 while (true) {
buzbee5abfa3e2012-01-31 17:01:43 -08001974 predBB = (BasicBlock*)oatGrowableListIteratorNext(&iter);
1975 if (!predBB) break;
buzbeeaad72012011-09-21 21:52:09 -07001976 if ((predBB->dataFlowInfo == NULL) ||
1977 (predBB->dataFlowInfo->endingNullCheckV == NULL)) {
1978 continue;
1979 }
buzbee43a36422011-09-14 14:00:13 -07001980 oatIntersectBitVectors(cUnit->tempSSARegisterV,
1981 cUnit->tempSSARegisterV,
1982 predBB->dataFlowInfo->endingNullCheckV);
1983 }
1984 }
1985
1986 // Walk through the instruction in the block, updating as necessary
1987 for (MIR* mir = bb->firstMIRInsn; mir; mir = mir->next) {
1988 if (mir->ssaRep == NULL) {
1989 continue;
1990 }
1991 int dfAttributes =
1992 oatDataFlowAttributes[mir->dalvikInsn.opcode];
1993
1994 // Mark target of NEW* as non-null
1995 if (dfAttributes & DF_NON_NULL_DST) {
buzbeeba938cb2012-02-03 14:47:55 -08001996 oatSetBit(cUnit, cUnit->tempSSARegisterV, mir->ssaRep->defs[0]);
buzbee43a36422011-09-14 14:00:13 -07001997 }
1998
1999 // Mark non-null returns from invoke-style NEW*
2000 if (dfAttributes & DF_NON_NULL_RET) {
2001 MIR* nextMir = mir->next;
Elliott Hughesadb8c672012-03-06 16:49:32 -08002002 // Next should be an MOVE_RESULT_OBJECT
2003 if (nextMir && nextMir->dalvikInsn.opcode == Instruction::MOVE_RESULT_OBJECT) {
buzbee43a36422011-09-14 14:00:13 -07002004 // Mark as null checked
buzbeeba938cb2012-02-03 14:47:55 -08002005 oatSetBit(cUnit, cUnit->tempSSARegisterV,
2006 nextMir->ssaRep->defs[0]);
buzbee43a36422011-09-14 14:00:13 -07002007 } else {
2008 if (nextMir) {
2009 LOG(WARNING) << "Unexpected opcode following new: " <<
2010 (int)nextMir->dalvikInsn.opcode;
buzbee949f56e2011-10-06 11:05:45 -07002011 } else if (bb->fallThrough) {
2012 // Look in next basic block
2013 struct BasicBlock* nextBB = bb->fallThrough;
2014 for (MIR* tmir = nextBB->firstMIRInsn; tmir;
Elliott Hughesb25c3f62012-03-26 16:35:06 -07002015 tmir =tmir->next) {
buzbee949f56e2011-10-06 11:05:45 -07002016 if ((int)tmir->dalvikInsn.opcode >= (int)kMirOpFirst) {
2017 continue;
2018 }
Elliott Hughesadb8c672012-03-06 16:49:32 -08002019 // First non-pseudo should be MOVE_RESULT_OBJECT
2020 if (tmir->dalvikInsn.opcode == Instruction::MOVE_RESULT_OBJECT) {
buzbee949f56e2011-10-06 11:05:45 -07002021 // Mark as null checked
buzbeeba938cb2012-02-03 14:47:55 -08002022 oatSetBit(cUnit, cUnit->tempSSARegisterV,
buzbee949f56e2011-10-06 11:05:45 -07002023 tmir->ssaRep->defs[0]);
2024 } else {
2025 LOG(WARNING) << "Unexpected op after new: " <<
2026 (int)tmir->dalvikInsn.opcode;
2027 }
2028 break;
2029 }
buzbee43a36422011-09-14 14:00:13 -07002030 }
2031 }
2032 }
2033
2034 /*
2035 * Propagate nullcheck state on register copies (including
2036 * Phi pseudo copies. For the latter, nullcheck state is
2037 * the "and" of all the Phi's operands.
2038 */
2039 if (dfAttributes & (DF_NULL_TRANSFER_0 | DF_NULL_TRANSFER_N)) {
2040 int tgtSreg = mir->ssaRep->defs[0];
2041 int operands = (dfAttributes & DF_NULL_TRANSFER_0) ? 1 :
2042 mir->ssaRep->numUses;
2043 bool nullChecked = true;
2044 for (int i = 0; i < operands; i++) {
2045 nullChecked &= oatIsBitSet(cUnit->tempSSARegisterV,
2046 mir->ssaRep->uses[i]);
2047 }
2048 if (nullChecked) {
buzbeeba938cb2012-02-03 14:47:55 -08002049 oatSetBit(cUnit, cUnit->tempSSARegisterV, tgtSreg);
buzbee43a36422011-09-14 14:00:13 -07002050 }
2051 }
2052
2053 // Already nullchecked?
2054 if (dfAttributes & DF_HAS_NULL_CHKS) {
buzbee239c4e72012-03-16 08:42:29 -07002055 int srcIdx;
2056 if (dfAttributes & DF_NULL_CHK_1) {
2057 srcIdx = 1;
2058 } else if (dfAttributes & DF_NULL_CHK_2) {
2059 srcIdx = 2;
2060 } else {
2061 srcIdx = 0;
2062 }
2063 int srcSreg = mir->ssaRep->uses[srcIdx];
buzbee43a36422011-09-14 14:00:13 -07002064 if (oatIsBitSet(cUnit->tempSSARegisterV, srcSreg)) {
2065 // Eliminate the null check
2066 mir->optimizationFlags |= MIR_IGNORE_NULL_CHECK;
2067 } else {
2068 // Mark sReg as null-checked
buzbeeba938cb2012-02-03 14:47:55 -08002069 oatSetBit(cUnit, cUnit->tempSSARegisterV, srcSreg);
buzbee43a36422011-09-14 14:00:13 -07002070 }
2071 }
2072 }
2073
2074 // Did anything change?
2075 bool res = oatCompareBitVectors(bb->dataFlowInfo->endingNullCheckV,
2076 cUnit->tempSSARegisterV);
2077 if (res) {
2078 oatCopyBitVector(bb->dataFlowInfo->endingNullCheckV,
2079 cUnit->tempSSARegisterV);
2080 }
2081 return res;
2082}
2083
2084void oatMethodNullCheckElimination(CompilationUnit *cUnit)
2085{
2086 if (!(cUnit->disableOpt & (1 << kNullCheckElimination))) {
2087 DCHECK(cUnit->tempSSARegisterV != NULL);
2088 oatDataFlowAnalysisDispatcher(cUnit, nullCheckEliminationInit,
2089 kAllNodes,
2090 false /* isIterative */);
2091 oatDataFlowAnalysisDispatcher(cUnit, eliminateNullChecks,
2092 kPreOrderDFSTraversal,
2093 true /* isIterative */);
2094 }
2095}
Elliott Hughes11d1b0c2012-01-23 16:57:47 -08002096
buzbeee1965672012-03-11 18:39:19 -07002097void oatMethodBasicBlockOptimization(CompilationUnit *cUnit)
2098{
buzbeee1965672012-03-11 18:39:19 -07002099 if (!(cUnit->disableOpt & (1 << kBBOpt))) {
buzbee239c4e72012-03-16 08:42:29 -07002100 oatInitGrowableList(cUnit, &cUnit->compilerTemps, 6, kListMisc);
2101 DCHECK_EQ(cUnit->numCompilerTemps, 0);
2102 if (!(cUnit->disableOpt & (1 << kBBOpt))) {
2103 oatDataFlowAnalysisDispatcher(cUnit, basicBlockOpt,
2104 kAllNodes, false /* isIterative */);
2105 }
buzbeee1965672012-03-11 18:39:19 -07002106 }
2107}
2108
buzbee239c4e72012-03-16 08:42:29 -07002109void addLoopHeader(CompilationUnit* cUnit, BasicBlock* header,
2110 BasicBlock* backEdge)
2111{
2112 GrowableListIterator iter;
2113 oatGrowableListIteratorInit(&cUnit->loopHeaders, &iter);
2114 for (LoopInfo* loop = (LoopInfo*)oatGrowableListIteratorNext(&iter);
2115 (loop != NULL); loop = (LoopInfo*)oatGrowableListIteratorNext(&iter)) {
2116 if (loop->header == header) {
2117 oatInsertGrowableList(cUnit, &loop->incomingBackEdges,
2118 (intptr_t)backEdge);
2119 return;
2120 }
2121 }
2122 LoopInfo* info = (LoopInfo*)oatNew(cUnit, sizeof(LoopInfo), true,
2123 kAllocDFInfo);
2124 info->header = header;
2125 oatInitGrowableList(cUnit, &info->incomingBackEdges, 2, kListMisc);
2126 oatInsertGrowableList(cUnit, &info->incomingBackEdges, (intptr_t)backEdge);
2127 oatInsertGrowableList(cUnit, &cUnit->loopHeaders, (intptr_t)info);
2128}
2129
2130bool findBackEdges(struct CompilationUnit* cUnit, struct BasicBlock* bb)
2131{
2132 if ((bb->dataFlowInfo == NULL) || (bb->lastMIRInsn == NULL)) {
2133 return false;
2134 }
2135 Instruction::Code opcode = bb->lastMIRInsn->dalvikInsn.opcode;
2136 if (Instruction::Flags(opcode) & Instruction::kBranch) {
2137 if (bb->taken && (bb->taken->startOffset <= bb->startOffset)) {
2138 DCHECK(bb->dominators != NULL);
2139 if (oatIsBitSet(bb->dominators, bb->taken->id)) {
2140 if (cUnit->printMe) {
2141 LOG(INFO) << "Loop backedge from 0x"
2142 << std::hex << bb->lastMIRInsn->offset
2143 << " to 0x" << std::hex << bb->taken->startOffset;
2144 }
2145 addLoopHeader(cUnit, bb->taken, bb);
2146 }
2147 }
2148 }
2149 return false;
2150}
2151
2152void addBlocksToLoop(CompilationUnit* cUnit, ArenaBitVector* blocks,
2153 BasicBlock* bb, int headId)
2154{
2155 if (!oatIsBitSet(bb->dominators, headId) ||
2156 oatIsBitSet(blocks, bb->id)) {
2157 return;
2158 }
2159 oatSetBit(cUnit, blocks, bb->id);
2160 GrowableListIterator iter;
2161 oatGrowableListIteratorInit(bb->predecessors, &iter);
2162 BasicBlock* predBB;
2163 for (predBB = (BasicBlock*)oatGrowableListIteratorNext(&iter); predBB;
2164 predBB = (BasicBlock*)oatGrowableListIteratorNext(&iter)) {
2165 addBlocksToLoop(cUnit, blocks, predBB, headId);
2166 }
2167}
2168
2169void oatDumpLoops(CompilationUnit *cUnit)
2170{
2171 GrowableListIterator iter;
2172 oatGrowableListIteratorInit(&cUnit->loopHeaders, &iter);
2173 for (LoopInfo* loop = (LoopInfo*)oatGrowableListIteratorNext(&iter);
2174 (loop != NULL); loop = (LoopInfo*)oatGrowableListIteratorNext(&iter)) {
2175 LOG(INFO) << "Loop head block id " << loop->header->id
2176 << ", offset 0x" << std::hex << loop->header->startOffset
2177 << ", Depth: " << loop->header->nestingDepth;
2178 GrowableListIterator iter;
2179 oatGrowableListIteratorInit(&loop->incomingBackEdges, &iter);
2180 BasicBlock* edgeBB;
2181 for (edgeBB = (BasicBlock*)oatGrowableListIteratorNext(&iter); edgeBB;
2182 edgeBB = (BasicBlock*)oatGrowableListIteratorNext(&iter)) {
2183 LOG(INFO) << " Backedge block id " << edgeBB->id
2184 << ", offset 0x" << std::hex << edgeBB->startOffset;
2185 ArenaBitVectorIterator bIter;
2186 oatBitVectorIteratorInit(loop->blocks, &bIter);
2187 for (int bbId = oatBitVectorIteratorNext(&bIter); bbId != -1;
2188 bbId = oatBitVectorIteratorNext(&bIter)) {
2189 BasicBlock *bb;
2190 bb = (BasicBlock*)
2191 oatGrowableListGetElement(&cUnit->blockList, bbId);
2192 LOG(INFO) << " (" << bb->id << ", 0x" << std::hex
2193 << bb->startOffset << ")";
2194 }
2195 }
2196 }
2197}
2198
2199void oatMethodLoopDetection(CompilationUnit *cUnit)
2200{
2201 if (cUnit->disableOpt & (1 << kPromoteRegs)) {
2202 return;
2203 }
2204 oatInitGrowableList(cUnit, &cUnit->loopHeaders, 6, kListMisc);
2205 // Find the loop headers
2206 oatDataFlowAnalysisDispatcher(cUnit, findBackEdges,
2207 kAllNodes, false /* isIterative */);
2208 GrowableListIterator iter;
2209 oatGrowableListIteratorInit(&cUnit->loopHeaders, &iter);
2210 // Add blocks to each header
2211 for (LoopInfo* loop = (LoopInfo*)oatGrowableListIteratorNext(&iter);
2212 loop; loop = (LoopInfo*)oatGrowableListIteratorNext(&iter)) {
2213 loop->blocks = oatAllocBitVector(cUnit, cUnit->numBlocks, true,
2214 kBitMapMisc);
2215 oatSetBit(cUnit, loop->blocks, loop->header->id);
2216 GrowableListIterator iter;
2217 oatGrowableListIteratorInit(&loop->incomingBackEdges, &iter);
2218 BasicBlock* edgeBB;
2219 for (edgeBB = (BasicBlock*)oatGrowableListIteratorNext(&iter); edgeBB;
2220 edgeBB = (BasicBlock*)oatGrowableListIteratorNext(&iter)) {
2221 addBlocksToLoop(cUnit, loop->blocks, edgeBB, loop->header->id);
2222 }
2223 }
2224 // Compute the nesting depth of each header
2225 oatGrowableListIteratorInit(&cUnit->loopHeaders, &iter);
2226 for (LoopInfo* loop = (LoopInfo*)oatGrowableListIteratorNext(&iter);
2227 loop; loop = (LoopInfo*)oatGrowableListIteratorNext(&iter)) {
2228 GrowableListIterator iter2;
2229 oatGrowableListIteratorInit(&cUnit->loopHeaders, &iter2);
2230 LoopInfo* loop2;
2231 for (loop2 = (LoopInfo*)oatGrowableListIteratorNext(&iter2);
2232 loop2; loop2 = (LoopInfo*)oatGrowableListIteratorNext(&iter2)) {
2233 if (oatIsBitSet(loop2->blocks, loop->header->id)) {
2234 loop->header->nestingDepth++;
2235 }
2236 }
2237 }
2238 // Assign nesting depth to each block in all loops
2239 oatGrowableListIteratorInit(&cUnit->loopHeaders, &iter);
2240 for (LoopInfo* loop = (LoopInfo*)oatGrowableListIteratorNext(&iter);
2241 (loop != NULL); loop = (LoopInfo*)oatGrowableListIteratorNext(&iter)) {
2242 ArenaBitVectorIterator bIter;
2243 oatBitVectorIteratorInit(loop->blocks, &bIter);
2244 for (int bbId = oatBitVectorIteratorNext(&bIter); bbId != -1;
2245 bbId = oatBitVectorIteratorNext(&bIter)) {
2246 BasicBlock *bb;
2247 bb = (BasicBlock*) oatGrowableListGetElement(&cUnit->blockList,
2248 bbId);
2249 bb->nestingDepth = std::max(bb->nestingDepth,
2250 loop->header->nestingDepth);
2251 }
2252 }
2253 if (cUnit->printMe) {
2254 oatDumpLoops(cUnit);
2255 }
2256}
2257
2258/*
buzbee9c044ce2012-03-18 13:24:07 -07002259 * This function will make a best guess at whether the invoke will
2260 * end up using Method*. It isn't critical to get it exactly right,
2261 * and attempting to do would involve more complexity than it's
2262 * worth.
2263 */
2264bool invokeUsesMethodStar(CompilationUnit* cUnit, MIR* mir)
2265{
2266 InvokeType type;
2267 Instruction::Code opcode = mir->dalvikInsn.opcode;
Elliott Hughesb25c3f62012-03-26 16:35:06 -07002268 switch (opcode) {
buzbee9c044ce2012-03-18 13:24:07 -07002269 case Instruction::INVOKE_STATIC:
2270 case Instruction::INVOKE_STATIC_RANGE:
2271 type = kStatic;
2272 break;
2273 case Instruction::INVOKE_DIRECT:
2274 case Instruction::INVOKE_DIRECT_RANGE:
2275 type = kDirect;
2276 break;
2277 case Instruction::INVOKE_VIRTUAL:
2278 case Instruction::INVOKE_VIRTUAL_RANGE:
2279 type = kVirtual;
2280 break;
2281 case Instruction::INVOKE_INTERFACE:
2282 case Instruction::INVOKE_INTERFACE_RANGE:
2283 return false;
2284 case Instruction::INVOKE_SUPER_RANGE:
2285 case Instruction::INVOKE_SUPER:
2286 type = kSuper;
2287 break;
2288 default:
2289 LOG(WARNING) << "Unexpected invoke op: " << (int)opcode;
2290 return false;
2291 }
2292 OatCompilationUnit mUnit(cUnit->class_loader, cUnit->class_linker,
2293 *cUnit->dex_file, *cUnit->dex_cache,
2294 cUnit->code_item, cUnit->method_idx,
2295 cUnit->access_flags);
2296 // TODO: add a flag so we don't counts the stats for this twice
2297 uint32_t dexMethodIdx = mir->dalvikInsn.vB;
2298 int vtableIdx;
2299 uintptr_t directCode;
2300 uintptr_t directMethod;
2301 bool fastPath =
2302 cUnit->compiler->ComputeInvokeInfo(dexMethodIdx, &mUnit, type,
2303 vtableIdx, directCode,
2304 directMethod) &&
2305 !SLOW_INVOKE_PATH;
2306 return (((type == kDirect) || (type == kStatic)) &&
2307 fastPath && ((directCode == 0) || (directMethod == 0)));
2308}
2309
2310/*
buzbee239c4e72012-03-16 08:42:29 -07002311 * Count uses, weighting by loop nesting depth. This code only
2312 * counts explicitly used sRegs. A later phase will add implicit
2313 * counts for things such as Method*, null-checked references, etc.
2314 */
2315bool countUses(struct CompilationUnit* cUnit, struct BasicBlock* bb)
2316{
2317 if (bb->blockType != kDalvikByteCode) {
2318 return false;
2319 }
2320 for (MIR* mir = bb->firstMIRInsn; (mir != NULL); mir = mir->next) {
2321 if (mir->ssaRep == NULL) {
2322 continue;
2323 }
buzbee9c044ce2012-03-18 13:24:07 -07002324 uint32_t weight = std::min(16U, (uint32_t)bb->nestingDepth);
buzbee239c4e72012-03-16 08:42:29 -07002325 for (int i = 0; i < mir->ssaRep->numUses; i++) {
2326 int sReg = mir->ssaRep->uses[i];
buzbee239c4e72012-03-16 08:42:29 -07002327 DCHECK_LT(sReg, (int)cUnit->useCounts.numUsed);
buzbee84fd6932012-03-29 16:44:16 -07002328 cUnit->rawUseCounts.elemList[sReg]++;
buzbee239c4e72012-03-16 08:42:29 -07002329 cUnit->useCounts.elemList[sReg] += (1 << weight);
2330 }
buzbee9c044ce2012-03-18 13:24:07 -07002331 if (!(cUnit->disableOpt & (1 << kPromoteCompilerTemps))) {
2332 int dfAttributes = oatDataFlowAttributes[mir->dalvikInsn.opcode];
2333 // Implicit use of Method* ? */
2334 if (dfAttributes & DF_UMS) {
2335 /*
2336 * Some invokes will not use Method* - need to perform test similar
2337 * to that found in genInvoke() to decide whether to count refs
2338 * for Method* on invoke-class opcodes.
2339 * TODO: refactor for common test here, save results for genInvoke
2340 */
2341 int usesMethodStar = true;
2342 if ((dfAttributes & (DF_FORMAT_35C | DF_FORMAT_3RC)) &&
2343 !(dfAttributes & DF_NON_NULL_RET)) {
2344 usesMethodStar &= invokeUsesMethodStar(cUnit, mir);
2345 }
2346 if (usesMethodStar) {
buzbee84fd6932012-03-29 16:44:16 -07002347 cUnit->rawUseCounts.elemList[cUnit->methodSReg]++;
buzbee9c044ce2012-03-18 13:24:07 -07002348 cUnit->useCounts.elemList[cUnit->methodSReg] += (1 << weight);
2349 }
2350 }
2351 }
buzbee239c4e72012-03-16 08:42:29 -07002352 }
2353 return false;
2354}
2355
2356void oatMethodUseCount(CompilationUnit *cUnit)
2357{
buzbee239c4e72012-03-16 08:42:29 -07002358 oatInitGrowableList(cUnit, &cUnit->useCounts, cUnit->numSSARegs + 32,
2359 kListMisc);
buzbee84fd6932012-03-29 16:44:16 -07002360 oatInitGrowableList(cUnit, &cUnit->rawUseCounts, cUnit->numSSARegs + 32,
2361 kListMisc);
buzbee239c4e72012-03-16 08:42:29 -07002362 // Initialize list
2363 for (int i = 0; i < cUnit->numSSARegs; i++) {
2364 oatInsertGrowableList(cUnit, &cUnit->useCounts, 0);
buzbee84fd6932012-03-29 16:44:16 -07002365 oatInsertGrowableList(cUnit, &cUnit->rawUseCounts, 0);
2366 }
2367 if (cUnit->disableOpt & (1 << kPromoteRegs)) {
2368 return;
buzbee239c4e72012-03-16 08:42:29 -07002369 }
2370 oatDataFlowAnalysisDispatcher(cUnit, countUses,
2371 kAllNodes, false /* isIterative */);
2372}
2373
Elliott Hughes11d1b0c2012-01-23 16:57:47 -08002374} // namespace art