blob: 354410e6bfb12a4c8a6472fdbedb6ea2ef424682 [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) {
145#ifdef USE_BAKER_READ_BARRIER
146 CHECK(kUseBakerReadBarrier);
147#if defined(__arm__)
148 uintptr_t obj = reinterpret_cast<uintptr_t>(this);
149 uintptr_t result;
150 DCHECK_EQ(OFFSETOF_MEMBER(Object, monitor_), 4U);
151 // Use inline assembly to prevent the compiler from optimizing away the false dependency.
152 __asm__ __volatile__(
153 "ldr %[result], [%[obj], #4]\n\t"
154 // This instruction is enough to "fool the compiler and the CPU" by having `fad` always be
155 // null, without them being able to assume that fact.
156 "eor %[fad], %[result], %[result]\n\t"
157 : [result] "+r" (result), [fad] "=r" (*fake_address_dependency)
158 : [obj] "r" (obj));
159 DCHECK_EQ(*fake_address_dependency, 0U);
160 LockWord lw(static_cast<uint32_t>(result));
161 uint32_t rb_state = lw.ReadBarrierState();
162 return rb_state;
163#elif defined(__aarch64__)
164 uintptr_t obj = reinterpret_cast<uintptr_t>(this);
165 uintptr_t result;
166 DCHECK_EQ(OFFSETOF_MEMBER(Object, monitor_), 4U);
167 // Use inline assembly to prevent the compiler from optimizing away the false dependency.
168 __asm__ __volatile__(
169 "ldr %w[result], [%[obj], #4]\n\t"
170 // This instruction is enough to "fool the compiler and the CPU" by having `fad` always be
171 // null, without them being able to assume that fact.
172 "eor %[fad], %[result], %[result]\n\t"
173 : [result] "+r" (result), [fad] "=r" (*fake_address_dependency)
174 : [obj] "r" (obj));
175 DCHECK_EQ(*fake_address_dependency, 0U);
176 LockWord lw(static_cast<uint32_t>(result));
177 uint32_t rb_state = lw.ReadBarrierState();
178 return rb_state;
179#elif defined(__i386__) || defined(__x86_64__)
180 LockWord lw = GetLockWord(false);
181 // i386/x86_64 don't need fake address dependency. Use a compiler fence to avoid compiler
182 // reordering.
183 *fake_address_dependency = 0;
184 std::atomic_signal_fence(std::memory_order_acquire);
185 uint32_t rb_state = lw.ReadBarrierState();
186 return rb_state;
187#else
188 // mips/mips64
189 LOG(FATAL) << "Unreachable";
190 UNREACHABLE();
191 UNUSED(fake_address_dependency);
192#endif
193#else // !USE_BAKER_READ_BARRIER
194 LOG(FATAL) << "Unreachable";
195 UNREACHABLE();
196 UNUSED(fake_address_dependency);
197#endif
198}
199
200inline uint32_t Object::GetReadBarrierState() {
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700201#ifdef USE_BAKER_READ_BARRIER
202 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#else
208 LOG(FATAL) << "Unreachable";
Ian Rogers2c4257b2014-10-24 14:20:06 -0700209 UNREACHABLE();
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800210#endif
211}
212
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700213inline uint32_t Object::GetReadBarrierStateAcquire() {
Mathieu Chartierc381c362016-08-23 13:27:53 -0700214#ifdef USE_BAKER_READ_BARRIER
215 DCHECK(kUseBakerReadBarrier);
216 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#else
221 LOG(FATAL) << "Unreachable";
222 UNREACHABLE();
223#endif
224}
225
Mathieu Chartier36a270a2016-07-28 18:08:51 -0700226inline uint32_t Object::GetMarkBit() {
227#ifdef USE_READ_BARRIER
Mathieu Chartier36a270a2016-07-28 18:08:51 -0700228 return GetLockWord(false).MarkBitState();
229#else
230 LOG(FATAL) << "Unreachable";
231 UNREACHABLE();
232#endif
233}
234
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700235inline void Object::SetReadBarrierState(uint32_t rb_state) {
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700236#ifdef USE_BAKER_READ_BARRIER
237 DCHECK(kUseBakerReadBarrier);
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700238 DCHECK(ReadBarrier::IsValidReadBarrierState(rb_state)) << rb_state;
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700239 LockWord lw = GetLockWord(false);
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700240 lw.SetReadBarrierState(rb_state);
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700241 SetLockWord(lw, false);
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800242#else
243 LOG(FATAL) << "Unreachable";
Ian Rogers2c4257b2014-10-24 14:20:06 -0700244 UNREACHABLE();
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700245 UNUSED(rb_state);
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800246#endif
247}
248
Hiroshi Yamauchied70b4a2015-11-17 17:52:15 -0800249template<bool kCasRelease>
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700250inline bool Object::AtomicSetReadBarrierState(uint32_t expected_rb_state, uint32_t rb_state) {
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700251#ifdef USE_BAKER_READ_BARRIER
252 DCHECK(kUseBakerReadBarrier);
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700253 DCHECK(ReadBarrier::IsValidReadBarrierState(expected_rb_state)) << expected_rb_state;
254 DCHECK(ReadBarrier::IsValidReadBarrierState(rb_state)) << rb_state;
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700255 LockWord expected_lw;
256 LockWord new_lw;
257 do {
258 LockWord lw = GetLockWord(false);
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700259 if (UNLIKELY(lw.ReadBarrierState() != expected_rb_state)) {
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700260 // Lost the race.
261 return false;
262 }
263 expected_lw = lw;
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700264 expected_lw.SetReadBarrierState(expected_rb_state);
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700265 new_lw = lw;
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700266 new_lw.SetReadBarrierState(rb_state);
Hiroshi Yamauchied70b4a2015-11-17 17:52:15 -0800267 // ConcurrentCopying::ProcessMarkStackRef uses this with kCasRelease == true.
268 // If kCasRelease == true, use a CAS release so that when GC updates all the fields of
269 // an object and then changes the object from gray to black, the field updates (stores) will be
270 // visible (won't be reordered after this CAS.)
271 } while (!(kCasRelease ?
272 CasLockWordWeakRelease(expected_lw, new_lw) :
273 CasLockWordWeakRelaxed(expected_lw, new_lw)));
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700274 return true;
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700275#else
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700276 UNUSED(expected_rb_state, rb_state);
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700277 LOG(FATAL) << "Unreachable";
Ian Rogers2c4257b2014-10-24 14:20:06 -0700278 UNREACHABLE();
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700279#endif
280}
281
Mathieu Chartier36a270a2016-07-28 18:08:51 -0700282inline bool Object::AtomicSetMarkBit(uint32_t expected_mark_bit, uint32_t mark_bit) {
283 LockWord expected_lw;
284 LockWord new_lw;
285 do {
286 LockWord lw = GetLockWord(false);
287 if (UNLIKELY(lw.MarkBitState() != expected_mark_bit)) {
288 // Lost the race.
289 return false;
290 }
291 expected_lw = lw;
292 new_lw = lw;
293 new_lw.SetMarkBitState(mark_bit);
294 // Since this is only set from the mutator, we can use the non release Cas.
295 } while (!CasLockWordWeakRelaxed(expected_lw, new_lw));
296 return true;
297}
298
299
Hiroshi Yamauchi12b58b22016-11-01 11:55:29 -0700300inline void Object::AssertReadBarrierState() const {
301 CHECK(kUseBakerReadBarrier);
302 Object* obj = const_cast<Object*>(this);
303 DCHECK(obj->GetReadBarrierState() == ReadBarrier::WhiteState())
304 << "Bad Baker pointer: obj=" << reinterpret_cast<void*>(obj)
305 << " rb_state" << reinterpret_cast<void*>(obj->GetReadBarrierState());
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800306}
307
Mathieu Chartier4e305412014-02-19 10:54:44 -0800308template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartieref41db72016-10-25 15:08:01 -0700309inline bool Object::VerifierInstanceOf(ObjPtr<Class> klass) {
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700310 DCHECK(klass != nullptr);
311 DCHECK(GetClass<kVerifyFlags>() != nullptr);
Jeff Haoa3faaf42013-09-03 19:07:00 -0700312 return klass->IsInterface() || InstanceOf(klass);
313}
314
Mathieu Chartier4e305412014-02-19 10:54:44 -0800315template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiera59d9b22016-09-26 18:13:17 -0700316inline bool Object::InstanceOf(ObjPtr<Class> klass) {
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700317 DCHECK(klass != nullptr);
318 DCHECK(GetClass<kVerifyNone>() != nullptr);
Mathieu Chartier4e305412014-02-19 10:54:44 -0800319 return klass->IsAssignableFrom(GetClass<kVerifyFlags>());
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800320}
321
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700322template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800323inline bool Object::IsClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700324 Class* java_lang_Class = GetClass<kVerifyFlags, kReadBarrierOption>()->
325 template GetClass<kVerifyFlags, kReadBarrierOption>();
326 return GetClass<static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis),
327 kReadBarrierOption>() == java_lang_Class;
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800328}
329
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700330template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800331inline Class* Object::AsClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700332 DCHECK((IsClass<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800333 return down_cast<Class*>(this);
334}
335
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800336template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800337inline bool Object::IsObjectArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800338 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800339 return IsArrayInstance<kVerifyFlags, kReadBarrierOption>() &&
340 !GetClass<kNewFlags, kReadBarrierOption>()->
341 template GetComponentType<kNewFlags, kReadBarrierOption>()->IsPrimitive();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800342}
343
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800344template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800345inline ObjectArray<T>* Object::AsObjectArray() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800346 DCHECK((IsObjectArray<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800347 return down_cast<ObjectArray<T>*>(this);
348}
349
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700350template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800351inline bool Object::IsArrayInstance() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700352 return GetClass<kVerifyFlags, kReadBarrierOption>()->
353 template IsArrayClass<kVerifyFlags, kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800354}
355
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800356template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800357inline bool Object::IsReferenceInstance() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800358 return GetClass<kVerifyFlags, kReadBarrierOption>()->IsTypeOfReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800359}
360
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800361template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700362inline Reference* Object::AsReference() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800363 DCHECK((IsReferenceInstance<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700364 return down_cast<Reference*>(this);
365}
366
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700367template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers05f30572013-02-20 12:13:11 -0800368inline Array* Object::AsArray() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700369 DCHECK((IsArrayInstance<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800370 return down_cast<Array*>(this);
371}
372
Mathieu Chartier4e305412014-02-19 10:54:44 -0800373template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800374inline BooleanArray* Object::AsBooleanArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800375 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
376 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
377 DCHECK(GetClass<kNewFlags>()->GetComponentType()->IsPrimitiveBoolean());
Ian Rogers05f30572013-02-20 12:13:11 -0800378 return down_cast<BooleanArray*>(this);
379}
380
Mathieu Chartier4e305412014-02-19 10:54:44 -0800381template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800382inline ByteArray* Object::AsByteArray() {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700383 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Mathieu Chartier4e305412014-02-19 10:54:44 -0800384 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
385 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveByte());
Ian Rogers05f30572013-02-20 12:13:11 -0800386 return down_cast<ByteArray*>(this);
387}
388
Mathieu Chartier4e305412014-02-19 10:54:44 -0800389template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800390inline ByteArray* Object::AsByteSizedArray() {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700391 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Mathieu Chartier4e305412014-02-19 10:54:44 -0800392 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
393 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveByte() ||
394 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveBoolean());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800395 return down_cast<ByteArray*>(this);
396}
397
Mathieu Chartier4e305412014-02-19 10:54:44 -0800398template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800399inline CharArray* Object::AsCharArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800400 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
401 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
402 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveChar());
Ian Rogers05f30572013-02-20 12:13:11 -0800403 return down_cast<CharArray*>(this);
404}
405
Mathieu Chartier4e305412014-02-19 10:54:44 -0800406template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800407inline ShortArray* Object::AsShortArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800408 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
409 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
410 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveShort());
Ian Rogers05f30572013-02-20 12:13:11 -0800411 return down_cast<ShortArray*>(this);
412}
413
Mathieu Chartier4e305412014-02-19 10:54:44 -0800414template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800415inline ShortArray* Object::AsShortSizedArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800416 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
417 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
418 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveShort() ||
419 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveChar());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800420 return down_cast<ShortArray*>(this);
421}
422
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800423template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700424inline bool Object::IsIntArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800425 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Mathieu Chartier31e88222016-10-14 18:43:19 -0700426 ObjPtr<Class> klass = GetClass<kVerifyFlags, kReadBarrierOption>();
427 ObjPtr<Class> component_type = klass->GetComponentType<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartiere401d142015-04-22 13:56:20 -0700428 return component_type != nullptr && component_type->template IsPrimitiveInt<kNewFlags>();
429}
430
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800431template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700432inline IntArray* Object::AsIntArray() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800433 DCHECK((IsIntArray<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800434 return down_cast<IntArray*>(this);
435}
436
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800437template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700438inline bool Object::IsLongArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800439 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Mathieu Chartier31e88222016-10-14 18:43:19 -0700440 ObjPtr<Class> klass = GetClass<kVerifyFlags, kReadBarrierOption>();
441 ObjPtr<Class> component_type = klass->GetComponentType<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartiere401d142015-04-22 13:56:20 -0700442 return component_type != nullptr && component_type->template IsPrimitiveLong<kNewFlags>();
443}
444
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800445template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700446inline LongArray* Object::AsLongArray() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800447 DCHECK((IsLongArray<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800448 return down_cast<LongArray*>(this);
449}
450
Mathieu Chartier4e305412014-02-19 10:54:44 -0800451template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700452inline bool Object::IsFloatArray() {
453 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
454 auto* component_type = GetClass<kVerifyFlags>()->GetComponentType();
455 return component_type != nullptr && component_type->template IsPrimitiveFloat<kNewFlags>();
456}
457
458template<VerifyObjectFlags kVerifyFlags>
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100459inline FloatArray* Object::AsFloatArray() {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700460 DCHECK(IsFloatArray<kVerifyFlags>());
Mathieu Chartier4e305412014-02-19 10:54:44 -0800461 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
462 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
463 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveFloat());
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100464 return down_cast<FloatArray*>(this);
465}
466
Mathieu Chartier4e305412014-02-19 10:54:44 -0800467template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700468inline bool Object::IsDoubleArray() {
469 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
470 auto* component_type = GetClass<kVerifyFlags>()->GetComponentType();
471 return component_type != nullptr && component_type->template IsPrimitiveDouble<kNewFlags>();
472}
473
474template<VerifyObjectFlags kVerifyFlags>
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100475inline DoubleArray* Object::AsDoubleArray() {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700476 DCHECK(IsDoubleArray<kVerifyFlags>());
Mathieu Chartier4e305412014-02-19 10:54:44 -0800477 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
478 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
479 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveDouble());
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100480 return down_cast<DoubleArray*>(this);
481}
482
Jeff Hao848f70a2014-01-15 13:49:50 -0800483template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
484inline bool Object::IsString() {
485 return GetClass<kVerifyFlags, kReadBarrierOption>()->IsStringClass();
486}
487
488template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers05f30572013-02-20 12:13:11 -0800489inline String* Object::AsString() {
Jeff Hao848f70a2014-01-15 13:49:50 -0800490 DCHECK((IsString<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800491 return down_cast<String*>(this);
492}
493
Mathieu Chartier4e305412014-02-19 10:54:44 -0800494template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800495inline Throwable* Object::AsThrowable() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800496 DCHECK(GetClass<kVerifyFlags>()->IsThrowableClass());
Ian Rogers05f30572013-02-20 12:13:11 -0800497 return down_cast<Throwable*>(this);
498}
499
Mathieu Chartier4e305412014-02-19 10:54:44 -0800500template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800501inline bool Object::IsWeakReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800502 return GetClass<kVerifyFlags>()->IsWeakReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800503}
504
Mathieu Chartier4e305412014-02-19 10:54:44 -0800505template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800506inline bool Object::IsSoftReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800507 return GetClass<kVerifyFlags>()->IsSoftReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800508}
509
Mathieu Chartier4e305412014-02-19 10:54:44 -0800510template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800511inline bool Object::IsFinalizerReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800512 return GetClass<kVerifyFlags>()->IsFinalizerReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800513}
514
Mathieu Chartier4e305412014-02-19 10:54:44 -0800515template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700516inline FinalizerReference* Object::AsFinalizerReference() {
517 DCHECK(IsFinalizerReferenceInstance<kVerifyFlags>());
518 return down_cast<FinalizerReference*>(this);
519}
520
521template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800522inline bool Object::IsPhantomReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800523 return GetClass<kVerifyFlags>()->IsPhantomReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800524}
525
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700526template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800527inline size_t Object::SizeOf() {
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800528 size_t result;
Mathieu Chartier4e305412014-02-19 10:54:44 -0800529 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700530 if (IsArrayInstance<kVerifyFlags, kReadBarrierOption>()) {
531 result = AsArray<kNewFlags, kReadBarrierOption>()->
532 template SizeOf<kNewFlags, kReadBarrierOption>();
533 } else if (IsClass<kNewFlags, kReadBarrierOption>()) {
534 result = AsClass<kNewFlags, kReadBarrierOption>()->
535 template SizeOf<kNewFlags, kReadBarrierOption>();
Jeff Hao848f70a2014-01-15 13:49:50 -0800536 } else if (GetClass<kNewFlags, kReadBarrierOption>()->IsStringClass()) {
537 result = AsString<kNewFlags, kReadBarrierOption>()->
538 template SizeOf<kNewFlags>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800539 } else {
Hiroshi Yamauchi25023c72014-05-09 11:45:53 -0700540 result = GetClass<kNewFlags, kReadBarrierOption>()->
541 template GetObjectSize<kNewFlags, kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800542 }
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700543 DCHECK_GE(result, sizeof(Object))
David Sehr709b0702016-10-13 09:12:37 -0700544 << " class=" << Class::PrettyClass(GetClass<kNewFlags, kReadBarrierOption>());
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800545 return result;
546}
547
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700548template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
Fred Shih37f05ef2014-07-16 18:38:08 -0700549inline uint8_t Object::GetFieldBoolean(MemberOffset field_offset) {
550 if (kVerifyFlags & kVerifyThis) {
551 VerifyObject(this);
552 }
553 return GetField<uint8_t, kIsVolatile>(field_offset);
554}
555
556template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
557inline int8_t Object::GetFieldByte(MemberOffset field_offset) {
558 if (kVerifyFlags & kVerifyThis) {
559 VerifyObject(this);
560 }
561 return GetField<int8_t, kIsVolatile>(field_offset);
562}
563
564template<VerifyObjectFlags kVerifyFlags>
565inline uint8_t Object::GetFieldBooleanVolatile(MemberOffset field_offset) {
566 return GetFieldBoolean<kVerifyFlags, true>(field_offset);
567}
568
569template<VerifyObjectFlags kVerifyFlags>
570inline int8_t Object::GetFieldByteVolatile(MemberOffset field_offset) {
571 return GetFieldByte<kVerifyFlags, true>(field_offset);
572}
573
574template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
575 bool kIsVolatile>
576inline void Object::SetFieldBoolean(MemberOffset field_offset, uint8_t new_value)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700577 REQUIRES_SHARED(Locks::mutator_lock_) {
Fred Shih37f05ef2014-07-16 18:38:08 -0700578 if (kCheckTransaction) {
579 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
580 }
581 if (kTransactionActive) {
582 Runtime::Current()->RecordWriteFieldBoolean(this, field_offset,
583 GetFieldBoolean<kVerifyFlags, kIsVolatile>(field_offset),
584 kIsVolatile);
585 }
586 if (kVerifyFlags & kVerifyThis) {
587 VerifyObject(this);
588 }
589 SetField<uint8_t, kIsVolatile>(field_offset, new_value);
590}
591
592template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
593 bool kIsVolatile>
594inline void Object::SetFieldByte(MemberOffset field_offset, int8_t new_value)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700595 REQUIRES_SHARED(Locks::mutator_lock_) {
Fred Shih37f05ef2014-07-16 18:38:08 -0700596 if (kCheckTransaction) {
597 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
598 }
599 if (kTransactionActive) {
600 Runtime::Current()->RecordWriteFieldByte(this, field_offset,
601 GetFieldByte<kVerifyFlags, kIsVolatile>(field_offset),
602 kIsVolatile);
603 }
604 if (kVerifyFlags & kVerifyThis) {
605 VerifyObject(this);
606 }
607 SetField<int8_t, kIsVolatile>(field_offset, new_value);
608}
609
610template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
611inline void Object::SetFieldBooleanVolatile(MemberOffset field_offset, uint8_t new_value) {
612 return SetFieldBoolean<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
613 field_offset, new_value);
614}
615
616template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
617inline void Object::SetFieldByteVolatile(MemberOffset field_offset, int8_t new_value) {
618 return SetFieldByte<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
619 field_offset, new_value);
620}
621
622template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
623inline uint16_t Object::GetFieldChar(MemberOffset field_offset) {
624 if (kVerifyFlags & kVerifyThis) {
625 VerifyObject(this);
626 }
627 return GetField<uint16_t, kIsVolatile>(field_offset);
628}
629
630template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
631inline int16_t Object::GetFieldShort(MemberOffset field_offset) {
632 if (kVerifyFlags & kVerifyThis) {
633 VerifyObject(this);
634 }
635 return GetField<int16_t, kIsVolatile>(field_offset);
636}
637
638template<VerifyObjectFlags kVerifyFlags>
639inline uint16_t Object::GetFieldCharVolatile(MemberOffset field_offset) {
640 return GetFieldChar<kVerifyFlags, true>(field_offset);
641}
642
643template<VerifyObjectFlags kVerifyFlags>
644inline int16_t Object::GetFieldShortVolatile(MemberOffset field_offset) {
645 return GetFieldShort<kVerifyFlags, true>(field_offset);
646}
647
648template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
649 bool kIsVolatile>
650inline void Object::SetFieldChar(MemberOffset field_offset, uint16_t new_value) {
651 if (kCheckTransaction) {
652 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
653 }
654 if (kTransactionActive) {
655 Runtime::Current()->RecordWriteFieldChar(this, field_offset,
656 GetFieldChar<kVerifyFlags, kIsVolatile>(field_offset),
657 kIsVolatile);
658 }
659 if (kVerifyFlags & kVerifyThis) {
660 VerifyObject(this);
661 }
662 SetField<uint16_t, kIsVolatile>(field_offset, new_value);
663}
664
665template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
666 bool kIsVolatile>
667inline void Object::SetFieldShort(MemberOffset field_offset, int16_t new_value) {
668 if (kCheckTransaction) {
669 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
670 }
671 if (kTransactionActive) {
672 Runtime::Current()->RecordWriteFieldChar(this, field_offset,
673 GetFieldShort<kVerifyFlags, kIsVolatile>(field_offset),
674 kIsVolatile);
675 }
676 if (kVerifyFlags & kVerifyThis) {
677 VerifyObject(this);
678 }
679 SetField<int16_t, kIsVolatile>(field_offset, new_value);
680}
681
682template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
683inline void Object::SetFieldCharVolatile(MemberOffset field_offset, uint16_t new_value) {
684 return SetFieldChar<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
685 field_offset, new_value);
686}
687
688template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
689inline void Object::SetFieldShortVolatile(MemberOffset field_offset, int16_t new_value) {
690 return SetFieldShort<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
691 field_offset, new_value);
692}
693
694template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700695inline int32_t Object::GetField32(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800696 if (kVerifyFlags & kVerifyThis) {
697 VerifyObject(this);
698 }
Fred Shih37f05ef2014-07-16 18:38:08 -0700699 return GetField<int32_t, kIsVolatile>(field_offset);
Ian Rogersb122a4b2013-11-19 18:00:50 -0800700}
701
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700702template<VerifyObjectFlags kVerifyFlags>
703inline int32_t Object::GetField32Volatile(MemberOffset field_offset) {
704 return GetField32<kVerifyFlags, true>(field_offset);
705}
706
707template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
708 bool kIsVolatile>
709inline void Object::SetField32(MemberOffset field_offset, int32_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100710 if (kCheckTransaction) {
711 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
712 }
713 if (kTransactionActive) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700714 Runtime::Current()->RecordWriteField32(this, field_offset,
715 GetField32<kVerifyFlags, kIsVolatile>(field_offset),
716 kIsVolatile);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100717 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800718 if (kVerifyFlags & kVerifyThis) {
Ian Rogersb122a4b2013-11-19 18:00:50 -0800719 VerifyObject(this);
720 }
Fred Shih37f05ef2014-07-16 18:38:08 -0700721 SetField<int32_t, kIsVolatile>(field_offset, new_value);
Ian Rogersb122a4b2013-11-19 18:00:50 -0800722}
723
Mathieu Chartier4e305412014-02-19 10:54:44 -0800724template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700725inline void Object::SetField32Volatile(MemberOffset field_offset, int32_t new_value) {
726 SetField32<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset, new_value);
727}
728
Hans Boehmd8434432014-07-11 09:56:07 -0700729// TODO: Pass memory_order_ and strong/weak as arguments to avoid code duplication?
730
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700731template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -0700732inline bool Object::CasFieldWeakSequentiallyConsistent32(MemberOffset field_offset,
733 int32_t old_value, int32_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100734 if (kCheckTransaction) {
735 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
736 }
737 if (kTransactionActive) {
738 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
739 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800740 if (kVerifyFlags & kVerifyThis) {
741 VerifyObject(this);
742 }
Ian Rogers13735952014-10-08 12:43:28 -0700743 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogers228602f2014-07-10 02:07:54 -0700744 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
Hans Boehm30359612014-05-21 17:46:23 -0700745
Ian Rogers228602f2014-07-10 02:07:54 -0700746 return atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_value, new_value);
Ian Rogersd9c4fc92013-10-01 19:45:43 -0700747}
748
Hans Boehmd8434432014-07-11 09:56:07 -0700749template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
750inline bool Object::CasFieldWeakRelaxed32(MemberOffset field_offset,
751 int32_t old_value, int32_t new_value) {
752 if (kCheckTransaction) {
753 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
754 }
755 if (kTransactionActive) {
756 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
757 }
758 if (kVerifyFlags & kVerifyThis) {
759 VerifyObject(this);
760 }
Ian Rogers13735952014-10-08 12:43:28 -0700761 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Hans Boehmd8434432014-07-11 09:56:07 -0700762 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
763
764 return atomic_addr->CompareExchangeWeakRelaxed(old_value, new_value);
765}
766
767template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Hans Boehmb3da36c2016-12-15 13:12:59 -0800768inline bool Object::CasFieldWeakAcquire32(MemberOffset field_offset,
769 int32_t old_value, int32_t new_value) {
770 if (kCheckTransaction) {
771 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
772 }
773 if (kTransactionActive) {
774 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
775 }
776 if (kVerifyFlags & kVerifyThis) {
777 VerifyObject(this);
778 }
779 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
780 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
781
782 return atomic_addr->CompareExchangeWeakAcquire(old_value, new_value);
783}
784
785template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -0700786inline bool Object::CasFieldWeakRelease32(MemberOffset field_offset,
787 int32_t old_value, int32_t new_value) {
788 if (kCheckTransaction) {
789 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
790 }
791 if (kTransactionActive) {
792 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
793 }
794 if (kVerifyFlags & kVerifyThis) {
795 VerifyObject(this);
796 }
797 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
798 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
799
800 return atomic_addr->CompareExchangeWeakRelease(old_value, new_value);
801}
802
803template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Hans Boehmd8434432014-07-11 09:56:07 -0700804inline bool Object::CasFieldStrongSequentiallyConsistent32(MemberOffset field_offset,
805 int32_t old_value, int32_t new_value) {
806 if (kCheckTransaction) {
807 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
808 }
809 if (kTransactionActive) {
810 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
811 }
812 if (kVerifyFlags & kVerifyThis) {
813 VerifyObject(this);
814 }
Ian Rogers13735952014-10-08 12:43:28 -0700815 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Hans Boehmd8434432014-07-11 09:56:07 -0700816 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
817
818 return atomic_addr->CompareExchangeStrongSequentiallyConsistent(old_value, new_value);
819}
820
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700821template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
822inline int64_t Object::GetField64(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800823 if (kVerifyFlags & kVerifyThis) {
824 VerifyObject(this);
825 }
Fred Shih37f05ef2014-07-16 18:38:08 -0700826 return GetField<int64_t, kIsVolatile>(field_offset);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800827}
828
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700829template<VerifyObjectFlags kVerifyFlags>
830inline int64_t Object::GetField64Volatile(MemberOffset field_offset) {
831 return GetField64<kVerifyFlags, true>(field_offset);
832}
833
834template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
835 bool kIsVolatile>
836inline void Object::SetField64(MemberOffset field_offset, int64_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100837 if (kCheckTransaction) {
838 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
839 }
840 if (kTransactionActive) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700841 Runtime::Current()->RecordWriteField64(this, field_offset,
842 GetField64<kVerifyFlags, kIsVolatile>(field_offset),
843 kIsVolatile);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100844 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800845 if (kVerifyFlags & kVerifyThis) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800846 VerifyObject(this);
847 }
Fred Shih37f05ef2014-07-16 18:38:08 -0700848 SetField<int64_t, kIsVolatile>(field_offset, new_value);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800849}
850
Mathieu Chartier4e305412014-02-19 10:54:44 -0800851template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700852inline void Object::SetField64Volatile(MemberOffset field_offset, int64_t new_value) {
853 return SetField64<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
854 new_value);
855}
856
Fred Shih37f05ef2014-07-16 18:38:08 -0700857template<typename kSize, bool kIsVolatile>
858inline void Object::SetField(MemberOffset field_offset, kSize new_value) {
Ian Rogers13735952014-10-08 12:43:28 -0700859 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Fred Shih37f05ef2014-07-16 18:38:08 -0700860 kSize* addr = reinterpret_cast<kSize*>(raw_addr);
861 if (kIsVolatile) {
862 reinterpret_cast<Atomic<kSize>*>(addr)->StoreSequentiallyConsistent(new_value);
863 } else {
864 reinterpret_cast<Atomic<kSize>*>(addr)->StoreJavaData(new_value);
865 }
866}
867
868template<typename kSize, bool kIsVolatile>
869inline kSize Object::GetField(MemberOffset field_offset) {
Ian Rogers13735952014-10-08 12:43:28 -0700870 const uint8_t* raw_addr = reinterpret_cast<const uint8_t*>(this) + field_offset.Int32Value();
Fred Shih37f05ef2014-07-16 18:38:08 -0700871 const kSize* addr = reinterpret_cast<const kSize*>(raw_addr);
872 if (kIsVolatile) {
873 return reinterpret_cast<const Atomic<kSize>*>(addr)->LoadSequentiallyConsistent();
874 } else {
875 return reinterpret_cast<const Atomic<kSize>*>(addr)->LoadJavaData();
876 }
877}
878
Mathieu Chartierc381c362016-08-23 13:27:53 -0700879template<typename kSize>
880inline kSize Object::GetFieldAcquire(MemberOffset field_offset) {
881 const uint8_t* raw_addr = reinterpret_cast<const uint8_t*>(this) + field_offset.Int32Value();
882 const kSize* addr = reinterpret_cast<const kSize*>(raw_addr);
883 return reinterpret_cast<const Atomic<kSize>*>(addr)->LoadAcquire();
884}
885
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700886template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -0700887inline bool Object::CasFieldWeakSequentiallyConsistent64(MemberOffset field_offset,
888 int64_t old_value, int64_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100889 if (kCheckTransaction) {
890 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
891 }
892 if (kTransactionActive) {
893 Runtime::Current()->RecordWriteField64(this, field_offset, old_value, true);
894 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800895 if (kVerifyFlags & kVerifyThis) {
896 VerifyObject(this);
897 }
Ian Rogers13735952014-10-08 12:43:28 -0700898 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogers228602f2014-07-10 02:07:54 -0700899 Atomic<int64_t>* atomic_addr = reinterpret_cast<Atomic<int64_t>*>(raw_addr);
900 return atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_value, new_value);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800901}
902
Hans Boehmd8434432014-07-11 09:56:07 -0700903template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
904inline bool Object::CasFieldStrongSequentiallyConsistent64(MemberOffset field_offset,
905 int64_t old_value, int64_t new_value) {
906 if (kCheckTransaction) {
907 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
908 }
909 if (kTransactionActive) {
910 Runtime::Current()->RecordWriteField64(this, field_offset, old_value, true);
911 }
912 if (kVerifyFlags & kVerifyThis) {
913 VerifyObject(this);
914 }
Ian Rogers13735952014-10-08 12:43:28 -0700915 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Hans Boehmd8434432014-07-11 09:56:07 -0700916 Atomic<int64_t>* atomic_addr = reinterpret_cast<Atomic<int64_t>*>(raw_addr);
917 return atomic_addr->CompareExchangeStrongSequentiallyConsistent(old_value, new_value);
918}
919
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700920template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption,
921 bool kIsVolatile>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700922inline T* Object::GetFieldObject(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800923 if (kVerifyFlags & kVerifyThis) {
924 VerifyObject(this);
925 }
Ian Rogers13735952014-10-08 12:43:28 -0700926 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogersef7d42f2014-01-06 12:55:46 -0800927 HeapReference<T>* objref_addr = reinterpret_cast<HeapReference<T>*>(raw_addr);
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700928 T* result = ReadBarrier::Barrier<T, kReadBarrierOption>(this, field_offset, objref_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700929 if (kIsVolatile) {
Hans Boehm30359612014-05-21 17:46:23 -0700930 // TODO: Refactor to use a SequentiallyConsistent load instead.
931 QuasiAtomic::ThreadFenceAcquire(); // Ensure visibility of operations preceding store.
Ian Rogersef7d42f2014-01-06 12:55:46 -0800932 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800933 if (kVerifyFlags & kVerifyReads) {
934 VerifyObject(result);
935 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800936 return result;
937}
938
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700939template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700940inline T* Object::GetFieldObjectVolatile(MemberOffset field_offset) {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700941 return GetFieldObject<T, kVerifyFlags, kReadBarrierOption, true>(field_offset);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700942}
943
944template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
945 bool kIsVolatile>
946inline void Object::SetFieldObjectWithoutWriteBarrier(MemberOffset field_offset,
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700947 ObjPtr<Object> new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100948 if (kCheckTransaction) {
949 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
950 }
951 if (kTransactionActive) {
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700952 ObjPtr<Object> obj;
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700953 if (kIsVolatile) {
954 obj = GetFieldObjectVolatile<Object>(field_offset);
955 } else {
956 obj = GetFieldObject<Object>(field_offset);
957 }
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700958 Runtime::Current()->RecordWriteFieldReference(this, field_offset, obj.Ptr(), true);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100959 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800960 if (kVerifyFlags & kVerifyThis) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800961 VerifyObject(this);
962 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800963 if (kVerifyFlags & kVerifyWrites) {
964 VerifyObject(new_value);
965 }
Ian Rogers13735952014-10-08 12:43:28 -0700966 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogersef7d42f2014-01-06 12:55:46 -0800967 HeapReference<Object>* objref_addr = reinterpret_cast<HeapReference<Object>*>(raw_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700968 if (kIsVolatile) {
Hans Boehm30359612014-05-21 17:46:23 -0700969 // TODO: Refactor to use a SequentiallyConsistent store instead.
970 QuasiAtomic::ThreadFenceRelease(); // Ensure that prior accesses are visible before store.
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700971 objref_addr->Assign(new_value.Ptr());
Hans Boehm30359612014-05-21 17:46:23 -0700972 QuasiAtomic::ThreadFenceSequentiallyConsistent();
973 // Ensure this store occurs before any volatile loads.
Ian Rogersef7d42f2014-01-06 12:55:46 -0800974 } else {
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700975 objref_addr->Assign(new_value.Ptr());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800976 }
977}
978
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700979template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
980 bool kIsVolatile>
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700981inline void Object::SetFieldObject(MemberOffset field_offset, ObjPtr<Object> new_value) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700982 SetFieldObjectWithoutWriteBarrier<kTransactionActive, kCheckTransaction, kVerifyFlags,
983 kIsVolatile>(field_offset, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800984 if (new_value != nullptr) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800985 Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700986 // TODO: Check field assignment could theoretically cause thread suspension, TODO: fix this.
987 CheckFieldAssignment(field_offset, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800988 }
989}
990
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700991template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Mathieu Chartiera058fdf2016-10-06 15:13:58 -0700992inline void Object::SetFieldObjectVolatile(MemberOffset field_offset, ObjPtr<Object> new_value) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700993 SetFieldObject<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
994 new_value);
995}
996
Mathieu Chartier4e305412014-02-19 10:54:44 -0800997template <VerifyObjectFlags kVerifyFlags>
998inline HeapReference<Object>* Object::GetFieldObjectReferenceAddr(MemberOffset field_offset) {
999 if (kVerifyFlags & kVerifyThis) {
1000 VerifyObject(this);
1001 }
Ian Rogers13735952014-10-08 12:43:28 -07001002 return reinterpret_cast<HeapReference<Object>*>(reinterpret_cast<uint8_t*>(this) +
Mathieu Chartier4e305412014-02-19 10:54:44 -08001003 field_offset.Int32Value());
1004}
1005
1006template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -07001007inline bool Object::CasFieldWeakSequentiallyConsistentObject(MemberOffset field_offset,
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001008 ObjPtr<Object> old_value,
1009 ObjPtr<Object> new_value) {
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -08001010 bool success = CasFieldWeakSequentiallyConsistentObjectWithoutWriteBarrier<
1011 kTransactionActive, kCheckTransaction, kVerifyFlags>(field_offset, old_value, new_value);
1012 if (success) {
1013 Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
1014 }
1015 return success;
1016}
1017
1018template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
1019inline bool Object::CasFieldWeakSequentiallyConsistentObjectWithoutWriteBarrier(
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001020 MemberOffset field_offset,
1021 ObjPtr<Object> old_value,
1022 ObjPtr<Object> new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +01001023 if (kCheckTransaction) {
1024 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
1025 }
Mathieu Chartier4e305412014-02-19 10:54:44 -08001026 if (kVerifyFlags & kVerifyThis) {
1027 VerifyObject(this);
1028 }
1029 if (kVerifyFlags & kVerifyWrites) {
1030 VerifyObject(new_value);
1031 }
1032 if (kVerifyFlags & kVerifyReads) {
1033 VerifyObject(old_value);
1034 }
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +01001035 if (kTransactionActive) {
1036 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
1037 }
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001038 HeapReference<Object> old_ref(HeapReference<Object>::FromObjPtr(old_value));
1039 HeapReference<Object> new_ref(HeapReference<Object>::FromObjPtr(new_value));
Ian Rogers13735952014-10-08 12:43:28 -07001040 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogers228602f2014-07-10 02:07:54 -07001041 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
1042
1043 bool success = atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_ref.reference_,
1044 new_ref.reference_);
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -08001045 return success;
1046}
Ian Rogers228602f2014-07-10 02:07:54 -07001047
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -08001048template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
1049inline bool Object::CasFieldStrongSequentiallyConsistentObject(MemberOffset field_offset,
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001050 ObjPtr<Object> old_value,
1051 ObjPtr<Object> new_value) {
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -08001052 bool success = CasFieldStrongSequentiallyConsistentObjectWithoutWriteBarrier<
1053 kTransactionActive, kCheckTransaction, kVerifyFlags>(field_offset, old_value, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -08001054 if (success) {
1055 Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
1056 }
1057 return success;
1058}
1059
Hans Boehmd8434432014-07-11 09:56:07 -07001060template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -08001061inline bool Object::CasFieldStrongSequentiallyConsistentObjectWithoutWriteBarrier(
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001062 MemberOffset field_offset,
1063 ObjPtr<Object> old_value,
1064 ObjPtr<Object> new_value) {
Hans Boehmd8434432014-07-11 09:56:07 -07001065 if (kCheckTransaction) {
1066 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
1067 }
1068 if (kVerifyFlags & kVerifyThis) {
1069 VerifyObject(this);
1070 }
1071 if (kVerifyFlags & kVerifyWrites) {
1072 VerifyObject(new_value);
1073 }
1074 if (kVerifyFlags & kVerifyReads) {
1075 VerifyObject(old_value);
1076 }
1077 if (kTransactionActive) {
1078 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
1079 }
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001080 HeapReference<Object> old_ref(HeapReference<Object>::FromObjPtr(old_value));
1081 HeapReference<Object> new_ref(HeapReference<Object>::FromObjPtr(new_value));
Ian Rogers13735952014-10-08 12:43:28 -07001082 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Hans Boehmd8434432014-07-11 09:56:07 -07001083 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
1084
1085 bool success = atomic_addr->CompareExchangeStrongSequentiallyConsistent(old_ref.reference_,
1086 new_ref.reference_);
Hans Boehmd8434432014-07-11 09:56:07 -07001087 return success;
1088}
1089
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -07001090template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
1091inline bool Object::CasFieldWeakRelaxedObjectWithoutWriteBarrier(
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001092 MemberOffset field_offset,
1093 ObjPtr<Object> old_value,
1094 ObjPtr<Object> new_value) {
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -07001095 if (kCheckTransaction) {
1096 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
1097 }
1098 if (kVerifyFlags & kVerifyThis) {
1099 VerifyObject(this);
1100 }
1101 if (kVerifyFlags & kVerifyWrites) {
1102 VerifyObject(new_value);
1103 }
1104 if (kVerifyFlags & kVerifyReads) {
1105 VerifyObject(old_value);
1106 }
1107 if (kTransactionActive) {
1108 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
1109 }
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001110 HeapReference<Object> old_ref(HeapReference<Object>::FromObjPtr(old_value));
1111 HeapReference<Object> new_ref(HeapReference<Object>::FromObjPtr(new_value));
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -07001112 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
1113 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
1114
1115 bool success = atomic_addr->CompareExchangeWeakRelaxed(old_ref.reference_,
1116 new_ref.reference_);
1117 return success;
1118}
1119
1120template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
1121inline bool Object::CasFieldStrongRelaxedObjectWithoutWriteBarrier(
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001122 MemberOffset field_offset,
1123 ObjPtr<Object> old_value,
1124 ObjPtr<Object> new_value) {
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -07001125 if (kCheckTransaction) {
1126 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
1127 }
1128 if (kVerifyFlags & kVerifyThis) {
1129 VerifyObject(this);
1130 }
1131 if (kVerifyFlags & kVerifyWrites) {
1132 VerifyObject(new_value);
1133 }
1134 if (kVerifyFlags & kVerifyReads) {
1135 VerifyObject(old_value);
1136 }
1137 if (kTransactionActive) {
1138 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
1139 }
Mathieu Chartiera058fdf2016-10-06 15:13:58 -07001140 HeapReference<Object> old_ref(HeapReference<Object>::FromObjPtr(old_value));
1141 HeapReference<Object> new_ref(HeapReference<Object>::FromObjPtr(new_value));
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -07001142 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
1143 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
1144
1145 bool success = atomic_addr->CompareExchangeStrongRelaxed(old_ref.reference_,
1146 new_ref.reference_);
1147 return success;
1148}
1149
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001150template<bool kIsStatic,
1151 VerifyObjectFlags kVerifyFlags,
1152 ReadBarrierOption kReadBarrierOption,
1153 typename Visitor>
Mathieu Chartier407f7022014-02-18 14:37:05 -08001154inline void Object::VisitFieldsReferences(uint32_t ref_offsets, const Visitor& visitor) {
Ian Rogerscdc1aaf2014-10-09 13:21:38 -07001155 if (!kIsStatic && (ref_offsets != mirror::Class::kClassWalkSuper)) {
1156 // Instance fields and not the slow-path.
Ian Rogerscdc1aaf2014-10-09 13:21:38 -07001157 uint32_t field_offset = mirror::kObjectHeaderSize;
Mathieu Chartier407f7022014-02-18 14:37:05 -08001158 while (ref_offsets != 0) {
Ian Rogerscdc1aaf2014-10-09 13:21:38 -07001159 if ((ref_offsets & 1) != 0) {
1160 visitor(this, MemberOffset(field_offset), kIsStatic);
1161 }
1162 ref_offsets >>= 1;
1163 field_offset += sizeof(mirror::HeapReference<mirror::Object>);
Mathieu Chartier407f7022014-02-18 14:37:05 -08001164 }
1165 } else {
Mingyao Yangfaff0f02014-09-10 12:03:22 -07001166 // There is no reference offset bitmap. In the non-static case, walk up the class
Mathieu Chartier407f7022014-02-18 14:37:05 -08001167 // inheritance hierarchy and find reference offsets the hard way. In the static case, just
1168 // consider this class.
Mathieu Chartier31e88222016-10-14 18:43:19 -07001169 for (ObjPtr<Class> klass = kIsStatic
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001170 ? AsClass<kVerifyFlags, kReadBarrierOption>()
1171 : GetClass<kVerifyFlags, kReadBarrierOption>();
1172 klass != nullptr;
1173 klass = kIsStatic ? nullptr : klass->GetSuperClass<kVerifyFlags, kReadBarrierOption>()) {
1174 const size_t num_reference_fields =
Mathieu Chartier407f7022014-02-18 14:37:05 -08001175 kIsStatic ? klass->NumReferenceStaticFields() : klass->NumReferenceInstanceFields();
Vladimir Marko76649e82014-11-10 18:32:59 +00001176 if (num_reference_fields == 0u) {
1177 continue;
1178 }
Mathieu Chartiere401d142015-04-22 13:56:20 -07001179 // Presumably GC can happen when we are cross compiling, it should not cause performance
1180 // problems to do pointer size logic.
Vladimir Marko76649e82014-11-10 18:32:59 +00001181 MemberOffset field_offset = kIsStatic
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001182 ? klass->GetFirstReferenceStaticFieldOffset<kVerifyFlags, kReadBarrierOption>(
Mathieu Chartiere401d142015-04-22 13:56:20 -07001183 Runtime::Current()->GetClassLinker()->GetImagePointerSize())
Hiroshi Yamauchi5496f692016-02-17 13:29:59 -08001184 : klass->GetFirstReferenceInstanceFieldOffset<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartier059ef3d2015-08-18 13:54:21 -07001185 for (size_t i = 0u; i < num_reference_fields; ++i) {
Mathieu Chartier407f7022014-02-18 14:37:05 -08001186 // TODO: Do a simpler check?
Mathieu Chartier059ef3d2015-08-18 13:54:21 -07001187 if (field_offset.Uint32Value() != ClassOffset().Uint32Value()) {
Mathieu Chartier52e4b432014-06-10 11:22:31 -07001188 visitor(this, field_offset, kIsStatic);
Mathieu Chartier407f7022014-02-18 14:37:05 -08001189 }
Vladimir Marko76649e82014-11-10 18:32:59 +00001190 field_offset = MemberOffset(field_offset.Uint32Value() +
1191 sizeof(mirror::HeapReference<mirror::Object>));
Mathieu Chartier407f7022014-02-18 14:37:05 -08001192 }
1193 }
1194 }
1195}
1196
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001197template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption, typename Visitor>
Mathieu Chartier31e88222016-10-14 18:43:19 -07001198inline void Object::VisitInstanceFieldsReferences(ObjPtr<Class> klass, const Visitor& visitor) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001199 VisitFieldsReferences<false, kVerifyFlags, kReadBarrierOption>(
1200 klass->GetReferenceInstanceOffsets<kVerifyFlags>(), visitor);
Mathieu Chartier407f7022014-02-18 14:37:05 -08001201}
1202
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001203template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption, typename Visitor>
Mathieu Chartier31e88222016-10-14 18:43:19 -07001204inline void Object::VisitStaticFieldsReferences(ObjPtr<Class> klass, const Visitor& visitor) {
Mingyao Yang98d1cc82014-05-15 17:02:16 -07001205 DCHECK(!klass->IsTemp());
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001206 klass->VisitFieldsReferences<true, kVerifyFlags, kReadBarrierOption>(0, visitor);
Mathieu Chartier407f7022014-02-18 14:37:05 -08001207}
1208
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001209template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere4275c02015-08-06 15:34:15 -07001210inline bool Object::IsClassLoader() {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001211 return GetClass<kVerifyFlags, kReadBarrierOption>()->IsClassLoaderClass();
Mathieu Chartiere4275c02015-08-06 15:34:15 -07001212}
1213
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001214template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere4275c02015-08-06 15:34:15 -07001215inline mirror::ClassLoader* Object::AsClassLoader() {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001216 DCHECK((IsClassLoader<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartiere4275c02015-08-06 15:34:15 -07001217 return down_cast<mirror::ClassLoader*>(this);
1218}
1219
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001220template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Vladimir Marko05792b92015-08-03 11:56:49 +01001221inline bool Object::IsDexCache() {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001222 return GetClass<kVerifyFlags, kReadBarrierOption>()->IsDexCacheClass();
Vladimir Marko05792b92015-08-03 11:56:49 +01001223}
1224
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001225template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Vladimir Marko05792b92015-08-03 11:56:49 +01001226inline mirror::DexCache* Object::AsDexCache() {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001227 DCHECK((IsDexCache<kVerifyFlags, kReadBarrierOption>()));
Vladimir Marko05792b92015-08-03 11:56:49 +01001228 return down_cast<mirror::DexCache*>(this);
1229}
1230
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001231template <bool kVisitNativeRoots,
1232 VerifyObjectFlags kVerifyFlags,
1233 ReadBarrierOption kReadBarrierOption,
1234 typename Visitor,
1235 typename JavaLangRefVisitor>
Mathieu Chartier407f7022014-02-18 14:37:05 -08001236inline void Object::VisitReferences(const Visitor& visitor,
1237 const JavaLangRefVisitor& ref_visitor) {
Mathieu Chartier31e88222016-10-14 18:43:19 -07001238 ObjPtr<Class> klass = GetClass<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartier059ef3d2015-08-18 13:54:21 -07001239 visitor(this, ClassOffset(), false);
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001240 const uint32_t class_flags = klass->GetClassFlags<kVerifyNone>();
1241 if (LIKELY(class_flags == kClassFlagNormal)) {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001242 DCHECK((!klass->IsVariableSize<kVerifyFlags, kReadBarrierOption>()));
1243 VisitInstanceFieldsReferences<kVerifyFlags, kReadBarrierOption>(klass, visitor);
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001244 DCHECK((!klass->IsClassClass<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartier66c2d2d2015-08-25 14:32:32 -07001245 DCHECK(!klass->IsStringClass());
1246 DCHECK(!klass->IsClassLoaderClass());
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001247 DCHECK((!klass->IsArrayClass<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001248 } else {
1249 if ((class_flags & kClassFlagNoReferenceFields) == 0) {
1250 DCHECK(!klass->IsStringClass());
1251 if (class_flags == kClassFlagClass) {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001252 DCHECK((klass->IsClassClass<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartier31e88222016-10-14 18:43:19 -07001253 ObjPtr<Class> as_klass = AsClass<kVerifyNone, kReadBarrierOption>();
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001254 as_klass->VisitReferences<kVisitNativeRoots, kVerifyFlags, kReadBarrierOption>(klass,
1255 visitor);
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001256 } else if (class_flags == kClassFlagObjectArray) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001257 DCHECK((klass->IsObjectArrayClass<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001258 AsObjectArray<mirror::Object, kVerifyNone, kReadBarrierOption>()->VisitReferences(visitor);
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001259 } else if ((class_flags & kClassFlagReference) != 0) {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001260 VisitInstanceFieldsReferences<kVerifyFlags, kReadBarrierOption>(klass, visitor);
1261 ref_visitor(klass, AsReference<kVerifyFlags, kReadBarrierOption>());
Vladimir Marko05792b92015-08-03 11:56:49 +01001262 } else if (class_flags == kClassFlagDexCache) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001263 mirror::DexCache* const dex_cache = AsDexCache<kVerifyFlags, kReadBarrierOption>();
1264 dex_cache->VisitReferences<kVisitNativeRoots,
1265 kVerifyFlags,
1266 kReadBarrierOption>(klass, visitor);
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001267 } else {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001268 mirror::ClassLoader* const class_loader = AsClassLoader<kVerifyFlags, kReadBarrierOption>();
1269 class_loader->VisitReferences<kVisitNativeRoots,
1270 kVerifyFlags,
1271 kReadBarrierOption>(klass, visitor);
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001272 }
1273 } else if (kIsDebugBuild) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001274 CHECK((!klass->IsClassClass<kVerifyFlags, kReadBarrierOption>()));
1275 CHECK((!klass->IsObjectArrayClass<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001276 // String still has instance fields for reflection purposes but these don't exist in
1277 // actual string instances.
1278 if (!klass->IsStringClass()) {
1279 size_t total_reference_instance_fields = 0;
Mathieu Chartier31e88222016-10-14 18:43:19 -07001280 ObjPtr<Class> super_class = klass;
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001281 do {
1282 total_reference_instance_fields += super_class->NumReferenceInstanceFields();
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001283 super_class = super_class->GetSuperClass<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001284 } while (super_class != nullptr);
1285 // The only reference field should be the object's class. This field is handled at the
1286 // beginning of the function.
1287 CHECK_EQ(total_reference_instance_fields, 1u);
1288 }
Mathieu Chartier407f7022014-02-18 14:37:05 -08001289 }
1290 }
1291}
Ian Rogers2dd0e2c2013-01-24 12:42:14 -08001292} // namespace mirror
1293} // namespace art
1294
Brian Carlstromfc0e3212013-07-17 14:40:12 -07001295#endif // ART_RUNTIME_MIRROR_OBJECT_INL_H_