blob: 4aff01c066ac8b2b8cb9dead9ff46106048ba4ac [file] [log] [blame]
/*
* Copyright (C) 2013 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_COMPILER_DEX_QUICK_DEX_FILE_METHOD_INLINER_H_
#define ART_COMPILER_DEX_QUICK_DEX_FILE_METHOD_INLINER_H_
#include <stdint.h>
#include "base/mutex.h"
#include "base/macros.h"
#include "safe_map.h"
#include "dex/compiler_enums.h"
#include "dex_file.h"
namespace art {
namespace verifier {
class MethodVerifier;
} // namespace verifier
struct CallInfo;
class Mir2Lir;
enum InlineMethodOpcode : uint16_t {
kIntrinsicDoubleCvt,
kIntrinsicFloatCvt,
kIntrinsicReverseBytes,
kIntrinsicAbsInt,
kIntrinsicAbsLong,
kIntrinsicAbsFloat,
kIntrinsicAbsDouble,
kIntrinsicMinMaxInt,
kIntrinsicSqrt,
kIntrinsicCharAt,
kIntrinsicCompareTo,
kIntrinsicIsEmptyOrLength,
kIntrinsicIndexOf,
kIntrinsicCurrentThread,
kIntrinsicPeek,
kIntrinsicPoke,
kIntrinsicCas,
kIntrinsicUnsafeGet,
kIntrinsicUnsafePut,
kInlineOpNop,
kInlineOpReturnArg,
kInlineOpNonWideConst,
kInlineOpIGet,
kInlineOpIPut,
};
std::ostream& operator<<(std::ostream& os, const InlineMethodOpcode& rhs);
enum InlineMethodFlags : uint16_t {
kNoInlineMethodFlags = 0x0000,
kInlineIntrinsic = 0x0001,
kInlineSpecial = 0x0002,
};
// IntrinsicFlags are stored in InlineMethod::d::raw_data
enum IntrinsicFlags {
kIntrinsicFlagNone = 0,
// kIntrinsicMinMaxInt
kIntrinsicFlagMax = kIntrinsicFlagNone,
kIntrinsicFlagMin = 1,
// kIntrinsicIsEmptyOrLength
kIntrinsicFlagLength = kIntrinsicFlagNone,
kIntrinsicFlagIsEmpty = kIntrinsicFlagMin,
// kIntrinsicIndexOf
kIntrinsicFlagBase0 = kIntrinsicFlagMin,
// kIntrinsicUnsafeGet, kIntrinsicUnsafePut, kIntrinsicUnsafeCas
kIntrinsicFlagIsLong = kIntrinsicFlagMin,
// kIntrinsicUnsafeGet, kIntrinsicUnsafePut
kIntrinsicFlagIsVolatile = 2,
// kIntrinsicUnsafePut, kIntrinsicUnsafeCas
kIntrinsicFlagIsObject = 4,
// kIntrinsicUnsafePut
kIntrinsicFlagIsOrdered = 8,
};
// Check that OpSize fits into 3 bits (at least the values the inliner uses).
COMPILE_ASSERT(kWord < 8 && kLong < 8 && kSingle < 8 && kDouble < 8 && kUnsignedHalf < 8 &&
kSignedHalf < 8 && kUnsignedByte < 8 && kSignedByte < 8, op_size_field_too_narrow);
struct InlineIGetIPutData {
uint16_t op_size : 3; // OpSize
uint16_t is_object : 1;
uint16_t object_arg : 4;
uint16_t src_arg : 4; // iput only
uint16_t method_is_static : 1;
uint16_t reserved : 3;
uint16_t field_idx;
uint32_t is_volatile : 1;
uint32_t field_offset : 31;
};
COMPILE_ASSERT(sizeof(InlineIGetIPutData) == sizeof(uint64_t), InvalidSizeOfInlineIGetIPutData);
struct InlineReturnArgData {
uint16_t arg;
uint16_t op_size : 3; // OpSize
uint16_t is_object : 1;
uint16_t reserved : 12;
uint32_t reserved2;
};
COMPILE_ASSERT(sizeof(InlineReturnArgData) == sizeof(uint64_t), InvalidSizeOfInlineReturnArgData);
struct InlineMethod {
InlineMethodOpcode opcode;
InlineMethodFlags flags;
union {
uint64_t data;
InlineIGetIPutData ifield_data;
InlineReturnArgData return_data;
} d;
};
/**
* Handles inlining of methods from a particular DexFile.
*
* Intrinsics are a special case of inline methods. The DexFile indices for
* all the supported intrinsic methods are looked up once by the FindIntrinsics
* function and cached by this class for quick lookup by the method index.
*
* TODO: Detect short methods (at least getters, setters and empty functions)
* from the verifier and mark them for inlining. Inline these methods early
* during compilation to allow further optimizations. Similarly, provide
* additional information about intrinsics to the early phases of compilation.
*/
class DexFileMethodInliner {
public:
DexFileMethodInliner();
~DexFileMethodInliner();
/**
* Analyse method code to determine if the method is a candidate for inlining.
* If it is, record its data for later.
*
* @param verifier the method verifier holding data about the method to analyse.
* @return true if the method is a candidate for inlining, false otherwise.
*/
bool AnalyseMethodCode(verifier::MethodVerifier* verifier)
SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) LOCKS_EXCLUDED(lock_);
/**
* Analyse method code to determine if the method is a candidate for inlining.
* If it is, record the inlining data.
*
* @param verifier the method verifier holding data about the method to analyse.
* @param method placeholder for the inline method data.
* @return true if the method is a candidate for inlining, false otherwise.
*/
bool AnalyseMethodCode(verifier::MethodVerifier* verifier, InlineMethod* method)
SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) LOCKS_EXCLUDED(lock_);
/**
* Check whether a particular method index corresponds to an intrinsic function.
*/
bool IsIntrinsic(uint32_t method_index) LOCKS_EXCLUDED(lock_);
/**
* Generate code for an intrinsic function invocation.
*/
bool GenIntrinsic(Mir2Lir* backend, CallInfo* info) LOCKS_EXCLUDED(lock_);
/**
* Check whether a particular method index corresponds to a special function.
*/
bool IsSpecial(uint32_t method_index) LOCKS_EXCLUDED(lock_);
/**
* Generate code for a special function.
*/
bool GenSpecial(Mir2Lir* backend, uint32_t method_idx);
/**
* To avoid multiple lookups of a class by its descriptor, we cache its
* type index in the IndexCache. These are the indexes into the IndexCache
* class_indexes array.
*/
enum ClassCacheIndex : uint8_t { // unit8_t to save space, make larger if needed
kClassCacheFirst = 0,
kClassCacheBoolean = kClassCacheFirst,
kClassCacheByte,
kClassCacheChar,
kClassCacheShort,
kClassCacheInt,
kClassCacheLong,
kClassCacheFloat,
kClassCacheDouble,
kClassCacheVoid,
kClassCacheJavaLangObject,
kClassCacheJavaLangString,
kClassCacheJavaLangDouble,
kClassCacheJavaLangFloat,
kClassCacheJavaLangInteger,
kClassCacheJavaLangLong,
kClassCacheJavaLangShort,
kClassCacheJavaLangMath,
kClassCacheJavaLangStrictMath,
kClassCacheJavaLangThread,
kClassCacheLibcoreIoMemory,
kClassCacheSunMiscUnsafe,
kClassCacheLast
};
/**
* To avoid multiple lookups of a method name string, we cache its string
* index in the IndexCache. These are the indexes into the IndexCache
* name_indexes array.
*/
enum NameCacheIndex : uint8_t { // unit8_t to save space, make larger if needed
kNameCacheFirst = 0,
kNameCacheReverseBytes = kNameCacheFirst,
kNameCacheDoubleToRawLongBits,
kNameCacheLongBitsToDouble,
kNameCacheFloatToRawIntBits,
kNameCacheIntBitsToFloat,
kNameCacheAbs,
kNameCacheMax,
kNameCacheMin,
kNameCacheSqrt,
kNameCacheCharAt,
kNameCacheCompareTo,
kNameCacheIsEmpty,
kNameCacheIndexOf,
kNameCacheLength,
kNameCacheCurrentThread,
kNameCachePeekByte,
kNameCachePeekIntNative,
kNameCachePeekLongNative,
kNameCachePeekShortNative,
kNameCachePokeByte,
kNameCachePokeIntNative,
kNameCachePokeLongNative,
kNameCachePokeShortNative,
kNameCacheCompareAndSwapInt,
kNameCacheCompareAndSwapLong,
kNameCacheCompareAndSwapObject,
kNameCacheGetInt,
kNameCacheGetIntVolatile,
kNameCachePutInt,
kNameCachePutIntVolatile,
kNameCachePutOrderedInt,
kNameCacheGetLong,
kNameCacheGetLongVolatile,
kNameCachePutLong,
kNameCachePutLongVolatile,
kNameCachePutOrderedLong,
kNameCacheGetObject,
kNameCacheGetObjectVolatile,
kNameCachePutObject,
kNameCachePutObjectVolatile,
kNameCachePutOrderedObject,
kNameCacheLast
};
/**
* To avoid multiple lookups of a method signature, we cache its proto
* index in the IndexCache. These are the indexes into the IndexCache
* proto_indexes array.
*/
enum ProtoCacheIndex : uint8_t { // unit8_t to save space, make larger if needed
kProtoCacheFirst = 0,
kProtoCacheI_I = kProtoCacheFirst,
kProtoCacheJ_J,
kProtoCacheS_S,
kProtoCacheD_D,
kProtoCacheF_F,
kProtoCacheD_J,
kProtoCacheJ_D,
kProtoCacheF_I,
kProtoCacheI_F,
kProtoCacheII_I,
kProtoCacheI_C,
kProtoCacheString_I,
kProtoCache_Z,
kProtoCache_I,
kProtoCache_Thread,
kProtoCacheJ_B,
kProtoCacheJ_I,
kProtoCacheJ_S,
kProtoCacheJB_V,
kProtoCacheJI_V,
kProtoCacheJJ_V,
kProtoCacheJS_V,
kProtoCacheObjectJII_Z,
kProtoCacheObjectJJJ_Z,
kProtoCacheObjectJObjectObject_Z,
kProtoCacheObjectJ_I,
kProtoCacheObjectJI_V,
kProtoCacheObjectJ_J,
kProtoCacheObjectJJ_V,
kProtoCacheObjectJ_Object,
kProtoCacheObjectJObject_V,
kProtoCacheLast
};
private:
/**
* The maximum number of method parameters we support in the ProtoDef.
*/
static constexpr uint32_t kProtoMaxParams = 6;
/**
* The method signature (proto) definition using cached class indexes.
* The return_type and params are used with the IndexCache to look up
* appropriate class indexes to be passed to DexFile::FindProtoId().
*/
struct ProtoDef {
ClassCacheIndex return_type;
uint8_t param_count;
ClassCacheIndex params[kProtoMaxParams];
};
/**
* The method definition using cached class, name and proto indexes.
* The class index, method name index and proto index are used with
* IndexCache to look up appropriate parameters for DexFile::FindMethodId().
*/
struct MethodDef {
ClassCacheIndex declaring_class;
NameCacheIndex name;
ProtoCacheIndex proto;
};
/**
* The definition of an intrinsic function binds the method definition
* to an Intrinsic.
*/
struct IntrinsicDef {
MethodDef method_def;
InlineMethod intrinsic;
};
/**
* Cache for class, method name and method signature indexes used during
* intrinsic function lookup to avoid multiple lookups of the same items.
*
* Many classes have multiple intrinsics and/or they are used in multiple
* method signatures and we want to avoid repeated lookups since they are
* not exactly cheap. The method names and method signatures are sometimes
* reused and therefore cached as well.
*/
struct IndexCache {
IndexCache();
uint32_t class_indexes[kClassCacheLast - kClassCacheFirst];
uint32_t name_indexes[kNameCacheLast - kNameCacheFirst];
uint32_t proto_indexes[kProtoCacheLast - kProtoCacheFirst];
};
static const char* const kClassCacheNames[];
static const char* const kNameCacheNames[];
static const ProtoDef kProtoCacheDefs[];
static const IntrinsicDef kIntrinsicMethods[];
static const uint32_t kIndexNotFound = static_cast<uint32_t>(-1);
static const uint32_t kIndexUnresolved = static_cast<uint32_t>(-2);
static uint32_t FindClassIndex(const DexFile* dex_file, IndexCache* cache,
ClassCacheIndex index);
static uint32_t FindNameIndex(const DexFile* dex_file, IndexCache* cache,
NameCacheIndex index);
static uint32_t FindProtoIndex(const DexFile* dex_file, IndexCache* cache,
ProtoCacheIndex index);
static uint32_t FindMethodIndex(const DexFile* dex_file, IndexCache* cache,
const MethodDef& method_def);
/**
* Find all known intrinsic methods in the dex_file and cache their indices.
*
* Only DexFileToMethodInlinerMap may call this function to initialize the inliner.
*/
void FindIntrinsics(const DexFile* dex_file) EXCLUSIVE_LOCKS_REQUIRED(lock_);
friend class DexFileToMethodInlinerMap;
bool AddInlineMethod(int32_t method_idx, const InlineMethod& method) LOCKS_EXCLUDED(lock_);
static bool AnalyseReturnMethod(const DexFile::CodeItem* code_item, InlineMethod* result);
static bool AnalyseConstMethod(const DexFile::CodeItem* code_item, InlineMethod* result);
static bool AnalyseIGetMethod(verifier::MethodVerifier* verifier, InlineMethod* result)
SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
static bool AnalyseIPutMethod(verifier::MethodVerifier* verifier, InlineMethod* result)
SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
ReaderWriterMutex lock_;
/*
* Maps method indexes (for the particular DexFile) to Intrinsic defintions.
*/
SafeMap<uint32_t, InlineMethod> inline_methods_ GUARDED_BY(lock_);
const DexFile* dex_file_;
DISALLOW_COPY_AND_ASSIGN(DexFileMethodInliner);
};
} // namespace art
#endif // ART_COMPILER_DEX_QUICK_DEX_FILE_METHOD_INLINER_H_