blob: ad522bf482673c0851271d1bc611df87b8642759 [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
buzbee67bc2362011-10-11 18:08:40 -0700126 DF_UA | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -0700127
Elliott Hughesadb8c672012-03-06 16:49:32 -0800128 // 20 INSTANCE_OF vA, vB, type@CCCC
buzbee67bc2362011-10-11 18:08:40 -0700129 DF_DA | DF_UB | DF_CORE_A | DF_CORE_B,
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
buzbee67bc2362011-10-11 18:08:40 -0700135 DF_DA | DF_NON_NULL_DST | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -0700136
Elliott Hughesadb8c672012-03-06 16:49:32 -0800137 // 23 NEW_ARRAY vA, vB, type@CCCC
buzbee67bc2362011-10-11 18:08:40 -0700138 DF_DA | DF_UB | DF_NON_NULL_DST | DF_CORE_A | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700139
Elliott Hughesadb8c672012-03-06 16:49:32 -0800140 // 24 FILLED_NEW_ARRAY {vD, vE, vF, vG, vA}
buzbee43a36422011-09-14 14:00:13 -0700141 DF_FORMAT_35C | DF_NON_NULL_RET,
buzbee67bf8852011-08-17 17:51:35 -0700142
Elliott Hughesadb8c672012-03-06 16:49:32 -0800143 // 25 FILLED_NEW_ARRAY_RANGE {vCCCC .. vNNNN}, type@BBBB
buzbee43a36422011-09-14 14:00:13 -0700144 DF_FORMAT_3RC | DF_NON_NULL_RET,
buzbee67bf8852011-08-17 17:51:35 -0700145
Elliott Hughesadb8c672012-03-06 16:49:32 -0800146 // 26 FILL_ARRAY_DATA vAA, +BBBBBBBB
buzbee67bc2362011-10-11 18:08:40 -0700147 DF_UA | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -0700148
Elliott Hughesadb8c672012-03-06 16:49:32 -0800149 // 27 THROW vAA
buzbee67bc2362011-10-11 18:08:40 -0700150 DF_UA | DF_CORE_A,
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
buzbee67bc2362011-10-11 18:08:40 -0700238 DF_DA | DF_UB | DF_UC | DF_NULL_CHK_0 | DF_RANGE_CHK_1 | DF_IS_GETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700241 DF_DA_WIDE | DF_UB | DF_UC | DF_NULL_CHK_0 | DF_RANGE_CHK_1 | DF_IS_GETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700244 DF_DA | DF_UB | DF_UC | DF_NULL_CHK_0 | DF_RANGE_CHK_1 | DF_IS_GETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700247 DF_DA | DF_UB | DF_UC | DF_NULL_CHK_0 | DF_RANGE_CHK_1 | DF_IS_GETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700250 DF_DA | DF_UB | DF_UC | DF_NULL_CHK_0 | DF_RANGE_CHK_1 | DF_IS_GETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700253 DF_DA | DF_UB | DF_UC | DF_NULL_CHK_0 | DF_RANGE_CHK_1 | DF_IS_GETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700256 DF_DA | DF_UB | DF_UC | DF_NULL_CHK_0 | DF_RANGE_CHK_1 | DF_IS_GETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700259 DF_UA | DF_UB | DF_UC | DF_NULL_CHK_1 | DF_RANGE_CHK_2 | DF_IS_SETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700262 DF_UA_WIDE | DF_UB | DF_UC | DF_NULL_CHK_1 | DF_RANGE_CHK_2 | DF_IS_SETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700265 DF_UA | DF_UB | DF_UC | DF_NULL_CHK_1 | DF_RANGE_CHK_2 | DF_IS_SETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700268 DF_UA | DF_UB | DF_UC | DF_NULL_CHK_1 | DF_RANGE_CHK_2 | DF_IS_SETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700271 DF_UA | DF_UB | DF_UC | DF_NULL_CHK_1 | DF_RANGE_CHK_2 | DF_IS_SETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700274 DF_UA | DF_UB | DF_UC | DF_NULL_CHK_1 | DF_RANGE_CHK_2 | DF_IS_SETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700277 DF_UA | DF_UB | DF_UC | DF_NULL_CHK_1 | DF_RANGE_CHK_2 | DF_IS_SETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700280 DF_DA | DF_UB | DF_NULL_CHK_0 | DF_IS_GETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700283 DF_DA_WIDE | DF_UB | DF_NULL_CHK_0 | DF_IS_GETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700286 DF_DA | DF_UB | DF_NULL_CHK_0 | DF_IS_GETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700289 DF_DA | DF_UB | DF_NULL_CHK_0 | DF_IS_GETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700292 DF_DA | DF_UB | DF_NULL_CHK_0 | DF_IS_GETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700295 DF_DA | DF_UB | DF_NULL_CHK_0 | DF_IS_GETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700298 DF_DA | DF_UB | DF_NULL_CHK_0 | DF_IS_GETTER | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700299
Elliott Hughesadb8c672012-03-06 16:49:32 -0800300 // 59 IPUT vA, vB, field@CCCC
buzbee67bc2362011-10-11 18:08:40 -0700301 DF_UA | DF_UB | DF_NULL_CHK_1 | DF_IS_SETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700304 DF_UA_WIDE | DF_UB | DF_NULL_CHK_1 | DF_IS_SETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700307 DF_UA | DF_UB | DF_NULL_CHK_1 | DF_IS_SETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700310 DF_UA | DF_UB | DF_NULL_CHK_1 | DF_IS_SETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700313 DF_UA | DF_UB | DF_NULL_CHK_1 | DF_IS_SETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700316 DF_UA | DF_UB | DF_NULL_CHK_1 | DF_IS_SETTER | 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
buzbee67bc2362011-10-11 18:08:40 -0700319 DF_UA | DF_UB | DF_NULL_CHK_1 | DF_IS_SETTER | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700320
Elliott Hughesadb8c672012-03-06 16:49:32 -0800321 // 60 SGET vAA, field@BBBB
buzbee67bf8852011-08-17 17:51:35 -0700322 DF_DA | DF_IS_GETTER,
323
Elliott Hughesadb8c672012-03-06 16:49:32 -0800324 // 61 SGET_WIDE vAA, field@BBBB
buzbee67bf8852011-08-17 17:51:35 -0700325 DF_DA_WIDE | DF_IS_GETTER,
326
Elliott Hughesadb8c672012-03-06 16:49:32 -0800327 // 62 SGET_OBJECT vAA, field@BBBB
buzbee67bc2362011-10-11 18:08:40 -0700328 DF_DA | DF_IS_GETTER | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -0700329
Elliott Hughesadb8c672012-03-06 16:49:32 -0800330 // 63 SGET_BOOLEAN vAA, field@BBBB
buzbee67bf8852011-08-17 17:51:35 -0700331 DF_DA | DF_IS_GETTER,
332
Elliott Hughesadb8c672012-03-06 16:49:32 -0800333 // 64 SGET_BYTE vAA, field@BBBB
buzbee67bf8852011-08-17 17:51:35 -0700334 DF_DA | DF_IS_GETTER,
335
Elliott Hughesadb8c672012-03-06 16:49:32 -0800336 // 65 SGET_CHAR vAA, field@BBBB
buzbee67bf8852011-08-17 17:51:35 -0700337 DF_DA | DF_IS_GETTER,
338
Elliott Hughesadb8c672012-03-06 16:49:32 -0800339 // 66 SGET_SHORT vAA, field@BBBB
buzbee67bf8852011-08-17 17:51:35 -0700340 DF_DA | DF_IS_GETTER,
341
Elliott Hughesadb8c672012-03-06 16:49:32 -0800342 // 67 SPUT vAA, field@BBBB
buzbee67bf8852011-08-17 17:51:35 -0700343 DF_UA | DF_IS_SETTER,
344
Elliott Hughesadb8c672012-03-06 16:49:32 -0800345 // 68 SPUT_WIDE vAA, field@BBBB
buzbee67bf8852011-08-17 17:51:35 -0700346 DF_UA_WIDE | DF_IS_SETTER,
347
Elliott Hughesadb8c672012-03-06 16:49:32 -0800348 // 69 SPUT_OBJECT vAA, field@BBBB
buzbee67bc2362011-10-11 18:08:40 -0700349 DF_UA | DF_IS_SETTER | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -0700350
Elliott Hughesadb8c672012-03-06 16:49:32 -0800351 // 6A SPUT_BOOLEAN vAA, field@BBBB
buzbee67bf8852011-08-17 17:51:35 -0700352 DF_UA | DF_IS_SETTER,
353
Elliott Hughesadb8c672012-03-06 16:49:32 -0800354 // 6B SPUT_BYTE vAA, field@BBBB
buzbee67bf8852011-08-17 17:51:35 -0700355 DF_UA | DF_IS_SETTER,
356
Elliott Hughesadb8c672012-03-06 16:49:32 -0800357 // 6C SPUT_CHAR vAA, field@BBBB
buzbee67bf8852011-08-17 17:51:35 -0700358 DF_UA | DF_IS_SETTER,
359
Elliott Hughesadb8c672012-03-06 16:49:32 -0800360 // 6D SPUT_SHORT vAA, field@BBBB
buzbee67bf8852011-08-17 17:51:35 -0700361 DF_UA | DF_IS_SETTER,
362
Elliott Hughesadb8c672012-03-06 16:49:32 -0800363 // 6E INVOKE_VIRTUAL {vD, vE, vF, vG, vA}
buzbee43a36422011-09-14 14:00:13 -0700364 DF_FORMAT_35C | DF_NULL_CHK_OUT0,
buzbee67bf8852011-08-17 17:51:35 -0700365
Elliott Hughesadb8c672012-03-06 16:49:32 -0800366 // 6F INVOKE_SUPER {vD, vE, vF, vG, vA}
buzbee43a36422011-09-14 14:00:13 -0700367 DF_FORMAT_35C | DF_NULL_CHK_OUT0,
buzbee67bf8852011-08-17 17:51:35 -0700368
Elliott Hughesadb8c672012-03-06 16:49:32 -0800369 // 70 INVOKE_DIRECT {vD, vE, vF, vG, vA}
buzbee43a36422011-09-14 14:00:13 -0700370 DF_FORMAT_35C | DF_NULL_CHK_OUT0,
buzbee67bf8852011-08-17 17:51:35 -0700371
Elliott Hughesadb8c672012-03-06 16:49:32 -0800372 // 71 INVOKE_STATIC {vD, vE, vF, vG, vA}
buzbee67bf8852011-08-17 17:51:35 -0700373 DF_FORMAT_35C,
374
Elliott Hughesadb8c672012-03-06 16:49:32 -0800375 // 72 INVOKE_INTERFACE {vD, vE, vF, vG, vA}
buzbee67bf8852011-08-17 17:51:35 -0700376 DF_FORMAT_35C,
377
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}
buzbee43a36422011-09-14 14:00:13 -0700382 DF_FORMAT_3RC | DF_NULL_CHK_OUT0,
buzbee67bf8852011-08-17 17:51:35 -0700383
Elliott Hughesadb8c672012-03-06 16:49:32 -0800384 // 75 INVOKE_SUPER_RANGE {vCCCC .. vNNNN}
buzbee43a36422011-09-14 14:00:13 -0700385 DF_FORMAT_3RC | DF_NULL_CHK_OUT0,
buzbee67bf8852011-08-17 17:51:35 -0700386
Elliott Hughesadb8c672012-03-06 16:49:32 -0800387 // 76 INVOKE_DIRECT_RANGE {vCCCC .. vNNNN}
buzbee43a36422011-09-14 14:00:13 -0700388 DF_FORMAT_3RC | DF_NULL_CHK_OUT0,
buzbee67bf8852011-08-17 17:51:35 -0700389
Elliott Hughesadb8c672012-03-06 16:49:32 -0800390 // 77 INVOKE_STATIC_RANGE {vCCCC .. vNNNN}
buzbee67bf8852011-08-17 17:51:35 -0700391 DF_FORMAT_3RC,
392
Elliott Hughesadb8c672012-03-06 16:49:32 -0800393 // 78 INVOKE_INTERFACE_RANGE {vCCCC .. vNNNN}
buzbee67bf8852011-08-17 17:51:35 -0700394 DF_FORMAT_3RC,
395
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
buzbee67bf8852011-08-17 17:51:35 -0700721 DF_DA,
722
Elliott Hughesadb8c672012-03-06 16:49:32 -0800723 // E6 SPUT_VOLATILE
buzbee67bf8852011-08-17 17:51:35 -0700724 DF_UA,
725
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
buzbee67bc2362011-10-11 18:08:40 -0700733 DF_UA_WIDE | DF_UB | DF_NULL_CHK_1 | DF_CORE_B,
buzbee67bf8852011-08-17 17:51:35 -0700734
Elliott Hughesadb8c672012-03-06 16:49:32 -0800735 // EA SGET_WIDE_VOLATILE
buzbee67bf8852011-08-17 17:51:35 -0700736 DF_DA_WIDE,
737
Elliott Hughesadb8c672012-03-06 16:49:32 -0800738 // EB SPUT_WIDE_VOLATILE
buzbee67bf8852011-08-17 17:51:35 -0700739 DF_UA_WIDE,
740
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
buzbee67bf8852011-08-17 17:51:35 -0700745 DF_NOP,
746
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
buzbee43a36422011-09-14 14:00:13 -0700760 DF_DA | DF_UB | DF_NULL_CHK_0 | DF_IS_GETTER,
buzbee67bf8852011-08-17 17:51:35 -0700761
Elliott Hughesadb8c672012-03-06 16:49:32 -0800762 // F3 IGET_WIDE_QUICK
buzbee43a36422011-09-14 14:00:13 -0700763 DF_DA_WIDE | DF_UB | DF_NULL_CHK_0 | DF_IS_GETTER,
buzbee67bf8852011-08-17 17:51:35 -0700764
Elliott Hughesadb8c672012-03-06 16:49:32 -0800765 // F4 IGET_OBJECT_QUICK
buzbee43a36422011-09-14 14:00:13 -0700766 DF_DA | DF_UB | DF_NULL_CHK_0 | DF_IS_GETTER,
buzbee67bf8852011-08-17 17:51:35 -0700767
Elliott Hughesadb8c672012-03-06 16:49:32 -0800768 // F5 IPUT_QUICK
buzbee43a36422011-09-14 14:00:13 -0700769 DF_UA | DF_UB | DF_NULL_CHK_1 | DF_IS_SETTER,
buzbee67bf8852011-08-17 17:51:35 -0700770
Elliott Hughesadb8c672012-03-06 16:49:32 -0800771 // F6 IPUT_WIDE_QUICK
buzbee43a36422011-09-14 14:00:13 -0700772 DF_UA_WIDE | DF_UB | DF_NULL_CHK_1 |DF_IS_SETTER,
buzbee67bf8852011-08-17 17:51:35 -0700773
Elliott Hughesadb8c672012-03-06 16:49:32 -0800774 // F7 IPUT_OBJECT_QUICK
buzbee43a36422011-09-14 14:00:13 -0700775 DF_UA | DF_UB | DF_NULL_CHK_1 | DF_IS_SETTER,
buzbee67bf8852011-08-17 17:51:35 -0700776
Elliott Hughesadb8c672012-03-06 16:49:32 -0800777 // F8 INVOKE_VIRTUAL_QUICK
buzbee43a36422011-09-14 14:00:13 -0700778 DF_FORMAT_35C | DF_NULL_CHK_OUT0,
buzbee67bf8852011-08-17 17:51:35 -0700779
Elliott Hughesadb8c672012-03-06 16:49:32 -0800780 // F9 INVOKE_VIRTUAL_QUICK_RANGE
buzbee43a36422011-09-14 14:00:13 -0700781 DF_FORMAT_3RC | DF_NULL_CHK_OUT0,
buzbee67bf8852011-08-17 17:51:35 -0700782
Elliott Hughesadb8c672012-03-06 16:49:32 -0800783 // FA INVOKE_SUPER_QUICK
buzbee43a36422011-09-14 14:00:13 -0700784 DF_FORMAT_35C | DF_NULL_CHK_OUT0,
buzbee67bf8852011-08-17 17:51:35 -0700785
Elliott Hughesadb8c672012-03-06 16:49:32 -0800786 // FB INVOKE_SUPER_QUICK_RANGE
buzbee43a36422011-09-14 14:00:13 -0700787 DF_FORMAT_3RC | DF_NULL_CHK_OUT0,
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
buzbee67bc2362011-10-11 18:08:40 -0700793 DF_DA | DF_CORE_A,
buzbee67bf8852011-08-17 17:51:35 -0700794
Elliott Hughesadb8c672012-03-06 16:49:32 -0800795 // FE SPUT_OBJECT_VOLATILE
buzbee67bc2362011-10-11 18:08:40 -0700796 DF_UA | DF_CORE_A,
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,
buzbee67bf8852011-08-17 17:51:35 -0700804 /*
805 * For extended MIR inserted at the MIR2LIR stage, it is okay to have
806 * undefined values here.
807 */
808};
809
810/* Return the Dalvik register/subscript pair of a given SSA register */
811int oatConvertSSARegToDalvik(const CompilationUnit* cUnit, int ssaReg)
812{
813 return GET_ELEM_N(cUnit->ssaToDalvikMap, int, ssaReg);
814}
815
816/*
817 * Utility function to convert encoded SSA register value into Dalvik register
818 * and subscript pair. Each SSA register can be used to index the
819 * ssaToDalvikMap list to get the subscript[31..16]/dalvik_reg[15..0] mapping.
820 */
buzbeeba938cb2012-02-03 14:47:55 -0800821char* oatGetDalvikDisassembly(CompilationUnit* cUnit,
Elliott Hughesadb8c672012-03-06 16:49:32 -0800822 const DecodedInstruction& insn, const char* note)
buzbee67bf8852011-08-17 17:51:35 -0700823{
824 char buffer[256];
Elliott Hughesadb8c672012-03-06 16:49:32 -0800825 Instruction::Code opcode = insn.opcode;
buzbee67bf8852011-08-17 17:51:35 -0700826 int dfAttributes = oatDataFlowAttributes[opcode];
827 int flags;
828 char* ret;
829
830 buffer[0] = 0;
831 if ((int)opcode >= (int)kMirOpFirst) {
832 if ((int)opcode == (int)kMirOpPhi) {
833 strcpy(buffer, "PHI");
Elliott Hughesadb8c672012-03-06 16:49:32 -0800834 } else {
buzbee67bf8852011-08-17 17:51:35 -0700835 sprintf(buffer, "Opcode %#x", opcode);
836 }
837 flags = 0;
838 } else {
Elliott Hughesadb8c672012-03-06 16:49:32 -0800839 strcpy(buffer, Instruction::Name(opcode));
840 flags = Instruction::Flags(opcode);
buzbee67bf8852011-08-17 17:51:35 -0700841 }
842
843 if (note)
844 strcat(buffer, note);
845
846 /* For branches, decode the instructions to print out the branch targets */
Elliott Hughesadb8c672012-03-06 16:49:32 -0800847 if (flags & Instruction::kBranch) {
848 Instruction::Format dalvikFormat = Instruction::FormatOf(insn.opcode);
buzbee67bf8852011-08-17 17:51:35 -0700849 int offset = 0;
850 switch (dalvikFormat) {
Elliott Hughesadb8c672012-03-06 16:49:32 -0800851 case Instruction::k21t:
852 snprintf(buffer + strlen(buffer), 256, " v%d,", insn.vA);
853 offset = (int) insn.vB;
buzbee67bf8852011-08-17 17:51:35 -0700854 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -0800855 case Instruction::k22t:
856 snprintf(buffer + strlen(buffer), 256, " v%d, v%d,", insn.vA, insn.vB);
857 offset = (int) insn.vC;
buzbee67bf8852011-08-17 17:51:35 -0700858 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -0800859 case Instruction::k10t:
860 case Instruction::k20t:
861 case Instruction::k30t:
862 offset = (int) insn.vA;
buzbee67bf8852011-08-17 17:51:35 -0700863 break;
864 default:
865 LOG(FATAL) << "Unexpected branch format " << (int)dalvikFormat
866 << " / opcode " << (int)opcode;
867 }
868 snprintf(buffer + strlen(buffer), 256, " (%c%x)",
869 offset > 0 ? '+' : '-',
870 offset > 0 ? offset : -offset);
871 } else if (dfAttributes & DF_FORMAT_35C) {
872 unsigned int i;
Elliott Hughesadb8c672012-03-06 16:49:32 -0800873 for (i = 0; i < insn.vA; i++) {
buzbee67bf8852011-08-17 17:51:35 -0700874 if (i != 0) strcat(buffer, ",");
Elliott Hughesadb8c672012-03-06 16:49:32 -0800875 snprintf(buffer + strlen(buffer), 256, " v%d", insn.arg[i]);
buzbee67bf8852011-08-17 17:51:35 -0700876 }
877 }
878 else if (dfAttributes & DF_FORMAT_3RC) {
879 snprintf(buffer + strlen(buffer), 256,
Elliott Hughesadb8c672012-03-06 16:49:32 -0800880 " v%d..v%d", insn.vC, insn.vC + insn.vA - 1);
buzbee67bf8852011-08-17 17:51:35 -0700881 }
882 else {
883 if (dfAttributes & DF_A_IS_REG) {
Elliott Hughesadb8c672012-03-06 16:49:32 -0800884 snprintf(buffer + strlen(buffer), 256, " v%d", insn.vA);
buzbee67bf8852011-08-17 17:51:35 -0700885 }
886 if (dfAttributes & DF_B_IS_REG) {
Elliott Hughesadb8c672012-03-06 16:49:32 -0800887 snprintf(buffer + strlen(buffer), 256, ", v%d", insn.vB);
buzbee67bf8852011-08-17 17:51:35 -0700888 }
889 else if ((int)opcode < (int)kMirOpFirst) {
Elliott Hughesadb8c672012-03-06 16:49:32 -0800890 snprintf(buffer + strlen(buffer), 256, ", (#%d)", insn.vB);
buzbee67bf8852011-08-17 17:51:35 -0700891 }
892 if (dfAttributes & DF_C_IS_REG) {
Elliott Hughesadb8c672012-03-06 16:49:32 -0800893 snprintf(buffer + strlen(buffer), 256, ", v%d", insn.vC);
buzbee67bf8852011-08-17 17:51:35 -0700894 }
895 else if ((int)opcode < (int)kMirOpFirst) {
Elliott Hughesadb8c672012-03-06 16:49:32 -0800896 snprintf(buffer + strlen(buffer), 256, ", (#%d)", insn.vC);
buzbee67bf8852011-08-17 17:51:35 -0700897 }
898 }
899 int length = strlen(buffer) + 1;
buzbeeba938cb2012-02-03 14:47:55 -0800900 ret = (char*)oatNew(cUnit, length, false, kAllocDFInfo);
buzbee67bf8852011-08-17 17:51:35 -0700901 memcpy(ret, buffer, length);
902 return ret;
903}
904
Elliott Hughesc1f143d2011-12-01 17:31:10 -0800905char* getSSAName(const CompilationUnit* cUnit, int ssaReg, char* name)
buzbee67bf8852011-08-17 17:51:35 -0700906{
907 int ssa2DalvikValue = oatConvertSSARegToDalvik(cUnit, ssaReg);
908
909 sprintf(name, "v%d_%d",
910 DECODE_REG(ssa2DalvikValue), DECODE_SUB(ssa2DalvikValue));
911 return name;
912}
913
914/*
915 * Dalvik instruction disassembler with optional SSA printing.
916 */
buzbee31a4a6f2012-02-28 15:36:15 -0800917char* oatFullDisassembler(CompilationUnit* cUnit, const MIR* mir)
buzbee67bf8852011-08-17 17:51:35 -0700918{
919 char buffer[256];
Elliott Hughes3b6baaa2011-10-14 19:13:56 -0700920 char operand0[32], operand1[32];
Elliott Hughesadb8c672012-03-06 16:49:32 -0800921 const DecodedInstruction* insn = &mir->dalvikInsn;
922 Instruction::Code opcode = insn->opcode;
buzbee67bf8852011-08-17 17:51:35 -0700923 int dfAttributes = oatDataFlowAttributes[opcode];
Elliott Hughesc1f143d2011-12-01 17:31:10 -0800924 char* ret;
buzbee67bf8852011-08-17 17:51:35 -0700925 int length;
buzbee67bf8852011-08-17 17:51:35 -0700926
927 buffer[0] = 0;
Elliott Hughesadb8c672012-03-06 16:49:32 -0800928 if (static_cast<int>(opcode) >= static_cast<int>(kMirOpFirst)) {
929 if (static_cast<int>(opcode) == static_cast<int>(kMirOpPhi)) {
buzbee67bf8852011-08-17 17:51:35 -0700930 snprintf(buffer, 256, "PHI %s = (%s",
931 getSSAName(cUnit, mir->ssaRep->defs[0], operand0),
932 getSSAName(cUnit, mir->ssaRep->uses[0], operand1));
933 int i;
934 for (i = 1; i < mir->ssaRep->numUses; i++) {
935 snprintf(buffer + strlen(buffer), 256, ", %s",
936 getSSAName(cUnit, mir->ssaRep->uses[i], operand0));
937 }
938 snprintf(buffer + strlen(buffer), 256, ")");
939 }
940 else {
941 sprintf(buffer, "Opcode %#x", opcode);
942 }
943 goto done;
944 } else {
Elliott Hughesadb8c672012-03-06 16:49:32 -0800945 strcpy(buffer, Instruction::Name(opcode));
buzbee67bf8852011-08-17 17:51:35 -0700946 }
947
buzbee67bf8852011-08-17 17:51:35 -0700948 /* For branches, decode the instructions to print out the branch targets */
Elliott Hughesadb8c672012-03-06 16:49:32 -0800949 if (Instruction::Flags(insn->opcode) & Instruction::kBranch) {
950 Instruction::Format dalvikFormat = Instruction::FormatOf(insn->opcode);
buzbee67bf8852011-08-17 17:51:35 -0700951 int delta = 0;
952 switch (dalvikFormat) {
Elliott Hughesadb8c672012-03-06 16:49:32 -0800953 case Instruction::k21t:
buzbee67bf8852011-08-17 17:51:35 -0700954 snprintf(buffer + strlen(buffer), 256, " %s, ",
955 getSSAName(cUnit, mir->ssaRep->uses[0], operand0));
956 delta = (int) insn->vB;
957 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -0800958 case Instruction::k22t:
buzbee67bf8852011-08-17 17:51:35 -0700959 snprintf(buffer + strlen(buffer), 256, " %s, %s, ",
960 getSSAName(cUnit, mir->ssaRep->uses[0], operand0),
961 getSSAName(cUnit, mir->ssaRep->uses[1], operand1));
962 delta = (int) insn->vC;
963 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -0800964 case Instruction::k10t:
965 case Instruction::k20t:
966 case Instruction::k30t:
buzbee67bf8852011-08-17 17:51:35 -0700967 delta = (int) insn->vA;
968 break;
969 default:
970 LOG(FATAL) << "Unexpected branch format: " <<
971 (int)dalvikFormat;
972 }
973 snprintf(buffer + strlen(buffer), 256, " %04x",
974 mir->offset + delta);
975 } else if (dfAttributes & (DF_FORMAT_35C | DF_FORMAT_3RC)) {
976 unsigned int i;
977 for (i = 0; i < insn->vA; i++) {
978 if (i != 0) strcat(buffer, ",");
979 snprintf(buffer + strlen(buffer), 256, " %s",
980 getSSAName(cUnit, mir->ssaRep->uses[i], operand0));
981 }
982 } else {
983 int udIdx;
984 if (mir->ssaRep->numDefs) {
985
986 for (udIdx = 0; udIdx < mir->ssaRep->numDefs; udIdx++) {
987 snprintf(buffer + strlen(buffer), 256, " %s",
988 getSSAName(cUnit, mir->ssaRep->defs[udIdx], operand0));
989 }
990 strcat(buffer, ",");
991 }
992 if (mir->ssaRep->numUses) {
993 /* No leading ',' for the first use */
994 snprintf(buffer + strlen(buffer), 256, " %s",
995 getSSAName(cUnit, mir->ssaRep->uses[0], operand0));
996 for (udIdx = 1; udIdx < mir->ssaRep->numUses; udIdx++) {
997 snprintf(buffer + strlen(buffer), 256, ", %s",
998 getSSAName(cUnit, mir->ssaRep->uses[udIdx], operand0));
999 }
1000 }
Elliott Hughesadb8c672012-03-06 16:49:32 -08001001 if (static_cast<int>(opcode) < static_cast<int>(kMirOpFirst)) {
1002 Instruction::Format dalvikFormat = Instruction::FormatOf(opcode);
buzbee67bf8852011-08-17 17:51:35 -07001003 switch (dalvikFormat) {
Elliott Hughesadb8c672012-03-06 16:49:32 -08001004 case Instruction::k11n: // op vA, #+B
1005 case Instruction::k21s: // op vAA, #+BBBB
1006 case Instruction::k21h: // op vAA, #+BBBB00000[00000000]
1007 case Instruction::k31i: // op vAA, #+BBBBBBBB
1008 case Instruction::k51l: // op vAA, #+BBBBBBBBBBBBBBBB
buzbee67bf8852011-08-17 17:51:35 -07001009 snprintf(buffer + strlen(buffer), 256, " #%#x", insn->vB);
1010 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001011 case Instruction::k21c: // op vAA, thing@BBBB
1012 case Instruction::k31c: // op vAA, thing@BBBBBBBB
buzbee67bf8852011-08-17 17:51:35 -07001013 snprintf(buffer + strlen(buffer), 256, " @%#x", insn->vB);
1014 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001015 case Instruction::k22b: // op vAA, vBB, #+CC
1016 case Instruction::k22s: // op vA, vB, #+CCCC
buzbee67bf8852011-08-17 17:51:35 -07001017 snprintf(buffer + strlen(buffer), 256, " #%#x", insn->vC);
1018 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001019 case Instruction::k22c: // op vA, vB, thing@CCCC
buzbee67bf8852011-08-17 17:51:35 -07001020 snprintf(buffer + strlen(buffer), 256, " @%#x", insn->vC);
1021 break;
1022 /* No need for special printing */
1023 default:
1024 break;
1025 }
1026 }
1027 }
1028
1029done:
1030 length = strlen(buffer) + 1;
buzbeeba938cb2012-02-03 14:47:55 -08001031 ret = (char*) oatNew(cUnit, length, false, kAllocDFInfo);
buzbee67bf8852011-08-17 17:51:35 -07001032 memcpy(ret, buffer, length);
1033 return ret;
1034}
1035
1036/*
1037 * Utility function to convert encoded SSA register value into Dalvik register
1038 * and subscript pair. Each SSA register can be used to index the
1039 * ssaToDalvikMap list to get the subscript[31..16]/dalvik_reg[15..0] mapping.
1040 */
Elliott Hughesc1f143d2011-12-01 17:31:10 -08001041char* oatGetSSAString(CompilationUnit* cUnit, SSARepresentation* ssaRep)
buzbee67bf8852011-08-17 17:51:35 -07001042{
1043 char buffer[256];
1044 char* ret;
1045 int i;
1046
1047 buffer[0] = 0;
1048 for (i = 0; i < ssaRep->numDefs; i++) {
1049 int ssa2DalvikValue = oatConvertSSARegToDalvik(cUnit, ssaRep->defs[i]);
1050
1051 sprintf(buffer + strlen(buffer), "s%d(v%d_%d) ",
1052 ssaRep->defs[i], DECODE_REG(ssa2DalvikValue),
1053 DECODE_SUB(ssa2DalvikValue));
1054 }
1055
1056 if (ssaRep->numDefs) {
1057 strcat(buffer, "<- ");
1058 }
1059
1060 for (i = 0; i < ssaRep->numUses; i++) {
1061 int ssa2DalvikValue = oatConvertSSARegToDalvik(cUnit, ssaRep->uses[i]);
1062 int len = strlen(buffer);
1063
1064 if (snprintf(buffer + len, 250 - len, "s%d(v%d_%d) ",
1065 ssaRep->uses[i], DECODE_REG(ssa2DalvikValue),
1066 DECODE_SUB(ssa2DalvikValue)) >= (250 - len)) {
1067 strcat(buffer, "...");
1068 break;
1069 }
1070 }
1071
1072 int length = strlen(buffer) + 1;
buzbeeba938cb2012-02-03 14:47:55 -08001073 ret = (char*)oatNew(cUnit, length, false, kAllocDFInfo);
buzbee67bf8852011-08-17 17:51:35 -07001074 memcpy(ret, buffer, length);
1075 return ret;
1076}
1077
1078/* Any register that is used before being defined is considered live-in */
buzbee31a4a6f2012-02-28 15:36:15 -08001079inline void handleLiveInUse(CompilationUnit* cUnit, ArenaBitVector* useV,
1080 ArenaBitVector* defV, ArenaBitVector* liveInV,
1081 int dalvikRegId)
buzbee67bf8852011-08-17 17:51:35 -07001082{
buzbeeba938cb2012-02-03 14:47:55 -08001083 oatSetBit(cUnit, useV, dalvikRegId);
buzbee67bf8852011-08-17 17:51:35 -07001084 if (!oatIsBitSet(defV, dalvikRegId)) {
buzbeeba938cb2012-02-03 14:47:55 -08001085 oatSetBit(cUnit, liveInV, dalvikRegId);
buzbee67bf8852011-08-17 17:51:35 -07001086 }
1087}
1088
1089/* Mark a reg as being defined */
buzbee31a4a6f2012-02-28 15:36:15 -08001090inline void handleDef(CompilationUnit* cUnit, ArenaBitVector* defV,
1091 int dalvikRegId)
buzbee67bf8852011-08-17 17:51:35 -07001092{
buzbeeba938cb2012-02-03 14:47:55 -08001093 oatSetBit(cUnit, defV, dalvikRegId);
buzbee67bf8852011-08-17 17:51:35 -07001094}
1095
1096/*
1097 * Find out live-in variables for natural loops. Variables that are live-in in
1098 * the main loop body are considered to be defined in the entry block.
1099 */
1100bool oatFindLocalLiveIn(CompilationUnit* cUnit, BasicBlock* bb)
1101{
1102 MIR* mir;
1103 ArenaBitVector *useV, *defV, *liveInV;
1104
1105 if (bb->dataFlowInfo == NULL) return false;
1106
1107 useV = bb->dataFlowInfo->useV =
buzbeeba938cb2012-02-03 14:47:55 -08001108 oatAllocBitVector(cUnit, cUnit->numDalvikRegisters, false, kBitMapUse);
buzbee67bf8852011-08-17 17:51:35 -07001109 defV = bb->dataFlowInfo->defV =
buzbeeba938cb2012-02-03 14:47:55 -08001110 oatAllocBitVector(cUnit, cUnit->numDalvikRegisters, false, kBitMapDef);
buzbee67bf8852011-08-17 17:51:35 -07001111 liveInV = bb->dataFlowInfo->liveInV =
buzbeeba938cb2012-02-03 14:47:55 -08001112 oatAllocBitVector(cUnit, cUnit->numDalvikRegisters, false,
1113 kBitMapLiveIn);
buzbee67bf8852011-08-17 17:51:35 -07001114
1115 for (mir = bb->firstMIRInsn; mir; mir = mir->next) {
1116 int dfAttributes =
1117 oatDataFlowAttributes[mir->dalvikInsn.opcode];
1118 DecodedInstruction *dInsn = &mir->dalvikInsn;
1119
1120 if (dfAttributes & DF_HAS_USES) {
1121 if (dfAttributes & DF_UA) {
buzbeeba938cb2012-02-03 14:47:55 -08001122 handleLiveInUse(cUnit, useV, defV, liveInV, dInsn->vA);
buzbee67bf8852011-08-17 17:51:35 -07001123 } else if (dfAttributes & DF_UA_WIDE) {
buzbeeba938cb2012-02-03 14:47:55 -08001124 handleLiveInUse(cUnit, useV, defV, liveInV, dInsn->vA);
1125 handleLiveInUse(cUnit, useV, defV, liveInV, dInsn->vA+1);
buzbee67bf8852011-08-17 17:51:35 -07001126 }
1127 if (dfAttributes & DF_UB) {
buzbeeba938cb2012-02-03 14:47:55 -08001128 handleLiveInUse(cUnit, useV, defV, liveInV, dInsn->vB);
buzbee67bf8852011-08-17 17:51:35 -07001129 } else if (dfAttributes & DF_UB_WIDE) {
buzbeeba938cb2012-02-03 14:47:55 -08001130 handleLiveInUse(cUnit, useV, defV, liveInV, dInsn->vB);
1131 handleLiveInUse(cUnit, useV, defV, liveInV, dInsn->vB+1);
buzbee67bf8852011-08-17 17:51:35 -07001132 }
1133 if (dfAttributes & DF_UC) {
buzbeeba938cb2012-02-03 14:47:55 -08001134 handleLiveInUse(cUnit, useV, defV, liveInV, dInsn->vC);
buzbee67bf8852011-08-17 17:51:35 -07001135 } else if (dfAttributes & DF_UC_WIDE) {
buzbeeba938cb2012-02-03 14:47:55 -08001136 handleLiveInUse(cUnit, useV, defV, liveInV, dInsn->vC);
1137 handleLiveInUse(cUnit, useV, defV, liveInV, dInsn->vC+1);
buzbee67bf8852011-08-17 17:51:35 -07001138 }
1139 }
buzbee07ce1d72012-02-10 17:22:02 -08001140 if (dfAttributes & DF_FORMAT_35C) {
1141 for (unsigned int i = 0; i < dInsn->vA; i++) {
1142 handleLiveInUse(cUnit, useV, defV, liveInV, dInsn->arg[i]);
1143 }
1144 }
1145 if (dfAttributes & DF_FORMAT_3RC) {
1146 for (unsigned int i = 0; i < dInsn->vA; i++) {
1147 handleLiveInUse(cUnit, useV, defV, liveInV, dInsn->vC+i);
1148 }
1149 }
buzbee67bf8852011-08-17 17:51:35 -07001150 if (dfAttributes & DF_HAS_DEFS) {
buzbeeba938cb2012-02-03 14:47:55 -08001151 handleDef(cUnit, defV, dInsn->vA);
buzbee67bf8852011-08-17 17:51:35 -07001152 if (dfAttributes & DF_DA_WIDE) {
buzbeeba938cb2012-02-03 14:47:55 -08001153 handleDef(cUnit, defV, dInsn->vA+1);
buzbee67bf8852011-08-17 17:51:35 -07001154 }
1155 }
1156 }
1157 return true;
1158}
1159
1160/* Find out the latest SSA register for a given Dalvik register */
buzbee31a4a6f2012-02-28 15:36:15 -08001161void handleSSAUse(CompilationUnit* cUnit, int* uses, int dalvikReg,
1162 int regIndex)
buzbee67bf8852011-08-17 17:51:35 -07001163{
1164 int encodedValue = cUnit->dalvikToSSAMap[dalvikReg];
1165 int ssaReg = DECODE_REG(encodedValue);
1166 uses[regIndex] = ssaReg;
1167}
1168
1169/* Setup a new SSA register for a given Dalvik register */
buzbee31a4a6f2012-02-28 15:36:15 -08001170void handleSSADef(CompilationUnit* cUnit, int* defs, int dalvikReg,
1171 int regIndex)
buzbee67bf8852011-08-17 17:51:35 -07001172{
buzbee67bf8852011-08-17 17:51:35 -07001173 int ssaReg = cUnit->numSSARegs++;
1174 /* Bump up the subscript */
buzbeef0cde542011-09-13 14:55:02 -07001175 int dalvikSub = ++cUnit->SSALastDefs[dalvikReg];
buzbee67bf8852011-08-17 17:51:35 -07001176 int newD2SMapping = ENCODE_REG_SUB(ssaReg, dalvikSub);
1177
1178 cUnit->dalvikToSSAMap[dalvikReg] = newD2SMapping;
1179
1180 int newS2DMapping = ENCODE_REG_SUB(dalvikReg, dalvikSub);
buzbeeba938cb2012-02-03 14:47:55 -08001181 oatInsertGrowableList(cUnit, cUnit->ssaToDalvikMap, newS2DMapping);
buzbee67bf8852011-08-17 17:51:35 -07001182
1183 defs[regIndex] = ssaReg;
1184}
1185
buzbeeec5adf32011-09-11 15:25:43 -07001186/* Look up new SSA names for format_35c instructions */
buzbee31a4a6f2012-02-28 15:36:15 -08001187void dataFlowSSAFormat35C(CompilationUnit* cUnit, MIR* mir)
buzbee67bf8852011-08-17 17:51:35 -07001188{
1189 DecodedInstruction *dInsn = &mir->dalvikInsn;
1190 int numUses = dInsn->vA;
1191 int i;
1192
1193 mir->ssaRep->numUses = numUses;
buzbeeba938cb2012-02-03 14:47:55 -08001194 mir->ssaRep->uses = (int *)oatNew(cUnit, sizeof(int) * numUses, true,
buzbee5abfa3e2012-01-31 17:01:43 -08001195 kAllocDFInfo);
buzbeeed3e9302011-09-23 17:34:19 -07001196 // NOTE: will be filled in during type & size inference pass
buzbeeba938cb2012-02-03 14:47:55 -08001197 mir->ssaRep->fpUse = (bool *)oatNew(cUnit, sizeof(bool) * numUses, true,
buzbee5abfa3e2012-01-31 17:01:43 -08001198 kAllocDFInfo);
buzbee67bf8852011-08-17 17:51:35 -07001199
1200 for (i = 0; i < numUses; i++) {
1201 handleSSAUse(cUnit, mir->ssaRep->uses, dInsn->arg[i], i);
1202 }
1203}
1204
buzbeeec5adf32011-09-11 15:25:43 -07001205/* Look up new SSA names for format_3rc instructions */
buzbee31a4a6f2012-02-28 15:36:15 -08001206void dataFlowSSAFormat3RC(CompilationUnit* cUnit, MIR* mir)
buzbee67bf8852011-08-17 17:51:35 -07001207{
1208 DecodedInstruction *dInsn = &mir->dalvikInsn;
1209 int numUses = dInsn->vA;
1210 int i;
1211
1212 mir->ssaRep->numUses = numUses;
buzbeeba938cb2012-02-03 14:47:55 -08001213 mir->ssaRep->uses = (int *)oatNew(cUnit, sizeof(int) * numUses, true,
buzbee5abfa3e2012-01-31 17:01:43 -08001214 kAllocDFInfo);
buzbeeed3e9302011-09-23 17:34:19 -07001215 // NOTE: will be filled in during type & size inference pass
buzbeeba938cb2012-02-03 14:47:55 -08001216 mir->ssaRep->fpUse = (bool *)oatNew(cUnit, sizeof(bool) * numUses, true,
buzbee5abfa3e2012-01-31 17:01:43 -08001217 kAllocDFInfo);
buzbee67bf8852011-08-17 17:51:35 -07001218
1219 for (i = 0; i < numUses; i++) {
1220 handleSSAUse(cUnit, mir->ssaRep->uses, dInsn->vC+i, i);
1221 }
1222}
1223
1224/* Entry function to convert a block into SSA representation */
1225bool oatDoSSAConversion(CompilationUnit* cUnit, BasicBlock* bb)
1226{
1227 MIR* mir;
1228
1229 if (bb->dataFlowInfo == NULL) return false;
1230
1231 for (mir = bb->firstMIRInsn; mir; mir = mir->next) {
1232 mir->ssaRep = (struct SSARepresentation *)
buzbeeba938cb2012-02-03 14:47:55 -08001233 oatNew(cUnit, sizeof(SSARepresentation), true, kAllocDFInfo);
buzbee67bf8852011-08-17 17:51:35 -07001234
1235 int dfAttributes =
1236 oatDataFlowAttributes[mir->dalvikInsn.opcode];
1237
buzbeef0cde542011-09-13 14:55:02 -07001238 // If not a pseudo-op, note non-leaf or can throw
Elliott Hughesadb8c672012-03-06 16:49:32 -08001239 if (static_cast<int>(mir->dalvikInsn.opcode) < static_cast<int>(kNumPackedOpcodes)) {
1240 int flags = Instruction::Flags(mir->dalvikInsn.opcode);
buzbeecefd1872011-09-09 09:59:52 -07001241
Elliott Hughesadb8c672012-03-06 16:49:32 -08001242 if (flags & Instruction::kThrow) {
buzbeef0cde542011-09-13 14:55:02 -07001243 cUnit->attrs &= ~METHOD_IS_THROW_FREE;
1244 }
buzbeecefd1872011-09-09 09:59:52 -07001245
Elliott Hughesadb8c672012-03-06 16:49:32 -08001246 if (flags & Instruction::kInvoke) {
buzbeef0cde542011-09-13 14:55:02 -07001247 cUnit->attrs &= ~METHOD_IS_LEAF;
1248 }
buzbeecefd1872011-09-09 09:59:52 -07001249 }
1250
buzbee67bf8852011-08-17 17:51:35 -07001251 int numUses = 0;
1252
1253 if (dfAttributes & DF_FORMAT_35C) {
1254 dataFlowSSAFormat35C(cUnit, mir);
1255 continue;
1256 }
1257
1258 if (dfAttributes & DF_FORMAT_3RC) {
1259 dataFlowSSAFormat3RC(cUnit, mir);
1260 continue;
1261 }
1262
1263 if (dfAttributes & DF_HAS_USES) {
1264 if (dfAttributes & DF_UA) {
1265 numUses++;
1266 } else if (dfAttributes & DF_UA_WIDE) {
1267 numUses += 2;
1268 }
1269 if (dfAttributes & DF_UB) {
1270 numUses++;
1271 } else if (dfAttributes & DF_UB_WIDE) {
1272 numUses += 2;
1273 }
1274 if (dfAttributes & DF_UC) {
1275 numUses++;
1276 } else if (dfAttributes & DF_UC_WIDE) {
1277 numUses += 2;
1278 }
1279 }
1280
1281 if (numUses) {
1282 mir->ssaRep->numUses = numUses;
buzbeeba938cb2012-02-03 14:47:55 -08001283 mir->ssaRep->uses = (int *)oatNew(cUnit, sizeof(int) * numUses,
buzbee5abfa3e2012-01-31 17:01:43 -08001284 false, kAllocDFInfo);
buzbeeba938cb2012-02-03 14:47:55 -08001285 mir->ssaRep->fpUse = (bool *)oatNew(cUnit, sizeof(bool) * numUses,
buzbee5abfa3e2012-01-31 17:01:43 -08001286 false, kAllocDFInfo);
buzbee67bf8852011-08-17 17:51:35 -07001287 }
1288
1289 int numDefs = 0;
1290
1291 if (dfAttributes & DF_HAS_DEFS) {
1292 numDefs++;
1293 if (dfAttributes & DF_DA_WIDE) {
1294 numDefs++;
1295 }
1296 }
1297
1298 if (numDefs) {
1299 mir->ssaRep->numDefs = numDefs;
buzbeeba938cb2012-02-03 14:47:55 -08001300 mir->ssaRep->defs = (int *)oatNew(cUnit, sizeof(int) * numDefs,
buzbee5abfa3e2012-01-31 17:01:43 -08001301 false, kAllocDFInfo);
buzbeeba938cb2012-02-03 14:47:55 -08001302 mir->ssaRep->fpDef = (bool *)oatNew(cUnit, sizeof(bool) * numDefs,
buzbee5abfa3e2012-01-31 17:01:43 -08001303 false, kAllocDFInfo);
buzbee67bf8852011-08-17 17:51:35 -07001304 }
1305
1306 DecodedInstruction *dInsn = &mir->dalvikInsn;
1307
1308 if (dfAttributes & DF_HAS_USES) {
1309 numUses = 0;
1310 if (dfAttributes & DF_UA) {
1311 mir->ssaRep->fpUse[numUses] = dfAttributes & DF_FP_A;
1312 handleSSAUse(cUnit, mir->ssaRep->uses, dInsn->vA, numUses++);
1313 } else if (dfAttributes & DF_UA_WIDE) {
1314 mir->ssaRep->fpUse[numUses] = dfAttributes & DF_FP_A;
1315 handleSSAUse(cUnit, mir->ssaRep->uses, dInsn->vA, numUses++);
1316 mir->ssaRep->fpUse[numUses] = dfAttributes & DF_FP_A;
1317 handleSSAUse(cUnit, mir->ssaRep->uses, dInsn->vA+1, numUses++);
1318 }
1319 if (dfAttributes & DF_UB) {
1320 mir->ssaRep->fpUse[numUses] = dfAttributes & DF_FP_B;
1321 handleSSAUse(cUnit, mir->ssaRep->uses, dInsn->vB, numUses++);
1322 } else if (dfAttributes & DF_UB_WIDE) {
1323 mir->ssaRep->fpUse[numUses] = dfAttributes & DF_FP_B;
1324 handleSSAUse(cUnit, mir->ssaRep->uses, dInsn->vB, numUses++);
1325 mir->ssaRep->fpUse[numUses] = dfAttributes & DF_FP_B;
1326 handleSSAUse(cUnit, mir->ssaRep->uses, dInsn->vB+1, numUses++);
1327 }
1328 if (dfAttributes & DF_UC) {
1329 mir->ssaRep->fpUse[numUses] = dfAttributes & DF_FP_C;
1330 handleSSAUse(cUnit, mir->ssaRep->uses, dInsn->vC, numUses++);
1331 } else if (dfAttributes & DF_UC_WIDE) {
1332 mir->ssaRep->fpUse[numUses] = dfAttributes & DF_FP_C;
1333 handleSSAUse(cUnit, mir->ssaRep->uses, dInsn->vC, numUses++);
1334 mir->ssaRep->fpUse[numUses] = dfAttributes & DF_FP_C;
1335 handleSSAUse(cUnit, mir->ssaRep->uses, dInsn->vC+1, numUses++);
1336 }
1337 }
1338 if (dfAttributes & DF_HAS_DEFS) {
1339 mir->ssaRep->fpDef[0] = dfAttributes & DF_FP_A;
1340 handleSSADef(cUnit, mir->ssaRep->defs, dInsn->vA, 0);
1341 if (dfAttributes & DF_DA_WIDE) {
1342 mir->ssaRep->fpDef[1] = dfAttributes & DF_FP_A;
1343 handleSSADef(cUnit, mir->ssaRep->defs, dInsn->vA+1, 1);
1344 }
1345 }
1346 }
1347
buzbee5abfa3e2012-01-31 17:01:43 -08001348 if (!cUnit->disableDataflow) {
1349 /*
1350 * Take a snapshot of Dalvik->SSA mapping at the end of each block. The
1351 * input to PHI nodes can be derived from the snapshot of all
1352 * predecessor blocks.
1353 */
1354 bb->dataFlowInfo->dalvikToSSAMap =
buzbeeba938cb2012-02-03 14:47:55 -08001355 (int *)oatNew(cUnit, sizeof(int) * cUnit->numDalvikRegisters, false,
buzbee5abfa3e2012-01-31 17:01:43 -08001356 kAllocDFInfo);
buzbee67bf8852011-08-17 17:51:35 -07001357
buzbee5abfa3e2012-01-31 17:01:43 -08001358 memcpy(bb->dataFlowInfo->dalvikToSSAMap, cUnit->dalvikToSSAMap,
1359 sizeof(int) * cUnit->numDalvikRegisters);
1360 }
buzbee67bf8852011-08-17 17:51:35 -07001361 return true;
1362}
1363
1364/* Setup a constant value for opcodes thare have the DF_SETS_CONST attribute */
buzbee31a4a6f2012-02-28 15:36:15 -08001365void setConstant(CompilationUnit* cUnit, int ssaReg, int value)
buzbee67bf8852011-08-17 17:51:35 -07001366{
buzbeeba938cb2012-02-03 14:47:55 -08001367 oatSetBit(cUnit, cUnit->isConstantV, ssaReg);
buzbee67bf8852011-08-17 17:51:35 -07001368 cUnit->constantValues[ssaReg] = value;
1369}
1370
1371bool oatDoConstantPropagation(CompilationUnit* cUnit, BasicBlock* bb)
1372{
1373 MIR* mir;
1374 ArenaBitVector *isConstantV = cUnit->isConstantV;
1375
1376 for (mir = bb->firstMIRInsn; mir; mir = mir->next) {
1377 int dfAttributes =
1378 oatDataFlowAttributes[mir->dalvikInsn.opcode];
1379
1380 DecodedInstruction *dInsn = &mir->dalvikInsn;
1381
1382 if (!(dfAttributes & DF_HAS_DEFS)) continue;
1383
1384 /* Handle instructions that set up constants directly */
1385 if (dfAttributes & DF_SETS_CONST) {
1386 if (dfAttributes & DF_DA) {
1387 switch (dInsn->opcode) {
Elliott Hughesadb8c672012-03-06 16:49:32 -08001388 case Instruction::CONST_4:
1389 case Instruction::CONST_16:
1390 case Instruction::CONST:
buzbee67bf8852011-08-17 17:51:35 -07001391 setConstant(cUnit, mir->ssaRep->defs[0], dInsn->vB);
1392 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001393 case Instruction::CONST_HIGH16:
buzbee67bf8852011-08-17 17:51:35 -07001394 setConstant(cUnit, mir->ssaRep->defs[0],
1395 dInsn->vB << 16);
1396 break;
1397 default:
1398 break;
1399 }
1400 } else if (dfAttributes & DF_DA_WIDE) {
1401 switch (dInsn->opcode) {
Elliott Hughesadb8c672012-03-06 16:49:32 -08001402 case Instruction::CONST_WIDE_16:
1403 case Instruction::CONST_WIDE_32:
buzbee67bf8852011-08-17 17:51:35 -07001404 setConstant(cUnit, mir->ssaRep->defs[0], dInsn->vB);
1405 setConstant(cUnit, mir->ssaRep->defs[1], 0);
1406 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001407 case Instruction::CONST_WIDE:
buzbee67bf8852011-08-17 17:51:35 -07001408 setConstant(cUnit, mir->ssaRep->defs[0],
1409 (int) dInsn->vB_wide);
1410 setConstant(cUnit, mir->ssaRep->defs[1],
1411 (int) (dInsn->vB_wide >> 32));
1412 break;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001413 case Instruction::CONST_WIDE_HIGH16:
buzbee67bf8852011-08-17 17:51:35 -07001414 setConstant(cUnit, mir->ssaRep->defs[0], 0);
1415 setConstant(cUnit, mir->ssaRep->defs[1],
1416 dInsn->vB << 16);
1417 break;
1418 default:
1419 break;
1420 }
1421 }
1422 /* Handle instructions that set up constants directly */
1423 } else if (dfAttributes & DF_IS_MOVE) {
1424 int i;
1425
1426 for (i = 0; i < mir->ssaRep->numUses; i++) {
1427 if (!oatIsBitSet(isConstantV, mir->ssaRep->uses[i])) break;
1428 }
1429 /* Move a register holding a constant to another register */
1430 if (i == mir->ssaRep->numUses) {
1431 setConstant(cUnit, mir->ssaRep->defs[0],
1432 cUnit->constantValues[mir->ssaRep->uses[0]]);
1433 if (dfAttributes & DF_DA_WIDE) {
1434 setConstant(cUnit, mir->ssaRep->defs[1],
1435 cUnit->constantValues[mir->ssaRep->uses[1]]);
1436 }
1437 }
1438 }
1439 }
1440 /* TODO: implement code to handle arithmetic operations */
1441 return true;
1442}
1443
1444/* Setup the basic data structures for SSA conversion */
1445void oatInitializeSSAConversion(CompilationUnit* cUnit)
1446{
1447 int i;
Ian Rogersa3760aa2011-11-14 14:32:37 -08001448 int numDalvikReg = cUnit->numDalvikRegisters;
buzbee67bf8852011-08-17 17:51:35 -07001449
buzbeeba938cb2012-02-03 14:47:55 -08001450 cUnit->ssaToDalvikMap = (GrowableList *)oatNew(cUnit, sizeof(GrowableList),
buzbee5abfa3e2012-01-31 17:01:43 -08001451 false, kAllocDFInfo);
1452 // Create the SSAtoDalvikMap, estimating the max size
buzbeeba938cb2012-02-03 14:47:55 -08001453 oatInitGrowableList(cUnit, cUnit->ssaToDalvikMap,
buzbee5abfa3e2012-01-31 17:01:43 -08001454 numDalvikReg + cUnit->defCount + 128,
1455 kListSSAtoDalvikMap);
buzbee67bf8852011-08-17 17:51:35 -07001456 /*
1457 * Initial number of SSA registers is equal to the number of Dalvik
1458 * registers.
1459 */
1460 cUnit->numSSARegs = numDalvikReg;
1461
1462 /*
1463 * Initialize the SSA2Dalvik map list. For the first numDalvikReg elements,
1464 * the subscript is 0 so we use the ENCODE_REG_SUB macro to encode the value
1465 * into "(0 << 16) | i"
1466 */
1467 for (i = 0; i < numDalvikReg; i++) {
buzbeeba938cb2012-02-03 14:47:55 -08001468 oatInsertGrowableList(cUnit, cUnit->ssaToDalvikMap,
1469 ENCODE_REG_SUB(i, 0));
buzbee67bf8852011-08-17 17:51:35 -07001470 }
1471
1472 /*
1473 * Initialize the DalvikToSSAMap map. The low 16 bit is the SSA register id,
1474 * while the high 16 bit is the current subscript. The original Dalvik
1475 * register N is mapped to SSA register N with subscript 0.
1476 */
buzbeeba938cb2012-02-03 14:47:55 -08001477 cUnit->dalvikToSSAMap = (int *)oatNew(cUnit, sizeof(int) * numDalvikReg,
buzbee5abfa3e2012-01-31 17:01:43 -08001478 false, kAllocDFInfo);
buzbeef0cde542011-09-13 14:55:02 -07001479 /* Keep track of the higest def for each dalvik reg */
buzbeeba938cb2012-02-03 14:47:55 -08001480 cUnit->SSALastDefs = (int *)oatNew(cUnit, sizeof(int) * numDalvikReg,
buzbee5abfa3e2012-01-31 17:01:43 -08001481 false, kAllocDFInfo);
buzbeef0cde542011-09-13 14:55:02 -07001482
buzbee67bf8852011-08-17 17:51:35 -07001483 for (i = 0; i < numDalvikReg; i++) {
1484 cUnit->dalvikToSSAMap[i] = i;
buzbeef0cde542011-09-13 14:55:02 -07001485 cUnit->SSALastDefs[i] = 0;
buzbee67bf8852011-08-17 17:51:35 -07001486 }
1487
1488 /*
1489 * Allocate the BasicBlockDataFlow structure for the entry and code blocks
1490 */
1491 GrowableListIterator iterator;
1492
1493 oatGrowableListIteratorInit(&cUnit->blockList, &iterator);
1494
1495 while (true) {
1496 BasicBlock* bb = (BasicBlock *) oatGrowableListIteratorNext(&iterator);
1497 if (bb == NULL) break;
1498 if (bb->hidden == true) continue;
1499 if (bb->blockType == kDalvikByteCode ||
1500 bb->blockType == kEntryBlock ||
1501 bb->blockType == kExitBlock) {
1502 bb->dataFlowInfo = (BasicBlockDataFlow *)
buzbeeba938cb2012-02-03 14:47:55 -08001503 oatNew(cUnit, sizeof(BasicBlockDataFlow),
buzbee5abfa3e2012-01-31 17:01:43 -08001504 true, kAllocDFInfo);
buzbee67bf8852011-08-17 17:51:35 -07001505 }
1506 }
1507}
1508
1509/* Clear the visited flag for each BB */
buzbee31a4a6f2012-02-28 15:36:15 -08001510bool oatClearVisitedFlag(struct CompilationUnit* cUnit, struct BasicBlock* bb)
buzbee67bf8852011-08-17 17:51:35 -07001511{
1512 bb->visited = false;
1513 return true;
1514}
1515
1516void oatDataFlowAnalysisDispatcher(CompilationUnit* cUnit,
1517 bool (*func)(CompilationUnit*, BasicBlock*),
1518 DataFlowAnalysisMode dfaMode,
1519 bool isIterative)
1520{
1521 bool change = true;
1522
1523 while (change) {
1524 change = false;
1525
buzbee5b537102012-01-17 17:33:47 -08001526 switch (dfaMode) {
buzbee67bf8852011-08-17 17:51:35 -07001527 /* Scan all blocks and perform the operations specified in func */
buzbee5b537102012-01-17 17:33:47 -08001528 case kAllNodes:
1529 {
1530 GrowableListIterator iterator;
1531 oatGrowableListIteratorInit(&cUnit->blockList, &iterator);
1532 while (true) {
1533 BasicBlock* bb =
1534 (BasicBlock *) oatGrowableListIteratorNext(&iterator);
1535 if (bb == NULL) break;
1536 if (bb->hidden == true) continue;
1537 change |= (*func)(cUnit, bb);
1538 }
buzbee67bf8852011-08-17 17:51:35 -07001539 }
buzbee5b537102012-01-17 17:33:47 -08001540 break;
1541 /* Scan reachable blocks and perform the ops specified in func. */
1542 case kReachableNodes:
1543 {
1544 int numReachableBlocks = cUnit->numReachableBlocks;
1545 int idx;
1546 const GrowableList *blockList = &cUnit->blockList;
buzbee67bf8852011-08-17 17:51:35 -07001547
buzbee5b537102012-01-17 17:33:47 -08001548 for (idx = 0; idx < numReachableBlocks; idx++) {
1549 int blockIdx = cUnit->dfsOrder.elemList[idx];
1550 BasicBlock* bb =
1551 (BasicBlock *) oatGrowableListGetElement(blockList,
1552 blockIdx);
1553 change |= (*func)(cUnit, bb);
1554 }
buzbee67bf8852011-08-17 17:51:35 -07001555 }
buzbee5b537102012-01-17 17:33:47 -08001556 break;
buzbee67bf8852011-08-17 17:51:35 -07001557
buzbee5b537102012-01-17 17:33:47 -08001558 /* Scan reachable blocks by pre-order dfs and invoke func on each. */
1559 case kPreOrderDFSTraversal:
1560 {
1561 int numReachableBlocks = cUnit->numReachableBlocks;
1562 int idx;
1563 const GrowableList *blockList = &cUnit->blockList;
buzbee67bf8852011-08-17 17:51:35 -07001564
buzbee5b537102012-01-17 17:33:47 -08001565 for (idx = 0; idx < numReachableBlocks; idx++) {
1566 int dfsIdx = cUnit->dfsOrder.elemList[idx];
1567 BasicBlock* bb =
1568 (BasicBlock *) oatGrowableListGetElement(blockList,
1569 dfsIdx);
1570 change |= (*func)(cUnit, bb);
1571 }
buzbee67bf8852011-08-17 17:51:35 -07001572 }
buzbee5b537102012-01-17 17:33:47 -08001573 break;
1574 /* Scan reachable blocks post-order dfs and invoke func on each. */
1575 case kPostOrderDFSTraversal:
1576 {
1577 int numReachableBlocks = cUnit->numReachableBlocks;
1578 int idx;
1579 const GrowableList *blockList = &cUnit->blockList;
buzbee67bf8852011-08-17 17:51:35 -07001580
buzbee5b537102012-01-17 17:33:47 -08001581 for (idx = numReachableBlocks - 1; idx >= 0; idx--) {
1582 int dfsIdx = cUnit->dfsOrder.elemList[idx];
1583 BasicBlock* bb =
1584 (BasicBlock *) oatGrowableListGetElement(blockList,
1585 dfsIdx);
1586 change |= (*func)(cUnit, bb);
1587 }
buzbee67bf8852011-08-17 17:51:35 -07001588 }
buzbee5b537102012-01-17 17:33:47 -08001589 break;
1590 /* Scan reachable post-order dom tree and invoke func on each. */
1591 case kPostOrderDOMTraversal:
1592 {
1593 int numReachableBlocks = cUnit->numReachableBlocks;
1594 int idx;
1595 const GrowableList *blockList = &cUnit->blockList;
1596
1597 for (idx = 0; idx < numReachableBlocks; idx++) {
1598 int domIdx = cUnit->domPostOrderTraversal.elemList[idx];
1599 BasicBlock* bb =
1600 (BasicBlock *) oatGrowableListGetElement(blockList,
1601 domIdx);
1602 change |= (*func)(cUnit, bb);
1603 }
1604 }
1605 break;
1606 /* Scan reachable blocks reverse post-order dfs, invoke func on each */
1607 case kReversePostOrderTraversal:
1608 {
1609 int numReachableBlocks = cUnit->numReachableBlocks;
1610 int idx;
1611 const GrowableList *blockList = &cUnit->blockList;
1612
1613 for (idx = numReachableBlocks - 1; idx >= 0; idx--) {
1614 int revIdx = cUnit->dfsPostOrder.elemList[idx];
1615 BasicBlock* bb =
1616 (BasicBlock *) oatGrowableListGetElement(blockList,
1617 revIdx);
1618 change |= (*func)(cUnit, bb);
1619 }
1620 }
1621 break;
1622 default:
1623 LOG(FATAL) << "Unknown traversal mode " << (int)dfaMode;
buzbee67bf8852011-08-17 17:51:35 -07001624 }
1625 /* If isIterative is false, exit the loop after the first iteration */
1626 change &= isIterative;
1627 }
1628}
buzbee43a36422011-09-14 14:00:13 -07001629
buzbee31a4a6f2012-02-28 15:36:15 -08001630bool nullCheckEliminationInit(struct CompilationUnit* cUnit,
1631 struct BasicBlock* bb)
buzbee43a36422011-09-14 14:00:13 -07001632{
1633 if (bb->dataFlowInfo == NULL) return false;
1634 bb->dataFlowInfo->endingNullCheckV =
buzbeeba938cb2012-02-03 14:47:55 -08001635 oatAllocBitVector(cUnit, cUnit->numSSARegs, false, kBitMapNullCheck);
buzbee43a36422011-09-14 14:00:13 -07001636 oatClearAllBits(bb->dataFlowInfo->endingNullCheckV);
1637 return true;
1638}
1639
1640/* Eliminate unnecessary null checks for a basic block. */
buzbee31a4a6f2012-02-28 15:36:15 -08001641bool eliminateNullChecks( struct CompilationUnit* cUnit, struct BasicBlock* bb)
buzbee43a36422011-09-14 14:00:13 -07001642{
1643 if (bb->dataFlowInfo == NULL) return false;
buzbee5abfa3e2012-01-31 17:01:43 -08001644
buzbee43a36422011-09-14 14:00:13 -07001645 /*
1646 * Set initial state. Be conservative with catch
1647 * blocks and start with no assumptions about null check
1648 * status (except for "this").
1649 */
buzbee43a36422011-09-14 14:00:13 -07001650 if ((bb->blockType == kEntryBlock) | bb->catchEntry) {
1651 oatClearAllBits(cUnit->tempSSARegisterV);
Elliott Hughes11d1b0c2012-01-23 16:57:47 -08001652 if ((cUnit->access_flags & kAccStatic) == 0) {
buzbee43a36422011-09-14 14:00:13 -07001653 // If non-static method, mark "this" as non-null
Ian Rogersa3760aa2011-11-14 14:32:37 -08001654 int thisReg = cUnit->numDalvikRegisters - cUnit->numIns;
buzbeeba938cb2012-02-03 14:47:55 -08001655 oatSetBit(cUnit, cUnit->tempSSARegisterV, thisReg);
buzbee43a36422011-09-14 14:00:13 -07001656 }
1657 } else {
1658 // Starting state is intesection of all incoming arcs
buzbee5abfa3e2012-01-31 17:01:43 -08001659 GrowableListIterator iter;
1660 oatGrowableListIteratorInit(bb->predecessors, &iter);
1661 BasicBlock* predBB = (BasicBlock*)oatGrowableListIteratorNext(&iter);
1662 DCHECK(predBB != NULL);
buzbee43a36422011-09-14 14:00:13 -07001663 oatCopyBitVector(cUnit->tempSSARegisterV,
1664 predBB->dataFlowInfo->endingNullCheckV);
1665 while (true) {
buzbee5abfa3e2012-01-31 17:01:43 -08001666 predBB = (BasicBlock*)oatGrowableListIteratorNext(&iter);
1667 if (!predBB) break;
buzbeeaad72012011-09-21 21:52:09 -07001668 if ((predBB->dataFlowInfo == NULL) ||
1669 (predBB->dataFlowInfo->endingNullCheckV == NULL)) {
1670 continue;
1671 }
buzbee43a36422011-09-14 14:00:13 -07001672 oatIntersectBitVectors(cUnit->tempSSARegisterV,
1673 cUnit->tempSSARegisterV,
1674 predBB->dataFlowInfo->endingNullCheckV);
1675 }
1676 }
1677
1678 // Walk through the instruction in the block, updating as necessary
1679 for (MIR* mir = bb->firstMIRInsn; mir; mir = mir->next) {
1680 if (mir->ssaRep == NULL) {
1681 continue;
1682 }
1683 int dfAttributes =
1684 oatDataFlowAttributes[mir->dalvikInsn.opcode];
1685
1686 // Mark target of NEW* as non-null
1687 if (dfAttributes & DF_NON_NULL_DST) {
buzbeeba938cb2012-02-03 14:47:55 -08001688 oatSetBit(cUnit, cUnit->tempSSARegisterV, mir->ssaRep->defs[0]);
buzbee43a36422011-09-14 14:00:13 -07001689 }
1690
1691 // Mark non-null returns from invoke-style NEW*
1692 if (dfAttributes & DF_NON_NULL_RET) {
1693 MIR* nextMir = mir->next;
Elliott Hughesadb8c672012-03-06 16:49:32 -08001694 // Next should be an MOVE_RESULT_OBJECT
1695 if (nextMir && nextMir->dalvikInsn.opcode == Instruction::MOVE_RESULT_OBJECT) {
buzbee43a36422011-09-14 14:00:13 -07001696 // Mark as null checked
buzbeeba938cb2012-02-03 14:47:55 -08001697 oatSetBit(cUnit, cUnit->tempSSARegisterV,
1698 nextMir->ssaRep->defs[0]);
buzbee43a36422011-09-14 14:00:13 -07001699 } else {
1700 if (nextMir) {
1701 LOG(WARNING) << "Unexpected opcode following new: " <<
1702 (int)nextMir->dalvikInsn.opcode;
buzbee949f56e2011-10-06 11:05:45 -07001703 } else if (bb->fallThrough) {
1704 // Look in next basic block
1705 struct BasicBlock* nextBB = bb->fallThrough;
1706 for (MIR* tmir = nextBB->firstMIRInsn; tmir;
1707 tmir =tmir->next){
1708 if ((int)tmir->dalvikInsn.opcode >= (int)kMirOpFirst) {
1709 continue;
1710 }
Elliott Hughesadb8c672012-03-06 16:49:32 -08001711 // First non-pseudo should be MOVE_RESULT_OBJECT
1712 if (tmir->dalvikInsn.opcode == Instruction::MOVE_RESULT_OBJECT) {
buzbee949f56e2011-10-06 11:05:45 -07001713 // Mark as null checked
buzbeeba938cb2012-02-03 14:47:55 -08001714 oatSetBit(cUnit, cUnit->tempSSARegisterV,
buzbee949f56e2011-10-06 11:05:45 -07001715 tmir->ssaRep->defs[0]);
1716 } else {
1717 LOG(WARNING) << "Unexpected op after new: " <<
1718 (int)tmir->dalvikInsn.opcode;
1719 }
1720 break;
1721 }
buzbee43a36422011-09-14 14:00:13 -07001722 }
1723 }
1724 }
1725
1726 /*
1727 * Propagate nullcheck state on register copies (including
1728 * Phi pseudo copies. For the latter, nullcheck state is
1729 * the "and" of all the Phi's operands.
1730 */
1731 if (dfAttributes & (DF_NULL_TRANSFER_0 | DF_NULL_TRANSFER_N)) {
1732 int tgtSreg = mir->ssaRep->defs[0];
1733 int operands = (dfAttributes & DF_NULL_TRANSFER_0) ? 1 :
1734 mir->ssaRep->numUses;
1735 bool nullChecked = true;
1736 for (int i = 0; i < operands; i++) {
1737 nullChecked &= oatIsBitSet(cUnit->tempSSARegisterV,
1738 mir->ssaRep->uses[i]);
1739 }
1740 if (nullChecked) {
buzbeeba938cb2012-02-03 14:47:55 -08001741 oatSetBit(cUnit, cUnit->tempSSARegisterV, tgtSreg);
buzbee43a36422011-09-14 14:00:13 -07001742 }
1743 }
1744
1745 // Already nullchecked?
1746 if (dfAttributes & DF_HAS_NULL_CHKS) {
1747 int srcSreg = (dfAttributes & DF_NULL_CHK_1) ?
1748 mir->ssaRep->uses[1] : mir->ssaRep->uses[0];
1749 if (oatIsBitSet(cUnit->tempSSARegisterV, srcSreg)) {
1750 // Eliminate the null check
1751 mir->optimizationFlags |= MIR_IGNORE_NULL_CHECK;
1752 } else {
1753 // Mark sReg as null-checked
buzbeeba938cb2012-02-03 14:47:55 -08001754 oatSetBit(cUnit, cUnit->tempSSARegisterV, srcSreg);
buzbee43a36422011-09-14 14:00:13 -07001755 }
1756 }
1757 }
1758
1759 // Did anything change?
1760 bool res = oatCompareBitVectors(bb->dataFlowInfo->endingNullCheckV,
1761 cUnit->tempSSARegisterV);
1762 if (res) {
1763 oatCopyBitVector(bb->dataFlowInfo->endingNullCheckV,
1764 cUnit->tempSSARegisterV);
1765 }
1766 return res;
1767}
1768
1769void oatMethodNullCheckElimination(CompilationUnit *cUnit)
1770{
1771 if (!(cUnit->disableOpt & (1 << kNullCheckElimination))) {
1772 DCHECK(cUnit->tempSSARegisterV != NULL);
1773 oatDataFlowAnalysisDispatcher(cUnit, nullCheckEliminationInit,
1774 kAllNodes,
1775 false /* isIterative */);
1776 oatDataFlowAnalysisDispatcher(cUnit, eliminateNullChecks,
1777 kPreOrderDFSTraversal,
1778 true /* isIterative */);
1779 }
1780}
Elliott Hughes11d1b0c2012-01-23 16:57:47 -08001781
1782} // namespace art