blob: 2476b135ffdc28d857b7040c7731d843c01fca82 [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
Vladimir Markof52d92f2019-03-29 12:33:02 +000082inline ObjPtr<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
Vladimir Markof52d92f2019-03-29 12:33:02 +000086inline ObjPtr<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() {
Vladimir Markoa040ddc2019-03-15 13:14:11 +0000142 // OK to look at from-space copies since java.lang.Class.class is non-moveable
143 // (even when running without boot image, see ClassLinker::InitWithoutImage())
144 // and we're reading constant references for comparison only. See ReadBarrierOption.
Mathieu Chartierd7a7f2f2018-09-07 11:57:18 -0700145 ObjPtr<Class> klass = GetClass<kVerifyFlags, kWithoutReadBarrier>();
Vladimir Marko98db89c2018-09-07 11:45:46 +0100146 ObjPtr<Class> java_lang_Class = klass->GetClass<kVerifyFlags, kWithoutReadBarrier>();
Mathieu Chartierd7a7f2f2018-09-07 11:57:18 -0700147 return klass == java_lang_Class;
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800148}
149
Mathieu Chartierd7a7f2f2018-09-07 11:57:18 -0700150template<VerifyObjectFlags kVerifyFlags>
Vladimir Marko4617d582019-03-28 13:48:31 +0000151inline ObjPtr<Class> Object::AsClass() {
Mathieu Chartierd7a7f2f2018-09-07 11:57:18 -0700152 DCHECK((IsClass<kVerifyFlags>()));
Vladimir Marko4617d582019-03-28 13:48:31 +0000153 return ObjPtr<Class>::DownCast(this);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800154}
155
Vladimir Marko98db89c2018-09-07 11:45:46 +0100156template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800157inline bool Object::IsObjectArray() {
Vladimir Marko98db89c2018-09-07 11:45:46 +0100158 // We do not need a read barrier here as the primitive type is constant,
159 // both from-space and to-space component type classes shall yield the same result.
Vladimir Markodbcb48f2018-11-12 11:47:04 +0000160 constexpr VerifyObjectFlags kNewFlags = RemoveThisFlags(kVerifyFlags);
Vladimir Marko98db89c2018-09-07 11:45:46 +0100161 return IsArrayInstance<kVerifyFlags>() &&
162 !GetClass<kNewFlags, kWithoutReadBarrier>()->
163 template GetComponentType<kNewFlags, kWithoutReadBarrier>()->IsPrimitive();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800164}
165
Vladimir Marko98db89c2018-09-07 11:45:46 +0100166template<class T, VerifyObjectFlags kVerifyFlags>
Vladimir Marko4617d582019-03-28 13:48:31 +0000167inline ObjPtr<ObjectArray<T>> Object::AsObjectArray() {
Vladimir Marko98db89c2018-09-07 11:45:46 +0100168 DCHECK((IsObjectArray<kVerifyFlags>()));
Vladimir Marko4617d582019-03-28 13:48:31 +0000169 return ObjPtr<ObjectArray<T>>::DownCast(this);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800170}
171
Vladimir Marko98db89c2018-09-07 11:45:46 +0100172template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800173inline bool Object::IsArrayInstance() {
Vladimir Marko98db89c2018-09-07 11:45:46 +0100174 // We do not need a read barrier here, both from-space and to-space version of the class
175 // shall return the same result from IsArrayClass().
176 return GetClass<kVerifyFlags, kWithoutReadBarrier>()->template IsArrayClass<kVerifyFlags>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800177}
178
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800179template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800180inline bool Object::IsReferenceInstance() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800181 return GetClass<kVerifyFlags, kReadBarrierOption>()->IsTypeOfReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800182}
183
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800184template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Vladimir Marko4617d582019-03-28 13:48:31 +0000185inline ObjPtr<Reference> Object::AsReference() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800186 DCHECK((IsReferenceInstance<kVerifyFlags, kReadBarrierOption>()));
Vladimir Marko4617d582019-03-28 13:48:31 +0000187 return ObjPtr<Reference>::DownCast(this);
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700188}
189
Vladimir Marko98db89c2018-09-07 11:45:46 +0100190template<VerifyObjectFlags kVerifyFlags>
Vladimir Marko4617d582019-03-28 13:48:31 +0000191inline ObjPtr<Array> Object::AsArray() {
Vladimir Marko98db89c2018-09-07 11:45:46 +0100192 DCHECK((IsArrayInstance<kVerifyFlags>()));
Vladimir Marko4617d582019-03-28 13:48:31 +0000193 return ObjPtr<Array>::DownCast(this);
Ian Rogers05f30572013-02-20 12:13:11 -0800194}
195
Vladimir Marko104883b2018-11-09 17:12:23 +0000196template<VerifyObjectFlags kVerifyFlags, Primitive::Type kType>
197ALWAYS_INLINE bool Object::IsSpecificPrimitiveArray() {
Vladimir Markoa040ddc2019-03-15 13:14:11 +0000198 // We do not need a read barrier here as the primitive type is constant, both from-space
199 // and to-space component type classes shall yield the same result. See ReadBarrierOption.
Vladimir Marko0984e482019-03-27 16:41:41 +0000200 const ObjPtr<Class> klass = GetClass<kVerifyFlags, kWithoutReadBarrier>();
Vladimir Markodbcb48f2018-11-12 11:47:04 +0000201 constexpr VerifyObjectFlags kNewFlags = RemoveThisFlags(kVerifyFlags);
Vladimir Marko0984e482019-03-27 16:41:41 +0000202 const ObjPtr<Class> component_type = klass->GetComponentType<kNewFlags, kWithoutReadBarrier>();
Vladimir Marko104883b2018-11-09 17:12:23 +0000203 return component_type != nullptr &&
204 component_type->GetPrimitiveType<kNewFlags>() == kType;
205}
206
207template<VerifyObjectFlags kVerifyFlags>
208inline bool Object::IsBooleanArray() {
209 return IsSpecificPrimitiveArray<kVerifyFlags, Primitive::kPrimBoolean>();
210}
211
Mathieu Chartier4e305412014-02-19 10:54:44 -0800212template<VerifyObjectFlags kVerifyFlags>
Vladimir Marko4617d582019-03-28 13:48:31 +0000213inline ObjPtr<BooleanArray> Object::AsBooleanArray() {
Vladimir Marko104883b2018-11-09 17:12:23 +0000214 DCHECK(IsBooleanArray<kVerifyFlags>());
Vladimir Marko4617d582019-03-28 13:48:31 +0000215 return ObjPtr<BooleanArray>::DownCast(this);
Ian Rogers05f30572013-02-20 12:13:11 -0800216}
217
Mathieu Chartier4e305412014-02-19 10:54:44 -0800218template<VerifyObjectFlags kVerifyFlags>
Vladimir Marko104883b2018-11-09 17:12:23 +0000219inline bool Object::IsByteArray() {
220 return IsSpecificPrimitiveArray<kVerifyFlags, Primitive::kPrimByte>();
221}
222
223template<VerifyObjectFlags kVerifyFlags>
Vladimir Marko4617d582019-03-28 13:48:31 +0000224inline ObjPtr<ByteArray> Object::AsByteArray() {
Vladimir Marko104883b2018-11-09 17:12:23 +0000225 DCHECK(IsByteArray<kVerifyFlags>());
Vladimir Marko4617d582019-03-28 13:48:31 +0000226 return ObjPtr<ByteArray>::DownCast(this);
Ian Rogers05f30572013-02-20 12:13:11 -0800227}
228
Mathieu Chartier4e305412014-02-19 10:54:44 -0800229template<VerifyObjectFlags kVerifyFlags>
Vladimir Marko104883b2018-11-09 17:12:23 +0000230inline bool Object::IsCharArray() {
231 return IsSpecificPrimitiveArray<kVerifyFlags, Primitive::kPrimChar>();
Ian Rogersef7d42f2014-01-06 12:55:46 -0800232}
233
Mathieu Chartier4e305412014-02-19 10:54:44 -0800234template<VerifyObjectFlags kVerifyFlags>
Vladimir Marko4617d582019-03-28 13:48:31 +0000235inline ObjPtr<CharArray> Object::AsCharArray() {
Vladimir Marko104883b2018-11-09 17:12:23 +0000236 DCHECK(IsCharArray<kVerifyFlags>());
Vladimir Marko4617d582019-03-28 13:48:31 +0000237 return ObjPtr<CharArray>::DownCast(this);
Ian Rogers05f30572013-02-20 12:13:11 -0800238}
239
Mathieu Chartier4e305412014-02-19 10:54:44 -0800240template<VerifyObjectFlags kVerifyFlags>
Vladimir Marko104883b2018-11-09 17:12:23 +0000241inline bool Object::IsShortArray() {
242 return IsSpecificPrimitiveArray<kVerifyFlags, Primitive::kPrimShort>();
243}
244
245template<VerifyObjectFlags kVerifyFlags>
Vladimir Marko4617d582019-03-28 13:48:31 +0000246inline ObjPtr<ShortArray> Object::AsShortArray() {
Vladimir Marko104883b2018-11-09 17:12:23 +0000247 DCHECK(IsShortArray<kVerifyFlags>());
Vladimir Marko4617d582019-03-28 13:48:31 +0000248 return ObjPtr<ShortArray>::DownCast(this);
Ian Rogers05f30572013-02-20 12:13:11 -0800249}
250
Mathieu Chartier4e305412014-02-19 10:54:44 -0800251template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700252inline bool Object::IsIntArray() {
Vladimir Marko104883b2018-11-09 17:12:23 +0000253 return IsSpecificPrimitiveArray<kVerifyFlags, Primitive::kPrimInt>();
Mathieu Chartiere401d142015-04-22 13:56:20 -0700254}
255
Vladimir Marko104883b2018-11-09 17:12:23 +0000256template<VerifyObjectFlags kVerifyFlags>
Vladimir Marko4617d582019-03-28 13:48:31 +0000257inline ObjPtr<IntArray> Object::AsIntArrayUnchecked() {
258 return ObjPtr<IntArray>::DownCast(this);
Andreas Gampe3aa868a2019-02-04 11:18:43 -0800259}
260template<VerifyObjectFlags kVerifyFlags>
Vladimir Marko4617d582019-03-28 13:48:31 +0000261inline ObjPtr<IntArray> Object::AsIntArray() {
Vladimir Marko104883b2018-11-09 17:12:23 +0000262 DCHECK((IsIntArray<kVerifyFlags>()));
Andreas Gampe3aa868a2019-02-04 11:18:43 -0800263 return AsIntArrayUnchecked<kVerifyFlags>();
Ian Rogers05f30572013-02-20 12:13:11 -0800264}
265
Vladimir Marko104883b2018-11-09 17:12:23 +0000266template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700267inline bool Object::IsLongArray() {
Vladimir Marko104883b2018-11-09 17:12:23 +0000268 return IsSpecificPrimitiveArray<kVerifyFlags, Primitive::kPrimLong>();
Mathieu Chartiere401d142015-04-22 13:56:20 -0700269}
270
Vladimir Marko104883b2018-11-09 17:12:23 +0000271template<VerifyObjectFlags kVerifyFlags>
Vladimir Marko4617d582019-03-28 13:48:31 +0000272inline ObjPtr<LongArray> Object::AsLongArrayUnchecked() {
273 return ObjPtr<LongArray>::DownCast(this);
Andreas Gampe3aa868a2019-02-04 11:18:43 -0800274}
275template<VerifyObjectFlags kVerifyFlags>
Vladimir Marko4617d582019-03-28 13:48:31 +0000276inline ObjPtr<LongArray> Object::AsLongArray() {
Vladimir Marko104883b2018-11-09 17:12:23 +0000277 DCHECK((IsLongArray<kVerifyFlags>()));
Andreas Gampe3aa868a2019-02-04 11:18:43 -0800278 return AsLongArrayUnchecked<kVerifyFlags>();
Ian Rogers05f30572013-02-20 12:13:11 -0800279}
280
Mathieu Chartier4e305412014-02-19 10:54:44 -0800281template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700282inline bool Object::IsFloatArray() {
Vladimir Marko104883b2018-11-09 17:12:23 +0000283 return IsSpecificPrimitiveArray<kVerifyFlags, Primitive::kPrimFloat>();
Mathieu Chartiere401d142015-04-22 13:56:20 -0700284}
285
286template<VerifyObjectFlags kVerifyFlags>
Vladimir Marko4617d582019-03-28 13:48:31 +0000287inline ObjPtr<FloatArray> Object::AsFloatArray() {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700288 DCHECK(IsFloatArray<kVerifyFlags>());
Vladimir Marko4617d582019-03-28 13:48:31 +0000289 return ObjPtr<FloatArray>::DownCast(this);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100290}
291
Mathieu Chartier4e305412014-02-19 10:54:44 -0800292template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700293inline bool Object::IsDoubleArray() {
Vladimir Marko104883b2018-11-09 17:12:23 +0000294 return IsSpecificPrimitiveArray<kVerifyFlags, Primitive::kPrimDouble>();
Mathieu Chartiere401d142015-04-22 13:56:20 -0700295}
296
297template<VerifyObjectFlags kVerifyFlags>
Vladimir Marko4617d582019-03-28 13:48:31 +0000298inline ObjPtr<DoubleArray> Object::AsDoubleArray() {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700299 DCHECK(IsDoubleArray<kVerifyFlags>());
Vladimir Marko4617d582019-03-28 13:48:31 +0000300 return ObjPtr<DoubleArray>::DownCast(this);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100301}
302
Vladimir Markod355acf2019-03-21 17:09:40 +0000303template<VerifyObjectFlags kVerifyFlags>
Jeff Hao848f70a2014-01-15 13:49:50 -0800304inline bool Object::IsString() {
Vladimir Markod355acf2019-03-21 17:09:40 +0000305 // No read barrier is needed for reading a constant primitive field through
306 // constant reference field. See ReadBarrierOption.
307 return GetClass<kVerifyFlags, kWithoutReadBarrier>()->IsStringClass();
Jeff Hao848f70a2014-01-15 13:49:50 -0800308}
309
Vladimir Markod355acf2019-03-21 17:09:40 +0000310template<VerifyObjectFlags kVerifyFlags>
Vladimir Marko4617d582019-03-28 13:48:31 +0000311inline ObjPtr<String> Object::AsString() {
Vladimir Markod355acf2019-03-21 17:09:40 +0000312 DCHECK((IsString<kVerifyFlags>()));
Vladimir Marko4617d582019-03-28 13:48:31 +0000313 return ObjPtr<String>::DownCast(this);
Ian Rogers05f30572013-02-20 12:13:11 -0800314}
315
Mathieu Chartier4e305412014-02-19 10:54:44 -0800316template<VerifyObjectFlags kVerifyFlags>
Vladimir Marko4617d582019-03-28 13:48:31 +0000317inline ObjPtr<Throwable> Object::AsThrowable() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800318 DCHECK(GetClass<kVerifyFlags>()->IsThrowableClass());
Vladimir Marko4617d582019-03-28 13:48:31 +0000319 return ObjPtr<Throwable>::DownCast(this);
Ian Rogers05f30572013-02-20 12:13:11 -0800320}
321
Mathieu Chartier4e305412014-02-19 10:54:44 -0800322template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800323inline bool Object::IsWeakReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800324 return GetClass<kVerifyFlags>()->IsWeakReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800325}
326
Mathieu Chartier4e305412014-02-19 10:54:44 -0800327template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800328inline bool Object::IsSoftReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800329 return GetClass<kVerifyFlags>()->IsSoftReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800330}
331
Mathieu Chartier4e305412014-02-19 10:54:44 -0800332template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800333inline bool Object::IsFinalizerReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800334 return GetClass<kVerifyFlags>()->IsFinalizerReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800335}
336
Mathieu Chartier4e305412014-02-19 10:54:44 -0800337template<VerifyObjectFlags kVerifyFlags>
Vladimir Marko4617d582019-03-28 13:48:31 +0000338inline ObjPtr<FinalizerReference> Object::AsFinalizerReference() {
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700339 DCHECK(IsFinalizerReferenceInstance<kVerifyFlags>());
Vladimir Marko4617d582019-03-28 13:48:31 +0000340 return ObjPtr<FinalizerReference>::DownCast(this);
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700341}
342
343template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800344inline bool Object::IsPhantomReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800345 return GetClass<kVerifyFlags>()->IsPhantomReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800346}
347
Mathieu Chartierd08f66f2017-04-13 11:47:53 -0700348template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800349inline size_t Object::SizeOf() {
Mathieu Chartierd08f66f2017-04-13 11:47:53 -0700350 // Read barrier is never required for SizeOf since objects sizes are constant. Reading from-space
351 // values is OK because of that.
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800352 size_t result;
Vladimir Markodbcb48f2018-11-12 11:47:04 +0000353 constexpr VerifyObjectFlags kNewFlags = RemoveThisFlags(kVerifyFlags);
Vladimir Marko98db89c2018-09-07 11:45:46 +0100354 if (IsArrayInstance<kVerifyFlags>()) {
Vladimir Markod355acf2019-03-21 17:09:40 +0000355 result = AsArray<kNewFlags>()->template SizeOf<kNewFlags>();
Mathieu Chartierd7a7f2f2018-09-07 11:57:18 -0700356 } else if (IsClass<kNewFlags>()) {
Vladimir Markod355acf2019-03-21 17:09:40 +0000357 result = AsClass<kNewFlags>()->template SizeOf<kNewFlags>();
358 } else if (IsString<kNewFlags>()) {
359 result = AsString<kNewFlags>()->template SizeOf<kNewFlags>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800360 } else {
Vladimir Markod355acf2019-03-21 17:09:40 +0000361 result = GetClass<kNewFlags, kWithoutReadBarrier>()->template GetObjectSize<kNewFlags>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800362 }
Vladimir Markod355acf2019-03-21 17:09:40 +0000363 DCHECK_GE(result, sizeof(Object)) << " class="
364 // Note: Class::PrettyClass() is reading constant reference fields to get to constant
365 // primitive fields and safely avoids read barriers, so it is safe to call on a Class
366 // reference read without read barrier from a constant reference field.
367 // See ReadBarrierOption. And, for correctness, we actually have to avoid the read
368 // barrier here if Object::SizeOf() is called on a from-space reference.
369 << GetClass<kNewFlags, kWithoutReadBarrier>()->PrettyClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800370 return result;
371}
372
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700373template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
Fred Shih37f05ef2014-07-16 18:38:08 -0700374inline int8_t Object::GetFieldByte(MemberOffset field_offset) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700375 Verify<kVerifyFlags>();
David Srbeckyce32c102018-08-31 07:21:07 +0100376 return GetFieldPrimitive<int8_t, kIsVolatile>(field_offset);
Fred Shih37f05ef2014-07-16 18:38:08 -0700377}
378
379template<VerifyObjectFlags kVerifyFlags>
380inline uint8_t Object::GetFieldBooleanVolatile(MemberOffset field_offset) {
381 return GetFieldBoolean<kVerifyFlags, true>(field_offset);
382}
383
384template<VerifyObjectFlags kVerifyFlags>
385inline int8_t Object::GetFieldByteVolatile(MemberOffset field_offset) {
386 return GetFieldByte<kVerifyFlags, true>(field_offset);
387}
388
Roland Levillaind32ead22018-05-30 17:38:21 +0100389template<bool kTransactionActive,
390 bool kCheckTransaction,
391 VerifyObjectFlags kVerifyFlags,
392 bool kIsVolatile>
Mathieu Chartier99111282018-06-19 12:30:56 -0700393inline void Object::SetFieldBoolean(MemberOffset field_offset, uint8_t new_value) {
394 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Fred Shih37f05ef2014-07-16 18:38:08 -0700395 if (kTransactionActive) {
Roland Levillaind32ead22018-05-30 17:38:21 +0100396 Runtime::Current()->RecordWriteFieldBoolean(
397 this,
398 field_offset,
399 GetFieldBoolean<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<uint8_t, kIsVolatile>(field_offset, new_value);
Fred Shih37f05ef2014-07-16 18:38:08 -0700404}
405
Roland Levillaind32ead22018-05-30 17:38:21 +0100406template<bool kTransactionActive,
407 bool kCheckTransaction,
408 VerifyObjectFlags kVerifyFlags,
409 bool kIsVolatile>
Mathieu Chartier99111282018-06-19 12:30:56 -0700410inline void Object::SetFieldByte(MemberOffset field_offset, int8_t new_value) {
411 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Fred Shih37f05ef2014-07-16 18:38:08 -0700412 if (kTransactionActive) {
Roland Levillaind32ead22018-05-30 17:38:21 +0100413 Runtime::Current()->RecordWriteFieldByte(this,
414 field_offset,
415 GetFieldByte<kVerifyFlags, kIsVolatile>(field_offset),
416 kIsVolatile);
Fred Shih37f05ef2014-07-16 18:38:08 -0700417 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700418 Verify<kVerifyFlags>();
David Srbeckyce32c102018-08-31 07:21:07 +0100419 SetFieldPrimitive<int8_t, kIsVolatile>(field_offset, new_value);
Fred Shih37f05ef2014-07-16 18:38:08 -0700420}
421
422template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
423inline void Object::SetFieldBooleanVolatile(MemberOffset field_offset, uint8_t new_value) {
424 return SetFieldBoolean<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
425 field_offset, new_value);
426}
427
428template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
429inline void Object::SetFieldByteVolatile(MemberOffset field_offset, int8_t new_value) {
430 return SetFieldByte<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
431 field_offset, new_value);
432}
433
434template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
435inline uint16_t Object::GetFieldChar(MemberOffset field_offset) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700436 Verify<kVerifyFlags>();
David Srbeckyce32c102018-08-31 07:21:07 +0100437 return GetFieldPrimitive<uint16_t, kIsVolatile>(field_offset);
Fred Shih37f05ef2014-07-16 18:38:08 -0700438}
439
440template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
441inline int16_t Object::GetFieldShort(MemberOffset field_offset) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700442 Verify<kVerifyFlags>();
David Srbeckyce32c102018-08-31 07:21:07 +0100443 return GetFieldPrimitive<int16_t, kIsVolatile>(field_offset);
Fred Shih37f05ef2014-07-16 18:38:08 -0700444}
445
446template<VerifyObjectFlags kVerifyFlags>
447inline uint16_t Object::GetFieldCharVolatile(MemberOffset field_offset) {
448 return GetFieldChar<kVerifyFlags, true>(field_offset);
449}
450
451template<VerifyObjectFlags kVerifyFlags>
452inline int16_t Object::GetFieldShortVolatile(MemberOffset field_offset) {
453 return GetFieldShort<kVerifyFlags, true>(field_offset);
454}
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::SetFieldChar(MemberOffset field_offset, uint16_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 GetFieldChar<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<uint16_t, kIsVolatile>(field_offset, new_value);
Fred Shih37f05ef2014-07-16 18:38:08 -0700470}
471
Roland Levillaind32ead22018-05-30 17:38:21 +0100472template<bool kTransactionActive,
473 bool kCheckTransaction,
474 VerifyObjectFlags kVerifyFlags,
475 bool kIsVolatile>
Fred Shih37f05ef2014-07-16 18:38:08 -0700476inline void Object::SetFieldShort(MemberOffset field_offset, int16_t new_value) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700477 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Fred Shih37f05ef2014-07-16 18:38:08 -0700478 if (kTransactionActive) {
Roland Levillaind32ead22018-05-30 17:38:21 +0100479 Runtime::Current()->RecordWriteFieldChar(this,
480 field_offset,
481 GetFieldShort<kVerifyFlags, kIsVolatile>(field_offset),
482 kIsVolatile);
Fred Shih37f05ef2014-07-16 18:38:08 -0700483 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700484 Verify<kVerifyFlags>();
David Srbeckyce32c102018-08-31 07:21:07 +0100485 SetFieldPrimitive<int16_t, kIsVolatile>(field_offset, new_value);
Fred Shih37f05ef2014-07-16 18:38:08 -0700486}
487
488template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
489inline void Object::SetFieldCharVolatile(MemberOffset field_offset, uint16_t new_value) {
490 return SetFieldChar<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
491 field_offset, new_value);
492}
493
494template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
495inline void Object::SetFieldShortVolatile(MemberOffset field_offset, int16_t new_value) {
496 return SetFieldShort<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
497 field_offset, new_value);
498}
499
Roland Levillaind32ead22018-05-30 17:38:21 +0100500template<bool kTransactionActive,
501 bool kCheckTransaction,
502 VerifyObjectFlags kVerifyFlags,
503 bool kIsVolatile>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700504inline void Object::SetField32(MemberOffset field_offset, int32_t new_value) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700505 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100506 if (kTransactionActive) {
Roland Levillaind32ead22018-05-30 17:38:21 +0100507 Runtime::Current()->RecordWriteField32(this,
508 field_offset,
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700509 GetField32<kVerifyFlags, kIsVolatile>(field_offset),
510 kIsVolatile);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100511 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700512 Verify<kVerifyFlags>();
David Srbeckyce32c102018-08-31 07:21:07 +0100513 SetFieldPrimitive<int32_t, kIsVolatile>(field_offset, new_value);
Ian Rogersb122a4b2013-11-19 18:00:50 -0800514}
515
Mathieu Chartier4e305412014-02-19 10:54:44 -0800516template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700517inline void Object::SetField32Volatile(MemberOffset field_offset, int32_t new_value) {
518 SetField32<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset, new_value);
519}
520
Chang Xing6d3e7682017-07-11 10:31:29 -0700521template<bool kCheckTransaction, VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
522inline void Object::SetField32Transaction(MemberOffset field_offset, int32_t new_value) {
523 if (Runtime::Current()->IsActiveTransaction()) {
524 SetField32<true, kCheckTransaction, kVerifyFlags, kIsVolatile>(field_offset, new_value);
525 } else {
526 SetField32<false, kCheckTransaction, kVerifyFlags, kIsVolatile>(field_offset, new_value);
527 }
528}
529
Roland Levillaind32ead22018-05-30 17:38:21 +0100530template<bool kTransactionActive,
531 bool kCheckTransaction,
532 VerifyObjectFlags kVerifyFlags,
533 bool kIsVolatile>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700534inline void Object::SetField64(MemberOffset field_offset, int64_t new_value) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700535 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100536 if (kTransactionActive) {
Roland Levillaind32ead22018-05-30 17:38:21 +0100537 Runtime::Current()->RecordWriteField64(this,
538 field_offset,
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700539 GetField64<kVerifyFlags, kIsVolatile>(field_offset),
540 kIsVolatile);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100541 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700542 Verify<kVerifyFlags>();
David Srbeckyce32c102018-08-31 07:21:07 +0100543 SetFieldPrimitive<int64_t, kIsVolatile>(field_offset, new_value);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800544}
545
Mathieu Chartier4e305412014-02-19 10:54:44 -0800546template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700547inline void Object::SetField64Volatile(MemberOffset field_offset, int64_t new_value) {
548 return SetField64<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
549 new_value);
550}
551
Chang Xing6d3e7682017-07-11 10:31:29 -0700552template<bool kCheckTransaction, VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
553inline void Object::SetField64Transaction(MemberOffset field_offset, int32_t new_value) {
554 if (Runtime::Current()->IsActiveTransaction()) {
555 SetField64<true, kCheckTransaction, kVerifyFlags, kIsVolatile>(field_offset, new_value);
556 } else {
557 SetField64<false, kCheckTransaction, kVerifyFlags, kIsVolatile>(field_offset, new_value);
558 }
559}
560
Mathieu Chartierc381c362016-08-23 13:27:53 -0700561template<typename kSize>
562inline kSize Object::GetFieldAcquire(MemberOffset field_offset) {
563 const uint8_t* raw_addr = reinterpret_cast<const uint8_t*>(this) + field_offset.Int32Value();
564 const kSize* addr = reinterpret_cast<const kSize*>(raw_addr);
Orion Hodson88591fe2018-03-06 13:35:43 +0000565 return reinterpret_cast<const Atomic<kSize>*>(addr)->load(std::memory_order_acquire);
Mathieu Chartierc381c362016-08-23 13:27:53 -0700566}
567
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700568template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -0700569inline bool Object::CasFieldWeakSequentiallyConsistent64(MemberOffset field_offset,
Roland Levillaind32ead22018-05-30 17:38:21 +0100570 int64_t old_value,
571 int64_t new_value) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700572 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100573 if (kTransactionActive) {
574 Runtime::Current()->RecordWriteField64(this, field_offset, old_value, true);
575 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700576 Verify<kVerifyFlags>();
Ian Rogers13735952014-10-08 12:43:28 -0700577 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogers228602f2014-07-10 02:07:54 -0700578 Atomic<int64_t>* atomic_addr = reinterpret_cast<Atomic<int64_t>*>(raw_addr);
Orion Hodson4557b382018-01-03 11:47:54 +0000579 return atomic_addr->CompareAndSetWeakSequentiallyConsistent(old_value, new_value);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800580}
581
Hans Boehmd8434432014-07-11 09:56:07 -0700582template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
583inline bool Object::CasFieldStrongSequentiallyConsistent64(MemberOffset field_offset,
Roland Levillaind32ead22018-05-30 17:38:21 +0100584 int64_t old_value,
585 int64_t new_value) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700586 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Hans Boehmd8434432014-07-11 09:56:07 -0700587 if (kTransactionActive) {
588 Runtime::Current()->RecordWriteField64(this, field_offset, old_value, true);
589 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700590 Verify<kVerifyFlags>();
Ian Rogers13735952014-10-08 12:43:28 -0700591 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Hans Boehmd8434432014-07-11 09:56:07 -0700592 Atomic<int64_t>* atomic_addr = reinterpret_cast<Atomic<int64_t>*>(raw_addr);
Orion Hodson4557b382018-01-03 11:47:54 +0000593 return atomic_addr->CompareAndSetStrongSequentiallyConsistent(old_value, new_value);
Hans Boehmd8434432014-07-11 09:56:07 -0700594}
595
Chris Wailes0c61be42018-09-26 17:27:34 -0700596/*
597 * Returns a pointer to an object representing what the field points to, not an
598 * object representing the field.
599 */
Roland Levillaind32ead22018-05-30 17:38:21 +0100600template<class T,
601 VerifyObjectFlags kVerifyFlags,
602 ReadBarrierOption kReadBarrierOption,
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700603 bool kIsVolatile>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700604inline T* Object::GetFieldObject(MemberOffset field_offset) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700605 Verify<kVerifyFlags>();
Ian Rogers13735952014-10-08 12:43:28 -0700606 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogersef7d42f2014-01-06 12:55:46 -0800607 HeapReference<T>* objref_addr = reinterpret_cast<HeapReference<T>*>(raw_addr);
Hans Boehmcc55e1d2017-07-27 15:28:07 -0700608 T* result = ReadBarrier::Barrier<T, kIsVolatile, kReadBarrierOption>(
609 this,
610 field_offset,
611 objref_addr);
Mathieu Chartier99111282018-06-19 12:30:56 -0700612 VerifyRead<kVerifyFlags>(result);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800613 return result;
614}
615
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700616template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700617inline T* Object::GetFieldObjectVolatile(MemberOffset field_offset) {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700618 return GetFieldObject<T, kVerifyFlags, kReadBarrierOption, true>(field_offset);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700619}
620
Roland Levillaind32ead22018-05-30 17:38:21 +0100621template<bool kTransactionActive,
622 bool kCheckTransaction,
623 VerifyObjectFlags kVerifyFlags,
624 bool kIsVolatile>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700625inline void Object::SetFieldObjectWithoutWriteBarrier(MemberOffset field_offset,
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700626 ObjPtr<Object> new_value) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700627 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100628 if (kTransactionActive) {
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700629 ObjPtr<Object> obj;
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700630 if (kIsVolatile) {
631 obj = GetFieldObjectVolatile<Object>(field_offset);
632 } else {
633 obj = GetFieldObject<Object>(field_offset);
634 }
Vladimir Markobcf17522018-06-01 13:14:32 +0100635 Runtime::Current()->RecordWriteFieldReference(this, field_offset, obj, true);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100636 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700637 Verify<kVerifyFlags>();
638 VerifyWrite<kVerifyFlags>(new_value);
Ian Rogers13735952014-10-08 12:43:28 -0700639 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogersef7d42f2014-01-06 12:55:46 -0800640 HeapReference<Object>* objref_addr = reinterpret_cast<HeapReference<Object>*>(raw_addr);
Hans Boehmcc55e1d2017-07-27 15:28:07 -0700641 objref_addr->Assign<kIsVolatile>(new_value.Ptr());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800642}
643
Roland Levillaind32ead22018-05-30 17:38:21 +0100644template<bool kTransactionActive,
645 bool kCheckTransaction,
646 VerifyObjectFlags kVerifyFlags,
647 bool kIsVolatile>
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700648inline void Object::SetFieldObject(MemberOffset field_offset, ObjPtr<Object> new_value) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700649 SetFieldObjectWithoutWriteBarrier<kTransactionActive, kCheckTransaction, kVerifyFlags,
650 kIsVolatile>(field_offset, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800651 if (new_value != nullptr) {
Mathieu Chartier88ea61e2018-06-20 17:45:41 -0700652 WriteBarrier::ForFieldWrite<WriteBarrier::kWithoutNullCheck>(this, field_offset, new_value);
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700653 // TODO: Check field assignment could theoretically cause thread suspension, TODO: fix this.
654 CheckFieldAssignment(field_offset, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800655 }
656}
657
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700658template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700659inline void Object::SetFieldObjectVolatile(MemberOffset field_offset, ObjPtr<Object> new_value) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700660 SetFieldObject<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
661 new_value);
662}
663
Chang Xing6d3e7682017-07-11 10:31:29 -0700664template<bool kCheckTransaction, VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
665inline void Object::SetFieldObjectTransaction(MemberOffset field_offset, ObjPtr<Object> new_value) {
666 if (Runtime::Current()->IsActiveTransaction()) {
667 SetFieldObject<true, kCheckTransaction, kVerifyFlags, kIsVolatile>(field_offset, new_value);
668 } else {
669 SetFieldObject<false, kCheckTransaction, kVerifyFlags, kIsVolatile>(field_offset, new_value);
670 }
671}
672
Mathieu Chartier4e305412014-02-19 10:54:44 -0800673template <VerifyObjectFlags kVerifyFlags>
674inline HeapReference<Object>* Object::GetFieldObjectReferenceAddr(MemberOffset field_offset) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700675 Verify<kVerifyFlags>();
Ian Rogers13735952014-10-08 12:43:28 -0700676 return reinterpret_cast<HeapReference<Object>*>(reinterpret_cast<uint8_t*>(this) +
Mathieu Chartier4e305412014-02-19 10:54:44 -0800677 field_offset.Int32Value());
678}
679
680template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Mathieu Chartiera9746b92018-06-22 10:25:40 -0700681inline bool Object::CasFieldObjectWithoutWriteBarrier(MemberOffset field_offset,
682 ObjPtr<Object> old_value,
683 ObjPtr<Object> new_value,
684 CASMode mode,
685 std::memory_order memory_order) {
686 VerifyTransaction<kTransactionActive, kCheckTransaction>();
687 VerifyCAS<kVerifyFlags>(new_value, old_value);
688 if (kTransactionActive) {
689 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
690 }
691 uint32_t old_ref(PtrCompression<kPoisonHeapReferences, Object>::Compress(old_value));
692 uint32_t new_ref(PtrCompression<kPoisonHeapReferences, Object>::Compress(new_value));
693 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
694 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
695 return atomic_addr->CompareAndSet(old_ref, new_ref, mode, memory_order);
696}
697
698template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
699inline bool Object::CasFieldObject(MemberOffset field_offset,
700 ObjPtr<Object> old_value,
701 ObjPtr<Object> new_value,
702 CASMode mode,
703 std::memory_order memory_order) {
704 bool success = CasFieldObjectWithoutWriteBarrier<
705 kTransactionActive, kCheckTransaction, kVerifyFlags>(field_offset,
706 old_value,
707 new_value,
708 mode,
709 memory_order);
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800710 if (success) {
Mathieu Chartier88ea61e2018-06-20 17:45:41 -0700711 WriteBarrier::ForFieldWrite(this, field_offset, new_value);
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800712 }
713 return success;
714}
715
716template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Orion Hodson928033d2018-02-07 05:30:54 +0000717inline ObjPtr<Object> Object::CompareAndExchangeFieldObject(MemberOffset field_offset,
718 ObjPtr<Object> old_value,
719 ObjPtr<Object> new_value) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700720 VerifyTransaction<kTransactionActive, kCheckTransaction>();
721 VerifyCAS<kVerifyFlags>(new_value, old_value);
Orion Hodson928033d2018-02-07 05:30:54 +0000722 uint32_t old_ref(PtrCompression<kPoisonHeapReferences, Object>::Compress(old_value));
723 uint32_t new_ref(PtrCompression<kPoisonHeapReferences, Object>::Compress(new_value));
724 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
725 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
Orion Hodson88591fe2018-03-06 13:35:43 +0000726 bool success = atomic_addr->compare_exchange_strong(old_ref, new_ref, std::memory_order_seq_cst);
Orion Hodson928033d2018-02-07 05:30:54 +0000727 ObjPtr<Object> witness_value(PtrCompression<kPoisonHeapReferences, Object>::Decompress(old_ref));
728 if (kIsDebugBuild) {
729 // Ensure caller has done read barrier on the reference field so it's in the to-space.
730 ReadBarrier::AssertToSpaceInvariant(witness_value.Ptr());
731 }
Mathieu Chartier1d2e2662018-06-19 14:02:12 -0700732 if (success) {
733 if (kTransactionActive) {
734 Runtime::Current()->RecordWriteFieldReference(this, field_offset, witness_value, true);
735 }
Mathieu Chartier88ea61e2018-06-20 17:45:41 -0700736 WriteBarrier::ForFieldWrite(this, field_offset, new_value);
Orion Hodson928033d2018-02-07 05:30:54 +0000737 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700738 VerifyRead<kVerifyFlags>(witness_value);
Orion Hodson928033d2018-02-07 05:30:54 +0000739 return witness_value;
740}
741
742template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
743inline ObjPtr<Object> Object::ExchangeFieldObject(MemberOffset field_offset,
744 ObjPtr<Object> new_value) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700745 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Andreas Gampe98ea9d92018-10-19 14:06:15 -0700746 VerifyCAS<kVerifyFlags>(new_value, /*old_value=*/ nullptr);
Mathieu Chartier99111282018-06-19 12:30:56 -0700747
Orion Hodson928033d2018-02-07 05:30:54 +0000748 uint32_t new_ref(PtrCompression<kPoisonHeapReferences, Object>::Compress(new_value));
749 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
750 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
Orion Hodson88591fe2018-03-06 13:35:43 +0000751 uint32_t old_ref = atomic_addr->exchange(new_ref, std::memory_order_seq_cst);
Orion Hodson928033d2018-02-07 05:30:54 +0000752 ObjPtr<Object> old_value(PtrCompression<kPoisonHeapReferences, Object>::Decompress(old_ref));
753 if (kIsDebugBuild) {
754 // Ensure caller has done read barrier on the reference field so it's in the to-space.
755 ReadBarrier::AssertToSpaceInvariant(old_value.Ptr());
756 }
757 if (kTransactionActive) {
758 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
759 }
Mathieu Chartier88ea61e2018-06-20 17:45:41 -0700760 WriteBarrier::ForFieldWrite(this, field_offset, new_value);
Mathieu Chartier99111282018-06-19 12:30:56 -0700761 VerifyRead<kVerifyFlags>(old_value);
Orion Hodson928033d2018-02-07 05:30:54 +0000762 return old_value;
763}
764
765template<typename T, VerifyObjectFlags kVerifyFlags>
766inline void Object::GetPrimitiveFieldViaAccessor(MemberOffset field_offset, Accessor<T>* accessor) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700767 Verify<kVerifyFlags>();
Orion Hodson928033d2018-02-07 05:30:54 +0000768 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
769 T* addr = reinterpret_cast<T*>(raw_addr);
770 accessor->Access(addr);
771}
772
773template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
774inline void Object::UpdateFieldBooleanViaAccessor(MemberOffset field_offset,
775 Accessor<uint8_t>* accessor) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700776 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Orion Hodson928033d2018-02-07 05:30:54 +0000777 if (kTransactionActive) {
778 static const bool kIsVolatile = true;
779 uint8_t old_value = GetFieldBoolean<kVerifyFlags, kIsVolatile>(field_offset);
780 Runtime::Current()->RecordWriteFieldBoolean(this, field_offset, old_value, kIsVolatile);
781 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700782 Verify<kVerifyFlags>();
Orion Hodson928033d2018-02-07 05:30:54 +0000783 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
784 uint8_t* addr = raw_addr;
785 accessor->Access(addr);
786}
787
788template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
789inline void Object::UpdateFieldByteViaAccessor(MemberOffset field_offset,
790 Accessor<int8_t>* accessor) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700791 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Orion Hodson928033d2018-02-07 05:30:54 +0000792 if (kTransactionActive) {
793 static const bool kIsVolatile = true;
794 int8_t old_value = GetFieldByte<kVerifyFlags, kIsVolatile>(field_offset);
795 Runtime::Current()->RecordWriteFieldByte(this, field_offset, old_value, kIsVolatile);
796 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700797 Verify<kVerifyFlags>();
Orion Hodson928033d2018-02-07 05:30:54 +0000798 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
799 int8_t* addr = reinterpret_cast<int8_t*>(raw_addr);
800 accessor->Access(addr);
801}
802
803template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
804inline void Object::UpdateFieldCharViaAccessor(MemberOffset field_offset,
805 Accessor<uint16_t>* accessor) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700806 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Orion Hodson928033d2018-02-07 05:30:54 +0000807 if (kTransactionActive) {
808 static const bool kIsVolatile = true;
809 uint16_t old_value = GetFieldChar<kVerifyFlags, kIsVolatile>(field_offset);
810 Runtime::Current()->RecordWriteFieldChar(this, field_offset, old_value, kIsVolatile);
811 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700812 Verify<kVerifyFlags>();
Orion Hodson928033d2018-02-07 05:30:54 +0000813 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
814 uint16_t* addr = reinterpret_cast<uint16_t*>(raw_addr);
815 accessor->Access(addr);
816}
817
818template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
819inline void Object::UpdateFieldShortViaAccessor(MemberOffset field_offset,
820 Accessor<int16_t>* accessor) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700821 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Orion Hodson928033d2018-02-07 05:30:54 +0000822 if (kTransactionActive) {
823 static const bool kIsVolatile = true;
824 int16_t old_value = GetFieldShort<kVerifyFlags, kIsVolatile>(field_offset);
825 Runtime::Current()->RecordWriteFieldShort(this, field_offset, old_value, kIsVolatile);
826 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700827 Verify<kVerifyFlags>();
Orion Hodson928033d2018-02-07 05:30:54 +0000828 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
829 int16_t* addr = reinterpret_cast<int16_t*>(raw_addr);
830 accessor->Access(addr);
831}
832
833template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
834inline void Object::UpdateField32ViaAccessor(MemberOffset field_offset,
835 Accessor<int32_t>* accessor) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700836 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Orion Hodson928033d2018-02-07 05:30:54 +0000837 if (kTransactionActive) {
838 static const bool kIsVolatile = true;
839 int32_t old_value = GetField32<kVerifyFlags, kIsVolatile>(field_offset);
840 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, kIsVolatile);
841 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700842 Verify<kVerifyFlags>();
Orion Hodson928033d2018-02-07 05:30:54 +0000843 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
844 int32_t* addr = reinterpret_cast<int32_t*>(raw_addr);
845 accessor->Access(addr);
846}
847
848template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
849inline void Object::UpdateField64ViaAccessor(MemberOffset field_offset,
850 Accessor<int64_t>* accessor) {
Mathieu Chartier99111282018-06-19 12:30:56 -0700851 VerifyTransaction<kTransactionActive, kCheckTransaction>();
Orion Hodson928033d2018-02-07 05:30:54 +0000852 if (kTransactionActive) {
853 static const bool kIsVolatile = true;
854 int64_t old_value = GetField64<kVerifyFlags, kIsVolatile>(field_offset);
855 Runtime::Current()->RecordWriteField64(this, field_offset, old_value, kIsVolatile);
856 }
Mathieu Chartier99111282018-06-19 12:30:56 -0700857 Verify<kVerifyFlags>();
Orion Hodson928033d2018-02-07 05:30:54 +0000858 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
859 int64_t* addr = reinterpret_cast<int64_t*>(raw_addr);
860 accessor->Access(addr);
861}
862
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800863template<bool kIsStatic,
864 VerifyObjectFlags kVerifyFlags,
865 ReadBarrierOption kReadBarrierOption,
866 typename Visitor>
Mathieu Chartier407f7022014-02-18 14:37:05 -0800867inline void Object::VisitFieldsReferences(uint32_t ref_offsets, const Visitor& visitor) {
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700868 if (!kIsStatic && (ref_offsets != mirror::Class::kClassWalkSuper)) {
869 // Instance fields and not the slow-path.
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700870 uint32_t field_offset = mirror::kObjectHeaderSize;
Mathieu Chartier407f7022014-02-18 14:37:05 -0800871 while (ref_offsets != 0) {
Ian Rogerscdc1aaf2014-10-09 13:21:38 -0700872 if ((ref_offsets & 1) != 0) {
873 visitor(this, MemberOffset(field_offset), kIsStatic);
874 }
875 ref_offsets >>= 1;
876 field_offset += sizeof(mirror::HeapReference<mirror::Object>);
Mathieu Chartier407f7022014-02-18 14:37:05 -0800877 }
878 } else {
Mingyao Yangfaff0f02014-09-10 12:03:22 -0700879 // There is no reference offset bitmap. In the non-static case, walk up the class
Mathieu Chartier407f7022014-02-18 14:37:05 -0800880 // inheritance hierarchy and find reference offsets the hard way. In the static case, just
881 // consider this class.
Mathieu Chartier31e88222016-10-14 18:43:19 -0700882 for (ObjPtr<Class> klass = kIsStatic
Mathieu Chartierd7a7f2f2018-09-07 11:57:18 -0700883 ? AsClass<kVerifyFlags>()
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800884 : GetClass<kVerifyFlags, kReadBarrierOption>();
885 klass != nullptr;
886 klass = kIsStatic ? nullptr : klass->GetSuperClass<kVerifyFlags, kReadBarrierOption>()) {
887 const size_t num_reference_fields =
Mathieu Chartier407f7022014-02-18 14:37:05 -0800888 kIsStatic ? klass->NumReferenceStaticFields() : klass->NumReferenceInstanceFields();
Vladimir Marko76649e82014-11-10 18:32:59 +0000889 if (num_reference_fields == 0u) {
890 continue;
891 }
Mathieu Chartiere401d142015-04-22 13:56:20 -0700892 // Presumably GC can happen when we are cross compiling, it should not cause performance
893 // problems to do pointer size logic.
Vladimir Marko76649e82014-11-10 18:32:59 +0000894 MemberOffset field_offset = kIsStatic
Vladimir Marko98db89c2018-09-07 11:45:46 +0100895 ? klass->GetFirstReferenceStaticFieldOffset<kVerifyFlags>(
Mathieu Chartiere401d142015-04-22 13:56:20 -0700896 Runtime::Current()->GetClassLinker()->GetImagePointerSize())
Hiroshi Yamauchi5496f692016-02-17 13:29:59 -0800897 : klass->GetFirstReferenceInstanceFieldOffset<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartier059ef3d2015-08-18 13:54:21 -0700898 for (size_t i = 0u; i < num_reference_fields; ++i) {
Mathieu Chartier407f7022014-02-18 14:37:05 -0800899 // TODO: Do a simpler check?
Mathieu Chartier059ef3d2015-08-18 13:54:21 -0700900 if (field_offset.Uint32Value() != ClassOffset().Uint32Value()) {
Mathieu Chartier52e4b432014-06-10 11:22:31 -0700901 visitor(this, field_offset, kIsStatic);
Mathieu Chartier407f7022014-02-18 14:37:05 -0800902 }
Vladimir Marko76649e82014-11-10 18:32:59 +0000903 field_offset = MemberOffset(field_offset.Uint32Value() +
904 sizeof(mirror::HeapReference<mirror::Object>));
Mathieu Chartier407f7022014-02-18 14:37:05 -0800905 }
906 }
907 }
908}
909
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800910template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption, typename Visitor>
Mathieu Chartier31e88222016-10-14 18:43:19 -0700911inline void Object::VisitInstanceFieldsReferences(ObjPtr<Class> klass, const Visitor& visitor) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800912 VisitFieldsReferences<false, kVerifyFlags, kReadBarrierOption>(
913 klass->GetReferenceInstanceOffsets<kVerifyFlags>(), visitor);
Mathieu Chartier407f7022014-02-18 14:37:05 -0800914}
915
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800916template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption, typename Visitor>
Mathieu Chartier31e88222016-10-14 18:43:19 -0700917inline void Object::VisitStaticFieldsReferences(ObjPtr<Class> klass, const Visitor& visitor) {
Vladimir Marko98db89c2018-09-07 11:45:46 +0100918 DCHECK(!klass->IsTemp<kVerifyFlags>());
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800919 klass->VisitFieldsReferences<true, kVerifyFlags, kReadBarrierOption>(0, visitor);
Mathieu Chartier407f7022014-02-18 14:37:05 -0800920}
921
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800922template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere4275c02015-08-06 15:34:15 -0700923inline bool Object::IsClassLoader() {
Vladimir Marko98db89c2018-09-07 11:45:46 +0100924 return GetClass<kVerifyFlags, kReadBarrierOption>()->template IsClassLoaderClass<kVerifyFlags>();
Mathieu Chartiere4275c02015-08-06 15:34:15 -0700925}
926
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800927template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Vladimir Marko4617d582019-03-28 13:48:31 +0000928inline ObjPtr<ClassLoader> Object::AsClassLoader() {
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800929 DCHECK((IsClassLoader<kVerifyFlags, kReadBarrierOption>()));
Vladimir Marko4617d582019-03-28 13:48:31 +0000930 return ObjPtr<ClassLoader>::DownCast(this);
Mathieu Chartiere4275c02015-08-06 15:34:15 -0700931}
932
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800933template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Vladimir Marko05792b92015-08-03 11:56:49 +0100934inline bool Object::IsDexCache() {
Vladimir Marko98db89c2018-09-07 11:45:46 +0100935 return GetClass<kVerifyFlags, kReadBarrierOption>()->template IsDexCacheClass<kVerifyFlags>();
Vladimir Marko05792b92015-08-03 11:56:49 +0100936}
937
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800938template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Vladimir Marko4617d582019-03-28 13:48:31 +0000939inline ObjPtr<mirror::DexCache> Object::AsDexCache() {
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800940 DCHECK((IsDexCache<kVerifyFlags, kReadBarrierOption>()));
Vladimir Marko4617d582019-03-28 13:48:31 +0000941 return ObjPtr<DexCache>::DownCast(this);
Vladimir Marko05792b92015-08-03 11:56:49 +0100942}
943
Mathieu Chartier99111282018-06-19 12:30:56 -0700944template<bool kTransactionActive, bool kCheckTransaction>
945inline void Object::VerifyTransaction() {
946 if (kCheckTransaction) {
947 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
948 }
949}
950
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800951} // namespace mirror
952} // namespace art
953
Brian Carlstromfc0e3212013-07-17 14:40:12 -0700954#endif // ART_RUNTIME_MIRROR_OBJECT_INL_H_