blob: 811f1ea726ab097ff8f4b97392394a90bcd37159 [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
Hans Boehmb3da36c2016-12-15 13:12:59 -0800100inline bool Object::CasLockWordWeakAcquire(LockWord old_val, LockWord new_val) {
101 // Force use of non-transactional mode and do not check.
102 return CasFieldWeakAcquire32<false, false>(
103 OFFSET_OF_OBJECT_MEMBER(Object, monitor_), old_val.GetValue(), new_val.GetValue());
104}
105
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -0700106inline bool Object::CasLockWordWeakRelease(LockWord old_val, LockWord new_val) {
107 // Force use of non-transactional mode and do not check.
108 return CasFieldWeakRelease32<false, false>(
109 OFFSET_OF_OBJECT_MEMBER(Object, monitor_), old_val.GetValue(), new_val.GetValue());
110}
111
Ian Rogersd9c4fc92013-10-01 19:45:43 -0700112inline uint32_t Object::GetLockOwnerThreadId() {
113 return Monitor::GetLockOwnerThreadId(this);
Ian Rogers05f30572013-02-20 12:13:11 -0800114}
115
Mathieu Chartiere7e8a5f2014-02-14 16:59:41 -0800116inline mirror::Object* Object::MonitorEnter(Thread* self) {
Mathieu Chartiera704eda2016-07-13 09:53:35 -0700117 return Monitor::MonitorEnter(self, this, /*trylock*/false);
118}
119
120inline mirror::Object* Object::MonitorTryEnter(Thread* self) {
121 return Monitor::MonitorEnter(self, this, /*trylock*/true);
Ian Rogers05f30572013-02-20 12:13:11 -0800122}
123
124inline bool Object::MonitorExit(Thread* self) {
125 return Monitor::MonitorExit(self, this);
126}
127
128inline void Object::Notify(Thread* self) {
129 Monitor::Notify(self, this);
130}
131
132inline void Object::NotifyAll(Thread* self) {
133 Monitor::NotifyAll(self, this);
134}
135
136inline void Object::Wait(Thread* self) {
137 Monitor::Wait(self, this, 0, 0, true, kWaiting);
138}
139
140inline void Object::Wait(Thread* self, int64_t ms, int32_t ns) {
141 Monitor::Wait(self, this, ms, ns, true, kTimedWaiting);
142}
143
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700144inline uint32_t Object::GetReadBarrierState(uintptr_t* fake_address_dependency) {
Andreas Gampe90b936d2017-01-31 08:58:55 -0800145 if (!kUseBakerReadBarrier) {
146 LOG(FATAL) << "Unreachable";
147 UNREACHABLE();
148 }
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700149#if defined(__arm__)
150 uintptr_t obj = reinterpret_cast<uintptr_t>(this);
151 uintptr_t result;
152 DCHECK_EQ(OFFSETOF_MEMBER(Object, monitor_), 4U);
153 // Use inline assembly to prevent the compiler from optimizing away the false dependency.
154 __asm__ __volatile__(
155 "ldr %[result], [%[obj], #4]\n\t"
156 // This instruction is enough to "fool the compiler and the CPU" by having `fad` always be
157 // null, without them being able to assume that fact.
158 "eor %[fad], %[result], %[result]\n\t"
159 : [result] "+r" (result), [fad] "=r" (*fake_address_dependency)
160 : [obj] "r" (obj));
161 DCHECK_EQ(*fake_address_dependency, 0U);
162 LockWord lw(static_cast<uint32_t>(result));
163 uint32_t rb_state = lw.ReadBarrierState();
164 return rb_state;
165#elif defined(__aarch64__)
166 uintptr_t obj = reinterpret_cast<uintptr_t>(this);
167 uintptr_t result;
168 DCHECK_EQ(OFFSETOF_MEMBER(Object, monitor_), 4U);
169 // Use inline assembly to prevent the compiler from optimizing away the false dependency.
170 __asm__ __volatile__(
171 "ldr %w[result], [%[obj], #4]\n\t"
172 // This instruction is enough to "fool the compiler and the CPU" by having `fad` always be
173 // null, without them being able to assume that fact.
174 "eor %[fad], %[result], %[result]\n\t"
175 : [result] "+r" (result), [fad] "=r" (*fake_address_dependency)
176 : [obj] "r" (obj));
177 DCHECK_EQ(*fake_address_dependency, 0U);
178 LockWord lw(static_cast<uint32_t>(result));
179 uint32_t rb_state = lw.ReadBarrierState();
180 return rb_state;
181#elif defined(__i386__) || defined(__x86_64__)
182 LockWord lw = GetLockWord(false);
183 // i386/x86_64 don't need fake address dependency. Use a compiler fence to avoid compiler
184 // reordering.
185 *fake_address_dependency = 0;
186 std::atomic_signal_fence(std::memory_order_acquire);
187 uint32_t rb_state = lw.ReadBarrierState();
188 return rb_state;
189#else
Alexey Frunze15958152017-02-09 19:08:30 -0800190 // MIPS32/MIPS64: use a memory barrier to prevent load-load reordering.
191 LockWord lw = GetLockWord(false);
192 *fake_address_dependency = 0;
193 std::atomic_thread_fence(std::memory_order_acquire);
194 uint32_t rb_state = lw.ReadBarrierState();
195 return rb_state;
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700196#endif
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700197}
198
199inline uint32_t Object::GetReadBarrierState() {
Andreas Gampe90b936d2017-01-31 08:58:55 -0800200 if (!kUseBakerReadBarrier) {
201 LOG(FATAL) << "Unreachable";
202 UNREACHABLE();
203 }
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700204 DCHECK(kUseBakerReadBarrier);
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700205 LockWord lw(GetField<uint32_t, /*kIsVolatile*/false>(OFFSET_OF_OBJECT_MEMBER(Object, monitor_)));
206 uint32_t rb_state = lw.ReadBarrierState();
207 DCHECK(ReadBarrier::IsValidReadBarrierState(rb_state)) << rb_state;
208 return rb_state;
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800209}
210
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700211inline uint32_t Object::GetReadBarrierStateAcquire() {
Andreas Gampe90b936d2017-01-31 08:58:55 -0800212 if (!kUseBakerReadBarrier) {
213 LOG(FATAL) << "Unreachable";
214 UNREACHABLE();
215 }
Mathieu Chartierc381c362016-08-23 13:27:53 -0700216 LockWord lw(GetFieldAcquire<uint32_t>(OFFSET_OF_OBJECT_MEMBER(Object, monitor_)));
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700217 uint32_t rb_state = lw.ReadBarrierState();
218 DCHECK(ReadBarrier::IsValidReadBarrierState(rb_state)) << rb_state;
219 return rb_state;
Mathieu Chartierc381c362016-08-23 13:27:53 -0700220}
221
Mathieu Chartier36a270a2016-07-28 18:08:51 -0700222inline uint32_t Object::GetMarkBit() {
223#ifdef USE_READ_BARRIER
Mathieu Chartier36a270a2016-07-28 18:08:51 -0700224 return GetLockWord(false).MarkBitState();
225#else
226 LOG(FATAL) << "Unreachable";
227 UNREACHABLE();
228#endif
229}
230
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700231inline void Object::SetReadBarrierState(uint32_t rb_state) {
Andreas Gampe90b936d2017-01-31 08:58:55 -0800232 if (!kUseBakerReadBarrier) {
233 LOG(FATAL) << "Unreachable";
234 UNREACHABLE();
235 }
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700236 DCHECK(ReadBarrier::IsValidReadBarrierState(rb_state)) << rb_state;
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700237 LockWord lw = GetLockWord(false);
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700238 lw.SetReadBarrierState(rb_state);
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700239 SetLockWord(lw, false);
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800240}
241
Hiroshi Yamauchied70b4a2015-11-17 17:52:15 -0800242template<bool kCasRelease>
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700243inline bool Object::AtomicSetReadBarrierState(uint32_t expected_rb_state, uint32_t rb_state) {
Andreas Gampe90b936d2017-01-31 08:58:55 -0800244 if (!kUseBakerReadBarrier) {
245 LOG(FATAL) << "Unreachable";
246 UNREACHABLE();
247 }
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700248 DCHECK(ReadBarrier::IsValidReadBarrierState(expected_rb_state)) << expected_rb_state;
249 DCHECK(ReadBarrier::IsValidReadBarrierState(rb_state)) << rb_state;
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700250 LockWord expected_lw;
251 LockWord new_lw;
252 do {
253 LockWord lw = GetLockWord(false);
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700254 if (UNLIKELY(lw.ReadBarrierState() != expected_rb_state)) {
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700255 // Lost the race.
256 return false;
257 }
258 expected_lw = lw;
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700259 expected_lw.SetReadBarrierState(expected_rb_state);
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700260 new_lw = lw;
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700261 new_lw.SetReadBarrierState(rb_state);
Hiroshi Yamauchied70b4a2015-11-17 17:52:15 -0800262 // ConcurrentCopying::ProcessMarkStackRef uses this with kCasRelease == true.
263 // If kCasRelease == true, use a CAS release so that when GC updates all the fields of
264 // an object and then changes the object from gray to black, the field updates (stores) will be
265 // visible (won't be reordered after this CAS.)
266 } while (!(kCasRelease ?
267 CasLockWordWeakRelease(expected_lw, new_lw) :
268 CasLockWordWeakRelaxed(expected_lw, new_lw)));
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700269 return true;
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700270}
271
Mathieu Chartier36a270a2016-07-28 18:08:51 -0700272inline bool Object::AtomicSetMarkBit(uint32_t expected_mark_bit, uint32_t mark_bit) {
273 LockWord expected_lw;
274 LockWord new_lw;
275 do {
276 LockWord lw = GetLockWord(false);
277 if (UNLIKELY(lw.MarkBitState() != expected_mark_bit)) {
278 // Lost the race.
279 return false;
280 }
281 expected_lw = lw;
282 new_lw = lw;
283 new_lw.SetMarkBitState(mark_bit);
284 // Since this is only set from the mutator, we can use the non release Cas.
285 } while (!CasLockWordWeakRelaxed(expected_lw, new_lw));
286 return true;
287}
288
289
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700290inline void Object::AssertReadBarrierState() const {
291 CHECK(kUseBakerReadBarrier);
292 Object* obj = const_cast<Object*>(this);
293 DCHECK(obj->GetReadBarrierState() == ReadBarrier::WhiteState())
294 << "Bad Baker pointer: obj=" << reinterpret_cast<void*>(obj)
295 << " rb_state" << reinterpret_cast<void*>(obj->GetReadBarrierState());
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800296}
297
Mathieu Chartier4e305412014-02-19 10:54:44 -0800298template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartieref41db72016-10-25 15:08:01 -0700299inline bool Object::VerifierInstanceOf(ObjPtr<Class> klass) {
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700300 DCHECK(klass != nullptr);
301 DCHECK(GetClass<kVerifyFlags>() != nullptr);
Jeff Haoa3faaf42013-09-03 19:07:00 -0700302 return klass->IsInterface() || InstanceOf(klass);
303}
304
Mathieu Chartier4e305412014-02-19 10:54:44 -0800305template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiera59d9b22016-09-26 18:13:17 -0700306inline bool Object::InstanceOf(ObjPtr<Class> klass) {
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700307 DCHECK(klass != nullptr);
308 DCHECK(GetClass<kVerifyNone>() != nullptr);
Mathieu Chartier4e305412014-02-19 10:54:44 -0800309 return klass->IsAssignableFrom(GetClass<kVerifyFlags>());
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800310}
311
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700312template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800313inline bool Object::IsClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700314 Class* java_lang_Class = GetClass<kVerifyFlags, kReadBarrierOption>()->
315 template GetClass<kVerifyFlags, kReadBarrierOption>();
316 return GetClass<static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis),
317 kReadBarrierOption>() == java_lang_Class;
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800318}
319
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700320template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800321inline Class* Object::AsClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700322 DCHECK((IsClass<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800323 return down_cast<Class*>(this);
324}
325
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800326template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800327inline bool Object::IsObjectArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800328 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800329 return IsArrayInstance<kVerifyFlags, kReadBarrierOption>() &&
330 !GetClass<kNewFlags, kReadBarrierOption>()->
331 template GetComponentType<kNewFlags, kReadBarrierOption>()->IsPrimitive();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800332}
333
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800334template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800335inline ObjectArray<T>* Object::AsObjectArray() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800336 DCHECK((IsObjectArray<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800337 return down_cast<ObjectArray<T>*>(this);
338}
339
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700340template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800341inline bool Object::IsArrayInstance() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700342 return GetClass<kVerifyFlags, kReadBarrierOption>()->
343 template IsArrayClass<kVerifyFlags, kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800344}
345
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800346template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800347inline bool Object::IsReferenceInstance() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800348 return GetClass<kVerifyFlags, kReadBarrierOption>()->IsTypeOfReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800349}
350
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800351template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700352inline Reference* Object::AsReference() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800353 DCHECK((IsReferenceInstance<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700354 return down_cast<Reference*>(this);
355}
356
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700357template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers05f30572013-02-20 12:13:11 -0800358inline Array* Object::AsArray() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700359 DCHECK((IsArrayInstance<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800360 return down_cast<Array*>(this);
361}
362
Mathieu Chartier4e305412014-02-19 10:54:44 -0800363template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800364inline BooleanArray* Object::AsBooleanArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800365 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
366 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
367 DCHECK(GetClass<kNewFlags>()->GetComponentType()->IsPrimitiveBoolean());
Ian Rogers05f30572013-02-20 12:13:11 -0800368 return down_cast<BooleanArray*>(this);
369}
370
Mathieu Chartier4e305412014-02-19 10:54:44 -0800371template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800372inline ByteArray* Object::AsByteArray() {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700373 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Mathieu Chartier4e305412014-02-19 10:54:44 -0800374 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
375 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveByte());
Ian Rogers05f30572013-02-20 12:13:11 -0800376 return down_cast<ByteArray*>(this);
377}
378
Mathieu Chartier4e305412014-02-19 10:54:44 -0800379template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800380inline ByteArray* Object::AsByteSizedArray() {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700381 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Mathieu Chartier4e305412014-02-19 10:54:44 -0800382 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
383 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveByte() ||
384 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveBoolean());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800385 return down_cast<ByteArray*>(this);
386}
387
Mathieu Chartier4e305412014-02-19 10:54:44 -0800388template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800389inline CharArray* Object::AsCharArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800390 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
391 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
392 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveChar());
Ian Rogers05f30572013-02-20 12:13:11 -0800393 return down_cast<CharArray*>(this);
394}
395
Mathieu Chartier4e305412014-02-19 10:54:44 -0800396template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800397inline ShortArray* Object::AsShortArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800398 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
399 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
400 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveShort());
Ian Rogers05f30572013-02-20 12:13:11 -0800401 return down_cast<ShortArray*>(this);
402}
403
Mathieu Chartier4e305412014-02-19 10:54:44 -0800404template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800405inline ShortArray* Object::AsShortSizedArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800406 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
407 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
408 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveShort() ||
409 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveChar());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800410 return down_cast<ShortArray*>(this);
411}
412
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800413template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700414inline bool Object::IsIntArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800415 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Mathieu Chartier31e88222016-10-14 18:43:19 -0700416 ObjPtr<Class> klass = GetClass<kVerifyFlags, kReadBarrierOption>();
417 ObjPtr<Class> component_type = klass->GetComponentType<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartiere401d142015-04-22 13:56:20 -0700418 return component_type != nullptr && component_type->template IsPrimitiveInt<kNewFlags>();
419}
420
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800421template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700422inline IntArray* Object::AsIntArray() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800423 DCHECK((IsIntArray<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800424 return down_cast<IntArray*>(this);
425}
426
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800427template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700428inline bool Object::IsLongArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800429 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Mathieu Chartier31e88222016-10-14 18:43:19 -0700430 ObjPtr<Class> klass = GetClass<kVerifyFlags, kReadBarrierOption>();
431 ObjPtr<Class> component_type = klass->GetComponentType<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartiere401d142015-04-22 13:56:20 -0700432 return component_type != nullptr && component_type->template IsPrimitiveLong<kNewFlags>();
433}
434
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800435template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700436inline LongArray* Object::AsLongArray() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800437 DCHECK((IsLongArray<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800438 return down_cast<LongArray*>(this);
439}
440
Mathieu Chartier4e305412014-02-19 10:54:44 -0800441template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700442inline bool Object::IsFloatArray() {
443 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
444 auto* component_type = GetClass<kVerifyFlags>()->GetComponentType();
445 return component_type != nullptr && component_type->template IsPrimitiveFloat<kNewFlags>();
446}
447
448template<VerifyObjectFlags kVerifyFlags>
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100449inline FloatArray* Object::AsFloatArray() {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700450 DCHECK(IsFloatArray<kVerifyFlags>());
Mathieu Chartier4e305412014-02-19 10:54:44 -0800451 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
452 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
453 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveFloat());
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100454 return down_cast<FloatArray*>(this);
455}
456
Mathieu Chartier4e305412014-02-19 10:54:44 -0800457template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700458inline bool Object::IsDoubleArray() {
459 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
460 auto* component_type = GetClass<kVerifyFlags>()->GetComponentType();
461 return component_type != nullptr && component_type->template IsPrimitiveDouble<kNewFlags>();
462}
463
464template<VerifyObjectFlags kVerifyFlags>
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100465inline DoubleArray* Object::AsDoubleArray() {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700466 DCHECK(IsDoubleArray<kVerifyFlags>());
Mathieu Chartier4e305412014-02-19 10:54:44 -0800467 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
468 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
469 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveDouble());
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100470 return down_cast<DoubleArray*>(this);
471}
472
Jeff Hao848f70a2014-01-15 13:49:50 -0800473template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
474inline bool Object::IsString() {
475 return GetClass<kVerifyFlags, kReadBarrierOption>()->IsStringClass();
476}
477
478template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers05f30572013-02-20 12:13:11 -0800479inline String* Object::AsString() {
Jeff Hao848f70a2014-01-15 13:49:50 -0800480 DCHECK((IsString<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800481 return down_cast<String*>(this);
482}
483
Mathieu Chartier4e305412014-02-19 10:54:44 -0800484template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800485inline Throwable* Object::AsThrowable() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800486 DCHECK(GetClass<kVerifyFlags>()->IsThrowableClass());
Ian Rogers05f30572013-02-20 12:13:11 -0800487 return down_cast<Throwable*>(this);
488}
489
Mathieu Chartier4e305412014-02-19 10:54:44 -0800490template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800491inline bool Object::IsWeakReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800492 return GetClass<kVerifyFlags>()->IsWeakReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800493}
494
Mathieu Chartier4e305412014-02-19 10:54:44 -0800495template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800496inline bool Object::IsSoftReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800497 return GetClass<kVerifyFlags>()->IsSoftReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800498}
499
Mathieu Chartier4e305412014-02-19 10:54:44 -0800500template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800501inline bool Object::IsFinalizerReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800502 return GetClass<kVerifyFlags>()->IsFinalizerReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800503}
504
Mathieu Chartier4e305412014-02-19 10:54:44 -0800505template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700506inline FinalizerReference* Object::AsFinalizerReference() {
507 DCHECK(IsFinalizerReferenceInstance<kVerifyFlags>());
508 return down_cast<FinalizerReference*>(this);
509}
510
511template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800512inline bool Object::IsPhantomReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800513 return GetClass<kVerifyFlags>()->IsPhantomReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800514}
515
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700516template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800517inline size_t Object::SizeOf() {
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800518 size_t result;
Mathieu Chartier4e305412014-02-19 10:54:44 -0800519 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700520 if (IsArrayInstance<kVerifyFlags, kReadBarrierOption>()) {
521 result = AsArray<kNewFlags, kReadBarrierOption>()->
522 template SizeOf<kNewFlags, kReadBarrierOption>();
523 } else if (IsClass<kNewFlags, kReadBarrierOption>()) {
524 result = AsClass<kNewFlags, kReadBarrierOption>()->
525 template SizeOf<kNewFlags, kReadBarrierOption>();
Jeff Hao848f70a2014-01-15 13:49:50 -0800526 } else if (GetClass<kNewFlags, kReadBarrierOption>()->IsStringClass()) {
527 result = AsString<kNewFlags, kReadBarrierOption>()->
528 template SizeOf<kNewFlags>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800529 } else {
Hiroshi Yamauchi25023c72014-05-09 11:45:53 -0700530 result = GetClass<kNewFlags, kReadBarrierOption>()->
531 template GetObjectSize<kNewFlags, kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800532 }
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700533 DCHECK_GE(result, sizeof(Object))
David Sehr709b0702016-10-13 09:12:37 -0700534 << " class=" << Class::PrettyClass(GetClass<kNewFlags, kReadBarrierOption>());
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800535 return result;
536}
537
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700538template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
Fred Shih37f05ef2014-07-16 18:38:08 -0700539inline uint8_t Object::GetFieldBoolean(MemberOffset field_offset) {
540 if (kVerifyFlags & kVerifyThis) {
541 VerifyObject(this);
542 }
543 return GetField<uint8_t, kIsVolatile>(field_offset);
544}
545
546template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
547inline int8_t Object::GetFieldByte(MemberOffset field_offset) {
548 if (kVerifyFlags & kVerifyThis) {
549 VerifyObject(this);
550 }
551 return GetField<int8_t, kIsVolatile>(field_offset);
552}
553
554template<VerifyObjectFlags kVerifyFlags>
555inline uint8_t Object::GetFieldBooleanVolatile(MemberOffset field_offset) {
556 return GetFieldBoolean<kVerifyFlags, true>(field_offset);
557}
558
559template<VerifyObjectFlags kVerifyFlags>
560inline int8_t Object::GetFieldByteVolatile(MemberOffset field_offset) {
561 return GetFieldByte<kVerifyFlags, true>(field_offset);
562}
563
564template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
565 bool kIsVolatile>
566inline void Object::SetFieldBoolean(MemberOffset field_offset, uint8_t new_value)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700567 REQUIRES_SHARED(Locks::mutator_lock_) {
Fred Shih37f05ef2014-07-16 18:38:08 -0700568 if (kCheckTransaction) {
569 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
570 }
571 if (kTransactionActive) {
572 Runtime::Current()->RecordWriteFieldBoolean(this, field_offset,
573 GetFieldBoolean<kVerifyFlags, kIsVolatile>(field_offset),
574 kIsVolatile);
575 }
576 if (kVerifyFlags & kVerifyThis) {
577 VerifyObject(this);
578 }
579 SetField<uint8_t, kIsVolatile>(field_offset, new_value);
580}
581
582template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
583 bool kIsVolatile>
584inline void Object::SetFieldByte(MemberOffset field_offset, int8_t new_value)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700585 REQUIRES_SHARED(Locks::mutator_lock_) {
Fred Shih37f05ef2014-07-16 18:38:08 -0700586 if (kCheckTransaction) {
587 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
588 }
589 if (kTransactionActive) {
590 Runtime::Current()->RecordWriteFieldByte(this, field_offset,
591 GetFieldByte<kVerifyFlags, kIsVolatile>(field_offset),
592 kIsVolatile);
593 }
594 if (kVerifyFlags & kVerifyThis) {
595 VerifyObject(this);
596 }
597 SetField<int8_t, kIsVolatile>(field_offset, new_value);
598}
599
600template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
601inline void Object::SetFieldBooleanVolatile(MemberOffset field_offset, uint8_t new_value) {
602 return SetFieldBoolean<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
603 field_offset, new_value);
604}
605
606template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
607inline void Object::SetFieldByteVolatile(MemberOffset field_offset, int8_t new_value) {
608 return SetFieldByte<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
609 field_offset, new_value);
610}
611
612template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
613inline uint16_t Object::GetFieldChar(MemberOffset field_offset) {
614 if (kVerifyFlags & kVerifyThis) {
615 VerifyObject(this);
616 }
617 return GetField<uint16_t, kIsVolatile>(field_offset);
618}
619
620template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
621inline int16_t Object::GetFieldShort(MemberOffset field_offset) {
622 if (kVerifyFlags & kVerifyThis) {
623 VerifyObject(this);
624 }
625 return GetField<int16_t, kIsVolatile>(field_offset);
626}
627
628template<VerifyObjectFlags kVerifyFlags>
629inline uint16_t Object::GetFieldCharVolatile(MemberOffset field_offset) {
630 return GetFieldChar<kVerifyFlags, true>(field_offset);
631}
632
633template<VerifyObjectFlags kVerifyFlags>
634inline int16_t Object::GetFieldShortVolatile(MemberOffset field_offset) {
635 return GetFieldShort<kVerifyFlags, true>(field_offset);
636}
637
638template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
639 bool kIsVolatile>
640inline void Object::SetFieldChar(MemberOffset field_offset, uint16_t new_value) {
641 if (kCheckTransaction) {
642 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
643 }
644 if (kTransactionActive) {
645 Runtime::Current()->RecordWriteFieldChar(this, field_offset,
646 GetFieldChar<kVerifyFlags, kIsVolatile>(field_offset),
647 kIsVolatile);
648 }
649 if (kVerifyFlags & kVerifyThis) {
650 VerifyObject(this);
651 }
652 SetField<uint16_t, kIsVolatile>(field_offset, new_value);
653}
654
655template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
656 bool kIsVolatile>
657inline void Object::SetFieldShort(MemberOffset field_offset, int16_t new_value) {
658 if (kCheckTransaction) {
659 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
660 }
661 if (kTransactionActive) {
662 Runtime::Current()->RecordWriteFieldChar(this, field_offset,
663 GetFieldShort<kVerifyFlags, kIsVolatile>(field_offset),
664 kIsVolatile);
665 }
666 if (kVerifyFlags & kVerifyThis) {
667 VerifyObject(this);
668 }
669 SetField<int16_t, kIsVolatile>(field_offset, new_value);
670}
671
672template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
673inline void Object::SetFieldCharVolatile(MemberOffset field_offset, uint16_t new_value) {
674 return SetFieldChar<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
675 field_offset, new_value);
676}
677
678template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
679inline void Object::SetFieldShortVolatile(MemberOffset field_offset, int16_t new_value) {
680 return SetFieldShort<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
681 field_offset, new_value);
682}
683
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700684template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
685 bool kIsVolatile>
686inline void Object::SetField32(MemberOffset field_offset, int32_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100687 if (kCheckTransaction) {
688 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
689 }
690 if (kTransactionActive) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700691 Runtime::Current()->RecordWriteField32(this, field_offset,
692 GetField32<kVerifyFlags, kIsVolatile>(field_offset),
693 kIsVolatile);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100694 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800695 if (kVerifyFlags & kVerifyThis) {
Ian Rogersb122a4b2013-11-19 18:00:50 -0800696 VerifyObject(this);
697 }
Fred Shih37f05ef2014-07-16 18:38:08 -0700698 SetField<int32_t, kIsVolatile>(field_offset, new_value);
Ian Rogersb122a4b2013-11-19 18:00:50 -0800699}
700
Mathieu Chartier4e305412014-02-19 10:54:44 -0800701template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700702inline void Object::SetField32Volatile(MemberOffset field_offset, int32_t new_value) {
703 SetField32<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset, new_value);
704}
705
Hans Boehmd8434432014-07-11 09:56:07 -0700706// TODO: Pass memory_order_ and strong/weak as arguments to avoid code duplication?
707
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700708template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -0700709inline bool Object::CasFieldWeakSequentiallyConsistent32(MemberOffset field_offset,
710 int32_t old_value, int32_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100711 if (kCheckTransaction) {
712 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
713 }
714 if (kTransactionActive) {
715 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
716 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800717 if (kVerifyFlags & kVerifyThis) {
718 VerifyObject(this);
719 }
Ian Rogers13735952014-10-08 12:43:28 -0700720 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogers228602f2014-07-10 02:07:54 -0700721 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
Hans Boehm30359612014-05-21 17:46:23 -0700722
Ian Rogers228602f2014-07-10 02:07:54 -0700723 return atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_value, new_value);
Ian Rogersd9c4fc92013-10-01 19:45:43 -0700724}
725
Hans Boehmd8434432014-07-11 09:56:07 -0700726template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
727inline bool Object::CasFieldWeakRelaxed32(MemberOffset field_offset,
728 int32_t old_value, int32_t new_value) {
729 if (kCheckTransaction) {
730 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
731 }
732 if (kTransactionActive) {
733 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
734 }
735 if (kVerifyFlags & kVerifyThis) {
736 VerifyObject(this);
737 }
Ian Rogers13735952014-10-08 12:43:28 -0700738 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Hans Boehmd8434432014-07-11 09:56:07 -0700739 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
740
741 return atomic_addr->CompareExchangeWeakRelaxed(old_value, new_value);
742}
743
744template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Hans Boehmb3da36c2016-12-15 13:12:59 -0800745inline bool Object::CasFieldWeakAcquire32(MemberOffset field_offset,
746 int32_t old_value, int32_t new_value) {
747 if (kCheckTransaction) {
748 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
749 }
750 if (kTransactionActive) {
751 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
752 }
753 if (kVerifyFlags & kVerifyThis) {
754 VerifyObject(this);
755 }
756 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
757 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
758
759 return atomic_addr->CompareExchangeWeakAcquire(old_value, new_value);
760}
761
762template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -0700763inline bool Object::CasFieldWeakRelease32(MemberOffset field_offset,
764 int32_t old_value, int32_t new_value) {
765 if (kCheckTransaction) {
766 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
767 }
768 if (kTransactionActive) {
769 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
770 }
771 if (kVerifyFlags & kVerifyThis) {
772 VerifyObject(this);
773 }
774 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
775 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
776
777 return atomic_addr->CompareExchangeWeakRelease(old_value, new_value);
778}
779
780template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Hans Boehmd8434432014-07-11 09:56:07 -0700781inline bool Object::CasFieldStrongSequentiallyConsistent32(MemberOffset field_offset,
782 int32_t old_value, int32_t new_value) {
783 if (kCheckTransaction) {
784 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
785 }
786 if (kTransactionActive) {
787 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
788 }
789 if (kVerifyFlags & kVerifyThis) {
790 VerifyObject(this);
791 }
Ian Rogers13735952014-10-08 12:43:28 -0700792 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Hans Boehmd8434432014-07-11 09:56:07 -0700793 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
794
795 return atomic_addr->CompareExchangeStrongSequentiallyConsistent(old_value, new_value);
796}
797
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700798template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
799inline int64_t Object::GetField64(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800800 if (kVerifyFlags & kVerifyThis) {
801 VerifyObject(this);
802 }
Fred Shih37f05ef2014-07-16 18:38:08 -0700803 return GetField<int64_t, kIsVolatile>(field_offset);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800804}
805
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700806template<VerifyObjectFlags kVerifyFlags>
807inline int64_t Object::GetField64Volatile(MemberOffset field_offset) {
808 return GetField64<kVerifyFlags, true>(field_offset);
809}
810
811template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
812 bool kIsVolatile>
813inline void Object::SetField64(MemberOffset field_offset, int64_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100814 if (kCheckTransaction) {
815 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
816 }
817 if (kTransactionActive) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700818 Runtime::Current()->RecordWriteField64(this, field_offset,
819 GetField64<kVerifyFlags, kIsVolatile>(field_offset),
820 kIsVolatile);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100821 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800822 if (kVerifyFlags & kVerifyThis) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800823 VerifyObject(this);
824 }
Fred Shih37f05ef2014-07-16 18:38:08 -0700825 SetField<int64_t, kIsVolatile>(field_offset, new_value);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800826}
827
Mathieu Chartier4e305412014-02-19 10:54:44 -0800828template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700829inline void Object::SetField64Volatile(MemberOffset field_offset, int64_t new_value) {
830 return SetField64<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
831 new_value);
832}
833
Mathieu Chartierc381c362016-08-23 13:27:53 -0700834template<typename kSize>
835inline kSize Object::GetFieldAcquire(MemberOffset field_offset) {
836 const uint8_t* raw_addr = reinterpret_cast<const uint8_t*>(this) + field_offset.Int32Value();
837 const kSize* addr = reinterpret_cast<const kSize*>(raw_addr);
838 return reinterpret_cast<const Atomic<kSize>*>(addr)->LoadAcquire();
839}
840
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700841template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -0700842inline bool Object::CasFieldWeakSequentiallyConsistent64(MemberOffset field_offset,
843 int64_t old_value, int64_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100844 if (kCheckTransaction) {
845 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
846 }
847 if (kTransactionActive) {
848 Runtime::Current()->RecordWriteField64(this, field_offset, old_value, true);
849 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800850 if (kVerifyFlags & kVerifyThis) {
851 VerifyObject(this);
852 }
Ian Rogers13735952014-10-08 12:43:28 -0700853 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogers228602f2014-07-10 02:07:54 -0700854 Atomic<int64_t>* atomic_addr = reinterpret_cast<Atomic<int64_t>*>(raw_addr);
855 return atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_value, new_value);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800856}
857
Hans Boehmd8434432014-07-11 09:56:07 -0700858template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
859inline bool Object::CasFieldStrongSequentiallyConsistent64(MemberOffset field_offset,
860 int64_t old_value, int64_t new_value) {
861 if (kCheckTransaction) {
862 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
863 }
864 if (kTransactionActive) {
865 Runtime::Current()->RecordWriteField64(this, field_offset, old_value, true);
866 }
867 if (kVerifyFlags & kVerifyThis) {
868 VerifyObject(this);
869 }
Ian Rogers13735952014-10-08 12:43:28 -0700870 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Hans Boehmd8434432014-07-11 09:56:07 -0700871 Atomic<int64_t>* atomic_addr = reinterpret_cast<Atomic<int64_t>*>(raw_addr);
872 return atomic_addr->CompareExchangeStrongSequentiallyConsistent(old_value, new_value);
873}
874
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700875template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption,
876 bool kIsVolatile>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700877inline T* Object::GetFieldObject(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800878 if (kVerifyFlags & kVerifyThis) {
879 VerifyObject(this);
880 }
Ian Rogers13735952014-10-08 12:43:28 -0700881 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogersef7d42f2014-01-06 12:55:46 -0800882 HeapReference<T>* objref_addr = reinterpret_cast<HeapReference<T>*>(raw_addr);
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700883 T* result = ReadBarrier::Barrier<T, kReadBarrierOption>(this, field_offset, objref_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700884 if (kIsVolatile) {
Hans Boehm30359612014-05-21 17:46:23 -0700885 // TODO: Refactor to use a SequentiallyConsistent load instead.
886 QuasiAtomic::ThreadFenceAcquire(); // Ensure visibility of operations preceding store.
Ian Rogersef7d42f2014-01-06 12:55:46 -0800887 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800888 if (kVerifyFlags & kVerifyReads) {
889 VerifyObject(result);
890 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800891 return result;
892}
893
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700894template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700895inline T* Object::GetFieldObjectVolatile(MemberOffset field_offset) {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700896 return GetFieldObject<T, kVerifyFlags, kReadBarrierOption, true>(field_offset);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700897}
898
899template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
900 bool kIsVolatile>
901inline void Object::SetFieldObjectWithoutWriteBarrier(MemberOffset field_offset,
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700902 ObjPtr<Object> new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100903 if (kCheckTransaction) {
904 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
905 }
906 if (kTransactionActive) {
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700907 ObjPtr<Object> obj;
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700908 if (kIsVolatile) {
909 obj = GetFieldObjectVolatile<Object>(field_offset);
910 } else {
911 obj = GetFieldObject<Object>(field_offset);
912 }
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700913 Runtime::Current()->RecordWriteFieldReference(this, field_offset, obj.Ptr(), true);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100914 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800915 if (kVerifyFlags & kVerifyThis) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800916 VerifyObject(this);
917 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800918 if (kVerifyFlags & kVerifyWrites) {
919 VerifyObject(new_value);
920 }
Ian Rogers13735952014-10-08 12:43:28 -0700921 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogersef7d42f2014-01-06 12:55:46 -0800922 HeapReference<Object>* objref_addr = reinterpret_cast<HeapReference<Object>*>(raw_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700923 if (kIsVolatile) {
Hans Boehm30359612014-05-21 17:46:23 -0700924 // TODO: Refactor to use a SequentiallyConsistent store instead.
925 QuasiAtomic::ThreadFenceRelease(); // Ensure that prior accesses are visible before store.
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700926 objref_addr->Assign(new_value.Ptr());
Hans Boehm30359612014-05-21 17:46:23 -0700927 QuasiAtomic::ThreadFenceSequentiallyConsistent();
928 // Ensure this store occurs before any volatile loads.
Ian Rogersef7d42f2014-01-06 12:55:46 -0800929 } else {
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700930 objref_addr->Assign(new_value.Ptr());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800931 }
932}
933
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700934template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
935 bool kIsVolatile>
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700936inline void Object::SetFieldObject(MemberOffset field_offset, ObjPtr<Object> new_value) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700937 SetFieldObjectWithoutWriteBarrier<kTransactionActive, kCheckTransaction, kVerifyFlags,
938 kIsVolatile>(field_offset, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800939 if (new_value != nullptr) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800940 Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700941 // TODO: Check field assignment could theoretically cause thread suspension, TODO: fix this.
942 CheckFieldAssignment(field_offset, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800943 }
944}
945
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700946template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700947inline void Object::SetFieldObjectVolatile(MemberOffset field_offset, ObjPtr<Object> new_value) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700948 SetFieldObject<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
949 new_value);
950}
951
Mathieu Chartier4e305412014-02-19 10:54:44 -0800952template <VerifyObjectFlags kVerifyFlags>
953inline HeapReference<Object>* Object::GetFieldObjectReferenceAddr(MemberOffset field_offset) {
954 if (kVerifyFlags & kVerifyThis) {
955 VerifyObject(this);
956 }
Ian Rogers13735952014-10-08 12:43:28 -0700957 return reinterpret_cast<HeapReference<Object>*>(reinterpret_cast<uint8_t*>(this) +
Mathieu Chartier4e305412014-02-19 10:54:44 -0800958 field_offset.Int32Value());
959}
960
961template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -0700962inline bool Object::CasFieldWeakSequentiallyConsistentObject(MemberOffset field_offset,
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700963 ObjPtr<Object> old_value,
964 ObjPtr<Object> new_value) {
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800965 bool success = CasFieldWeakSequentiallyConsistentObjectWithoutWriteBarrier<
966 kTransactionActive, kCheckTransaction, kVerifyFlags>(field_offset, old_value, new_value);
967 if (success) {
968 Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
969 }
970 return success;
971}
972
973template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
974inline bool Object::CasFieldWeakSequentiallyConsistentObjectWithoutWriteBarrier(
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700975 MemberOffset field_offset,
976 ObjPtr<Object> old_value,
977 ObjPtr<Object> new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100978 if (kCheckTransaction) {
979 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
980 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800981 if (kVerifyFlags & kVerifyThis) {
982 VerifyObject(this);
983 }
984 if (kVerifyFlags & kVerifyWrites) {
985 VerifyObject(new_value);
986 }
987 if (kVerifyFlags & kVerifyReads) {
988 VerifyObject(old_value);
989 }
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100990 if (kTransactionActive) {
991 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
992 }
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700993 HeapReference<Object> old_ref(HeapReference<Object>::FromObjPtr(old_value));
994 HeapReference<Object> new_ref(HeapReference<Object>::FromObjPtr(new_value));
Ian Rogers13735952014-10-08 12:43:28 -0700995 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogers228602f2014-07-10 02:07:54 -0700996 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
997
998 bool success = atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_ref.reference_,
999 new_ref.reference_);
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -08001000 return success;
1001}
Ian Rogers228602f2014-07-10 02:07:54 -07001002
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -08001003template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
1004inline bool Object::CasFieldStrongSequentiallyConsistentObject(MemberOffset field_offset,
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001005 ObjPtr<Object> old_value,
1006 ObjPtr<Object> new_value) {
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -08001007 bool success = CasFieldStrongSequentiallyConsistentObjectWithoutWriteBarrier<
1008 kTransactionActive, kCheckTransaction, kVerifyFlags>(field_offset, old_value, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -08001009 if (success) {
1010 Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
1011 }
1012 return success;
1013}
1014
Hans Boehmd8434432014-07-11 09:56:07 -07001015template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -08001016inline bool Object::CasFieldStrongSequentiallyConsistentObjectWithoutWriteBarrier(
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001017 MemberOffset field_offset,
1018 ObjPtr<Object> old_value,
1019 ObjPtr<Object> new_value) {
Hans Boehmd8434432014-07-11 09:56:07 -07001020 if (kCheckTransaction) {
1021 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
1022 }
1023 if (kVerifyFlags & kVerifyThis) {
1024 VerifyObject(this);
1025 }
1026 if (kVerifyFlags & kVerifyWrites) {
1027 VerifyObject(new_value);
1028 }
1029 if (kVerifyFlags & kVerifyReads) {
1030 VerifyObject(old_value);
1031 }
1032 if (kTransactionActive) {
1033 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
1034 }
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001035 HeapReference<Object> old_ref(HeapReference<Object>::FromObjPtr(old_value));
1036 HeapReference<Object> new_ref(HeapReference<Object>::FromObjPtr(new_value));
Ian Rogers13735952014-10-08 12:43:28 -07001037 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Hans Boehmd8434432014-07-11 09:56:07 -07001038 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
1039
1040 bool success = atomic_addr->CompareExchangeStrongSequentiallyConsistent(old_ref.reference_,
1041 new_ref.reference_);
Hans Boehmd8434432014-07-11 09:56:07 -07001042 return success;
1043}
1044
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -07001045template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
1046inline bool Object::CasFieldWeakRelaxedObjectWithoutWriteBarrier(
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001047 MemberOffset field_offset,
1048 ObjPtr<Object> old_value,
1049 ObjPtr<Object> new_value) {
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -07001050 if (kCheckTransaction) {
1051 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
1052 }
1053 if (kVerifyFlags & kVerifyThis) {
1054 VerifyObject(this);
1055 }
1056 if (kVerifyFlags & kVerifyWrites) {
1057 VerifyObject(new_value);
1058 }
1059 if (kVerifyFlags & kVerifyReads) {
1060 VerifyObject(old_value);
1061 }
1062 if (kTransactionActive) {
1063 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
1064 }
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001065 HeapReference<Object> old_ref(HeapReference<Object>::FromObjPtr(old_value));
1066 HeapReference<Object> new_ref(HeapReference<Object>::FromObjPtr(new_value));
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -07001067 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
1068 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
1069
1070 bool success = atomic_addr->CompareExchangeWeakRelaxed(old_ref.reference_,
1071 new_ref.reference_);
1072 return success;
1073}
1074
1075template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
1076inline bool Object::CasFieldStrongRelaxedObjectWithoutWriteBarrier(
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001077 MemberOffset field_offset,
1078 ObjPtr<Object> old_value,
1079 ObjPtr<Object> new_value) {
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -07001080 if (kCheckTransaction) {
1081 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
1082 }
1083 if (kVerifyFlags & kVerifyThis) {
1084 VerifyObject(this);
1085 }
1086 if (kVerifyFlags & kVerifyWrites) {
1087 VerifyObject(new_value);
1088 }
1089 if (kVerifyFlags & kVerifyReads) {
1090 VerifyObject(old_value);
1091 }
1092 if (kTransactionActive) {
1093 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
1094 }
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001095 HeapReference<Object> old_ref(HeapReference<Object>::FromObjPtr(old_value));
1096 HeapReference<Object> new_ref(HeapReference<Object>::FromObjPtr(new_value));
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -07001097 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
1098 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
1099
1100 bool success = atomic_addr->CompareExchangeStrongRelaxed(old_ref.reference_,
1101 new_ref.reference_);
1102 return success;
1103}
1104
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001105template<bool kIsStatic,
1106 VerifyObjectFlags kVerifyFlags,
1107 ReadBarrierOption kReadBarrierOption,
1108 typename Visitor>
Mathieu Chartier407f7022014-02-18 14:37:05 -08001109inline void Object::VisitFieldsReferences(uint32_t ref_offsets, const Visitor& visitor) {
Ian Rogerscdc1aaf2014-10-09 13:21:38 -07001110 if (!kIsStatic && (ref_offsets != mirror::Class::kClassWalkSuper)) {
1111 // Instance fields and not the slow-path.
Ian Rogerscdc1aaf2014-10-09 13:21:38 -07001112 uint32_t field_offset = mirror::kObjectHeaderSize;
Mathieu Chartier407f7022014-02-18 14:37:05 -08001113 while (ref_offsets != 0) {
Ian Rogerscdc1aaf2014-10-09 13:21:38 -07001114 if ((ref_offsets & 1) != 0) {
1115 visitor(this, MemberOffset(field_offset), kIsStatic);
1116 }
1117 ref_offsets >>= 1;
1118 field_offset += sizeof(mirror::HeapReference<mirror::Object>);
Mathieu Chartier407f7022014-02-18 14:37:05 -08001119 }
1120 } else {
Mingyao Yangfaff0f02014-09-10 12:03:22 -07001121 // There is no reference offset bitmap. In the non-static case, walk up the class
Mathieu Chartier407f7022014-02-18 14:37:05 -08001122 // inheritance hierarchy and find reference offsets the hard way. In the static case, just
1123 // consider this class.
Mathieu Chartier31e88222016-10-14 18:43:19 -07001124 for (ObjPtr<Class> klass = kIsStatic
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001125 ? AsClass<kVerifyFlags, kReadBarrierOption>()
1126 : GetClass<kVerifyFlags, kReadBarrierOption>();
1127 klass != nullptr;
1128 klass = kIsStatic ? nullptr : klass->GetSuperClass<kVerifyFlags, kReadBarrierOption>()) {
1129 const size_t num_reference_fields =
Mathieu Chartier407f7022014-02-18 14:37:05 -08001130 kIsStatic ? klass->NumReferenceStaticFields() : klass->NumReferenceInstanceFields();
Vladimir Marko76649e82014-11-10 18:32:59 +00001131 if (num_reference_fields == 0u) {
1132 continue;
1133 }
Mathieu Chartiere401d142015-04-22 13:56:20 -07001134 // Presumably GC can happen when we are cross compiling, it should not cause performance
1135 // problems to do pointer size logic.
Vladimir Marko76649e82014-11-10 18:32:59 +00001136 MemberOffset field_offset = kIsStatic
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001137 ? klass->GetFirstReferenceStaticFieldOffset<kVerifyFlags, kReadBarrierOption>(
Mathieu Chartiere401d142015-04-22 13:56:20 -07001138 Runtime::Current()->GetClassLinker()->GetImagePointerSize())
Hiroshi Yamauchi5496f692016-02-17 13:29:59 -08001139 : klass->GetFirstReferenceInstanceFieldOffset<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartier059ef3d2015-08-18 13:54:21 -07001140 for (size_t i = 0u; i < num_reference_fields; ++i) {
Mathieu Chartier407f7022014-02-18 14:37:05 -08001141 // TODO: Do a simpler check?
Mathieu Chartier059ef3d2015-08-18 13:54:21 -07001142 if (field_offset.Uint32Value() != ClassOffset().Uint32Value()) {
Mathieu Chartier52e4b432014-06-10 11:22:31 -07001143 visitor(this, field_offset, kIsStatic);
Mathieu Chartier407f7022014-02-18 14:37:05 -08001144 }
Vladimir Marko76649e82014-11-10 18:32:59 +00001145 field_offset = MemberOffset(field_offset.Uint32Value() +
1146 sizeof(mirror::HeapReference<mirror::Object>));
Mathieu Chartier407f7022014-02-18 14:37:05 -08001147 }
1148 }
1149 }
1150}
1151
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001152template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption, typename Visitor>
Mathieu Chartier31e88222016-10-14 18:43:19 -07001153inline void Object::VisitInstanceFieldsReferences(ObjPtr<Class> klass, const Visitor& visitor) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001154 VisitFieldsReferences<false, kVerifyFlags, kReadBarrierOption>(
1155 klass->GetReferenceInstanceOffsets<kVerifyFlags>(), visitor);
Mathieu Chartier407f7022014-02-18 14:37:05 -08001156}
1157
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001158template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption, typename Visitor>
Mathieu Chartier31e88222016-10-14 18:43:19 -07001159inline void Object::VisitStaticFieldsReferences(ObjPtr<Class> klass, const Visitor& visitor) {
Mingyao Yang98d1cc82014-05-15 17:02:16 -07001160 DCHECK(!klass->IsTemp());
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001161 klass->VisitFieldsReferences<true, kVerifyFlags, kReadBarrierOption>(0, visitor);
Mathieu Chartier407f7022014-02-18 14:37:05 -08001162}
1163
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001164template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere4275c02015-08-06 15:34:15 -07001165inline bool Object::IsClassLoader() {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001166 return GetClass<kVerifyFlags, kReadBarrierOption>()->IsClassLoaderClass();
Mathieu Chartiere4275c02015-08-06 15:34:15 -07001167}
1168
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001169template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere4275c02015-08-06 15:34:15 -07001170inline mirror::ClassLoader* Object::AsClassLoader() {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001171 DCHECK((IsClassLoader<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartiere4275c02015-08-06 15:34:15 -07001172 return down_cast<mirror::ClassLoader*>(this);
1173}
1174
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001175template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Vladimir Marko05792b92015-08-03 11:56:49 +01001176inline bool Object::IsDexCache() {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001177 return GetClass<kVerifyFlags, kReadBarrierOption>()->IsDexCacheClass();
Vladimir Marko05792b92015-08-03 11:56:49 +01001178}
1179
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001180template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Vladimir Marko05792b92015-08-03 11:56:49 +01001181inline mirror::DexCache* Object::AsDexCache() {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001182 DCHECK((IsDexCache<kVerifyFlags, kReadBarrierOption>()));
Vladimir Marko05792b92015-08-03 11:56:49 +01001183 return down_cast<mirror::DexCache*>(this);
1184}
1185
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001186template <bool kVisitNativeRoots,
1187 VerifyObjectFlags kVerifyFlags,
1188 ReadBarrierOption kReadBarrierOption,
1189 typename Visitor,
1190 typename JavaLangRefVisitor>
Mathieu Chartier407f7022014-02-18 14:37:05 -08001191inline void Object::VisitReferences(const Visitor& visitor,
1192 const JavaLangRefVisitor& ref_visitor) {
Mathieu Chartier31e88222016-10-14 18:43:19 -07001193 ObjPtr<Class> klass = GetClass<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartier059ef3d2015-08-18 13:54:21 -07001194 visitor(this, ClassOffset(), false);
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001195 const uint32_t class_flags = klass->GetClassFlags<kVerifyNone>();
1196 if (LIKELY(class_flags == kClassFlagNormal)) {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001197 DCHECK((!klass->IsVariableSize<kVerifyFlags, kReadBarrierOption>()));
1198 VisitInstanceFieldsReferences<kVerifyFlags, kReadBarrierOption>(klass, visitor);
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001199 DCHECK((!klass->IsClassClass<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartier66c2d2d2015-08-25 14:32:32 -07001200 DCHECK(!klass->IsStringClass());
1201 DCHECK(!klass->IsClassLoaderClass());
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001202 DCHECK((!klass->IsArrayClass<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001203 } else {
1204 if ((class_flags & kClassFlagNoReferenceFields) == 0) {
1205 DCHECK(!klass->IsStringClass());
1206 if (class_flags == kClassFlagClass) {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001207 DCHECK((klass->IsClassClass<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartier31e88222016-10-14 18:43:19 -07001208 ObjPtr<Class> as_klass = AsClass<kVerifyNone, kReadBarrierOption>();
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001209 as_klass->VisitReferences<kVisitNativeRoots, kVerifyFlags, kReadBarrierOption>(klass,
1210 visitor);
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001211 } else if (class_flags == kClassFlagObjectArray) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001212 DCHECK((klass->IsObjectArrayClass<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001213 AsObjectArray<mirror::Object, kVerifyNone, kReadBarrierOption>()->VisitReferences(visitor);
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001214 } else if ((class_flags & kClassFlagReference) != 0) {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001215 VisitInstanceFieldsReferences<kVerifyFlags, kReadBarrierOption>(klass, visitor);
1216 ref_visitor(klass, AsReference<kVerifyFlags, kReadBarrierOption>());
Vladimir Marko05792b92015-08-03 11:56:49 +01001217 } else if (class_flags == kClassFlagDexCache) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001218 mirror::DexCache* const dex_cache = AsDexCache<kVerifyFlags, kReadBarrierOption>();
1219 dex_cache->VisitReferences<kVisitNativeRoots,
1220 kVerifyFlags,
1221 kReadBarrierOption>(klass, visitor);
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001222 } else {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001223 mirror::ClassLoader* const class_loader = AsClassLoader<kVerifyFlags, kReadBarrierOption>();
1224 class_loader->VisitReferences<kVisitNativeRoots,
1225 kVerifyFlags,
1226 kReadBarrierOption>(klass, visitor);
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001227 }
1228 } else if (kIsDebugBuild) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001229 CHECK((!klass->IsClassClass<kVerifyFlags, kReadBarrierOption>()));
1230 CHECK((!klass->IsObjectArrayClass<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001231 // String still has instance fields for reflection purposes but these don't exist in
1232 // actual string instances.
1233 if (!klass->IsStringClass()) {
1234 size_t total_reference_instance_fields = 0;
Mathieu Chartier31e88222016-10-14 18:43:19 -07001235 ObjPtr<Class> super_class = klass;
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001236 do {
1237 total_reference_instance_fields += super_class->NumReferenceInstanceFields();
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001238 super_class = super_class->GetSuperClass<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001239 } while (super_class != nullptr);
1240 // The only reference field should be the object's class. This field is handled at the
1241 // beginning of the function.
1242 CHECK_EQ(total_reference_instance_fields, 1u);
1243 }
Mathieu Chartier407f7022014-02-18 14:37:05 -08001244 }
1245 }
1246}
Ian Rogers2dd0e2c2013-01-24 12:42:14 -08001247} // namespace mirror
1248} // namespace art
1249
Brian Carlstromfc0e3212013-07-17 14:40:12 -07001250#endif // ART_RUNTIME_MIRROR_OBJECT_INL_H_