blob: 0f5cbb2b71ede2ab042657d725f2879bd5c18696 [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"
Hiroshi Yamauchi800ac2d2014-04-02 17:32:54 -070034#include "read_barrier-inl.h"
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -070035#include "reference.h"
Jeff Hao848f70a2014-01-15 13:49:50 -080036#include "runtime.h"
37#include "string-inl.h"
Ian Rogers05f30572013-02-20 12:13:11 -080038#include "throwable.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080039
40namespace art {
41namespace mirror {
42
Andreas Gampe542451c2016-07-26 09:02:02 -070043inline uint32_t Object::ClassSize(PointerSize pointer_size) {
Mingyao Yang98d1cc82014-05-15 17:02:16 -070044 uint32_t vtable_entries = kVTableLength;
Mathieu Chartiere401d142015-04-22 13:56:20 -070045 return Class::ComputeClassSize(true, vtable_entries, 0, 0, 0, 0, 0, pointer_size);
Mingyao Yang98d1cc82014-05-15 17:02:16 -070046}
47
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -070048template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -080049inline Class* Object::GetClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -070050 return GetFieldObject<Class, kVerifyFlags, kReadBarrierOption>(
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070051 OFFSET_OF_OBJECT_MEMBER(Object, klass_));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080052}
53
Mathieu Chartier4e305412014-02-19 10:54:44 -080054template<VerifyObjectFlags kVerifyFlags>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080055inline void Object::SetClass(Class* new_klass) {
Mathieu Chartier2cebb242015-04-21 16:50:40 -070056 // new_klass may be null prior to class linker initialization.
Ian Rogersef7d42f2014-01-06 12:55:46 -080057 // We don't mark the card as this occurs as part of object allocation. Not all objects have
58 // backing cards, such as large objects.
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +010059 // We use non transactional version since we can't undo this write. We also disable checking as
60 // we may run in transaction mode here.
Mathieu Chartier4e305412014-02-19 10:54:44 -080061 SetFieldObjectWithoutWriteBarrier<false, false,
62 static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis)>(
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070063 OFFSET_OF_OBJECT_MEMBER(Object, klass_), new_klass);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080064}
65
Andreas Gampe3b45ef22015-05-26 21:34:09 -070066template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartierbbd695c2014-04-16 09:48:48 -070067inline LockWord Object::GetLockWord(bool as_volatile) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070068 if (as_volatile) {
Andreas Gampe3b45ef22015-05-26 21:34:09 -070069 return LockWord(GetField32Volatile<kVerifyFlags>(OFFSET_OF_OBJECT_MEMBER(Object, monitor_)));
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070070 }
Andreas Gampe3b45ef22015-05-26 21:34:09 -070071 return LockWord(GetField32<kVerifyFlags>(OFFSET_OF_OBJECT_MEMBER(Object, monitor_)));
Ian Rogersd9c4fc92013-10-01 19:45:43 -070072}
73
Andreas Gampe3b45ef22015-05-26 21:34:09 -070074template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartierbbd695c2014-04-16 09:48:48 -070075inline void Object::SetLockWord(LockWord new_val, bool as_volatile) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +010076 // Force use of non-transactional mode and do not check.
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070077 if (as_volatile) {
Andreas Gampe3b45ef22015-05-26 21:34:09 -070078 SetField32Volatile<false, false, kVerifyFlags>(
79 OFFSET_OF_OBJECT_MEMBER(Object, monitor_), new_val.GetValue());
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070080 } else {
Andreas Gampe3b45ef22015-05-26 21:34:09 -070081 SetField32<false, false, kVerifyFlags>(
82 OFFSET_OF_OBJECT_MEMBER(Object, monitor_), new_val.GetValue());
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070083 }
Ian Rogersd9c4fc92013-10-01 19:45:43 -070084}
85
Ian Rogers228602f2014-07-10 02:07:54 -070086inline bool Object::CasLockWordWeakSequentiallyConsistent(LockWord old_val, LockWord new_val) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +010087 // Force use of non-transactional mode and do not check.
Ian Rogers228602f2014-07-10 02:07:54 -070088 return CasFieldWeakSequentiallyConsistent32<false, false>(
89 OFFSET_OF_OBJECT_MEMBER(Object, monitor_), old_val.GetValue(), new_val.GetValue());
Ian Rogersd9c4fc92013-10-01 19:45:43 -070090}
91
Hans Boehmd8434432014-07-11 09:56:07 -070092inline bool Object::CasLockWordWeakRelaxed(LockWord old_val, LockWord new_val) {
93 // Force use of non-transactional mode and do not check.
94 return CasFieldWeakRelaxed32<false, false>(
95 OFFSET_OF_OBJECT_MEMBER(Object, monitor_), old_val.GetValue(), new_val.GetValue());
96}
97
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -070098inline bool Object::CasLockWordWeakRelease(LockWord old_val, LockWord new_val) {
99 // Force use of non-transactional mode and do not check.
100 return CasFieldWeakRelease32<false, false>(
101 OFFSET_OF_OBJECT_MEMBER(Object, monitor_), old_val.GetValue(), new_val.GetValue());
102}
103
Ian Rogersd9c4fc92013-10-01 19:45:43 -0700104inline uint32_t Object::GetLockOwnerThreadId() {
105 return Monitor::GetLockOwnerThreadId(this);
Ian Rogers05f30572013-02-20 12:13:11 -0800106}
107
Mathieu Chartiere7e8a5f2014-02-14 16:59:41 -0800108inline mirror::Object* Object::MonitorEnter(Thread* self) {
Mathieu Chartiera704eda2016-07-13 09:53:35 -0700109 return Monitor::MonitorEnter(self, this, /*trylock*/false);
110}
111
112inline mirror::Object* Object::MonitorTryEnter(Thread* self) {
113 return Monitor::MonitorEnter(self, this, /*trylock*/true);
Ian Rogers05f30572013-02-20 12:13:11 -0800114}
115
116inline bool Object::MonitorExit(Thread* self) {
117 return Monitor::MonitorExit(self, this);
118}
119
120inline void Object::Notify(Thread* self) {
121 Monitor::Notify(self, this);
122}
123
124inline void Object::NotifyAll(Thread* self) {
125 Monitor::NotifyAll(self, this);
126}
127
128inline void Object::Wait(Thread* self) {
129 Monitor::Wait(self, this, 0, 0, true, kWaiting);
130}
131
132inline void Object::Wait(Thread* self, int64_t ms, int32_t ns) {
133 Monitor::Wait(self, this, ms, ns, true, kTimedWaiting);
134}
135
Hiroshi Yamauchi624468c2014-03-31 15:14:47 -0700136inline Object* Object::GetReadBarrierPointer() {
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700137#ifdef USE_BAKER_READ_BARRIER
138 DCHECK(kUseBakerReadBarrier);
139 return reinterpret_cast<Object*>(GetLockWord(false).ReadBarrierState());
140#elif USE_BROOKS_READ_BARRIER
141 DCHECK(kUseBrooksReadBarrier);
Hiroshi Yamauchi6a154a42014-05-02 14:26:13 -0700142 return GetFieldObject<Object, kVerifyNone, kWithoutReadBarrier>(
143 OFFSET_OF_OBJECT_MEMBER(Object, x_rb_ptr_));
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800144#else
145 LOG(FATAL) << "Unreachable";
Ian Rogers2c4257b2014-10-24 14:20:06 -0700146 UNREACHABLE();
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800147#endif
148}
149
Mathieu Chartierc381c362016-08-23 13:27:53 -0700150inline Object* Object::GetReadBarrierPointerAcquire() {
151#ifdef USE_BAKER_READ_BARRIER
152 DCHECK(kUseBakerReadBarrier);
153 LockWord lw(GetFieldAcquire<uint32_t>(OFFSET_OF_OBJECT_MEMBER(Object, monitor_)));
154 return reinterpret_cast<Object*>(lw.ReadBarrierState());
155#else
156 LOG(FATAL) << "Unreachable";
157 UNREACHABLE();
158#endif
159}
160
161
Mathieu Chartier36a270a2016-07-28 18:08:51 -0700162inline uint32_t Object::GetMarkBit() {
163#ifdef USE_READ_BARRIER
Mathieu Chartier36a270a2016-07-28 18:08:51 -0700164 return GetLockWord(false).MarkBitState();
165#else
166 LOG(FATAL) << "Unreachable";
167 UNREACHABLE();
168#endif
169}
170
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700171inline void Object::SetReadBarrierPointer(Object* rb_ptr) {
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700172#ifdef USE_BAKER_READ_BARRIER
173 DCHECK(kUseBakerReadBarrier);
174 DCHECK_EQ(reinterpret_cast<uint64_t>(rb_ptr) >> 32, 0U);
Mathieu Chartier36a270a2016-07-28 18:08:51 -0700175 DCHECK_NE(rb_ptr, ReadBarrier::BlackPtr()) << "Setting to black is not supported";
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700176 LockWord lw = GetLockWord(false);
177 lw.SetReadBarrierState(static_cast<uint32_t>(reinterpret_cast<uintptr_t>(rb_ptr)));
178 SetLockWord(lw, false);
179#elif USE_BROOKS_READ_BARRIER
180 DCHECK(kUseBrooksReadBarrier);
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800181 // We don't mark the card as this occurs as part of object allocation. Not all objects have
182 // backing cards, such as large objects.
183 SetFieldObjectWithoutWriteBarrier<false, false, kVerifyNone>(
Hiroshi Yamauchi6a154a42014-05-02 14:26:13 -0700184 OFFSET_OF_OBJECT_MEMBER(Object, x_rb_ptr_), rb_ptr);
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800185#else
186 LOG(FATAL) << "Unreachable";
Ian Rogers2c4257b2014-10-24 14:20:06 -0700187 UNREACHABLE();
Ian Rogers6a3c1fc2014-10-31 00:33:20 -0700188 UNUSED(rb_ptr);
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800189#endif
190}
191
Hiroshi Yamauchied70b4a2015-11-17 17:52:15 -0800192template<bool kCasRelease>
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700193inline bool Object::AtomicSetReadBarrierPointer(Object* expected_rb_ptr, Object* rb_ptr) {
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700194#ifdef USE_BAKER_READ_BARRIER
195 DCHECK(kUseBakerReadBarrier);
196 DCHECK_EQ(reinterpret_cast<uint64_t>(expected_rb_ptr) >> 32, 0U);
197 DCHECK_EQ(reinterpret_cast<uint64_t>(rb_ptr) >> 32, 0U);
Mathieu Chartier36a270a2016-07-28 18:08:51 -0700198 DCHECK_NE(expected_rb_ptr, ReadBarrier::BlackPtr()) << "Setting to black is not supported";
199 DCHECK_NE(rb_ptr, ReadBarrier::BlackPtr()) << "Setting to black is not supported";
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700200 LockWord expected_lw;
201 LockWord new_lw;
202 do {
203 LockWord lw = GetLockWord(false);
204 if (UNLIKELY(reinterpret_cast<Object*>(lw.ReadBarrierState()) != expected_rb_ptr)) {
205 // Lost the race.
206 return false;
207 }
208 expected_lw = lw;
209 expected_lw.SetReadBarrierState(
210 static_cast<uint32_t>(reinterpret_cast<uintptr_t>(expected_rb_ptr)));
211 new_lw = lw;
212 new_lw.SetReadBarrierState(static_cast<uint32_t>(reinterpret_cast<uintptr_t>(rb_ptr)));
Hiroshi Yamauchied70b4a2015-11-17 17:52:15 -0800213 // ConcurrentCopying::ProcessMarkStackRef uses this with kCasRelease == true.
214 // If kCasRelease == true, use a CAS release so that when GC updates all the fields of
215 // an object and then changes the object from gray to black, the field updates (stores) will be
216 // visible (won't be reordered after this CAS.)
217 } while (!(kCasRelease ?
218 CasLockWordWeakRelease(expected_lw, new_lw) :
219 CasLockWordWeakRelaxed(expected_lw, new_lw)));
Hiroshi Yamauchi60f63f52015-04-23 16:12:40 -0700220 return true;
221#elif USE_BROOKS_READ_BARRIER
222 DCHECK(kUseBrooksReadBarrier);
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700223 MemberOffset offset = OFFSET_OF_OBJECT_MEMBER(Object, x_rb_ptr_);
Ian Rogers13735952014-10-08 12:43:28 -0700224 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + offset.SizeValue();
Ian Rogers228602f2014-07-10 02:07:54 -0700225 Atomic<uint32_t>* atomic_rb_ptr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700226 HeapReference<Object> expected_ref(HeapReference<Object>::FromMirrorPtr(expected_rb_ptr));
227 HeapReference<Object> new_ref(HeapReference<Object>::FromMirrorPtr(rb_ptr));
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700228 do {
Ian Rogers228602f2014-07-10 02:07:54 -0700229 if (UNLIKELY(atomic_rb_ptr->LoadRelaxed() != expected_ref.reference_)) {
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700230 // Lost the race.
231 return false;
232 }
Ian Rogers228602f2014-07-10 02:07:54 -0700233 } while (!atomic_rb_ptr->CompareExchangeWeakSequentiallyConsistent(expected_ref.reference_,
234 new_ref.reference_));
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700235 return true;
236#else
Ian Rogers6a3c1fc2014-10-31 00:33:20 -0700237 UNUSED(expected_rb_ptr, rb_ptr);
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700238 LOG(FATAL) << "Unreachable";
Ian Rogers2c4257b2014-10-24 14:20:06 -0700239 UNREACHABLE();
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700240#endif
241}
242
Mathieu Chartier36a270a2016-07-28 18:08:51 -0700243inline bool Object::AtomicSetMarkBit(uint32_t expected_mark_bit, uint32_t mark_bit) {
244 LockWord expected_lw;
245 LockWord new_lw;
246 do {
247 LockWord lw = GetLockWord(false);
248 if (UNLIKELY(lw.MarkBitState() != expected_mark_bit)) {
249 // Lost the race.
250 return false;
251 }
252 expected_lw = lw;
253 new_lw = lw;
254 new_lw.SetMarkBitState(mark_bit);
255 // Since this is only set from the mutator, we can use the non release Cas.
256 } while (!CasLockWordWeakRelaxed(expected_lw, new_lw));
257 return true;
258}
259
260
Hiroshi Yamauchi624468c2014-03-31 15:14:47 -0700261inline void Object::AssertReadBarrierPointer() const {
Hiroshi Yamauchi800ac2d2014-04-02 17:32:54 -0700262 if (kUseBakerReadBarrier) {
263 Object* obj = const_cast<Object*>(this);
264 DCHECK(obj->GetReadBarrierPointer() == nullptr)
265 << "Bad Baker pointer: obj=" << reinterpret_cast<void*>(obj)
266 << " ptr=" << reinterpret_cast<void*>(obj->GetReadBarrierPointer());
Ian Rogers2c4257b2014-10-24 14:20:06 -0700267 } else {
268 CHECK(kUseBrooksReadBarrier);
Hiroshi Yamauchi800ac2d2014-04-02 17:32:54 -0700269 Object* obj = const_cast<Object*>(this);
270 DCHECK_EQ(obj, obj->GetReadBarrierPointer())
271 << "Bad Brooks pointer: obj=" << reinterpret_cast<void*>(obj)
272 << " ptr=" << reinterpret_cast<void*>(obj->GetReadBarrierPointer());
Hiroshi Yamauchi800ac2d2014-04-02 17:32:54 -0700273 }
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800274}
275
Mathieu Chartier4e305412014-02-19 10:54:44 -0800276template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800277inline bool Object::VerifierInstanceOf(Class* klass) {
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700278 DCHECK(klass != nullptr);
279 DCHECK(GetClass<kVerifyFlags>() != nullptr);
Jeff Haoa3faaf42013-09-03 19:07:00 -0700280 return klass->IsInterface() || InstanceOf(klass);
281}
282
Mathieu Chartier4e305412014-02-19 10:54:44 -0800283template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800284inline bool Object::InstanceOf(Class* klass) {
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700285 DCHECK(klass != nullptr);
286 DCHECK(GetClass<kVerifyNone>() != nullptr);
Mathieu Chartier4e305412014-02-19 10:54:44 -0800287 return klass->IsAssignableFrom(GetClass<kVerifyFlags>());
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800288}
289
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700290template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800291inline bool Object::IsClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700292 Class* java_lang_Class = GetClass<kVerifyFlags, kReadBarrierOption>()->
293 template GetClass<kVerifyFlags, kReadBarrierOption>();
294 return GetClass<static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis),
295 kReadBarrierOption>() == java_lang_Class;
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800296}
297
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700298template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800299inline Class* Object::AsClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700300 DCHECK((IsClass<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800301 return down_cast<Class*>(this);
302}
303
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800304template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800305inline bool Object::IsObjectArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800306 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800307 return IsArrayInstance<kVerifyFlags, kReadBarrierOption>() &&
308 !GetClass<kNewFlags, kReadBarrierOption>()->
309 template GetComponentType<kNewFlags, kReadBarrierOption>()->IsPrimitive();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800310}
311
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800312template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800313inline ObjectArray<T>* Object::AsObjectArray() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800314 DCHECK((IsObjectArray<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800315 return down_cast<ObjectArray<T>*>(this);
316}
317
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700318template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800319inline bool Object::IsArrayInstance() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700320 return GetClass<kVerifyFlags, kReadBarrierOption>()->
321 template IsArrayClass<kVerifyFlags, kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800322}
323
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800324template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800325inline bool Object::IsReferenceInstance() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800326 return GetClass<kVerifyFlags, kReadBarrierOption>()->IsTypeOfReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800327}
328
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800329template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700330inline Reference* Object::AsReference() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800331 DCHECK((IsReferenceInstance<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700332 return down_cast<Reference*>(this);
333}
334
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700335template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers05f30572013-02-20 12:13:11 -0800336inline Array* Object::AsArray() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700337 DCHECK((IsArrayInstance<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800338 return down_cast<Array*>(this);
339}
340
Mathieu Chartier4e305412014-02-19 10:54:44 -0800341template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800342inline BooleanArray* Object::AsBooleanArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800343 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
344 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
345 DCHECK(GetClass<kNewFlags>()->GetComponentType()->IsPrimitiveBoolean());
Ian Rogers05f30572013-02-20 12:13:11 -0800346 return down_cast<BooleanArray*>(this);
347}
348
Mathieu Chartier4e305412014-02-19 10:54:44 -0800349template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800350inline ByteArray* Object::AsByteArray() {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700351 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Mathieu Chartier4e305412014-02-19 10:54:44 -0800352 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
353 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveByte());
Ian Rogers05f30572013-02-20 12:13:11 -0800354 return down_cast<ByteArray*>(this);
355}
356
Mathieu Chartier4e305412014-02-19 10:54:44 -0800357template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800358inline ByteArray* Object::AsByteSizedArray() {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700359 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Mathieu Chartier4e305412014-02-19 10:54:44 -0800360 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
361 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveByte() ||
362 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveBoolean());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800363 return down_cast<ByteArray*>(this);
364}
365
Mathieu Chartier4e305412014-02-19 10:54:44 -0800366template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800367inline CharArray* Object::AsCharArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800368 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
369 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
370 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveChar());
Ian Rogers05f30572013-02-20 12:13:11 -0800371 return down_cast<CharArray*>(this);
372}
373
Mathieu Chartier4e305412014-02-19 10:54:44 -0800374template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800375inline ShortArray* Object::AsShortArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800376 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
377 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
378 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveShort());
Ian Rogers05f30572013-02-20 12:13:11 -0800379 return down_cast<ShortArray*>(this);
380}
381
Mathieu Chartier4e305412014-02-19 10:54:44 -0800382template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800383inline ShortArray* Object::AsShortSizedArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800384 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
385 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
386 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveShort() ||
387 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveChar());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800388 return down_cast<ShortArray*>(this);
389}
390
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800391template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700392inline bool Object::IsIntArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800393 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800394 mirror::Class* klass = GetClass<kVerifyFlags, kReadBarrierOption>();
395 mirror::Class* component_type = klass->GetComponentType<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartiere401d142015-04-22 13:56:20 -0700396 return component_type != nullptr && component_type->template IsPrimitiveInt<kNewFlags>();
397}
398
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800399template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700400inline IntArray* Object::AsIntArray() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800401 DCHECK((IsIntArray<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800402 return down_cast<IntArray*>(this);
403}
404
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800405template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700406inline bool Object::IsLongArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800407 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800408 mirror::Class* klass = GetClass<kVerifyFlags, kReadBarrierOption>();
409 mirror::Class* component_type = klass->GetComponentType<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartiere401d142015-04-22 13:56:20 -0700410 return component_type != nullptr && component_type->template IsPrimitiveLong<kNewFlags>();
411}
412
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800413template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700414inline LongArray* Object::AsLongArray() {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -0800415 DCHECK((IsLongArray<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800416 return down_cast<LongArray*>(this);
417}
418
Mathieu Chartier4e305412014-02-19 10:54:44 -0800419template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700420inline bool Object::IsFloatArray() {
421 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
422 auto* component_type = GetClass<kVerifyFlags>()->GetComponentType();
423 return component_type != nullptr && component_type->template IsPrimitiveFloat<kNewFlags>();
424}
425
426template<VerifyObjectFlags kVerifyFlags>
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100427inline FloatArray* Object::AsFloatArray() {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700428 DCHECK(IsFloatArray<kVerifyFlags>());
Mathieu Chartier4e305412014-02-19 10:54:44 -0800429 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
430 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
431 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveFloat());
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100432 return down_cast<FloatArray*>(this);
433}
434
Mathieu Chartier4e305412014-02-19 10:54:44 -0800435template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartiere401d142015-04-22 13:56:20 -0700436inline bool Object::IsDoubleArray() {
437 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
438 auto* component_type = GetClass<kVerifyFlags>()->GetComponentType();
439 return component_type != nullptr && component_type->template IsPrimitiveDouble<kNewFlags>();
440}
441
442template<VerifyObjectFlags kVerifyFlags>
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100443inline DoubleArray* Object::AsDoubleArray() {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700444 DCHECK(IsDoubleArray<kVerifyFlags>());
Mathieu Chartier4e305412014-02-19 10:54:44 -0800445 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
446 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
447 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveDouble());
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100448 return down_cast<DoubleArray*>(this);
449}
450
Jeff Hao848f70a2014-01-15 13:49:50 -0800451template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
452inline bool Object::IsString() {
453 return GetClass<kVerifyFlags, kReadBarrierOption>()->IsStringClass();
454}
455
456template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers05f30572013-02-20 12:13:11 -0800457inline String* Object::AsString() {
Jeff Hao848f70a2014-01-15 13:49:50 -0800458 DCHECK((IsString<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800459 return down_cast<String*>(this);
460}
461
Mathieu Chartier4e305412014-02-19 10:54:44 -0800462template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800463inline Throwable* Object::AsThrowable() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800464 DCHECK(GetClass<kVerifyFlags>()->IsThrowableClass());
Ian Rogers05f30572013-02-20 12:13:11 -0800465 return down_cast<Throwable*>(this);
466}
467
Mathieu Chartier4e305412014-02-19 10:54:44 -0800468template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800469inline bool Object::IsWeakReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800470 return GetClass<kVerifyFlags>()->IsWeakReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800471}
472
Mathieu Chartier4e305412014-02-19 10:54:44 -0800473template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800474inline bool Object::IsSoftReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800475 return GetClass<kVerifyFlags>()->IsSoftReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800476}
477
Mathieu Chartier4e305412014-02-19 10:54:44 -0800478template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800479inline bool Object::IsFinalizerReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800480 return GetClass<kVerifyFlags>()->IsFinalizerReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800481}
482
Mathieu Chartier4e305412014-02-19 10:54:44 -0800483template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700484inline FinalizerReference* Object::AsFinalizerReference() {
485 DCHECK(IsFinalizerReferenceInstance<kVerifyFlags>());
486 return down_cast<FinalizerReference*>(this);
487}
488
489template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800490inline bool Object::IsPhantomReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800491 return GetClass<kVerifyFlags>()->IsPhantomReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800492}
493
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700494template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800495inline size_t Object::SizeOf() {
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800496 size_t result;
Mathieu Chartier4e305412014-02-19 10:54:44 -0800497 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700498 if (IsArrayInstance<kVerifyFlags, kReadBarrierOption>()) {
499 result = AsArray<kNewFlags, kReadBarrierOption>()->
500 template SizeOf<kNewFlags, kReadBarrierOption>();
501 } else if (IsClass<kNewFlags, kReadBarrierOption>()) {
502 result = AsClass<kNewFlags, kReadBarrierOption>()->
503 template SizeOf<kNewFlags, kReadBarrierOption>();
Jeff Hao848f70a2014-01-15 13:49:50 -0800504 } else if (GetClass<kNewFlags, kReadBarrierOption>()->IsStringClass()) {
505 result = AsString<kNewFlags, kReadBarrierOption>()->
506 template SizeOf<kNewFlags>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800507 } else {
Hiroshi Yamauchi25023c72014-05-09 11:45:53 -0700508 result = GetClass<kNewFlags, kReadBarrierOption>()->
509 template GetObjectSize<kNewFlags, kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800510 }
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700511 DCHECK_GE(result, sizeof(Object))
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700512 << " class=" << PrettyTypeOf(GetClass<kNewFlags, kReadBarrierOption>());
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800513 return result;
514}
515
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700516template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
Fred Shih37f05ef2014-07-16 18:38:08 -0700517inline uint8_t Object::GetFieldBoolean(MemberOffset field_offset) {
518 if (kVerifyFlags & kVerifyThis) {
519 VerifyObject(this);
520 }
521 return GetField<uint8_t, kIsVolatile>(field_offset);
522}
523
524template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
525inline int8_t Object::GetFieldByte(MemberOffset field_offset) {
526 if (kVerifyFlags & kVerifyThis) {
527 VerifyObject(this);
528 }
529 return GetField<int8_t, kIsVolatile>(field_offset);
530}
531
532template<VerifyObjectFlags kVerifyFlags>
533inline uint8_t Object::GetFieldBooleanVolatile(MemberOffset field_offset) {
534 return GetFieldBoolean<kVerifyFlags, true>(field_offset);
535}
536
537template<VerifyObjectFlags kVerifyFlags>
538inline int8_t Object::GetFieldByteVolatile(MemberOffset field_offset) {
539 return GetFieldByte<kVerifyFlags, true>(field_offset);
540}
541
542template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
543 bool kIsVolatile>
544inline void Object::SetFieldBoolean(MemberOffset field_offset, uint8_t new_value)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700545 REQUIRES_SHARED(Locks::mutator_lock_) {
Fred Shih37f05ef2014-07-16 18:38:08 -0700546 if (kCheckTransaction) {
547 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
548 }
549 if (kTransactionActive) {
550 Runtime::Current()->RecordWriteFieldBoolean(this, field_offset,
551 GetFieldBoolean<kVerifyFlags, kIsVolatile>(field_offset),
552 kIsVolatile);
553 }
554 if (kVerifyFlags & kVerifyThis) {
555 VerifyObject(this);
556 }
557 SetField<uint8_t, kIsVolatile>(field_offset, new_value);
558}
559
560template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
561 bool kIsVolatile>
562inline void Object::SetFieldByte(MemberOffset field_offset, int8_t new_value)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700563 REQUIRES_SHARED(Locks::mutator_lock_) {
Fred Shih37f05ef2014-07-16 18:38:08 -0700564 if (kCheckTransaction) {
565 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
566 }
567 if (kTransactionActive) {
568 Runtime::Current()->RecordWriteFieldByte(this, field_offset,
569 GetFieldByte<kVerifyFlags, kIsVolatile>(field_offset),
570 kIsVolatile);
571 }
572 if (kVerifyFlags & kVerifyThis) {
573 VerifyObject(this);
574 }
575 SetField<int8_t, kIsVolatile>(field_offset, new_value);
576}
577
578template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
579inline void Object::SetFieldBooleanVolatile(MemberOffset field_offset, uint8_t new_value) {
580 return SetFieldBoolean<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
581 field_offset, new_value);
582}
583
584template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
585inline void Object::SetFieldByteVolatile(MemberOffset field_offset, int8_t new_value) {
586 return SetFieldByte<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
587 field_offset, new_value);
588}
589
590template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
591inline uint16_t Object::GetFieldChar(MemberOffset field_offset) {
592 if (kVerifyFlags & kVerifyThis) {
593 VerifyObject(this);
594 }
595 return GetField<uint16_t, kIsVolatile>(field_offset);
596}
597
598template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
599inline int16_t Object::GetFieldShort(MemberOffset field_offset) {
600 if (kVerifyFlags & kVerifyThis) {
601 VerifyObject(this);
602 }
603 return GetField<int16_t, kIsVolatile>(field_offset);
604}
605
606template<VerifyObjectFlags kVerifyFlags>
607inline uint16_t Object::GetFieldCharVolatile(MemberOffset field_offset) {
608 return GetFieldChar<kVerifyFlags, true>(field_offset);
609}
610
611template<VerifyObjectFlags kVerifyFlags>
612inline int16_t Object::GetFieldShortVolatile(MemberOffset field_offset) {
613 return GetFieldShort<kVerifyFlags, true>(field_offset);
614}
615
616template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
617 bool kIsVolatile>
618inline void Object::SetFieldChar(MemberOffset field_offset, uint16_t new_value) {
619 if (kCheckTransaction) {
620 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
621 }
622 if (kTransactionActive) {
623 Runtime::Current()->RecordWriteFieldChar(this, field_offset,
624 GetFieldChar<kVerifyFlags, kIsVolatile>(field_offset),
625 kIsVolatile);
626 }
627 if (kVerifyFlags & kVerifyThis) {
628 VerifyObject(this);
629 }
630 SetField<uint16_t, kIsVolatile>(field_offset, new_value);
631}
632
633template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
634 bool kIsVolatile>
635inline void Object::SetFieldShort(MemberOffset field_offset, int16_t new_value) {
636 if (kCheckTransaction) {
637 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
638 }
639 if (kTransactionActive) {
640 Runtime::Current()->RecordWriteFieldChar(this, field_offset,
641 GetFieldShort<kVerifyFlags, kIsVolatile>(field_offset),
642 kIsVolatile);
643 }
644 if (kVerifyFlags & kVerifyThis) {
645 VerifyObject(this);
646 }
647 SetField<int16_t, kIsVolatile>(field_offset, new_value);
648}
649
650template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
651inline void Object::SetFieldCharVolatile(MemberOffset field_offset, uint16_t new_value) {
652 return SetFieldChar<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
653 field_offset, new_value);
654}
655
656template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
657inline void Object::SetFieldShortVolatile(MemberOffset field_offset, int16_t new_value) {
658 return SetFieldShort<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(
659 field_offset, new_value);
660}
661
662template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700663inline int32_t Object::GetField32(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800664 if (kVerifyFlags & kVerifyThis) {
665 VerifyObject(this);
666 }
Fred Shih37f05ef2014-07-16 18:38:08 -0700667 return GetField<int32_t, kIsVolatile>(field_offset);
Ian Rogersb122a4b2013-11-19 18:00:50 -0800668}
669
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700670template<VerifyObjectFlags kVerifyFlags>
671inline int32_t Object::GetField32Volatile(MemberOffset field_offset) {
672 return GetField32<kVerifyFlags, true>(field_offset);
673}
674
675template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
676 bool kIsVolatile>
677inline void Object::SetField32(MemberOffset field_offset, int32_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100678 if (kCheckTransaction) {
679 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
680 }
681 if (kTransactionActive) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700682 Runtime::Current()->RecordWriteField32(this, field_offset,
683 GetField32<kVerifyFlags, kIsVolatile>(field_offset),
684 kIsVolatile);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100685 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800686 if (kVerifyFlags & kVerifyThis) {
Ian Rogersb122a4b2013-11-19 18:00:50 -0800687 VerifyObject(this);
688 }
Fred Shih37f05ef2014-07-16 18:38:08 -0700689 SetField<int32_t, kIsVolatile>(field_offset, new_value);
Ian Rogersb122a4b2013-11-19 18:00:50 -0800690}
691
Mathieu Chartier4e305412014-02-19 10:54:44 -0800692template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700693inline void Object::SetField32Volatile(MemberOffset field_offset, int32_t new_value) {
694 SetField32<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset, new_value);
695}
696
Hans Boehmd8434432014-07-11 09:56:07 -0700697// TODO: Pass memory_order_ and strong/weak as arguments to avoid code duplication?
698
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700699template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -0700700inline bool Object::CasFieldWeakSequentiallyConsistent32(MemberOffset field_offset,
701 int32_t old_value, int32_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100702 if (kCheckTransaction) {
703 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
704 }
705 if (kTransactionActive) {
706 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
707 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800708 if (kVerifyFlags & kVerifyThis) {
709 VerifyObject(this);
710 }
Ian Rogers13735952014-10-08 12:43:28 -0700711 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogers228602f2014-07-10 02:07:54 -0700712 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
Hans Boehm30359612014-05-21 17:46:23 -0700713
Ian Rogers228602f2014-07-10 02:07:54 -0700714 return atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_value, new_value);
Ian Rogersd9c4fc92013-10-01 19:45:43 -0700715}
716
Hans Boehmd8434432014-07-11 09:56:07 -0700717template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
718inline bool Object::CasFieldWeakRelaxed32(MemberOffset field_offset,
719 int32_t old_value, int32_t new_value) {
720 if (kCheckTransaction) {
721 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
722 }
723 if (kTransactionActive) {
724 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
725 }
726 if (kVerifyFlags & kVerifyThis) {
727 VerifyObject(this);
728 }
Ian Rogers13735952014-10-08 12:43:28 -0700729 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Hans Boehmd8434432014-07-11 09:56:07 -0700730 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
731
732 return atomic_addr->CompareExchangeWeakRelaxed(old_value, new_value);
733}
734
735template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -0700736inline bool Object::CasFieldWeakRelease32(MemberOffset field_offset,
737 int32_t old_value, int32_t new_value) {
738 if (kCheckTransaction) {
739 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
740 }
741 if (kTransactionActive) {
742 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
743 }
744 if (kVerifyFlags & kVerifyThis) {
745 VerifyObject(this);
746 }
747 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
748 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
749
750 return atomic_addr->CompareExchangeWeakRelease(old_value, new_value);
751}
752
753template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Hans Boehmd8434432014-07-11 09:56:07 -0700754inline bool Object::CasFieldStrongSequentiallyConsistent32(MemberOffset field_offset,
755 int32_t old_value, int32_t new_value) {
756 if (kCheckTransaction) {
757 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
758 }
759 if (kTransactionActive) {
760 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
761 }
762 if (kVerifyFlags & kVerifyThis) {
763 VerifyObject(this);
764 }
Ian Rogers13735952014-10-08 12:43:28 -0700765 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Hans Boehmd8434432014-07-11 09:56:07 -0700766 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
767
768 return atomic_addr->CompareExchangeStrongSequentiallyConsistent(old_value, new_value);
769}
770
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700771template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
772inline int64_t Object::GetField64(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800773 if (kVerifyFlags & kVerifyThis) {
774 VerifyObject(this);
775 }
Fred Shih37f05ef2014-07-16 18:38:08 -0700776 return GetField<int64_t, kIsVolatile>(field_offset);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800777}
778
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700779template<VerifyObjectFlags kVerifyFlags>
780inline int64_t Object::GetField64Volatile(MemberOffset field_offset) {
781 return GetField64<kVerifyFlags, true>(field_offset);
782}
783
784template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
785 bool kIsVolatile>
786inline void Object::SetField64(MemberOffset field_offset, int64_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100787 if (kCheckTransaction) {
788 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
789 }
790 if (kTransactionActive) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700791 Runtime::Current()->RecordWriteField64(this, field_offset,
792 GetField64<kVerifyFlags, kIsVolatile>(field_offset),
793 kIsVolatile);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100794 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800795 if (kVerifyFlags & kVerifyThis) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800796 VerifyObject(this);
797 }
Fred Shih37f05ef2014-07-16 18:38:08 -0700798 SetField<int64_t, kIsVolatile>(field_offset, new_value);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800799}
800
Mathieu Chartier4e305412014-02-19 10:54:44 -0800801template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700802inline void Object::SetField64Volatile(MemberOffset field_offset, int64_t new_value) {
803 return SetField64<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
804 new_value);
805}
806
Fred Shih37f05ef2014-07-16 18:38:08 -0700807template<typename kSize, bool kIsVolatile>
808inline void Object::SetField(MemberOffset field_offset, kSize new_value) {
Ian Rogers13735952014-10-08 12:43:28 -0700809 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Fred Shih37f05ef2014-07-16 18:38:08 -0700810 kSize* addr = reinterpret_cast<kSize*>(raw_addr);
811 if (kIsVolatile) {
812 reinterpret_cast<Atomic<kSize>*>(addr)->StoreSequentiallyConsistent(new_value);
813 } else {
814 reinterpret_cast<Atomic<kSize>*>(addr)->StoreJavaData(new_value);
815 }
816}
817
818template<typename kSize, bool kIsVolatile>
819inline kSize Object::GetField(MemberOffset field_offset) {
Ian Rogers13735952014-10-08 12:43:28 -0700820 const uint8_t* raw_addr = reinterpret_cast<const uint8_t*>(this) + field_offset.Int32Value();
Fred Shih37f05ef2014-07-16 18:38:08 -0700821 const kSize* addr = reinterpret_cast<const kSize*>(raw_addr);
822 if (kIsVolatile) {
823 return reinterpret_cast<const Atomic<kSize>*>(addr)->LoadSequentiallyConsistent();
824 } else {
825 return reinterpret_cast<const Atomic<kSize>*>(addr)->LoadJavaData();
826 }
827}
828
Mathieu Chartierc381c362016-08-23 13:27:53 -0700829template<typename kSize>
830inline kSize Object::GetFieldAcquire(MemberOffset field_offset) {
831 const uint8_t* raw_addr = reinterpret_cast<const uint8_t*>(this) + field_offset.Int32Value();
832 const kSize* addr = reinterpret_cast<const kSize*>(raw_addr);
833 return reinterpret_cast<const Atomic<kSize>*>(addr)->LoadAcquire();
834}
835
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700836template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -0700837inline bool Object::CasFieldWeakSequentiallyConsistent64(MemberOffset field_offset,
838 int64_t old_value, int64_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100839 if (kCheckTransaction) {
840 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
841 }
842 if (kTransactionActive) {
843 Runtime::Current()->RecordWriteField64(this, field_offset, old_value, true);
844 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800845 if (kVerifyFlags & kVerifyThis) {
846 VerifyObject(this);
847 }
Ian Rogers13735952014-10-08 12:43:28 -0700848 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogers228602f2014-07-10 02:07:54 -0700849 Atomic<int64_t>* atomic_addr = reinterpret_cast<Atomic<int64_t>*>(raw_addr);
850 return atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_value, new_value);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800851}
852
Hans Boehmd8434432014-07-11 09:56:07 -0700853template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
854inline bool Object::CasFieldStrongSequentiallyConsistent64(MemberOffset field_offset,
855 int64_t old_value, int64_t new_value) {
856 if (kCheckTransaction) {
857 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
858 }
859 if (kTransactionActive) {
860 Runtime::Current()->RecordWriteField64(this, field_offset, old_value, true);
861 }
862 if (kVerifyFlags & kVerifyThis) {
863 VerifyObject(this);
864 }
Ian Rogers13735952014-10-08 12:43:28 -0700865 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Hans Boehmd8434432014-07-11 09:56:07 -0700866 Atomic<int64_t>* atomic_addr = reinterpret_cast<Atomic<int64_t>*>(raw_addr);
867 return atomic_addr->CompareExchangeStrongSequentiallyConsistent(old_value, new_value);
868}
869
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700870template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption,
871 bool kIsVolatile>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700872inline T* Object::GetFieldObject(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800873 if (kVerifyFlags & kVerifyThis) {
874 VerifyObject(this);
875 }
Ian Rogers13735952014-10-08 12:43:28 -0700876 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogersef7d42f2014-01-06 12:55:46 -0800877 HeapReference<T>* objref_addr = reinterpret_cast<HeapReference<T>*>(raw_addr);
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700878 T* result = ReadBarrier::Barrier<T, kReadBarrierOption>(this, field_offset, objref_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700879 if (kIsVolatile) {
Hans Boehm30359612014-05-21 17:46:23 -0700880 // TODO: Refactor to use a SequentiallyConsistent load instead.
881 QuasiAtomic::ThreadFenceAcquire(); // Ensure visibility of operations preceding store.
Ian Rogersef7d42f2014-01-06 12:55:46 -0800882 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800883 if (kVerifyFlags & kVerifyReads) {
884 VerifyObject(result);
885 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800886 return result;
887}
888
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700889template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700890inline T* Object::GetFieldObjectVolatile(MemberOffset field_offset) {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700891 return GetFieldObject<T, kVerifyFlags, kReadBarrierOption, true>(field_offset);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700892}
893
894template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
895 bool kIsVolatile>
896inline void Object::SetFieldObjectWithoutWriteBarrier(MemberOffset field_offset,
897 Object* new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100898 if (kCheckTransaction) {
899 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
900 }
901 if (kTransactionActive) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700902 mirror::Object* obj;
903 if (kIsVolatile) {
904 obj = GetFieldObjectVolatile<Object>(field_offset);
905 } else {
906 obj = GetFieldObject<Object>(field_offset);
907 }
908 Runtime::Current()->RecordWriteFieldReference(this, field_offset, obj, true);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100909 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800910 if (kVerifyFlags & kVerifyThis) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800911 VerifyObject(this);
912 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800913 if (kVerifyFlags & kVerifyWrites) {
914 VerifyObject(new_value);
915 }
Ian Rogers13735952014-10-08 12:43:28 -0700916 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogersef7d42f2014-01-06 12:55:46 -0800917 HeapReference<Object>* objref_addr = reinterpret_cast<HeapReference<Object>*>(raw_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700918 if (kIsVolatile) {
Hans Boehm30359612014-05-21 17:46:23 -0700919 // TODO: Refactor to use a SequentiallyConsistent store instead.
920 QuasiAtomic::ThreadFenceRelease(); // Ensure that prior accesses are visible before store.
Ian Rogersef7d42f2014-01-06 12:55:46 -0800921 objref_addr->Assign(new_value);
Hans Boehm30359612014-05-21 17:46:23 -0700922 QuasiAtomic::ThreadFenceSequentiallyConsistent();
923 // Ensure this store occurs before any volatile loads.
Ian Rogersef7d42f2014-01-06 12:55:46 -0800924 } else {
925 objref_addr->Assign(new_value);
926 }
927}
928
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700929template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
930 bool kIsVolatile>
931inline void Object::SetFieldObject(MemberOffset field_offset, Object* new_value) {
932 SetFieldObjectWithoutWriteBarrier<kTransactionActive, kCheckTransaction, kVerifyFlags,
933 kIsVolatile>(field_offset, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800934 if (new_value != nullptr) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800935 Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700936 // TODO: Check field assignment could theoretically cause thread suspension, TODO: fix this.
937 CheckFieldAssignment(field_offset, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800938 }
939}
940
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700941template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
942inline void Object::SetFieldObjectVolatile(MemberOffset field_offset, Object* new_value) {
943 SetFieldObject<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
944 new_value);
945}
946
Mathieu Chartier4e305412014-02-19 10:54:44 -0800947template <VerifyObjectFlags kVerifyFlags>
948inline HeapReference<Object>* Object::GetFieldObjectReferenceAddr(MemberOffset field_offset) {
949 if (kVerifyFlags & kVerifyThis) {
950 VerifyObject(this);
951 }
Ian Rogers13735952014-10-08 12:43:28 -0700952 return reinterpret_cast<HeapReference<Object>*>(reinterpret_cast<uint8_t*>(this) +
Mathieu Chartier4e305412014-02-19 10:54:44 -0800953 field_offset.Int32Value());
954}
955
956template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -0700957inline bool Object::CasFieldWeakSequentiallyConsistentObject(MemberOffset field_offset,
958 Object* old_value, Object* new_value) {
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800959 bool success = CasFieldWeakSequentiallyConsistentObjectWithoutWriteBarrier<
960 kTransactionActive, kCheckTransaction, kVerifyFlags>(field_offset, old_value, new_value);
961 if (success) {
962 Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
963 }
964 return success;
965}
966
967template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
968inline bool Object::CasFieldWeakSequentiallyConsistentObjectWithoutWriteBarrier(
969 MemberOffset field_offset, Object* old_value, Object* new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100970 if (kCheckTransaction) {
971 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
972 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800973 if (kVerifyFlags & kVerifyThis) {
974 VerifyObject(this);
975 }
976 if (kVerifyFlags & kVerifyWrites) {
977 VerifyObject(new_value);
978 }
979 if (kVerifyFlags & kVerifyReads) {
980 VerifyObject(old_value);
981 }
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100982 if (kTransactionActive) {
983 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
984 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800985 HeapReference<Object> old_ref(HeapReference<Object>::FromMirrorPtr(old_value));
986 HeapReference<Object> new_ref(HeapReference<Object>::FromMirrorPtr(new_value));
Ian Rogers13735952014-10-08 12:43:28 -0700987 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Ian Rogers228602f2014-07-10 02:07:54 -0700988 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
989
990 bool success = atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_ref.reference_,
991 new_ref.reference_);
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800992 return success;
993}
Ian Rogers228602f2014-07-10 02:07:54 -0700994
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800995template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
996inline bool Object::CasFieldStrongSequentiallyConsistentObject(MemberOffset field_offset,
997 Object* old_value, Object* new_value) {
998 bool success = CasFieldStrongSequentiallyConsistentObjectWithoutWriteBarrier<
999 kTransactionActive, kCheckTransaction, kVerifyFlags>(field_offset, old_value, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -08001000 if (success) {
1001 Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
1002 }
1003 return success;
1004}
1005
Hans Boehmd8434432014-07-11 09:56:07 -07001006template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -08001007inline bool Object::CasFieldStrongSequentiallyConsistentObjectWithoutWriteBarrier(
1008 MemberOffset field_offset, Object* old_value, Object* new_value) {
Hans Boehmd8434432014-07-11 09:56:07 -07001009 if (kCheckTransaction) {
1010 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
1011 }
1012 if (kVerifyFlags & kVerifyThis) {
1013 VerifyObject(this);
1014 }
1015 if (kVerifyFlags & kVerifyWrites) {
1016 VerifyObject(new_value);
1017 }
1018 if (kVerifyFlags & kVerifyReads) {
1019 VerifyObject(old_value);
1020 }
1021 if (kTransactionActive) {
1022 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
1023 }
1024 HeapReference<Object> old_ref(HeapReference<Object>::FromMirrorPtr(old_value));
1025 HeapReference<Object> new_ref(HeapReference<Object>::FromMirrorPtr(new_value));
Ian Rogers13735952014-10-08 12:43:28 -07001026 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
Hans Boehmd8434432014-07-11 09:56:07 -07001027 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
1028
1029 bool success = atomic_addr->CompareExchangeStrongSequentiallyConsistent(old_ref.reference_,
1030 new_ref.reference_);
Hans Boehmd8434432014-07-11 09:56:07 -07001031 return success;
1032}
1033
Hiroshi Yamauchifed3e2f2015-10-20 11:11:56 -07001034template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
1035inline bool Object::CasFieldWeakRelaxedObjectWithoutWriteBarrier(
1036 MemberOffset field_offset, Object* old_value, Object* new_value) {
1037 if (kCheckTransaction) {
1038 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
1039 }
1040 if (kVerifyFlags & kVerifyThis) {
1041 VerifyObject(this);
1042 }
1043 if (kVerifyFlags & kVerifyWrites) {
1044 VerifyObject(new_value);
1045 }
1046 if (kVerifyFlags & kVerifyReads) {
1047 VerifyObject(old_value);
1048 }
1049 if (kTransactionActive) {
1050 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
1051 }
1052 HeapReference<Object> old_ref(HeapReference<Object>::FromMirrorPtr(old_value));
1053 HeapReference<Object> new_ref(HeapReference<Object>::FromMirrorPtr(new_value));
1054 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
1055 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
1056
1057 bool success = atomic_addr->CompareExchangeWeakRelaxed(old_ref.reference_,
1058 new_ref.reference_);
1059 return success;
1060}
1061
1062template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
1063inline bool Object::CasFieldStrongRelaxedObjectWithoutWriteBarrier(
1064 MemberOffset field_offset, Object* old_value, Object* new_value) {
1065 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 }
1080 HeapReference<Object> old_ref(HeapReference<Object>::FromMirrorPtr(old_value));
1081 HeapReference<Object> new_ref(HeapReference<Object>::FromMirrorPtr(new_value));
1082 uint8_t* raw_addr = reinterpret_cast<uint8_t*>(this) + field_offset.Int32Value();
1083 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
1084
1085 bool success = atomic_addr->CompareExchangeStrongRelaxed(old_ref.reference_,
1086 new_ref.reference_);
1087 return success;
1088}
1089
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001090template<bool kIsStatic,
1091 VerifyObjectFlags kVerifyFlags,
1092 ReadBarrierOption kReadBarrierOption,
1093 typename Visitor>
Mathieu Chartier407f7022014-02-18 14:37:05 -08001094inline void Object::VisitFieldsReferences(uint32_t ref_offsets, const Visitor& visitor) {
Ian Rogerscdc1aaf2014-10-09 13:21:38 -07001095 if (!kIsStatic && (ref_offsets != mirror::Class::kClassWalkSuper)) {
1096 // Instance fields and not the slow-path.
Ian Rogerscdc1aaf2014-10-09 13:21:38 -07001097 uint32_t field_offset = mirror::kObjectHeaderSize;
Mathieu Chartier407f7022014-02-18 14:37:05 -08001098 while (ref_offsets != 0) {
Ian Rogerscdc1aaf2014-10-09 13:21:38 -07001099 if ((ref_offsets & 1) != 0) {
1100 visitor(this, MemberOffset(field_offset), kIsStatic);
1101 }
1102 ref_offsets >>= 1;
1103 field_offset += sizeof(mirror::HeapReference<mirror::Object>);
Mathieu Chartier407f7022014-02-18 14:37:05 -08001104 }
1105 } else {
Mingyao Yangfaff0f02014-09-10 12:03:22 -07001106 // There is no reference offset bitmap. In the non-static case, walk up the class
Mathieu Chartier407f7022014-02-18 14:37:05 -08001107 // inheritance hierarchy and find reference offsets the hard way. In the static case, just
1108 // consider this class.
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001109 for (mirror::Class* klass = kIsStatic
1110 ? AsClass<kVerifyFlags, kReadBarrierOption>()
1111 : GetClass<kVerifyFlags, kReadBarrierOption>();
1112 klass != nullptr;
1113 klass = kIsStatic ? nullptr : klass->GetSuperClass<kVerifyFlags, kReadBarrierOption>()) {
1114 const size_t num_reference_fields =
Mathieu Chartier407f7022014-02-18 14:37:05 -08001115 kIsStatic ? klass->NumReferenceStaticFields() : klass->NumReferenceInstanceFields();
Vladimir Marko76649e82014-11-10 18:32:59 +00001116 if (num_reference_fields == 0u) {
1117 continue;
1118 }
Mathieu Chartiere401d142015-04-22 13:56:20 -07001119 // Presumably GC can happen when we are cross compiling, it should not cause performance
1120 // problems to do pointer size logic.
Vladimir Marko76649e82014-11-10 18:32:59 +00001121 MemberOffset field_offset = kIsStatic
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001122 ? klass->GetFirstReferenceStaticFieldOffset<kVerifyFlags, kReadBarrierOption>(
Mathieu Chartiere401d142015-04-22 13:56:20 -07001123 Runtime::Current()->GetClassLinker()->GetImagePointerSize())
Hiroshi Yamauchi5496f692016-02-17 13:29:59 -08001124 : klass->GetFirstReferenceInstanceFieldOffset<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartier059ef3d2015-08-18 13:54:21 -07001125 for (size_t i = 0u; i < num_reference_fields; ++i) {
Mathieu Chartier407f7022014-02-18 14:37:05 -08001126 // TODO: Do a simpler check?
Mathieu Chartier059ef3d2015-08-18 13:54:21 -07001127 if (field_offset.Uint32Value() != ClassOffset().Uint32Value()) {
Mathieu Chartier52e4b432014-06-10 11:22:31 -07001128 visitor(this, field_offset, kIsStatic);
Mathieu Chartier407f7022014-02-18 14:37:05 -08001129 }
Vladimir Marko76649e82014-11-10 18:32:59 +00001130 field_offset = MemberOffset(field_offset.Uint32Value() +
1131 sizeof(mirror::HeapReference<mirror::Object>));
Mathieu Chartier407f7022014-02-18 14:37:05 -08001132 }
1133 }
1134 }
1135}
1136
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001137template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption, typename Visitor>
Mathieu Chartier407f7022014-02-18 14:37:05 -08001138inline void Object::VisitInstanceFieldsReferences(mirror::Class* klass, const Visitor& visitor) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001139 VisitFieldsReferences<false, kVerifyFlags, kReadBarrierOption>(
1140 klass->GetReferenceInstanceOffsets<kVerifyFlags>(), visitor);
Mathieu Chartier407f7022014-02-18 14:37:05 -08001141}
1142
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001143template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption, typename Visitor>
Mathieu Chartier407f7022014-02-18 14:37:05 -08001144inline void Object::VisitStaticFieldsReferences(mirror::Class* klass, const Visitor& visitor) {
Mingyao Yang98d1cc82014-05-15 17:02:16 -07001145 DCHECK(!klass->IsTemp());
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001146 klass->VisitFieldsReferences<true, kVerifyFlags, kReadBarrierOption>(0, visitor);
Mathieu Chartier407f7022014-02-18 14:37:05 -08001147}
1148
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001149template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere4275c02015-08-06 15:34:15 -07001150inline bool Object::IsClassLoader() {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001151 return GetClass<kVerifyFlags, kReadBarrierOption>()->IsClassLoaderClass();
Mathieu Chartiere4275c02015-08-06 15:34:15 -07001152}
1153
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001154template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Mathieu Chartiere4275c02015-08-06 15:34:15 -07001155inline mirror::ClassLoader* Object::AsClassLoader() {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001156 DCHECK((IsClassLoader<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartiere4275c02015-08-06 15:34:15 -07001157 return down_cast<mirror::ClassLoader*>(this);
1158}
1159
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001160template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Vladimir Marko05792b92015-08-03 11:56:49 +01001161inline bool Object::IsDexCache() {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001162 return GetClass<kVerifyFlags, kReadBarrierOption>()->IsDexCacheClass();
Vladimir Marko05792b92015-08-03 11:56:49 +01001163}
1164
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001165template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Vladimir Marko05792b92015-08-03 11:56:49 +01001166inline mirror::DexCache* Object::AsDexCache() {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001167 DCHECK((IsDexCache<kVerifyFlags, kReadBarrierOption>()));
Vladimir Marko05792b92015-08-03 11:56:49 +01001168 return down_cast<mirror::DexCache*>(this);
1169}
1170
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001171template <bool kVisitNativeRoots,
1172 VerifyObjectFlags kVerifyFlags,
1173 ReadBarrierOption kReadBarrierOption,
1174 typename Visitor,
1175 typename JavaLangRefVisitor>
Mathieu Chartier407f7022014-02-18 14:37:05 -08001176inline void Object::VisitReferences(const Visitor& visitor,
1177 const JavaLangRefVisitor& ref_visitor) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001178 mirror::Class* klass = GetClass<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartier059ef3d2015-08-18 13:54:21 -07001179 visitor(this, ClassOffset(), false);
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001180 const uint32_t class_flags = klass->GetClassFlags<kVerifyNone>();
1181 if (LIKELY(class_flags == kClassFlagNormal)) {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001182 DCHECK((!klass->IsVariableSize<kVerifyFlags, kReadBarrierOption>()));
1183 VisitInstanceFieldsReferences<kVerifyFlags, kReadBarrierOption>(klass, visitor);
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001184 DCHECK((!klass->IsClassClass<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartier66c2d2d2015-08-25 14:32:32 -07001185 DCHECK(!klass->IsStringClass());
1186 DCHECK(!klass->IsClassLoaderClass());
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001187 DCHECK((!klass->IsArrayClass<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001188 } else {
1189 if ((class_flags & kClassFlagNoReferenceFields) == 0) {
1190 DCHECK(!klass->IsStringClass());
1191 if (class_flags == kClassFlagClass) {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001192 DCHECK((klass->IsClassClass<kVerifyFlags, kReadBarrierOption>()));
1193 mirror::Class* as_klass = AsClass<kVerifyNone, kReadBarrierOption>();
1194 as_klass->VisitReferences<kVisitNativeRoots, kVerifyFlags, kReadBarrierOption>(klass,
1195 visitor);
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001196 } else if (class_flags == kClassFlagObjectArray) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001197 DCHECK((klass->IsObjectArrayClass<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001198 AsObjectArray<mirror::Object, kVerifyNone, kReadBarrierOption>()->VisitReferences(visitor);
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001199 } else if ((class_flags & kClassFlagReference) != 0) {
Mathieu Chartierdfe02f62016-02-01 20:15:11 -08001200 VisitInstanceFieldsReferences<kVerifyFlags, kReadBarrierOption>(klass, visitor);
1201 ref_visitor(klass, AsReference<kVerifyFlags, kReadBarrierOption>());
Vladimir Marko05792b92015-08-03 11:56:49 +01001202 } else if (class_flags == kClassFlagDexCache) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001203 mirror::DexCache* const dex_cache = AsDexCache<kVerifyFlags, kReadBarrierOption>();
1204 dex_cache->VisitReferences<kVisitNativeRoots,
1205 kVerifyFlags,
1206 kReadBarrierOption>(klass, visitor);
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001207 } else {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001208 mirror::ClassLoader* const class_loader = AsClassLoader<kVerifyFlags, kReadBarrierOption>();
1209 class_loader->VisitReferences<kVisitNativeRoots,
1210 kVerifyFlags,
1211 kReadBarrierOption>(klass, visitor);
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001212 }
1213 } else if (kIsDebugBuild) {
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001214 CHECK((!klass->IsClassClass<kVerifyFlags, kReadBarrierOption>()));
1215 CHECK((!klass->IsObjectArrayClass<kVerifyFlags, kReadBarrierOption>()));
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001216 // String still has instance fields for reflection purposes but these don't exist in
1217 // actual string instances.
1218 if (!klass->IsStringClass()) {
1219 size_t total_reference_instance_fields = 0;
1220 mirror::Class* super_class = klass;
1221 do {
1222 total_reference_instance_fields += super_class->NumReferenceInstanceFields();
Mathieu Chartierfbc31082016-01-24 11:59:56 -08001223 super_class = super_class->GetSuperClass<kVerifyFlags, kReadBarrierOption>();
Mathieu Chartier52a7f5c2015-08-18 18:35:52 -07001224 } while (super_class != nullptr);
1225 // The only reference field should be the object's class. This field is handled at the
1226 // beginning of the function.
1227 CHECK_EQ(total_reference_instance_fields, 1u);
1228 }
Mathieu Chartier407f7022014-02-18 14:37:05 -08001229 }
1230 }
1231}
Ian Rogers2dd0e2c2013-01-24 12:42:14 -08001232} // namespace mirror
1233} // namespace art
1234
Brian Carlstromfc0e3212013-07-17 14:40:12 -07001235#endif // ART_RUNTIME_MIRROR_OBJECT_INL_H_