| /* |
| * Copyright (C) 2014 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| #ifndef ART_RUNTIME_DEX_INSTRUCTION_UTILS_H_ |
| #define ART_RUNTIME_DEX_INSTRUCTION_UTILS_H_ |
| |
| #include "dex_instruction.h" |
| |
| namespace art { |
| |
| // Dex invoke type corresponds to the ordering of INVOKE instructions; |
| // this order is the same for range and non-range invokes. |
| enum DexInvokeType : uint8_t { |
| kDexInvokeVirtual = 0, // invoke-virtual, invoke-virtual-range |
| kDexInvokeSuper, // invoke-super, invoke-super-range |
| kDexInvokeDirect, // invoke-direct, invoke-direct-range |
| kDexInvokeStatic, // invoke-static, invoke-static-range |
| kDexInvokeInterface, // invoke-interface, invoke-interface-range |
| kDexInvokeTypeCount |
| }; |
| |
| // Dex instruction memory access types correspond to the ordering of GET/PUT instructions; |
| // this order is the same for IGET, IPUT, SGET, SPUT, AGET and APUT. |
| enum DexMemAccessType : uint8_t { |
| kDexMemAccessWord = 0, // op 0; int or float, the actual type is not encoded. |
| kDexMemAccessWide, // op_WIDE 1; long or double, the actual type is not encoded. |
| kDexMemAccessObject, // op_OBJECT 2; the actual reference type is not encoded. |
| kDexMemAccessBoolean, // op_BOOLEAN 3 |
| kDexMemAccessByte, // op_BYTE 4 |
| kDexMemAccessChar, // op_CHAR 5 |
| kDexMemAccessShort, // op_SHORT 6 |
| kDexMemAccessTypeCount |
| }; |
| |
| std::ostream& operator<<(std::ostream& os, const DexMemAccessType& type); |
| |
| // NOTE: The following functions disregard quickened instructions. |
| |
| // By "direct" const we mean to exclude const-string and const-class |
| // which load data from somewhere else, i.e. indirectly. |
| constexpr bool IsInstructionDirectConst(Instruction::Code opcode) { |
| return Instruction::CONST_4 <= opcode && opcode <= Instruction::CONST_WIDE_HIGH16; |
| } |
| |
| constexpr bool IsInstructionConstWide(Instruction::Code opcode) { |
| return Instruction::CONST_WIDE_16 <= opcode && opcode <= Instruction::CONST_WIDE_HIGH16; |
| } |
| |
| constexpr bool IsInstructionReturn(Instruction::Code opcode) { |
| return Instruction::RETURN_VOID <= opcode && opcode <= Instruction::RETURN_OBJECT; |
| } |
| |
| constexpr bool IsInstructionInvoke(Instruction::Code opcode) { |
| return Instruction::INVOKE_VIRTUAL <= opcode && opcode <= Instruction::INVOKE_INTERFACE_RANGE && |
| opcode != Instruction::RETURN_VOID_NO_BARRIER; |
| } |
| |
| constexpr bool IsInstructionQuickInvoke(Instruction::Code opcode) { |
| return opcode == Instruction::INVOKE_VIRTUAL_QUICK || |
| opcode == Instruction::INVOKE_VIRTUAL_RANGE_QUICK; |
| } |
| |
| constexpr bool IsInstructionInvokeStatic(Instruction::Code opcode) { |
| return opcode == Instruction::INVOKE_STATIC || opcode == Instruction::INVOKE_STATIC_RANGE; |
| } |
| |
| constexpr bool IsInstructionGoto(Instruction::Code opcode) { |
| return Instruction::GOTO <= opcode && opcode <= Instruction::GOTO_32; |
| } |
| |
| constexpr bool IsInstructionIfCc(Instruction::Code opcode) { |
| return Instruction::IF_EQ <= opcode && opcode <= Instruction::IF_LE; |
| } |
| |
| constexpr bool IsInstructionIfCcZ(Instruction::Code opcode) { |
| return Instruction::IF_EQZ <= opcode && opcode <= Instruction::IF_LEZ; |
| } |
| |
| constexpr bool IsInstructionIGet(Instruction::Code code) { |
| return Instruction::IGET <= code && code <= Instruction::IGET_SHORT; |
| } |
| |
| constexpr bool IsInstructionIPut(Instruction::Code code) { |
| return Instruction::IPUT <= code && code <= Instruction::IPUT_SHORT; |
| } |
| |
| constexpr bool IsInstructionSGet(Instruction::Code code) { |
| return Instruction::SGET <= code && code <= Instruction::SGET_SHORT; |
| } |
| |
| constexpr bool IsInstructionSPut(Instruction::Code code) { |
| return Instruction::SPUT <= code && code <= Instruction::SPUT_SHORT; |
| } |
| |
| constexpr bool IsInstructionAGet(Instruction::Code code) { |
| return Instruction::AGET <= code && code <= Instruction::AGET_SHORT; |
| } |
| |
| constexpr bool IsInstructionAPut(Instruction::Code code) { |
| return Instruction::APUT <= code && code <= Instruction::APUT_SHORT; |
| } |
| |
| constexpr bool IsInstructionIGetOrIPut(Instruction::Code code) { |
| return Instruction::IGET <= code && code <= Instruction::IPUT_SHORT; |
| } |
| |
| constexpr bool IsInstructionIGetQuickOrIPutQuick(Instruction::Code code) { |
| return (code >= Instruction::IGET_QUICK && code <= Instruction::IPUT_OBJECT_QUICK) || |
| (code >= Instruction::IPUT_BOOLEAN_QUICK && code <= Instruction::IGET_SHORT_QUICK); |
| } |
| |
| constexpr bool IsInstructionSGetOrSPut(Instruction::Code code) { |
| return Instruction::SGET <= code && code <= Instruction::SPUT_SHORT; |
| } |
| |
| constexpr bool IsInstructionAGetOrAPut(Instruction::Code code) { |
| return Instruction::AGET <= code && code <= Instruction::APUT_SHORT; |
| } |
| |
| constexpr bool IsInstructionBinOp2Addr(Instruction::Code code) { |
| return Instruction::ADD_INT_2ADDR <= code && code <= Instruction::REM_DOUBLE_2ADDR; |
| } |
| |
| // TODO: Remove the #if guards below when we fully migrate to C++14. |
| |
| constexpr bool IsInvokeInstructionRange(Instruction::Code opcode) { |
| #if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions. |
| DCHECK(IsInstructionInvoke(opcode)); |
| #endif |
| return opcode >= Instruction::INVOKE_VIRTUAL_RANGE; |
| } |
| |
| constexpr DexInvokeType InvokeInstructionType(Instruction::Code opcode) { |
| #if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions. |
| DCHECK(IsInstructionInvoke(opcode)); |
| #endif |
| return static_cast<DexInvokeType>(IsInvokeInstructionRange(opcode) |
| ? (opcode - Instruction::INVOKE_VIRTUAL_RANGE) |
| : (opcode - Instruction::INVOKE_VIRTUAL)); |
| } |
| |
| constexpr DexMemAccessType IGetMemAccessType(Instruction::Code code) { |
| #if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions. |
| DCHECK(IsInstructionIGet(code)); |
| #endif |
| return static_cast<DexMemAccessType>(code - Instruction::IGET); |
| } |
| |
| constexpr DexMemAccessType IPutMemAccessType(Instruction::Code code) { |
| #if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions. |
| DCHECK(IsInstructionIPut(code)); |
| #endif |
| return static_cast<DexMemAccessType>(code - Instruction::IPUT); |
| } |
| |
| constexpr DexMemAccessType SGetMemAccessType(Instruction::Code code) { |
| #if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions. |
| DCHECK(IsInstructionSGet(code)); |
| #endif |
| return static_cast<DexMemAccessType>(code - Instruction::SGET); |
| } |
| |
| constexpr DexMemAccessType SPutMemAccessType(Instruction::Code code) { |
| #if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions. |
| DCHECK(IsInstructionSPut(code)); |
| #endif |
| return static_cast<DexMemAccessType>(code - Instruction::SPUT); |
| } |
| |
| constexpr DexMemAccessType AGetMemAccessType(Instruction::Code code) { |
| #if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions. |
| DCHECK(IsInstructionAGet(code)); |
| #endif |
| return static_cast<DexMemAccessType>(code - Instruction::AGET); |
| } |
| |
| constexpr DexMemAccessType APutMemAccessType(Instruction::Code code) { |
| #if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions. |
| DCHECK(IsInstructionAPut(code)); |
| #endif |
| return static_cast<DexMemAccessType>(code - Instruction::APUT); |
| } |
| |
| constexpr DexMemAccessType IGetOrIPutMemAccessType(Instruction::Code code) { |
| #if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions. |
| DCHECK(IsInstructionIGetOrIPut(code)); |
| #endif |
| return (code >= Instruction::IPUT) ? IPutMemAccessType(code) : IGetMemAccessType(code); |
| } |
| |
| static inline DexMemAccessType IGetQuickOrIPutQuickMemAccessType(Instruction::Code code) { |
| DCHECK(IsInstructionIGetQuickOrIPutQuick(code)); |
| switch (code) { |
| case Instruction::IGET_QUICK: case Instruction::IPUT_QUICK: |
| return kDexMemAccessWord; |
| case Instruction::IGET_WIDE_QUICK: case Instruction::IPUT_WIDE_QUICK: |
| return kDexMemAccessWide; |
| case Instruction::IGET_OBJECT_QUICK: case Instruction::IPUT_OBJECT_QUICK: |
| return kDexMemAccessObject; |
| case Instruction::IGET_BOOLEAN_QUICK: case Instruction::IPUT_BOOLEAN_QUICK: |
| return kDexMemAccessBoolean; |
| case Instruction::IGET_BYTE_QUICK: case Instruction::IPUT_BYTE_QUICK: |
| return kDexMemAccessByte; |
| case Instruction::IGET_CHAR_QUICK: case Instruction::IPUT_CHAR_QUICK: |
| return kDexMemAccessChar; |
| case Instruction::IGET_SHORT_QUICK: case Instruction::IPUT_SHORT_QUICK: |
| return kDexMemAccessShort; |
| default: |
| LOG(FATAL) << code; |
| UNREACHABLE(); |
| } |
| } |
| |
| constexpr DexMemAccessType SGetOrSPutMemAccessType(Instruction::Code code) { |
| #if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions. |
| DCHECK(IsInstructionSGetOrSPut(code)); |
| #endif |
| return (code >= Instruction::SPUT) ? SPutMemAccessType(code) : SGetMemAccessType(code); |
| } |
| |
| constexpr DexMemAccessType AGetOrAPutMemAccessType(Instruction::Code code) { |
| #if __cplusplus >= 201402 // C++14 allows the DCHECK() in constexpr functions. |
| DCHECK(IsInstructionAGetOrAPut(code)); |
| #endif |
| return (code >= Instruction::APUT) ? APutMemAccessType(code) : AGetMemAccessType(code); |
| } |
| |
| } // namespace art |
| |
| #endif // ART_RUNTIME_DEX_INSTRUCTION_UTILS_H_ |