blob: 6d29ed379d1989505affa0a5dedd437e5b6458ca [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
Brian Carlstromea46f952013-07-30 01:26:50 -070022#include "art_field.h"
23#include "art_method.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080024#include "atomic.h"
Ian Rogers4f6ad8a2013-03-18 15:27:28 -070025#include "array-inl.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080026#include "class.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"
Mathieu Chartiere4275c02015-08-06 15:34:15 -070029#include "class_loader-inl.h"
Vladimir Marko05792b92015-08-03 11:56:49 +010030#include "dex_cache-inl.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"
Mathieu Chartier52e4b432014-06-10 11:22:31 -070033#include "object_array-inl.h"
Mathieu Chartiera058fdf2016-10-06 15:13:58 -070034#include "object_reference-inl.h"
Mathieu Chartiera59d9b22016-09-26 18:13:17 -070035#include "obj_ptr-inl.h"
Hiroshi Yamauchi800ac2d2014-04-02 17:32:54 -070036#include "read_barrier-inl.h"
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -070037#include "reference.h"
Jeff Hao848f70a2014-01-15 13:49:50 -080038#include "runtime.h"
39#include "string-inl.h"
Ian Rogers05f30572013-02-20 12:13:11 -080040#include "throwable.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080041
42namespace art {
43namespace mirror {
44
Andreas Gampe542451c2016-07-26 09:02:02 -070045inline uint32_t Object::ClassSize(PointerSize pointer_size) {
Mingyao Yang98d1cc82014-05-15 17:02:16 -070046 uint32_t vtable_entries = kVTableLength;
Mathieu Chartiere401d142015-04-22 13:56:20 -070047 return Class::ComputeClassSize(true, vtable_entries, 0, 0, 0, 0, 0, pointer_size);
Mingyao Yang98d1cc82014-05-15 17:02:16 -070048}
49
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -070050template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -080051inline Class* Object::GetClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -070052 return GetFieldObject<Class, kVerifyFlags, kReadBarrierOption>(
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070053 OFFSET_OF_OBJECT_MEMBER(Object, klass_));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080054}
55
Mathieu Chartier4e305412014-02-19 10:54:44 -080056template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiera058fdf2016-10-06 15:13:58 -070057inline void Object::SetClass(ObjPtr<Class> new_klass) {
Mathieu Chartier2cebb242015-04-21 16:50:40 -070058 // new_klass may be null prior to class linker initialization.
Ian Rogersef7d42f2014-01-06 12:55:46 -080059 // We don't mark the card as this occurs as part of object allocation. Not all objects have
60 // backing cards, such as large objects.
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +010061 // We use non transactional version since we can't undo this write. We also disable checking as
62 // we may run in transaction mode here.
Mathieu Chartier4e305412014-02-19 10:54:44 -080063 SetFieldObjectWithoutWriteBarrier<false, false,
64 static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis)>(
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070065 OFFSET_OF_OBJECT_MEMBER(Object, klass_), 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 LockWord Object::GetLockWord(bool as_volatile) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070070 if (as_volatile) {
Andreas Gampe3b45ef22015-05-26 21:34:09 -070071 return LockWord(GetField32Volatile<kVerifyFlags>(OFFSET_OF_OBJECT_MEMBER(Object, monitor_)));
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070072 }
Andreas Gampe3b45ef22015-05-26 21:34:09 -070073 return LockWord(GetField32<kVerifyFlags>(OFFSET_OF_OBJECT_MEMBER(Object, monitor_)));
Ian Rogersd9c4fc92013-10-01 19:45:43 -070074}
75
Andreas Gampe3b45ef22015-05-26 21:34:09 -070076template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartierbbd695c2014-04-16 09:48:48 -070077inline void Object::SetLockWord(LockWord new_val, bool as_volatile) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +010078 // Force use of non-transactional mode and do not check.
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070079 if (as_volatile) {
Andreas Gampe3b45ef22015-05-26 21:34:09 -070080 SetField32Volatile<false, false, kVerifyFlags>(
81 OFFSET_OF_OBJECT_MEMBER(Object, monitor_), new_val.GetValue());
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070082 } else {
Andreas Gampe3b45ef22015-05-26 21:34:09 -070083 SetField32<false, false, kVerifyFlags>(
84 OFFSET_OF_OBJECT_MEMBER(Object, monitor_), new_val.GetValue());
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070085 }
Ian Rogersd9c4fc92013-10-01 19:45:43 -070086}
87
Ian Rogers228602f2014-07-10 02:07:54 -070088inline bool Object::CasLockWordWeakSequentiallyConsistent(LockWord old_val, LockWord new_val) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +010089 // Force use of non-transactional mode and do not check.
Ian Rogers228602f2014-07-10 02:07:54 -070090 return CasFieldWeakSequentiallyConsistent32<false, false>(
91 OFFSET_OF_OBJECT_MEMBER(Object, monitor_), old_val.GetValue(), new_val.GetValue());
Ian Rogersd9c4fc92013-10-01 19:45:43 -070092}
93
Hans Boehmd8434432014-07-11 09:56:07 -070094inline bool Object::CasLockWordWeakRelaxed(LockWord old_val, LockWord new_val) {
95 // Force use of non-transactional mode and do not check.
96 return CasFieldWeakRelaxed32<false, false>(
97 OFFSET_OF_OBJECT_MEMBER(Object, monitor_), old_val.GetValue(), new_val.GetValue());
98}
99
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -0700100inline bool Object::CasLockWordWeakRelease(LockWord old_val, LockWord new_val) {
101 // Force use of non-transactional mode and do not check.
102 return CasFieldWeakRelease32<false, false>(
103 OFFSET_OF_OBJECT_MEMBER(Object, monitor_), old_val.GetValue(), new_val.GetValue());
104}
105
Ian Rogersd9c4fc92013-10-01 19:45:43 -0700106inline uint32_t Object::GetLockOwnerThreadId() {
107 return Monitor::GetLockOwnerThreadId(this);
Ian Rogers05f30572013-02-20 12:13:11 -0800108}
109
Mathieu Chartiere7e8a5f2014-02-14 16:59:41 -0800110inline mirror::Object* Object::MonitorEnter(Thread* self) {
Mathieu Chartiera704eda2016-07-13 09:53:35 -0700111 return Monitor::MonitorEnter(self, this, /*trylock*/false);
112}
113
114inline mirror::Object* Object::MonitorTryEnter(Thread* self) {
115 return Monitor::MonitorEnter(self, this, /*trylock*/true);
Ian Rogers05f30572013-02-20 12:13:11 -0800116}
117
118inline bool Object::MonitorExit(Thread* self) {
119 return Monitor::MonitorExit(self, this);
120}
121
122inline void Object::Notify(Thread* self) {
123 Monitor::Notify(self, this);
124}
125
126inline void Object::NotifyAll(Thread* self) {
127 Monitor::NotifyAll(self, this);
128}
129
130inline void Object::Wait(Thread* self) {
131 Monitor::Wait(self, this, 0, 0, true, kWaiting);
132}
133
134inline void Object::Wait(Thread* self, int64_t ms, int32_t ns) {
135 Monitor::Wait(self, this, ms, ns, true, kTimedWaiting);
136}
137
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700138inline uint32_t Object::GetReadBarrierState(uintptr_t* fake_address_dependency) {
139#ifdef USE_BAKER_READ_BARRIER
140 CHECK(kUseBakerReadBarrier);
141#if defined(__arm__)
142 uintptr_t obj = reinterpret_cast<uintptr_t>(this);
143 uintptr_t result;
144 DCHECK_EQ(OFFSETOF_MEMBER(Object, monitor_), 4U);
145 // Use inline assembly to prevent the compiler from optimizing away the false dependency.
146 __asm__ __volatile__(
147 "ldr %[result], [%[obj], #4]\n\t"
148 // This instruction is enough to "fool the compiler and the CPU" by having `fad` always be
149 // null, without them being able to assume that fact.
150 "eor %[fad], %[result], %[result]\n\t"
151 : [result] "+r" (result), [fad] "=r" (*fake_address_dependency)
152 : [obj] "r" (obj));
153 DCHECK_EQ(*fake_address_dependency, 0U);
154 LockWord lw(static_cast<uint32_t>(result));
155 uint32_t rb_state = lw.ReadBarrierState();
156 return rb_state;
157#elif defined(__aarch64__)
158 uintptr_t obj = reinterpret_cast<uintptr_t>(this);
159 uintptr_t result;
160 DCHECK_EQ(OFFSETOF_MEMBER(Object, monitor_), 4U);
161 // Use inline assembly to prevent the compiler from optimizing away the false dependency.
162 __asm__ __volatile__(
163 "ldr %w[result], [%[obj], #4]\n\t"
164 // This instruction is enough to "fool the compiler and the CPU" by having `fad` always be
165 // null, without them being able to assume that fact.
166 "eor %[fad], %[result], %[result]\n\t"
167 : [result] "+r" (result), [fad] "=r" (*fake_address_dependency)
168 : [obj] "r" (obj));
169 DCHECK_EQ(*fake_address_dependency, 0U);
170 LockWord lw(static_cast<uint32_t>(result));
171 uint32_t rb_state = lw.ReadBarrierState();
172 return rb_state;
173#elif defined(__i386__) || defined(__x86_64__)
174 LockWord lw = GetLockWord(false);
175 // i386/x86_64 don't need fake address dependency. Use a compiler fence to avoid compiler
176 // reordering.
177 *fake_address_dependency = 0;
178 std::atomic_signal_fence(std::memory_order_acquire);
179 uint32_t rb_state = lw.ReadBarrierState();
180 return rb_state;
181#else
182 // mips/mips64
183 LOG(FATAL) << "Unreachable";
184 UNREACHABLE();
185 UNUSED(fake_address_dependency);
186#endif
187#else // !USE_BAKER_READ_BARRIER
188 LOG(FATAL) << "Unreachable";
189 UNREACHABLE();
190 UNUSED(fake_address_dependency);
191#endif
192}
193
194inline uint32_t Object::GetReadBarrierState() {
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700195#ifdef USE_BAKER_READ_BARRIER
196 DCHECK(kUseBakerReadBarrier);
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700197 LockWord lw(GetField<uint32_t, /*kIsVolatile*/false>(OFFSET_OF_OBJECT_MEMBER(Object, monitor_)));
198 uint32_t rb_state = lw.ReadBarrierState();
199 DCHECK(ReadBarrier::IsValidReadBarrierState(rb_state)) << rb_state;
200 return rb_state;
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800201#else
202 LOG(FATAL) << "Unreachable";
Ian Rogers2c4257b2014-10-24 14:20:06 -0700203 UNREACHABLE();
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800204#endif
205}
206
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700207inline uint32_t Object::GetReadBarrierStateAcquire() {
Mathieu Chartierc381c362016-08-23 13:27:53 -0700208#ifdef USE_BAKER_READ_BARRIER
209 DCHECK(kUseBakerReadBarrier);
210 LockWord lw(GetFieldAcquire<uint32_t>(OFFSET_OF_OBJECT_MEMBER(Object, monitor_)));
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700211 uint32_t rb_state = lw.ReadBarrierState();
212 DCHECK(ReadBarrier::IsValidReadBarrierState(rb_state)) << rb_state;
213 return rb_state;
Mathieu Chartierc381c362016-08-23 13:27:53 -0700214#else
215 LOG(FATAL) << "Unreachable";
216 UNREACHABLE();
217#endif
218}
219
Mathieu Chartier36a270a2016-07-28 18:08:51 -0700220inline uint32_t Object::GetMarkBit() {
221#ifdef USE_READ_BARRIER
Mathieu Chartier36a270a2016-07-28 18:08:51 -0700222 return GetLockWord(false).MarkBitState();
223#else
224 LOG(FATAL) << "Unreachable";
225 UNREACHABLE();
226#endif
227}
228
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700229inline void Object::SetReadBarrierState(uint32_t rb_state) {
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700230#ifdef USE_BAKER_READ_BARRIER
231 DCHECK(kUseBakerReadBarrier);
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700232 DCHECK(ReadBarrier::IsValidReadBarrierState(rb_state)) << rb_state;
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700233 LockWord lw = GetLockWord(false);
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700234 lw.SetReadBarrierState(rb_state);
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700235 SetLockWord(lw, false);
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800236#else
237 LOG(FATAL) << "Unreachable";
Ian Rogers2c4257b2014-10-24 14:20:06 -0700238 UNREACHABLE();
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700239 UNUSED(rb_state);
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800240#endif
241}
242
Hiroshi Yamauchied70b4a2015-11-17 17:52:15 -0800243template<bool kCasRelease>
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700244inline bool Object::AtomicSetReadBarrierState(uint32_t expected_rb_state, uint32_t rb_state) {
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700245#ifdef USE_BAKER_READ_BARRIER
246 DCHECK(kUseBakerReadBarrier);
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700247 DCHECK(ReadBarrier::IsValidReadBarrierState(expected_rb_state)) << expected_rb_state;
248 DCHECK(ReadBarrier::IsValidReadBarrierState(rb_state)) << rb_state;
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700249 LockWord expected_lw;
250 LockWord new_lw;
251 do {
252 LockWord lw = GetLockWord(false);
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700253 if (UNLIKELY(lw.ReadBarrierState() != expected_rb_state)) {
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700254 // Lost the race.
255 return false;
256 }
257 expected_lw = lw;
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700258 expected_lw.SetReadBarrierState(expected_rb_state);
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700259 new_lw = lw;
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700260 new_lw.SetReadBarrierState(rb_state);
Hiroshi Yamauchied70b4a2015-11-17 17:52:15 -0800261 // ConcurrentCopying::ProcessMarkStackRef uses this with kCasRelease == true.
262 // If kCasRelease == true, use a CAS release so that when GC updates all the fields of
263 // an object and then changes the object from gray to black, the field updates (stores) will be
264 // visible (won't be reordered after this CAS.)
265 } while (!(kCasRelease ?
266 CasLockWordWeakRelease(expected_lw, new_lw) :
267 CasLockWordWeakRelaxed(expected_lw, new_lw)));
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700268 return true;
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700269#else
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700270 UNUSED(expected_rb_state, rb_state);
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700271 LOG(FATAL) << "Unreachable";
Ian Rogers2c4257b2014-10-24 14:20:06 -0700272 UNREACHABLE();
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700273#endif
274}
275
Mathieu Chartier36a270a2016-07-28 18:08:51 -0700276inline bool Object::AtomicSetMarkBit(uint32_t expected_mark_bit, uint32_t mark_bit) {
277 LockWord expected_lw;
278 LockWord new_lw;
279 do {
280 LockWord lw = GetLockWord(false);
281 if (UNLIKELY(lw.MarkBitState() != expected_mark_bit)) {
282 // Lost the race.
283 return false;
284 }
285 expected_lw = lw;
286 new_lw = lw;
287 new_lw.SetMarkBitState(mark_bit);
288 // Since this is only set from the mutator, we can use the non release Cas.
289 } while (!CasLockWordWeakRelaxed(expected_lw, new_lw));
290 return true;
291}
292
293
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700294inline void Object::AssertReadBarrierState() const {
295 CHECK(kUseBakerReadBarrier);
296 Object* obj = const_cast<Object*>(this);
297 DCHECK(obj->GetReadBarrierState() == ReadBarrier::WhiteState())
298 << "Bad Baker pointer: obj=" << reinterpret_cast<void*>(obj)
299 << " rb_state" << reinterpret_cast<void*>(obj->GetReadBarrierState());
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800300}
301
Mathieu Chartier4e305412014-02-19 10:54:44 -0800302template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartieref41db72016-10-25 15:08:01 -0700303inline bool Object::VerifierInstanceOf(ObjPtr<Class> klass) {
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700304 DCHECK(klass != nullptr);
305 DCHECK(GetClass<kVerifyFlags>() != nullptr);
Jeff Haoa3faaf42013-09-03 19:07:00 -0700306 return klass->IsInterface() || InstanceOf(klass);
307}
308
Mathieu Chartier4e305412014-02-19 10:54:44 -0800309template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiera59d9b22016-09-26 18:13:17 -0700310inline bool Object::InstanceOf(ObjPtr<Class> klass) {
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700311 DCHECK(klass != nullptr);
312 DCHECK(GetClass<kVerifyNone>() != nullptr);
Mathieu Chartier4e305412014-02-19 10:54:44 -0800313 return klass->IsAssignableFrom(GetClass<kVerifyFlags>());
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800314}
315
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700316template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800317inline bool Object::IsClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700318 Class* java_lang_Class = GetClass<kVerifyFlags, kReadBarrierOption>()->
319 template GetClass<kVerifyFlags, kReadBarrierOption>();
320 return GetClass<static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis),
321 kReadBarrierOption>() == java_lang_Class;
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800322}
323
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700324template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800325inline Class* Object::AsClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700326 DCHECK((IsClass<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800327 return down_cast<Class*>(this);
328}
329
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800330template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800331inline bool Object::IsObjectArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800332 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800333 return IsArrayInstance<kVerifyFlags, kReadBarrierOption>() &&
334 !GetClass<kNewFlags, kReadBarrierOption>()->
335 template GetComponentType<kNewFlags, kReadBarrierOption>()->IsPrimitive();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800336}
337
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800338template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800339inline ObjectArray<T>* Object::AsObjectArray() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800340 DCHECK((IsObjectArray<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800341 return down_cast<ObjectArray<T>*>(this);
342}
343
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700344template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800345inline bool Object::IsArrayInstance() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700346 return GetClass<kVerifyFlags, kReadBarrierOption>()->
347 template IsArrayClass<kVerifyFlags, kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800348}
349
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800350template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800351inline bool Object::IsReferenceInstance() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800352 return GetClass<kVerifyFlags, kReadBarrierOption>()->IsTypeOfReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800353}
354
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800355template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700356inline Reference* Object::AsReference() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800357 DCHECK((IsReferenceInstance<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700358 return down_cast<Reference*>(this);
359}
360
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700361template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers05f30572013-02-20 12:13:11 -0800362inline Array* Object::AsArray() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700363 DCHECK((IsArrayInstance<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800364 return down_cast<Array*>(this);
365}
366
Mathieu Chartier4e305412014-02-19 10:54:44 -0800367template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800368inline BooleanArray* Object::AsBooleanArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800369 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
370 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
371 DCHECK(GetClass<kNewFlags>()->GetComponentType()->IsPrimitiveBoolean());
Ian Rogers05f30572013-02-20 12:13:11 -0800372 return down_cast<BooleanArray*>(this);
373}
374
Mathieu Chartier4e305412014-02-19 10:54:44 -0800375template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800376inline ByteArray* Object::AsByteArray() {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700377 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Mathieu Chartier4e305412014-02-19 10:54:44 -0800378 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
379 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveByte());
Ian Rogers05f30572013-02-20 12:13:11 -0800380 return down_cast<ByteArray*>(this);
381}
382
Mathieu Chartier4e305412014-02-19 10:54:44 -0800383template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800384inline ByteArray* Object::AsByteSizedArray() {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700385 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Mathieu Chartier4e305412014-02-19 10:54:44 -0800386 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
387 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveByte() ||
388 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveBoolean());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800389 return down_cast<ByteArray*>(this);
390}
391
Mathieu Chartier4e305412014-02-19 10:54:44 -0800392template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800393inline CharArray* Object::AsCharArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800394 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
395 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
396 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveChar());
Ian Rogers05f30572013-02-20 12:13:11 -0800397 return down_cast<CharArray*>(this);
398}
399
Mathieu Chartier4e305412014-02-19 10:54:44 -0800400template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800401inline ShortArray* Object::AsShortArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800402 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
403 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
404 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveShort());
Ian Rogers05f30572013-02-20 12:13:11 -0800405 return down_cast<ShortArray*>(this);
406}
407
Mathieu Chartier4e305412014-02-19 10:54:44 -0800408template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800409inline ShortArray* Object::AsShortSizedArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800410 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
411 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
412 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveShort() ||
413 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveChar());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800414 return down_cast<ShortArray*>(this);
415}
416
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800417template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700418inline bool Object::IsIntArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800419 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Mathieu Chartier31e88222016-10-14 18:43:19 -0700420 ObjPtr<Class> klass = GetClass<kVerifyFlags, kReadBarrierOption>();
421 ObjPtr<Class> component_type = klass->GetComponentType<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartiere401d142015-04-22 13:56:20 -0700422 return component_type != nullptr && component_type->template IsPrimitiveInt<kNewFlags>();
423}
424
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800425template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700426inline IntArray* Object::AsIntArray() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800427 DCHECK((IsIntArray<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800428 return down_cast<IntArray*>(this);
429}
430
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800431template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700432inline bool Object::IsLongArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800433 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Mathieu Chartier31e88222016-10-14 18:43:19 -0700434 ObjPtr<Class> klass = GetClass<kVerifyFlags, kReadBarrierOption>();
435 ObjPtr<Class> component_type = klass->GetComponentType<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartiere401d142015-04-22 13:56:20 -0700436 return component_type != nullptr && component_type->template IsPrimitiveLong<kNewFlags>();
437}
438
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800439template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700440inline LongArray* Object::AsLongArray() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800441 DCHECK((IsLongArray<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800442 return down_cast<LongArray*>(this);
443}
444
Mathieu Chartier4e305412014-02-19 10:54:44 -0800445template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700446inline bool Object::IsFloatArray() {
447 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
448 auto* component_type = GetClass<kVerifyFlags>()->GetComponentType();
449 return component_type != nullptr && component_type->template IsPrimitiveFloat<kNewFlags>();
450}
451
452template<VerifyObjectFlags kVerifyFlags>
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100453inline FloatArray* Object::AsFloatArray() {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700454 DCHECK(IsFloatArray<kVerifyFlags>());
Mathieu Chartier4e305412014-02-19 10:54:44 -0800455 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
456 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
457 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveFloat());
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100458 return down_cast<FloatArray*>(this);
459}
460
Mathieu Chartier4e305412014-02-19 10:54:44 -0800461template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700462inline bool Object::IsDoubleArray() {
463 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
464 auto* component_type = GetClass<kVerifyFlags>()->GetComponentType();
465 return component_type != nullptr && component_type->template IsPrimitiveDouble<kNewFlags>();
466}
467
468template<VerifyObjectFlags kVerifyFlags>
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100469inline DoubleArray* Object::AsDoubleArray() {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700470 DCHECK(IsDoubleArray<kVerifyFlags>());
Mathieu Chartier4e305412014-02-19 10:54:44 -0800471 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
472 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
473 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveDouble());
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100474 return down_cast<DoubleArray*>(this);
475}
476
Jeff Hao848f70a2014-01-15 13:49:50 -0800477template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
478inline bool Object::IsString() {
479 return GetClass<kVerifyFlags, kReadBarrierOption>()->IsStringClass();
480}
481
482template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers05f30572013-02-20 12:13:11 -0800483inline String* Object::AsString() {
Jeff Hao848f70a2014-01-15 13:49:50 -0800484 DCHECK((IsString<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800485 return down_cast<String*>(this);
486}
487
Mathieu Chartier4e305412014-02-19 10:54:44 -0800488template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800489inline Throwable* Object::AsThrowable() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800490 DCHECK(GetClass<kVerifyFlags>()->IsThrowableClass());
Ian Rogers05f30572013-02-20 12:13:11 -0800491 return down_cast<Throwable*>(this);
492}
493
Mathieu Chartier4e305412014-02-19 10:54:44 -0800494template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800495inline bool Object::IsWeakReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800496 return GetClass<kVerifyFlags>()->IsWeakReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800497}
498
Mathieu Chartier4e305412014-02-19 10:54:44 -0800499template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800500inline bool Object::IsSoftReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800501 return GetClass<kVerifyFlags>()->IsSoftReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800502}
503
Mathieu Chartier4e305412014-02-19 10:54:44 -0800504template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800505inline bool Object::IsFinalizerReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800506 return GetClass<kVerifyFlags>()->IsFinalizerReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800507}
508
Mathieu Chartier4e305412014-02-19 10:54:44 -0800509template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700510inline FinalizerReference* Object::AsFinalizerReference() {
511 DCHECK(IsFinalizerReferenceInstance<kVerifyFlags>());
512 return down_cast<FinalizerReference*>(this);
513}
514
515template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800516inline bool Object::IsPhantomReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800517 return GetClass<kVerifyFlags>()->IsPhantomReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800518}
519
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700520template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800521inline size_t Object::SizeOf() {
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800522 size_t result;
Mathieu Chartier4e305412014-02-19 10:54:44 -0800523 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700524 if (IsArrayInstance<kVerifyFlags, kReadBarrierOption>()) {
525 result = AsArray<kNewFlags, kReadBarrierOption>()->
526 template SizeOf<kNewFlags, kReadBarrierOption>();
527 } else if (IsClass<kNewFlags, kReadBarrierOption>()) {
528 result = AsClass<kNewFlags, kReadBarrierOption>()->
529 template SizeOf<kNewFlags, kReadBarrierOption>();
Jeff Hao848f70a2014-01-15 13:49:50 -0800530 } else if (GetClass<kNewFlags, kReadBarrierOption>()->IsStringClass()) {
531 result = AsString<kNewFlags, kReadBarrierOption>()->
532 template SizeOf<kNewFlags>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800533 } else {
Hiroshi Yamauchi25023c72014-05-09 11:45:53 -0700534 result = GetClass<kNewFlags, kReadBarrierOption>()->
535 template GetObjectSize<kNewFlags, kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800536 }
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700537 DCHECK_GE(result, sizeof(Object))
David Sehr709b0702016-10-13 09:12:37 -0700538 << " class=" << Class::PrettyClass(GetClass<kNewFlags, kReadBarrierOption>());
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800539 return result;
540}
541
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700542template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
Fred Shih37f05ef2014-07-16 18:38:08 -0700543inline uint8_t Object::GetFieldBoolean(MemberOffset field_offset) {
544 if (kVerifyFlags & kVerifyThis) {
545 VerifyObject(this);
546 }
547 return GetField<uint8_t, kIsVolatile>(field_offset);
548}
549
550template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
551inline int8_t Object::GetFieldByte(MemberOffset field_offset) {
552 if (kVerifyFlags & kVerifyThis) {
553 VerifyObject(this);
554 }
555 return GetField<int8_t, kIsVolatile>(field_offset);
556}
557
558template<VerifyObjectFlags kVerifyFlags>
559inline uint8_t Object::GetFieldBooleanVolatile(MemberOffset field_offset) {
560 return GetFieldBoolean<kVerifyFlags, true>(field_offset);
561}
562
563template<VerifyObjectFlags kVerifyFlags>
564inline int8_t Object::GetFieldByteVolatile(MemberOffset field_offset) {
565 return GetFieldByte<kVerifyFlags, true>(field_offset);
566}
567
568template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
569 bool kIsVolatile>
570inline void Object::SetFieldBoolean(MemberOffset field_offset, uint8_t new_value)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700571 REQUIRES_SHARED(Locks::mutator_lock_) {
Fred Shih37f05ef2014-07-16 18:38:08 -0700572 if (kCheckTransaction) {
573 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
574 }
575 if (kTransactionActive) {
576 Runtime::Current()->RecordWriteFieldBoolean(this, field_offset,
577 GetFieldBoolean<kVerifyFlags, kIsVolatile>(field_offset),
578 kIsVolatile);
579 }
580 if (kVerifyFlags & kVerifyThis) {
581 VerifyObject(this);
582 }
583 SetField<uint8_t, kIsVolatile>(field_offset, new_value);
584}
585
586template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
587 bool kIsVolatile>
588inline void Object::SetFieldByte(MemberOffset field_offset, int8_t new_value)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700589 REQUIRES_SHARED(Locks::mutator_lock_) {
Fred Shih37f05ef2014-07-16 18:38:08 -0700590 if (kCheckTransaction) {
591 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
592 }
593 if (kTransactionActive) {
594 Runtime::Current()->RecordWriteFieldByte(this, field_offset,
595 GetFieldByte<kVerifyFlags, kIsVolatile>(field_offset),
596 kIsVolatile);
597 }
598 if (kVerifyFlags & kVerifyThis) {
599 VerifyObject(this);
600 }
601 SetField<int8_t, kIsVolatile>(field_offset, new_value);
602}
603
604template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
605inline void Object::SetFieldBooleanVolatile(MemberOffset field_offset, uint8_t new_value) {
606 return SetFieldBoolean<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
607 field_offset, new_value);
608}
609
610template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
611inline void Object::SetFieldByteVolatile(MemberOffset field_offset, int8_t new_value) {
612 return SetFieldByte<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
613 field_offset, new_value);
614}
615
616template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
617inline uint16_t Object::GetFieldChar(MemberOffset field_offset) {
618 if (kVerifyFlags & kVerifyThis) {
619 VerifyObject(this);
620 }
621 return GetField<uint16_t, kIsVolatile>(field_offset);
622}
623
624template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
625inline int16_t Object::GetFieldShort(MemberOffset field_offset) {
626 if (kVerifyFlags & kVerifyThis) {
627 VerifyObject(this);
628 }
629 return GetField<int16_t, kIsVolatile>(field_offset);
630}
631
632template<VerifyObjectFlags kVerifyFlags>
633inline uint16_t Object::GetFieldCharVolatile(MemberOffset field_offset) {
634 return GetFieldChar<kVerifyFlags, true>(field_offset);
635}
636
637template<VerifyObjectFlags kVerifyFlags>
638inline int16_t Object::GetFieldShortVolatile(MemberOffset field_offset) {
639 return GetFieldShort<kVerifyFlags, true>(field_offset);
640}
641
642template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
643 bool kIsVolatile>
644inline void Object::SetFieldChar(MemberOffset field_offset, uint16_t new_value) {
645 if (kCheckTransaction) {
646 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
647 }
648 if (kTransactionActive) {
649 Runtime::Current()->RecordWriteFieldChar(this, field_offset,
650 GetFieldChar<kVerifyFlags, kIsVolatile>(field_offset),
651 kIsVolatile);
652 }
653 if (kVerifyFlags & kVerifyThis) {
654 VerifyObject(this);
655 }
656 SetField<uint16_t, kIsVolatile>(field_offset, new_value);
657}
658
659template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
660 bool kIsVolatile>
661inline void Object::SetFieldShort(MemberOffset field_offset, int16_t new_value) {
662 if (kCheckTransaction) {
663 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
664 }
665 if (kTransactionActive) {
666 Runtime::Current()->RecordWriteFieldChar(this, field_offset,
667 GetFieldShort<kVerifyFlags, kIsVolatile>(field_offset),
668 kIsVolatile);
669 }
670 if (kVerifyFlags & kVerifyThis) {
671 VerifyObject(this);
672 }
673 SetField<int16_t, kIsVolatile>(field_offset, new_value);
674}
675
676template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
677inline void Object::SetFieldCharVolatile(MemberOffset field_offset, uint16_t new_value) {
678 return SetFieldChar<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
679 field_offset, new_value);
680}
681
682template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
683inline void Object::SetFieldShortVolatile(MemberOffset field_offset, int16_t new_value) {
684 return SetFieldShort<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
685 field_offset, new_value);
686}
687
688template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700689inline int32_t Object::GetField32(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800690 if (kVerifyFlags & kVerifyThis) {
691 VerifyObject(this);
692 }
Fred Shih37f05ef2014-07-16 18:38:08 -0700693 return GetField<int32_t, kIsVolatile>(field_offset);
Ian Rogersb122a4b2013-11-19 18:00:50 -0800694}
695
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700696template<VerifyObjectFlags kVerifyFlags>
697inline int32_t Object::GetField32Volatile(MemberOffset field_offset) {
698 return GetField32<kVerifyFlags, true>(field_offset);
699}
700
701template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
702 bool kIsVolatile>
703inline void Object::SetField32(MemberOffset field_offset, int32_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100704 if (kCheckTransaction) {
705 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
706 }
707 if (kTransactionActive) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700708 Runtime::Current()->RecordWriteField32(this, field_offset,
709 GetField32<kVerifyFlags, kIsVolatile>(field_offset),
710 kIsVolatile);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100711 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800712 if (kVerifyFlags & kVerifyThis) {
Ian Rogersb122a4b2013-11-19 18:00:50 -0800713 VerifyObject(this);
714 }
Fred Shih37f05ef2014-07-16 18:38:08 -0700715 SetField<int32_t, kIsVolatile>(field_offset, new_value);
Ian Rogersb122a4b2013-11-19 18:00:50 -0800716}
717
Mathieu Chartier4e305412014-02-19 10:54:44 -0800718template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700719inline void Object::SetField32Volatile(MemberOffset field_offset, int32_t new_value) {
720 SetField32<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset, new_value);
721}
722
Hans Boehmd8434432014-07-11 09:56:07 -0700723// TODO: Pass memory_order_ and strong/weak as arguments to avoid code duplication?
724
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700725template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -0700726inline bool Object::CasFieldWeakSequentiallyConsistent32(MemberOffset field_offset,
727 int32_t old_value, int32_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100728 if (kCheckTransaction) {
729 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
730 }
731 if (kTransactionActive) {
732 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
733 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800734 if (kVerifyFlags & kVerifyThis) {
735 VerifyObject(this);
736 }
Ian Rogers13735952014-10-08 12:43:28 -0700737 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogers228602f2014-07-10 02:07:54 -0700738 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
Hans Boehm30359612014-05-21 17:46:23 -0700739
Ian Rogers228602f2014-07-10 02:07:54 -0700740 return atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_value, new_value);
Ian Rogersd9c4fc92013-10-01 19:45:43 -0700741}
742
Hans Boehmd8434432014-07-11 09:56:07 -0700743template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
744inline bool Object::CasFieldWeakRelaxed32(MemberOffset field_offset,
745 int32_t old_value, int32_t new_value) {
746 if (kCheckTransaction) {
747 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
748 }
749 if (kTransactionActive) {
750 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
751 }
752 if (kVerifyFlags & kVerifyThis) {
753 VerifyObject(this);
754 }
Ian Rogers13735952014-10-08 12:43:28 -0700755 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Hans Boehmd8434432014-07-11 09:56:07 -0700756 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
757
758 return atomic_addr->CompareExchangeWeakRelaxed(old_value, new_value);
759}
760
761template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -0700762inline bool Object::CasFieldWeakRelease32(MemberOffset field_offset,
763 int32_t old_value, int32_t new_value) {
764 if (kCheckTransaction) {
765 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
766 }
767 if (kTransactionActive) {
768 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
769 }
770 if (kVerifyFlags & kVerifyThis) {
771 VerifyObject(this);
772 }
773 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
774 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
775
776 return atomic_addr->CompareExchangeWeakRelease(old_value, new_value);
777}
778
779template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Hans Boehmd8434432014-07-11 09:56:07 -0700780inline bool Object::CasFieldStrongSequentiallyConsistent32(MemberOffset field_offset,
781 int32_t old_value, int32_t new_value) {
782 if (kCheckTransaction) {
783 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
784 }
785 if (kTransactionActive) {
786 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
787 }
788 if (kVerifyFlags & kVerifyThis) {
789 VerifyObject(this);
790 }
Ian Rogers13735952014-10-08 12:43:28 -0700791 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Hans Boehmd8434432014-07-11 09:56:07 -0700792 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
793
794 return atomic_addr->CompareExchangeStrongSequentiallyConsistent(old_value, new_value);
795}
796
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700797template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
798inline int64_t Object::GetField64(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800799 if (kVerifyFlags & kVerifyThis) {
800 VerifyObject(this);
801 }
Fred Shih37f05ef2014-07-16 18:38:08 -0700802 return GetField<int64_t, kIsVolatile>(field_offset);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800803}
804
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700805template<VerifyObjectFlags kVerifyFlags>
806inline int64_t Object::GetField64Volatile(MemberOffset field_offset) {
807 return GetField64<kVerifyFlags, true>(field_offset);
808}
809
810template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
811 bool kIsVolatile>
812inline void Object::SetField64(MemberOffset field_offset, int64_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100813 if (kCheckTransaction) {
814 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
815 }
816 if (kTransactionActive) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700817 Runtime::Current()->RecordWriteField64(this, field_offset,
818 GetField64<kVerifyFlags, kIsVolatile>(field_offset),
819 kIsVolatile);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100820 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800821 if (kVerifyFlags & kVerifyThis) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800822 VerifyObject(this);
823 }
Fred Shih37f05ef2014-07-16 18:38:08 -0700824 SetField<int64_t, kIsVolatile>(field_offset, new_value);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800825}
826
Mathieu Chartier4e305412014-02-19 10:54:44 -0800827template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700828inline void Object::SetField64Volatile(MemberOffset field_offset, int64_t new_value) {
829 return SetField64<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
830 new_value);
831}
832
Fred Shih37f05ef2014-07-16 18:38:08 -0700833template<typename kSize, bool kIsVolatile>
834inline void Object::SetField(MemberOffset field_offset, kSize new_value) {
Ian Rogers13735952014-10-08 12:43:28 -0700835 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Fred Shih37f05ef2014-07-16 18:38:08 -0700836 kSize* addr = reinterpret_cast<kSize*>(raw_addr);
837 if (kIsVolatile) {
838 reinterpret_cast<Atomic<kSize>*>(addr)->StoreSequentiallyConsistent(new_value);
839 } else {
840 reinterpret_cast<Atomic<kSize>*>(addr)->StoreJavaData(new_value);
841 }
842}
843
844template<typename kSize, bool kIsVolatile>
845inline kSize Object::GetField(MemberOffset field_offset) {
Ian Rogers13735952014-10-08 12:43:28 -0700846 const uint8_t* raw_addr = reinterpret_cast<const uint8_t*>(this) + field_offset.Int32Value();
Fred Shih37f05ef2014-07-16 18:38:08 -0700847 const kSize* addr = reinterpret_cast<const kSize*>(raw_addr);
848 if (kIsVolatile) {
849 return reinterpret_cast<const Atomic<kSize>*>(addr)->LoadSequentiallyConsistent();
850 } else {
851 return reinterpret_cast<const Atomic<kSize>*>(addr)->LoadJavaData();
852 }
853}
854
Mathieu Chartierc381c362016-08-23 13:27:53 -0700855template<typename kSize>
856inline kSize Object::GetFieldAcquire(MemberOffset field_offset) {
857 const uint8_t* raw_addr = reinterpret_cast<const uint8_t*>(this) + field_offset.Int32Value();
858 const kSize* addr = reinterpret_cast<const kSize*>(raw_addr);
859 return reinterpret_cast<const Atomic<kSize>*>(addr)->LoadAcquire();
860}
861
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700862template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -0700863inline bool Object::CasFieldWeakSequentiallyConsistent64(MemberOffset field_offset,
864 int64_t old_value, int64_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100865 if (kCheckTransaction) {
866 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
867 }
868 if (kTransactionActive) {
869 Runtime::Current()->RecordWriteField64(this, field_offset, old_value, true);
870 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800871 if (kVerifyFlags & kVerifyThis) {
872 VerifyObject(this);
873 }
Ian Rogers13735952014-10-08 12:43:28 -0700874 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogers228602f2014-07-10 02:07:54 -0700875 Atomic<int64_t>* atomic_addr = reinterpret_cast<Atomic<int64_t>*>(raw_addr);
876 return atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_value, new_value);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800877}
878
Hans Boehmd8434432014-07-11 09:56:07 -0700879template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
880inline bool Object::CasFieldStrongSequentiallyConsistent64(MemberOffset field_offset,
881 int64_t old_value, int64_t new_value) {
882 if (kCheckTransaction) {
883 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
884 }
885 if (kTransactionActive) {
886 Runtime::Current()->RecordWriteField64(this, field_offset, old_value, true);
887 }
888 if (kVerifyFlags & kVerifyThis) {
889 VerifyObject(this);
890 }
Ian Rogers13735952014-10-08 12:43:28 -0700891 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Hans Boehmd8434432014-07-11 09:56:07 -0700892 Atomic<int64_t>* atomic_addr = reinterpret_cast<Atomic<int64_t>*>(raw_addr);
893 return atomic_addr->CompareExchangeStrongSequentiallyConsistent(old_value, new_value);
894}
895
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700896template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption,
897 bool kIsVolatile>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700898inline T* Object::GetFieldObject(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800899 if (kVerifyFlags & kVerifyThis) {
900 VerifyObject(this);
901 }
Ian Rogers13735952014-10-08 12:43:28 -0700902 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogersef7d42f2014-01-06 12:55:46 -0800903 HeapReference<T>* objref_addr = reinterpret_cast<HeapReference<T>*>(raw_addr);
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700904 T* result = ReadBarrier::Barrier<T, kReadBarrierOption>(this, field_offset, objref_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700905 if (kIsVolatile) {
Hans Boehm30359612014-05-21 17:46:23 -0700906 // TODO: Refactor to use a SequentiallyConsistent load instead.
907 QuasiAtomic::ThreadFenceAcquire(); // Ensure visibility of operations preceding store.
Ian Rogersef7d42f2014-01-06 12:55:46 -0800908 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800909 if (kVerifyFlags & kVerifyReads) {
910 VerifyObject(result);
911 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800912 return result;
913}
914
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700915template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700916inline T* Object::GetFieldObjectVolatile(MemberOffset field_offset) {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700917 return GetFieldObject<T, kVerifyFlags, kReadBarrierOption, true>(field_offset);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700918}
919
920template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
921 bool kIsVolatile>
922inline void Object::SetFieldObjectWithoutWriteBarrier(MemberOffset field_offset,
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700923 ObjPtr<Object> new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100924 if (kCheckTransaction) {
925 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
926 }
927 if (kTransactionActive) {
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700928 ObjPtr<Object> obj;
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700929 if (kIsVolatile) {
930 obj = GetFieldObjectVolatile<Object>(field_offset);
931 } else {
932 obj = GetFieldObject<Object>(field_offset);
933 }
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700934 Runtime::Current()->RecordWriteFieldReference(this, field_offset, obj.Ptr(), true);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100935 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800936 if (kVerifyFlags & kVerifyThis) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800937 VerifyObject(this);
938 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800939 if (kVerifyFlags & kVerifyWrites) {
940 VerifyObject(new_value);
941 }
Ian Rogers13735952014-10-08 12:43:28 -0700942 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogersef7d42f2014-01-06 12:55:46 -0800943 HeapReference<Object>* objref_addr = reinterpret_cast<HeapReference<Object>*>(raw_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700944 if (kIsVolatile) {
Hans Boehm30359612014-05-21 17:46:23 -0700945 // TODO: Refactor to use a SequentiallyConsistent store instead.
946 QuasiAtomic::ThreadFenceRelease(); // Ensure that prior accesses are visible before store.
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700947 objref_addr->Assign(new_value.Ptr());
Hans Boehm30359612014-05-21 17:46:23 -0700948 QuasiAtomic::ThreadFenceSequentiallyConsistent();
949 // Ensure this store occurs before any volatile loads.
Ian Rogersef7d42f2014-01-06 12:55:46 -0800950 } else {
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700951 objref_addr->Assign(new_value.Ptr());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800952 }
953}
954
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700955template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
956 bool kIsVolatile>
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700957inline void Object::SetFieldObject(MemberOffset field_offset, ObjPtr<Object> new_value) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700958 SetFieldObjectWithoutWriteBarrier<kTransactionActive, kCheckTransaction, kVerifyFlags,
959 kIsVolatile>(field_offset, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800960 if (new_value != nullptr) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800961 Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700962 // TODO: Check field assignment could theoretically cause thread suspension, TODO: fix this.
963 CheckFieldAssignment(field_offset, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800964 }
965}
966
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700967template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700968inline void Object::SetFieldObjectVolatile(MemberOffset field_offset, ObjPtr<Object> new_value) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700969 SetFieldObject<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
970 new_value);
971}
972
Mathieu Chartier4e305412014-02-19 10:54:44 -0800973template <VerifyObjectFlags kVerifyFlags>
974inline HeapReference<Object>* Object::GetFieldObjectReferenceAddr(MemberOffset field_offset) {
975 if (kVerifyFlags & kVerifyThis) {
976 VerifyObject(this);
977 }
Ian Rogers13735952014-10-08 12:43:28 -0700978 return reinterpret_cast<HeapReference<Object>*>(reinterpret_cast<uint8_t*>(this) +
Mathieu Chartier4e305412014-02-19 10:54:44 -0800979 field_offset.Int32Value());
980}
981
982template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -0700983inline bool Object::CasFieldWeakSequentiallyConsistentObject(MemberOffset field_offset,
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700984 ObjPtr<Object> old_value,
985 ObjPtr<Object> new_value) {
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800986 bool success = CasFieldWeakSequentiallyConsistentObjectWithoutWriteBarrier<
987 kTransactionActive, kCheckTransaction, kVerifyFlags>(field_offset, old_value, new_value);
988 if (success) {
989 Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
990 }
991 return success;
992}
993
994template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
995inline bool Object::CasFieldWeakSequentiallyConsistentObjectWithoutWriteBarrier(
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700996 MemberOffset field_offset,
997 ObjPtr<Object> old_value,
998 ObjPtr<Object> new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100999 if (kCheckTransaction) {
1000 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
1001 }
Mathieu Chartier4e305412014-02-19 10:54:44 -08001002 if (kVerifyFlags & kVerifyThis) {
1003 VerifyObject(this);
1004 }
1005 if (kVerifyFlags & kVerifyWrites) {
1006 VerifyObject(new_value);
1007 }
1008 if (kVerifyFlags & kVerifyReads) {
1009 VerifyObject(old_value);
1010 }
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +01001011 if (kTransactionActive) {
1012 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
1013 }
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001014 HeapReference<Object> old_ref(HeapReference<Object>::FromObjPtr(old_value));
1015 HeapReference<Object> new_ref(HeapReference<Object>::FromObjPtr(new_value));
Ian Rogers13735952014-10-08 12:43:28 -07001016 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogers228602f2014-07-10 02:07:54 -07001017 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
1018
1019 bool success = atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_ref.reference_,
1020 new_ref.reference_);
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -08001021 return success;
1022}
Ian Rogers228602f2014-07-10 02:07:54 -07001023
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -08001024template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
1025inline bool Object::CasFieldStrongSequentiallyConsistentObject(MemberOffset field_offset,
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001026 ObjPtr<Object> old_value,
1027 ObjPtr<Object> new_value) {
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -08001028 bool success = CasFieldStrongSequentiallyConsistentObjectWithoutWriteBarrier<
1029 kTransactionActive, kCheckTransaction, kVerifyFlags>(field_offset, old_value, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -08001030 if (success) {
1031 Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
1032 }
1033 return success;
1034}
1035
Hans Boehmd8434432014-07-11 09:56:07 -07001036template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -08001037inline bool Object::CasFieldStrongSequentiallyConsistentObjectWithoutWriteBarrier(
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001038 MemberOffset field_offset,
1039 ObjPtr<Object> old_value,
1040 ObjPtr<Object> new_value) {
Hans Boehmd8434432014-07-11 09:56:07 -07001041 if (kCheckTransaction) {
1042 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
1043 }
1044 if (kVerifyFlags & kVerifyThis) {
1045 VerifyObject(this);
1046 }
1047 if (kVerifyFlags & kVerifyWrites) {
1048 VerifyObject(new_value);
1049 }
1050 if (kVerifyFlags & kVerifyReads) {
1051 VerifyObject(old_value);
1052 }
1053 if (kTransactionActive) {
1054 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
1055 }
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001056 HeapReference<Object> old_ref(HeapReference<Object>::FromObjPtr(old_value));
1057 HeapReference<Object> new_ref(HeapReference<Object>::FromObjPtr(new_value));
Ian Rogers13735952014-10-08 12:43:28 -07001058 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Hans Boehmd8434432014-07-11 09:56:07 -07001059 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
1060
1061 bool success = atomic_addr->CompareExchangeStrongSequentiallyConsistent(old_ref.reference_,
1062 new_ref.reference_);
Hans Boehmd8434432014-07-11 09:56:07 -07001063 return success;
1064}
1065
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -07001066template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
1067inline bool Object::CasFieldWeakRelaxedObjectWithoutWriteBarrier(
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001068 MemberOffset field_offset,
1069 ObjPtr<Object> old_value,
1070 ObjPtr<Object> new_value) {
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -07001071 if (kCheckTransaction) {
1072 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
1073 }
1074 if (kVerifyFlags & kVerifyThis) {
1075 VerifyObject(this);
1076 }
1077 if (kVerifyFlags & kVerifyWrites) {
1078 VerifyObject(new_value);
1079 }
1080 if (kVerifyFlags & kVerifyReads) {
1081 VerifyObject(old_value);
1082 }
1083 if (kTransactionActive) {
1084 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
1085 }
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001086 HeapReference<Object> old_ref(HeapReference<Object>::FromObjPtr(old_value));
1087 HeapReference<Object> new_ref(HeapReference<Object>::FromObjPtr(new_value));
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -07001088 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
1089 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
1090
1091 bool success = atomic_addr->CompareExchangeWeakRelaxed(old_ref.reference_,
1092 new_ref.reference_);
1093 return success;
1094}
1095
1096template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
1097inline bool Object::CasFieldStrongRelaxedObjectWithoutWriteBarrier(
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001098 MemberOffset field_offset,
1099 ObjPtr<Object> old_value,
1100 ObjPtr<Object> new_value) {
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -07001101 if (kCheckTransaction) {
1102 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
1103 }
1104 if (kVerifyFlags & kVerifyThis) {
1105 VerifyObject(this);
1106 }
1107 if (kVerifyFlags & kVerifyWrites) {
1108 VerifyObject(new_value);
1109 }
1110 if (kVerifyFlags & kVerifyReads) {
1111 VerifyObject(old_value);
1112 }
1113 if (kTransactionActive) {
1114 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
1115 }
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001116 HeapReference<Object> old_ref(HeapReference<Object>::FromObjPtr(old_value));
1117 HeapReference<Object> new_ref(HeapReference<Object>::FromObjPtr(new_value));
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -07001118 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
1119 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
1120
1121 bool success = atomic_addr->CompareExchangeStrongRelaxed(old_ref.reference_,
1122 new_ref.reference_);
1123 return success;
1124}
1125
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001126template<bool kIsStatic,
1127 VerifyObjectFlags kVerifyFlags,
1128 ReadBarrierOption kReadBarrierOption,
1129 typename Visitor>
Mathieu Chartier407f7022014-02-18 14:37:05 -08001130inline void Object::VisitFieldsReferences(uint32_t ref_offsets, const Visitor& visitor) {
Ian Rogerscdc1aaf2014-10-09 13:21:38 -07001131 if (!kIsStatic && (ref_offsets != mirror::Class::kClassWalkSuper)) {
1132 // Instance fields and not the slow-path.
Ian Rogerscdc1aaf2014-10-09 13:21:38 -07001133 uint32_t field_offset = mirror::kObjectHeaderSize;
Mathieu Chartier407f7022014-02-18 14:37:05 -08001134 while (ref_offsets != 0) {
Ian Rogerscdc1aaf2014-10-09 13:21:38 -07001135 if ((ref_offsets & 1) != 0) {
1136 visitor(this, MemberOffset(field_offset), kIsStatic);
1137 }
1138 ref_offsets >>= 1;
1139 field_offset += sizeof(mirror::HeapReference<mirror::Object>);
Mathieu Chartier407f7022014-02-18 14:37:05 -08001140 }
1141 } else {
Mingyao Yangfaff0f02014-09-10 12:03:22 -07001142 // There is no reference offset bitmap. In the non-static case, walk up the class
Mathieu Chartier407f7022014-02-18 14:37:05 -08001143 // inheritance hierarchy and find reference offsets the hard way. In the static case, just
1144 // consider this class.
Mathieu Chartier31e88222016-10-14 18:43:19 -07001145 for (ObjPtr<Class> klass = kIsStatic
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001146 ? AsClass<kVerifyFlags, kReadBarrierOption>()
1147 : GetClass<kVerifyFlags, kReadBarrierOption>();
1148 klass != nullptr;
1149 klass = kIsStatic ? nullptr : klass->GetSuperClass<kVerifyFlags, kReadBarrierOption>()) {
1150 const size_t num_reference_fields =
Mathieu Chartier407f7022014-02-18 14:37:05 -08001151 kIsStatic ? klass->NumReferenceStaticFields() : klass->NumReferenceInstanceFields();
Vladimir Marko76649e82014-11-10 18:32:59 +00001152 if (num_reference_fields == 0u) {
1153 continue;
1154 }
Mathieu Chartiere401d142015-04-22 13:56:20 -07001155 // Presumably GC can happen when we are cross compiling, it should not cause performance
1156 // problems to do pointer size logic.
Vladimir Marko76649e82014-11-10 18:32:59 +00001157 MemberOffset field_offset = kIsStatic
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001158 ? klass->GetFirstReferenceStaticFieldOffset<kVerifyFlags, kReadBarrierOption>(
Mathieu Chartiere401d142015-04-22 13:56:20 -07001159 Runtime::Current()->GetClassLinker()->GetImagePointerSize())
Hiroshi Yamauchi5496f692016-02-17 13:29:59 -08001160 : klass->GetFirstReferenceInstanceFieldOffset<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartier059ef3d2015-08-18 13:54:21 -07001161 for (size_t i = 0u; i < num_reference_fields; ++i) {
Mathieu Chartier407f7022014-02-18 14:37:05 -08001162 // TODO: Do a simpler check?
Mathieu Chartier059ef3d2015-08-18 13:54:21 -07001163 if (field_offset.Uint32Value() != ClassOffset().Uint32Value()) {
Mathieu Chartier52e4b432014-06-10 11:22:31 -07001164 visitor(this, field_offset, kIsStatic);
Mathieu Chartier407f7022014-02-18 14:37:05 -08001165 }
Vladimir Marko76649e82014-11-10 18:32:59 +00001166 field_offset = MemberOffset(field_offset.Uint32Value() +
1167 sizeof(mirror::HeapReference<mirror::Object>));
Mathieu Chartier407f7022014-02-18 14:37:05 -08001168 }
1169 }
1170 }
1171}
1172
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001173template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption, typename Visitor>
Mathieu Chartier31e88222016-10-14 18:43:19 -07001174inline void Object::VisitInstanceFieldsReferences(ObjPtr<Class> klass, const Visitor& visitor) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001175 VisitFieldsReferences<false, kVerifyFlags, kReadBarrierOption>(
1176 klass->GetReferenceInstanceOffsets<kVerifyFlags>(), visitor);
Mathieu Chartier407f7022014-02-18 14:37:05 -08001177}
1178
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001179template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption, typename Visitor>
Mathieu Chartier31e88222016-10-14 18:43:19 -07001180inline void Object::VisitStaticFieldsReferences(ObjPtr<Class> klass, const Visitor& visitor) {
Mingyao Yang98d1cc82014-05-15 17:02:16 -07001181 DCHECK(!klass->IsTemp());
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001182 klass->VisitFieldsReferences<true, kVerifyFlags, kReadBarrierOption>(0, visitor);
Mathieu Chartier407f7022014-02-18 14:37:05 -08001183}
1184
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001185template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere4275c02015-08-06 15:34:15 -07001186inline bool Object::IsClassLoader() {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001187 return GetClass<kVerifyFlags, kReadBarrierOption>()->IsClassLoaderClass();
Mathieu Chartiere4275c02015-08-06 15:34:15 -07001188}
1189
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001190template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere4275c02015-08-06 15:34:15 -07001191inline mirror::ClassLoader* Object::AsClassLoader() {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001192 DCHECK((IsClassLoader<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartiere4275c02015-08-06 15:34:15 -07001193 return down_cast<mirror::ClassLoader*>(this);
1194}
1195
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001196template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Vladimir Marko05792b92015-08-03 11:56:49 +01001197inline bool Object::IsDexCache() {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001198 return GetClass<kVerifyFlags, kReadBarrierOption>()->IsDexCacheClass();
Vladimir Marko05792b92015-08-03 11:56:49 +01001199}
1200
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001201template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Vladimir Marko05792b92015-08-03 11:56:49 +01001202inline mirror::DexCache* Object::AsDexCache() {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001203 DCHECK((IsDexCache<kVerifyFlags, kReadBarrierOption>()));
Vladimir Marko05792b92015-08-03 11:56:49 +01001204 return down_cast<mirror::DexCache*>(this);
1205}
1206
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001207template <bool kVisitNativeRoots,
1208 VerifyObjectFlags kVerifyFlags,
1209 ReadBarrierOption kReadBarrierOption,
1210 typename Visitor,
1211 typename JavaLangRefVisitor>
Mathieu Chartier407f7022014-02-18 14:37:05 -08001212inline void Object::VisitReferences(const Visitor& visitor,
1213 const JavaLangRefVisitor& ref_visitor) {
Mathieu Chartier31e88222016-10-14 18:43:19 -07001214 ObjPtr<Class> klass = GetClass<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartier059ef3d2015-08-18 13:54:21 -07001215 visitor(this, ClassOffset(), false);
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001216 const uint32_t class_flags = klass->GetClassFlags<kVerifyNone>();
1217 if (LIKELY(class_flags == kClassFlagNormal)) {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001218 DCHECK((!klass->IsVariableSize<kVerifyFlags, kReadBarrierOption>()));
1219 VisitInstanceFieldsReferences<kVerifyFlags, kReadBarrierOption>(klass, visitor);
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001220 DCHECK((!klass->IsClassClass<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartier66c2d2d2015-08-25 14:32:32 -07001221 DCHECK(!klass->IsStringClass());
1222 DCHECK(!klass->IsClassLoaderClass());
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001223 DCHECK((!klass->IsArrayClass<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001224 } else {
1225 if ((class_flags & kClassFlagNoReferenceFields) == 0) {
1226 DCHECK(!klass->IsStringClass());
1227 if (class_flags == kClassFlagClass) {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001228 DCHECK((klass->IsClassClass<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartier31e88222016-10-14 18:43:19 -07001229 ObjPtr<Class> as_klass = AsClass<kVerifyNone, kReadBarrierOption>();
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001230 as_klass->VisitReferences<kVisitNativeRoots, kVerifyFlags, kReadBarrierOption>(klass,
1231 visitor);
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001232 } else if (class_flags == kClassFlagObjectArray) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001233 DCHECK((klass->IsObjectArrayClass<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001234 AsObjectArray<mirror::Object, kVerifyNone, kReadBarrierOption>()->VisitReferences(visitor);
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001235 } else if ((class_flags & kClassFlagReference) != 0) {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001236 VisitInstanceFieldsReferences<kVerifyFlags, kReadBarrierOption>(klass, visitor);
1237 ref_visitor(klass, AsReference<kVerifyFlags, kReadBarrierOption>());
Vladimir Marko05792b92015-08-03 11:56:49 +01001238 } else if (class_flags == kClassFlagDexCache) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001239 mirror::DexCache* const dex_cache = AsDexCache<kVerifyFlags, kReadBarrierOption>();
1240 dex_cache->VisitReferences<kVisitNativeRoots,
1241 kVerifyFlags,
1242 kReadBarrierOption>(klass, visitor);
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001243 } else {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001244 mirror::ClassLoader* const class_loader = AsClassLoader<kVerifyFlags, kReadBarrierOption>();
1245 class_loader->VisitReferences<kVisitNativeRoots,
1246 kVerifyFlags,
1247 kReadBarrierOption>(klass, visitor);
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001248 }
1249 } else if (kIsDebugBuild) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001250 CHECK((!klass->IsClassClass<kVerifyFlags, kReadBarrierOption>()));
1251 CHECK((!klass->IsObjectArrayClass<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001252 // String still has instance fields for reflection purposes but these don't exist in
1253 // actual string instances.
1254 if (!klass->IsStringClass()) {
1255 size_t total_reference_instance_fields = 0;
Mathieu Chartier31e88222016-10-14 18:43:19 -07001256 ObjPtr<Class> super_class = klass;
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001257 do {
1258 total_reference_instance_fields += super_class->NumReferenceInstanceFields();
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001259 super_class = super_class->GetSuperClass<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001260 } while (super_class != nullptr);
1261 // The only reference field should be the object's class. This field is handled at the
1262 // beginning of the function.
1263 CHECK_EQ(total_reference_instance_fields, 1u);
1264 }
Mathieu Chartier407f7022014-02-18 14:37:05 -08001265 }
1266 }
1267}
Ian Rogers2dd0e2c2013-01-24 12:42:14 -08001268} // namespace mirror
1269} // namespace art
1270
Brian Carlstromfc0e3212013-07-17 14:40:12 -07001271#endif // ART_RUNTIME_MIRROR_OBJECT_INL_H_