blob: a2072a2ef216519a33742fef6a8c7d67942457b2 [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"
Hiroshi Yamauchi800ac2d2014-04-02 17:32:54 -070029#include "read_barrier-inl.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080030#include "runtime.h"
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -070031#include "reference.h"
Ian Rogers05f30572013-02-20 12:13:11 -080032#include "throwable.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080033
34namespace art {
35namespace mirror {
36
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -070037template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -080038inline Class* Object::GetClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -070039 return GetFieldObject<Class, kVerifyFlags, kReadBarrierOption>(
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070040 OFFSET_OF_OBJECT_MEMBER(Object, klass_));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080041}
42
Mathieu Chartier4e305412014-02-19 10:54:44 -080043template<VerifyObjectFlags kVerifyFlags>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080044inline void Object::SetClass(Class* new_klass) {
Ian Rogersef7d42f2014-01-06 12:55:46 -080045 // new_klass may be NULL prior to class linker initialization.
46 // We don't mark the card as this occurs as part of object allocation. Not all objects have
47 // backing cards, such as large objects.
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +010048 // We use non transactional version since we can't undo this write. We also disable checking as
49 // we may run in transaction mode here.
Mathieu Chartier4e305412014-02-19 10:54:44 -080050 SetFieldObjectWithoutWriteBarrier<false, false,
51 static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis)>(
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070052 OFFSET_OF_OBJECT_MEMBER(Object, klass_), new_klass);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080053}
54
Mathieu Chartierbbd695c2014-04-16 09:48:48 -070055inline LockWord Object::GetLockWord(bool as_volatile) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070056 if (as_volatile) {
57 return LockWord(GetField32Volatile(OFFSET_OF_OBJECT_MEMBER(Object, monitor_)));
58 }
59 return LockWord(GetField32(OFFSET_OF_OBJECT_MEMBER(Object, monitor_)));
Ian Rogersd9c4fc92013-10-01 19:45:43 -070060}
61
Mathieu Chartierbbd695c2014-04-16 09:48:48 -070062inline void Object::SetLockWord(LockWord new_val, bool as_volatile) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +010063 // Force use of non-transactional mode and do not check.
Ian Rogersb0fa5dc2014-04-28 16:47:08 -070064 if (as_volatile) {
65 SetField32Volatile<false, false>(OFFSET_OF_OBJECT_MEMBER(Object, monitor_), new_val.GetValue());
66 } else {
67 SetField32<false, false>(OFFSET_OF_OBJECT_MEMBER(Object, monitor_), new_val.GetValue());
68 }
Ian Rogersd9c4fc92013-10-01 19:45:43 -070069}
70
71inline bool Object::CasLockWord(LockWord old_val, LockWord new_val) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +010072 // Force use of non-transactional mode and do not check.
73 return CasField32<false, false>(OFFSET_OF_OBJECT_MEMBER(Object, monitor_), old_val.GetValue(),
74 new_val.GetValue());
Ian Rogersd9c4fc92013-10-01 19:45:43 -070075}
76
77inline uint32_t Object::GetLockOwnerThreadId() {
78 return Monitor::GetLockOwnerThreadId(this);
Ian Rogers05f30572013-02-20 12:13:11 -080079}
80
Mathieu Chartiere7e8a5f2014-02-14 16:59:41 -080081inline mirror::Object* Object::MonitorEnter(Thread* self) {
82 return Monitor::MonitorEnter(self, this);
Ian Rogers05f30572013-02-20 12:13:11 -080083}
84
85inline bool Object::MonitorExit(Thread* self) {
86 return Monitor::MonitorExit(self, this);
87}
88
89inline void Object::Notify(Thread* self) {
90 Monitor::Notify(self, this);
91}
92
93inline void Object::NotifyAll(Thread* self) {
94 Monitor::NotifyAll(self, this);
95}
96
97inline void Object::Wait(Thread* self) {
98 Monitor::Wait(self, this, 0, 0, true, kWaiting);
99}
100
101inline void Object::Wait(Thread* self, int64_t ms, int32_t ns) {
102 Monitor::Wait(self, this, ms, ns, true, kTimedWaiting);
103}
104
Hiroshi Yamauchi624468c2014-03-31 15:14:47 -0700105inline Object* Object::GetReadBarrierPointer() {
106#ifdef USE_BAKER_OR_BROOKS_READ_BARRIER
107 DCHECK(kUseBakerOrBrooksReadBarrier);
Hiroshi Yamauchi6a154a42014-05-02 14:26:13 -0700108 return GetFieldObject<Object, kVerifyNone, kWithoutReadBarrier>(
109 OFFSET_OF_OBJECT_MEMBER(Object, x_rb_ptr_));
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800110#else
111 LOG(FATAL) << "Unreachable";
112 return nullptr;
113#endif
114}
115
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700116inline void Object::SetReadBarrierPointer(Object* rb_ptr) {
Hiroshi Yamauchi624468c2014-03-31 15:14:47 -0700117#ifdef USE_BAKER_OR_BROOKS_READ_BARRIER
118 DCHECK(kUseBakerOrBrooksReadBarrier);
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800119 // We don't mark the card as this occurs as part of object allocation. Not all objects have
120 // backing cards, such as large objects.
121 SetFieldObjectWithoutWriteBarrier<false, false, kVerifyNone>(
Hiroshi Yamauchi6a154a42014-05-02 14:26:13 -0700122 OFFSET_OF_OBJECT_MEMBER(Object, x_rb_ptr_), rb_ptr);
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800123#else
124 LOG(FATAL) << "Unreachable";
125#endif
126}
127
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700128inline bool Object::AtomicSetReadBarrierPointer(Object* expected_rb_ptr, Object* rb_ptr) {
129#ifdef USE_BAKER_OR_BROOKS_READ_BARRIER
130 DCHECK(kUseBakerOrBrooksReadBarrier);
131 MemberOffset offset = OFFSET_OF_OBJECT_MEMBER(Object, x_rb_ptr_);
132 byte* raw_addr = reinterpret_cast<byte*>(this) + offset.SizeValue();
133 HeapReference<Object>* ref = reinterpret_cast<HeapReference<Object>*>(raw_addr);
134 HeapReference<Object> expected_ref(HeapReference<Object>::FromMirrorPtr(expected_rb_ptr));
135 HeapReference<Object> new_ref(HeapReference<Object>::FromMirrorPtr(rb_ptr));
136 uint32_t expected_val = expected_ref.reference_;
137 uint32_t new_val;
138 do {
139 uint32_t old_val = ref->reference_;
140 if (old_val != expected_val) {
141 // Lost the race.
142 return false;
143 }
144 new_val = new_ref.reference_;
145 } while (!__sync_bool_compare_and_swap(
146 reinterpret_cast<uint32_t*>(raw_addr), expected_val, new_val));
147 DCHECK_EQ(new_val, ref->reference_);
148 return true;
149#else
150 LOG(FATAL) << "Unreachable";
151 return false;
152#endif
153}
154
Hiroshi Yamauchi624468c2014-03-31 15:14:47 -0700155inline void Object::AssertReadBarrierPointer() const {
Hiroshi Yamauchi800ac2d2014-04-02 17:32:54 -0700156 if (kUseBakerReadBarrier) {
157 Object* obj = const_cast<Object*>(this);
158 DCHECK(obj->GetReadBarrierPointer() == nullptr)
159 << "Bad Baker pointer: obj=" << reinterpret_cast<void*>(obj)
160 << " ptr=" << reinterpret_cast<void*>(obj->GetReadBarrierPointer());
161 } else if (kUseBrooksReadBarrier) {
162 Object* obj = const_cast<Object*>(this);
163 DCHECK_EQ(obj, obj->GetReadBarrierPointer())
164 << "Bad Brooks pointer: obj=" << reinterpret_cast<void*>(obj)
165 << " ptr=" << reinterpret_cast<void*>(obj->GetReadBarrierPointer());
166 } else {
167 LOG(FATAL) << "Unreachable";
168 }
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800169}
170
Mathieu Chartier4e305412014-02-19 10:54:44 -0800171template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800172inline bool Object::VerifierInstanceOf(Class* klass) {
Jeff Haoa3faaf42013-09-03 19:07:00 -0700173 DCHECK(klass != NULL);
Mathieu Chartier4e305412014-02-19 10:54:44 -0800174 DCHECK(GetClass<kVerifyFlags>() != NULL);
Jeff Haoa3faaf42013-09-03 19:07:00 -0700175 return klass->IsInterface() || InstanceOf(klass);
176}
177
Mathieu Chartier4e305412014-02-19 10:54:44 -0800178template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800179inline bool Object::InstanceOf(Class* klass) {
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800180 DCHECK(klass != NULL);
Mathieu Chartier4e305412014-02-19 10:54:44 -0800181 DCHECK(GetClass<kVerifyNone>() != NULL);
182 return klass->IsAssignableFrom(GetClass<kVerifyFlags>());
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800183}
184
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700185template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800186inline bool Object::IsClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700187 Class* java_lang_Class = GetClass<kVerifyFlags, kReadBarrierOption>()->
188 template GetClass<kVerifyFlags, kReadBarrierOption>();
189 return GetClass<static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis),
190 kReadBarrierOption>() == java_lang_Class;
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800191}
192
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700193template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800194inline Class* Object::AsClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700195 DCHECK((IsClass<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800196 return down_cast<Class*>(this);
197}
198
Mathieu Chartier4e305412014-02-19 10:54:44 -0800199template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800200inline bool Object::IsObjectArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800201 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
202 return IsArrayInstance<kVerifyFlags>() &&
203 !GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitive();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800204}
205
Mathieu Chartier4e305412014-02-19 10:54:44 -0800206template<class T, VerifyObjectFlags kVerifyFlags>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800207inline ObjectArray<T>* Object::AsObjectArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800208 DCHECK(IsObjectArray<kVerifyFlags>());
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800209 return down_cast<ObjectArray<T>*>(this);
210}
211
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700212template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800213inline bool Object::IsArrayInstance() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700214 return GetClass<kVerifyFlags, kReadBarrierOption>()->
215 template IsArrayClass<kVerifyFlags, kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800216}
217
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700218template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800219inline bool Object::IsArtField() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700220 return GetClass<kVerifyFlags, kReadBarrierOption>()->
221 template IsArtFieldClass<kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800222}
223
Mathieu Chartier4e305412014-02-19 10:54:44 -0800224template<VerifyObjectFlags kVerifyFlags>
225inline ArtField* Object::AsArtField() {
226 DCHECK(IsArtField<kVerifyFlags>());
Brian Carlstromea46f952013-07-30 01:26:50 -0700227 return down_cast<ArtField*>(this);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800228}
229
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700230template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800231inline bool Object::IsArtMethod() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700232 return GetClass<kVerifyFlags, kReadBarrierOption>()->
233 template IsArtMethodClass<kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800234}
235
Mathieu Chartier4e305412014-02-19 10:54:44 -0800236template<VerifyObjectFlags kVerifyFlags>
Brian Carlstromea46f952013-07-30 01:26:50 -0700237inline ArtMethod* Object::AsArtMethod() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800238 DCHECK(IsArtMethod<kVerifyFlags>());
Brian Carlstromea46f952013-07-30 01:26:50 -0700239 return down_cast<ArtMethod*>(this);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800240}
241
Mathieu Chartier4e305412014-02-19 10:54:44 -0800242template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800243inline bool Object::IsReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800244 return GetClass<kVerifyFlags>()->IsReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800245}
246
Mathieu Chartier4e305412014-02-19 10:54:44 -0800247template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700248inline Reference* Object::AsReference() {
249 DCHECK(IsReferenceInstance<kVerifyFlags>());
250 return down_cast<Reference*>(this);
251}
252
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700253template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers05f30572013-02-20 12:13:11 -0800254inline Array* Object::AsArray() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700255 DCHECK((IsArrayInstance<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800256 return down_cast<Array*>(this);
257}
258
Mathieu Chartier4e305412014-02-19 10:54:44 -0800259template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800260inline BooleanArray* Object::AsBooleanArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800261 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
262 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
263 DCHECK(GetClass<kNewFlags>()->GetComponentType()->IsPrimitiveBoolean());
Ian Rogers05f30572013-02-20 12:13:11 -0800264 return down_cast<BooleanArray*>(this);
265}
266
Mathieu Chartier4e305412014-02-19 10:54:44 -0800267template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800268inline ByteArray* Object::AsByteArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800269 static const VerifyObjectFlags kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
270 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
271 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveByte());
Ian Rogers05f30572013-02-20 12:13:11 -0800272 return down_cast<ByteArray*>(this);
273}
274
Mathieu Chartier4e305412014-02-19 10:54:44 -0800275template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800276inline ByteArray* Object::AsByteSizedArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800277 constexpr VerifyObjectFlags kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
278 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
279 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveByte() ||
280 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveBoolean());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800281 return down_cast<ByteArray*>(this);
282}
283
Mathieu Chartier4e305412014-02-19 10:54:44 -0800284template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800285inline CharArray* Object::AsCharArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800286 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
287 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
288 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveChar());
Ian Rogers05f30572013-02-20 12:13:11 -0800289 return down_cast<CharArray*>(this);
290}
291
Mathieu Chartier4e305412014-02-19 10:54:44 -0800292template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800293inline ShortArray* Object::AsShortArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800294 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
295 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
296 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveShort());
Ian Rogers05f30572013-02-20 12:13:11 -0800297 return down_cast<ShortArray*>(this);
298}
299
Mathieu Chartier4e305412014-02-19 10:54:44 -0800300template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800301inline ShortArray* Object::AsShortSizedArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800302 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
303 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
304 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveShort() ||
305 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveChar());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800306 return down_cast<ShortArray*>(this);
307}
308
Mathieu Chartier4e305412014-02-19 10:54:44 -0800309template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800310inline IntArray* Object::AsIntArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800311 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
312 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
313 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveInt() ||
314 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveFloat());
Ian Rogers05f30572013-02-20 12:13:11 -0800315 return down_cast<IntArray*>(this);
316}
317
Mathieu Chartier4e305412014-02-19 10:54:44 -0800318template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800319inline LongArray* Object::AsLongArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800320 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
321 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
322 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveLong() ||
323 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveDouble());
Ian Rogers05f30572013-02-20 12:13:11 -0800324 return down_cast<LongArray*>(this);
325}
326
Mathieu Chartier4e305412014-02-19 10:54:44 -0800327template<VerifyObjectFlags kVerifyFlags>
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100328inline FloatArray* Object::AsFloatArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800329 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
330 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
331 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveFloat());
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100332 return down_cast<FloatArray*>(this);
333}
334
Mathieu Chartier4e305412014-02-19 10:54:44 -0800335template<VerifyObjectFlags kVerifyFlags>
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100336inline DoubleArray* Object::AsDoubleArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800337 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
338 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
339 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveDouble());
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100340 return down_cast<DoubleArray*>(this);
341}
342
Mathieu Chartier4e305412014-02-19 10:54:44 -0800343template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800344inline String* Object::AsString() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800345 DCHECK(GetClass<kVerifyFlags>()->IsStringClass());
Ian Rogers05f30572013-02-20 12:13:11 -0800346 return down_cast<String*>(this);
347}
348
Mathieu Chartier4e305412014-02-19 10:54:44 -0800349template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800350inline Throwable* Object::AsThrowable() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800351 DCHECK(GetClass<kVerifyFlags>()->IsThrowableClass());
Ian Rogers05f30572013-02-20 12:13:11 -0800352 return down_cast<Throwable*>(this);
353}
354
Mathieu Chartier4e305412014-02-19 10:54:44 -0800355template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800356inline bool Object::IsWeakReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800357 return GetClass<kVerifyFlags>()->IsWeakReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800358}
359
Mathieu Chartier4e305412014-02-19 10:54:44 -0800360template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800361inline bool Object::IsSoftReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800362 return GetClass<kVerifyFlags>()->IsSoftReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800363}
364
Mathieu Chartier4e305412014-02-19 10:54:44 -0800365template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800366inline bool Object::IsFinalizerReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800367 return GetClass<kVerifyFlags>()->IsFinalizerReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800368}
369
Mathieu Chartier4e305412014-02-19 10:54:44 -0800370template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700371inline FinalizerReference* Object::AsFinalizerReference() {
372 DCHECK(IsFinalizerReferenceInstance<kVerifyFlags>());
373 return down_cast<FinalizerReference*>(this);
374}
375
376template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800377inline bool Object::IsPhantomReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800378 return GetClass<kVerifyFlags>()->IsPhantomReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800379}
380
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700381template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800382inline size_t Object::SizeOf() {
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800383 size_t result;
Mathieu Chartier4e305412014-02-19 10:54:44 -0800384 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700385 if (IsArrayInstance<kVerifyFlags, kReadBarrierOption>()) {
386 result = AsArray<kNewFlags, kReadBarrierOption>()->
387 template SizeOf<kNewFlags, kReadBarrierOption>();
388 } else if (IsClass<kNewFlags, kReadBarrierOption>()) {
389 result = AsClass<kNewFlags, kReadBarrierOption>()->
390 template SizeOf<kNewFlags, kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800391 } else {
Hiroshi Yamauchi25023c72014-05-09 11:45:53 -0700392 result = GetClass<kNewFlags, kReadBarrierOption>()->
393 template GetObjectSize<kNewFlags, kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800394 }
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700395 DCHECK_GE(result, sizeof(Object))
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700396 << " class=" << PrettyTypeOf(GetClass<kNewFlags, kReadBarrierOption>());
397 DCHECK(!(IsArtField<kNewFlags, kReadBarrierOption>()) || result == sizeof(ArtField));
398 DCHECK(!(IsArtMethod<kNewFlags, kReadBarrierOption>()) || result == sizeof(ArtMethod));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800399 return result;
400}
401
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700402template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
403inline int32_t Object::GetField32(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800404 if (kVerifyFlags & kVerifyThis) {
405 VerifyObject(this);
406 }
Ian Rogersb122a4b2013-11-19 18:00:50 -0800407 const byte* raw_addr = reinterpret_cast<const byte*>(this) + field_offset.Int32Value();
408 const int32_t* word_addr = reinterpret_cast<const int32_t*>(raw_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700409 if (UNLIKELY(kIsVolatile)) {
Ian Rogersb122a4b2013-11-19 18:00:50 -0800410 int32_t result = *(reinterpret_cast<volatile int32_t*>(const_cast<int32_t*>(word_addr)));
Ian Rogersef7d42f2014-01-06 12:55:46 -0800411 QuasiAtomic::MembarLoadLoad(); // Ensure volatile loads don't re-order.
Ian Rogersb122a4b2013-11-19 18:00:50 -0800412 return result;
413 } else {
414 return *word_addr;
415 }
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) {
Ian Rogersb122a4b2013-11-19 18:00:50 -0800440 QuasiAtomic::MembarStoreStore(); // Ensure this store occurs after others in the queue.
441 *word_addr = new_value;
Ian Rogersef7d42f2014-01-06 12:55:46 -0800442 QuasiAtomic::MembarStoreLoad(); // Ensure this store occurs before any volatile loads.
Ian Rogersb122a4b2013-11-19 18:00:50 -0800443 } else {
444 *word_addr = new_value;
445 }
446}
447
Mathieu Chartier4e305412014-02-19 10:54:44 -0800448template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700449inline void Object::SetField32Volatile(MemberOffset field_offset, int32_t new_value) {
450 SetField32<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset, new_value);
451}
452
453template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers29501cf2014-02-07 21:00:25 -0800454inline bool Object::CasField32(MemberOffset field_offset, int32_t old_value, int32_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100455 if (kCheckTransaction) {
456 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
457 }
458 if (kTransactionActive) {
459 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
460 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800461 if (kVerifyFlags & kVerifyThis) {
462 VerifyObject(this);
463 }
Ian Rogersd9c4fc92013-10-01 19:45:43 -0700464 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
Ian Rogers29501cf2014-02-07 21:00:25 -0800465 volatile int32_t* addr = reinterpret_cast<volatile int32_t*>(raw_addr);
Ian Rogersb122a4b2013-11-19 18:00:50 -0800466 return __sync_bool_compare_and_swap(addr, 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) {
Ian Rogers29501cf2014-02-07 21:00:25 -0800477 int64_t result = QuasiAtomic::Read64(addr);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800478 QuasiAtomic::MembarLoadLoad(); // Ensure volatile loads don't re-order.
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800479 return result;
480 } else {
481 return *addr;
482 }
483}
484
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700485template<VerifyObjectFlags kVerifyFlags>
486inline int64_t Object::GetField64Volatile(MemberOffset field_offset) {
487 return GetField64<kVerifyFlags, true>(field_offset);
488}
489
490template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
491 bool kIsVolatile>
492inline void Object::SetField64(MemberOffset field_offset, int64_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100493 if (kCheckTransaction) {
494 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
495 }
496 if (kTransactionActive) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700497 Runtime::Current()->RecordWriteField64(this, field_offset,
498 GetField64<kVerifyFlags, kIsVolatile>(field_offset),
499 kIsVolatile);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100500 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800501 if (kVerifyFlags & kVerifyThis) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800502 VerifyObject(this);
503 }
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800504 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
505 int64_t* addr = reinterpret_cast<int64_t*>(raw_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700506 if (kIsVolatile) {
Ian Rogersb122a4b2013-11-19 18:00:50 -0800507 QuasiAtomic::MembarStoreStore(); // Ensure this store occurs after others in the queue.
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800508 QuasiAtomic::Write64(addr, new_value);
Ian Rogersb122a4b2013-11-19 18:00:50 -0800509 if (!QuasiAtomic::LongAtomicsUseMutexes()) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800510 QuasiAtomic::MembarStoreLoad(); // Ensure this store occurs before any volatile loads.
Ian Rogersb122a4b2013-11-19 18:00:50 -0800511 } else {
512 // Fence from from mutex is enough.
513 }
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800514 } else {
515 *addr = new_value;
516 }
517}
518
Mathieu Chartier4e305412014-02-19 10:54:44 -0800519template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700520inline void Object::SetField64Volatile(MemberOffset field_offset, int64_t new_value) {
521 return SetField64<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
522 new_value);
523}
524
525template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers29501cf2014-02-07 21:00:25 -0800526inline bool Object::CasField64(MemberOffset field_offset, int64_t old_value, int64_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100527 if (kCheckTransaction) {
528 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
529 }
530 if (kTransactionActive) {
531 Runtime::Current()->RecordWriteField64(this, field_offset, old_value, true);
532 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800533 if (kVerifyFlags & kVerifyThis) {
534 VerifyObject(this);
535 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800536 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
Ian Rogers29501cf2014-02-07 21:00:25 -0800537 volatile int64_t* addr = reinterpret_cast<volatile int64_t*>(raw_addr);
538 return QuasiAtomic::Cas64(old_value, new_value, addr);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800539}
540
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700541template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption,
542 bool kIsVolatile>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700543inline T* Object::GetFieldObject(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800544 if (kVerifyFlags & kVerifyThis) {
545 VerifyObject(this);
546 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800547 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
548 HeapReference<T>* objref_addr = reinterpret_cast<HeapReference<T>*>(raw_addr);
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700549 T* result = ReadBarrier::Barrier<T, kReadBarrierOption>(this, field_offset, objref_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700550 if (kIsVolatile) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800551 QuasiAtomic::MembarLoadLoad(); // Ensure loads don't re-order.
552 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800553 if (kVerifyFlags & kVerifyReads) {
554 VerifyObject(result);
555 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800556 return result;
557}
558
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700559template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700560inline T* Object::GetFieldObjectVolatile(MemberOffset field_offset) {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700561 return GetFieldObject<T, kVerifyFlags, kReadBarrierOption, true>(field_offset);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700562}
563
564template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
565 bool kIsVolatile>
566inline void Object::SetFieldObjectWithoutWriteBarrier(MemberOffset field_offset,
567 Object* new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100568 if (kCheckTransaction) {
569 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
570 }
571 if (kTransactionActive) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700572 mirror::Object* obj;
573 if (kIsVolatile) {
574 obj = GetFieldObjectVolatile<Object>(field_offset);
575 } else {
576 obj = GetFieldObject<Object>(field_offset);
577 }
578 Runtime::Current()->RecordWriteFieldReference(this, field_offset, obj, true);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100579 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800580 if (kVerifyFlags & kVerifyThis) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800581 VerifyObject(this);
582 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800583 if (kVerifyFlags & kVerifyWrites) {
584 VerifyObject(new_value);
585 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800586 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
587 HeapReference<Object>* objref_addr = reinterpret_cast<HeapReference<Object>*>(raw_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700588 if (kIsVolatile) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800589 QuasiAtomic::MembarStoreStore(); // Ensure this store occurs after others in the queue.
590 objref_addr->Assign(new_value);
591 QuasiAtomic::MembarStoreLoad(); // Ensure this store occurs before any loads.
592 } else {
593 objref_addr->Assign(new_value);
594 }
595}
596
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700597template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
598 bool kIsVolatile>
599inline void Object::SetFieldObject(MemberOffset field_offset, Object* new_value) {
600 SetFieldObjectWithoutWriteBarrier<kTransactionActive, kCheckTransaction, kVerifyFlags,
601 kIsVolatile>(field_offset, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800602 if (new_value != nullptr) {
603 CheckFieldAssignment(field_offset, new_value);
604 Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
605 }
606}
607
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700608template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
609inline void Object::SetFieldObjectVolatile(MemberOffset field_offset, Object* new_value) {
610 SetFieldObject<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
611 new_value);
612}
613
Mathieu Chartier4e305412014-02-19 10:54:44 -0800614template <VerifyObjectFlags kVerifyFlags>
615inline HeapReference<Object>* Object::GetFieldObjectReferenceAddr(MemberOffset field_offset) {
616 if (kVerifyFlags & kVerifyThis) {
617 VerifyObject(this);
618 }
619 return reinterpret_cast<HeapReference<Object>*>(reinterpret_cast<byte*>(this) +
620 field_offset.Int32Value());
621}
622
623template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
624inline bool Object::CasFieldObject(MemberOffset field_offset, Object* old_value,
625 Object* new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100626 if (kCheckTransaction) {
627 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
628 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800629 if (kVerifyFlags & kVerifyThis) {
630 VerifyObject(this);
631 }
632 if (kVerifyFlags & kVerifyWrites) {
633 VerifyObject(new_value);
634 }
635 if (kVerifyFlags & kVerifyReads) {
636 VerifyObject(old_value);
637 }
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100638 if (kTransactionActive) {
639 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
640 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800641 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
Ian Rogers29501cf2014-02-07 21:00:25 -0800642 volatile int32_t* addr = reinterpret_cast<volatile int32_t*>(raw_addr);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800643 HeapReference<Object> old_ref(HeapReference<Object>::FromMirrorPtr(old_value));
644 HeapReference<Object> new_ref(HeapReference<Object>::FromMirrorPtr(new_value));
645 bool success = __sync_bool_compare_and_swap(addr, old_ref.reference_, new_ref.reference_);
646 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?
679 if (!kVisitClass && UNLIKELY(field_offset.Uint32Value() == ClassOffset().Uint32Value())) {
680 continue;
681 }
682 visitor(this, field_offset, kIsStatic);
683 }
684 }
685 }
686}
687
688template<bool kVisitClass, typename Visitor>
689inline void Object::VisitInstanceFieldsReferences(mirror::Class* klass, const Visitor& visitor) {
690 VisitFieldsReferences<kVisitClass, false>(
691 klass->GetReferenceInstanceOffsets<kVerifyNone>(), visitor);
692}
693
694template<bool kVisitClass, typename Visitor>
695inline void Object::VisitStaticFieldsReferences(mirror::Class* klass, const Visitor& visitor) {
696 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>();
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700705 if (klass->IsVariableSize()) {
706 if (klass->IsClassClass()) {
707 AsClass<kVerifyNone>()->VisitReferences<kVisitClass>(klass, visitor);
708 } else {
709 DCHECK(klass->IsArrayClass<kVerifyFlags>());
710 if (klass->IsObjectArrayClass<kVerifyNone>()) {
711 AsObjectArray<mirror::Object, kVerifyNone>()->VisitReferences<kVisitClass>(visitor);
712 } else if (kVisitClass) {
713 visitor(this, ClassOffset(), false);
714 }
Mathieu Chartier407f7022014-02-18 14:37:05 -0800715 }
716 } else {
Mathieu Chartier580a8df2014-03-26 15:15:57 -0700717 VisitInstanceFieldsReferences<kVisitClass>(klass, visitor);
718 if (UNLIKELY(klass->IsReferenceClass<kVerifyNone>())) {
Mathieu Chartier407f7022014-02-18 14:37:05 -0800719 ref_visitor(klass, AsReference());
720 }
721 }
722}
723
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800724} // namespace mirror
725} // namespace art
726
Brian Carlstromfc0e3212013-07-17 14:40:12 -0700727#endif // ART_RUNTIME_MIRROR_OBJECT_INL_H_