blob: 86824ff538a4a22fc1ccdbed7c65699d025e005e [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 Yamauchi800ac2d2014-04-02 17:32:54 -0700108 return GetFieldObject<Object, kVerifyNone, false>(OFFSET_OF_OBJECT_MEMBER(Object, x_rb_ptr_), false);
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800109#else
110 LOG(FATAL) << "Unreachable";
111 return nullptr;
112#endif
113}
114
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700115inline void Object::SetReadBarrierPointer(Object* rb_ptr) {
Hiroshi Yamauchi624468c2014-03-31 15:14:47 -0700116#ifdef USE_BAKER_OR_BROOKS_READ_BARRIER
117 DCHECK(kUseBakerOrBrooksReadBarrier);
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800118 // We don't mark the card as this occurs as part of object allocation. Not all objects have
119 // backing cards, such as large objects.
120 SetFieldObjectWithoutWriteBarrier<false, false, kVerifyNone>(
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700121 OFFSET_OF_OBJECT_MEMBER(Object, x_rb_ptr_), rb_ptr, false);
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800122#else
123 LOG(FATAL) << "Unreachable";
124#endif
125}
126
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700127inline bool Object::AtomicSetReadBarrierPointer(Object* expected_rb_ptr, Object* rb_ptr) {
128#ifdef USE_BAKER_OR_BROOKS_READ_BARRIER
129 DCHECK(kUseBakerOrBrooksReadBarrier);
130 MemberOffset offset = OFFSET_OF_OBJECT_MEMBER(Object, x_rb_ptr_);
131 byte* raw_addr = reinterpret_cast<byte*>(this) + offset.SizeValue();
132 HeapReference<Object>* ref = reinterpret_cast<HeapReference<Object>*>(raw_addr);
133 HeapReference<Object> expected_ref(HeapReference<Object>::FromMirrorPtr(expected_rb_ptr));
134 HeapReference<Object> new_ref(HeapReference<Object>::FromMirrorPtr(rb_ptr));
135 uint32_t expected_val = expected_ref.reference_;
136 uint32_t new_val;
137 do {
138 uint32_t old_val = ref->reference_;
139 if (old_val != expected_val) {
140 // Lost the race.
141 return false;
142 }
143 new_val = new_ref.reference_;
144 } while (!__sync_bool_compare_and_swap(
145 reinterpret_cast<uint32_t*>(raw_addr), expected_val, new_val));
146 DCHECK_EQ(new_val, ref->reference_);
147 return true;
148#else
149 LOG(FATAL) << "Unreachable";
150 return false;
151#endif
152}
153
Hiroshi Yamauchi624468c2014-03-31 15:14:47 -0700154inline void Object::AssertReadBarrierPointer() const {
Hiroshi Yamauchi800ac2d2014-04-02 17:32:54 -0700155 if (kUseBakerReadBarrier) {
156 Object* obj = const_cast<Object*>(this);
157 DCHECK(obj->GetReadBarrierPointer() == nullptr)
158 << "Bad Baker pointer: obj=" << reinterpret_cast<void*>(obj)
159 << " ptr=" << reinterpret_cast<void*>(obj->GetReadBarrierPointer());
160 } else if (kUseBrooksReadBarrier) {
161 Object* obj = const_cast<Object*>(this);
162 DCHECK_EQ(obj, obj->GetReadBarrierPointer())
163 << "Bad Brooks pointer: obj=" << reinterpret_cast<void*>(obj)
164 << " ptr=" << reinterpret_cast<void*>(obj->GetReadBarrierPointer());
165 } else {
166 LOG(FATAL) << "Unreachable";
167 }
Hiroshi Yamauchi9d04a202014-01-31 13:35:49 -0800168}
169
Mathieu Chartier4e305412014-02-19 10:54:44 -0800170template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800171inline bool Object::VerifierInstanceOf(Class* klass) {
Jeff Haoa3faaf42013-09-03 19:07:00 -0700172 DCHECK(klass != NULL);
Mathieu Chartier4e305412014-02-19 10:54:44 -0800173 DCHECK(GetClass<kVerifyFlags>() != NULL);
Jeff Haoa3faaf42013-09-03 19:07:00 -0700174 return klass->IsInterface() || InstanceOf(klass);
175}
176
Mathieu Chartier4e305412014-02-19 10:54:44 -0800177template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800178inline bool Object::InstanceOf(Class* klass) {
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800179 DCHECK(klass != NULL);
Mathieu Chartier4e305412014-02-19 10:54:44 -0800180 DCHECK(GetClass<kVerifyNone>() != NULL);
181 return klass->IsAssignableFrom(GetClass<kVerifyFlags>());
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800182}
183
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700184template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800185inline bool Object::IsClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700186 Class* java_lang_Class = GetClass<kVerifyFlags, kReadBarrierOption>()->
187 template GetClass<kVerifyFlags, kReadBarrierOption>();
188 return GetClass<static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis),
189 kReadBarrierOption>() == java_lang_Class;
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800190}
191
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700192template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800193inline Class* Object::AsClass() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700194 DCHECK((IsClass<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800195 return down_cast<Class*>(this);
196}
197
Mathieu Chartier4e305412014-02-19 10:54:44 -0800198template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800199inline bool Object::IsObjectArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800200 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
201 return IsArrayInstance<kVerifyFlags>() &&
202 !GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitive();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800203}
204
Mathieu Chartier4e305412014-02-19 10:54:44 -0800205template<class T, VerifyObjectFlags kVerifyFlags>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800206inline ObjectArray<T>* Object::AsObjectArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800207 DCHECK(IsObjectArray<kVerifyFlags>());
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800208 return down_cast<ObjectArray<T>*>(this);
209}
210
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700211template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800212inline bool Object::IsArrayInstance() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700213 return GetClass<kVerifyFlags, kReadBarrierOption>()->
214 template IsArrayClass<kVerifyFlags, kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800215}
216
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700217template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800218inline bool Object::IsArtField() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700219 return GetClass<kVerifyFlags, kReadBarrierOption>()->
220 template IsArtFieldClass<kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800221}
222
Mathieu Chartier4e305412014-02-19 10:54:44 -0800223template<VerifyObjectFlags kVerifyFlags>
224inline ArtField* Object::AsArtField() {
225 DCHECK(IsArtField<kVerifyFlags>());
Brian Carlstromea46f952013-07-30 01:26:50 -0700226 return down_cast<ArtField*>(this);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800227}
228
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700229template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800230inline bool Object::IsArtMethod() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700231 return GetClass<kVerifyFlags, kReadBarrierOption>()->
232 template IsArtMethodClass<kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800233}
234
Mathieu Chartier4e305412014-02-19 10:54:44 -0800235template<VerifyObjectFlags kVerifyFlags>
Brian Carlstromea46f952013-07-30 01:26:50 -0700236inline ArtMethod* Object::AsArtMethod() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800237 DCHECK(IsArtMethod<kVerifyFlags>());
Brian Carlstromea46f952013-07-30 01:26:50 -0700238 return down_cast<ArtMethod*>(this);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800239}
240
Mathieu Chartier4e305412014-02-19 10:54:44 -0800241template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800242inline bool Object::IsReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800243 return GetClass<kVerifyFlags>()->IsReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800244}
245
Mathieu Chartier4e305412014-02-19 10:54:44 -0800246template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700247inline Reference* Object::AsReference() {
248 DCHECK(IsReferenceInstance<kVerifyFlags>());
249 return down_cast<Reference*>(this);
250}
251
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700252template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogers05f30572013-02-20 12:13:11 -0800253inline Array* Object::AsArray() {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700254 DCHECK((IsArrayInstance<kVerifyFlags, kReadBarrierOption>()));
Ian Rogers05f30572013-02-20 12:13:11 -0800255 return down_cast<Array*>(this);
256}
257
Mathieu Chartier4e305412014-02-19 10:54:44 -0800258template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800259inline BooleanArray* Object::AsBooleanArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800260 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
261 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
262 DCHECK(GetClass<kNewFlags>()->GetComponentType()->IsPrimitiveBoolean());
Ian Rogers05f30572013-02-20 12:13:11 -0800263 return down_cast<BooleanArray*>(this);
264}
265
Mathieu Chartier4e305412014-02-19 10:54:44 -0800266template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800267inline ByteArray* Object::AsByteArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800268 static const VerifyObjectFlags kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
269 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
270 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveByte());
Ian Rogers05f30572013-02-20 12:13:11 -0800271 return down_cast<ByteArray*>(this);
272}
273
Mathieu Chartier4e305412014-02-19 10:54:44 -0800274template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800275inline ByteArray* Object::AsByteSizedArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800276 constexpr VerifyObjectFlags kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
277 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
278 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveByte() ||
279 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveBoolean());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800280 return down_cast<ByteArray*>(this);
281}
282
Mathieu Chartier4e305412014-02-19 10:54:44 -0800283template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800284inline CharArray* Object::AsCharArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800285 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
286 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
287 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveChar());
Ian Rogers05f30572013-02-20 12:13:11 -0800288 return down_cast<CharArray*>(this);
289}
290
Mathieu Chartier4e305412014-02-19 10:54:44 -0800291template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800292inline ShortArray* Object::AsShortArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800293 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
294 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
295 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveShort());
Ian Rogers05f30572013-02-20 12:13:11 -0800296 return down_cast<ShortArray*>(this);
297}
298
Mathieu Chartier4e305412014-02-19 10:54:44 -0800299template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800300inline ShortArray* Object::AsShortSizedArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800301 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
302 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
303 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveShort() ||
304 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveChar());
Ian Rogersef7d42f2014-01-06 12:55:46 -0800305 return down_cast<ShortArray*>(this);
306}
307
Mathieu Chartier4e305412014-02-19 10:54:44 -0800308template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800309inline IntArray* Object::AsIntArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800310 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
311 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
312 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveInt() ||
313 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveFloat());
Ian Rogers05f30572013-02-20 12:13:11 -0800314 return down_cast<IntArray*>(this);
315}
316
Mathieu Chartier4e305412014-02-19 10:54:44 -0800317template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800318inline LongArray* Object::AsLongArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800319 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
320 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
321 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveLong() ||
322 GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveDouble());
Ian Rogers05f30572013-02-20 12:13:11 -0800323 return down_cast<LongArray*>(this);
324}
325
Mathieu Chartier4e305412014-02-19 10:54:44 -0800326template<VerifyObjectFlags kVerifyFlags>
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100327inline FloatArray* Object::AsFloatArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800328 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
329 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
330 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveFloat());
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100331 return down_cast<FloatArray*>(this);
332}
333
Mathieu Chartier4e305412014-02-19 10:54:44 -0800334template<VerifyObjectFlags kVerifyFlags>
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100335inline DoubleArray* Object::AsDoubleArray() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800336 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
337 DCHECK(GetClass<kVerifyFlags>()->IsArrayClass());
338 DCHECK(GetClass<kNewFlags>()->template GetComponentType<kNewFlags>()->IsPrimitiveDouble());
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100339 return down_cast<DoubleArray*>(this);
340}
341
Mathieu Chartier4e305412014-02-19 10:54:44 -0800342template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800343inline String* Object::AsString() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800344 DCHECK(GetClass<kVerifyFlags>()->IsStringClass());
Ian Rogers05f30572013-02-20 12:13:11 -0800345 return down_cast<String*>(this);
346}
347
Mathieu Chartier4e305412014-02-19 10:54:44 -0800348template<VerifyObjectFlags kVerifyFlags>
Ian Rogers05f30572013-02-20 12:13:11 -0800349inline Throwable* Object::AsThrowable() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800350 DCHECK(GetClass<kVerifyFlags>()->IsThrowableClass());
Ian Rogers05f30572013-02-20 12:13:11 -0800351 return down_cast<Throwable*>(this);
352}
353
Mathieu Chartier4e305412014-02-19 10:54:44 -0800354template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800355inline bool Object::IsWeakReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800356 return GetClass<kVerifyFlags>()->IsWeakReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800357}
358
Mathieu Chartier4e305412014-02-19 10:54:44 -0800359template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800360inline bool Object::IsSoftReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800361 return GetClass<kVerifyFlags>()->IsSoftReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800362}
363
Mathieu Chartier4e305412014-02-19 10:54:44 -0800364template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800365inline bool Object::IsFinalizerReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800366 return GetClass<kVerifyFlags>()->IsFinalizerReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800367}
368
Mathieu Chartier4e305412014-02-19 10:54:44 -0800369template<VerifyObjectFlags kVerifyFlags>
Mathieu Chartier8fa2dad2014-03-13 12:22:56 -0700370inline FinalizerReference* Object::AsFinalizerReference() {
371 DCHECK(IsFinalizerReferenceInstance<kVerifyFlags>());
372 return down_cast<FinalizerReference*>(this);
373}
374
375template<VerifyObjectFlags kVerifyFlags>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800376inline bool Object::IsPhantomReferenceInstance() {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800377 return GetClass<kVerifyFlags>()->IsPhantomReferenceClass();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800378}
379
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700380template<VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersef7d42f2014-01-06 12:55:46 -0800381inline size_t Object::SizeOf() {
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800382 size_t result;
Mathieu Chartier4e305412014-02-19 10:54:44 -0800383 constexpr auto kNewFlags = static_cast<VerifyObjectFlags>(kVerifyFlags & ~kVerifyThis);
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700384 if (IsArrayInstance<kVerifyFlags, kReadBarrierOption>()) {
385 result = AsArray<kNewFlags, kReadBarrierOption>()->
386 template SizeOf<kNewFlags, kReadBarrierOption>();
387 } else if (IsClass<kNewFlags, kReadBarrierOption>()) {
388 result = AsClass<kNewFlags, kReadBarrierOption>()->
389 template SizeOf<kNewFlags, kReadBarrierOption>();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800390 } else {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700391 result = GetClass<kNewFlags, kReadBarrierOption>()->GetObjectSize();
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800392 }
Hiroshi Yamauchi9103c862014-04-22 13:51:07 -0700393 DCHECK_GE(result, sizeof(Object))
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700394 << " class=" << PrettyTypeOf(GetClass<kNewFlags, kReadBarrierOption>());
395 DCHECK(!(IsArtField<kNewFlags, kReadBarrierOption>()) || result == sizeof(ArtField));
396 DCHECK(!(IsArtMethod<kNewFlags, kReadBarrierOption>()) || result == sizeof(ArtMethod));
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800397 return result;
398}
399
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700400template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
401inline int32_t Object::GetField32(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800402 if (kVerifyFlags & kVerifyThis) {
403 VerifyObject(this);
404 }
Ian Rogersb122a4b2013-11-19 18:00:50 -0800405 const byte* raw_addr = reinterpret_cast<const byte*>(this) + field_offset.Int32Value();
406 const int32_t* word_addr = reinterpret_cast<const int32_t*>(raw_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700407 if (UNLIKELY(kIsVolatile)) {
Ian Rogersb122a4b2013-11-19 18:00:50 -0800408 int32_t result = *(reinterpret_cast<volatile int32_t*>(const_cast<int32_t*>(word_addr)));
Ian Rogersef7d42f2014-01-06 12:55:46 -0800409 QuasiAtomic::MembarLoadLoad(); // Ensure volatile loads don't re-order.
Ian Rogersb122a4b2013-11-19 18:00:50 -0800410 return result;
411 } else {
412 return *word_addr;
413 }
414}
415
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700416template<VerifyObjectFlags kVerifyFlags>
417inline int32_t Object::GetField32Volatile(MemberOffset field_offset) {
418 return GetField32<kVerifyFlags, true>(field_offset);
419}
420
421template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
422 bool kIsVolatile>
423inline void Object::SetField32(MemberOffset field_offset, int32_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100424 if (kCheckTransaction) {
425 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
426 }
427 if (kTransactionActive) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700428 Runtime::Current()->RecordWriteField32(this, field_offset,
429 GetField32<kVerifyFlags, kIsVolatile>(field_offset),
430 kIsVolatile);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100431 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800432 if (kVerifyFlags & kVerifyThis) {
Ian Rogersb122a4b2013-11-19 18:00:50 -0800433 VerifyObject(this);
434 }
435 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
Ian Rogers29501cf2014-02-07 21:00:25 -0800436 int32_t* word_addr = reinterpret_cast<int32_t*>(raw_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700437 if (kIsVolatile) {
Ian Rogersb122a4b2013-11-19 18:00:50 -0800438 QuasiAtomic::MembarStoreStore(); // Ensure this store occurs after others in the queue.
439 *word_addr = new_value;
Ian Rogersef7d42f2014-01-06 12:55:46 -0800440 QuasiAtomic::MembarStoreLoad(); // Ensure this store occurs before any volatile loads.
Ian Rogersb122a4b2013-11-19 18:00:50 -0800441 } else {
442 *word_addr = new_value;
443 }
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 Rogers29501cf2014-02-07 21:00:25 -0800452inline bool Object::CasField32(MemberOffset field_offset, int32_t old_value, int32_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100453 if (kCheckTransaction) {
454 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
455 }
456 if (kTransactionActive) {
457 Runtime::Current()->RecordWriteField32(this, field_offset, old_value, true);
458 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800459 if (kVerifyFlags & kVerifyThis) {
460 VerifyObject(this);
461 }
Ian Rogersd9c4fc92013-10-01 19:45:43 -0700462 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
Ian Rogers29501cf2014-02-07 21:00:25 -0800463 volatile int32_t* addr = reinterpret_cast<volatile int32_t*>(raw_addr);
Ian Rogersb122a4b2013-11-19 18:00:50 -0800464 return __sync_bool_compare_and_swap(addr, old_value, new_value);
Ian Rogersd9c4fc92013-10-01 19:45:43 -0700465}
466
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700467template<VerifyObjectFlags kVerifyFlags, bool kIsVolatile>
468inline int64_t Object::GetField64(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800469 if (kVerifyFlags & kVerifyThis) {
470 VerifyObject(this);
471 }
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800472 const byte* raw_addr = reinterpret_cast<const byte*>(this) + field_offset.Int32Value();
473 const int64_t* addr = reinterpret_cast<const int64_t*>(raw_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700474 if (kIsVolatile) {
Ian Rogers29501cf2014-02-07 21:00:25 -0800475 int64_t result = QuasiAtomic::Read64(addr);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800476 QuasiAtomic::MembarLoadLoad(); // Ensure volatile loads don't re-order.
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800477 return result;
478 } else {
479 return *addr;
480 }
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) {
Ian Rogersb122a4b2013-11-19 18:00:50 -0800505 QuasiAtomic::MembarStoreStore(); // Ensure this store occurs after others in the queue.
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800506 QuasiAtomic::Write64(addr, new_value);
Ian Rogersb122a4b2013-11-19 18:00:50 -0800507 if (!QuasiAtomic::LongAtomicsUseMutexes()) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800508 QuasiAtomic::MembarStoreLoad(); // Ensure this store occurs before any volatile loads.
Ian Rogersb122a4b2013-11-19 18:00:50 -0800509 } else {
510 // Fence from from mutex is enough.
511 }
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800512 } else {
513 *addr = new_value;
514 }
515}
516
Mathieu Chartier4e305412014-02-19 10:54:44 -0800517template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700518inline void Object::SetField64Volatile(MemberOffset field_offset, int64_t new_value) {
519 return SetField64<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
520 new_value);
521}
522
523template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
Ian Rogers29501cf2014-02-07 21:00:25 -0800524inline bool Object::CasField64(MemberOffset field_offset, int64_t old_value, int64_t new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100525 if (kCheckTransaction) {
526 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
527 }
528 if (kTransactionActive) {
529 Runtime::Current()->RecordWriteField64(this, field_offset, old_value, true);
530 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800531 if (kVerifyFlags & kVerifyThis) {
532 VerifyObject(this);
533 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800534 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
Ian Rogers29501cf2014-02-07 21:00:25 -0800535 volatile int64_t* addr = reinterpret_cast<volatile int64_t*>(raw_addr);
536 return QuasiAtomic::Cas64(old_value, new_value, addr);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800537}
538
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700539template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption,
540 bool kIsVolatile>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700541inline T* Object::GetFieldObject(MemberOffset field_offset) {
Mathieu Chartier4e305412014-02-19 10:54:44 -0800542 if (kVerifyFlags & kVerifyThis) {
543 VerifyObject(this);
544 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800545 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
546 HeapReference<T>* objref_addr = reinterpret_cast<HeapReference<T>*>(raw_addr);
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700547 T* result = ReadBarrier::Barrier<T, kReadBarrierOption>(this, field_offset, objref_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700548 if (kIsVolatile) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800549 QuasiAtomic::MembarLoadLoad(); // Ensure loads don't re-order.
550 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800551 if (kVerifyFlags & kVerifyReads) {
552 VerifyObject(result);
553 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800554 return result;
555}
556
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700557template<class T, VerifyObjectFlags kVerifyFlags, ReadBarrierOption kReadBarrierOption>
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700558inline T* Object::GetFieldObjectVolatile(MemberOffset field_offset) {
Hiroshi Yamauchi6e83c172014-05-01 21:25:41 -0700559 return GetFieldObject<T, kVerifyFlags, kReadBarrierOption, true>(field_offset);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700560}
561
562template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
563 bool kIsVolatile>
564inline void Object::SetFieldObjectWithoutWriteBarrier(MemberOffset field_offset,
565 Object* new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100566 if (kCheckTransaction) {
567 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
568 }
569 if (kTransactionActive) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700570 mirror::Object* obj;
571 if (kIsVolatile) {
572 obj = GetFieldObjectVolatile<Object>(field_offset);
573 } else {
574 obj = GetFieldObject<Object>(field_offset);
575 }
576 Runtime::Current()->RecordWriteFieldReference(this, field_offset, obj, true);
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100577 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800578 if (kVerifyFlags & kVerifyThis) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800579 VerifyObject(this);
580 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800581 if (kVerifyFlags & kVerifyWrites) {
582 VerifyObject(new_value);
583 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800584 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
585 HeapReference<Object>* objref_addr = reinterpret_cast<HeapReference<Object>*>(raw_addr);
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700586 if (kIsVolatile) {
Ian Rogersef7d42f2014-01-06 12:55:46 -0800587 QuasiAtomic::MembarStoreStore(); // Ensure this store occurs after others in the queue.
588 objref_addr->Assign(new_value);
589 QuasiAtomic::MembarStoreLoad(); // Ensure this store occurs before any loads.
590 } else {
591 objref_addr->Assign(new_value);
592 }
593}
594
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700595template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags,
596 bool kIsVolatile>
597inline void Object::SetFieldObject(MemberOffset field_offset, Object* new_value) {
598 SetFieldObjectWithoutWriteBarrier<kTransactionActive, kCheckTransaction, kVerifyFlags,
599 kIsVolatile>(field_offset, new_value);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800600 if (new_value != nullptr) {
601 CheckFieldAssignment(field_offset, new_value);
602 Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
603 }
604}
605
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700606template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
607inline void Object::SetFieldObjectVolatile(MemberOffset field_offset, Object* new_value) {
608 SetFieldObject<kTransactionActive, kCheckTransaction, kVerifyFlags, true>(field_offset,
609 new_value);
610}
611
Mathieu Chartier4e305412014-02-19 10:54:44 -0800612template <VerifyObjectFlags kVerifyFlags>
613inline HeapReference<Object>* Object::GetFieldObjectReferenceAddr(MemberOffset field_offset) {
614 if (kVerifyFlags & kVerifyThis) {
615 VerifyObject(this);
616 }
617 return reinterpret_cast<HeapReference<Object>*>(reinterpret_cast<byte*>(this) +
618 field_offset.Int32Value());
619}
620
621template<bool kTransactionActive, bool kCheckTransaction, VerifyObjectFlags kVerifyFlags>
622inline bool Object::CasFieldObject(MemberOffset field_offset, Object* old_value,
623 Object* new_value) {
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100624 if (kCheckTransaction) {
625 DCHECK_EQ(kTransactionActive, Runtime::Current()->IsActiveTransaction());
626 }
Mathieu Chartier4e305412014-02-19 10:54:44 -0800627 if (kVerifyFlags & kVerifyThis) {
628 VerifyObject(this);
629 }
630 if (kVerifyFlags & kVerifyWrites) {
631 VerifyObject(new_value);
632 }
633 if (kVerifyFlags & kVerifyReads) {
634 VerifyObject(old_value);
635 }
Sebastien Hertzd2fe10a2014-01-15 10:20:56 +0100636 if (kTransactionActive) {
637 Runtime::Current()->RecordWriteFieldReference(this, field_offset, old_value, true);
638 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800639 byte* raw_addr = reinterpret_cast<byte*>(this) + field_offset.Int32Value();
Ian Rogers29501cf2014-02-07 21:00:25 -0800640 volatile int32_t* addr = reinterpret_cast<volatile int32_t*>(raw_addr);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800641 HeapReference<Object> old_ref(HeapReference<Object>::FromMirrorPtr(old_value));
642 HeapReference<Object> new_ref(HeapReference<Object>::FromMirrorPtr(new_value));
643 bool success = __sync_bool_compare_and_swap(addr, old_ref.reference_, new_ref.reference_);
644 if (success) {
645 Runtime::Current()->GetHeap()->WriteBarrierField(this, field_offset, new_value);
646 }
647 return success;
648}
649
Mathieu Chartier407f7022014-02-18 14:37:05 -0800650template<bool kVisitClass, bool kIsStatic, typename Visitor>
651inline void Object::VisitFieldsReferences(uint32_t ref_offsets, const Visitor& visitor) {
652 if (LIKELY(ref_offsets != CLASS_WALK_SUPER)) {
653 if (!kVisitClass) {
654 // Mask out the class from the reference offsets.
655 ref_offsets ^= kWordHighBitMask;
656 }
657 DCHECK_EQ(ClassOffset().Uint32Value(), 0U);
658 // Found a reference offset bitmap. Visit the specified offsets.
659 while (ref_offsets != 0) {
660 size_t right_shift = CLZ(ref_offsets);
661 MemberOffset field_offset = CLASS_OFFSET_FROM_CLZ(right_shift);
662 visitor(this, field_offset, kIsStatic);
663 ref_offsets &= ~(CLASS_HIGH_BIT >> right_shift);
664 }
665 } else {
666 // There is no reference offset bitmap. In the non-static case, walk up the class
667 // inheritance hierarchy and find reference offsets the hard way. In the static case, just
668 // consider this class.
669 for (mirror::Class* klass = kIsStatic ? AsClass() : GetClass(); klass != nullptr;
670 klass = kIsStatic ? nullptr : klass->GetSuperClass()) {
671 size_t num_reference_fields =
672 kIsStatic ? klass->NumReferenceStaticFields() : klass->NumReferenceInstanceFields();
673 for (size_t i = 0; i < num_reference_fields; ++i) {
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700674 mirror::ArtField* field = kIsStatic ? klass->GetStaticField(i) : klass->GetInstanceField(i);
Mathieu Chartier407f7022014-02-18 14:37:05 -0800675 MemberOffset field_offset = field->GetOffset();
676 // TODO: Do a simpler check?
677 if (!kVisitClass && UNLIKELY(field_offset.Uint32Value() == ClassOffset().Uint32Value())) {
678 continue;
679 }
680 visitor(this, field_offset, kIsStatic);
681 }
682 }
683 }
684}
685
686template<bool kVisitClass, typename Visitor>
687inline void Object::VisitInstanceFieldsReferences(mirror::Class* klass, const Visitor& visitor) {
688 VisitFieldsReferences<kVisitClass, false>(
689 klass->GetReferenceInstanceOffsets<kVerifyNone>(), visitor);
690}
691
692template<bool kVisitClass, typename Visitor>
693inline void Object::VisitStaticFieldsReferences(mirror::Class* klass, const Visitor& visitor) {
694 klass->VisitFieldsReferences<kVisitClass, true>(
695 klass->GetReferenceStaticOffsets<kVerifyNone>(), visitor);
696}
697
698template <const bool kVisitClass, VerifyObjectFlags kVerifyFlags, typename Visitor,
699 typename JavaLangRefVisitor>
700inline void Object::VisitReferences(const Visitor& visitor,
701 const JavaLangRefVisitor& ref_visitor) {
702 mirror::Class* klass = GetClass<kVerifyFlags>();
Ian Rogersb0fa5dc2014-04-28 16:47:08 -0700703 if (klass->IsVariableSize()) {
704 if (klass->IsClassClass()) {
705 AsClass<kVerifyNone>()->VisitReferences<kVisitClass>(klass, visitor);
706 } else {
707 DCHECK(klass->IsArrayClass<kVerifyFlags>());
708 if (klass->IsObjectArrayClass<kVerifyNone>()) {
709 AsObjectArray<mirror::Object, kVerifyNone>()->VisitReferences<kVisitClass>(visitor);
710 } else if (kVisitClass) {
711 visitor(this, ClassOffset(), false);
712 }
Mathieu Chartier407f7022014-02-18 14:37:05 -0800713 }
714 } else {
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_