blob: 52a6b81f0e4371d4417888089aae0508532ca874 [file] [log] [blame]
/*
* Copyright (C) 2018 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_OPTIMIZING_INTRINSIC_OBJECTS_H_
#define ART_COMPILER_OPTIMIZING_INTRINSIC_OBJECTS_H_
#include "base/bit_field.h"
#include "base/bit_utils.h"
#include "base/macros.h"
#include "base/mutex.h"
#include "obj_ptr.h"
#include "offsets.h"
namespace art HIDDEN {
class ClassLinker;
class MemberOffset;
class Thread;
namespace mirror {
class Object;
template <class T> class ObjectArray;
} // namespace mirror
#define BOXED_TYPES(V) \
V(Byte, -128, 127, DataType::Type::kInt8, 0) \
V(Short, -128, 127, DataType::Type::kInt16, kByteCacheLastIndex) \
V(Character, 0, 127, DataType::Type::kUint16, kShortCacheLastIndex) \
V(Integer, -128, 127, DataType::Type::kInt32, kCharacterCacheLastIndex)
#define DEFINE_BOXED_CONSTANTS(name, low, high, unused, start_index) \
static constexpr size_t k ##name ##CacheLastIndex = start_index + (high - low + 1); \
static constexpr size_t k ##name ##CacheFirstIndex = start_index;
BOXED_TYPES(DEFINE_BOXED_CONSTANTS)
static constexpr size_t kNumberOfBoxedCaches = kIntegerCacheLastIndex;
#undef DEFINE_BOXED_CONSTANTS
class IntrinsicObjects {
public:
enum class PatchType {
kValueOfObject,
kValueOfArray,
kLast = kValueOfArray
};
static uint32_t EncodePatch(PatchType patch_type, uint32_t index = 0u) {
return PatchTypeField::Encode(static_cast<uint32_t>(patch_type)) | IndexField::Encode(index);
}
static PatchType DecodePatchType(uint32_t intrinsic_data) {
return static_cast<PatchType>(PatchTypeField::Decode(intrinsic_data));
}
static uint32_t DecodePatchIndex(uint32_t intrinsic_data) {
return IndexField::Decode(intrinsic_data);
}
// Helpers returning addresses of objects, suitable for embedding in generated code.
#define DEFINE_BOXED_ACCESSES(name, unused1, unused2, unused3, start_index) \
static ObjPtr<mirror::Object> Get ##name ##ValueOfObject( \
ObjPtr<mirror::ObjectArray<mirror::Object>> boot_image_live_objects, \
uint32_t index) REQUIRES_SHARED(Locks::mutator_lock_) { \
return GetValueOfObject(boot_image_live_objects, k ##name ##CacheFirstIndex, index); \
} \
static MemberOffset Get ##name ##ValueOfArrayDataOffset( \
ObjPtr<mirror::ObjectArray<mirror::Object>> boot_image_live_objects) \
REQUIRES_SHARED(Locks::mutator_lock_) { \
return GetValueOfArrayDataOffset(boot_image_live_objects, k ##name ##CacheFirstIndex); \
}
BOXED_TYPES(DEFINE_BOXED_ACCESSES)
#undef DEFINED_BOXED_ACCESSES
EXPORT static void FillIntrinsicObjects(
ObjPtr<mirror::ObjectArray<mirror::Object>> boot_image_live_objects, size_t start_index)
REQUIRES_SHARED(Locks::mutator_lock_);
static constexpr size_t GetNumberOfIntrinsicObjects() {
return kNumberOfBoxedCaches;
}
EXPORT static ObjPtr<mirror::Object> GetValueOfObject(
ObjPtr<mirror::ObjectArray<mirror::Object>> boot_image_live_objects,
size_t start_index,
uint32_t index) REQUIRES_SHARED(Locks::mutator_lock_);
EXPORT static MemberOffset GetValueOfArrayDataOffset(
ObjPtr<mirror::ObjectArray<mirror::Object>> boot_image_live_objects,
size_t start_index) REQUIRES_SHARED(Locks::mutator_lock_);
private:
static constexpr size_t kPatchTypeBits =
MinimumBitsToStore(static_cast<uint32_t>(PatchType::kLast));
static constexpr size_t kIndexBits = BitSizeOf<uint32_t>() - kPatchTypeBits;
using PatchTypeField = BitField<uint32_t, 0u, kPatchTypeBits>;
using IndexField = BitField<uint32_t, kPatchTypeBits, kIndexBits>;
};
} // namespace art
#endif // ART_COMPILER_OPTIMIZING_INTRINSIC_OBJECTS_H_