blob: a85d0332c31d05ad51b9e31f8a7217432077f37f [file] [log] [blame]
Vladimir Marko05792b92015-08-03 11:56:49 +01001/*
2 * Copyright (C) 2015 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_UTILS_DEX_CACHE_ARRAYS_LAYOUT_INL_H_
18#define ART_RUNTIME_UTILS_DEX_CACHE_ARRAYS_LAYOUT_INL_H_
19
20#include "dex_cache_arrays_layout.h"
21
22#include "base/bit_utils.h"
23#include "base/logging.h"
24#include "gc_root.h"
25#include "globals.h"
Christina Wadsworthbf44e0e2016-08-18 10:37:42 -070026#include "mirror/dex_cache.h"
Vladimir Marko05792b92015-08-03 11:56:49 +010027#include "primitive.h"
28
29namespace art {
30
Andreas Gampe542451c2016-07-26 09:02:02 -070031inline DexCacheArraysLayout::DexCacheArraysLayout(PointerSize pointer_size,
Vladimir Marko09d09432015-09-08 13:47:48 +010032 const DexFile::Header& header)
Vladimir Marko05792b92015-08-03 11:56:49 +010033 : pointer_size_(pointer_size),
34 /* types_offset_ is always 0u, so it's constexpr */
Vladimir Marko0d4909e2016-02-02 20:27:08 +000035 methods_offset_(
36 RoundUp(types_offset_ + TypesSize(header.type_ids_size_), MethodsAlignment())),
37 strings_offset_(
38 RoundUp(methods_offset_ + MethodsSize(header.method_ids_size_), StringsAlignment())),
39 fields_offset_(
40 RoundUp(strings_offset_ + StringsSize(header.string_ids_size_), FieldsAlignment())),
41 size_(
42 RoundUp(fields_offset_ + FieldsSize(header.field_ids_size_), Alignment())) {
Vladimir Marko05792b92015-08-03 11:56:49 +010043}
44
Andreas Gampe542451c2016-07-26 09:02:02 -070045inline DexCacheArraysLayout::DexCacheArraysLayout(PointerSize pointer_size, const DexFile* dex_file)
Vladimir Marko09d09432015-09-08 13:47:48 +010046 : DexCacheArraysLayout(pointer_size, dex_file->GetHeader()) {
47}
48
Christina Wadsworthbf44e0e2016-08-18 10:37:42 -070049inline constexpr size_t DexCacheArraysLayout::Alignment() {
Vladimir Marko05792b92015-08-03 11:56:49 +010050 // GcRoot<> alignment is 4, i.e. lower than or equal to the pointer alignment.
51 static_assert(alignof(GcRoot<mirror::Class>) == 4, "Expecting alignof(GcRoot<>) == 4");
Christina Wadsworthbf44e0e2016-08-18 10:37:42 -070052 static_assert(alignof(mirror::StringDexCacheType) == 8, "Expecting alignof(StringDexCacheType) == 8");
Christina Wadsworth9210ce92016-08-19 13:28:19 -070053 return alignof(mirror::StringDexCacheType);
Andreas Gampe542451c2016-07-26 09:02:02 -070054}
55
56template <typename T>
57static constexpr PointerSize GcRootAsPointerSize() {
Andreas Gampebda1d602016-08-29 17:43:45 -070058 static_assert(sizeof(GcRoot<T>) == 4U, "Unexpected GcRoot size");
59 return PointerSize::k32;
Vladimir Marko05792b92015-08-03 11:56:49 +010060}
61
62inline size_t DexCacheArraysLayout::TypeOffset(uint32_t type_idx) const {
Andreas Gampe542451c2016-07-26 09:02:02 -070063 return types_offset_ + ElementOffset(GcRootAsPointerSize<mirror::Class>(), type_idx);
Vladimir Marko05792b92015-08-03 11:56:49 +010064}
65
66inline size_t DexCacheArraysLayout::TypesSize(size_t num_elements) const {
Mathieu Chartierfbc31082016-01-24 11:59:56 -080067 // App image patching relies on having enough room for a forwarding pointer in the types array.
68 // See FixupArtMethodArrayVisitor and ClassLinker::AddImageSpace.
Andreas Gampe542451c2016-07-26 09:02:02 -070069 return std::max(ArraySize(GcRootAsPointerSize<mirror::Class>(), num_elements),
70 static_cast<size_t>(pointer_size_));
Vladimir Marko05792b92015-08-03 11:56:49 +010071}
72
73inline size_t DexCacheArraysLayout::TypesAlignment() const {
74 return alignof(GcRoot<mirror::Class>);
75}
76
77inline size_t DexCacheArraysLayout::MethodOffset(uint32_t method_idx) const {
78 return methods_offset_ + ElementOffset(pointer_size_, method_idx);
79}
80
81inline size_t DexCacheArraysLayout::MethodsSize(size_t num_elements) const {
Mathieu Chartierfbc31082016-01-24 11:59:56 -080082 // App image patching relies on having enough room for a forwarding pointer in the methods array.
Andreas Gampe542451c2016-07-26 09:02:02 -070083 return std::max(ArraySize(pointer_size_, num_elements), static_cast<size_t>(pointer_size_));
Vladimir Marko05792b92015-08-03 11:56:49 +010084}
85
86inline size_t DexCacheArraysLayout::MethodsAlignment() const {
Andreas Gampe542451c2016-07-26 09:02:02 -070087 return static_cast<size_t>(pointer_size_);
Vladimir Marko05792b92015-08-03 11:56:49 +010088}
89
90inline size_t DexCacheArraysLayout::StringOffset(uint32_t string_idx) const {
Christina Wadsworthbf44e0e2016-08-18 10:37:42 -070091 return strings_offset_ + ElementOffset(PointerSize::k64,
92 string_idx % mirror::DexCache::kDexCacheStringCacheSize);
Vladimir Marko05792b92015-08-03 11:56:49 +010093}
94
95inline size_t DexCacheArraysLayout::StringsSize(size_t num_elements) const {
Christina Wadsworthbf44e0e2016-08-18 10:37:42 -070096 size_t cache_size = mirror::DexCache::kDexCacheStringCacheSize;
97 if (num_elements < cache_size) {
98 cache_size = num_elements;
99 }
100 return ArraySize(PointerSize::k64, cache_size);
Vladimir Marko05792b92015-08-03 11:56:49 +0100101}
102
103inline size_t DexCacheArraysLayout::StringsAlignment() const {
Christina Wadsworth6353a632016-08-19 15:58:05 -0700104 static_assert(alignof(mirror::StringDexCacheType) == 8,
105 "Expecting alignof(StringDexCacheType) == 8");
Christina Wadsworth9210ce92016-08-19 13:28:19 -0700106 return alignof(mirror::StringDexCacheType);
Vladimir Marko05792b92015-08-03 11:56:49 +0100107}
108
109inline size_t DexCacheArraysLayout::FieldOffset(uint32_t field_idx) const {
110 return fields_offset_ + ElementOffset(pointer_size_, field_idx);
111}
112
113inline size_t DexCacheArraysLayout::FieldsSize(size_t num_elements) const {
114 return ArraySize(pointer_size_, num_elements);
115}
116
117inline size_t DexCacheArraysLayout::FieldsAlignment() const {
Andreas Gampe542451c2016-07-26 09:02:02 -0700118 return static_cast<size_t>(pointer_size_);
Vladimir Marko05792b92015-08-03 11:56:49 +0100119}
120
Andreas Gampe542451c2016-07-26 09:02:02 -0700121inline size_t DexCacheArraysLayout::ElementOffset(PointerSize element_size, uint32_t idx) {
122 return static_cast<size_t>(element_size) * idx;
Vladimir Marko05792b92015-08-03 11:56:49 +0100123}
124
Andreas Gampe542451c2016-07-26 09:02:02 -0700125inline size_t DexCacheArraysLayout::ArraySize(PointerSize element_size, uint32_t num_elements) {
126 return static_cast<size_t>(element_size) * num_elements;
Vladimir Marko05792b92015-08-03 11:56:49 +0100127}
128
129} // namespace art
130
131#endif // ART_RUNTIME_UTILS_DEX_CACHE_ARRAYS_LAYOUT_INL_H_