blob: 72d82442b5c6ef91c3ffa8117a03e49c9a6b0c1f [file] [log] [blame]
Vladimir Markoaf6925b2014-10-31 16:37:32 +00001/*
2 * Copyright (C) 2014 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
Mathieu Chartiere5f13e52015-02-24 09:37:21 -080017#ifndef ART_RUNTIME_DEX_INSTRUCTION_UTILS_H_
18#define ART_RUNTIME_DEX_INSTRUCTION_UTILS_H_
Vladimir Markoaf6925b2014-10-31 16:37:32 +000019
20#include "dex_instruction.h"
21
22namespace art {
23
24// Dex invoke type corresponds to the ordering of INVOKE instructions;
25// this order is the same for range and non-range invokes.
26enum DexInvokeType : uint8_t {
27 kDexInvokeVirtual = 0, // invoke-virtual, invoke-virtual-range
28 kDexInvokeSuper, // invoke-super, invoke-super-range
29 kDexInvokeDirect, // invoke-direct, invoke-direct-range
30 kDexInvokeStatic, // invoke-static, invoke-static-range
31 kDexInvokeInterface, // invoke-interface, invoke-interface-range
32 kDexInvokeTypeCount
33};
34
35// Dex instruction memory access types correspond to the ordering of GET/PUT instructions;
36// this order is the same for IGET, IPUT, SGET, SPUT, AGET and APUT.
37enum DexMemAccessType : uint8_t {
38 kDexMemAccessWord = 0, // op 0; int or float, the actual type is not encoded.
39 kDexMemAccessWide, // op_WIDE 1; long or double, the actual type is not encoded.
40 kDexMemAccessObject, // op_OBJECT 2; the actual reference type is not encoded.
41 kDexMemAccessBoolean, // op_BOOLEAN 3
42 kDexMemAccessByte, // op_BYTE 4
43 kDexMemAccessChar, // op_CHAR 5
44 kDexMemAccessShort, // op_SHORT 6
45 kDexMemAccessTypeCount
46};
47
48std::ostream& operator<<(std::ostream& os, const DexMemAccessType& type);
49
50// NOTE: The following functions disregard quickened instructions.
51
Vladimir Markof8b3b8b2016-02-04 19:46:56 +000052// By "direct" const we mean to exclude const-string and const-class
53// which load data from somewhere else, i.e. indirectly.
54constexpr bool IsInstructionDirectConst(Instruction::Code opcode) {
55 return Instruction::CONST_4 <= opcode && opcode <= Instruction::CONST_WIDE_HIGH16;
56}
57
58constexpr bool IsInstructionConstWide(Instruction::Code opcode) {
59 return Instruction::CONST_WIDE_16 <= opcode && opcode <= Instruction::CONST_WIDE_HIGH16;
60}
61
Vladimir Marko321b9872014-11-24 16:33:51 +000062constexpr bool IsInstructionReturn(Instruction::Code opcode) {
63 return Instruction::RETURN_VOID <= opcode && opcode <= Instruction::RETURN_OBJECT;
64}
65
Vladimir Markoaf6925b2014-10-31 16:37:32 +000066constexpr bool IsInstructionInvoke(Instruction::Code opcode) {
67 return Instruction::INVOKE_VIRTUAL <= opcode && opcode <= Instruction::INVOKE_INTERFACE_RANGE &&
Mathieu Chartierd7cbf8a2015-03-19 12:43:20 -070068 opcode != Instruction::RETURN_VOID_NO_BARRIER;
Vladimir Markoaf6925b2014-10-31 16:37:32 +000069}
70
Mathieu Chartiere5f13e52015-02-24 09:37:21 -080071constexpr bool IsInstructionQuickInvoke(Instruction::Code opcode) {
72 return opcode == Instruction::INVOKE_VIRTUAL_QUICK ||
73 opcode == Instruction::INVOKE_VIRTUAL_RANGE_QUICK;
74}
75
Vladimir Markoaf6925b2014-10-31 16:37:32 +000076constexpr bool IsInstructionInvokeStatic(Instruction::Code opcode) {
77 return opcode == Instruction::INVOKE_STATIC || opcode == Instruction::INVOKE_STATIC_RANGE;
78}
79
Vladimir Marko26e7d452014-11-24 14:09:46 +000080constexpr bool IsInstructionGoto(Instruction::Code opcode) {
81 return Instruction::GOTO <= opcode && opcode <= Instruction::GOTO_32;
82}
83
Vladimir Markoaf6925b2014-10-31 16:37:32 +000084constexpr bool IsInstructionIfCc(Instruction::Code opcode) {
85 return Instruction::IF_EQ <= opcode && opcode <= Instruction::IF_LE;
86}
87
88constexpr bool IsInstructionIfCcZ(Instruction::Code opcode) {
89 return Instruction::IF_EQZ <= opcode && opcode <= Instruction::IF_LEZ;
90}
91
92constexpr bool IsInstructionIGet(Instruction::Code code) {
93 return Instruction::IGET <= code && code <= Instruction::IGET_SHORT;
94}
95
96constexpr bool IsInstructionIPut(Instruction::Code code) {
97 return Instruction::IPUT <= code && code <= Instruction::IPUT_SHORT;
98}
99
100constexpr bool IsInstructionSGet(Instruction::Code code) {
101 return Instruction::SGET <= code && code <= Instruction::SGET_SHORT;
102}
103
104constexpr bool IsInstructionSPut(Instruction::Code code) {
105 return Instruction::SPUT <= code && code <= Instruction::SPUT_SHORT;
106}
107
108constexpr bool IsInstructionAGet(Instruction::Code code) {
109 return Instruction::AGET <= code && code <= Instruction::AGET_SHORT;
110}
111
112constexpr bool IsInstructionAPut(Instruction::Code code) {
113 return Instruction::APUT <= code && code <= Instruction::APUT_SHORT;
114}
115
116constexpr bool IsInstructionIGetOrIPut(Instruction::Code code) {
117 return Instruction::IGET <= code && code <= Instruction::IPUT_SHORT;
118}
119
Mathieu Chartiere5f13e52015-02-24 09:37:21 -0800120constexpr bool IsInstructionIGetQuickOrIPutQuick(Instruction::Code code) {
121 return (code >= Instruction::IGET_QUICK && code <= Instruction::IPUT_OBJECT_QUICK) ||
122 (code >= Instruction::IPUT_BOOLEAN_QUICK && code <= Instruction::IGET_SHORT_QUICK);
123}
124
Vladimir Markoaf6925b2014-10-31 16:37:32 +0000125constexpr bool IsInstructionSGetOrSPut(Instruction::Code code) {
126 return Instruction::SGET <= code && code <= Instruction::SPUT_SHORT;
127}
128
129constexpr bool IsInstructionAGetOrAPut(Instruction::Code code) {
130 return Instruction::AGET <= code && code <= Instruction::APUT_SHORT;
131}
132
Vladimir Marko7a01dc22015-01-02 17:00:44 +0000133constexpr bool IsInstructionBinOp2Addr(Instruction::Code code) {
134 return Instruction::ADD_INT_2ADDR <= code && code <= Instruction::REM_DOUBLE_2ADDR;
135}
136
Vladimir Markoaf6925b2014-10-31 16:37:32 +0000137constexpr bool IsInvokeInstructionRange(Instruction::Code opcode) {
Vladimir Markoaf6925b2014-10-31 16:37:32 +0000138 DCHECK(IsInstructionInvoke(opcode));
Vladimir Markoaf6925b2014-10-31 16:37:32 +0000139 return opcode >= Instruction::INVOKE_VIRTUAL_RANGE;
140}
141
142constexpr DexInvokeType InvokeInstructionType(Instruction::Code opcode) {
Vladimir Markoaf6925b2014-10-31 16:37:32 +0000143 DCHECK(IsInstructionInvoke(opcode));
Vladimir Markoaf6925b2014-10-31 16:37:32 +0000144 return static_cast<DexInvokeType>(IsInvokeInstructionRange(opcode)
145 ? (opcode - Instruction::INVOKE_VIRTUAL_RANGE)
146 : (opcode - Instruction::INVOKE_VIRTUAL));
147}
148
149constexpr DexMemAccessType IGetMemAccessType(Instruction::Code code) {
Dan Alberteb3bd882015-08-07 16:19:01 -0700150 DCHECK(IsInstructionIGet(code));
Vladimir Markoaf6925b2014-10-31 16:37:32 +0000151 return static_cast<DexMemAccessType>(code - Instruction::IGET);
152}
153
154constexpr DexMemAccessType IPutMemAccessType(Instruction::Code code) {
Dan Alberteb3bd882015-08-07 16:19:01 -0700155 DCHECK(IsInstructionIPut(code));
Vladimir Markoaf6925b2014-10-31 16:37:32 +0000156 return static_cast<DexMemAccessType>(code - Instruction::IPUT);
157}
158
159constexpr DexMemAccessType SGetMemAccessType(Instruction::Code code) {
Dan Alberteb3bd882015-08-07 16:19:01 -0700160 DCHECK(IsInstructionSGet(code));
Vladimir Markoaf6925b2014-10-31 16:37:32 +0000161 return static_cast<DexMemAccessType>(code - Instruction::SGET);
162}
163
164constexpr DexMemAccessType SPutMemAccessType(Instruction::Code code) {
Dan Alberteb3bd882015-08-07 16:19:01 -0700165 DCHECK(IsInstructionSPut(code));
Vladimir Markoaf6925b2014-10-31 16:37:32 +0000166 return static_cast<DexMemAccessType>(code - Instruction::SPUT);
167}
168
169constexpr DexMemAccessType AGetMemAccessType(Instruction::Code code) {
Dan Alberteb3bd882015-08-07 16:19:01 -0700170 DCHECK(IsInstructionAGet(code));
Vladimir Markoaf6925b2014-10-31 16:37:32 +0000171 return static_cast<DexMemAccessType>(code - Instruction::AGET);
172}
173
174constexpr DexMemAccessType APutMemAccessType(Instruction::Code code) {
Dan Alberteb3bd882015-08-07 16:19:01 -0700175 DCHECK(IsInstructionAPut(code));
Vladimir Markoaf6925b2014-10-31 16:37:32 +0000176 return static_cast<DexMemAccessType>(code - Instruction::APUT);
177}
178
179constexpr DexMemAccessType IGetOrIPutMemAccessType(Instruction::Code code) {
Dan Alberteb3bd882015-08-07 16:19:01 -0700180 DCHECK(IsInstructionIGetOrIPut(code));
Vladimir Markoaf6925b2014-10-31 16:37:32 +0000181 return (code >= Instruction::IPUT) ? IPutMemAccessType(code) : IGetMemAccessType(code);
182}
183
Vladimir Markof04cf542016-08-31 15:25:25 +0100184inline DexMemAccessType IGetQuickOrIPutQuickMemAccessType(Instruction::Code code) {
Mathieu Chartiere5f13e52015-02-24 09:37:21 -0800185 DCHECK(IsInstructionIGetQuickOrIPutQuick(code));
186 switch (code) {
187 case Instruction::IGET_QUICK: case Instruction::IPUT_QUICK:
188 return kDexMemAccessWord;
189 case Instruction::IGET_WIDE_QUICK: case Instruction::IPUT_WIDE_QUICK:
190 return kDexMemAccessWide;
191 case Instruction::IGET_OBJECT_QUICK: case Instruction::IPUT_OBJECT_QUICK:
192 return kDexMemAccessObject;
193 case Instruction::IGET_BOOLEAN_QUICK: case Instruction::IPUT_BOOLEAN_QUICK:
194 return kDexMemAccessBoolean;
195 case Instruction::IGET_BYTE_QUICK: case Instruction::IPUT_BYTE_QUICK:
196 return kDexMemAccessByte;
197 case Instruction::IGET_CHAR_QUICK: case Instruction::IPUT_CHAR_QUICK:
198 return kDexMemAccessChar;
199 case Instruction::IGET_SHORT_QUICK: case Instruction::IPUT_SHORT_QUICK:
200 return kDexMemAccessShort;
201 default:
202 LOG(FATAL) << code;
203 UNREACHABLE();
204 }
205}
206
Vladimir Markoaf6925b2014-10-31 16:37:32 +0000207constexpr DexMemAccessType SGetOrSPutMemAccessType(Instruction::Code code) {
Dan Alberteb3bd882015-08-07 16:19:01 -0700208 DCHECK(IsInstructionSGetOrSPut(code));
Vladimir Markoaf6925b2014-10-31 16:37:32 +0000209 return (code >= Instruction::SPUT) ? SPutMemAccessType(code) : SGetMemAccessType(code);
210}
211
212constexpr DexMemAccessType AGetOrAPutMemAccessType(Instruction::Code code) {
Dan Alberteb3bd882015-08-07 16:19:01 -0700213 DCHECK(IsInstructionAGetOrAPut(code));
Vladimir Markoaf6925b2014-10-31 16:37:32 +0000214 return (code >= Instruction::APUT) ? APutMemAccessType(code) : AGetMemAccessType(code);
215}
216
217} // namespace art
218
Mathieu Chartiere5f13e52015-02-24 09:37:21 -0800219#endif // ART_RUNTIME_DEX_INSTRUCTION_UTILS_H_