blob: 8e591e4434fc6371e73b0dcb53d2e814dae74959 [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
190 // mips/mips64
191 LOG(FATAL) << "Unreachable";
192 UNREACHABLE();
193 UNUSED(fake_address_dependency);
194#endif
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700195}
196
197inline uint32_t Object::GetReadBarrierState() {
Andreas Gampe90b936d2017-01-31 08:58:55 -0800198 if (!kUseBakerReadBarrier) {
199 LOG(FATAL) << "Unreachable";
200 UNREACHABLE();
201 }
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700202 DCHECK(kUseBakerReadBarrier);
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700203 LockWord lw(GetField<uint32_t, /*kIsVolatile*/false>(OFFSET_OF_OBJECT_MEMBER(Object, monitor_)));
204 uint32_t rb_state = lw.ReadBarrierState();
205 DCHECK(ReadBarrier::IsValidReadBarrierState(rb_state)) << rb_state;
206 return rb_state;
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800207}
208
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700209inline uint32_t Object::GetReadBarrierStateAcquire() {
Andreas Gampe90b936d2017-01-31 08:58:55 -0800210 if (!kUseBakerReadBarrier) {
211 LOG(FATAL) << "Unreachable";
212 UNREACHABLE();
213 }
Mathieu Chartierc381c362016-08-23 13:27:53 -0700214 LockWord lw(GetFieldAcquire<uint32_t>(OFFSET_OF_OBJECT_MEMBER(Object, monitor_)));
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700215 uint32_t rb_state = lw.ReadBarrierState();
216 DCHECK(ReadBarrier::IsValidReadBarrierState(rb_state)) << rb_state;
217 return rb_state;
Mathieu Chartierc381c362016-08-23 13:27:53 -0700218}
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) {
Andreas Gampe90b936d2017-01-31 08:58:55 -0800230 if (!kUseBakerReadBarrier) {
231 LOG(FATAL) << "Unreachable";
232 UNREACHABLE();
233 }
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700234 DCHECK(ReadBarrier::IsValidReadBarrierState(rb_state)) << rb_state;
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700235 LockWord lw = GetLockWord(false);
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700236 lw.SetReadBarrierState(rb_state);
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700237 SetLockWord(lw, false);
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800238}
239
Hiroshi Yamauchied70b4a2015-11-17 17:52:15 -0800240template<bool kCasRelease>
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700241inline bool Object::AtomicSetReadBarrierState(uint32_t expected_rb_state, uint32_t rb_state) {
Andreas Gampe90b936d2017-01-31 08:58:55 -0800242 if (!kUseBakerReadBarrier) {
243 LOG(FATAL) << "Unreachable";
244 UNREACHABLE();
245 }
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700246 DCHECK(ReadBarrier::IsValidReadBarrierState(expected_rb_state)) << expected_rb_state;
247 DCHECK(ReadBarrier::IsValidReadBarrierState(rb_state)) << rb_state;
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700248 LockWord expected_lw;
249 LockWord new_lw;
250 do {
251 LockWord lw = GetLockWord(false);
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700252 if (UNLIKELY(lw.ReadBarrierState() != expected_rb_state)) {
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700253 // Lost the race.
254 return false;
255 }
256 expected_lw = lw;
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700257 expected_lw.SetReadBarrierState(expected_rb_state);
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700258 new_lw = lw;
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700259 new_lw.SetReadBarrierState(rb_state);
Hiroshi Yamauchied70b4a2015-11-17 17:52:15 -0800260 // ConcurrentCopying::ProcessMarkStackRef uses this with kCasRelease == true.
261 // If kCasRelease == true, use a CAS release so that when GC updates all the fields of
262 // an object and then changes the object from gray to black, the field updates (stores) will be
263 // visible (won't be reordered after this CAS.)
264 } while (!(kCasRelease ?
265 CasLockWordWeakRelease(expected_lw, new_lw) :
266 CasLockWordWeakRelaxed(expected_lw, new_lw)));
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700267 return true;
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700268}
269
Mathieu Chartier36a270a2016-07-28 18:08:51 -0700270inline bool Object::AtomicSetMarkBit(uint32_t expected_mark_bit, uint32_t mark_bit) {
271 LockWord expected_lw;
272 LockWord new_lw;
273 do {
274 LockWord lw = GetLockWord(false);
275 if (UNLIKELY(lw.MarkBitState() != expected_mark_bit)) {
276 // Lost the race.
277 return false;
278 }
279 expected_lw = lw;
280 new_lw = lw;
281 new_lw.SetMarkBitState(mark_bit);
282 // Since this is only set from the mutator, we can use the non release Cas.
283 } while (!CasLockWordWeakRelaxed(expected_lw, new_lw));
284 return true;
285}
286
287
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700288inline void Object::AssertReadBarrierState() const {
289 CHECK(kUseBakerReadBarrier);
290 Object* obj = const_cast<Object*>(this);
291 DCHECK(obj->GetReadBarrierState() == ReadBarrier::WhiteState())
292 << "Bad Baker pointer: obj=" << reinterpret_cast<void*>(obj)
293 << " rb_state" << reinterpret_cast<void*>(obj->GetReadBarrierState());
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800294}
295
Mathieu Chartier4e305412014-02-19 10:54:44 -0800296template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartieref41db72016-10-25 15:08:01 -0700297inline bool Object::VerifierInstanceOf(ObjPtr<Class> klass) {
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700298 DCHECK(klass != nullptr);
299 DCHECK(GetClass<kVerifyFlags>() != nullptr);
Jeff Haoa3faaf42013-09-03 19:07:00 -0700300 return klass->IsInterface() || InstanceOf(klass);
301}
302
Mathieu Chartier4e305412014-02-19 10:54:44 -0800303template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiera59d9b22016-09-26 18:13:17 -0700304inline bool Object::InstanceOf(ObjPtr<Class> klass) {
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700305 DCHECK(klass != nullptr);
306 DCHECK(GetClass<kVerifyNone>() != nullptr);
Mathieu Chartier4e305412014-02-19 10:54:44 -0800307 return klass->IsAssignableFrom(GetClass<kVerifyFlags>());
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800308}
309
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700310template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800311inline bool Object::IsClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700312 Class* java_lang_Class = GetClass<kVerifyFlags, kReadBarrierOption>()->
313 template GetClass<kVerifyFlags, kReadBarrierOption>();
314 return GetClass<static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis),
315 kReadBarrierOption>() == java_lang_Class;
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800316}
317
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700318template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800319inline Class* Object::AsClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700320 DCHECK((IsClass<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800321 return down_cast<Class*>(this);
322}
323
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800324template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800325inline bool Object::IsObjectArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800326 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800327 return IsArrayInstance<kVerifyFlags, kReadBarrierOption>() &&
328 !GetClass<kNewFlags, kReadBarrierOption>()->
329 template GetComponentType<kNewFlags, kReadBarrierOption>()->IsPrimitive();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800330}
331
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800332template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800333inline ObjectArray<T>* Object::AsObjectArray() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800334 DCHECK((IsObjectArray<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800335 return down_cast<ObjectArray<T>*>(this);
336}
337
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700338template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800339inline bool Object::IsArrayInstance() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700340 return GetClass<kVerifyFlags, kReadBarrierOption>()->
341 template IsArrayClass<kVerifyFlags, kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800342}
343
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800344template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800345inline bool Object::IsReferenceInstance() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800346 return GetClass<kVerifyFlags, kReadBarrierOption>()->IsTypeOfReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800347}
348
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800349template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700350inline Reference* Object::AsReference() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800351 DCHECK((IsReferenceInstance<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700352 return down_cast<Reference*>(this);
353}
354
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700355template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers05f30572013-02-20 12:13:11 -0800356inline Array* Object::AsArray() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700357 DCHECK((IsArrayInstance<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800358 return down_cast<Array*>(this);
359}
360
Mathieu Chartier4e305412014-02-19 10:54:44 -0800361template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800362inline BooleanArray* Object::AsBooleanArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800363 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
364 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
365 DCHECK(GetClass<kNewFlags>()->GetComponentType()->IsPrimitiveBoolean());
Ian Rogers05f30572013-02-20 12:13:11 -0800366 return down_cast<BooleanArray*>(this);
367}
368
Mathieu Chartier4e305412014-02-19 10:54:44 -0800369template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800370inline ByteArray* Object::AsByteArray() {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700371 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Mathieu Chartier4e305412014-02-19 10:54:44 -0800372 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
373 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveByte());
Ian Rogers05f30572013-02-20 12:13:11 -0800374 return down_cast<ByteArray*>(this);
375}
376
Mathieu Chartier4e305412014-02-19 10:54:44 -0800377template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800378inline ByteArray* Object::AsByteSizedArray() {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700379 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Mathieu Chartier4e305412014-02-19 10:54:44 -0800380 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
381 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveByte() ||
382 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveBoolean());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800383 return down_cast<ByteArray*>(this);
384}
385
Mathieu Chartier4e305412014-02-19 10:54:44 -0800386template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800387inline CharArray* Object::AsCharArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800388 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
389 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
390 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveChar());
Ian Rogers05f30572013-02-20 12:13:11 -0800391 return down_cast<CharArray*>(this);
392}
393
Mathieu Chartier4e305412014-02-19 10:54:44 -0800394template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800395inline ShortArray* Object::AsShortArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800396 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
397 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
398 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveShort());
Ian Rogers05f30572013-02-20 12:13:11 -0800399 return down_cast<ShortArray*>(this);
400}
401
Mathieu Chartier4e305412014-02-19 10:54:44 -0800402template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800403inline ShortArray* Object::AsShortSizedArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800404 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
405 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
406 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveShort() ||
407 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveChar());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800408 return down_cast<ShortArray*>(this);
409}
410
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800411template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700412inline bool Object::IsIntArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800413 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Mathieu Chartier31e88222016-10-14 18:43:19 -0700414 ObjPtr<Class> klass = GetClass<kVerifyFlags, kReadBarrierOption>();
415 ObjPtr<Class> component_type = klass->GetComponentType<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartiere401d142015-04-22 13:56:20 -0700416 return component_type != nullptr && component_type->template IsPrimitiveInt<kNewFlags>();
417}
418
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800419template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700420inline IntArray* Object::AsIntArray() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800421 DCHECK((IsIntArray<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800422 return down_cast<IntArray*>(this);
423}
424
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800425template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700426inline bool Object::IsLongArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800427 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Mathieu Chartier31e88222016-10-14 18:43:19 -0700428 ObjPtr<Class> klass = GetClass<kVerifyFlags, kReadBarrierOption>();
429 ObjPtr<Class> component_type = klass->GetComponentType<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartiere401d142015-04-22 13:56:20 -0700430 return component_type != nullptr && component_type->template IsPrimitiveLong<kNewFlags>();
431}
432
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800433template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700434inline LongArray* Object::AsLongArray() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800435 DCHECK((IsLongArray<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800436 return down_cast<LongArray*>(this);
437}
438
Mathieu Chartier4e305412014-02-19 10:54:44 -0800439template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700440inline bool Object::IsFloatArray() {
441 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
442 auto* component_type = GetClass<kVerifyFlags>()->GetComponentType();
443 return component_type != nullptr && component_type->template IsPrimitiveFloat<kNewFlags>();
444}
445
446template<VerifyObjectFlags kVerifyFlags>
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100447inline FloatArray* Object::AsFloatArray() {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700448 DCHECK(IsFloatArray<kVerifyFlags>());
Mathieu Chartier4e305412014-02-19 10:54:44 -0800449 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
450 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
451 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveFloat());
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100452 return down_cast<FloatArray*>(this);
453}
454
Mathieu Chartier4e305412014-02-19 10:54:44 -0800455template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700456inline bool Object::IsDoubleArray() {
457 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
458 auto* component_type = GetClass<kVerifyFlags>()->GetComponentType();
459 return component_type != nullptr && component_type->template IsPrimitiveDouble<kNewFlags>();
460}
461
462template<VerifyObjectFlags kVerifyFlags>
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100463inline DoubleArray* Object::AsDoubleArray() {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700464 DCHECK(IsDoubleArray<kVerifyFlags>());
Mathieu Chartier4e305412014-02-19 10:54:44 -0800465 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
466 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
467 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveDouble());
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100468 return down_cast<DoubleArray*>(this);
469}
470
Jeff Hao848f70a2014-01-15 13:49:50 -0800471template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
472inline bool Object::IsString() {
473 return GetClass<kVerifyFlags, kReadBarrierOption>()->IsStringClass();
474}
475
476template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers05f30572013-02-20 12:13:11 -0800477inline String* Object::AsString() {
Jeff Hao848f70a2014-01-15 13:49:50 -0800478 DCHECK((IsString<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800479 return down_cast<String*>(this);
480}
481
Mathieu Chartier4e305412014-02-19 10:54:44 -0800482template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800483inline Throwable* Object::AsThrowable() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800484 DCHECK(GetClass<kVerifyFlags>()->IsThrowableClass());
Ian Rogers05f30572013-02-20 12:13:11 -0800485 return down_cast<Throwable*>(this);
486}
487
Mathieu Chartier4e305412014-02-19 10:54:44 -0800488template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800489inline bool Object::IsWeakReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800490 return GetClass<kVerifyFlags>()->IsWeakReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800491}
492
Mathieu Chartier4e305412014-02-19 10:54:44 -0800493template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800494inline bool Object::IsSoftReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800495 return GetClass<kVerifyFlags>()->IsSoftReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800496}
497
Mathieu Chartier4e305412014-02-19 10:54:44 -0800498template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800499inline bool Object::IsFinalizerReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800500 return GetClass<kVerifyFlags>()->IsFinalizerReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800501}
502
Mathieu Chartier4e305412014-02-19 10:54:44 -0800503template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700504inline FinalizerReference* Object::AsFinalizerReference() {
505 DCHECK(IsFinalizerReferenceInstance<kVerifyFlags>());
506 return down_cast<FinalizerReference*>(this);
507}
508
509template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800510inline bool Object::IsPhantomReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800511 return GetClass<kVerifyFlags>()->IsPhantomReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800512}
513
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700514template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800515inline size_t Object::SizeOf() {
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800516 size_t result;
Mathieu Chartier4e305412014-02-19 10:54:44 -0800517 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700518 if (IsArrayInstance<kVerifyFlags, kReadBarrierOption>()) {
519 result = AsArray<kNewFlags, kReadBarrierOption>()->
520 template SizeOf<kNewFlags, kReadBarrierOption>();
521 } else if (IsClass<kNewFlags, kReadBarrierOption>()) {
522 result = AsClass<kNewFlags, kReadBarrierOption>()->
523 template SizeOf<kNewFlags, kReadBarrierOption>();
Jeff Hao848f70a2014-01-15 13:49:50 -0800524 } else if (GetClass<kNewFlags, kReadBarrierOption>()->IsStringClass()) {
525 result = AsString<kNewFlags, kReadBarrierOption>()->
526 template SizeOf<kNewFlags>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800527 } else {
Hiroshi Yamauchi25023c72014-05-09 11:45:53 -0700528 result = GetClass<kNewFlags, kReadBarrierOption>()->
529 template GetObjectSize<kNewFlags, kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800530 }
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700531 DCHECK_GE(result, sizeof(Object))
David Sehr709b0702016-10-13 09:12:37 -0700532 << " class=" << Class::PrettyClass(GetClass<kNewFlags, kReadBarrierOption>());
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800533 return result;
534}
535
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700536template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
Fred Shih37f05ef2014-07-16 18:38:08 -0700537inline uint8_t Object::GetFieldBoolean(MemberOffset field_offset) {
538 if (kVerifyFlags & kVerifyThis) {
539 VerifyObject(this);
540 }
541 return GetField<uint8_t, kIsVolatile>(field_offset);
542}
543
544template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
545inline int8_t Object::GetFieldByte(MemberOffset field_offset) {
546 if (kVerifyFlags & kVerifyThis) {
547 VerifyObject(this);
548 }
549 return GetField<int8_t, kIsVolatile>(field_offset);
550}
551
552template<VerifyObjectFlags kVerifyFlags>
553inline uint8_t Object::GetFieldBooleanVolatile(MemberOffset field_offset) {
554 return GetFieldBoolean<kVerifyFlags, true>(field_offset);
555}
556
557template<VerifyObjectFlags kVerifyFlags>
558inline int8_t Object::GetFieldByteVolatile(MemberOffset field_offset) {
559 return GetFieldByte<kVerifyFlags, true>(field_offset);
560}
561
562template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
563 bool kIsVolatile>
564inline void Object::SetFieldBoolean(MemberOffset field_offset, uint8_t new_value)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700565 REQUIRES_SHARED(Locks::mutator_lock_) {
Fred Shih37f05ef2014-07-16 18:38:08 -0700566 if (kCheckTransaction) {
567 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
568 }
569 if (kTransactionActive) {
570 Runtime::Current()->RecordWriteFieldBoolean(this, field_offset,
571 GetFieldBoolean<kVerifyFlags, kIsVolatile>(field_offset),
572 kIsVolatile);
573 }
574 if (kVerifyFlags & kVerifyThis) {
575 VerifyObject(this);
576 }
577 SetField<uint8_t, kIsVolatile>(field_offset, new_value);
578}
579
580template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
581 bool kIsVolatile>
582inline void Object::SetFieldByte(MemberOffset field_offset, int8_t new_value)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700583 REQUIRES_SHARED(Locks::mutator_lock_) {
Fred Shih37f05ef2014-07-16 18:38:08 -0700584 if (kCheckTransaction) {
585 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
586 }
587 if (kTransactionActive) {
588 Runtime::Current()->RecordWriteFieldByte(this, field_offset,
589 GetFieldByte<kVerifyFlags, kIsVolatile>(field_offset),
590 kIsVolatile);
591 }
592 if (kVerifyFlags & kVerifyThis) {
593 VerifyObject(this);
594 }
595 SetField<int8_t, kIsVolatile>(field_offset, new_value);
596}
597
598template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
599inline void Object::SetFieldBooleanVolatile(MemberOffset field_offset, uint8_t new_value) {
600 return SetFieldBoolean<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
601 field_offset, new_value);
602}
603
604template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
605inline void Object::SetFieldByteVolatile(MemberOffset field_offset, int8_t new_value) {
606 return SetFieldByte<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
607 field_offset, new_value);
608}
609
610template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
611inline uint16_t Object::GetFieldChar(MemberOffset field_offset) {
612 if (kVerifyFlags & kVerifyThis) {
613 VerifyObject(this);
614 }
615 return GetField<uint16_t, kIsVolatile>(field_offset);
616}
617
618template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
619inline int16_t Object::GetFieldShort(MemberOffset field_offset) {
620 if (kVerifyFlags & kVerifyThis) {
621 VerifyObject(this);
622 }
623 return GetField<int16_t, kIsVolatile>(field_offset);
624}
625
626template<VerifyObjectFlags kVerifyFlags>
627inline uint16_t Object::GetFieldCharVolatile(MemberOffset field_offset) {
628 return GetFieldChar<kVerifyFlags, true>(field_offset);
629}
630
631template<VerifyObjectFlags kVerifyFlags>
632inline int16_t Object::GetFieldShortVolatile(MemberOffset field_offset) {
633 return GetFieldShort<kVerifyFlags, true>(field_offset);
634}
635
636template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
637 bool kIsVolatile>
638inline void Object::SetFieldChar(MemberOffset field_offset, uint16_t new_value) {
639 if (kCheckTransaction) {
640 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
641 }
642 if (kTransactionActive) {
643 Runtime::Current()->RecordWriteFieldChar(this, field_offset,
644 GetFieldChar<kVerifyFlags, kIsVolatile>(field_offset),
645 kIsVolatile);
646 }
647 if (kVerifyFlags & kVerifyThis) {
648 VerifyObject(this);
649 }
650 SetField<uint16_t, kIsVolatile>(field_offset, new_value);
651}
652
653template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
654 bool kIsVolatile>
655inline void Object::SetFieldShort(MemberOffset field_offset, int16_t new_value) {
656 if (kCheckTransaction) {
657 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
658 }
659 if (kTransactionActive) {
660 Runtime::Current()->RecordWriteFieldChar(this, field_offset,
661 GetFieldShort<kVerifyFlags, kIsVolatile>(field_offset),
662 kIsVolatile);
663 }
664 if (kVerifyFlags & kVerifyThis) {
665 VerifyObject(this);
666 }
667 SetField<int16_t, kIsVolatile>(field_offset, new_value);
668}
669
670template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
671inline void Object::SetFieldCharVolatile(MemberOffset field_offset, uint16_t new_value) {
672 return SetFieldChar<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
673 field_offset, new_value);
674}
675
676template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
677inline void Object::SetFieldShortVolatile(MemberOffset field_offset, int16_t new_value) {
678 return SetFieldShort<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
679 field_offset, new_value);
680}
681
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700682template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
683 bool kIsVolatile>
684inline void Object::SetField32(MemberOffset field_offset, int32_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100685 if (kCheckTransaction) {
686 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
687 }
688 if (kTransactionActive) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700689 Runtime::Current()->RecordWriteField32(this, field_offset,
690 GetField32<kVerifyFlags, kIsVolatile>(field_offset),
691 kIsVolatile);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100692 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800693 if (kVerifyFlags & kVerifyThis) {
Ian Rogersb122a4b2013-11-19 18:00:50 -0800694 VerifyObject(this);
695 }
Fred Shih37f05ef2014-07-16 18:38:08 -0700696 SetField<int32_t, kIsVolatile>(field_offset, new_value);
Ian Rogersb122a4b2013-11-19 18:00:50 -0800697}
698
Mathieu Chartier4e305412014-02-19 10:54:44 -0800699template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700700inline void Object::SetField32Volatile(MemberOffset field_offset, int32_t new_value) {
701 SetField32<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset, new_value);
702}
703
Hans Boehmd8434432014-07-11 09:56:07 -0700704// TODO: Pass memory_order_ and strong/weak as arguments to avoid code duplication?
705
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700706template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -0700707inline bool Object::CasFieldWeakSequentiallyConsistent32(MemberOffset field_offset,
708 int32_t old_value, int32_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100709 if (kCheckTransaction) {
710 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
711 }
712 if (kTransactionActive) {
713 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
714 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800715 if (kVerifyFlags & kVerifyThis) {
716 VerifyObject(this);
717 }
Ian Rogers13735952014-10-08 12:43:28 -0700718 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogers228602f2014-07-10 02:07:54 -0700719 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
Hans Boehm30359612014-05-21 17:46:23 -0700720
Ian Rogers228602f2014-07-10 02:07:54 -0700721 return atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_value, new_value);
Ian Rogersd9c4fc92013-10-01 19:45:43 -0700722}
723
Hans Boehmd8434432014-07-11 09:56:07 -0700724template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
725inline bool Object::CasFieldWeakRelaxed32(MemberOffset field_offset,
726 int32_t old_value, int32_t new_value) {
727 if (kCheckTransaction) {
728 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
729 }
730 if (kTransactionActive) {
731 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
732 }
733 if (kVerifyFlags & kVerifyThis) {
734 VerifyObject(this);
735 }
Ian Rogers13735952014-10-08 12:43:28 -0700736 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Hans Boehmd8434432014-07-11 09:56:07 -0700737 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
738
739 return atomic_addr->CompareExchangeWeakRelaxed(old_value, new_value);
740}
741
742template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Hans Boehmb3da36c2016-12-15 13:12:59 -0800743inline bool Object::CasFieldWeakAcquire32(MemberOffset field_offset,
744 int32_t old_value, int32_t new_value) {
745 if (kCheckTransaction) {
746 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
747 }
748 if (kTransactionActive) {
749 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
750 }
751 if (kVerifyFlags & kVerifyThis) {
752 VerifyObject(this);
753 }
754 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
755 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
756
757 return atomic_addr->CompareExchangeWeakAcquire(old_value, new_value);
758}
759
760template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -0700761inline bool Object::CasFieldWeakRelease32(MemberOffset field_offset,
762 int32_t old_value, int32_t new_value) {
763 if (kCheckTransaction) {
764 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
765 }
766 if (kTransactionActive) {
767 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
768 }
769 if (kVerifyFlags & kVerifyThis) {
770 VerifyObject(this);
771 }
772 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
773 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
774
775 return atomic_addr->CompareExchangeWeakRelease(old_value, new_value);
776}
777
778template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Hans Boehmd8434432014-07-11 09:56:07 -0700779inline bool Object::CasFieldStrongSequentiallyConsistent32(MemberOffset field_offset,
780 int32_t old_value, int32_t new_value) {
781 if (kCheckTransaction) {
782 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
783 }
784 if (kTransactionActive) {
785 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
786 }
787 if (kVerifyFlags & kVerifyThis) {
788 VerifyObject(this);
789 }
Ian Rogers13735952014-10-08 12:43:28 -0700790 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Hans Boehmd8434432014-07-11 09:56:07 -0700791 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
792
793 return atomic_addr->CompareExchangeStrongSequentiallyConsistent(old_value, new_value);
794}
795
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700796template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
797inline int64_t Object::GetField64(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800798 if (kVerifyFlags & kVerifyThis) {
799 VerifyObject(this);
800 }
Fred Shih37f05ef2014-07-16 18:38:08 -0700801 return GetField<int64_t, kIsVolatile>(field_offset);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800802}
803
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700804template<VerifyObjectFlags kVerifyFlags>
805inline int64_t Object::GetField64Volatile(MemberOffset field_offset) {
806 return GetField64<kVerifyFlags, true>(field_offset);
807}
808
809template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
810 bool kIsVolatile>
811inline void Object::SetField64(MemberOffset field_offset, int64_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100812 if (kCheckTransaction) {
813 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
814 }
815 if (kTransactionActive) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700816 Runtime::Current()->RecordWriteField64(this, field_offset,
817 GetField64<kVerifyFlags, kIsVolatile>(field_offset),
818 kIsVolatile);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100819 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800820 if (kVerifyFlags & kVerifyThis) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800821 VerifyObject(this);
822 }
Fred Shih37f05ef2014-07-16 18:38:08 -0700823 SetField<int64_t, kIsVolatile>(field_offset, new_value);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800824}
825
Mathieu Chartier4e305412014-02-19 10:54:44 -0800826template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700827inline void Object::SetField64Volatile(MemberOffset field_offset, int64_t new_value) {
828 return SetField64<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
829 new_value);
830}
831
Mathieu Chartierc381c362016-08-23 13:27:53 -0700832template<typename kSize>
833inline kSize Object::GetFieldAcquire(MemberOffset field_offset) {
834 const uint8_t* raw_addr = reinterpret_cast<const uint8_t*>(this) + field_offset.Int32Value();
835 const kSize* addr = reinterpret_cast<const kSize*>(raw_addr);
836 return reinterpret_cast<const Atomic<kSize>*>(addr)->LoadAcquire();
837}
838
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700839template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -0700840inline bool Object::CasFieldWeakSequentiallyConsistent64(MemberOffset field_offset,
841 int64_t old_value, int64_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100842 if (kCheckTransaction) {
843 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
844 }
845 if (kTransactionActive) {
846 Runtime::Current()->RecordWriteField64(this, field_offset, old_value, true);
847 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800848 if (kVerifyFlags & kVerifyThis) {
849 VerifyObject(this);
850 }
Ian Rogers13735952014-10-08 12:43:28 -0700851 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogers228602f2014-07-10 02:07:54 -0700852 Atomic<int64_t>* atomic_addr = reinterpret_cast<Atomic<int64_t>*>(raw_addr);
853 return atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_value, new_value);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800854}
855
Hans Boehmd8434432014-07-11 09:56:07 -0700856template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
857inline bool Object::CasFieldStrongSequentiallyConsistent64(MemberOffset field_offset,
858 int64_t old_value, int64_t new_value) {
859 if (kCheckTransaction) {
860 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
861 }
862 if (kTransactionActive) {
863 Runtime::Current()->RecordWriteField64(this, field_offset, old_value, true);
864 }
865 if (kVerifyFlags & kVerifyThis) {
866 VerifyObject(this);
867 }
Ian Rogers13735952014-10-08 12:43:28 -0700868 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Hans Boehmd8434432014-07-11 09:56:07 -0700869 Atomic<int64_t>* atomic_addr = reinterpret_cast<Atomic<int64_t>*>(raw_addr);
870 return atomic_addr->CompareExchangeStrongSequentiallyConsistent(old_value, new_value);
871}
872
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700873template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption,
874 bool kIsVolatile>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700875inline T* Object::GetFieldObject(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800876 if (kVerifyFlags & kVerifyThis) {
877 VerifyObject(this);
878 }
Ian Rogers13735952014-10-08 12:43:28 -0700879 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogersef7d42f2014-01-06 12:55:46 -0800880 HeapReference<T>* objref_addr = reinterpret_cast<HeapReference<T>*>(raw_addr);
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700881 T* result = ReadBarrier::Barrier<T, kReadBarrierOption>(this, field_offset, objref_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700882 if (kIsVolatile) {
Hans Boehm30359612014-05-21 17:46:23 -0700883 // TODO: Refactor to use a SequentiallyConsistent load instead.
884 QuasiAtomic::ThreadFenceAcquire(); // Ensure visibility of operations preceding store.
Ian Rogersef7d42f2014-01-06 12:55:46 -0800885 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800886 if (kVerifyFlags & kVerifyReads) {
887 VerifyObject(result);
888 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800889 return result;
890}
891
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700892template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700893inline T* Object::GetFieldObjectVolatile(MemberOffset field_offset) {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700894 return GetFieldObject<T, kVerifyFlags, kReadBarrierOption, true>(field_offset);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700895}
896
897template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
898 bool kIsVolatile>
899inline void Object::SetFieldObjectWithoutWriteBarrier(MemberOffset field_offset,
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700900 ObjPtr<Object> new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100901 if (kCheckTransaction) {
902 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
903 }
904 if (kTransactionActive) {
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700905 ObjPtr<Object> obj;
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700906 if (kIsVolatile) {
907 obj = GetFieldObjectVolatile<Object>(field_offset);
908 } else {
909 obj = GetFieldObject<Object>(field_offset);
910 }
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700911 Runtime::Current()->RecordWriteFieldReference(this, field_offset, obj.Ptr(), true);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100912 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800913 if (kVerifyFlags & kVerifyThis) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800914 VerifyObject(this);
915 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800916 if (kVerifyFlags & kVerifyWrites) {
917 VerifyObject(new_value);
918 }
Ian Rogers13735952014-10-08 12:43:28 -0700919 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogersef7d42f2014-01-06 12:55:46 -0800920 HeapReference<Object>* objref_addr = reinterpret_cast<HeapReference<Object>*>(raw_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700921 if (kIsVolatile) {
Hans Boehm30359612014-05-21 17:46:23 -0700922 // TODO: Refactor to use a SequentiallyConsistent store instead.
923 QuasiAtomic::ThreadFenceRelease(); // Ensure that prior accesses are visible before store.
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700924 objref_addr->Assign(new_value.Ptr());
Hans Boehm30359612014-05-21 17:46:23 -0700925 QuasiAtomic::ThreadFenceSequentiallyConsistent();
926 // Ensure this store occurs before any volatile loads.
Ian Rogersef7d42f2014-01-06 12:55:46 -0800927 } else {
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700928 objref_addr->Assign(new_value.Ptr());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800929 }
930}
931
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700932template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
933 bool kIsVolatile>
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700934inline void Object::SetFieldObject(MemberOffset field_offset, ObjPtr<Object> new_value) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700935 SetFieldObjectWithoutWriteBarrier<kTransactionActive, kCheckTransaction, kVerifyFlags,
936 kIsVolatile>(field_offset, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800937 if (new_value != nullptr) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800938 Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700939 // TODO: Check field assignment could theoretically cause thread suspension, TODO: fix this.
940 CheckFieldAssignment(field_offset, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800941 }
942}
943
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700944template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700945inline void Object::SetFieldObjectVolatile(MemberOffset field_offset, ObjPtr<Object> new_value) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700946 SetFieldObject<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
947 new_value);
948}
949
Mathieu Chartier4e305412014-02-19 10:54:44 -0800950template <VerifyObjectFlags kVerifyFlags>
951inline HeapReference<Object>* Object::GetFieldObjectReferenceAddr(MemberOffset field_offset) {
952 if (kVerifyFlags & kVerifyThis) {
953 VerifyObject(this);
954 }
Ian Rogers13735952014-10-08 12:43:28 -0700955 return reinterpret_cast<HeapReference<Object>*>(reinterpret_cast<uint8_t*>(this) +
Mathieu Chartier4e305412014-02-19 10:54:44 -0800956 field_offset.Int32Value());
957}
958
959template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -0700960inline bool Object::CasFieldWeakSequentiallyConsistentObject(MemberOffset field_offset,
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700961 ObjPtr<Object> old_value,
962 ObjPtr<Object> new_value) {
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800963 bool success = CasFieldWeakSequentiallyConsistentObjectWithoutWriteBarrier<
964 kTransactionActive, kCheckTransaction, kVerifyFlags>(field_offset, old_value, new_value);
965 if (success) {
966 Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
967 }
968 return success;
969}
970
971template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
972inline bool Object::CasFieldWeakSequentiallyConsistentObjectWithoutWriteBarrier(
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700973 MemberOffset field_offset,
974 ObjPtr<Object> old_value,
975 ObjPtr<Object> new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100976 if (kCheckTransaction) {
977 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
978 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800979 if (kVerifyFlags & kVerifyThis) {
980 VerifyObject(this);
981 }
982 if (kVerifyFlags & kVerifyWrites) {
983 VerifyObject(new_value);
984 }
985 if (kVerifyFlags & kVerifyReads) {
986 VerifyObject(old_value);
987 }
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100988 if (kTransactionActive) {
989 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
990 }
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700991 HeapReference<Object> old_ref(HeapReference<Object>::FromObjPtr(old_value));
992 HeapReference<Object> new_ref(HeapReference<Object>::FromObjPtr(new_value));
Ian Rogers13735952014-10-08 12:43:28 -0700993 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogers228602f2014-07-10 02:07:54 -0700994 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
995
996 bool success = atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_ref.reference_,
997 new_ref.reference_);
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800998 return success;
999}
Ian Rogers228602f2014-07-10 02:07:54 -07001000
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -08001001template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
1002inline bool Object::CasFieldStrongSequentiallyConsistentObject(MemberOffset field_offset,
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001003 ObjPtr<Object> old_value,
1004 ObjPtr<Object> new_value) {
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -08001005 bool success = CasFieldStrongSequentiallyConsistentObjectWithoutWriteBarrier<
1006 kTransactionActive, kCheckTransaction, kVerifyFlags>(field_offset, old_value, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -08001007 if (success) {
1008 Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
1009 }
1010 return success;
1011}
1012
Hans Boehmd8434432014-07-11 09:56:07 -07001013template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -08001014inline bool Object::CasFieldStrongSequentiallyConsistentObjectWithoutWriteBarrier(
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001015 MemberOffset field_offset,
1016 ObjPtr<Object> old_value,
1017 ObjPtr<Object> new_value) {
Hans Boehmd8434432014-07-11 09:56:07 -07001018 if (kCheckTransaction) {
1019 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
1020 }
1021 if (kVerifyFlags & kVerifyThis) {
1022 VerifyObject(this);
1023 }
1024 if (kVerifyFlags & kVerifyWrites) {
1025 VerifyObject(new_value);
1026 }
1027 if (kVerifyFlags & kVerifyReads) {
1028 VerifyObject(old_value);
1029 }
1030 if (kTransactionActive) {
1031 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
1032 }
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001033 HeapReference<Object> old_ref(HeapReference<Object>::FromObjPtr(old_value));
1034 HeapReference<Object> new_ref(HeapReference<Object>::FromObjPtr(new_value));
Ian Rogers13735952014-10-08 12:43:28 -07001035 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Hans Boehmd8434432014-07-11 09:56:07 -07001036 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
1037
1038 bool success = atomic_addr->CompareExchangeStrongSequentiallyConsistent(old_ref.reference_,
1039 new_ref.reference_);
Hans Boehmd8434432014-07-11 09:56:07 -07001040 return success;
1041}
1042
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -07001043template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
1044inline bool Object::CasFieldWeakRelaxedObjectWithoutWriteBarrier(
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001045 MemberOffset field_offset,
1046 ObjPtr<Object> old_value,
1047 ObjPtr<Object> new_value) {
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -07001048 if (kCheckTransaction) {
1049 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
1050 }
1051 if (kVerifyFlags & kVerifyThis) {
1052 VerifyObject(this);
1053 }
1054 if (kVerifyFlags & kVerifyWrites) {
1055 VerifyObject(new_value);
1056 }
1057 if (kVerifyFlags & kVerifyReads) {
1058 VerifyObject(old_value);
1059 }
1060 if (kTransactionActive) {
1061 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
1062 }
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001063 HeapReference<Object> old_ref(HeapReference<Object>::FromObjPtr(old_value));
1064 HeapReference<Object> new_ref(HeapReference<Object>::FromObjPtr(new_value));
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -07001065 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
1066 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
1067
1068 bool success = atomic_addr->CompareExchangeWeakRelaxed(old_ref.reference_,
1069 new_ref.reference_);
1070 return success;
1071}
1072
1073template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
1074inline bool Object::CasFieldStrongRelaxedObjectWithoutWriteBarrier(
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001075 MemberOffset field_offset,
1076 ObjPtr<Object> old_value,
1077 ObjPtr<Object> new_value) {
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -07001078 if (kCheckTransaction) {
1079 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
1080 }
1081 if (kVerifyFlags & kVerifyThis) {
1082 VerifyObject(this);
1083 }
1084 if (kVerifyFlags & kVerifyWrites) {
1085 VerifyObject(new_value);
1086 }
1087 if (kVerifyFlags & kVerifyReads) {
1088 VerifyObject(old_value);
1089 }
1090 if (kTransactionActive) {
1091 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
1092 }
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001093 HeapReference<Object> old_ref(HeapReference<Object>::FromObjPtr(old_value));
1094 HeapReference<Object> new_ref(HeapReference<Object>::FromObjPtr(new_value));
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -07001095 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
1096 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
1097
1098 bool success = atomic_addr->CompareExchangeStrongRelaxed(old_ref.reference_,
1099 new_ref.reference_);
1100 return success;
1101}
1102
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001103template<bool kIsStatic,
1104 VerifyObjectFlags kVerifyFlags,
1105 ReadBarrierOption kReadBarrierOption,
1106 typename Visitor>
Mathieu Chartier407f7022014-02-18 14:37:05 -08001107inline void Object::VisitFieldsReferences(uint32_t ref_offsets, const Visitor& visitor) {
Ian Rogerscdc1aaf2014-10-09 13:21:38 -07001108 if (!kIsStatic && (ref_offsets != mirror::Class::kClassWalkSuper)) {
1109 // Instance fields and not the slow-path.
Ian Rogerscdc1aaf2014-10-09 13:21:38 -07001110 uint32_t field_offset = mirror::kObjectHeaderSize;
Mathieu Chartier407f7022014-02-18 14:37:05 -08001111 while (ref_offsets != 0) {
Ian Rogerscdc1aaf2014-10-09 13:21:38 -07001112 if ((ref_offsets & 1) != 0) {
1113 visitor(this, MemberOffset(field_offset), kIsStatic);
1114 }
1115 ref_offsets >>= 1;
1116 field_offset += sizeof(mirror::HeapReference<mirror::Object>);
Mathieu Chartier407f7022014-02-18 14:37:05 -08001117 }
1118 } else {
Mingyao Yangfaff0f02014-09-10 12:03:22 -07001119 // There is no reference offset bitmap. In the non-static case, walk up the class
Mathieu Chartier407f7022014-02-18 14:37:05 -08001120 // inheritance hierarchy and find reference offsets the hard way. In the static case, just
1121 // consider this class.
Mathieu Chartier31e88222016-10-14 18:43:19 -07001122 for (ObjPtr<Class> klass = kIsStatic
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001123 ? AsClass<kVerifyFlags, kReadBarrierOption>()
1124 : GetClass<kVerifyFlags, kReadBarrierOption>();
1125 klass != nullptr;
1126 klass = kIsStatic ? nullptr : klass->GetSuperClass<kVerifyFlags, kReadBarrierOption>()) {
1127 const size_t num_reference_fields =
Mathieu Chartier407f7022014-02-18 14:37:05 -08001128 kIsStatic ? klass->NumReferenceStaticFields() : klass->NumReferenceInstanceFields();
Vladimir Marko76649e82014-11-10 18:32:59 +00001129 if (num_reference_fields == 0u) {
1130 continue;
1131 }
Mathieu Chartiere401d142015-04-22 13:56:20 -07001132 // Presumably GC can happen when we are cross compiling, it should not cause performance
1133 // problems to do pointer size logic.
Vladimir Marko76649e82014-11-10 18:32:59 +00001134 MemberOffset field_offset = kIsStatic
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001135 ? klass->GetFirstReferenceStaticFieldOffset<kVerifyFlags, kReadBarrierOption>(
Mathieu Chartiere401d142015-04-22 13:56:20 -07001136 Runtime::Current()->GetClassLinker()->GetImagePointerSize())
Hiroshi Yamauchi5496f692016-02-17 13:29:59 -08001137 : klass->GetFirstReferenceInstanceFieldOffset<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartier059ef3d2015-08-18 13:54:21 -07001138 for (size_t i = 0u; i < num_reference_fields; ++i) {
Mathieu Chartier407f7022014-02-18 14:37:05 -08001139 // TODO: Do a simpler check?
Mathieu Chartier059ef3d2015-08-18 13:54:21 -07001140 if (field_offset.Uint32Value() != ClassOffset().Uint32Value()) {
Mathieu Chartier52e4b432014-06-10 11:22:31 -07001141 visitor(this, field_offset, kIsStatic);
Mathieu Chartier407f7022014-02-18 14:37:05 -08001142 }
Vladimir Marko76649e82014-11-10 18:32:59 +00001143 field_offset = MemberOffset(field_offset.Uint32Value() +
1144 sizeof(mirror::HeapReference<mirror::Object>));
Mathieu Chartier407f7022014-02-18 14:37:05 -08001145 }
1146 }
1147 }
1148}
1149
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001150template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption, typename Visitor>
Mathieu Chartier31e88222016-10-14 18:43:19 -07001151inline void Object::VisitInstanceFieldsReferences(ObjPtr<Class> klass, const Visitor& visitor) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001152 VisitFieldsReferences<false, kVerifyFlags, kReadBarrierOption>(
1153 klass->GetReferenceInstanceOffsets<kVerifyFlags>(), visitor);
Mathieu Chartier407f7022014-02-18 14:37:05 -08001154}
1155
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001156template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption, typename Visitor>
Mathieu Chartier31e88222016-10-14 18:43:19 -07001157inline void Object::VisitStaticFieldsReferences(ObjPtr<Class> klass, const Visitor& visitor) {
Mingyao Yang98d1cc82014-05-15 17:02:16 -07001158 DCHECK(!klass->IsTemp());
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001159 klass->VisitFieldsReferences<true, kVerifyFlags, kReadBarrierOption>(0, visitor);
Mathieu Chartier407f7022014-02-18 14:37:05 -08001160}
1161
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001162template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere4275c02015-08-06 15:34:15 -07001163inline bool Object::IsClassLoader() {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001164 return GetClass<kVerifyFlags, kReadBarrierOption>()->IsClassLoaderClass();
Mathieu Chartiere4275c02015-08-06 15:34:15 -07001165}
1166
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001167template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere4275c02015-08-06 15:34:15 -07001168inline mirror::ClassLoader* Object::AsClassLoader() {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001169 DCHECK((IsClassLoader<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartiere4275c02015-08-06 15:34:15 -07001170 return down_cast<mirror::ClassLoader*>(this);
1171}
1172
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001173template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Vladimir Marko05792b92015-08-03 11:56:49 +01001174inline bool Object::IsDexCache() {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001175 return GetClass<kVerifyFlags, kReadBarrierOption>()->IsDexCacheClass();
Vladimir Marko05792b92015-08-03 11:56:49 +01001176}
1177
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001178template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Vladimir Marko05792b92015-08-03 11:56:49 +01001179inline mirror::DexCache* Object::AsDexCache() {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001180 DCHECK((IsDexCache<kVerifyFlags, kReadBarrierOption>()));
Vladimir Marko05792b92015-08-03 11:56:49 +01001181 return down_cast<mirror::DexCache*>(this);
1182}
1183
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001184template <bool kVisitNativeRoots,
1185 VerifyObjectFlags kVerifyFlags,
1186 ReadBarrierOption kReadBarrierOption,
1187 typename Visitor,
1188 typename JavaLangRefVisitor>
Mathieu Chartier407f7022014-02-18 14:37:05 -08001189inline void Object::VisitReferences(const Visitor& visitor,
1190 const JavaLangRefVisitor& ref_visitor) {
Mathieu Chartier31e88222016-10-14 18:43:19 -07001191 ObjPtr<Class> klass = GetClass<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartier059ef3d2015-08-18 13:54:21 -07001192 visitor(this, ClassOffset(), false);
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001193 const uint32_t class_flags = klass->GetClassFlags<kVerifyNone>();
1194 if (LIKELY(class_flags == kClassFlagNormal)) {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001195 DCHECK((!klass->IsVariableSize<kVerifyFlags, kReadBarrierOption>()));
1196 VisitInstanceFieldsReferences<kVerifyFlags, kReadBarrierOption>(klass, visitor);
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001197 DCHECK((!klass->IsClassClass<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartier66c2d2d2015-08-25 14:32:32 -07001198 DCHECK(!klass->IsStringClass());
1199 DCHECK(!klass->IsClassLoaderClass());
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001200 DCHECK((!klass->IsArrayClass<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001201 } else {
1202 if ((class_flags & kClassFlagNoReferenceFields) == 0) {
1203 DCHECK(!klass->IsStringClass());
1204 if (class_flags == kClassFlagClass) {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001205 DCHECK((klass->IsClassClass<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartier31e88222016-10-14 18:43:19 -07001206 ObjPtr<Class> as_klass = AsClass<kVerifyNone, kReadBarrierOption>();
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001207 as_klass->VisitReferences<kVisitNativeRoots, kVerifyFlags, kReadBarrierOption>(klass,
1208 visitor);
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001209 } else if (class_flags == kClassFlagObjectArray) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001210 DCHECK((klass->IsObjectArrayClass<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001211 AsObjectArray<mirror::Object, kVerifyNone, kReadBarrierOption>()->VisitReferences(visitor);
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001212 } else if ((class_flags & kClassFlagReference) != 0) {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001213 VisitInstanceFieldsReferences<kVerifyFlags, kReadBarrierOption>(klass, visitor);
1214 ref_visitor(klass, AsReference<kVerifyFlags, kReadBarrierOption>());
Vladimir Marko05792b92015-08-03 11:56:49 +01001215 } else if (class_flags == kClassFlagDexCache) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001216 mirror::DexCache* const dex_cache = AsDexCache<kVerifyFlags, kReadBarrierOption>();
1217 dex_cache->VisitReferences<kVisitNativeRoots,
1218 kVerifyFlags,
1219 kReadBarrierOption>(klass, visitor);
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001220 } else {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001221 mirror::ClassLoader* const class_loader = AsClassLoader<kVerifyFlags, kReadBarrierOption>();
1222 class_loader->VisitReferences<kVisitNativeRoots,
1223 kVerifyFlags,
1224 kReadBarrierOption>(klass, visitor);
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001225 }
1226 } else if (kIsDebugBuild) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001227 CHECK((!klass->IsClassClass<kVerifyFlags, kReadBarrierOption>()));
1228 CHECK((!klass->IsObjectArrayClass<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001229 // String still has instance fields for reflection purposes but these don't exist in
1230 // actual string instances.
1231 if (!klass->IsStringClass()) {
1232 size_t total_reference_instance_fields = 0;
Mathieu Chartier31e88222016-10-14 18:43:19 -07001233 ObjPtr<Class> super_class = klass;
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001234 do {
1235 total_reference_instance_fields += super_class->NumReferenceInstanceFields();
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001236 super_class = super_class->GetSuperClass<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001237 } while (super_class != nullptr);
1238 // The only reference field should be the object's class. This field is handled at the
1239 // beginning of the function.
1240 CHECK_EQ(total_reference_instance_fields, 1u);
1241 }
Mathieu Chartier407f7022014-02-18 14:37:05 -08001242 }
1243 }
1244}
Ian Rogers2dd0e2c2013-01-24 12:42:14 -08001245} // namespace mirror
1246} // namespace art
1247
Brian Carlstromfc0e3212013-07-17 14:40:12 -07001248#endif // ART_RUNTIME_MIRROR_OBJECT_INL_H_