blob: d9f442c3e41833c12e6a6b2d0fa22cbd7d6b17b2 [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"
Ian Rogersd9c4fc92013-10-01 19:45:43 -070027#include "lock_word-inl.h"
Ian Rogers05f30572013-02-20 12:13:11 -080028#include "monitor.h"
Mathieu Chartier52e4b432014-06-10 11:22:31 -070029#include "object_array-inl.h"
Hiroshi Yamauchi800ac2d2014-04-02 17:32:54 -070030#include "read_barrier-inl.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080031#include "runtime.h"
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -070032#include "reference.h"
Ian Rogers05f30572013-02-20 12:13:11 -080033#include "throwable.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080034
35namespace art {
36namespace mirror {
37
Mingyao Yang98d1cc82014-05-15 17:02:16 -070038inline uint32_t Object::ClassSize() {
39 uint32_t vtable_entries = kVTableLength;
40 return Class::ComputeClassSize(true, vtable_entries, 0, 0, 0);
41}
42
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -070043template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -080044inline Class* Object::GetClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -070045 return GetFieldObject<Class, kVerifyFlags, kReadBarrierOption>(
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070046 OFFSET_OF_OBJECT_MEMBER(Object, klass_));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080047}
48
Mathieu Chartier4e305412014-02-19 10:54:44 -080049template<VerifyObjectFlags kVerifyFlags>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080050inline void Object::SetClass(Class* new_klass) {
Ian Rogersef7d42f2014-01-06 12:55:46 -080051 // new_klass may be NULL prior to class linker initialization.
52 // We don't mark the card as this occurs as part of object allocation. Not all objects have
53 // backing cards, such as large objects.
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +010054 // We use non transactional version since we can't undo this write. We also disable checking as
55 // we may run in transaction mode here.
Mathieu Chartier4e305412014-02-19 10:54:44 -080056 SetFieldObjectWithoutWriteBarrier<false, false,
57 static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis)>(
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070058 OFFSET_OF_OBJECT_MEMBER(Object, klass_), new_klass);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080059}
60
Mathieu Chartierbbd695c2014-04-16 09:48:48 -070061inline LockWord Object::GetLockWord(bool as_volatile) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070062 if (as_volatile) {
63 return LockWord(GetField32Volatile(OFFSET_OF_OBJECT_MEMBER(Object, monitor_)));
64 }
65 return LockWord(GetField32(OFFSET_OF_OBJECT_MEMBER(Object, monitor_)));
Ian Rogersd9c4fc92013-10-01 19:45:43 -070066}
67
Mathieu Chartierbbd695c2014-04-16 09:48:48 -070068inline void Object::SetLockWord(LockWord new_val, bool as_volatile) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +010069 // Force use of non-transactional mode and do not check.
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070070 if (as_volatile) {
71 SetField32Volatile<false, false>(OFFSET_OF_OBJECT_MEMBER(Object, monitor_), new_val.GetValue());
72 } else {
73 SetField32<false, false>(OFFSET_OF_OBJECT_MEMBER(Object, monitor_), new_val.GetValue());
74 }
Ian Rogersd9c4fc92013-10-01 19:45:43 -070075}
76
Ian Rogers228602f2014-07-10 02:07:54 -070077inline bool Object::CasLockWordWeakSequentiallyConsistent(LockWord old_val, LockWord new_val) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +010078 // Force use of non-transactional mode and do not check.
Ian Rogers228602f2014-07-10 02:07:54 -070079 return CasFieldWeakSequentiallyConsistent32<false, false>(
80 OFFSET_OF_OBJECT_MEMBER(Object, monitor_), old_val.GetValue(), new_val.GetValue());
Ian Rogersd9c4fc92013-10-01 19:45:43 -070081}
82
83inline uint32_t Object::GetLockOwnerThreadId() {
84 return Monitor::GetLockOwnerThreadId(this);
Ian Rogers05f30572013-02-20 12:13:11 -080085}
86
Mathieu Chartiere7e8a5f2014-02-14 16:59:41 -080087inline mirror::Object* Object::MonitorEnter(Thread* self) {
88 return Monitor::MonitorEnter(self, this);
Ian Rogers05f30572013-02-20 12:13:11 -080089}
90
91inline bool Object::MonitorExit(Thread* self) {
92 return Monitor::MonitorExit(self, this);
93}
94
95inline void Object::Notify(Thread* self) {
96 Monitor::Notify(self, this);
97}
98
99inline void Object::NotifyAll(Thread* self) {
100 Monitor::NotifyAll(self, this);
101}
102
103inline void Object::Wait(Thread* self) {
104 Monitor::Wait(self, this, 0, 0, true, kWaiting);
105}
106
107inline void Object::Wait(Thread* self, int64_t ms, int32_t ns) {
108 Monitor::Wait(self, this, ms, ns, true, kTimedWaiting);
109}
110
Hiroshi Yamauchi624468c2014-03-31 15:14:47 -0700111inline Object* Object::GetReadBarrierPointer() {
112#ifdef USE_BAKER_OR_BROOKS_READ_BARRIER
113 DCHECK(kUseBakerOrBrooksReadBarrier);
Hiroshi Yamauchi6a154a42014-05-02 14:26:13 -0700114 return GetFieldObject<Object, kVerifyNone, kWithoutReadBarrier>(
115 OFFSET_OF_OBJECT_MEMBER(Object, x_rb_ptr_));
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800116#else
117 LOG(FATAL) << "Unreachable";
118 return nullptr;
119#endif
120}
121
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700122inline void Object::SetReadBarrierPointer(Object* rb_ptr) {
Hiroshi Yamauchi624468c2014-03-31 15:14:47 -0700123#ifdef USE_BAKER_OR_BROOKS_READ_BARRIER
124 DCHECK(kUseBakerOrBrooksReadBarrier);
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800125 // We don't mark the card as this occurs as part of object allocation. Not all objects have
126 // backing cards, such as large objects.
127 SetFieldObjectWithoutWriteBarrier<false, false, kVerifyNone>(
Hiroshi Yamauchi6a154a42014-05-02 14:26:13 -0700128 OFFSET_OF_OBJECT_MEMBER(Object, x_rb_ptr_), rb_ptr);
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800129#else
130 LOG(FATAL) << "Unreachable";
131#endif
132}
133
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700134inline bool Object::AtomicSetReadBarrierPointer(Object* expected_rb_ptr, Object* rb_ptr) {
135#ifdef USE_BAKER_OR_BROOKS_READ_BARRIER
136 DCHECK(kUseBakerOrBrooksReadBarrier);
137 MemberOffset offset = OFFSET_OF_OBJECT_MEMBER(Object, x_rb_ptr_);
138 byte* raw_addr = reinterpret_cast<byte*>(this) + offset.SizeValue();
Ian Rogers228602f2014-07-10 02:07:54 -0700139 Atomic<uint32_t>* atomic_rb_ptr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700140 HeapReference<Object> expected_ref(HeapReference<Object>::FromMirrorPtr(expected_rb_ptr));
141 HeapReference<Object> new_ref(HeapReference<Object>::FromMirrorPtr(rb_ptr));
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700142 do {
Ian Rogers228602f2014-07-10 02:07:54 -0700143 if (UNLIKELY(atomic_rb_ptr->LoadRelaxed() != expected_ref.reference_)) {
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700144 // Lost the race.
145 return false;
146 }
Ian Rogers228602f2014-07-10 02:07:54 -0700147 } while (!atomic_rb_ptr->CompareExchangeWeakSequentiallyConsistent(expected_ref.reference_,
148 new_ref.reference_));
149 DCHECK_EQ(new_ref.reference_, atomic_rb_ptr->LoadRelaxed());
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700150 return true;
151#else
152 LOG(FATAL) << "Unreachable";
153 return false;
154#endif
155}
156
Hiroshi Yamauchi624468c2014-03-31 15:14:47 -0700157inline void Object::AssertReadBarrierPointer() const {
Hiroshi Yamauchi800ac2d2014-04-02 17:32:54 -0700158 if (kUseBakerReadBarrier) {
159 Object* obj = const_cast<Object*>(this);
160 DCHECK(obj->GetReadBarrierPointer() == nullptr)
161 << "Bad Baker pointer: obj=" << reinterpret_cast<void*>(obj)
162 << " ptr=" << reinterpret_cast<void*>(obj->GetReadBarrierPointer());
163 } else if (kUseBrooksReadBarrier) {
164 Object* obj = const_cast<Object*>(this);
165 DCHECK_EQ(obj, obj->GetReadBarrierPointer())
166 << "Bad Brooks pointer: obj=" << reinterpret_cast<void*>(obj)
167 << " ptr=" << reinterpret_cast<void*>(obj->GetReadBarrierPointer());
168 } else {
169 LOG(FATAL) << "Unreachable";
170 }
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800171}
172
Mathieu Chartier4e305412014-02-19 10:54:44 -0800173template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800174inline bool Object::VerifierInstanceOf(Class* klass) {
Jeff Haoa3faaf42013-09-03 19:07:00 -0700175 DCHECK(klass != NULL);
Mathieu Chartier4e305412014-02-19 10:54:44 -0800176 DCHECK(GetClass<kVerifyFlags>() != NULL);
Jeff Haoa3faaf42013-09-03 19:07:00 -0700177 return klass->IsInterface() || InstanceOf(klass);
178}
179
Mathieu Chartier4e305412014-02-19 10:54:44 -0800180template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800181inline bool Object::InstanceOf(Class* klass) {
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800182 DCHECK(klass != NULL);
Mathieu Chartier4e305412014-02-19 10:54:44 -0800183 DCHECK(GetClass<kVerifyNone>() != NULL);
184 return klass->IsAssignableFrom(GetClass<kVerifyFlags>());
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800185}
186
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700187template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800188inline bool Object::IsClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700189 Class* java_lang_Class = GetClass<kVerifyFlags, kReadBarrierOption>()->
190 template GetClass<kVerifyFlags, kReadBarrierOption>();
191 return GetClass<static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis),
192 kReadBarrierOption>() == java_lang_Class;
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800193}
194
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700195template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800196inline Class* Object::AsClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700197 DCHECK((IsClass<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800198 return down_cast<Class*>(this);
199}
200
Mathieu Chartier4e305412014-02-19 10:54:44 -0800201template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800202inline bool Object::IsObjectArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800203 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
204 return IsArrayInstance<kVerifyFlags>() &&
205 !GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitive();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800206}
207
Mathieu Chartier4e305412014-02-19 10:54:44 -0800208template<class T, VerifyObjectFlags kVerifyFlags>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800209inline ObjectArray<T>* Object::AsObjectArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800210 DCHECK(IsObjectArray<kVerifyFlags>());
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800211 return down_cast<ObjectArray<T>*>(this);
212}
213
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700214template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800215inline bool Object::IsArrayInstance() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700216 return GetClass<kVerifyFlags, kReadBarrierOption>()->
217 template IsArrayClass<kVerifyFlags, kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800218}
219
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700220template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800221inline bool Object::IsArtField() {
Hiroshi Yamauchi4f1ebc22014-06-25 14:30:41 -0700222 return GetClass<kVerifyFlags, kReadBarrierOption>()->
223 template IsArtFieldClass<kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800224}
225
Mathieu Chartier4e305412014-02-19 10:54:44 -0800226template<VerifyObjectFlags kVerifyFlags>
227inline ArtField* Object::AsArtField() {
228 DCHECK(IsArtField<kVerifyFlags>());
Brian Carlstromea46f952013-07-30 01:26:50 -0700229 return down_cast<ArtField*>(this);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800230}
231
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700232template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800233inline bool Object::IsArtMethod() {
Hiroshi Yamauchi4f1ebc22014-06-25 14:30:41 -0700234 return GetClass<kVerifyFlags, kReadBarrierOption>()->
235 template IsArtMethodClass<kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800236}
237
Mathieu Chartier4e305412014-02-19 10:54:44 -0800238template<VerifyObjectFlags kVerifyFlags>
Brian Carlstromea46f952013-07-30 01:26:50 -0700239inline ArtMethod* Object::AsArtMethod() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800240 DCHECK(IsArtMethod<kVerifyFlags>());
Brian Carlstromea46f952013-07-30 01:26:50 -0700241 return down_cast<ArtMethod*>(this);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800242}
243
Mathieu Chartier4e305412014-02-19 10:54:44 -0800244template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800245inline bool Object::IsReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800246 return GetClass<kVerifyFlags>()->IsReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800247}
248
Mathieu Chartier4e305412014-02-19 10:54:44 -0800249template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700250inline Reference* Object::AsReference() {
251 DCHECK(IsReferenceInstance<kVerifyFlags>());
252 return down_cast<Reference*>(this);
253}
254
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700255template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers05f30572013-02-20 12:13:11 -0800256inline Array* Object::AsArray() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700257 DCHECK((IsArrayInstance<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800258 return down_cast<Array*>(this);
259}
260
Mathieu Chartier4e305412014-02-19 10:54:44 -0800261template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800262inline BooleanArray* Object::AsBooleanArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800263 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
264 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
265 DCHECK(GetClass<kNewFlags>()->GetComponentType()->IsPrimitiveBoolean());
Ian Rogers05f30572013-02-20 12:13:11 -0800266 return down_cast<BooleanArray*>(this);
267}
268
Mathieu Chartier4e305412014-02-19 10:54:44 -0800269template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800270inline ByteArray* Object::AsByteArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800271 static const VerifyObjectFlags kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
272 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
273 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveByte());
Ian Rogers05f30572013-02-20 12:13:11 -0800274 return down_cast<ByteArray*>(this);
275}
276
Mathieu Chartier4e305412014-02-19 10:54:44 -0800277template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800278inline ByteArray* Object::AsByteSizedArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800279 constexpr VerifyObjectFlags kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
280 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
281 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveByte() ||
282 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveBoolean());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800283 return down_cast<ByteArray*>(this);
284}
285
Mathieu Chartier4e305412014-02-19 10:54:44 -0800286template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800287inline CharArray* Object::AsCharArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800288 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
289 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
290 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveChar());
Ian Rogers05f30572013-02-20 12:13:11 -0800291 return down_cast<CharArray*>(this);
292}
293
Mathieu Chartier4e305412014-02-19 10:54:44 -0800294template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800295inline ShortArray* Object::AsShortArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800296 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
297 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
298 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveShort());
Ian Rogers05f30572013-02-20 12:13:11 -0800299 return down_cast<ShortArray*>(this);
300}
301
Mathieu Chartier4e305412014-02-19 10:54:44 -0800302template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800303inline ShortArray* Object::AsShortSizedArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800304 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
305 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
306 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveShort() ||
307 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveChar());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800308 return down_cast<ShortArray*>(this);
309}
310
Mathieu Chartier4e305412014-02-19 10:54:44 -0800311template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800312inline IntArray* Object::AsIntArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800313 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
314 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
315 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveInt() ||
316 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveFloat());
Ian Rogers05f30572013-02-20 12:13:11 -0800317 return down_cast<IntArray*>(this);
318}
319
Mathieu Chartier4e305412014-02-19 10:54:44 -0800320template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800321inline LongArray* Object::AsLongArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800322 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
323 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
324 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveLong() ||
325 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveDouble());
Ian Rogers05f30572013-02-20 12:13:11 -0800326 return down_cast<LongArray*>(this);
327}
328
Mathieu Chartier4e305412014-02-19 10:54:44 -0800329template<VerifyObjectFlags kVerifyFlags>
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100330inline FloatArray* Object::AsFloatArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800331 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
332 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
333 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveFloat());
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100334 return down_cast<FloatArray*>(this);
335}
336
Mathieu Chartier4e305412014-02-19 10:54:44 -0800337template<VerifyObjectFlags kVerifyFlags>
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100338inline DoubleArray* Object::AsDoubleArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800339 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
340 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
341 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveDouble());
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100342 return down_cast<DoubleArray*>(this);
343}
344
Mathieu Chartier4e305412014-02-19 10:54:44 -0800345template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800346inline String* Object::AsString() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800347 DCHECK(GetClass<kVerifyFlags>()->IsStringClass());
Ian Rogers05f30572013-02-20 12:13:11 -0800348 return down_cast<String*>(this);
349}
350
Mathieu Chartier4e305412014-02-19 10:54:44 -0800351template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800352inline Throwable* Object::AsThrowable() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800353 DCHECK(GetClass<kVerifyFlags>()->IsThrowableClass());
Ian Rogers05f30572013-02-20 12:13:11 -0800354 return down_cast<Throwable*>(this);
355}
356
Mathieu Chartier4e305412014-02-19 10:54:44 -0800357template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800358inline bool Object::IsWeakReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800359 return GetClass<kVerifyFlags>()->IsWeakReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800360}
361
Mathieu Chartier4e305412014-02-19 10:54:44 -0800362template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800363inline bool Object::IsSoftReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800364 return GetClass<kVerifyFlags>()->IsSoftReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800365}
366
Mathieu Chartier4e305412014-02-19 10:54:44 -0800367template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800368inline bool Object::IsFinalizerReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800369 return GetClass<kVerifyFlags>()->IsFinalizerReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800370}
371
Mathieu Chartier4e305412014-02-19 10:54:44 -0800372template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700373inline FinalizerReference* Object::AsFinalizerReference() {
374 DCHECK(IsFinalizerReferenceInstance<kVerifyFlags>());
375 return down_cast<FinalizerReference*>(this);
376}
377
378template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800379inline bool Object::IsPhantomReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800380 return GetClass<kVerifyFlags>()->IsPhantomReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800381}
382
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700383template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800384inline size_t Object::SizeOf() {
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800385 size_t result;
Mathieu Chartier4e305412014-02-19 10:54:44 -0800386 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700387 if (IsArrayInstance<kVerifyFlags, kReadBarrierOption>()) {
388 result = AsArray<kNewFlags, kReadBarrierOption>()->
389 template SizeOf<kNewFlags, kReadBarrierOption>();
390 } else if (IsClass<kNewFlags, kReadBarrierOption>()) {
391 result = AsClass<kNewFlags, kReadBarrierOption>()->
392 template SizeOf<kNewFlags, kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800393 } else {
Hiroshi Yamauchi25023c72014-05-09 11:45:53 -0700394 result = GetClass<kNewFlags, kReadBarrierOption>()->
395 template GetObjectSize<kNewFlags, kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800396 }
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700397 DCHECK_GE(result, sizeof(Object))
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700398 << " class=" << PrettyTypeOf(GetClass<kNewFlags, kReadBarrierOption>());
399 DCHECK(!(IsArtField<kNewFlags, kReadBarrierOption>()) || result == sizeof(ArtField));
400 DCHECK(!(IsArtMethod<kNewFlags, kReadBarrierOption>()) || result == sizeof(ArtMethod));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800401 return result;
402}
403
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700404template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
405inline int32_t Object::GetField32(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800406 if (kVerifyFlags & kVerifyThis) {
407 VerifyObject(this);
408 }
Ian Rogersb122a4b2013-11-19 18:00:50 -0800409 const byte* raw_addr = reinterpret_cast<const byte*>(this) + field_offset.Int32Value();
410 const int32_t* word_addr = reinterpret_cast<const int32_t*>(raw_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700411 if (UNLIKELY(kIsVolatile)) {
Hans Boehm30359612014-05-21 17:46:23 -0700412 return reinterpret_cast<const Atomic<int32_t>*>(word_addr)->LoadSequentiallyConsistent();
Ian Rogersb122a4b2013-11-19 18:00:50 -0800413 } else {
Hans Boehm30359612014-05-21 17:46:23 -0700414 return reinterpret_cast<const Atomic<int32_t>*>(word_addr)->LoadJavaData();
Ian Rogersb122a4b2013-11-19 18:00:50 -0800415 }
416}
417
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700418template<VerifyObjectFlags kVerifyFlags>
419inline int32_t Object::GetField32Volatile(MemberOffset field_offset) {
420 return GetField32<kVerifyFlags, true>(field_offset);
421}
422
423template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
424 bool kIsVolatile>
425inline void Object::SetField32(MemberOffset field_offset, int32_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100426 if (kCheckTransaction) {
427 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
428 }
429 if (kTransactionActive) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700430 Runtime::Current()->RecordWriteField32(this, field_offset,
431 GetField32<kVerifyFlags, kIsVolatile>(field_offset),
432 kIsVolatile);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100433 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800434 if (kVerifyFlags & kVerifyThis) {
Ian Rogersb122a4b2013-11-19 18:00:50 -0800435 VerifyObject(this);
436 }
437 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
Ian Rogers29501cf2014-02-07 21:00:25 -0800438 int32_t* word_addr = reinterpret_cast<int32_t*>(raw_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700439 if (kIsVolatile) {
Hans Boehm30359612014-05-21 17:46:23 -0700440 reinterpret_cast<Atomic<int32_t>*>(word_addr)->StoreSequentiallyConsistent(new_value);
Ian Rogersb122a4b2013-11-19 18:00:50 -0800441 } else {
Hans Boehm30359612014-05-21 17:46:23 -0700442 reinterpret_cast<Atomic<int32_t>*>(word_addr)->StoreJavaData(new_value);
Ian Rogersb122a4b2013-11-19 18:00:50 -0800443 }
444}
445
Mathieu Chartier4e305412014-02-19 10:54:44 -0800446template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700447inline void Object::SetField32Volatile(MemberOffset field_offset, int32_t new_value) {
448 SetField32<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset, new_value);
449}
450
451template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -0700452inline bool Object::CasFieldWeakSequentiallyConsistent32(MemberOffset field_offset,
453 int32_t old_value, int32_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100454 if (kCheckTransaction) {
455 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
456 }
457 if (kTransactionActive) {
458 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
459 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800460 if (kVerifyFlags & kVerifyThis) {
461 VerifyObject(this);
462 }
Ian Rogersd9c4fc92013-10-01 19:45:43 -0700463 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
Ian Rogers228602f2014-07-10 02:07:54 -0700464 AtomicInteger* atomic_addr = reinterpret_cast<AtomicInteger*>(raw_addr);
Hans Boehm30359612014-05-21 17:46:23 -0700465
Ian Rogers228602f2014-07-10 02:07:54 -0700466 return atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_value, new_value);
Ian Rogersd9c4fc92013-10-01 19:45:43 -0700467}
468
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700469template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
470inline int64_t Object::GetField64(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800471 if (kVerifyFlags & kVerifyThis) {
472 VerifyObject(this);
473 }
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800474 const byte* raw_addr = reinterpret_cast<const byte*>(this) + field_offset.Int32Value();
475 const int64_t* addr = reinterpret_cast<const int64_t*>(raw_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700476 if (kIsVolatile) {
Hans Boehm30359612014-05-21 17:46:23 -0700477 return reinterpret_cast<const Atomic<int64_t>*>(addr)->LoadSequentiallyConsistent();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800478 } else {
Hans Boehm30359612014-05-21 17:46:23 -0700479 return reinterpret_cast<const Atomic<int64_t>*>(addr)->LoadJavaData();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800480 }
481}
482
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700483template<VerifyObjectFlags kVerifyFlags>
484inline int64_t Object::GetField64Volatile(MemberOffset field_offset) {
485 return GetField64<kVerifyFlags, true>(field_offset);
486}
487
488template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
489 bool kIsVolatile>
490inline void Object::SetField64(MemberOffset field_offset, int64_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100491 if (kCheckTransaction) {
492 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
493 }
494 if (kTransactionActive) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700495 Runtime::Current()->RecordWriteField64(this, field_offset,
496 GetField64<kVerifyFlags, kIsVolatile>(field_offset),
497 kIsVolatile);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100498 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800499 if (kVerifyFlags & kVerifyThis) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800500 VerifyObject(this);
501 }
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800502 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
503 int64_t* addr = reinterpret_cast<int64_t*>(raw_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700504 if (kIsVolatile) {
Hans Boehm30359612014-05-21 17:46:23 -0700505 reinterpret_cast<Atomic<int64_t>*>(addr)->StoreSequentiallyConsistent(new_value);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800506 } else {
Hans Boehm30359612014-05-21 17:46:23 -0700507 reinterpret_cast<Atomic<int64_t>*>(addr)->StoreJavaData(new_value);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800508 }
509}
510
Mathieu Chartier4e305412014-02-19 10:54:44 -0800511template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700512inline void Object::SetField64Volatile(MemberOffset field_offset, int64_t new_value) {
513 return SetField64<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
514 new_value);
515}
516
517template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -0700518inline bool Object::CasFieldWeakSequentiallyConsistent64(MemberOffset field_offset,
519 int64_t old_value, int64_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100520 if (kCheckTransaction) {
521 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
522 }
523 if (kTransactionActive) {
524 Runtime::Current()->RecordWriteField64(this, field_offset, old_value, true);
525 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800526 if (kVerifyFlags & kVerifyThis) {
527 VerifyObject(this);
528 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800529 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
Ian Rogers228602f2014-07-10 02:07:54 -0700530 Atomic<int64_t>* atomic_addr = reinterpret_cast<Atomic<int64_t>*>(raw_addr);
531 return atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_value, new_value);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800532}
533
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700534template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption,
535 bool kIsVolatile>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700536inline T* Object::GetFieldObject(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800537 if (kVerifyFlags & kVerifyThis) {
538 VerifyObject(this);
539 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800540 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
541 HeapReference<T>* objref_addr = reinterpret_cast<HeapReference<T>*>(raw_addr);
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700542 T* result = ReadBarrier::Barrier<T, kReadBarrierOption>(this, field_offset, objref_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700543 if (kIsVolatile) {
Hans Boehm30359612014-05-21 17:46:23 -0700544 // TODO: Refactor to use a SequentiallyConsistent load instead.
545 QuasiAtomic::ThreadFenceAcquire(); // Ensure visibility of operations preceding store.
Ian Rogersef7d42f2014-01-06 12:55:46 -0800546 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800547 if (kVerifyFlags & kVerifyReads) {
548 VerifyObject(result);
549 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800550 return result;
551}
552
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700553template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700554inline T* Object::GetFieldObjectVolatile(MemberOffset field_offset) {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700555 return GetFieldObject<T, kVerifyFlags, kReadBarrierOption, true>(field_offset);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700556}
557
558template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
559 bool kIsVolatile>
560inline void Object::SetFieldObjectWithoutWriteBarrier(MemberOffset field_offset,
561 Object* new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100562 if (kCheckTransaction) {
563 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
564 }
565 if (kTransactionActive) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700566 mirror::Object* obj;
567 if (kIsVolatile) {
568 obj = GetFieldObjectVolatile<Object>(field_offset);
569 } else {
570 obj = GetFieldObject<Object>(field_offset);
571 }
572 Runtime::Current()->RecordWriteFieldReference(this, field_offset, obj, true);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100573 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800574 if (kVerifyFlags & kVerifyThis) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800575 VerifyObject(this);
576 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800577 if (kVerifyFlags & kVerifyWrites) {
578 VerifyObject(new_value);
579 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800580 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
581 HeapReference<Object>* objref_addr = reinterpret_cast<HeapReference<Object>*>(raw_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700582 if (kIsVolatile) {
Hans Boehm30359612014-05-21 17:46:23 -0700583 // TODO: Refactor to use a SequentiallyConsistent store instead.
584 QuasiAtomic::ThreadFenceRelease(); // Ensure that prior accesses are visible before store.
Ian Rogersef7d42f2014-01-06 12:55:46 -0800585 objref_addr->Assign(new_value);
Hans Boehm30359612014-05-21 17:46:23 -0700586 QuasiAtomic::ThreadFenceSequentiallyConsistent();
587 // Ensure this store occurs before any volatile loads.
Ian Rogersef7d42f2014-01-06 12:55:46 -0800588 } else {
589 objref_addr->Assign(new_value);
590 }
591}
592
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700593template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
594 bool kIsVolatile>
595inline void Object::SetFieldObject(MemberOffset field_offset, Object* new_value) {
596 SetFieldObjectWithoutWriteBarrier<kTransactionActive, kCheckTransaction, kVerifyFlags,
597 kIsVolatile>(field_offset, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800598 if (new_value != nullptr) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800599 Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
Mathieu Chartier61c5ebc2014-06-05 17:42:53 -0700600 // TODO: Check field assignment could theoretically cause thread suspension, TODO: fix this.
601 CheckFieldAssignment(field_offset, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800602 }
603}
604
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700605template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
606inline void Object::SetFieldObjectVolatile(MemberOffset field_offset, Object* new_value) {
607 SetFieldObject<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
608 new_value);
609}
610
Mathieu Chartier4e305412014-02-19 10:54:44 -0800611template <VerifyObjectFlags kVerifyFlags>
612inline HeapReference<Object>* Object::GetFieldObjectReferenceAddr(MemberOffset field_offset) {
613 if (kVerifyFlags & kVerifyThis) {
614 VerifyObject(this);
615 }
616 return reinterpret_cast<HeapReference<Object>*>(reinterpret_cast<byte*>(this) +
617 field_offset.Int32Value());
618}
619
620template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers228602f2014-07-10 02:07:54 -0700621inline bool Object::CasFieldWeakSequentiallyConsistentObject(MemberOffset field_offset,
622 Object* old_value, Object* new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100623 if (kCheckTransaction) {
624 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
625 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800626 if (kVerifyFlags & kVerifyThis) {
627 VerifyObject(this);
628 }
629 if (kVerifyFlags & kVerifyWrites) {
630 VerifyObject(new_value);
631 }
632 if (kVerifyFlags & kVerifyReads) {
633 VerifyObject(old_value);
634 }
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100635 if (kTransactionActive) {
636 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
637 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800638 HeapReference<Object> old_ref(HeapReference<Object>::FromMirrorPtr(old_value));
639 HeapReference<Object> new_ref(HeapReference<Object>::FromMirrorPtr(new_value));
Ian Rogers228602f2014-07-10 02:07:54 -0700640 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
641 Atomic<uint32_t>* atomic_addr = reinterpret_cast<Atomic<uint32_t>*>(raw_addr);
642
643 bool success = atomic_addr->CompareExchangeWeakSequentiallyConsistent(old_ref.reference_,
644 new_ref.reference_);
645
Ian Rogersef7d42f2014-01-06 12:55:46 -0800646 if (success) {
647 Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
648 }
649 return success;
650}
651
Mathieu Chartier407f7022014-02-18 14:37:05 -0800652template<bool kVisitClass, bool kIsStatic, typename Visitor>
653inline void Object::VisitFieldsReferences(uint32_t ref_offsets, const Visitor& visitor) {
654 if (LIKELY(ref_offsets != CLASS_WALK_SUPER)) {
655 if (!kVisitClass) {
656 // Mask out the class from the reference offsets.
657 ref_offsets ^= kWordHighBitMask;
658 }
659 DCHECK_EQ(ClassOffset().Uint32Value(), 0U);
660 // Found a reference offset bitmap. Visit the specified offsets.
661 while (ref_offsets != 0) {
662 size_t right_shift = CLZ(ref_offsets);
663 MemberOffset field_offset = CLASS_OFFSET_FROM_CLZ(right_shift);
664 visitor(this, field_offset, kIsStatic);
665 ref_offsets &= ~(CLASS_HIGH_BIT >> right_shift);
666 }
667 } else {
668 // There is no reference offset bitmap. In the non-static case, walk up the class
669 // inheritance hierarchy and find reference offsets the hard way. In the static case, just
670 // consider this class.
671 for (mirror::Class* klass = kIsStatic ? AsClass() : GetClass(); klass != nullptr;
672 klass = kIsStatic ? nullptr : klass->GetSuperClass()) {
673 size_t num_reference_fields =
674 kIsStatic ? klass->NumReferenceStaticFields() : klass->NumReferenceInstanceFields();
675 for (size_t i = 0; i < num_reference_fields; ++i) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700676 mirror::ArtField* field = kIsStatic ? klass->GetStaticField(i) : klass->GetInstanceField(i);
Mathieu Chartier407f7022014-02-18 14:37:05 -0800677 MemberOffset field_offset = field->GetOffset();
678 // TODO: Do a simpler check?
Mathieu Chartier52e4b432014-06-10 11:22:31 -0700679 if (kVisitClass || field_offset.Uint32Value() != ClassOffset().Uint32Value()) {
680 visitor(this, field_offset, kIsStatic);
Mathieu Chartier407f7022014-02-18 14:37:05 -0800681 }
Mathieu Chartier407f7022014-02-18 14:37:05 -0800682 }
683 }
684 }
685}
686
687template<bool kVisitClass, typename Visitor>
688inline void Object::VisitInstanceFieldsReferences(mirror::Class* klass, const Visitor& visitor) {
689 VisitFieldsReferences<kVisitClass, false>(
690 klass->GetReferenceInstanceOffsets<kVerifyNone>(), visitor);
691}
692
693template<bool kVisitClass, typename Visitor>
694inline void Object::VisitStaticFieldsReferences(mirror::Class* klass, const Visitor& visitor) {
Mingyao Yang98d1cc82014-05-15 17:02:16 -0700695 DCHECK(!klass->IsTemp());
Mathieu Chartier407f7022014-02-18 14:37:05 -0800696 klass->VisitFieldsReferences<kVisitClass, true>(
697 klass->GetReferenceStaticOffsets<kVerifyNone>(), visitor);
698}
699
700template <const bool kVisitClass, VerifyObjectFlags kVerifyFlags, typename Visitor,
701 typename JavaLangRefVisitor>
702inline void Object::VisitReferences(const Visitor& visitor,
703 const JavaLangRefVisitor& ref_visitor) {
704 mirror::Class* klass = GetClass<kVerifyFlags>();
Mathieu Chartier52e4b432014-06-10 11:22:31 -0700705 if (klass == Class::GetJavaLangClass()) {
706 AsClass<kVerifyNone>()->VisitReferences<kVisitClass>(klass, visitor);
707 } else if (klass->IsArrayClass()) {
708 if (klass->IsObjectArrayClass<kVerifyNone>()) {
709 AsObjectArray<mirror::Object, kVerifyNone>()->VisitReferences<kVisitClass>(visitor);
710 } else if (kVisitClass) {
711 visitor(this, ClassOffset(), false);
Mathieu Chartier407f7022014-02-18 14:37:05 -0800712 }
713 } else {
Mathieu Chartier52e4b432014-06-10 11:22:31 -0700714 DCHECK(!klass->IsVariableSize());
Mathieu Chartier580a8df2014-03-26 15:15:57 -0700715 VisitInstanceFieldsReferences<kVisitClass>(klass, visitor);
716 if (UNLIKELY(klass->IsReferenceClass<kVerifyNone>())) {
Mathieu Chartier407f7022014-02-18 14:37:05 -0800717 ref_visitor(klass, AsReference());
718 }
719 }
720}
721
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800722} // namespace mirror
723} // namespace art
724
Brian Carlstromfc0e3212013-07-17 14:40:12 -0700725#endif // ART_RUNTIME_MIRROR_OBJECT_INL_H_