dex_instruction_utils.h revision e5f13e57ff8fa36342beb33830b3ec5942a61cca
1/* 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_RUNTIME_DEX_INSTRUCTION_UTILS_H_ 18#define ART_RUNTIME_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 IsInstructionReturn(Instruction::Code opcode) { 53 return Instruction::RETURN_VOID <= opcode && opcode <= Instruction::RETURN_OBJECT; 54} 55 56constexpr bool IsInstructionInvoke(Instruction::Code opcode) { 57 return Instruction::INVOKE_VIRTUAL <= opcode && opcode <= Instruction::INVOKE_INTERFACE_RANGE && 58 opcode != Instruction::RETURN_VOID_BARRIER; 59} 60 61constexpr bool IsInstructionQuickInvoke(Instruction::Code opcode) { 62 return opcode == Instruction::INVOKE_VIRTUAL_QUICK || 63 opcode == Instruction::INVOKE_VIRTUAL_RANGE_QUICK; 64} 65 66constexpr bool IsInstructionInvokeStatic(Instruction::Code opcode) { 67 return opcode == Instruction::INVOKE_STATIC || opcode == Instruction::INVOKE_STATIC_RANGE; 68} 69 70constexpr bool IsInstructionGoto(Instruction::Code opcode) { 71 return Instruction::GOTO <= opcode && opcode <= Instruction::GOTO_32; 72} 73 74constexpr bool IsInstructionIfCc(Instruction::Code opcode) { 75 return Instruction::IF_EQ <= opcode && opcode <= Instruction::IF_LE; 76} 77 78constexpr bool IsInstructionIfCcZ(Instruction::Code opcode) { 79 return Instruction::IF_EQZ <= opcode && opcode <= Instruction::IF_LEZ; 80} 81 82constexpr bool IsInstructionIGet(Instruction::Code code) { 83 return Instruction::IGET <= code && code <= Instruction::IGET_SHORT; 84} 85 86constexpr bool IsInstructionIPut(Instruction::Code code) { 87 return Instruction::IPUT <= code && code <= Instruction::IPUT_SHORT; 88} 89 90constexpr bool IsInstructionSGet(Instruction::Code code) { 91 return Instruction::SGET <= code && code <= Instruction::SGET_SHORT; 92} 93 94constexpr bool IsInstructionSPut(Instruction::Code code) { 95 return Instruction::SPUT <= code && code <= Instruction::SPUT_SHORT; 96} 97 98constexpr bool IsInstructionAGet(Instruction::Code code) { 99 return Instruction::AGET <= code && code <= Instruction::AGET_SHORT; 100} 101 102constexpr bool IsInstructionAPut(Instruction::Code code) { 103 return Instruction::APUT <= code && code <= Instruction::APUT_SHORT; 104} 105 106constexpr bool IsInstructionIGetOrIPut(Instruction::Code code) { 107 return Instruction::IGET <= code && code <= Instruction::IPUT_SHORT; 108} 109 110constexpr bool IsInstructionIGetQuickOrIPutQuick(Instruction::Code code) { 111 return (code >= Instruction::IGET_QUICK && code <= Instruction::IPUT_OBJECT_QUICK) || 112 (code >= Instruction::IPUT_BOOLEAN_QUICK && code <= Instruction::IGET_SHORT_QUICK); 113} 114 115constexpr bool IsInstructionSGetOrSPut(Instruction::Code code) { 116 return Instruction::SGET <= code && code <= Instruction::SPUT_SHORT; 117} 118 119constexpr bool IsInstructionAGetOrAPut(Instruction::Code code) { 120 return Instruction::AGET <= code && code <= Instruction::APUT_SHORT; 121} 122 123constexpr bool IsInstructionBinOp2Addr(Instruction::Code code) { 124 return Instruction::ADD_INT_2ADDR <= code && code <= Instruction::REM_DOUBLE_2ADDR; 125} 126 127// TODO: Remove the #if guards below when we fully migrate to C++14. 128 129constexpr bool IsInvokeInstructionRange(Instruction::Code opcode) { 130#if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions. 131 DCHECK(IsInstructionInvoke(opcode)); 132#endif 133 return opcode >= Instruction::INVOKE_VIRTUAL_RANGE; 134} 135 136constexpr DexInvokeType InvokeInstructionType(Instruction::Code opcode) { 137#if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions. 138 DCHECK(IsInstructionInvoke(opcode)); 139#endif 140 return static_cast<DexInvokeType>(IsInvokeInstructionRange(opcode) 141 ? (opcode - Instruction::INVOKE_VIRTUAL_RANGE) 142 : (opcode - Instruction::INVOKE_VIRTUAL)); 143} 144 145constexpr DexMemAccessType IGetMemAccessType(Instruction::Code code) { 146#if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions. 147 DCHECK(IsInstructionIGet(opcode)); 148#endif 149 return static_cast<DexMemAccessType>(code - Instruction::IGET); 150} 151 152constexpr DexMemAccessType IPutMemAccessType(Instruction::Code code) { 153#if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions. 154 DCHECK(IsInstructionIPut(opcode)); 155#endif 156 return static_cast<DexMemAccessType>(code - Instruction::IPUT); 157} 158 159constexpr DexMemAccessType SGetMemAccessType(Instruction::Code code) { 160#if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions. 161 DCHECK(IsInstructionSGet(opcode)); 162#endif 163 return static_cast<DexMemAccessType>(code - Instruction::SGET); 164} 165 166constexpr DexMemAccessType SPutMemAccessType(Instruction::Code code) { 167#if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions. 168 DCHECK(IsInstructionSPut(opcode)); 169#endif 170 return static_cast<DexMemAccessType>(code - Instruction::SPUT); 171} 172 173constexpr DexMemAccessType AGetMemAccessType(Instruction::Code code) { 174#if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions. 175 DCHECK(IsInstructionAGet(opcode)); 176#endif 177 return static_cast<DexMemAccessType>(code - Instruction::AGET); 178} 179 180constexpr DexMemAccessType APutMemAccessType(Instruction::Code code) { 181#if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions. 182 DCHECK(IsInstructionAPut(opcode)); 183#endif 184 return static_cast<DexMemAccessType>(code - Instruction::APUT); 185} 186 187constexpr DexMemAccessType IGetOrIPutMemAccessType(Instruction::Code code) { 188#if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions. 189 DCHECK(IsInstructionIGetOrIPut(opcode)); 190#endif 191 return (code >= Instruction::IPUT) ? IPutMemAccessType(code) : IGetMemAccessType(code); 192} 193 194static inline DexMemAccessType IGetQuickOrIPutQuickMemAccessType(Instruction::Code code) { 195 DCHECK(IsInstructionIGetQuickOrIPutQuick(code)); 196 switch (code) { 197 case Instruction::IGET_QUICK: case Instruction::IPUT_QUICK: 198 return kDexMemAccessWord; 199 case Instruction::IGET_WIDE_QUICK: case Instruction::IPUT_WIDE_QUICK: 200 return kDexMemAccessWide; 201 case Instruction::IGET_OBJECT_QUICK: case Instruction::IPUT_OBJECT_QUICK: 202 return kDexMemAccessObject; 203 case Instruction::IGET_BOOLEAN_QUICK: case Instruction::IPUT_BOOLEAN_QUICK: 204 return kDexMemAccessBoolean; 205 case Instruction::IGET_BYTE_QUICK: case Instruction::IPUT_BYTE_QUICK: 206 return kDexMemAccessByte; 207 case Instruction::IGET_CHAR_QUICK: case Instruction::IPUT_CHAR_QUICK: 208 return kDexMemAccessChar; 209 case Instruction::IGET_SHORT_QUICK: case Instruction::IPUT_SHORT_QUICK: 210 return kDexMemAccessShort; 211 default: 212 LOG(FATAL) << code; 213 UNREACHABLE(); 214 } 215} 216 217constexpr DexMemAccessType SGetOrSPutMemAccessType(Instruction::Code code) { 218#if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions. 219 DCHECK(IsInstructionSGetOrSPut(opcode)); 220#endif 221 return (code >= Instruction::SPUT) ? SPutMemAccessType(code) : SGetMemAccessType(code); 222} 223 224constexpr DexMemAccessType AGetOrAPutMemAccessType(Instruction::Code code) { 225#if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions. 226 DCHECK(IsInstructionAGetOrAPut(opcode)); 227#endif 228 return (code >= Instruction::APUT) ? APutMemAccessType(code) : AGetMemAccessType(code); 229} 230 231} // namespace art 232 233#endif // ART_RUNTIME_DEX_INSTRUCTION_UTILS_H_ 234