blob: ad7d750dea2d0e5a48550a03941870f232f4608c [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
17#ifndef ART_COMPILER_UTILS_DEX_INSTRUCTION_UTILS_H_
18#define ART_COMPILER_UTILS_DEX_INSTRUCTION_UTILS_H_
19
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
52constexpr bool IsInstructionInvoke(Instruction::Code opcode) {
53 return Instruction::INVOKE_VIRTUAL <= opcode && opcode <= Instruction::INVOKE_INTERFACE_RANGE &&
54 opcode != Instruction::RETURN_VOID_BARRIER;
55}
56
57constexpr bool IsInstructionInvokeStatic(Instruction::Code opcode) {
58 return opcode == Instruction::INVOKE_STATIC || opcode == Instruction::INVOKE_STATIC_RANGE;
59}
60
Vladimir Marko26e7d452014-11-24 14:09:46 +000061constexpr bool IsInstructionGoto(Instruction::Code opcode) {
62 return Instruction::GOTO <= opcode && opcode <= Instruction::GOTO_32;
63}
64
Vladimir Markoaf6925b2014-10-31 16:37:32 +000065constexpr bool IsInstructionIfCc(Instruction::Code opcode) {
66 return Instruction::IF_EQ <= opcode && opcode <= Instruction::IF_LE;
67}
68
69constexpr bool IsInstructionIfCcZ(Instruction::Code opcode) {
70 return Instruction::IF_EQZ <= opcode && opcode <= Instruction::IF_LEZ;
71}
72
73constexpr bool IsInstructionIGet(Instruction::Code code) {
74 return Instruction::IGET <= code && code <= Instruction::IGET_SHORT;
75}
76
77constexpr bool IsInstructionIPut(Instruction::Code code) {
78 return Instruction::IPUT <= code && code <= Instruction::IPUT_SHORT;
79}
80
81constexpr bool IsInstructionSGet(Instruction::Code code) {
82 return Instruction::SGET <= code && code <= Instruction::SGET_SHORT;
83}
84
85constexpr bool IsInstructionSPut(Instruction::Code code) {
86 return Instruction::SPUT <= code && code <= Instruction::SPUT_SHORT;
87}
88
89constexpr bool IsInstructionAGet(Instruction::Code code) {
90 return Instruction::AGET <= code && code <= Instruction::AGET_SHORT;
91}
92
93constexpr bool IsInstructionAPut(Instruction::Code code) {
94 return Instruction::APUT <= code && code <= Instruction::APUT_SHORT;
95}
96
97constexpr bool IsInstructionIGetOrIPut(Instruction::Code code) {
98 return Instruction::IGET <= code && code <= Instruction::IPUT_SHORT;
99}
100
101constexpr bool IsInstructionSGetOrSPut(Instruction::Code code) {
102 return Instruction::SGET <= code && code <= Instruction::SPUT_SHORT;
103}
104
105constexpr bool IsInstructionAGetOrAPut(Instruction::Code code) {
106 return Instruction::AGET <= code && code <= Instruction::APUT_SHORT;
107}
108
109// TODO: Remove the #if guards below when we fully migrate to C++14.
110
111constexpr bool IsInvokeInstructionRange(Instruction::Code opcode) {
112#if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions.
113 DCHECK(IsInstructionInvoke(opcode));
114#endif
115 return opcode >= Instruction::INVOKE_VIRTUAL_RANGE;
116}
117
118constexpr DexInvokeType InvokeInstructionType(Instruction::Code opcode) {
119#if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions.
120 DCHECK(IsInstructionInvoke(opcode));
121#endif
122 return static_cast<DexInvokeType>(IsInvokeInstructionRange(opcode)
123 ? (opcode - Instruction::INVOKE_VIRTUAL_RANGE)
124 : (opcode - Instruction::INVOKE_VIRTUAL));
125}
126
127constexpr DexMemAccessType IGetMemAccessType(Instruction::Code code) {
128#if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions.
129 DCHECK(IsInstructionIGet(opcode));
130#endif
131 return static_cast<DexMemAccessType>(code - Instruction::IGET);
132}
133
134constexpr DexMemAccessType IPutMemAccessType(Instruction::Code code) {
135#if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions.
136 DCHECK(IsInstructionIPut(opcode));
137#endif
138 return static_cast<DexMemAccessType>(code - Instruction::IPUT);
139}
140
141constexpr DexMemAccessType SGetMemAccessType(Instruction::Code code) {
142#if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions.
143 DCHECK(IsInstructionSGet(opcode));
144#endif
145 return static_cast<DexMemAccessType>(code - Instruction::SGET);
146}
147
148constexpr DexMemAccessType SPutMemAccessType(Instruction::Code code) {
149#if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions.
150 DCHECK(IsInstructionSPut(opcode));
151#endif
152 return static_cast<DexMemAccessType>(code - Instruction::SPUT);
153}
154
155constexpr DexMemAccessType AGetMemAccessType(Instruction::Code code) {
156#if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions.
157 DCHECK(IsInstructionAGet(opcode));
158#endif
159 return static_cast<DexMemAccessType>(code - Instruction::AGET);
160}
161
162constexpr DexMemAccessType APutMemAccessType(Instruction::Code code) {
163#if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions.
164 DCHECK(IsInstructionAPut(opcode));
165#endif
166 return static_cast<DexMemAccessType>(code - Instruction::APUT);
167}
168
169constexpr DexMemAccessType IGetOrIPutMemAccessType(Instruction::Code code) {
170#if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions.
171 DCHECK(IsInstructionIGetOrIPut(opcode));
172#endif
173 return (code >= Instruction::IPUT) ? IPutMemAccessType(code) : IGetMemAccessType(code);
174}
175
176constexpr DexMemAccessType SGetOrSPutMemAccessType(Instruction::Code code) {
177#if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions.
178 DCHECK(IsInstructionSGetOrSPut(opcode));
179#endif
180 return (code >= Instruction::SPUT) ? SPutMemAccessType(code) : SGetMemAccessType(code);
181}
182
183constexpr DexMemAccessType AGetOrAPutMemAccessType(Instruction::Code code) {
184#if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions.
185 DCHECK(IsInstructionAGetOrAPut(opcode));
186#endif
187 return (code >= Instruction::APUT) ? APutMemAccessType(code) : AGetMemAccessType(code);
188}
189
190} // namespace art
191
192#endif // ART_COMPILER_UTILS_DEX_INSTRUCTION_UTILS_H_