blob: 6f88cc5df450b230a2a5151cf392e2c256ab1a27 [file] [log] [blame]
Elliott Hughes2faa5f12012-01-30 14:42:07 -08001/*
2 * Copyright (C) 2011 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 */
Brian Carlstrom7e49dca2011-07-22 18:07:34 -070016
Brian Carlstromfc0e3212013-07-17 14:40:12 -070017#ifndef ART_RUNTIME_MIRROR_DEX_CACHE_H_
18#define ART_RUNTIME_MIRROR_DEX_CACHE_H_
Brian Carlstrom7e49dca2011-07-22 18:07:34 -070019
Mathieu Chartiere401d142015-04-22 13:56:20 -070020#include "array.h"
Andreas Gampe58a5af82014-07-31 16:23:49 -070021#include "art_field.h"
Andreas Gampe58a5af82014-07-31 16:23:49 -070022#include "class.h"
Andreas Gampea5b09a62016-11-17 15:21:22 -080023#include "dex_file_types.h"
Brian Carlstrom7e49dca2011-07-22 18:07:34 -070024#include "object.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080025#include "object_array.h"
Brian Carlstrom7e49dca2011-07-22 18:07:34 -070026
27namespace art {
28
Alex Lightdba61482016-12-21 08:20:29 -080029class ArtMethod;
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080030struct DexCacheOffsets;
31class DexFile;
32class ImageWriter;
33union JValue;
Andreas Gampecc1b5352016-12-01 16:58:38 -080034class LinearAlloc;
35class Thread;
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080036
37namespace mirror {
38
Narayan Kamath25352fc2016-08-03 12:46:58 +010039class MethodType;
Mingyao Yang98d1cc82014-05-15 17:02:16 -070040class String;
Brian Carlstrom7e49dca2011-07-22 18:07:34 -070041
Narayan Kamathc38a6f82016-09-29 17:07:20 +010042template <typename T> struct PACKED(8) DexCachePair {
43 GcRoot<T> object;
44 uint32_t index;
Christina Wadsworthbf44e0e2016-08-18 10:37:42 -070045 // The array is initially [ {0,0}, {0,0}, {0,0} ... ]
46 // We maintain the invariant that once a dex cache entry is populated,
47 // the pointer is always non-0
48 // Any given entry would thus be:
49 // {non-0, non-0} OR {0,0}
50 //
51 // It's generally sufficiently enough then to check if the
Narayan Kamathc38a6f82016-09-29 17:07:20 +010052 // lookup index matches the stored index (for a >0 lookup index)
Christina Wadsworthbf44e0e2016-08-18 10:37:42 -070053 // because if it's true the pointer is also non-null.
54 //
55 // For the 0th entry which is a special case, the value is either
56 // {0,0} (initial state) or {non-0, 0} which indicates
Narayan Kamathc38a6f82016-09-29 17:07:20 +010057 // that a valid object is stored at that index for a dex section id of 0.
Christina Wadsworthbf44e0e2016-08-18 10:37:42 -070058 //
Narayan Kamathc38a6f82016-09-29 17:07:20 +010059 // As an optimization, we want to avoid branching on the object pointer since
60 // it's always non-null if the id branch succeeds (except for the 0th id).
Christina Wadsworthbf44e0e2016-08-18 10:37:42 -070061 // Set the initial state for the 0th entry to be {0,1} which is guaranteed to fail
Narayan Kamathc38a6f82016-09-29 17:07:20 +010062 // the lookup id == stored id branch.
63 DexCachePair(T* object, uint32_t index)
64 : object(object),
65 index(index) {}
66 DexCachePair() = default;
67 DexCachePair(const DexCachePair<T>&) = default;
68 DexCachePair& operator=(const DexCachePair<T>&) = default;
Mathieu Chartierbb816d62016-09-07 10:17:46 -070069
Narayan Kamathc38a6f82016-09-29 17:07:20 +010070 static void Initialize(std::atomic<DexCachePair<T>>* dex_cache) {
71 DexCachePair<T> first_elem;
72 first_elem.object = GcRoot<T>(nullptr);
73 first_elem.index = InvalidIndexForSlot(0);
74 dex_cache[0].store(first_elem, std::memory_order_relaxed);
Christina Wadsworthbf44e0e2016-08-18 10:37:42 -070075 }
Mathieu Chartierbb816d62016-09-07 10:17:46 -070076
Narayan Kamathc38a6f82016-09-29 17:07:20 +010077 static GcRoot<T> Lookup(std::atomic<DexCachePair<T>>* dex_cache,
78 uint32_t idx,
79 uint32_t cache_size) {
Narayan Kamath000e1882016-10-24 17:14:25 +010080 DCHECK_NE(cache_size, 0u);
Narayan Kamathc38a6f82016-09-29 17:07:20 +010081 DexCachePair<T> element = dex_cache[idx % cache_size].load(std::memory_order_relaxed);
82 if (idx != element.index) {
83 return GcRoot<T>(nullptr);
84 }
85
86 DCHECK(!element.object.IsNull());
87 return element.object;
Christina Wadsworthbf44e0e2016-08-18 10:37:42 -070088 }
Mathieu Chartierbb816d62016-09-07 10:17:46 -070089
Narayan Kamath23136d12016-09-30 16:29:19 +010090 static void Assign(std::atomic<DexCachePair<T>>* dex_cache,
91 uint32_t idx,
92 T* object,
93 uint32_t cache_size) {
94 DCHECK_LT(idx % cache_size, cache_size);
95 dex_cache[idx % cache_size].store(
96 DexCachePair<T>(object, idx), std::memory_order_relaxed);
97 }
98
Narayan Kamathc38a6f82016-09-29 17:07:20 +010099 static uint32_t InvalidIndexForSlot(uint32_t slot) {
Mathieu Chartierbb816d62016-09-07 10:17:46 -0700100 // Since the cache size is a power of two, 0 will always map to slot 0.
101 // Use 1 for slot 0 and 0 for all other slots.
102 return (slot == 0) ? 1u : 0u;
103 }
Christina Wadsworthbf44e0e2016-08-18 10:37:42 -0700104};
Narayan Kamathc38a6f82016-09-29 17:07:20 +0100105
106using StringDexCachePair = DexCachePair<mirror::String>;
Christina Wadsworthbf44e0e2016-08-18 10:37:42 -0700107using StringDexCacheType = std::atomic<StringDexCachePair>;
108
Narayan Kamath25352fc2016-08-03 12:46:58 +0100109using MethodTypeDexCachePair = DexCachePair<mirror::MethodType>;
110using MethodTypeDexCacheType = std::atomic<MethodTypeDexCachePair>;
111
Mingyao Yang98d1cc82014-05-15 17:02:16 -0700112// C++ mirror of java.lang.DexCache.
113class MANAGED DexCache FINAL : public Object {
Brian Carlstrom83db7722011-08-26 17:32:56 -0700114 public:
Mingyao Yang98d1cc82014-05-15 17:02:16 -0700115 // Size of java.lang.DexCache.class.
Andreas Gampe542451c2016-07-26 09:02:02 -0700116 static uint32_t ClassSize(PointerSize pointer_size);
Mingyao Yang98d1cc82014-05-15 17:02:16 -0700117
Christina Wadsworthbf44e0e2016-08-18 10:37:42 -0700118 // Size of string dex cache. Needs to be a power of 2 for entrypoint assumptions to hold.
119 static constexpr size_t kDexCacheStringCacheSize = 1024;
120 static_assert(IsPowerOfTwo(kDexCacheStringCacheSize),
121 "String dex cache size is not a power of 2.");
122
Narayan Kamath25352fc2016-08-03 12:46:58 +0100123 // Size of method type dex cache. Needs to be a power of 2 for entrypoint assumptions
124 // to hold.
125 static constexpr size_t kDexCacheMethodTypeCacheSize = 1024;
126 static_assert(IsPowerOfTwo(kDexCacheMethodTypeCacheSize),
127 "MethodType dex cache size is not a power of 2.");
128
Christina Wadsworthbf44e0e2016-08-18 10:37:42 -0700129 static constexpr size_t StaticStringSize() {
130 return kDexCacheStringCacheSize;
131 }
132
Narayan Kamath25352fc2016-08-03 12:46:58 +0100133 static constexpr size_t StaticMethodTypeSize() {
134 return kDexCacheMethodTypeCacheSize;
135 }
136
Mingyao Yang98d1cc82014-05-15 17:02:16 -0700137 // Size of an instance of java.lang.DexCache not including referenced values.
138 static constexpr uint32_t InstanceSize() {
139 return sizeof(DexCache);
140 }
141
Andreas Gampecc1b5352016-12-01 16:58:38 -0800142 static void InitializeDexCache(Thread* self,
143 ObjPtr<mirror::DexCache> dex_cache,
144 ObjPtr<mirror::String> location,
145 const DexFile* dex_file,
146 LinearAlloc* linear_alloc,
147 PointerSize image_pointer_size)
148 REQUIRES_SHARED(Locks::mutator_lock_)
149 REQUIRES(Locks::dex_lock_);
Brian Carlstrom7e49dca2011-07-22 18:07:34 -0700150
Andreas Gampe542451c2016-07-26 09:02:02 -0700151 void Fixup(ArtMethod* trampoline, PointerSize pointer_size)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700152 REQUIRES_SHARED(Locks::mutator_lock_);
Ian Rogers19846512012-02-24 11:42:47 -0800153
Mathieu Chartier60bc39c2016-01-27 18:37:48 -0800154 template <ReadBarrierOption kReadBarrierOption = kWithReadBarrier, typename Visitor>
Christina Wadsworthbf44e0e2016-08-18 10:37:42 -0700155 void FixupStrings(StringDexCacheType* dest, const Visitor& visitor)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700156 REQUIRES_SHARED(Locks::mutator_lock_);
Mathieu Chartier4b00d342015-11-13 10:42:08 -0800157
Mathieu Chartier60bc39c2016-01-27 18:37:48 -0800158 template <ReadBarrierOption kReadBarrierOption = kWithReadBarrier, typename Visitor>
Mathieu Chartier4b00d342015-11-13 10:42:08 -0800159 void FixupResolvedTypes(GcRoot<mirror::Class>* dest, const Visitor& visitor)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700160 REQUIRES_SHARED(Locks::mutator_lock_);
Mathieu Chartier4b00d342015-11-13 10:42:08 -0800161
Narayan Kamath7fe56582016-10-14 18:49:12 +0100162 template <ReadBarrierOption kReadBarrierOption = kWithReadBarrier, typename Visitor>
163 void FixupResolvedMethodTypes(MethodTypeDexCacheType* dest, const Visitor& visitor)
164 REQUIRES_SHARED(Locks::mutator_lock_);
165
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700166 String* GetLocation() REQUIRES_SHARED(Locks::mutator_lock_) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700167 return GetFieldObject<String>(OFFSET_OF_OBJECT_MEMBER(DexCache, location_));
Brian Carlstroma663ea52011-08-19 23:33:41 -0700168 }
169
Andreas Gampedd9d0552015-03-09 12:57:41 -0700170 static MemberOffset DexOffset() {
171 return OFFSET_OF_OBJECT_MEMBER(DexCache, dex_);
172 }
173
buzbee5cd21802011-08-26 10:40:14 -0700174 static MemberOffset StringsOffset() {
Mathieu Chartier66f19252012-09-18 08:57:04 -0700175 return OFFSET_OF_OBJECT_MEMBER(DexCache, strings_);
buzbeec5ef0462011-08-25 18:44:49 -0700176 }
177
Vladimir Marko05792b92015-08-03 11:56:49 +0100178 static MemberOffset ResolvedTypesOffset() {
179 return OFFSET_OF_OBJECT_MEMBER(DexCache, resolved_types_);
180 }
181
Brian Carlstrom1caa2c22011-08-28 13:02:33 -0700182 static MemberOffset ResolvedFieldsOffset() {
Mathieu Chartier66f19252012-09-18 08:57:04 -0700183 return OFFSET_OF_OBJECT_MEMBER(DexCache, resolved_fields_);
buzbeec5ef0462011-08-25 18:44:49 -0700184 }
185
Brian Carlstrom1caa2c22011-08-28 13:02:33 -0700186 static MemberOffset ResolvedMethodsOffset() {
Mathieu Chartier66f19252012-09-18 08:57:04 -0700187 return OFFSET_OF_OBJECT_MEMBER(DexCache, resolved_methods_);
buzbeec5ef0462011-08-25 18:44:49 -0700188 }
189
Narayan Kamath25352fc2016-08-03 12:46:58 +0100190 static MemberOffset ResolvedMethodTypesOffset() {
191 return OFFSET_OF_OBJECT_MEMBER(DexCache, resolved_method_types_);
192 }
193
Vladimir Marko05792b92015-08-03 11:56:49 +0100194 static MemberOffset NumStringsOffset() {
195 return OFFSET_OF_OBJECT_MEMBER(DexCache, num_strings_);
Brian Carlstrom7e49dca2011-07-22 18:07:34 -0700196 }
197
Vladimir Marko05792b92015-08-03 11:56:49 +0100198 static MemberOffset NumResolvedTypesOffset() {
199 return OFFSET_OF_OBJECT_MEMBER(DexCache, num_resolved_types_);
Brian Carlstrom7e49dca2011-07-22 18:07:34 -0700200 }
201
Vladimir Marko05792b92015-08-03 11:56:49 +0100202 static MemberOffset NumResolvedFieldsOffset() {
203 return OFFSET_OF_OBJECT_MEMBER(DexCache, num_resolved_fields_);
Brian Carlstrom7e49dca2011-07-22 18:07:34 -0700204 }
205
Vladimir Marko05792b92015-08-03 11:56:49 +0100206 static MemberOffset NumResolvedMethodsOffset() {
207 return OFFSET_OF_OBJECT_MEMBER(DexCache, num_resolved_methods_);
Brian Carlstrom7e49dca2011-07-22 18:07:34 -0700208 }
209
Narayan Kamath25352fc2016-08-03 12:46:58 +0100210 static MemberOffset NumResolvedMethodTypesOffset() {
211 return OFFSET_OF_OBJECT_MEMBER(DexCache, num_resolved_method_types_);
212 }
213
Andreas Gampe8a0128a2016-11-28 07:38:35 -0800214 mirror::String* GetResolvedString(dex::StringIndex string_idx) ALWAYS_INLINE
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700215 REQUIRES_SHARED(Locks::mutator_lock_);
Brian Carlstrom7e49dca2011-07-22 18:07:34 -0700216
Andreas Gampe8a0128a2016-11-28 07:38:35 -0800217 void SetResolvedString(dex::StringIndex string_idx, ObjPtr<mirror::String> resolved) ALWAYS_INLINE
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700218 REQUIRES_SHARED(Locks::mutator_lock_);
Brian Carlstrom7e49dca2011-07-22 18:07:34 -0700219
Mathieu Chartierbb816d62016-09-07 10:17:46 -0700220 // Clear a string for a string_idx, used to undo string intern transactions to make sure
221 // the string isn't kept live.
Andreas Gampe8a0128a2016-11-28 07:38:35 -0800222 void ClearString(dex::StringIndex string_idx) REQUIRES_SHARED(Locks::mutator_lock_);
Mathieu Chartierbb816d62016-09-07 10:17:46 -0700223
Andreas Gampea5b09a62016-11-17 15:21:22 -0800224 Class* GetResolvedType(dex::TypeIndex type_idx) REQUIRES_SHARED(Locks::mutator_lock_);
Vladimir Marko05792b92015-08-03 11:56:49 +0100225
Andreas Gampea5b09a62016-11-17 15:21:22 -0800226 void SetResolvedType(dex::TypeIndex type_idx, ObjPtr<Class> resolved)
Mathieu Chartier31e88222016-10-14 18:43:19 -0700227 REQUIRES_SHARED(Locks::mutator_lock_);
Vladimir Marko05792b92015-08-03 11:56:49 +0100228
Andreas Gampe542451c2016-07-26 09:02:02 -0700229 ALWAYS_INLINE ArtMethod* GetResolvedMethod(uint32_t method_idx, PointerSize ptr_size)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700230 REQUIRES_SHARED(Locks::mutator_lock_);
Brian Carlstrom7e49dca2011-07-22 18:07:34 -0700231
Andreas Gampe542451c2016-07-26 09:02:02 -0700232 ALWAYS_INLINE void SetResolvedMethod(uint32_t method_idx,
233 ArtMethod* resolved,
234 PointerSize ptr_size)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700235 REQUIRES_SHARED(Locks::mutator_lock_);
Brian Carlstrom7e49dca2011-07-22 18:07:34 -0700236
Mathieu Chartierc7853442015-03-27 14:35:38 -0700237 // Pointer sized variant, used for patching.
Andreas Gampe542451c2016-07-26 09:02:02 -0700238 ALWAYS_INLINE ArtField* GetResolvedField(uint32_t idx, PointerSize ptr_size)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700239 REQUIRES_SHARED(Locks::mutator_lock_);
Mathieu Chartiere401d142015-04-22 13:56:20 -0700240
241 // Pointer sized variant, used for patching.
Andreas Gampe542451c2016-07-26 09:02:02 -0700242 ALWAYS_INLINE void SetResolvedField(uint32_t idx, ArtField* field, PointerSize ptr_size)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700243 REQUIRES_SHARED(Locks::mutator_lock_);
Brian Carlstrom7e49dca2011-07-22 18:07:34 -0700244
Narayan Kamath25352fc2016-08-03 12:46:58 +0100245 MethodType* GetResolvedMethodType(uint32_t proto_idx) REQUIRES_SHARED(Locks::mutator_lock_);
246
247 void SetResolvedMethodType(uint32_t proto_idx, MethodType* resolved) REQUIRES_SHARED(Locks::mutator_lock_);
248
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700249 StringDexCacheType* GetStrings() ALWAYS_INLINE REQUIRES_SHARED(Locks::mutator_lock_) {
Christina Wadsworthbf44e0e2016-08-18 10:37:42 -0700250 return GetFieldPtr64<StringDexCacheType*>(StringsOffset());
Brian Carlstrom1caa2c22011-08-28 13:02:33 -0700251 }
Brian Carlstrom83db7722011-08-26 17:32:56 -0700252
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700253 void SetStrings(StringDexCacheType* strings) ALWAYS_INLINE REQUIRES_SHARED(Locks::mutator_lock_) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800254 SetFieldPtr<false>(StringsOffset(), strings);
255 }
256
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700257 GcRoot<Class>* GetResolvedTypes() ALWAYS_INLINE REQUIRES_SHARED(Locks::mutator_lock_) {
Vladimir Marko05792b92015-08-03 11:56:49 +0100258 return GetFieldPtr<GcRoot<Class>*>(ResolvedTypesOffset());
Mathieu Chartier66f19252012-09-18 08:57:04 -0700259 }
260
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800261 void SetResolvedTypes(GcRoot<Class>* resolved_types)
262 ALWAYS_INLINE
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700263 REQUIRES_SHARED(Locks::mutator_lock_) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800264 SetFieldPtr<false>(ResolvedTypesOffset(), resolved_types);
265 }
266
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700267 ArtMethod** GetResolvedMethods() ALWAYS_INLINE REQUIRES_SHARED(Locks::mutator_lock_) {
Vladimir Marko05792b92015-08-03 11:56:49 +0100268 return GetFieldPtr<ArtMethod**>(ResolvedMethodsOffset());
Mathieu Chartier66f19252012-09-18 08:57:04 -0700269 }
270
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800271 void SetResolvedMethods(ArtMethod** resolved_methods)
272 ALWAYS_INLINE
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700273 REQUIRES_SHARED(Locks::mutator_lock_) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800274 SetFieldPtr<false>(ResolvedMethodsOffset(), resolved_methods);
275 }
276
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700277 ArtField** GetResolvedFields() ALWAYS_INLINE REQUIRES_SHARED(Locks::mutator_lock_) {
Vladimir Marko05792b92015-08-03 11:56:49 +0100278 return GetFieldPtr<ArtField**>(ResolvedFieldsOffset());
279 }
280
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800281 void SetResolvedFields(ArtField** resolved_fields)
282 ALWAYS_INLINE
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700283 REQUIRES_SHARED(Locks::mutator_lock_) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800284 SetFieldPtr<false>(ResolvedFieldsOffset(), resolved_fields);
285 }
286
Narayan Kamath25352fc2016-08-03 12:46:58 +0100287 MethodTypeDexCacheType* GetResolvedMethodTypes()
288 ALWAYS_INLINE REQUIRES_SHARED(Locks::mutator_lock_) {
Narayan Kamath7fe56582016-10-14 18:49:12 +0100289 return GetFieldPtr64<MethodTypeDexCacheType*>(ResolvedMethodTypesOffset());
Narayan Kamath25352fc2016-08-03 12:46:58 +0100290 }
291
292 void SetResolvedMethodTypes(MethodTypeDexCacheType* resolved_method_types)
293 ALWAYS_INLINE
294 REQUIRES_SHARED(Locks::mutator_lock_) {
295 SetFieldPtr<false>(ResolvedMethodTypesOffset(), resolved_method_types);
296 }
297
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700298 size_t NumStrings() REQUIRES_SHARED(Locks::mutator_lock_) {
Vladimir Marko05792b92015-08-03 11:56:49 +0100299 return GetField32(NumStringsOffset());
300 }
301
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700302 size_t NumResolvedTypes() REQUIRES_SHARED(Locks::mutator_lock_) {
Vladimir Marko05792b92015-08-03 11:56:49 +0100303 return GetField32(NumResolvedTypesOffset());
304 }
305
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700306 size_t NumResolvedMethods() REQUIRES_SHARED(Locks::mutator_lock_) {
Vladimir Marko05792b92015-08-03 11:56:49 +0100307 return GetField32(NumResolvedMethodsOffset());
308 }
309
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700310 size_t NumResolvedFields() REQUIRES_SHARED(Locks::mutator_lock_) {
Vladimir Marko05792b92015-08-03 11:56:49 +0100311 return GetField32(NumResolvedFieldsOffset());
Mathieu Chartier66f19252012-09-18 08:57:04 -0700312 }
313
Narayan Kamath25352fc2016-08-03 12:46:58 +0100314 size_t NumResolvedMethodTypes() REQUIRES_SHARED(Locks::mutator_lock_) {
315 return GetField32(NumResolvedMethodTypesOffset());
316 }
317
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700318 const DexFile* GetDexFile() ALWAYS_INLINE REQUIRES_SHARED(Locks::mutator_lock_) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700319 return GetFieldPtr<const DexFile*>(OFFSET_OF_OBJECT_MEMBER(DexCache, dex_file_));
Mathieu Chartier66f19252012-09-18 08:57:04 -0700320 }
321
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700322 void SetDexFile(const DexFile* dex_file) REQUIRES_SHARED(Locks::mutator_lock_) {
Mathieu Chartier76172162016-01-26 14:54:06 -0800323 SetFieldPtr<false>(OFFSET_OF_OBJECT_MEMBER(DexCache, dex_file_), dex_file);
Brian Carlstrom4b620ff2011-09-11 01:11:01 -0700324 }
Brian Carlstromc4fa2c02011-08-21 03:00:12 -0700325
Mathieu Chartier28357fa2016-10-18 16:27:40 -0700326 void SetLocation(ObjPtr<mirror::String> location) REQUIRES_SHARED(Locks::mutator_lock_);
Mathieu Chartier76172162016-01-26 14:54:06 -0800327
Vladimir Marko05792b92015-08-03 11:56:49 +0100328 // NOTE: Get/SetElementPtrSize() are intended for working with ArtMethod** and ArtField**
329 // provided by GetResolvedMethods/Fields() and ArtMethod::GetDexCacheResolvedMethods(),
330 // so they need to be public.
331
332 template <typename PtrType>
Andreas Gampe542451c2016-07-26 09:02:02 -0700333 static PtrType GetElementPtrSize(PtrType* ptr_array, size_t idx, PointerSize ptr_size);
Vladimir Marko05792b92015-08-03 11:56:49 +0100334
335 template <typename PtrType>
Andreas Gampe542451c2016-07-26 09:02:02 -0700336 static void SetElementPtrSize(PtrType* ptr_array, size_t idx, PtrType ptr, PointerSize ptr_size);
Vladimir Marko05792b92015-08-03 11:56:49 +0100337
Brian Carlstromc4fa2c02011-08-21 03:00:12 -0700338 private:
Andreas Gampecc1b5352016-12-01 16:58:38 -0800339 void Init(const DexFile* dex_file,
340 ObjPtr<String> location,
341 StringDexCacheType* strings,
342 uint32_t num_strings,
343 GcRoot<Class>* resolved_types,
344 uint32_t num_resolved_types,
345 ArtMethod** resolved_methods,
346 uint32_t num_resolved_methods,
347 ArtField** resolved_fields,
348 uint32_t num_resolved_fields,
349 MethodTypeDexCacheType* resolved_methodtypes,
350 uint32_t num_resolved_methodtypes,
351 PointerSize pointer_size)
352 REQUIRES_SHARED(Locks::mutator_lock_);
353
Vladimir Marko05792b92015-08-03 11:56:49 +0100354 // Visit instance fields of the dex cache as well as its associated arrays.
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800355 template <bool kVisitNativeRoots,
356 VerifyObjectFlags kVerifyFlags = kDefaultVerifyFlags,
357 ReadBarrierOption kReadBarrierOption = kWithReadBarrier,
358 typename Visitor>
Mathieu Chartier31e88222016-10-14 18:43:19 -0700359 void VisitReferences(ObjPtr<mirror::Class> klass, const Visitor& visitor)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700360 REQUIRES_SHARED(Locks::mutator_lock_) REQUIRES(Locks::heap_bitmap_lock_);
Vladimir Marko05792b92015-08-03 11:56:49 +0100361
Ian Rogersef7d42f2014-01-06 12:55:46 -0800362 HeapReference<Object> dex_;
363 HeapReference<String> location_;
Narayan Kamath25352fc2016-08-03 12:46:58 +0100364 uint64_t dex_file_; // const DexFile*
365 uint64_t resolved_fields_; // ArtField*, array with num_resolved_fields_ elements.
366 uint64_t resolved_method_types_; // std::atomic<MethodTypeDexCachePair>* array with
367 // num_resolved_method_types_ elements.
368 uint64_t resolved_methods_; // ArtMethod*, array with num_resolved_methods_ elements.
369 uint64_t resolved_types_; // GcRoot<Class>*, array with num_resolved_types_ elements.
370 uint64_t strings_; // std::atomic<StringDexCachePair>*, array with num_strings_
371 // elements.
372
373 uint32_t num_resolved_fields_; // Number of elements in the resolved_fields_ array.
374 uint32_t num_resolved_method_types_; // Number of elements in the resolved_method_types_ array.
375 uint32_t num_resolved_methods_; // Number of elements in the resolved_methods_ array.
376 uint32_t num_resolved_types_; // Number of elements in the resolved_types_ array.
377 uint32_t num_strings_; // Number of elements in the strings_ array.
Brian Carlstrom83db7722011-08-26 17:32:56 -0700378
Brian Carlstrom7934ac22013-07-26 10:54:15 -0700379 friend struct art::DexCacheOffsets; // for verifying offset information
Vladimir Marko05792b92015-08-03 11:56:49 +0100380 friend class Object; // For VisitReferences
Brian Carlstromc4fa2c02011-08-21 03:00:12 -0700381 DISALLOW_IMPLICIT_CONSTRUCTORS(DexCache);
Brian Carlstrom7e49dca2011-07-22 18:07:34 -0700382};
383
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800384} // namespace mirror
Brian Carlstrom7e49dca2011-07-22 18:07:34 -0700385} // namespace art
386
Brian Carlstromfc0e3212013-07-17 14:40:12 -0700387#endif // ART_RUNTIME_MIRROR_DEX_CACHE_H_