blob: 2c2ad9b9b0de8b190db8bf676fe44ff01079cc11 [file] [log] [blame]
Ian Rogers2dd0e2c2013-01-24 12:42:14 -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 */
16
Brian Carlstromfc0e3212013-07-17 14:40:12 -070017#ifndef ART_RUNTIME_MIRROR_OBJECT_INL_H_
18#define ART_RUNTIME_MIRROR_OBJECT_INL_H_
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080019
20#include "object.h"
21
Andreas Gampe8cf9cb32017-07-19 09:28:38 -070022#include "array-inl.h"
Brian Carlstromea46f952013-07-30 01:26:50 -070023#include "art_field.h"
24#include "art_method.h"
David Sehrc431b9d2018-03-02 12:01:51 -080025#include "base/atomic.h"
Andreas Gampec6ea7d02017-02-01 16:46:28 -080026#include "class-inl.h"
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -070027#include "class_flags.h"
Mathieu Chartiere401d142015-04-22 13:56:20 -070028#include "class_linker.h"
Andreas Gampe2ff3b972017-06-05 18:14:53 -070029#include "dex_cache.h"
Andreas Gampe09659c22017-09-18 18:23:32 -070030#include "heap_poisoning.h"
Ian Rogersd9c4fc92013-10-01 19:45:43 -070031#include "lock_word-inl.h"
Ian Rogers05f30572013-02-20 12:13:11 -080032#include "monitor.h"
Andreas Gampe8cf9cb32017-07-19 09:28:38 -070033#include "obj_ptr-inl.h"
34#include "object-readbarrier-inl.h"
Mathieu Chartier52e4b432014-06-10 11:22:31 -070035#include "object_array-inl.h"
Mathieu Chartiera058fdf2016-10-06 15:13:58 -070036#include "object_reference-inl.h"
Hiroshi Yamauchi800ac2d2014-04-02 17:32:54 -070037#include "read_barrier-inl.h"
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -070038#include "reference.h"
Jeff Hao848f70a2014-01-15 13:49:50 -080039#include "runtime.h"
Vladimir Marko5924a4a2018-05-29 17:40:41 +010040#include "string.h"
Ian Rogers05f30572013-02-20 12:13:11 -080041#include "throwable.h"
Mathieu Chartier88ea61e2018-06-20 17:45:41 -070042#include "write_barrier-inl.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080043
44namespace art {
45namespace mirror {
46
Andreas Gampe542451c2016-07-26 09:02:02 -070047inline uint32_t Object::ClassSize(PointerSize pointer_size) {
Mingyao Yang98d1cc82014-05-15 17:02:16 -070048 uint32_t vtable_entries = kVTableLength;
Mathieu Chartiere401d142015-04-22 13:56:20 -070049 return Class::ComputeClassSize(true, vtable_entries, 0, 0, 0, 0, 0, pointer_size);
Mingyao Yang98d1cc82014-05-15 17:02:16 -070050}
51
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -070052template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -080053inline Class* Object::GetClass() {
Mathieu Chartier99111282018-06-19 12:30:56 -070054 return GetFieldObject<Class, kVerifyFlags, kReadBarrierOption>(ClassOffset());
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080055}
56
Mathieu Chartier4e305412014-02-19 10:54:44 -080057template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiera058fdf2016-10-06 15:13:58 -070058inline void Object::SetClass(ObjPtr<Class> new_klass) {
Mathieu Chartier2cebb242015-04-21 16:50:40 -070059 // new_klass may be null prior to class linker initialization.
Ian Rogersef7d42f2014-01-06 12:55:46 -080060 // We don't mark the card as this occurs as part of object allocation. Not all objects have
61 // backing cards, such as large objects.
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +010062 // We use non transactional version since we can't undo this write. We also disable checking as
63 // we may run in transaction mode here.
Mathieu Chartier99111282018-06-19 12:30:56 -070064 SetFieldObjectWithoutWriteBarrier<false, false, RemoveThisFlags(kVerifyFlags)>(ClassOffset(),
65 new_klass);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080066}
67
Andreas Gampe3b45ef22015-05-26 21:34:09 -070068template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartierbbd695c2014-04-16 09:48:48 -070069inline void Object::SetLockWord(LockWord new_val, bool as_volatile) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +010070 // Force use of non-transactional mode and do not check.
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070071 if (as_volatile) {
Mathieu Chartier99111282018-06-19 12:30:56 -070072 SetField32Volatile<false, false, kVerifyFlags>(MonitorOffset(), new_val.GetValue());
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070073 } else {
Mathieu Chartier99111282018-06-19 12:30:56 -070074 SetField32<false, false, kVerifyFlags>(MonitorOffset(), new_val.GetValue());
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070075 }
Ian Rogersd9c4fc92013-10-01 19:45:43 -070076}
77
Ian Rogersd9c4fc92013-10-01 19:45:43 -070078inline uint32_t Object::GetLockOwnerThreadId() {
79 return Monitor::GetLockOwnerThreadId(this);
Ian Rogers05f30572013-02-20 12:13:11 -080080}
81
Mathieu Chartiere7e8a5f2014-02-14 16:59:41 -080082inline mirror::Object* Object::MonitorEnter(Thread* self) {
Andreas Gampe98ea9d92018-10-19 14:06:15 -070083 return Monitor::MonitorEnter(self, this, /*trylock=*/false);
Mathieu Chartiera704eda2016-07-13 09:53:35 -070084}
85
86inline mirror::Object* Object::MonitorTryEnter(Thread* self) {
Andreas Gampe98ea9d92018-10-19 14:06:15 -070087 return Monitor::MonitorEnter(self, this, /*trylock=*/true);
Ian Rogers05f30572013-02-20 12:13:11 -080088}
89
90inline bool Object::MonitorExit(Thread* self) {
91 return Monitor::MonitorExit(self, this);
92}
93
94inline void Object::Notify(Thread* self) {
95 Monitor::Notify(self, this);
96}
97
98inline void Object::NotifyAll(Thread* self) {
99 Monitor::NotifyAll(self, this);
100}
101
Ian Rogers05f30572013-02-20 12:13:11 -0800102inline void Object::Wait(Thread* self, int64_t ms, int32_t ns) {
103 Monitor::Wait(self, this, ms, ns, true, kTimedWaiting);
104}
105
Mathieu Chartier36a270a2016-07-28 18:08:51 -0700106inline uint32_t Object::GetMarkBit() {
Mathieu Chartier99111282018-06-19 12:30:56 -0700107 CHECK(kUseReadBarrier);
Mathieu Chartier36a270a2016-07-28 18:08:51 -0700108 return GetLockWord(false).MarkBitState();
Mathieu Chartier36a270a2016-07-28 18:08:51 -0700109}
110
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700111inline void Object::SetReadBarrierState(uint32_t rb_state) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700112 CHECK(kUseBakerReadBarrier);
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700113 DCHECK(ReadBarrier::IsValidReadBarrierState(rb_state)) << rb_state;
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700114 LockWord lw = GetLockWord(false);
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700115 lw.SetReadBarrierState(rb_state);
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700116 SetLockWord(lw, false);
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800117}
118
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700119inline void Object::AssertReadBarrierState() const {
120 CHECK(kUseBakerReadBarrier);
121 Object* obj = const_cast<Object*>(this);
Roland Levillain14e5a292018-06-28 12:00:56 +0100122 DCHECK_EQ(obj->GetReadBarrierState(), ReadBarrier::NonGrayState())
Mathieu Chartier99111282018-06-19 12:30:56 -0700123 << "Bad Baker pointer: obj=" << obj << " rb_state" << obj->GetReadBarrierState();
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800124}
125
Mathieu Chartier4e305412014-02-19 10:54:44 -0800126template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartieref41db72016-10-25 15:08:01 -0700127inline bool Object::VerifierInstanceOf(ObjPtr<Class> klass) {
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700128 DCHECK(klass != nullptr);
129 DCHECK(GetClass<kVerifyFlags>() != nullptr);
Jeff Haoa3faaf42013-09-03 19:07:00 -0700130 return klass->IsInterface() || InstanceOf(klass);
131}
132
Mathieu Chartier4e305412014-02-19 10:54:44 -0800133template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiera59d9b22016-09-26 18:13:17 -0700134inline bool Object::InstanceOf(ObjPtr<Class> klass) {
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700135 DCHECK(klass != nullptr);
Mathieu Chartier99111282018-06-19 12:30:56 -0700136 DCHECK(GetClass<kVerifyNone>() != nullptr) << "this=" << this;
Mathieu Chartier4e305412014-02-19 10:54:44 -0800137 return klass->IsAssignableFrom(GetClass<kVerifyFlags>());
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800138}
139
Mathieu Chartierd7a7f2f2018-09-07 11:57:18 -0700140template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800141inline bool Object::IsClass() {
Mathieu Chartierd7a7f2f2018-09-07 11:57:18 -0700142 // OK to look at from-space copies since java.lang.Class.class is not movable.
143 // See b/114413743
144 ObjPtr<Class> klass = GetClass<kVerifyFlags, kWithoutReadBarrier>();
Vladimir Marko98db89c2018-09-07 11:45:46 +0100145 ObjPtr<Class> java_lang_Class = klass->GetClass<kVerifyFlags, kWithoutReadBarrier>();
Mathieu Chartierd7a7f2f2018-09-07 11:57:18 -0700146 return klass == java_lang_Class;
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800147}
148
Mathieu Chartierd7a7f2f2018-09-07 11:57:18 -0700149template<VerifyObjectFlags kVerifyFlags>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800150inline Class* Object::AsClass() {
Mathieu Chartierd7a7f2f2018-09-07 11:57:18 -0700151 DCHECK((IsClass<kVerifyFlags>()));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800152 return down_cast<Class*>(this);
153}
154
Vladimir Marko98db89c2018-09-07 11:45:46 +0100155template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800156inline bool Object::IsObjectArray() {
Vladimir Marko98db89c2018-09-07 11:45:46 +0100157 // We do not need a read barrier here as the primitive type is constant,
158 // both from-space and to-space component type classes shall yield the same result.
Vladimir Markodbcb48f2018-11-12 11:47:04 +0000159 constexpr VerifyObjectFlags kNewFlags = RemoveThisFlags(kVerifyFlags);
Vladimir Marko98db89c2018-09-07 11:45:46 +0100160 return IsArrayInstance<kVerifyFlags>() &&
161 !GetClass<kNewFlags, kWithoutReadBarrier>()->
162 template GetComponentType<kNewFlags, kWithoutReadBarrier>()->IsPrimitive();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800163}
164
Vladimir Marko98db89c2018-09-07 11:45:46 +0100165template<class T, VerifyObjectFlags kVerifyFlags>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800166inline ObjectArray<T>* Object::AsObjectArray() {
Vladimir Marko98db89c2018-09-07 11:45:46 +0100167 DCHECK((IsObjectArray<kVerifyFlags>()));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800168 return down_cast<ObjectArray<T>*>(this);
169}
170
Vladimir Marko98db89c2018-09-07 11:45:46 +0100171template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800172inline bool Object::IsArrayInstance() {
Vladimir Marko98db89c2018-09-07 11:45:46 +0100173 // We do not need a read barrier here, both from-space and to-space version of the class
174 // shall return the same result from IsArrayClass().
175 return GetClass<kVerifyFlags, kWithoutReadBarrier>()->template IsArrayClass<kVerifyFlags>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800176}
177
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800178template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800179inline bool Object::IsReferenceInstance() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800180 return GetClass<kVerifyFlags, kReadBarrierOption>()->IsTypeOfReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800181}
182
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800183template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700184inline Reference* Object::AsReference() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800185 DCHECK((IsReferenceInstance<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700186 return down_cast<Reference*>(this);
187}
188
Vladimir Marko98db89c2018-09-07 11:45:46 +0100189template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800190inline Array* Object::AsArray() {
Vladimir Marko98db89c2018-09-07 11:45:46 +0100191 DCHECK((IsArrayInstance<kVerifyFlags>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800192 return down_cast<Array*>(this);
193}
194
Vladimir Marko104883b2018-11-09 17:12:23 +0000195template<VerifyObjectFlags kVerifyFlags, Primitive::Type kType>
196ALWAYS_INLINE bool Object::IsSpecificPrimitiveArray() {
197 // We do not need a read barrier here as the primitive type is constant,
198 // both from-space and to-space component type classes shall yield the same result.
199 ObjPtr<Class> klass = GetClass<kVerifyFlags, kWithoutReadBarrier>();
Vladimir Markodbcb48f2018-11-12 11:47:04 +0000200 constexpr VerifyObjectFlags kNewFlags = RemoveThisFlags(kVerifyFlags);
Vladimir Marko104883b2018-11-09 17:12:23 +0000201 ObjPtr<Class> const component_type = klass->GetComponentType<kNewFlags, kWithoutReadBarrier>();
202 return component_type != nullptr &&
203 component_type->GetPrimitiveType<kNewFlags>() == kType;
204}
205
206template<VerifyObjectFlags kVerifyFlags>
207inline bool Object::IsBooleanArray() {
208 return IsSpecificPrimitiveArray<kVerifyFlags, Primitive::kPrimBoolean>();
209}
210
Mathieu Chartier4e305412014-02-19 10:54:44 -0800211template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800212inline BooleanArray* Object::AsBooleanArray() {
Vladimir Marko104883b2018-11-09 17:12:23 +0000213 DCHECK(IsBooleanArray<kVerifyFlags>());
Ian Rogers05f30572013-02-20 12:13:11 -0800214 return down_cast<BooleanArray*>(this);
215}
216
Mathieu Chartier4e305412014-02-19 10:54:44 -0800217template<VerifyObjectFlags kVerifyFlags>
Vladimir Marko104883b2018-11-09 17:12:23 +0000218inline bool Object::IsByteArray() {
219 return IsSpecificPrimitiveArray<kVerifyFlags, Primitive::kPrimByte>();
220}
221
222template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800223inline ByteArray* Object::AsByteArray() {
Vladimir Marko104883b2018-11-09 17:12:23 +0000224 DCHECK(IsByteArray<kVerifyFlags>());
Ian Rogers05f30572013-02-20 12:13:11 -0800225 return down_cast<ByteArray*>(this);
226}
227
Mathieu Chartier4e305412014-02-19 10:54:44 -0800228template<VerifyObjectFlags kVerifyFlags>
Vladimir Marko104883b2018-11-09 17:12:23 +0000229inline bool Object::IsCharArray() {
230 return IsSpecificPrimitiveArray<kVerifyFlags, Primitive::kPrimChar>();
Ian Rogersef7d42f2014-01-06 12:55:46 -0800231}
232
Mathieu Chartier4e305412014-02-19 10:54:44 -0800233template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800234inline CharArray* Object::AsCharArray() {
Vladimir Marko104883b2018-11-09 17:12:23 +0000235 DCHECK(IsCharArray<kVerifyFlags>());
Ian Rogers05f30572013-02-20 12:13:11 -0800236 return down_cast<CharArray*>(this);
237}
238
Mathieu Chartier4e305412014-02-19 10:54:44 -0800239template<VerifyObjectFlags kVerifyFlags>
Vladimir Marko104883b2018-11-09 17:12:23 +0000240inline bool Object::IsShortArray() {
241 return IsSpecificPrimitiveArray<kVerifyFlags, Primitive::kPrimShort>();
242}
243
244template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800245inline ShortArray* Object::AsShortArray() {
Vladimir Marko104883b2018-11-09 17:12:23 +0000246 DCHECK(IsShortArray<kVerifyFlags>());
Ian Rogers05f30572013-02-20 12:13:11 -0800247 return down_cast<ShortArray*>(this);
248}
249
Mathieu Chartier4e305412014-02-19 10:54:44 -0800250template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700251inline bool Object::IsIntArray() {
Vladimir Marko104883b2018-11-09 17:12:23 +0000252 return IsSpecificPrimitiveArray<kVerifyFlags, Primitive::kPrimInt>();
Mathieu Chartiere401d142015-04-22 13:56:20 -0700253}
254
Vladimir Marko104883b2018-11-09 17:12:23 +0000255template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700256inline IntArray* Object::AsIntArray() {
Vladimir Marko104883b2018-11-09 17:12:23 +0000257 DCHECK((IsIntArray<kVerifyFlags>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800258 return down_cast<IntArray*>(this);
259}
260
Vladimir Marko104883b2018-11-09 17:12:23 +0000261template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700262inline bool Object::IsLongArray() {
Vladimir Marko104883b2018-11-09 17:12:23 +0000263 return IsSpecificPrimitiveArray<kVerifyFlags, Primitive::kPrimLong>();
Mathieu Chartiere401d142015-04-22 13:56:20 -0700264}
265
Vladimir Marko104883b2018-11-09 17:12:23 +0000266template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700267inline LongArray* Object::AsLongArray() {
Vladimir Marko104883b2018-11-09 17:12:23 +0000268 DCHECK((IsLongArray<kVerifyFlags>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800269 return down_cast<LongArray*>(this);
270}
271
Mathieu Chartier4e305412014-02-19 10:54:44 -0800272template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700273inline bool Object::IsFloatArray() {
Vladimir Marko104883b2018-11-09 17:12:23 +0000274 return IsSpecificPrimitiveArray<kVerifyFlags, Primitive::kPrimFloat>();
Mathieu Chartiere401d142015-04-22 13:56:20 -0700275}
276
277template<VerifyObjectFlags kVerifyFlags>
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100278inline FloatArray* Object::AsFloatArray() {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700279 DCHECK(IsFloatArray<kVerifyFlags>());
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100280 return down_cast<FloatArray*>(this);
281}
282
Mathieu Chartier4e305412014-02-19 10:54:44 -0800283template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700284inline bool Object::IsDoubleArray() {
Vladimir Marko104883b2018-11-09 17:12:23 +0000285 return IsSpecificPrimitiveArray<kVerifyFlags, Primitive::kPrimDouble>();
Mathieu Chartiere401d142015-04-22 13:56:20 -0700286}
287
288template<VerifyObjectFlags kVerifyFlags>
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100289inline DoubleArray* Object::AsDoubleArray() {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700290 DCHECK(IsDoubleArray<kVerifyFlags>());
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100291 return down_cast<DoubleArray*>(this);
292}
293
Jeff Hao848f70a2014-01-15 13:49:50 -0800294template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
295inline bool Object::IsString() {
296 return GetClass<kVerifyFlags, kReadBarrierOption>()->IsStringClass();
297}
298
299template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers05f30572013-02-20 12:13:11 -0800300inline String* Object::AsString() {
Jeff Hao848f70a2014-01-15 13:49:50 -0800301 DCHECK((IsString<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800302 return down_cast<String*>(this);
303}
304
Mathieu Chartier4e305412014-02-19 10:54:44 -0800305template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800306inline Throwable* Object::AsThrowable() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800307 DCHECK(GetClass<kVerifyFlags>()->IsThrowableClass());
Ian Rogers05f30572013-02-20 12:13:11 -0800308 return down_cast<Throwable*>(this);
309}
310
Mathieu Chartier4e305412014-02-19 10:54:44 -0800311template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800312inline bool Object::IsWeakReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800313 return GetClass<kVerifyFlags>()->IsWeakReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800314}
315
Mathieu Chartier4e305412014-02-19 10:54:44 -0800316template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800317inline bool Object::IsSoftReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800318 return GetClass<kVerifyFlags>()->IsSoftReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800319}
320
Mathieu Chartier4e305412014-02-19 10:54:44 -0800321template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800322inline bool Object::IsFinalizerReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800323 return GetClass<kVerifyFlags>()->IsFinalizerReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800324}
325
Mathieu Chartier4e305412014-02-19 10:54:44 -0800326template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700327inline FinalizerReference* Object::AsFinalizerReference() {
328 DCHECK(IsFinalizerReferenceInstance<kVerifyFlags>());
329 return down_cast<FinalizerReference*>(this);
330}
331
332template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800333inline bool Object::IsPhantomReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800334 return GetClass<kVerifyFlags>()->IsPhantomReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800335}
336
Mathieu Chartierd08f66f2017-04-13 11:47:53 -0700337template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800338inline size_t Object::SizeOf() {
Mathieu Chartierd08f66f2017-04-13 11:47:53 -0700339 // Read barrier is never required for SizeOf since objects sizes are constant. Reading from-space
340 // values is OK because of that.
Mathieu Chartier99111282018-06-19 12:30:56 -0700341 static constexpr ReadBarrierOption kRBO = kWithoutReadBarrier;
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800342 size_t result;
Vladimir Markodbcb48f2018-11-12 11:47:04 +0000343 constexpr VerifyObjectFlags kNewFlags = RemoveThisFlags(kVerifyFlags);
Vladimir Marko98db89c2018-09-07 11:45:46 +0100344 if (IsArrayInstance<kVerifyFlags>()) {
345 result = AsArray<kNewFlags>()->template SizeOf<kNewFlags, kRBO>();
Mathieu Chartierd7a7f2f2018-09-07 11:57:18 -0700346 } else if (IsClass<kNewFlags>()) {
347 result = AsClass<kNewFlags>()->template SizeOf<kNewFlags, kRBO>();
Mathieu Chartier99111282018-06-19 12:30:56 -0700348 } else if (GetClass<kNewFlags, kRBO>()->IsStringClass()) {
349 result = AsString<kNewFlags, kRBO>()->template SizeOf<kNewFlags>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800350 } else {
Vladimir Marko98db89c2018-09-07 11:45:46 +0100351 result = GetClass<kNewFlags, kRBO>()->template GetObjectSize<kNewFlags>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800352 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700353 DCHECK_GE(result, sizeof(Object)) << " class=" << Class::PrettyClass(GetClass<kNewFlags, kRBO>());
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800354 return result;
355}
356
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700357template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
Fred Shih37f05ef2014-07-16 18:38:08 -0700358inline int8_t Object::GetFieldByte(MemberOffset field_offset) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700359 Verify<kVerifyFlags>();
David Srbeckyce32c102018-08-31 07:21:07 +0100360 return GetFieldPrimitive<int8_t, kIsVolatile>(field_offset);
Fred Shih37f05ef2014-07-16 18:38:08 -0700361}
362
363template<VerifyObjectFlags kVerifyFlags>
364inline uint8_t Object::GetFieldBooleanVolatile(MemberOffset field_offset) {
365 return GetFieldBoolean<kVerifyFlags, true>(field_offset);
366}
367
368template<VerifyObjectFlags kVerifyFlags>
369inline int8_t Object::GetFieldByteVolatile(MemberOffset field_offset) {
370 return GetFieldByte<kVerifyFlags, true>(field_offset);
371}
372
Roland Levillaind32ead22018-05-30 17:38:21 +0100373template<bool kTransactionActive,
374 bool kCheckTransaction,
375 VerifyObjectFlags kVerifyFlags,
376 bool kIsVolatile>
Mathieu Chartier99111282018-06-19 12:30:56 -0700377inline void Object::SetFieldBoolean(MemberOffset field_offset, uint8_t new_value) {
378 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Fred Shih37f05ef2014-07-16 18:38:08 -0700379 if (kTransactionActive) {
Roland Levillaind32ead22018-05-30 17:38:21 +0100380 Runtime::Current()->RecordWriteFieldBoolean(
381 this,
382 field_offset,
383 GetFieldBoolean<kVerifyFlags, kIsVolatile>(field_offset),
384 kIsVolatile);
Fred Shih37f05ef2014-07-16 18:38:08 -0700385 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700386 Verify<kVerifyFlags>();
David Srbeckyce32c102018-08-31 07:21:07 +0100387 SetFieldPrimitive<uint8_t, kIsVolatile>(field_offset, new_value);
Fred Shih37f05ef2014-07-16 18:38:08 -0700388}
389
Roland Levillaind32ead22018-05-30 17:38:21 +0100390template<bool kTransactionActive,
391 bool kCheckTransaction,
392 VerifyObjectFlags kVerifyFlags,
393 bool kIsVolatile>
Mathieu Chartier99111282018-06-19 12:30:56 -0700394inline void Object::SetFieldByte(MemberOffset field_offset, int8_t new_value) {
395 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Fred Shih37f05ef2014-07-16 18:38:08 -0700396 if (kTransactionActive) {
Roland Levillaind32ead22018-05-30 17:38:21 +0100397 Runtime::Current()->RecordWriteFieldByte(this,
398 field_offset,
399 GetFieldByte<kVerifyFlags, kIsVolatile>(field_offset),
400 kIsVolatile);
Fred Shih37f05ef2014-07-16 18:38:08 -0700401 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700402 Verify<kVerifyFlags>();
David Srbeckyce32c102018-08-31 07:21:07 +0100403 SetFieldPrimitive<int8_t, kIsVolatile>(field_offset, new_value);
Fred Shih37f05ef2014-07-16 18:38:08 -0700404}
405
406template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
407inline void Object::SetFieldBooleanVolatile(MemberOffset field_offset, uint8_t new_value) {
408 return SetFieldBoolean<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
409 field_offset, new_value);
410}
411
412template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
413inline void Object::SetFieldByteVolatile(MemberOffset field_offset, int8_t new_value) {
414 return SetFieldByte<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
415 field_offset, new_value);
416}
417
418template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
419inline uint16_t Object::GetFieldChar(MemberOffset field_offset) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700420 Verify<kVerifyFlags>();
David Srbeckyce32c102018-08-31 07:21:07 +0100421 return GetFieldPrimitive<uint16_t, kIsVolatile>(field_offset);
Fred Shih37f05ef2014-07-16 18:38:08 -0700422}
423
424template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
425inline int16_t Object::GetFieldShort(MemberOffset field_offset) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700426 Verify<kVerifyFlags>();
David Srbeckyce32c102018-08-31 07:21:07 +0100427 return GetFieldPrimitive<int16_t, kIsVolatile>(field_offset);
Fred Shih37f05ef2014-07-16 18:38:08 -0700428}
429
430template<VerifyObjectFlags kVerifyFlags>
431inline uint16_t Object::GetFieldCharVolatile(MemberOffset field_offset) {
432 return GetFieldChar<kVerifyFlags, true>(field_offset);
433}
434
435template<VerifyObjectFlags kVerifyFlags>
436inline int16_t Object::GetFieldShortVolatile(MemberOffset field_offset) {
437 return GetFieldShort<kVerifyFlags, true>(field_offset);
438}
439
Roland Levillaind32ead22018-05-30 17:38:21 +0100440template<bool kTransactionActive,
441 bool kCheckTransaction,
442 VerifyObjectFlags kVerifyFlags,
443 bool kIsVolatile>
Fred Shih37f05ef2014-07-16 18:38:08 -0700444inline void Object::SetFieldChar(MemberOffset field_offset, uint16_t new_value) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700445 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Fred Shih37f05ef2014-07-16 18:38:08 -0700446 if (kTransactionActive) {
Roland Levillaind32ead22018-05-30 17:38:21 +0100447 Runtime::Current()->RecordWriteFieldChar(this,
448 field_offset,
449 GetFieldChar<kVerifyFlags, kIsVolatile>(field_offset),
450 kIsVolatile);
Fred Shih37f05ef2014-07-16 18:38:08 -0700451 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700452 Verify<kVerifyFlags>();
David Srbeckyce32c102018-08-31 07:21:07 +0100453 SetFieldPrimitive<uint16_t, kIsVolatile>(field_offset, new_value);
Fred Shih37f05ef2014-07-16 18:38:08 -0700454}
455
Roland Levillaind32ead22018-05-30 17:38:21 +0100456template<bool kTransactionActive,
457 bool kCheckTransaction,
458 VerifyObjectFlags kVerifyFlags,
459 bool kIsVolatile>
Fred Shih37f05ef2014-07-16 18:38:08 -0700460inline void Object::SetFieldShort(MemberOffset field_offset, int16_t new_value) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700461 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Fred Shih37f05ef2014-07-16 18:38:08 -0700462 if (kTransactionActive) {
Roland Levillaind32ead22018-05-30 17:38:21 +0100463 Runtime::Current()->RecordWriteFieldChar(this,
464 field_offset,
465 GetFieldShort<kVerifyFlags, kIsVolatile>(field_offset),
466 kIsVolatile);
Fred Shih37f05ef2014-07-16 18:38:08 -0700467 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700468 Verify<kVerifyFlags>();
David Srbeckyce32c102018-08-31 07:21:07 +0100469 SetFieldPrimitive<int16_t, kIsVolatile>(field_offset, new_value);
Fred Shih37f05ef2014-07-16 18:38:08 -0700470}
471
472template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
473inline void Object::SetFieldCharVolatile(MemberOffset field_offset, uint16_t new_value) {
474 return SetFieldChar<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
475 field_offset, new_value);
476}
477
478template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
479inline void Object::SetFieldShortVolatile(MemberOffset field_offset, int16_t new_value) {
480 return SetFieldShort<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
481 field_offset, new_value);
482}
483
Roland Levillaind32ead22018-05-30 17:38:21 +0100484template<bool kTransactionActive,
485 bool kCheckTransaction,
486 VerifyObjectFlags kVerifyFlags,
487 bool kIsVolatile>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700488inline void Object::SetField32(MemberOffset field_offset, int32_t new_value) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700489 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100490 if (kTransactionActive) {
Roland Levillaind32ead22018-05-30 17:38:21 +0100491 Runtime::Current()->RecordWriteField32(this,
492 field_offset,
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700493 GetField32<kVerifyFlags, kIsVolatile>(field_offset),
494 kIsVolatile);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100495 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700496 Verify<kVerifyFlags>();
David Srbeckyce32c102018-08-31 07:21:07 +0100497 SetFieldPrimitive<int32_t, kIsVolatile>(field_offset, new_value);
Ian Rogersb122a4b2013-11-19 18:00:50 -0800498}
499
Mathieu Chartier4e305412014-02-19 10:54:44 -0800500template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700501inline void Object::SetField32Volatile(MemberOffset field_offset, int32_t new_value) {
502 SetField32<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset, new_value);
503}
504
Chang Xing6d3e7682017-07-11 10:31:29 -0700505template<bool kCheckTransaction, VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
506inline void Object::SetField32Transaction(MemberOffset field_offset, int32_t new_value) {
507 if (Runtime::Current()->IsActiveTransaction()) {
508 SetField32<true, kCheckTransaction, kVerifyFlags, kIsVolatile>(field_offset, new_value);
509 } else {
510 SetField32<false, kCheckTransaction, kVerifyFlags, kIsVolatile>(field_offset, new_value);
511 }
512}
513
Roland Levillaind32ead22018-05-30 17:38:21 +0100514template<bool kTransactionActive,
515 bool kCheckTransaction,
516 VerifyObjectFlags kVerifyFlags,
517 bool kIsVolatile>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700518inline void Object::SetField64(MemberOffset field_offset, int64_t new_value) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700519 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100520 if (kTransactionActive) {
Roland Levillaind32ead22018-05-30 17:38:21 +0100521 Runtime::Current()->RecordWriteField64(this,
522 field_offset,
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700523 GetField64<kVerifyFlags, kIsVolatile>(field_offset),
524 kIsVolatile);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100525 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700526 Verify<kVerifyFlags>();
David Srbeckyce32c102018-08-31 07:21:07 +0100527 SetFieldPrimitive<int64_t, kIsVolatile>(field_offset, new_value);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800528}
529
Mathieu Chartier4e305412014-02-19 10:54:44 -0800530template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700531inline void Object::SetField64Volatile(MemberOffset field_offset, int64_t new_value) {
532 return SetField64<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
533 new_value);
534}
535
Chang Xing6d3e7682017-07-11 10:31:29 -0700536template<bool kCheckTransaction, VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
537inline void Object::SetField64Transaction(MemberOffset field_offset, int32_t new_value) {
538 if (Runtime::Current()->IsActiveTransaction()) {
539 SetField64<true, kCheckTransaction, kVerifyFlags, kIsVolatile>(field_offset, new_value);
540 } else {
541 SetField64<false, kCheckTransaction, kVerifyFlags, kIsVolatile>(field_offset, new_value);
542 }
543}
544
Mathieu Chartierc381c362016-08-23 13:27:53 -0700545template<typename kSize>
546inline kSize Object::GetFieldAcquire(MemberOffset field_offset) {
547 const uint8_t* raw_addr = reinterpret_cast<const uint8_t*>(this) + field_offset.Int32Value();
548 const kSize* addr = reinterpret_cast<const kSize*>(raw_addr);
Orion Hodson88591fe2018-03-06 13:35:43 +0000549 return reinterpret_cast<const Atomic<kSize>*>(addr)->load(std::memory_order_acquire);
Mathieu Chartierc381c362016-08-23 13:27:53 -0700550}
551
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700552template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -0700553inline bool Object::CasFieldWeakSequentiallyConsistent64(MemberOffset field_offset,
Roland Levillaind32ead22018-05-30 17:38:21 +0100554 int64_t old_value,
555 int64_t new_value) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700556 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100557 if (kTransactionActive) {
558 Runtime::Current()->RecordWriteField64(this, field_offset, old_value, true);
559 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700560 Verify<kVerifyFlags>();
Ian Rogers13735952014-10-08 12:43:28 -0700561 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogers228602f2014-07-10 02:07:54 -0700562 Atomic<int64_t>* atomic_addr = reinterpret_cast<Atomic<int64_t>*>(raw_addr);
Orion Hodson4557b382018-01-03 11:47:54 +0000563 return atomic_addr->CompareAndSetWeakSequentiallyConsistent(old_value, new_value);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800564}
565
Hans Boehmd8434432014-07-11 09:56:07 -0700566template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
567inline bool Object::CasFieldStrongSequentiallyConsistent64(MemberOffset field_offset,
Roland Levillaind32ead22018-05-30 17:38:21 +0100568 int64_t old_value,
569 int64_t new_value) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700570 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Hans Boehmd8434432014-07-11 09:56:07 -0700571 if (kTransactionActive) {
572 Runtime::Current()->RecordWriteField64(this, field_offset, old_value, true);
573 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700574 Verify<kVerifyFlags>();
Ian Rogers13735952014-10-08 12:43:28 -0700575 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Hans Boehmd8434432014-07-11 09:56:07 -0700576 Atomic<int64_t>* atomic_addr = reinterpret_cast<Atomic<int64_t>*>(raw_addr);
Orion Hodson4557b382018-01-03 11:47:54 +0000577 return atomic_addr->CompareAndSetStrongSequentiallyConsistent(old_value, new_value);
Hans Boehmd8434432014-07-11 09:56:07 -0700578}
579
Chris Wailes0c61be42018-09-26 17:27:34 -0700580/*
581 * Returns a pointer to an object representing what the field points to, not an
582 * object representing the field.
583 */
Roland Levillaind32ead22018-05-30 17:38:21 +0100584template<class T,
585 VerifyObjectFlags kVerifyFlags,
586 ReadBarrierOption kReadBarrierOption,
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700587 bool kIsVolatile>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700588inline T* Object::GetFieldObject(MemberOffset field_offset) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700589 Verify<kVerifyFlags>();
Ian Rogers13735952014-10-08 12:43:28 -0700590 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogersef7d42f2014-01-06 12:55:46 -0800591 HeapReference<T>* objref_addr = reinterpret_cast<HeapReference<T>*>(raw_addr);
Hans Boehmcc55e1d2017-07-27 15:28:07 -0700592 T* result = ReadBarrier::Barrier<T, kIsVolatile, kReadBarrierOption>(
593 this,
594 field_offset,
595 objref_addr);
Mathieu Chartier99111282018-06-19 12:30:56 -0700596 VerifyRead<kVerifyFlags>(result);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800597 return result;
598}
599
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700600template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700601inline T* Object::GetFieldObjectVolatile(MemberOffset field_offset) {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700602 return GetFieldObject<T, kVerifyFlags, kReadBarrierOption, true>(field_offset);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700603}
604
Roland Levillaind32ead22018-05-30 17:38:21 +0100605template<bool kTransactionActive,
606 bool kCheckTransaction,
607 VerifyObjectFlags kVerifyFlags,
608 bool kIsVolatile>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700609inline void Object::SetFieldObjectWithoutWriteBarrier(MemberOffset field_offset,
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700610 ObjPtr<Object> new_value) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700611 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100612 if (kTransactionActive) {
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700613 ObjPtr<Object> obj;
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700614 if (kIsVolatile) {
615 obj = GetFieldObjectVolatile<Object>(field_offset);
616 } else {
617 obj = GetFieldObject<Object>(field_offset);
618 }
Vladimir Markobcf17522018-06-01 13:14:32 +0100619 Runtime::Current()->RecordWriteFieldReference(this, field_offset, obj, true);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100620 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700621 Verify<kVerifyFlags>();
622 VerifyWrite<kVerifyFlags>(new_value);
Ian Rogers13735952014-10-08 12:43:28 -0700623 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogersef7d42f2014-01-06 12:55:46 -0800624 HeapReference<Object>* objref_addr = reinterpret_cast<HeapReference<Object>*>(raw_addr);
Hans Boehmcc55e1d2017-07-27 15:28:07 -0700625 objref_addr->Assign<kIsVolatile>(new_value.Ptr());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800626}
627
Roland Levillaind32ead22018-05-30 17:38:21 +0100628template<bool kTransactionActive,
629 bool kCheckTransaction,
630 VerifyObjectFlags kVerifyFlags,
631 bool kIsVolatile>
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700632inline void Object::SetFieldObject(MemberOffset field_offset, ObjPtr<Object> new_value) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700633 SetFieldObjectWithoutWriteBarrier<kTransactionActive, kCheckTransaction, kVerifyFlags,
634 kIsVolatile>(field_offset, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800635 if (new_value != nullptr) {
Mathieu Chartier88ea61e2018-06-20 17:45:41 -0700636 WriteBarrier::ForFieldWrite<WriteBarrier::kWithoutNullCheck>(this, field_offset, new_value);
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700637 // TODO: Check field assignment could theoretically cause thread suspension, TODO: fix this.
638 CheckFieldAssignment(field_offset, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800639 }
640}
641
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700642template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700643inline void Object::SetFieldObjectVolatile(MemberOffset field_offset, ObjPtr<Object> new_value) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700644 SetFieldObject<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
645 new_value);
646}
647
Chang Xing6d3e7682017-07-11 10:31:29 -0700648template<bool kCheckTransaction, VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
649inline void Object::SetFieldObjectTransaction(MemberOffset field_offset, ObjPtr<Object> new_value) {
650 if (Runtime::Current()->IsActiveTransaction()) {
651 SetFieldObject<true, kCheckTransaction, kVerifyFlags, kIsVolatile>(field_offset, new_value);
652 } else {
653 SetFieldObject<false, kCheckTransaction, kVerifyFlags, kIsVolatile>(field_offset, new_value);
654 }
655}
656
Mathieu Chartier4e305412014-02-19 10:54:44 -0800657template <VerifyObjectFlags kVerifyFlags>
658inline HeapReference<Object>* Object::GetFieldObjectReferenceAddr(MemberOffset field_offset) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700659 Verify<kVerifyFlags>();
Ian Rogers13735952014-10-08 12:43:28 -0700660 return reinterpret_cast<HeapReference<Object>*>(reinterpret_cast<uint8_t*>(this) +
Mathieu Chartier4e305412014-02-19 10:54:44 -0800661 field_offset.Int32Value());
662}
663
664template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Mathieu Chartiera9746b92018-06-22 10:25:40 -0700665inline bool Object::CasFieldObjectWithoutWriteBarrier(MemberOffset field_offset,
666 ObjPtr<Object> old_value,
667 ObjPtr<Object> new_value,
668 CASMode mode,
669 std::memory_order memory_order) {
670 VerifyTransaction<kTransactionActive, kCheckTransaction>();
671 VerifyCAS<kVerifyFlags>(new_value, old_value);
672 if (kTransactionActive) {
673 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
674 }
675 uint32_t old_ref(PtrCompression<kPoisonHeapReferences, Object>::Compress(old_value));
676 uint32_t new_ref(PtrCompression<kPoisonHeapReferences, Object>::Compress(new_value));
677 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
678 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
679 return atomic_addr->CompareAndSet(old_ref, new_ref, mode, memory_order);
680}
681
682template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
683inline bool Object::CasFieldObject(MemberOffset field_offset,
684 ObjPtr<Object> old_value,
685 ObjPtr<Object> new_value,
686 CASMode mode,
687 std::memory_order memory_order) {
688 bool success = CasFieldObjectWithoutWriteBarrier<
689 kTransactionActive, kCheckTransaction, kVerifyFlags>(field_offset,
690 old_value,
691 new_value,
692 mode,
693 memory_order);
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800694 if (success) {
Mathieu Chartier88ea61e2018-06-20 17:45:41 -0700695 WriteBarrier::ForFieldWrite(this, field_offset, new_value);
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800696 }
697 return success;
698}
699
700template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Orion Hodson928033d2018-02-07 05:30:54 +0000701inline ObjPtr<Object> Object::CompareAndExchangeFieldObject(MemberOffset field_offset,
702 ObjPtr<Object> old_value,
703 ObjPtr<Object> new_value) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700704 VerifyTransaction<kTransactionActive, kCheckTransaction>();
705 VerifyCAS<kVerifyFlags>(new_value, old_value);
Orion Hodson928033d2018-02-07 05:30:54 +0000706 uint32_t old_ref(PtrCompression<kPoisonHeapReferences, Object>::Compress(old_value));
707 uint32_t new_ref(PtrCompression<kPoisonHeapReferences, Object>::Compress(new_value));
708 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
709 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
Orion Hodson88591fe2018-03-06 13:35:43 +0000710 bool success = atomic_addr->compare_exchange_strong(old_ref, new_ref, std::memory_order_seq_cst);
Orion Hodson928033d2018-02-07 05:30:54 +0000711 ObjPtr<Object> witness_value(PtrCompression<kPoisonHeapReferences, Object>::Decompress(old_ref));
712 if (kIsDebugBuild) {
713 // Ensure caller has done read barrier on the reference field so it's in the to-space.
714 ReadBarrier::AssertToSpaceInvariant(witness_value.Ptr());
715 }
Mathieu Chartier1d2e2662018-06-19 14:02:12 -0700716 if (success) {
717 if (kTransactionActive) {
718 Runtime::Current()->RecordWriteFieldReference(this, field_offset, witness_value, true);
719 }
Mathieu Chartier88ea61e2018-06-20 17:45:41 -0700720 WriteBarrier::ForFieldWrite(this, field_offset, new_value);
Orion Hodson928033d2018-02-07 05:30:54 +0000721 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700722 VerifyRead<kVerifyFlags>(witness_value);
Orion Hodson928033d2018-02-07 05:30:54 +0000723 return witness_value;
724}
725
726template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
727inline ObjPtr<Object> Object::ExchangeFieldObject(MemberOffset field_offset,
728 ObjPtr<Object> new_value) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700729 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Andreas Gampe98ea9d92018-10-19 14:06:15 -0700730 VerifyCAS<kVerifyFlags>(new_value, /*old_value=*/ nullptr);
Mathieu Chartier99111282018-06-19 12:30:56 -0700731
Orion Hodson928033d2018-02-07 05:30:54 +0000732 uint32_t new_ref(PtrCompression<kPoisonHeapReferences, Object>::Compress(new_value));
733 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
734 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
Orion Hodson88591fe2018-03-06 13:35:43 +0000735 uint32_t old_ref = atomic_addr->exchange(new_ref, std::memory_order_seq_cst);
Orion Hodson928033d2018-02-07 05:30:54 +0000736 ObjPtr<Object> old_value(PtrCompression<kPoisonHeapReferences, Object>::Decompress(old_ref));
737 if (kIsDebugBuild) {
738 // Ensure caller has done read barrier on the reference field so it's in the to-space.
739 ReadBarrier::AssertToSpaceInvariant(old_value.Ptr());
740 }
741 if (kTransactionActive) {
742 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
743 }
Mathieu Chartier88ea61e2018-06-20 17:45:41 -0700744 WriteBarrier::ForFieldWrite(this, field_offset, new_value);
Mathieu Chartier99111282018-06-19 12:30:56 -0700745 VerifyRead<kVerifyFlags>(old_value);
Orion Hodson928033d2018-02-07 05:30:54 +0000746 return old_value;
747}
748
749template<typename T, VerifyObjectFlags kVerifyFlags>
750inline void Object::GetPrimitiveFieldViaAccessor(MemberOffset field_offset, Accessor<T>* accessor) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700751 Verify<kVerifyFlags>();
Orion Hodson928033d2018-02-07 05:30:54 +0000752 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
753 T* addr = reinterpret_cast<T*>(raw_addr);
754 accessor->Access(addr);
755}
756
757template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
758inline void Object::UpdateFieldBooleanViaAccessor(MemberOffset field_offset,
759 Accessor<uint8_t>* accessor) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700760 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Orion Hodson928033d2018-02-07 05:30:54 +0000761 if (kTransactionActive) {
762 static const bool kIsVolatile = true;
763 uint8_t old_value = GetFieldBoolean<kVerifyFlags, kIsVolatile>(field_offset);
764 Runtime::Current()->RecordWriteFieldBoolean(this, field_offset, old_value, kIsVolatile);
765 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700766 Verify<kVerifyFlags>();
Orion Hodson928033d2018-02-07 05:30:54 +0000767 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
768 uint8_t* addr = raw_addr;
769 accessor->Access(addr);
770}
771
772template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
773inline void Object::UpdateFieldByteViaAccessor(MemberOffset field_offset,
774 Accessor<int8_t>* accessor) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700775 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Orion Hodson928033d2018-02-07 05:30:54 +0000776 if (kTransactionActive) {
777 static const bool kIsVolatile = true;
778 int8_t old_value = GetFieldByte<kVerifyFlags, kIsVolatile>(field_offset);
779 Runtime::Current()->RecordWriteFieldByte(this, field_offset, old_value, kIsVolatile);
780 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700781 Verify<kVerifyFlags>();
Orion Hodson928033d2018-02-07 05:30:54 +0000782 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
783 int8_t* addr = reinterpret_cast<int8_t*>(raw_addr);
784 accessor->Access(addr);
785}
786
787template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
788inline void Object::UpdateFieldCharViaAccessor(MemberOffset field_offset,
789 Accessor<uint16_t>* accessor) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700790 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Orion Hodson928033d2018-02-07 05:30:54 +0000791 if (kTransactionActive) {
792 static const bool kIsVolatile = true;
793 uint16_t old_value = GetFieldChar<kVerifyFlags, kIsVolatile>(field_offset);
794 Runtime::Current()->RecordWriteFieldChar(this, field_offset, old_value, kIsVolatile);
795 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700796 Verify<kVerifyFlags>();
Orion Hodson928033d2018-02-07 05:30:54 +0000797 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
798 uint16_t* addr = reinterpret_cast<uint16_t*>(raw_addr);
799 accessor->Access(addr);
800}
801
802template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
803inline void Object::UpdateFieldShortViaAccessor(MemberOffset field_offset,
804 Accessor<int16_t>* accessor) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700805 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Orion Hodson928033d2018-02-07 05:30:54 +0000806 if (kTransactionActive) {
807 static const bool kIsVolatile = true;
808 int16_t old_value = GetFieldShort<kVerifyFlags, kIsVolatile>(field_offset);
809 Runtime::Current()->RecordWriteFieldShort(this, field_offset, old_value, kIsVolatile);
810 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700811 Verify<kVerifyFlags>();
Orion Hodson928033d2018-02-07 05:30:54 +0000812 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
813 int16_t* addr = reinterpret_cast<int16_t*>(raw_addr);
814 accessor->Access(addr);
815}
816
817template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
818inline void Object::UpdateField32ViaAccessor(MemberOffset field_offset,
819 Accessor<int32_t>* accessor) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700820 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Orion Hodson928033d2018-02-07 05:30:54 +0000821 if (kTransactionActive) {
822 static const bool kIsVolatile = true;
823 int32_t old_value = GetField32<kVerifyFlags, kIsVolatile>(field_offset);
824 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, kIsVolatile);
825 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700826 Verify<kVerifyFlags>();
Orion Hodson928033d2018-02-07 05:30:54 +0000827 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
828 int32_t* addr = reinterpret_cast<int32_t*>(raw_addr);
829 accessor->Access(addr);
830}
831
832template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
833inline void Object::UpdateField64ViaAccessor(MemberOffset field_offset,
834 Accessor<int64_t>* accessor) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700835 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Orion Hodson928033d2018-02-07 05:30:54 +0000836 if (kTransactionActive) {
837 static const bool kIsVolatile = true;
838 int64_t old_value = GetField64<kVerifyFlags, kIsVolatile>(field_offset);
839 Runtime::Current()->RecordWriteField64(this, field_offset, old_value, kIsVolatile);
840 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700841 Verify<kVerifyFlags>();
Orion Hodson928033d2018-02-07 05:30:54 +0000842 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
843 int64_t* addr = reinterpret_cast<int64_t*>(raw_addr);
844 accessor->Access(addr);
845}
846
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800847template<bool kIsStatic,
848 VerifyObjectFlags kVerifyFlags,
849 ReadBarrierOption kReadBarrierOption,
850 typename Visitor>
Mathieu Chartier407f7022014-02-18 14:37:05 -0800851inline void Object::VisitFieldsReferences(uint32_t ref_offsets, const Visitor& visitor) {
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700852 if (!kIsStatic && (ref_offsets != mirror::Class::kClassWalkSuper)) {
853 // Instance fields and not the slow-path.
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700854 uint32_t field_offset = mirror::kObjectHeaderSize;
Mathieu Chartier407f7022014-02-18 14:37:05 -0800855 while (ref_offsets != 0) {
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700856 if ((ref_offsets & 1) != 0) {
857 visitor(this, MemberOffset(field_offset), kIsStatic);
858 }
859 ref_offsets >>= 1;
860 field_offset += sizeof(mirror::HeapReference<mirror::Object>);
Mathieu Chartier407f7022014-02-18 14:37:05 -0800861 }
862 } else {
Mingyao Yangfaff0f02014-09-10 12:03:22 -0700863 // There is no reference offset bitmap. In the non-static case, walk up the class
Mathieu Chartier407f7022014-02-18 14:37:05 -0800864 // inheritance hierarchy and find reference offsets the hard way. In the static case, just
865 // consider this class.
Mathieu Chartier31e88222016-10-14 18:43:19 -0700866 for (ObjPtr<Class> klass = kIsStatic
Mathieu Chartierd7a7f2f2018-09-07 11:57:18 -0700867 ? AsClass<kVerifyFlags>()
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800868 : GetClass<kVerifyFlags, kReadBarrierOption>();
869 klass != nullptr;
870 klass = kIsStatic ? nullptr : klass->GetSuperClass<kVerifyFlags, kReadBarrierOption>()) {
871 const size_t num_reference_fields =
Mathieu Chartier407f7022014-02-18 14:37:05 -0800872 kIsStatic ? klass->NumReferenceStaticFields() : klass->NumReferenceInstanceFields();
Vladimir Marko76649e82014-11-10 18:32:59 +0000873 if (num_reference_fields == 0u) {
874 continue;
875 }
Mathieu Chartiere401d142015-04-22 13:56:20 -0700876 // Presumably GC can happen when we are cross compiling, it should not cause performance
877 // problems to do pointer size logic.
Vladimir Marko76649e82014-11-10 18:32:59 +0000878 MemberOffset field_offset = kIsStatic
Vladimir Marko98db89c2018-09-07 11:45:46 +0100879 ? klass->GetFirstReferenceStaticFieldOffset<kVerifyFlags>(
Mathieu Chartiere401d142015-04-22 13:56:20 -0700880 Runtime::Current()->GetClassLinker()->GetImagePointerSize())
Hiroshi Yamauchi5496f692016-02-17 13:29:59 -0800881 : klass->GetFirstReferenceInstanceFieldOffset<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartier059ef3d2015-08-18 13:54:21 -0700882 for (size_t i = 0u; i < num_reference_fields; ++i) {
Mathieu Chartier407f7022014-02-18 14:37:05 -0800883 // TODO: Do a simpler check?
Mathieu Chartier059ef3d2015-08-18 13:54:21 -0700884 if (field_offset.Uint32Value() != ClassOffset().Uint32Value()) {
Mathieu Chartier52e4b432014-06-10 11:22:31 -0700885 visitor(this, field_offset, kIsStatic);
Mathieu Chartier407f7022014-02-18 14:37:05 -0800886 }
Vladimir Marko76649e82014-11-10 18:32:59 +0000887 field_offset = MemberOffset(field_offset.Uint32Value() +
888 sizeof(mirror::HeapReference<mirror::Object>));
Mathieu Chartier407f7022014-02-18 14:37:05 -0800889 }
890 }
891 }
892}
893
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800894template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption, typename Visitor>
Mathieu Chartier31e88222016-10-14 18:43:19 -0700895inline void Object::VisitInstanceFieldsReferences(ObjPtr<Class> klass, const Visitor& visitor) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800896 VisitFieldsReferences<false, kVerifyFlags, kReadBarrierOption>(
897 klass->GetReferenceInstanceOffsets<kVerifyFlags>(), visitor);
Mathieu Chartier407f7022014-02-18 14:37:05 -0800898}
899
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800900template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption, typename Visitor>
Mathieu Chartier31e88222016-10-14 18:43:19 -0700901inline void Object::VisitStaticFieldsReferences(ObjPtr<Class> klass, const Visitor& visitor) {
Vladimir Marko98db89c2018-09-07 11:45:46 +0100902 DCHECK(!klass->IsTemp<kVerifyFlags>());
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800903 klass->VisitFieldsReferences<true, kVerifyFlags, kReadBarrierOption>(0, visitor);
Mathieu Chartier407f7022014-02-18 14:37:05 -0800904}
905
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800906template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere4275c02015-08-06 15:34:15 -0700907inline bool Object::IsClassLoader() {
Vladimir Marko98db89c2018-09-07 11:45:46 +0100908 return GetClass<kVerifyFlags, kReadBarrierOption>()->template IsClassLoaderClass<kVerifyFlags>();
Mathieu Chartiere4275c02015-08-06 15:34:15 -0700909}
910
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800911template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere4275c02015-08-06 15:34:15 -0700912inline mirror::ClassLoader* Object::AsClassLoader() {
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800913 DCHECK((IsClassLoader<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartiere4275c02015-08-06 15:34:15 -0700914 return down_cast<mirror::ClassLoader*>(this);
915}
916
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800917template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Vladimir Marko05792b92015-08-03 11:56:49 +0100918inline bool Object::IsDexCache() {
Vladimir Marko98db89c2018-09-07 11:45:46 +0100919 return GetClass<kVerifyFlags, kReadBarrierOption>()->template IsDexCacheClass<kVerifyFlags>();
Vladimir Marko05792b92015-08-03 11:56:49 +0100920}
921
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800922template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Vladimir Marko05792b92015-08-03 11:56:49 +0100923inline mirror::DexCache* Object::AsDexCache() {
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800924 DCHECK((IsDexCache<kVerifyFlags, kReadBarrierOption>()));
Vladimir Marko05792b92015-08-03 11:56:49 +0100925 return down_cast<mirror::DexCache*>(this);
926}
927
Mathieu Chartier99111282018-06-19 12:30:56 -0700928template<bool kTransactionActive, bool kCheckTransaction>
929inline void Object::VerifyTransaction() {
930 if (kCheckTransaction) {
931 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
932 }
933}
934
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800935} // namespace mirror
936} // namespace art
937
Brian Carlstromfc0e3212013-07-17 14:40:12 -0700938#endif // ART_RUNTIME_MIRROR_OBJECT_INL_H_